2 /*--------------------------------------------------------------------*/
3 /*--- Dumping core. coredump-elf.c ---*/
4 /*--------------------------------------------------------------------*/
7 This file is part of Valgrind, a dynamic binary instrumentation
10 Copyright (C) 2000-2017 Julian Seward
13 This program is free software; you can redistribute it and/or
14 modify it under the terms of the GNU General Public License as
15 published by the Free Software Foundation; either version 2 of the
16 License, or (at your option) any later version.
18 This program is distributed in the hope that it will be useful, but
19 WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
21 General Public License for more details.
23 You should have received a copy of the GNU General Public License
24 along with this program; if not, write to the Free Software
25 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
28 The GNU General Public License is contained in the file COPYING.
31 #if defined(VGO_linux)
33 #include "pub_core_basics.h"
34 #include "pub_core_vki.h"
35 #include "pub_core_aspacehl.h"
36 #include "pub_core_aspacemgr.h"
37 #include "pub_core_libcbase.h"
38 #include "pub_core_machine.h"
39 #include "pub_core_coredump.h"
40 #include "pub_core_libcprint.h"
41 #include "pub_core_libcfile.h" // VG_(close) et al
42 #include "pub_core_libcproc.h" // VG_(geteuid), VG_(getegid)
43 #include "pub_core_libcassert.h" // VG_(exit), vg_assert
44 #include "pub_core_mallocfree.h" // VG_(malloc), VG_(free)
45 #include "pub_core_threadstate.h"
46 #include "pub_core_xarray.h"
47 #include "pub_core_clientstate.h"
48 #include "pub_core_options.h"
53 Generate a standard ELF core file corresponding to the client state
54 at the time of a crash.
58 #define NT_PRXFPREG 0x46e62b7f /* copied from gdb5.1/include/elf/common.h */
59 #endif /* NT_PRXFPREG */
62 #define ESZ(x) Elf64_##x
63 #elif VG_WORDSIZE == 4
64 #define ESZ(x) Elf32_##x
66 #error VG_WORDSIZE needs to ==4 or ==8
69 /* If true, then this Segment may be mentioned in the core */
70 static Bool
may_dump(const NSegment
*seg
)
72 if (seg
->kind
== SkAnonC
||
73 seg
->kind
== SkShmC
||
74 (seg
->kind
== SkFileC
&&
75 !VKI_S_ISCHR(seg
->mode
) && !VKI_S_ISBLK(seg
->mode
)))
81 /* If true, then this Segment's contents will be in the core */
82 static Bool
should_dump(const NSegment
*seg
)
84 return may_dump(seg
); // && seg->hasW;
87 static void fill_ehdr(ESZ(Ehdr
) *ehdr
, Int num_phdrs
)
89 VG_(memset
)(ehdr
, 0, sizeof(*ehdr
));
91 VG_(memcpy
)(ehdr
->e_ident
, ELFMAG
, SELFMAG
);
92 ehdr
->e_ident
[EI_CLASS
] = VG_ELF_CLASS
;
93 ehdr
->e_ident
[EI_DATA
] = VG_ELF_DATA2XXX
;
94 ehdr
->e_ident
[EI_VERSION
] = EV_CURRENT
;
96 ehdr
->e_type
= ET_CORE
;
97 ehdr
->e_machine
= VG_ELF_MACHINE
;
98 ehdr
->e_version
= EV_CURRENT
;
100 ehdr
->e_phoff
= sizeof(ESZ(Ehdr
));
103 ehdr
->e_ehsize
= sizeof(ESZ(Ehdr
));
104 ehdr
->e_phentsize
= sizeof(ESZ(Phdr
));
105 ehdr
->e_phnum
= num_phdrs
;
106 ehdr
->e_shentsize
= 0;
108 ehdr
->e_shstrndx
= 0;
112 static void fill_phdr(ESZ(Phdr
) *phdr
, const NSegment
*seg
, UInt off
, Bool write
)
114 SizeT len
= seg
->end
- seg
->start
+ 1;
116 write
= write
&& should_dump(seg
);
118 VG_(memset
)(phdr
, 0, sizeof(*phdr
));
120 phdr
->p_type
= PT_LOAD
;
121 phdr
->p_offset
= off
;
122 phdr
->p_vaddr
= seg
->start
;
124 phdr
->p_filesz
= write
? len
: 0;
129 phdr
->p_flags
|= PF_R
;
131 phdr
->p_flags
|= PF_W
;
133 phdr
->p_flags
|= PF_X
;
135 phdr
->p_align
= VKI_PAGE_SIZE
;
138 #if defined(VGPV_arm_linux_android) || defined(VGPV_x86_linux_android) \
139 || defined(VGPV_mips32_linux_android)
140 /* Android's libc doesn't provide a definition for this. Hence: */
156 static UInt
note_size(const struct note
*n
)
158 return sizeof(ESZ(Nhdr
)) + VG_ROUNDUP(VG_(strlen
)(n
->name
)+1, 4)
159 + VG_ROUNDUP(n
->note
.n_descsz
, 4);
162 #if !defined(VGPV_arm_linux_android) \
163 && !defined(VGPV_x86_linux_android) \
164 && !defined(VGPV_mips32_linux_android) \
165 && !defined(VGPV_arm64_linux_android)
166 static void add_note(struct note
**list
, const HChar
*name
, UInt type
,
167 const void *data
, UInt datasz
)
169 Int namelen
= VG_(strlen
)(name
)+1;
170 Int notelen
= sizeof(struct note
) +
171 VG_ROUNDUP(namelen
, 4) +
172 VG_ROUNDUP(datasz
, 4);
173 struct note
*n
= VG_(malloc
)("coredump-elf.an.1", notelen
);
175 VG_(memset
)(n
, 0, notelen
);
180 n
->note
.n_type
= type
;
181 n
->note
.n_namesz
= namelen
;
182 n
->note
.n_descsz
= datasz
;
184 VG_(memcpy
)(n
->name
, name
, namelen
);
185 VG_(memcpy
)(n
->name
+VG_ROUNDUP(namelen
,4), data
, datasz
);
187 #endif /* !defined(VGPV_*_linux_android) */
189 static void write_note(Int fd
, const struct note
*n
)
191 VG_(write
)(fd
, &n
->note
, note_size(n
));
194 static void fill_prpsinfo(const ThreadState
*tst
,
195 struct vki_elf_prpsinfo
*prpsinfo
)
197 VG_(memset
)(prpsinfo
, 0, sizeof(*prpsinfo
));
199 switch(tst
->status
) {
202 prpsinfo
->pr_sname
= 'R';
206 prpsinfo
->pr_sname
= 'S';
210 prpsinfo
->pr_sname
= 'Z';
215 prpsinfo
->pr_sname
= '?';
219 prpsinfo
->pr_uid
= 0;
220 prpsinfo
->pr_gid
= 0;
222 VG_(client_fname
)(prpsinfo
->pr_fname
, sizeof(prpsinfo
->pr_fname
), False
);
225 static void fill_prstatus(const ThreadState
*tst
,
226 /*OUT*/struct vki_elf_prstatus
*prs
,
227 const vki_siginfo_t
*si
)
229 #if defined(VGP_mips32_linux) || defined(VGP_mips64_linux)
230 vki_elf_greg_t
*regs
;
232 struct vki_user_regs_struct
*regs
;
234 const ThreadArchState
* arch
= &tst
->arch
;
236 VG_(memset
)(prs
, 0, sizeof(*prs
));
238 prs
->pr_info
.si_signo
= si
->si_signo
;
239 prs
->pr_info
.si_code
= si
->si_code
;
240 prs
->pr_info
.si_errno
= 0;
242 prs
->pr_cursig
= si
->si_signo
;
244 prs
->pr_pid
= tst
->os_state
.lwpid
;
246 prs
->pr_pgrp
= VG_(getpgrp
)();
247 prs
->pr_sid
= VG_(getpgrp
)();
249 #if defined(VGP_s390x_linux)
250 /* prs->pr_reg has struct type. Need to take address. */
251 regs
= (struct vki_user_regs_struct
*)&(prs
->pr_reg
);
252 #elif defined(VGP_mips32_linux) || defined(VGP_mips64_linux)
253 regs
= (vki_elf_greg_t
*)prs
->pr_reg
;
255 regs
= (struct vki_user_regs_struct
*)prs
->pr_reg
;
256 vg_assert(sizeof(*regs
) == sizeof(prs
->pr_reg
));
259 #if defined(VGP_x86_linux)
260 regs
->eflags
= LibVEX_GuestX86_get_eflags( &arch
->vex
);
261 regs
->esp
= arch
->vex
.guest_ESP
;
262 regs
->eip
= arch
->vex
.guest_EIP
;
264 regs
->ebx
= arch
->vex
.guest_EBX
;
265 regs
->ecx
= arch
->vex
.guest_ECX
;
266 regs
->edx
= arch
->vex
.guest_EDX
;
267 regs
->esi
= arch
->vex
.guest_ESI
;
268 regs
->edi
= arch
->vex
.guest_EDI
;
269 regs
->ebp
= arch
->vex
.guest_EBP
;
270 regs
->eax
= arch
->vex
.guest_EAX
;
272 regs
->cs
= arch
->vex
.guest_CS
;
273 regs
->ds
= arch
->vex
.guest_DS
;
274 regs
->ss
= arch
->vex
.guest_SS
;
275 regs
->es
= arch
->vex
.guest_ES
;
276 regs
->fs
= arch
->vex
.guest_FS
;
277 regs
->gs
= arch
->vex
.guest_GS
;
279 #elif defined(VGP_amd64_linux)
280 regs
->eflags
= LibVEX_GuestAMD64_get_rflags( &arch
->vex
);
281 regs
->rsp
= arch
->vex
.guest_RSP
;
282 regs
->rip
= arch
->vex
.guest_RIP
;
284 regs
->rbx
= arch
->vex
.guest_RBX
;
285 regs
->rcx
= arch
->vex
.guest_RCX
;
286 regs
->rdx
= arch
->vex
.guest_RDX
;
287 regs
->rsi
= arch
->vex
.guest_RSI
;
288 regs
->rdi
= arch
->vex
.guest_RDI
;
289 regs
->rbp
= arch
->vex
.guest_RBP
;
290 regs
->rax
= arch
->vex
.guest_RAX
;
291 regs
->r8
= arch
->vex
.guest_R8
;
292 regs
->r9
= arch
->vex
.guest_R9
;
293 regs
->r10
= arch
->vex
.guest_R10
;
294 regs
->r11
= arch
->vex
.guest_R11
;
295 regs
->r12
= arch
->vex
.guest_R12
;
296 regs
->r13
= arch
->vex
.guest_R13
;
297 regs
->r14
= arch
->vex
.guest_R14
;
298 regs
->r15
= arch
->vex
.guest_R15
;
300 #elif defined(VGP_ppc32_linux)
301 # define DO(n) regs->gpr[n] = arch->vex.guest_GPR##n
302 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
303 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
304 DO(16); DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23);
305 DO(24); DO(25); DO(26); DO(27); DO(28); DO(29); DO(30); DO(31);
308 regs
->nip
= arch
->vex
.guest_CIA
;
309 regs
->msr
= 0xf032; /* pretty arbitrary */
310 regs
->orig_gpr3
= arch
->vex
.guest_GPR3
;
311 regs
->ctr
= arch
->vex
.guest_CTR
;
312 regs
->link
= arch
->vex
.guest_LR
;
313 regs
->xer
= LibVEX_GuestPPC32_get_XER( &arch
->vex
);
314 regs
->ccr
= LibVEX_GuestPPC32_get_CR( &arch
->vex
);
317 regs
->dar
= 0; /* should be fault address? */
321 #elif defined(VGP_ppc64be_linux)
322 # define DO(n) regs->gpr[n] = arch->vex.guest_GPR##n
323 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
324 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
325 DO(16); DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23);
326 DO(24); DO(25); DO(26); DO(27); DO(28); DO(29); DO(30); DO(31);
329 regs
->nip
= arch
->vex
.guest_CIA
;
330 regs
->msr
= 0xf032; /* pretty arbitrary */
331 regs
->orig_gpr3
= arch
->vex
.guest_GPR3
;
332 regs
->ctr
= arch
->vex
.guest_CTR
;
333 regs
->link
= arch
->vex
.guest_LR
;
334 regs
->xer
= LibVEX_GuestPPC64_get_XER( &arch
->vex
);
335 regs
->ccr
= LibVEX_GuestPPC64_get_CR( &arch
->vex
);
338 regs
->dar
= 0; /* should be fault address? */
342 #elif defined(VGP_ppc64le_linux)
343 # define DO(n) regs->gpr[n] = arch->vex.guest_GPR##n
344 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
345 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
346 DO(16); DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23);
347 DO(24); DO(25); DO(26); DO(27); DO(28); DO(29); DO(30); DO(31);
350 regs
->nip
= arch
->vex
.guest_CIA
;
351 regs
->msr
= 0xf033; /* pretty arbitrary */
352 regs
->orig_gpr3
= arch
->vex
.guest_GPR3
;
353 regs
->ctr
= arch
->vex
.guest_CTR
;
354 regs
->link
= arch
->vex
.guest_LR
;
355 regs
->xer
= LibVEX_GuestPPC64_get_XER( &(arch
->vex
) );
356 regs
->ccr
= LibVEX_GuestPPC64_get_CR( &(arch
->vex
) );
359 regs
->dar
= 0; /* should be fault address? */
363 #elif defined(VGP_arm_linux)
364 regs
->ARM_r0
= arch
->vex
.guest_R0
;
365 regs
->ARM_r1
= arch
->vex
.guest_R1
;
366 regs
->ARM_r2
= arch
->vex
.guest_R2
;
367 regs
->ARM_r3
= arch
->vex
.guest_R3
;
368 regs
->ARM_r4
= arch
->vex
.guest_R4
;
369 regs
->ARM_r5
= arch
->vex
.guest_R5
;
370 regs
->ARM_r6
= arch
->vex
.guest_R6
;
371 regs
->ARM_r7
= arch
->vex
.guest_R7
;
372 regs
->ARM_r8
= arch
->vex
.guest_R8
;
373 regs
->ARM_r9
= arch
->vex
.guest_R9
;
374 regs
->ARM_r10
= arch
->vex
.guest_R10
;
375 regs
->ARM_fp
= arch
->vex
.guest_R11
;
376 regs
->ARM_ip
= arch
->vex
.guest_R12
;
377 regs
->ARM_sp
= arch
->vex
.guest_R13
;
378 regs
->ARM_lr
= arch
->vex
.guest_R14
;
379 regs
->ARM_pc
= arch
->vex
.guest_R15T
;
380 regs
->ARM_cpsr
= LibVEX_GuestARM_get_cpsr( &arch
->vex
);
382 #elif defined(VGP_arm64_linux)
386 #elif defined(VGP_s390x_linux)
387 # define DO(n) regs->gprs[n] = arch->vex.guest_r##n
388 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
389 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
391 # define DO(n) regs->acrs[n] = arch->vex.guest_a##n
392 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
393 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
395 regs
->orig_gpr2
= arch
->vex
.guest_r2
;
397 #elif defined(VGP_mips32_linux)
398 # define DO(n) regs[VKI_MIPS32_EF_R##n] = arch->vex.guest_r##n
399 DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7); DO(8);
400 DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15); DO(16);
401 DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23); DO(24);
402 DO(25); DO(28); DO(29); DO(30); DO(31);
404 regs
[VKI_MIPS32_EF_LO
] = arch
->vex
.guest_LO
;
405 regs
[VKI_MIPS32_EF_HI
] = arch
->vex
.guest_HI
;
406 regs
[VKI_MIPS32_EF_CP0_STATUS
] = arch
->vex
.guest_CP0_status
;
407 regs
[VKI_MIPS32_EF_CP0_EPC
] = arch
->vex
.guest_PC
;
408 #elif defined(VGP_mips64_linux)
409 # define DO(n) regs[VKI_MIPS64_EF_R##n] = arch->vex.guest_r##n
410 DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7); DO(8);
411 DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15); DO(16);
412 DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23); DO(24);
413 DO(25); DO(28); DO(29); DO(30); DO(31);
415 regs
[VKI_MIPS64_EF_LO
] = arch
->vex
.guest_LO
;
416 regs
[VKI_MIPS64_EF_HI
] = arch
->vex
.guest_HI
;
417 regs
[VKI_MIPS64_EF_CP0_STATUS
] = arch
->vex
.guest_CP0_status
;
418 regs
[VKI_MIPS64_EF_CP0_EPC
] = arch
->vex
.guest_PC
;
420 # error Unknown ELF platform
424 static void fill_fpu(const ThreadState
*tst
, vki_elf_fpregset_t
*fpu
)
426 __attribute__((unused
))
427 const ThreadArchState
* arch
= &tst
->arch
;
429 #if defined(VGP_x86_linux)
430 //:: static void fill_fpu(vki_elf_fpregset_t *fpu, const HChar *from)
432 //:: if (VG_(have_ssestate)) {
436 //:: /* This is what the kernel does */
437 //:: VG_(memcpy)(fpu, from, 7*sizeof(long));
439 //:: to = (UShort *)&fpu->st_space[0];
440 //:: from += 18 * sizeof(UShort);
442 //:: for (i = 0; i < 8; i++, to += 5, from += 8)
443 //:: VG_(memcpy)(to, from, 5*sizeof(UShort));
445 //:: VG_(memcpy)(fpu, from, sizeof(*fpu));
448 //:: fill_fpu(fpu, (const HChar *)&arch->m_sse);
450 #elif defined(VGP_amd64_linux)
458 //:: fpu->mxcsr_mask = ?;
459 //:: fpu->st_space = ?;
461 # define DO(n) VG_(memcpy)(fpu->xmm_space + n * 4, \
462 &arch->vex.guest_YMM##n[0], 16)
463 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
464 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
467 VG_(memset
)(fpu
->padding
, 0, sizeof(fpu
->padding
));
469 #elif defined(VGP_ppc32_linux)
470 /* The guest state has the FPR fields declared as ULongs, so need
471 to fish out the values without converting them.
472 NOTE: The 32 FP registers map to the first 32 VSX registers.*/
473 # define DO(n) (*fpu)[n] = *(double*)(&arch->vex.guest_VSR##n)
474 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
475 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
476 DO(16); DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23);
477 DO(24); DO(25); DO(26); DO(27); DO(28); DO(29); DO(30); DO(31);
480 #elif defined(VGP_ppc64be_linux) || defined(VGP_ppc64le_linux)
481 /* The guest state has the FPR fields declared as ULongs, so need
482 to fish out the values without converting them.
483 NOTE: The 32 FP registers map to the first 32 VSX registers.*/
484 # define DO(n) (*fpu)[n] = *(double*)(&arch->vex.guest_VSR##n)
485 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
486 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
487 DO(16); DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23);
488 DO(24); DO(25); DO(26); DO(27); DO(28); DO(29); DO(30); DO(31);
491 #elif defined(VGP_arm_linux)
494 #elif defined(VGP_arm64_linux)
497 #elif defined(VGP_s390x_linux)
498 /* NOTE: The 16 FP registers map to the first 16 VSX registers. */
499 # define DO(n) fpu->fprs[n].ui = *(const Double*)(&arch->vex.guest_v##n.w64[0])
500 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
501 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
503 #elif defined(VGP_mips32_linux) || defined(VGP_mips64_linux)
504 # define DO(n) (*fpu)[n] = *(const double*)(&arch->vex.guest_f##n)
505 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
506 DO(8); DO(9); DO(10); DO(11); DO(12); DO(13); DO(14); DO(15);
507 DO(16); DO(17); DO(18); DO(19); DO(20); DO(21); DO(22); DO(23);
508 DO(24); DO(25); DO(26); DO(27); DO(28); DO(29); DO(30); DO(31);
511 # error Unknown ELF platform
515 #if defined(VGP_x86_linux) && !defined(VGPV_x86_linux_android)
516 static void fill_xfpu(const ThreadState
*tst
, vki_elf_fpxregset_t
*xfpu
)
518 const ThreadArchState
* arch
= &tst
->arch
;
528 //:: xfpu->mxcsr = ?;
530 //:: xfpu->st_space = ?;
532 # define DO(n) VG_(memcpy)(xfpu->xmm_space + n * 4, &arch->vex.guest_XMM##n, sizeof(arch->vex.guest_XMM##n))
533 DO(0); DO(1); DO(2); DO(3); DO(4); DO(5); DO(6); DO(7);
536 VG_(memset
)(xfpu
->padding
, 0, sizeof(xfpu
->padding
));
541 void dump_one_thread(struct note
**notelist
, const vki_siginfo_t
*si
, ThreadId tid
)
543 vki_elf_fpregset_t fpu
;
544 struct vki_elf_prstatus prstatus
;
545 VG_(memset
)(&fpu
, 0, sizeof(fpu
));
546 VG_(memset
)(&prstatus
, 0, sizeof(prstatus
));
547 # if defined(VGP_x86_linux) && !defined(VGPV_x86_linux_android)
549 vki_elf_fpxregset_t xfpu
;
550 VG_(memset
)(&xfpu
, 0, sizeof(xfpu
));
551 fill_xfpu(&VG_(threads
)[tid
], &xfpu
);
552 add_note(notelist
, "LINUX", NT_PRXFPREG
, &xfpu
, sizeof(xfpu
));
556 fill_fpu(&VG_(threads
)[tid
], &fpu
);
557 # if !defined(VGPV_arm_linux_android) \
558 && !defined(VGPV_x86_linux_android) \
559 && !defined(VGPV_mips32_linux_android) \
560 && !defined(VGPV_arm64_linux_android)
561 add_note(notelist
, "CORE", NT_FPREGSET
, &fpu
, sizeof(fpu
));
564 fill_prstatus(&VG_(threads
)[tid
], &prstatus
, si
);
565 # if !defined(VGPV_arm_linux_android) \
566 && !defined(VGPV_x86_linux_android) \
567 && !defined(VGPV_mips32_linux_android) \
568 && !defined(VGPV_arm64_linux_android)
569 add_note(notelist
, "CORE", NT_PRSTATUS
, &prstatus
, sizeof(prstatus
));
574 void make_elf_coredump(ThreadId tid
, const vki_siginfo_t
*si
, ULong max_size
)
577 const HChar
*basename
= "vgcore";
578 const HChar
*coreext
= "";
581 NSegment
const * seg
;
587 struct note
*notelist
, *note
;
589 struct vki_elf_prpsinfo prpsinfo
;
593 if (VG_(clo_log_fname_unexpanded
) != NULL
) {
595 basename
= VG_(expand_file_name
)("--log-file",
596 VG_(clo_log_fname_unexpanded
));
601 buf
= VG_(malloc
)( "coredump-elf.mec.1",
602 VG_(strlen
)(coreext
) + VG_(strlen
)(basename
)
603 + 100/*for the two %ds. */ );
606 Int oflags
= VKI_O_CREAT
|VKI_O_WRONLY
|VKI_O_EXCL
|VKI_O_TRUNC
;
610 VG_(sprintf
)(buf
, "%s%s.%d",
611 basename
, coreext
, VG_(getpid
)());
613 VG_(sprintf
)(buf
, "%s%s.%d.%d",
614 basename
, coreext
, VG_(getpid
)(), seq
);
617 # if defined(VKI_O_LARGEFILE)
618 oflags
|= VKI_O_LARGEFILE
;
621 sres
= VG_(open
)(buf
, oflags
, VKI_S_IRUSR
|VKI_S_IWUSR
);
622 if (!sr_isError(sres
)) {
623 core_fd
= sr_Res(sres
);
627 if (sr_isError(sres
) && sr_Err(sres
) != VKI_EEXIST
)
628 return; /* can't create file */
631 /* Get the client segments */
632 seg_starts
= VG_(get_segment_starts
)(SkFileC
| SkAnonC
| SkShmC
,
635 /* First, count how many memory segments to dump */
636 num_phdrs
= 1; /* start with notes */
637 for(i
= 0; i
< n_seg_starts
; i
++) {
638 if (!may_dump(VG_(am_find_nsegment
)(seg_starts
[i
])))
644 fill_ehdr(&ehdr
, num_phdrs
);
648 /* Second, work out their layout */
649 phdrs
= VG_(malloc
)("coredump-elf.mec.1", sizeof(*phdrs
) * num_phdrs
);
651 /* Add details for all threads except the one that faulted */
652 for(i
= 1; i
< VG_N_THREADS
; i
++) {
654 if (VG_(threads
)[i
].status
== VgTs_Empty
)
660 dump_one_thread(¬elist
, si
, i
);
663 /* Add details for the faulting thread. Note that because we are
664 adding to the head of a linked list this thread will actually
665 come out first in the core file, which seems to be how
666 debuggers determine that it is the faulting thread. */
667 dump_one_thread(¬elist
, si
, tid
);
669 fill_prpsinfo(&VG_(threads
)[tid
], &prpsinfo
);
670 # if !defined(VGPV_arm_linux_android) \
671 && !defined(VGPV_x86_linux_android) \
672 && !defined(VGPV_mips32_linux_android) \
673 && !defined(VGPV_arm64_linux_android)
674 add_note(¬elist
, "CORE", NT_PRPSINFO
, &prpsinfo
, sizeof(prpsinfo
));
677 for (note
= notelist
, notesz
= 0; note
!= NULL
; note
= note
->next
)
678 notesz
+= note_size(note
);
680 off
= sizeof(ehdr
) + sizeof(*phdrs
) * num_phdrs
;
682 phdrs
[0].p_type
= PT_NOTE
;
683 phdrs
[0].p_offset
= off
;
684 phdrs
[0].p_vaddr
= 0;
685 phdrs
[0].p_paddr
= 0;
686 phdrs
[0].p_filesz
= notesz
;
687 phdrs
[0].p_memsz
= 0;
688 phdrs
[0].p_flags
= 0;
689 phdrs
[0].p_align
= 0;
693 off
= VG_PGROUNDUP(off
);
695 for(i
= 0, idx
= 1; i
< n_seg_starts
; i
++) {
696 seg
= VG_(am_find_nsegment
)(seg_starts
[i
]);
701 fill_phdr(&phdrs
[idx
], seg
, off
,
702 (seg
->end
- seg
->start
+ 1 + off
) < max_size
);
704 off
+= phdrs
[idx
].p_filesz
;
709 /* write everything out */
710 VG_(write
)(core_fd
, &ehdr
, sizeof(ehdr
));
711 VG_(write
)(core_fd
, phdrs
, sizeof(*phdrs
) * num_phdrs
);
713 for(note
= notelist
; note
!= NULL
; note
= note
->next
)
714 write_note(core_fd
, note
);
716 VG_(lseek
)(core_fd
, phdrs
[1].p_offset
, VKI_SEEK_SET
);
718 for(i
= 0, idx
= 1; i
< n_seg_starts
; i
++) {
719 seg
= VG_(am_find_nsegment
)(seg_starts
[i
]);
721 if (!should_dump(seg
))
724 if (phdrs
[idx
].p_filesz
> 0) {
725 vg_assert(VG_(lseek
)(core_fd
, phdrs
[idx
].p_offset
, VKI_SEEK_SET
)
726 == phdrs
[idx
].p_offset
);
727 vg_assert(seg
->end
- seg
->start
+ 1 >= phdrs
[idx
].p_filesz
);
729 (void)VG_(write
)(core_fd
, (void *)seg
->start
, phdrs
[idx
].p_filesz
);
734 VG_(free
)(seg_starts
);
739 void VG_(make_coredump
)(ThreadId tid
, const vki_siginfo_t
*si
, ULong max_size
)
741 make_elf_coredump(tid
, si
, max_size
);
744 #endif // defined(VGO_linux)
746 /*--------------------------------------------------------------------*/
748 /*--------------------------------------------------------------------*/