diff options
author | Artur Grabowski <art@cvs.openbsd.org> | 2002-09-23 01:41:10 +0000 |
---|---|---|
committer | Artur Grabowski <art@cvs.openbsd.org> | 2002-09-23 01:41:10 +0000 |
commit | 37e60d4e879bfb430b186cd3dcc80c4635e4623f (patch) | |
tree | 537c14119687270912a853b44c381f050cd9fde5 /sys | |
parent | dbe86ef0cad473922ebbee79d6382d5753a8ea84 (diff) |
Add support for vmcmds that load sections relative to a base section.
You mark one section with VMCMD_BASE and the rest are marked RELATIVE.
Use that to load ELF interpreter correctly in all cases.
Inspired by NetBSD. Great debugging help from drahn@
deraadt@ ok
Diffstat (limited to 'sys')
-rw-r--r-- | sys/kern/exec_elf.c | 63 | ||||
-rw-r--r-- | sys/kern/kern_exec.c | 15 | ||||
-rw-r--r-- | sys/sys/exec.h | 19 |
3 files changed, 68 insertions, 29 deletions
diff --git a/sys/kern/exec_elf.c b/sys/kern/exec_elf.c index 68077f357ca..7ecba4b234b 100644 --- a/sys/kern/exec_elf.c +++ b/sys/kern/exec_elf.c @@ -1,4 +1,4 @@ -/* $OpenBSD: exec_elf.c,v 1.40 2002/09/18 22:07:50 drahn Exp $ */ +/* $OpenBSD: exec_elf.c,v 1.41 2002/09/23 01:41:09 art Exp $ */ /* * Copyright (c) 1996 Per Fogelstrom @@ -99,7 +99,7 @@ int ELFNAME(check_header)(Elf_Ehdr *, int); int ELFNAME(olf_check_header)(Elf_Ehdr *, int, u_int8_t *); int ELFNAME(read_from)(struct proc *, struct vnode *, u_long, caddr_t, int); void ELFNAME(load_psection)(struct exec_vmcmd_set *, struct vnode *, - Elf_Phdr *, Elf_Addr *, Elf_Addr *, int *); + Elf_Phdr *, Elf_Addr *, Elf_Addr *, int *, int); extern char sigcode[], esigcode[]; #ifdef SYSCALL_DEBUG @@ -240,7 +240,7 @@ os_ok: */ void ELFNAME(load_psection)(struct exec_vmcmd_set *vcset, struct vnode *vp, - Elf_Phdr *ph, Elf_Addr *addr, Elf_Addr *size, int *prot) + Elf_Phdr *ph, Elf_Addr *addr, Elf_Addr *size, int *prot, int flags) { u_long uaddr, msize, psize, rm, rf; long diff, offset; @@ -250,11 +250,10 @@ ELFNAME(load_psection)(struct exec_vmcmd_set *vcset, struct vnode *vp, */ if (*addr != ELFDEFNNAME(NO_ADDR)) { if (ph->p_align > 1) { - *addr = ELF_ROUND(*addr, ph->p_align); - uaddr = ELF_TRUNC(ph->p_vaddr, ph->p_align); + *addr = ELF_TRUNC(*addr, ph->p_align); + diff = ph->p_vaddr - ELF_TRUNC(ph->p_vaddr, ph->p_align); } else - uaddr = ph->p_vaddr; - diff = ph->p_vaddr - uaddr; + diff = 0; } else { *addr = uaddr = ph->p_vaddr; if (ph->p_align > 1) @@ -277,15 +276,16 @@ ELFNAME(load_psection)(struct exec_vmcmd_set *vcset, struct vnode *vp, */ if (ph->p_flags & PF_W) { psize = trunc_page(*size); - NEW_VMCMD(vcset, vmcmd_map_pagedvn, psize, *addr, vp, - offset, *prot); + if (psize > 0) + NEW_VMCMD2(vcset, vmcmd_map_pagedvn, psize, *addr, vp, + offset, *prot, flags); if (psize != *size) { - NEW_VMCMD(vcset, vmcmd_map_readvn, *size - psize, - *addr + psize, vp, offset + psize, *prot); + NEW_VMCMD2(vcset, vmcmd_map_readvn, *size - psize, + *addr + psize, vp, offset + psize, *prot, flags); } } else { - NEW_VMCMD(vcset, vmcmd_map_pagedvn, psize, *addr, vp, offset, - *prot); + NEW_VMCMD2(vcset, vmcmd_map_pagedvn, psize, *addr, vp, offset, + *prot, flags); } /* @@ -295,8 +295,8 @@ ELFNAME(load_psection)(struct exec_vmcmd_set *vcset, struct vnode *vp, rf = round_page(*addr + *size); if (rm != rf) { - NEW_VMCMD(vcset, vmcmd_map_zero, rm - rf, rf, NULLVP, 0, - *prot); + NEW_VMCMD2(vcset, vmcmd_map_zero, rm - rf, rf, NULLVP, 0, + *prot, flags); *size = msize; } } @@ -336,9 +336,10 @@ ELFNAME(load_file)(struct proc *p, char *path, struct exec_package *epp, Elf_Phdr *ph = NULL; u_long phsize; char *bp = NULL; - Elf_Addr addr = *last; + Elf_Addr addr; struct vnode *vp; u_int8_t os; /* Just a dummy in this routine */ + Elf_Phdr *base_ph = NULL; bp = path; NDINIT(&nd, LOOKUP, FOLLOW | LOCKLEAF, UIO_SYSSPACE, path, p); @@ -381,15 +382,20 @@ ELFNAME(load_file)(struct proc *p, char *path, struct exec_package *epp, for (i = 0; i < eh.e_phnum; i++) { Elf_Addr size = 0; int prot = 0; -#if defined(__mips__) - if (*last == ELFDEFNNAME(NO_ADDR)) - addr = ELFDEFNNAME(NO_ADDR); /* GRRRRR!!!!! */ -#endif + int flags; switch (ph[i].p_type) { case PT_LOAD: + if (base_ph == NULL) { + flags = VMCMD_BASE; + addr = *last; + base_ph = &ph[i]; + } else { + flags = VMCMD_RELATIVE; + addr = ph[i].p_vaddr - base_ph->p_vaddr; + } ELFNAME(load_psection)(&epp->ep_vmcmds, nd.ni_vp, - &ph[i], &addr, &size, &prot); + &ph[i], &addr, &size, &prot, flags); /* If entry is within this section it must be text */ if (eh.e_entry >= ph[i].p_vaddr && eh.e_entry < (ph[i].p_vaddr + size)) { @@ -546,7 +552,7 @@ native: if (nload++ == 2) goto bad; ELFNAME(load_psection)(&epp->ep_vmcmds, epp->ep_vp, - &ph[i], &addr, &size, &prot); + &ph[i], &addr, &size, &prot, 0); /* * Decide whether it's text or data by looking * at the entry point. @@ -659,6 +665,7 @@ ELFNAME2(exec,fixup)(struct proc *p, struct exec_package *epp) struct elf_args *ap; AuxInfo ai[ELF_AUX_ENTRIES], *a; Elf_Addr pos = epp->ep_interp_pos; + struct exec_vmcmd *base_vc; if (epp->ep_interp == 0) { return (0); @@ -676,11 +683,21 @@ ELFNAME2(exec,fixup)(struct proc *p, struct exec_package *epp) /* * We have to do this ourselves... */ + base_vc = NULL; for (i = 0; i < epp->ep_vmcmds.evs_used && !error; i++) { struct exec_vmcmd *vcp; - vcp = &epp->ep_vmcmds.evs_cmds[i]; + + if (vcp->ev_flags & VMCMD_RELATIVE) { +#ifdef DIAGNOSTIC + if (base_vc == NULL) + panic("sys_execve: RELATIVE without base"); +#endif + vcp->ev_addr += base_vc->ev_addr; + } error = (*vcp->ev_proc)(p, vcp); + if (vcp->ev_flags & VMCMD_BASE) + base_vc = vcp; } kill_vmcmds(&epp->ep_vmcmds); diff --git a/sys/kern/kern_exec.c b/sys/kern/kern_exec.c index 2b0c25bc328..7d332543464 100644 --- a/sys/kern/kern_exec.c +++ b/sys/kern/kern_exec.c @@ -1,4 +1,4 @@ -/* $OpenBSD: kern_exec.c,v 1.70 2002/08/22 22:04:42 art Exp $ */ +/* $OpenBSD: kern_exec.c,v 1.71 2002/09/23 01:41:09 art Exp $ */ /* $NetBSD: kern_exec.c,v 1.75 1996/02/09 18:59:28 christos Exp $ */ /*- @@ -252,6 +252,7 @@ sys_execve(p, v, retval) struct vmspace *vm = p->p_vmspace; char **tmpfap; extern struct emul emul_native; + struct exec_vmcmd *base_vc; /* * Cheap solution to complicated problems. @@ -406,11 +407,23 @@ sys_execve(p, v, retval) if (pack.ep_vmcmds.evs_used == 0) panic("execve: no vmcmds"); #endif + base_vc = NULL; for (i = 0; i < pack.ep_vmcmds.evs_used && !error; i++) { struct exec_vmcmd *vcp; vcp = &pack.ep_vmcmds.evs_cmds[i]; + + if (vcp->ev_flags & VMCMD_RELATIVE) { +#ifdef DIAGNOSTIC + if (base_vc == NULL) + panic("sys_execve: RELATIVE without base"); +#endif + vcp->ev_addr += base_vc->ev_addr; + } + error = (*vcp->ev_proc)(p, vcp); + if (vcp->ev_flags & VMCMD_BASE) + base_vc = vcp; } /* free the vmspace-creation commands, and release their references */ diff --git a/sys/sys/exec.h b/sys/sys/exec.h index eee337c776f..b9bf179ca84 100644 --- a/sys/sys/exec.h +++ b/sys/sys/exec.h @@ -1,4 +1,4 @@ -/* $OpenBSD: exec.h,v 1.16 2002/07/20 19:24:57 art Exp $ */ +/* $OpenBSD: exec.h,v 1.17 2002/09/23 01:41:09 art Exp $ */ /* $NetBSD: exec.h,v 1.59 1996/02/09 18:25:09 christos Exp $ */ /*- @@ -124,6 +124,9 @@ struct exec_vmcmd { struct vnode *ev_vp; /* vnode pointer for the file w/the data */ u_long ev_offset; /* offset in the file for the data */ u_int ev_prot; /* protections for segment */ + int ev_flags; +#define VMCMD_RELATIVE 0x0001 /* ev_addr is relative to base entry */ +#define VMCMD_BASE 0x0002 /* marks a base entry */ }; #define EXEC_DEFAULT_VMCMD_SETSIZE 8 /* # of cmds in set to start */ @@ -193,11 +196,15 @@ int exec_setup_stack(struct proc *, struct exec_package *); void new_vmcmd(struct exec_vmcmd_set *evsp, int (*proc)(struct proc *p, struct exec_vmcmd *), u_long len, u_long addr, struct vnode *vp, u_long offset, - u_int prot); + u_int prot, int flags); #define NEW_VMCMD(evsp,proc,len,addr,vp,offset,prot) \ - new_vmcmd(evsp,proc,len,addr,vp,offset,prot); + new_vmcmd(evsp,proc,len,addr,vp,offset,prot, 0); +#define NEW_VMCMD2(evsp,proc,len,addr,vp,offset,prot,flags) \ + new_vmcmd(evsp,proc,len,addr,vp,offset,prot,flags) #else /* DEBUG */ -#define NEW_VMCMD(evsp,proc,len,addr,vp,offset,prot) { \ +#define NEW_VMCMD(evsp,proc,len,addr,vp,offset,prot) \ + NEW_VMCMD2(evsp,proc,len,addr,vp,offset,prot,0) +#define NEW_VMCMD2(evsp,proc,len,addr,vp,offset,prot,flags) do { \ struct exec_vmcmd *vcp; \ if ((evsp)->evs_used >= (evsp)->evs_cnt) \ vmcmdset_extend(evsp); \ @@ -209,7 +216,9 @@ void new_vmcmd(struct exec_vmcmd_set *evsp, VREF(vp); \ vcp->ev_offset = (offset); \ vcp->ev_prot = (prot); \ -} + vcp->ev_flags = (flags); \ +} while (0) + #endif /* DEBUG */ /* Initialize an empty vmcmd set */ |