blob: c1d3a451a6659949a9a56b5eb98c967f6fc5803c [file] [log] [blame]
Ulrich Drepperb08d5a82005-07-26 05:00:05 +00001/* Find CU for given offset.
2 Copyright (C) 2003, 2004, 2005 Red Hat, Inc.
3 Written by Ulrich Drepper <drepper@redhat.com>, 2003.
4
5 This program is Open Source software; you can redistribute it and/or
6 modify it under the terms of the Open Software License version 1.0 as
7 published by the Open Source Initiative.
8
9 You should have received a copy of the Open Software License along
10 with this program; if not, you may obtain a copy of the Open Software
11 License version 1.0 from http://www.opensource.org/licenses/osl.php or
12 by writing the Open Source Initiative c/o Lawrence Rosen, Esq.,
13 3001 King Ranch Road, Ukiah, CA 95482. */
14
15#ifdef HAVE_CONFIG_H
16# include <config.h>
17#endif
18
19#include <assert.h>
20#include <search.h>
21#include "libdwP.h"
22
23
24static int
25findcu_cb (const void *arg1, const void *arg2)
26{
27 struct Dwarf_CU *cu1 = (struct Dwarf_CU *) arg1;
28 struct Dwarf_CU *cu2 = (struct Dwarf_CU *) arg2;
29
30 /* Find out which of the two arguments is the search value. It has
31 end offset 0. */
32 if (cu1->end == 0)
33 {
34 if (cu1->start < cu2->start)
35 return -1;
36 if (cu1->start >= cu2->end)
37 return 1;
38 }
39 else
40 {
41 if (cu2->start < cu1->start)
42 return 1;
43 if (cu2->start >= cu1->end)
44 return -1;
45 }
46
47 return 0;
48}
49
50
51struct Dwarf_CU *
52__libdw_findcu (dbg, start)
53 Dwarf *dbg;
54 Dwarf_Off start;
55{
56 /* Maybe we already know that CU. */
57 struct Dwarf_CU fake = { .start = start, .end = 0 };
58 struct Dwarf_CU **found = tfind (&fake, &dbg->cu_tree, findcu_cb);
59 if (found != NULL)
60 return *found;
61
62 if (start < dbg->next_cu_offset)
63 {
64 __libdw_seterrno (DWARF_E_INVALID_DWARF);
65 return NULL;
66 }
67
68 /* No. Then read more CUs. */
69 while (1)
70 {
71 Dwarf_Off oldoff = dbg->next_cu_offset;
72 uint8_t address_size;
73 uint8_t offset_size;
74 Dwarf_Off abbrev_offset;
75
76 if (INTUSE(dwarf_nextcu) (dbg, oldoff, &dbg->next_cu_offset, NULL,
77 &abbrev_offset, &address_size, &offset_size)
78 != 0)
79 /* No more entries. */
80 return NULL;
81
82 /* Create an entry for this CU. */
83 struct Dwarf_CU *newp = libdw_typed_alloc (dbg, struct Dwarf_CU);
84
85 newp->dbg = dbg;
86 newp->start = oldoff;
87 newp->end = dbg->next_cu_offset;
88 newp->address_size = address_size;
89 newp->offset_size = offset_size;
90 Dwarf_Abbrev_Hash_init (&newp->abbrev_hash, 41);
91 newp->orig_abbrev_offset = newp->last_abbrev_offset = abbrev_offset;
92 newp->lines = NULL;
93 newp->locs = NULL;
94
95 /* Add the new entry to the search tree. */
96 if (tsearch (newp, &dbg->cu_tree, findcu_cb) == NULL)
97 {
98 /* Something went wrong. Unfo the operation. */
99 dbg->next_cu_offset = oldoff;
100 __libdw_seterrno (DWARF_E_NOMEM);
101 return NULL;
102 }
103
104 /* Is this the one we are looking for? */
105 if (start < dbg->next_cu_offset)
106 // XXX Match exact offset.
107 return newp;
108 }
109 /* NOTREACHED */
110}