2 * Copyright (c) 1998 John D. Polstra
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * $FreeBSD: src/usr.bin/gcore/elfcore.c,v 1.5.2.3 2001/07/11 23:59:11 obrien Exp $
27 * $DragonFly: src/usr.bin/gcore/elfcore.c,v 1.5 2008/06/05 18:06:33 swildner Exp $
30 #define _KERNEL_STRUCTURES
32 #include <sys/param.h>
34 #include <sys/procfs.h>
35 #include <machine/elf.h>
36 #include <vm/vm_param.h>
39 #include <vm/vm_map.h>
51 * Code for generating ELF core dumps.
54 typedef void (*segment_callback
)(vm_map_entry_t
, void *);
56 /* Closure for cb_put_phdr(). */
58 Elf_Phdr
*phdr
; /* Program header to fill in */
59 Elf_Off offset
; /* Offset of segment in core file */
62 /* Closure for cb_size_segment(). */
64 int count
; /* Count of writable segments. */
65 size_t size
; /* Total size of all writable segments. */
68 static void cb_put_phdr(vm_map_entry_t
, void *);
69 static void cb_size_segment(vm_map_entry_t
, void *);
70 static void each_writable_segment(vm_map_entry_t
, segment_callback
,
72 static void elf_corehdr(int fd
, pid_t
, vm_map_entry_t
, int numsegs
,
73 void *hdr
, size_t hdrsize
);
74 static void elf_puthdr(vm_map_entry_t
, void *, size_t *,
75 const prstatus_t
*, const prfpregset_t
*, const prpsinfo_t
*, int numsegs
);
76 static void elf_putnote(void *dst
, size_t *off
, const char *name
, int type
,
77 const void *desc
, size_t descsz
);
78 static void freemap(vm_map_entry_t
);
79 static void readhdrinfo(pid_t
, prstatus_t
*, prfpregset_t
*, prpsinfo_t
*);
80 static vm_map_entry_t
readmap(pid_t
);
83 * Write an ELF coredump for the given pid to the given fd.
86 elf_coredump(int fd
, pid_t pid
)
89 struct sseg_closure seginfo
;
97 /* Get the program's memory map. */
100 /* Size the program segments. */
103 each_writable_segment(map
, cb_size_segment
, &seginfo
);
106 * Calculate the size of the core file header area by making
107 * a dry run of generating it. Nothing is written, but the
108 * size is calculated.
111 elf_puthdr(map
, NULL
, &hdrsize
,
112 (const prstatus_t
*)NULL
, (const prfpregset_t
*)NULL
,
113 (const prpsinfo_t
*)NULL
, seginfo
.count
);
116 * Allocate memory for building the header, fill it up,
119 hdr
= malloc(hdrsize
);
120 if ((hdr
= malloc(hdrsize
)) == NULL
)
121 errx(1, "out of memory");
122 elf_corehdr(fd
, pid
, map
, seginfo
.count
, hdr
, hdrsize
);
124 /* Write the contents of all of the writable segments. */
125 snprintf(memname
, sizeof memname
, "/proc/%d/mem", pid
);
126 if ((memfd
= open(memname
, O_RDONLY
)) == -1)
127 err(1, "cannot open %s", memname
);
129 php
= (Elf_Phdr
*)((char *)hdr
+ sizeof(Elf_Ehdr
)) + 1;
130 for (i
= 0; i
< seginfo
.count
; i
++) {
131 int nleft
= php
->p_filesz
;
133 lseek(memfd
, (off_t
)php
->p_vaddr
, SEEK_SET
);
140 if (nwant
> sizeof buf
)
142 ngot
= read(memfd
, buf
, nwant
);
144 err(1, "read from %s", memname
);
146 errx(1, "short read from %s:"
147 " wanted %d, got %d\n", memname
,
149 ngot
= write(fd
, buf
, nwant
);
151 err(1, "write of segment %d failed", i
);
153 errx(1, "short write");
164 * A callback for each_writable_segment() to write out the segment's
165 * program header entry.
168 cb_put_phdr(vm_map_entry_t entry
, void *closure
)
170 struct phdr_closure
*phc
= (struct phdr_closure
*)closure
;
171 Elf_Phdr
*phdr
= phc
->phdr
;
173 phc
->offset
= round_page(phc
->offset
);
175 phdr
->p_type
= PT_LOAD
;
176 phdr
->p_offset
= phc
->offset
;
177 phdr
->p_vaddr
= entry
->start
;
179 phdr
->p_filesz
= phdr
->p_memsz
= entry
->end
- entry
->start
;
180 phdr
->p_align
= PAGE_SIZE
;
182 if (entry
->protection
& VM_PROT_READ
)
183 phdr
->p_flags
|= PF_R
;
184 if (entry
->protection
& VM_PROT_WRITE
)
185 phdr
->p_flags
|= PF_W
;
186 if (entry
->protection
& VM_PROT_EXECUTE
)
187 phdr
->p_flags
|= PF_X
;
189 phc
->offset
+= phdr
->p_filesz
;
194 * A callback for each_writable_segment() to gather information about
195 * the number of segments and their total size.
198 cb_size_segment(vm_map_entry_t entry
, void *closure
)
200 struct sseg_closure
*ssc
= (struct sseg_closure
*)closure
;
203 ssc
->size
+= entry
->end
- entry
->start
;
207 * For each segment in the given memory map, call the given function
208 * with a pointer to the map entry and some arbitrary caller-supplied
212 each_writable_segment(vm_map_entry_t map
, segment_callback func
, void *closure
)
214 vm_map_entry_t entry
;
216 for (entry
= map
; entry
!= NULL
; entry
= entry
->next
)
217 (*func
)(entry
, closure
);
221 * Write the core file header to the file, including padding up to
225 elf_corehdr(int fd
, pid_t pid
, vm_map_entry_t map
, int numsegs
, void *hdr
,
230 prfpregset_t fpregset
;
233 /* Gather the information for the header. */
234 readhdrinfo(pid
, &status
, &fpregset
, &psinfo
);
236 /* Fill in the header. */
237 memset(hdr
, 0, hdrsize
);
239 elf_puthdr(map
, hdr
, &off
, &status
, &fpregset
, &psinfo
, numsegs
);
241 /* Write it to the core file. */
242 if (write(fd
, hdr
, hdrsize
) == -1)
247 * Generate the ELF coredump header into the buffer at "dst". "dst" may
248 * be NULL, in which case the header is sized but not actually generated.
251 elf_puthdr(vm_map_entry_t map
, void *dst
, size_t *off
, const prstatus_t
*status
,
252 const prfpregset_t
*fpregset
, const prpsinfo_t
*psinfo
, int numsegs
)
260 *off
+= sizeof(Elf_Ehdr
);
263 *off
+= (numsegs
+ 1) * sizeof(Elf_Phdr
);
266 elf_putnote(dst
, off
, "FreeBSD", NT_PRSTATUS
, status
,
268 elf_putnote(dst
, off
, "FreeBSD", NT_FPREGSET
, fpregset
,
270 elf_putnote(dst
, off
, "FreeBSD", NT_PRPSINFO
, psinfo
,
272 notesz
= *off
- noteoff
;
274 /* Align up to a page boundary for the program segments. */
275 *off
= round_page(*off
);
280 struct phdr_closure phc
;
283 * Fill in the ELF header.
285 ehdr
= (Elf_Ehdr
*)((char *)dst
+ ehoff
);
286 ehdr
->e_ident
[EI_MAG0
] = ELFMAG0
;
287 ehdr
->e_ident
[EI_MAG1
] = ELFMAG1
;
288 ehdr
->e_ident
[EI_MAG2
] = ELFMAG2
;
289 ehdr
->e_ident
[EI_MAG3
] = ELFMAG3
;
290 ehdr
->e_ident
[EI_CLASS
] = ELF_CLASS
;
291 ehdr
->e_ident
[EI_DATA
] = ELF_DATA
;
292 ehdr
->e_ident
[EI_VERSION
] = EV_CURRENT
;
293 ehdr
->e_ident
[EI_OSABI
] = ELFOSABI_FREEBSD
;
294 ehdr
->e_ident
[EI_ABIVERSION
] = 0;
295 ehdr
->e_ident
[EI_PAD
] = 0;
296 ehdr
->e_type
= ET_CORE
;
297 ehdr
->e_machine
= ELF_ARCH
;
298 ehdr
->e_version
= EV_CURRENT
;
300 ehdr
->e_phoff
= phoff
;
302 ehdr
->e_ehsize
= sizeof(Elf_Ehdr
);
303 ehdr
->e_phentsize
= sizeof(Elf_Phdr
);
304 ehdr
->e_phnum
= numsegs
+ 1;
305 ehdr
->e_shentsize
= sizeof(Elf_Shdr
);
307 ehdr
->e_shstrndx
= SHN_UNDEF
;
310 * Fill in the program header entries.
312 phdr
= (Elf_Phdr
*)((char *)dst
+ phoff
);
314 /* The note segement. */
315 phdr
->p_type
= PT_NOTE
;
316 phdr
->p_offset
= noteoff
;
319 phdr
->p_filesz
= notesz
;
325 /* All the writable segments from the program. */
328 each_writable_segment(map
, cb_put_phdr
, &phc
);
333 * Emit one note section to "dst", or just size it if "dst" is NULL.
336 elf_putnote(void *dst
, size_t *off
, const char *name
, int type
,
337 const void *desc
, size_t descsz
)
341 note
.n_namesz
= strlen(name
) + 1;
342 note
.n_descsz
= descsz
;
345 bcopy(¬e
, (char *)dst
+ *off
, sizeof note
);
348 bcopy(name
, (char *)dst
+ *off
, note
.n_namesz
);
349 *off
+= roundup2(note
.n_namesz
, sizeof(Elf_Size
));
351 bcopy(desc
, (char *)dst
+ *off
, note
.n_descsz
);
352 *off
+= roundup2(note
.n_descsz
, sizeof(Elf_Size
));
356 * Free the memory map.
359 freemap(vm_map_entry_t map
)
361 while (map
!= NULL
) {
362 vm_map_entry_t next
= map
->next
;
369 * Read the process information necessary to fill in the core file's header.
372 readhdrinfo(pid_t pid
, prstatus_t
*status
, prfpregset_t
*fpregset
,
381 memset(status
, 0, sizeof *status
);
382 status
->pr_version
= PRSTATUS_VERSION
;
383 status
->pr_statussz
= sizeof(prstatus_t
);
384 status
->pr_gregsetsz
= sizeof(gregset_t
);
385 status
->pr_fpregsetsz
= sizeof(fpregset_t
);
386 status
->pr_osreldate
= __DragonFly_version
;
387 status
->pr_pid
= pid
;
389 memset(fpregset
, 0, sizeof *fpregset
);
391 memset(psinfo
, 0, sizeof *psinfo
);
392 psinfo
->pr_version
= PRPSINFO_VERSION
;
393 psinfo
->pr_psinfosz
= sizeof(prpsinfo_t
);
395 /* Read the general registers. */
396 snprintf(name
, sizeof name
, "/proc/%d/regs", pid
);
397 if ((fd
= open(name
, O_RDONLY
)) == -1)
398 err(1, "cannot open %s", name
);
399 if ((n
= read(fd
, &status
->pr_reg
, sizeof status
->pr_reg
)) == -1)
400 err(1, "read error from %s", name
);
401 if (n
< sizeof status
->pr_reg
)
402 errx(1, "short read from %s: wanted %u, got %d", name
,
403 sizeof status
->pr_reg
, n
);
406 /* Read the floating point registers. */
407 snprintf(name
, sizeof name
, "/proc/%d/fpregs", pid
);
408 if ((fd
= open(name
, O_RDONLY
)) == -1)
409 err(1, "cannot open %s", name
);
410 if ((n
= read(fd
, fpregset
, sizeof *fpregset
)) == -1)
411 err(1, "read error from %s", name
);
412 if (n
< sizeof *fpregset
)
413 errx(1, "short read from %s: wanted %u, got %d", name
,
414 sizeof *fpregset
, n
);
417 /* Read and parse the process status. */
418 snprintf(name
, sizeof name
, "/proc/%d/status", pid
);
419 if ((fd
= open(name
, O_RDONLY
)) == -1)
420 err(1, "cannot open %s", name
);
421 if ((n
= read(fd
, line
, sizeof line
- 1)) == -1)
422 err(1, "read error from %s", name
);
425 for (i
= 0; i
< n
&& line
[i
] != ' '; i
++)
426 psinfo
->pr_fname
[i
] = line
[i
];
427 strncpy(psinfo
->pr_psargs
, psinfo
->pr_fname
, PRARGSZ
);
432 * Read the process's memory map using procfs, and return a list of
433 * VM map entries. Only the non-device read/writable segments are
434 * returned. The map entries in the list aren't fully filled in; only
435 * the items we need are present.
437 static vm_map_entry_t
447 vm_map_entry_t
*linkp
;
449 snprintf(mapname
, sizeof mapname
, "/proc/%d/map", pid
);
450 if ((mapfd
= open(mapname
, O_RDONLY
)) == -1)
451 err(1, "cannot open %s", mapname
);
454 * Procfs requires (for consistency) that the entire memory map
455 * be read with a single read() call. Start with a reasonbly sized
456 * buffer, and double it until it is big enough.
461 if ((mapbuf
= realloc(mapbuf
, bufsize
+ 1)) == NULL
)
462 errx(1, "out of memory");
463 mapsize
= read(mapfd
, mapbuf
, bufsize
);
464 if (mapsize
!= -1 || errno
!= EFBIG
)
467 /* This lseek shouldn't be necessary, but it is. */
468 lseek(mapfd
, (off_t
)0, SEEK_SET
);
471 err(1, "read error from %s", mapname
);
473 errx(1, "empty map file %s", mapname
);
480 while (pos
< mapsize
) {
490 n
= sscanf(mapbuf
+ pos
, "%x %x %*d %*d %*x %3[-rwx]"
491 " %*d %*d %*x %*s %*s %16s%*[\n]%n",
492 &start
, &end
, prot
, type
, &len
);
494 errx(1, "ill-formed line in %s", mapname
);
497 /* Ignore segments of the wrong kind, and unwritable ones */
498 if (strncmp(prot
, "rw", 2) != 0 ||
499 (strcmp(type
, "default") != 0 &&
500 strcmp(type
, "vnode") != 0 &&
501 strcmp(type
, "swap") != 0))
504 if ((ent
= (vm_map_entry_t
)calloc(1, sizeof *ent
)) == NULL
)
505 errx(1, "out of memory");
508 ent
->protection
= VM_PROT_READ
| VM_PROT_WRITE
;
510 ent
->protection
|= VM_PROT_EXECUTE
;