d773820bd9
'fast path' size on the first path matches the actual size on the second) save all the notes (mostly the cpu registers for all the LWPs) in malloced memory on the first pass. Sanity check that the number of memory segments matches written matches the count obtained earlier. If gcore() is used they could differ. (Not sure that returning ENOMEM is ideal, but it is better than a crash.)
512 lines
13 KiB
C
512 lines
13 KiB
C
/* $NetBSD: core_elf32.c,v 1.39 2014/01/03 20:52:47 dsl Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2001 Wasabi Systems, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Written by Jason R. Thorpe for Wasabi Systems, Inc.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed for the NetBSD Project by
|
|
* Wasabi Systems, Inc.
|
|
* 4. The name of Wasabi Systems, Inc. may not be used to endorse
|
|
* or promote products derived from this software without specific prior
|
|
* written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/*
|
|
* core_elf32.c/core_elf64.c: Support for the Elf32/Elf64 core file format.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__KERNEL_RCSID(1, "$NetBSD: core_elf32.c,v 1.39 2014/01/03 20:52:47 dsl Exp $");
|
|
|
|
#ifdef _KERNEL_OPT
|
|
#include "opt_coredump.h"
|
|
#endif
|
|
|
|
#ifndef ELFSIZE
|
|
#define ELFSIZE 32
|
|
#endif
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/vnode.h>
|
|
#include <sys/exec.h>
|
|
#include <sys/exec_elf.h>
|
|
#include <sys/ptrace.h>
|
|
#include <sys/kmem.h>
|
|
#include <sys/kauth.h>
|
|
|
|
#include <machine/reg.h>
|
|
|
|
#include <uvm/uvm_extern.h>
|
|
|
|
#ifdef COREDUMP
|
|
|
|
struct writesegs_state {
|
|
Elf_Phdr *psections;
|
|
off_t secoff;
|
|
size_t npsections;
|
|
};
|
|
|
|
/*
|
|
* We need to know how big the 'notes' are before we write the main header.
|
|
* To avoid problems with double-processing we save the data.
|
|
*/
|
|
struct note_buf {
|
|
struct note_buf *nb_next;
|
|
unsigned char nb_data[4096 - sizeof (void *)];
|
|
};
|
|
|
|
struct note_state {
|
|
struct note_buf *ns_first;
|
|
struct note_buf *ns_last;
|
|
unsigned int ns_count; /* Of full buffers */
|
|
unsigned int ns_offset; /* Write point in last buffer */
|
|
};
|
|
|
|
static int ELFNAMEEND(coredump_getseghdrs)(struct proc *,
|
|
struct uvm_coredump_state *);
|
|
|
|
static int ELFNAMEEND(coredump_notes)(struct lwp *, struct note_state *);
|
|
static int ELFNAMEEND(coredump_note)(struct lwp *, struct note_state *);
|
|
|
|
/* The 'note' section names and data are always 4-byte aligned. */
|
|
#define ELFROUNDSIZE 4 /* XXX Should it be sizeof(Elf_Word)? */
|
|
|
|
#define elf_process_read_regs CONCAT(process_read_regs, ELFSIZE)
|
|
#ifdef __HAVE_PROCESS_XFPREGS
|
|
#define elf_process_read_xfpregs CONCAT(process_read_xfpregs, ELFSIZE)
|
|
#else
|
|
#define elf_process_read_fpregs CONCAT(process_read_fpregs, ELFSIZE)
|
|
#endif
|
|
#define elf_reg CONCAT(process_reg, ELFSIZE)
|
|
#define elf_fpreg CONCAT(process_fpreg, ELFSIZE)
|
|
|
|
int
|
|
ELFNAMEEND(coredump)(struct lwp *l, struct coredump_iostate *cookie)
|
|
{
|
|
Elf_Ehdr ehdr;
|
|
Elf_Phdr *psections;
|
|
size_t psectionssize;
|
|
int npsections;
|
|
struct writesegs_state ws;
|
|
off_t notestart;
|
|
size_t notesize;
|
|
int error, i;
|
|
|
|
struct note_state ns;
|
|
struct note_buf *nb;
|
|
|
|
psections = NULL;
|
|
|
|
/* Get all of the notes (mostly all the registers). */
|
|
ns.ns_first = kmem_alloc(sizeof *ns.ns_first, KM_SLEEP);
|
|
ns.ns_last = ns.ns_first;
|
|
ns.ns_count = 0;
|
|
ns.ns_offset = 0;
|
|
error = ELFNAMEEND(coredump_notes)(l, &ns);
|
|
ns.ns_last->nb_next = NULL;
|
|
if (error)
|
|
goto out;
|
|
notesize = ns.ns_count * sizeof nb->nb_data + ns.ns_offset;
|
|
|
|
/*
|
|
* We have to make a total of 3 passes across the map:
|
|
*
|
|
* 1. Count the number of map entries (the number of
|
|
* PT_LOAD sections in the dump).
|
|
*
|
|
* 2. Write the P-section headers.
|
|
*
|
|
* 3. Write the P-sections.
|
|
*/
|
|
|
|
/* Pass 1: count the entries. */
|
|
npsections = uvm_coredump_count_segs(l->l_proc);
|
|
/* Allow for the PT_NOTE section. */
|
|
npsections++;
|
|
|
|
/* Build the main elf header */
|
|
memset(&ehdr.e_ident[EI_PAD], 0, sizeof(ehdr.e_ident) - EI_PAD);
|
|
memcpy(ehdr.e_ident, ELFMAG, SELFMAG);
|
|
#if ELFSIZE == 32
|
|
ehdr.e_ident[EI_CLASS] = ELFCLASS32;
|
|
#elif ELFSIZE == 64
|
|
ehdr.e_ident[EI_CLASS] = ELFCLASS64;
|
|
#endif
|
|
ehdr.e_ident[EI_DATA] = ELFDEFNNAME(MACHDEP_ENDIANNESS);
|
|
ehdr.e_ident[EI_VERSION] = EV_CURRENT;
|
|
/* XXX Should be the OSABI/ABI version of the executable. */
|
|
ehdr.e_ident[EI_OSABI] = ELFOSABI_SYSV;
|
|
ehdr.e_ident[EI_ABIVERSION] = 0;
|
|
|
|
ehdr.e_type = ET_CORE;
|
|
/* XXX This should be the e_machine of the executable. */
|
|
ehdr.e_machine = ELFDEFNNAME(MACHDEP_ID);
|
|
ehdr.e_version = EV_CURRENT;
|
|
ehdr.e_entry = 0;
|
|
ehdr.e_phoff = sizeof(ehdr);
|
|
ehdr.e_shoff = 0;
|
|
ehdr.e_flags = 0;
|
|
ehdr.e_ehsize = sizeof(ehdr);
|
|
ehdr.e_phentsize = sizeof(Elf_Phdr);
|
|
ehdr.e_phnum = npsections;
|
|
ehdr.e_shentsize = 0;
|
|
ehdr.e_shnum = 0;
|
|
ehdr.e_shstrndx = 0;
|
|
|
|
#ifdef ELF_MD_COREDUMP_SETUP
|
|
ELF_MD_COREDUMP_SETUP(l, &ehdr);
|
|
#endif
|
|
|
|
/* Write out the ELF header. */
|
|
error = coredump_write(cookie, UIO_SYSSPACE, &ehdr, sizeof(ehdr));
|
|
if (error)
|
|
goto out;
|
|
|
|
psectionssize = npsections * sizeof(*psections);
|
|
notestart = sizeof(ehdr) + psectionssize;
|
|
|
|
psections = kmem_zalloc(psectionssize, KM_SLEEP);
|
|
|
|
/* Pass 2: now find the P-section headers. */
|
|
ws.secoff = notestart + notesize;
|
|
ws.psections = psections;
|
|
ws.npsections = npsections - 1;
|
|
error = uvm_coredump_walkmap(l->l_proc, ELFNAMEEND(coredump_getseghdrs),
|
|
&ws);
|
|
if (error)
|
|
goto out;
|
|
if (ws.npsections != 0) {
|
|
/* A section went away */
|
|
error = ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
/* Add the PT_NOTE header after the P-section headers. */
|
|
ws.psections->p_type = PT_NOTE;
|
|
ws.psections->p_offset = notestart;
|
|
ws.psections->p_vaddr = 0;
|
|
ws.psections->p_paddr = 0;
|
|
ws.psections->p_filesz = notesize;
|
|
ws.psections->p_memsz = 0;
|
|
ws.psections->p_flags = PF_R;
|
|
ws.psections->p_align = ELFROUNDSIZE;
|
|
|
|
/* Write the P-section headers followed by the PT_NOTE header */
|
|
error = coredump_write(cookie, UIO_SYSSPACE, psections, psectionssize);
|
|
if (error)
|
|
goto out;
|
|
|
|
#ifdef DIAGNOSTIC
|
|
if (coredump_offset(cookie) != notestart)
|
|
panic("coredump: offset %lld != notestart %lld",
|
|
(long long) coredump_offset(cookie),
|
|
(long long) notestart);
|
|
#endif
|
|
|
|
/* Write out the notes. */
|
|
for (nb = ns.ns_first; nb != NULL; nb = nb->nb_next) {
|
|
error = coredump_write(cookie, UIO_SYSSPACE, nb->nb_data,
|
|
nb->nb_next == NULL ? ns.ns_offset : sizeof nb->nb_data);
|
|
if (error)
|
|
goto out;
|
|
}
|
|
|
|
/* Finally, write the sections themselves. */
|
|
for (i = 0; i < npsections - 1; i++) {
|
|
if (psections[i].p_filesz == 0)
|
|
continue;
|
|
|
|
#ifdef DIAGNOSTIC
|
|
if (coredump_offset(cookie) != psections[i].p_offset)
|
|
panic("coredump: offset %lld != p_offset[%d] %lld",
|
|
(long long) coredump_offset(cookie), i,
|
|
(long long) psections[i].p_filesz);
|
|
#endif
|
|
|
|
error = coredump_write(cookie, UIO_USERSPACE,
|
|
(void *)(vaddr_t)psections[i].p_vaddr,
|
|
psections[i].p_filesz);
|
|
if (error)
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
if (psections)
|
|
kmem_free(psections, psectionssize);
|
|
for (; (nb = ns.ns_first) != NULL; ns.ns_first = nb->nb_next)
|
|
kmem_free(nb, sizeof *nb);
|
|
return (error);
|
|
}
|
|
|
|
static int
|
|
ELFNAMEEND(coredump_getseghdrs)(struct proc *p, struct uvm_coredump_state *us)
|
|
{
|
|
struct writesegs_state *ws = us->cookie;
|
|
Elf_Phdr phdr;
|
|
vsize_t size, realsize;
|
|
vaddr_t end;
|
|
int error;
|
|
|
|
/* Don't overrun if there are more sections */
|
|
if (ws->npsections == 0)
|
|
return ENOMEM;
|
|
ws->npsections--;
|
|
|
|
size = us->end - us->start;
|
|
realsize = us->realend - us->start;
|
|
end = us->realend;
|
|
|
|
/* Don't bother writing out trailing zeros */
|
|
while (realsize > 0) {
|
|
long buf[1024 / sizeof(long)];
|
|
size_t slen = realsize > sizeof(buf) ? sizeof(buf) : realsize;
|
|
const long *ep;
|
|
int i;
|
|
|
|
end -= slen;
|
|
if ((error = copyin_proc(p, (void *)end, buf, slen)) != 0)
|
|
return error;
|
|
|
|
ep = (const long *) &buf[slen / sizeof(buf[0])];
|
|
for (i = 0, ep--; buf <= ep; ep--, i++) {
|
|
if (*ep)
|
|
break;
|
|
}
|
|
realsize -= i * sizeof(buf[0]);
|
|
if (i * sizeof(buf[0]) < slen)
|
|
break;
|
|
}
|
|
|
|
phdr.p_type = PT_LOAD;
|
|
phdr.p_offset = ws->secoff;
|
|
phdr.p_vaddr = us->start;
|
|
phdr.p_paddr = 0;
|
|
phdr.p_filesz = realsize;
|
|
phdr.p_memsz = size;
|
|
phdr.p_flags = 0;
|
|
if (us->prot & VM_PROT_READ)
|
|
phdr.p_flags |= PF_R;
|
|
if (us->prot & VM_PROT_WRITE)
|
|
phdr.p_flags |= PF_W;
|
|
if (us->prot & VM_PROT_EXECUTE)
|
|
phdr.p_flags |= PF_X;
|
|
phdr.p_align = PAGE_SIZE;
|
|
|
|
ws->secoff += phdr.p_filesz;
|
|
*ws->psections++ = phdr;
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
ELFNAMEEND(coredump_notes)(struct lwp *l, struct note_state *ns)
|
|
{
|
|
struct proc *p;
|
|
struct netbsd_elfcore_procinfo cpi;
|
|
Elf_Nhdr nhdr;
|
|
int error;
|
|
struct lwp *l0;
|
|
sigset_t ss1, ss2;
|
|
|
|
p = l->l_proc;
|
|
|
|
/* First, write an elfcore_procinfo. */
|
|
cpi.cpi_version = NETBSD_ELFCORE_PROCINFO_VERSION;
|
|
cpi.cpi_cpisize = sizeof(cpi);
|
|
cpi.cpi_signo = p->p_sigctx.ps_signo;
|
|
cpi.cpi_sigcode = p->p_sigctx.ps_code;
|
|
cpi.cpi_siglwp = p->p_sigctx.ps_lwp;
|
|
|
|
/*
|
|
* XXX This should be per-LWP.
|
|
*/
|
|
ss1 = p->p_sigpend.sp_set;
|
|
sigemptyset(&ss2);
|
|
LIST_FOREACH(l0, &p->p_lwps, l_sibling) {
|
|
sigplusset(&l0->l_sigpend.sp_set, &ss1);
|
|
sigplusset(&l0->l_sigmask, &ss2);
|
|
}
|
|
memcpy(&cpi.cpi_sigpend, &ss1, sizeof(cpi.cpi_sigpend));
|
|
memcpy(&cpi.cpi_sigmask, &ss2, sizeof(cpi.cpi_sigmask));
|
|
memcpy(&cpi.cpi_sigignore, &p->p_sigctx.ps_sigignore,
|
|
sizeof(cpi.cpi_sigignore));
|
|
memcpy(&cpi.cpi_sigcatch, &p->p_sigctx.ps_sigcatch,
|
|
sizeof(cpi.cpi_sigcatch));
|
|
|
|
cpi.cpi_pid = p->p_pid;
|
|
mutex_enter(proc_lock);
|
|
cpi.cpi_ppid = p->p_pptr->p_pid;
|
|
cpi.cpi_pgrp = p->p_pgid;
|
|
cpi.cpi_sid = p->p_session->s_sid;
|
|
mutex_exit(proc_lock);
|
|
|
|
cpi.cpi_ruid = kauth_cred_getuid(l->l_cred);
|
|
cpi.cpi_euid = kauth_cred_geteuid(l->l_cred);
|
|
cpi.cpi_svuid = kauth_cred_getsvuid(l->l_cred);
|
|
|
|
cpi.cpi_rgid = kauth_cred_getgid(l->l_cred);
|
|
cpi.cpi_egid = kauth_cred_getegid(l->l_cred);
|
|
cpi.cpi_svgid = kauth_cred_getsvgid(l->l_cred);
|
|
|
|
cpi.cpi_nlwps = p->p_nlwps;
|
|
(void)strncpy(cpi.cpi_name, p->p_comm, sizeof(cpi.cpi_name));
|
|
cpi.cpi_name[sizeof(cpi.cpi_name) - 1] = '\0';
|
|
|
|
nhdr.n_namesz = sizeof(ELF_NOTE_NETBSD_CORE_NAME);
|
|
nhdr.n_descsz = sizeof(cpi);
|
|
nhdr.n_type = ELF_NOTE_NETBSD_CORE_PROCINFO;
|
|
|
|
ELFNAMEEND(coredump_savenote)(ns, &nhdr, ELF_NOTE_NETBSD_CORE_NAME,
|
|
&cpi);
|
|
|
|
/* XXX Add hook for machdep per-proc notes. */
|
|
|
|
/*
|
|
* Now write the register info for the thread that caused the
|
|
* coredump.
|
|
*/
|
|
error = ELFNAMEEND(coredump_note)(l, ns);
|
|
if (error)
|
|
return (error);
|
|
|
|
/*
|
|
* Now, for each LWP, write the register info and any other
|
|
* per-LWP notes. Since we're dumping core, we don't bother
|
|
* locking.
|
|
*/
|
|
LIST_FOREACH(l0, &p->p_lwps, l_sibling) {
|
|
if (l0 == l) /* we've taken care of this thread */
|
|
continue;
|
|
error = ELFNAMEEND(coredump_note)(l0, ns);
|
|
if (error)
|
|
return (error);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
ELFNAMEEND(coredump_note)(struct lwp *l, struct note_state *ns)
|
|
{
|
|
Elf_Nhdr nhdr;
|
|
int error;
|
|
int namesize;
|
|
char name[64];
|
|
elf_reg intreg;
|
|
#ifdef PT_GETFPREGS
|
|
elf_fpreg freg;
|
|
size_t freglen;
|
|
#endif
|
|
|
|
snprintf(name, sizeof(name), "%s@%d",
|
|
ELF_NOTE_NETBSD_CORE_NAME, l->l_lid);
|
|
namesize = strlen(name) + 1;
|
|
|
|
error = elf_process_read_regs(l, &intreg);
|
|
if (error)
|
|
return (error);
|
|
|
|
nhdr.n_namesz = namesize;
|
|
nhdr.n_descsz = sizeof(intreg);
|
|
nhdr.n_type = PT_GETREGS;
|
|
|
|
ELFNAMEEND(coredump_savenote)(ns, &nhdr, name, &intreg);
|
|
|
|
#ifdef PT_GETFPREGS
|
|
freglen = sizeof(freg);
|
|
#ifdef __HAVE_PROCESS_XFPREGS
|
|
error = elf_process_read_xfpregs(l, &freg, &freglen);
|
|
#else
|
|
error = elf_process_read_fpregs(l, &freg);
|
|
#endif
|
|
if (error)
|
|
return (error);
|
|
|
|
nhdr.n_namesz = namesize;
|
|
nhdr.n_descsz = freglen;
|
|
nhdr.n_type = PT_GETFPREGS;
|
|
|
|
ELFNAMEEND(coredump_savenote)(ns, &nhdr, name, &freg);
|
|
#endif
|
|
/* XXX Add hook for machdep per-LWP notes. */
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
save_note_bytes(struct note_state *ns, const void *data, size_t len)
|
|
{
|
|
struct note_buf *nb = ns->ns_last;
|
|
size_t copylen;
|
|
unsigned char *wp;
|
|
|
|
/*
|
|
* Just copy the data into a buffer list.
|
|
* All but the last buffer is full.
|
|
*/
|
|
for (;;) {
|
|
copylen = min(len, sizeof nb->nb_data - ns->ns_offset);
|
|
wp = nb->nb_data + ns->ns_offset;
|
|
memcpy(wp, data, copylen);
|
|
if (copylen == len)
|
|
break;
|
|
nb->nb_next = kmem_alloc(sizeof *nb->nb_next, KM_SLEEP);
|
|
nb = nb->nb_next;
|
|
ns->ns_last = nb;
|
|
ns->ns_count++;
|
|
ns->ns_offset = 0;
|
|
len -= copylen;
|
|
data = (const unsigned char *)data + copylen;
|
|
}
|
|
|
|
while (copylen & (ELFROUNDSIZE - 1))
|
|
wp[copylen++] = 0;
|
|
|
|
ns->ns_offset += copylen;
|
|
}
|
|
|
|
void
|
|
ELFNAMEEND(coredump_savenote)(struct note_state *ns, Elf_Nhdr *nhdr,
|
|
const char *name, void *data)
|
|
{
|
|
save_note_bytes(ns, nhdr, sizeof (*nhdr));
|
|
save_note_bytes(ns, name, nhdr->n_namesz);
|
|
save_note_bytes(ns, data, nhdr->n_descsz);
|
|
}
|
|
|
|
#else /* COREDUMP */
|
|
|
|
int
|
|
ELFNAMEEND(coredump)(struct lwp *l, void *cookie)
|
|
{
|
|
|
|
return ENOSYS;
|
|
}
|
|
|
|
#endif /* COREDUMP */
|