aboutsummaryrefslogtreecommitdiff
path: root/lib/libc/ia64/gen/unwind.c
blob: 7afd0eff496ea58f8970dd1f2312c01ade0d24fe (plain) (blame)
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
127
128
129
/*
 * 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 */
	unsigned long reloc, vaddr;
	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

	reloc = (ehdr->e_type == ET_DYN) ? (uintptr_t)info.dli_fbase : 0;
	*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;
		vaddr = phdr->p_vaddr + reloc;

		switch (phdr->p_type) {
		case PT_DYNAMIC:
			dyn = (Elf_Dyn*)vaddr;
			while (dyn->d_tag != DT_NULL) {
				if (dyn->d_tag == DT_PLTGOT) {
					*pgp = dyn->d_un.d_ptr + reloc;
					break;
				}
				dyn++;
			}
			break;
		case PT_LOAD:
			if (pc >= (void*)vaddr &&
			    pc < (void*)(vaddr + phdr->p_memsz))
				*pseg = vaddr;
			break;
		case PT_IA_64_UNWIND:
#if SANITY
			assert(*pseg != 0UL);
			assert(res == NULL);
#endif
			unw = (struct ia64_unwind_entry*)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;
}