marcel 7de93ed35c Implement _Unwind_FindTableEntry(). This function is part of GCC
for some configurations, but not for FreeBSD (yet?). Have one in
libc in the mean time.
2002-04-13 04:06:34 +00:00

127 lines
3.2 KiB
C

/*
* Copyright (c) 2002 Marcel Moolenaar
* All rights reserved.
*
* 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.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 THE AUTHOR 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.
*
* $FreeBSD$
*/
#include <sys/types.h>
#include <assert.h>
#include <dlfcn.h>
#include <stdlib.h>
#include <machine/elf.h>
#ifndef PT_IA_64_UNWIND
#define PT_IA_64_UNWIND 0x70000001
#endif
#define SANITY 0
struct ia64_unwind_entry
{
Elf64_Addr start;
Elf64_Addr end;
Elf64_Addr descr;
};
struct ia64_unwind_entry *
_Unwind_FindTableEntry(const void *pc, unsigned long *pseg, unsigned long *pgp)
{
Dl_info info;
Elf_Dyn *dyn;
Elf_Ehdr *ehdr;
Elf_Phdr *phdr;
char *p, *p_top;
struct ia64_unwind_entry *unw, *res;
register unsigned long gp __asm__("gp"); /* XXX assumes gcc */
size_t l, m, r;
if (!dladdr(pc, &info))
return NULL;
ehdr = (Elf_Ehdr*)info.dli_fbase;
#if SANITY
assert(IS_ELF(*ehdr));
assert(ehdr->e_ident[EI_CLASS] == ELFCLASS64);
assert(ehdr->e_ident[EI_DATA] == ELFDATA2LSB);
assert(ehdr->e_machine == EM_IA_64);
#endif
*pgp = gp;
*pseg = 0UL;
res = NULL;
p = (char*)info.dli_fbase + ehdr->e_phoff;
p_top = p + ehdr->e_phnum * ehdr->e_phentsize;
while (p < p_top) {
phdr = (Elf_Phdr*)p;
switch (phdr->p_type) {
case PT_DYNAMIC:
dyn = (Elf_Dyn*)phdr->p_vaddr;
while (dyn->d_tag != DT_NULL) {
if (dyn->d_tag == DT_PLTGOT) {
*pgp = dyn->d_un.d_ptr;
break;
}
dyn++;
}
break;
case PT_LOAD:
if (pc >= (void*)phdr->p_vaddr &&
pc < (void*)(phdr->p_vaddr + phdr->p_memsz))
*pseg = phdr->p_vaddr;
break;
case PT_IA_64_UNWIND:
#if SANITY
assert(*pseg != 0UL);
assert(res == NULL);
#endif
unw = (struct ia64_unwind_entry*)phdr->p_vaddr;
l = 0;
r = phdr->p_memsz / sizeof(struct ia64_unwind_entry);
while (l < r) {
m = (l + r) >> 1;
res = unw + m;
if (pc < (void*)(res->start + *pseg))
r = m;
else if (pc >= (void*)(res->end + *pseg))
l = m + 1;
else
break; /* found */
}
if (l >= r)
res = NULL;
break;
}
p += ehdr->e_phentsize;
}
return res;
}