Commit 93eb211e authored by Daisuke HATAYAMA's avatar Daisuke HATAYAMA Committed by Linus Torvalds

elf coredump: make offset calculation process and writing process explicit

By the next patch, elf_core_dump() and elf_fdpic_core_dump() will support
extended numbering and so will produce the corefiles with section header
table in a special case.

The problem is the process of writing a file header offset of the section
header table into e_shoff field of the ELF header.  ELF header is
positioned at the beginning of the corefile, while section header at the
end.  So, we need to take which of the following ways:

 1. Seek backward to retry writing operation for ELF header
    after writing process for a whole part

 2. Make offset calculation process and writing process
    totally sequential

The clause 1.  is not always possible: one cannot assume that file system
supports seek function.  Consider the no_llseek case.

Therefore, this patch adopts the clause 2.
Signed-off-by: default avatarDaisuke HATAYAMA <d.hatayama@jp.fujitsu.com>
Cc: "Luck, Tony" <tony.luck@intel.com>
Cc: Jeff Dike <jdike@addtoit.com>
Cc: David Howells <dhowells@redhat.com>
Cc: Greg Ungerer <gerg@snapgear.com>
Cc: Roland McGrath <roland@redhat.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Alexander Viro <viro@zeniv.linux.org.uk>
Cc: Andi Kleen <andi@firstfloor.org>
Cc: Alan Cox <alan@lxorguk.ukuu.org.uk>
Cc: <linux-arch@vger.kernel.org>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 1fcccbac
...@@ -1856,6 +1856,7 @@ static int elf_core_dump(struct coredump_params *cprm) ...@@ -1856,6 +1856,7 @@ static int elf_core_dump(struct coredump_params *cprm)
loff_t offset = 0, dataoff, foffset; loff_t offset = 0, dataoff, foffset;
unsigned long mm_flags; unsigned long mm_flags;
struct elf_note_info info; struct elf_note_info info;
struct elf_phdr *phdr4note = NULL;
/* /*
* We no longer stop all VM operations. * We no longer stop all VM operations.
...@@ -1898,28 +1899,22 @@ static int elf_core_dump(struct coredump_params *cprm) ...@@ -1898,28 +1899,22 @@ static int elf_core_dump(struct coredump_params *cprm)
fs = get_fs(); fs = get_fs();
set_fs(KERNEL_DS); set_fs(KERNEL_DS);
size += sizeof(*elf);
if (size > cprm->limit || !dump_write(cprm->file, elf, sizeof(*elf)))
goto end_coredump;
offset += sizeof(*elf); /* Elf header */ offset += sizeof(*elf); /* Elf header */
offset += (segs + 1) * sizeof(struct elf_phdr); /* Program headers */ offset += (segs + 1) * sizeof(struct elf_phdr); /* Program headers */
foffset = offset; foffset = offset;
/* Write notes phdr entry */ /* Write notes phdr entry */
{ {
struct elf_phdr phdr;
size_t sz = get_note_info_size(&info); size_t sz = get_note_info_size(&info);
sz += elf_coredump_extra_notes_size(); sz += elf_coredump_extra_notes_size();
fill_elf_note_phdr(&phdr, sz, offset); phdr4note = kmalloc(sizeof(*phdr4note), GFP_KERNEL);
offset += sz; if (!phdr4note)
size += sizeof(phdr);
if (size > cprm->limit
|| !dump_write(cprm->file, &phdr, sizeof(phdr)))
goto end_coredump; goto end_coredump;
fill_elf_note_phdr(phdr4note, sz, offset);
offset += sz;
} }
dataoff = offset = roundup(offset, ELF_EXEC_PAGESIZE); dataoff = offset = roundup(offset, ELF_EXEC_PAGESIZE);
...@@ -1931,6 +1926,15 @@ static int elf_core_dump(struct coredump_params *cprm) ...@@ -1931,6 +1926,15 @@ static int elf_core_dump(struct coredump_params *cprm)
*/ */
mm_flags = current->mm->flags; mm_flags = current->mm->flags;
size += sizeof(*elf);
if (size > cprm->limit || !dump_write(cprm->file, elf, sizeof(*elf)))
goto end_coredump;
size += sizeof(*phdr4note);
if (size > cprm->limit
|| !dump_write(cprm->file, phdr4note, sizeof(*phdr4note)))
goto end_coredump;
/* Write program headers for segments dump */ /* Write program headers for segments dump */
for (vma = first_vma(current, gate_vma); vma != NULL; for (vma = first_vma(current, gate_vma); vma != NULL;
vma = next_vma(vma, gate_vma)) { vma = next_vma(vma, gate_vma)) {
...@@ -2004,6 +2008,7 @@ static int elf_core_dump(struct coredump_params *cprm) ...@@ -2004,6 +2008,7 @@ static int elf_core_dump(struct coredump_params *cprm)
cleanup: cleanup:
free_note_info(&info); free_note_info(&info);
kfree(phdr4note);
kfree(elf); kfree(elf);
out: out:
return has_dumped; return has_dumped;
......
...@@ -1600,6 +1600,7 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm) ...@@ -1600,6 +1600,7 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)
int thread_status_size = 0; int thread_status_size = 0;
elf_addr_t *auxv; elf_addr_t *auxv;
unsigned long mm_flags; unsigned long mm_flags;
struct elf_phdr *phdr4note = NULL;
/* /*
* We no longer stop all VM operations. * We no longer stop all VM operations.
...@@ -1706,18 +1707,12 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm) ...@@ -1706,18 +1707,12 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)
fs = get_fs(); fs = get_fs();
set_fs(KERNEL_DS); set_fs(KERNEL_DS);
size += sizeof(*elf);
if (size > cprm->limit
|| !dump_write(cprm->file, elf, sizeof(*elf)))
goto end_coredump;
offset += sizeof(*elf); /* Elf header */ offset += sizeof(*elf); /* Elf header */
offset += (segs+1) * sizeof(struct elf_phdr); /* Program headers */ offset += (segs+1) * sizeof(struct elf_phdr); /* Program headers */
foffset = offset; foffset = offset;
/* Write notes phdr entry */ /* Write notes phdr entry */
{ {
struct elf_phdr phdr;
int sz = 0; int sz = 0;
for (i = 0; i < numnote; i++) for (i = 0; i < numnote; i++)
...@@ -1725,13 +1720,12 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm) ...@@ -1725,13 +1720,12 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)
sz += thread_status_size; sz += thread_status_size;
fill_elf_note_phdr(&phdr, sz, offset); phdr4note = kmalloc(sizeof(*phdr4note), GFP_KERNEL);
offset += sz; if (!phdr4note)
size += sizeof(phdr);
if (size > cprm->limit
|| !dump_write(cprm->file, &phdr, sizeof(phdr)))
goto end_coredump; goto end_coredump;
fill_elf_note_phdr(phdr4note, sz, offset);
offset += sz;
} }
/* Page-align dumped data */ /* Page-align dumped data */
...@@ -1744,6 +1738,15 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm) ...@@ -1744,6 +1738,15 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)
*/ */
mm_flags = current->mm->flags; mm_flags = current->mm->flags;
size += sizeof(*elf);
if (size > cprm->limit || !dump_write(cprm->file, elf, sizeof(*elf)))
goto end_coredump;
size += sizeof(*phdr4note);
if (size > cprm->limit
|| !dump_write(cprm->file, phdr4note, sizeof(*phdr4note)))
goto end_coredump;
/* write program headers for segments dump */ /* write program headers for segments dump */
for (vma = current->mm->mmap; vma; vma = vma->vm_next) { for (vma = current->mm->mmap; vma; vma = vma->vm_next) {
struct elf_phdr phdr; struct elf_phdr phdr;
...@@ -1815,7 +1818,7 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm) ...@@ -1815,7 +1818,7 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)
list_del(tmp); list_del(tmp);
kfree(list_entry(tmp, struct elf_thread_status, list)); kfree(list_entry(tmp, struct elf_thread_status, list));
} }
kfree(phdr4note);
kfree(elf); kfree(elf);
kfree(prstatus); kfree(prstatus);
kfree(psinfo); kfree(psinfo);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment