/src/elfutils/libdw/dwarf_get_units.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* Iterate through the CU units for a given Dwarf. |
2 | | Copyright (C) 2016, 2017 Red Hat, Inc. |
3 | | This file is part of elfutils. |
4 | | |
5 | | This file is free software; you can redistribute it and/or modify |
6 | | it under the terms of either |
7 | | |
8 | | * the GNU Lesser General Public License as published by the Free |
9 | | Software Foundation; either version 3 of the License, or (at |
10 | | your option) any later version |
11 | | |
12 | | or |
13 | | |
14 | | * the GNU General Public License as published by the Free |
15 | | Software Foundation; either version 2 of the License, or (at |
16 | | your option) any later version |
17 | | |
18 | | or both in parallel, as here. |
19 | | |
20 | | elfutils is distributed in the hope that it will be useful, but |
21 | | WITHOUT ANY WARRANTY; without even the implied warranty of |
22 | | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
23 | | General Public License for more details. |
24 | | |
25 | | You should have received copies of the GNU General Public License and |
26 | | the GNU Lesser General Public License along with this program. If |
27 | | not, see <http://www.gnu.org/licenses/>. */ |
28 | | |
29 | | |
30 | | #ifdef HAVE_CONFIG_H |
31 | | # include <config.h> |
32 | | #endif |
33 | | |
34 | | #include <string.h> |
35 | | |
36 | | #include "libdwP.h" |
37 | | |
38 | | int |
39 | | dwarf_get_units (Dwarf *dwarf, Dwarf_CU *cu, Dwarf_CU **next_cu, |
40 | | Dwarf_Half *version, uint8_t *unit_type, |
41 | | Dwarf_Die *cudie, Dwarf_Die *subdie) |
42 | 0 | { |
43 | | /* Handle existing error. */ |
44 | 0 | if (dwarf == NULL) |
45 | 0 | return -1; |
46 | | |
47 | 0 | Dwarf_Off off; |
48 | 0 | bool v4type; |
49 | 0 | if (cu == NULL) |
50 | 0 | { |
51 | 0 | off = 0; |
52 | 0 | v4type = false; |
53 | 0 | } |
54 | 0 | else |
55 | 0 | { |
56 | 0 | off = cu->end; |
57 | 0 | v4type = cu->sec_idx != IDX_debug_info; |
58 | | |
59 | | /* Make sure we got a real (not fake) CU. */ |
60 | 0 | if (cu->sec_idx != IDX_debug_info && cu->sec_idx != IDX_debug_types) |
61 | 0 | { |
62 | 0 | __libdw_seterrno (DWARF_E_INVALID_OFFSET); |
63 | 0 | return -1; |
64 | 0 | } |
65 | | |
66 | | /* Do we have to switch to the other section, or are we at the end? */ |
67 | 0 | if (! v4type) |
68 | 0 | { |
69 | 0 | if (off >= cu->dbg->sectiondata[IDX_debug_info]->d_size) |
70 | 0 | { |
71 | 0 | if (cu->dbg->sectiondata[IDX_debug_types] == NULL) |
72 | 0 | return 1; |
73 | | |
74 | 0 | off = 0; |
75 | 0 | v4type = true; |
76 | 0 | } |
77 | 0 | } |
78 | 0 | else |
79 | 0 | if (off >= cu->dbg->sectiondata[IDX_debug_types]->d_size) |
80 | 0 | return 1; |
81 | 0 | } |
82 | | |
83 | 0 | *next_cu = __libdw_findcu (dwarf, off, v4type); |
84 | 0 | if (*next_cu == NULL) |
85 | 0 | return -1; |
86 | | |
87 | 0 | Dwarf_CU *next = (*next_cu); |
88 | |
|
89 | 0 | if (version != NULL) |
90 | 0 | *version = next->version; |
91 | |
|
92 | 0 | if (unit_type != NULL) |
93 | 0 | *unit_type = next->unit_type; |
94 | |
|
95 | 0 | if (cudie != NULL) |
96 | 0 | { |
97 | 0 | if (next->version >= 2 && next->version <= 5 |
98 | 0 | && next->unit_type >= DW_UT_compile |
99 | 0 | && next->unit_type <= DW_UT_split_type) |
100 | 0 | *cudie = CUDIE (next); |
101 | 0 | else |
102 | 0 | memset (cudie, '\0', sizeof (Dwarf_Die)); |
103 | 0 | } |
104 | |
|
105 | 0 | if (subdie != NULL) |
106 | 0 | { |
107 | 0 | if (next->version >= 2 && next->version <= 5) |
108 | 0 | { |
109 | | /* For types, return the actual type DIE. For skeletons, |
110 | | find the associated split compile unit and return its |
111 | | DIE. */ |
112 | 0 | if (next->unit_type == DW_UT_type |
113 | 0 | || next->unit_type == DW_UT_split_type) |
114 | 0 | *subdie = SUBDIE(next); |
115 | 0 | else if (next->unit_type == DW_UT_skeleton) |
116 | 0 | { |
117 | 0 | Dwarf_CU *split_cu = __libdw_find_split_unit (next); |
118 | 0 | if (split_cu != NULL) |
119 | 0 | *subdie = CUDIE(split_cu); |
120 | 0 | else |
121 | 0 | memset (subdie, '\0', sizeof (Dwarf_Die)); |
122 | 0 | } |
123 | 0 | else |
124 | 0 | memset (subdie, '\0', sizeof (Dwarf_Die)); |
125 | 0 | } |
126 | 0 | else |
127 | 0 | memset (subdie, '\0', sizeof (Dwarf_Die)); |
128 | 0 | } |
129 | |
|
130 | 0 | return 0; |
131 | 0 | } |
132 | | INTDEF(dwarf_get_units) |