1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
/*
* 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;
}
|