3 Copyright (C) 2000-2004 Silicon Graphics, Inc. All Rights Reserved.
4 Portions Copyright (C) 2007-2010 David Anderson. All Rights Reserved.
6 This program is free software; you can redistribute it and/or modify it
7 under the terms of version 2.1 of the GNU Lesser General Public License
8 as published by the Free Software Foundation.
10 This program is distributed in the hope that it would be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
14 Further, this software is distributed without any warranty that it is
15 free of the rightful claim of any third person regarding infringement
16 or the like. Any license provided herein, whether implied or
17 otherwise, applies only to this software file. Patent licenses, if
18 any, provided herein do not apply to combinations of this program with
19 other software, or any other product whatsoever.
21 You should have received a copy of the GNU Lesser General Public
22 License along with this program; if not, write the Free Software
23 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
26 Contact information: Silicon Graphics, Inc., 1500 Crittenden Lane,
27 Mountain View, CA 94043, or:
31 For further information regarding this notice, see:
33 http://oss.sgi.com/projects/GenInfo/NoticeExplan
36 /* The address of the Free Software Foundation is
37 Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
38 Boston, MA 02110-1301, USA.
39 SGI has moved from the Crittenden Lane address.
48 #include "dwarf_incl.h"
50 #include "dwarf_arange.h"
51 #include "dwarf_global.h" /* for _dwarf_fixup_* */
54 /* Common code for two user-visible routines to share.
55 Errors here result in memory leaks, but errors here
56 are serious (making aranges unusable) so we assume
57 callers will not repeat the error often or mind the leaks.
60 dwarf_get_aranges_list(Dwarf_Debug dbg
,
61 Dwarf_Chain
* chain_out
,
62 Dwarf_Signed
* chain_count_out
,
65 /* Sweeps through the arange. */
66 Dwarf_Small
*arange_ptr
= 0;
67 Dwarf_Small
*arange_ptr_start
= 0;
69 /* Start of arange header. Used for rounding offset of arange_ptr
70 to twice the tuple size. Libdwarf requirement. */
71 Dwarf_Small
*header_ptr
= 0;
73 /* Version of .debug_aranges header. */
74 Dwarf_Half version
= 0;
76 /* Offset of current set of aranges into .debug_info. */
77 Dwarf_Off info_offset
= 0;
79 /* Size in bytes of addresses in target. */
80 Dwarf_Small address_size
= 0;
82 /* Size in bytes of segment offsets in target. */
83 Dwarf_Small segment_size
= 0;
85 /* Count of total number of aranges. */
86 Dwarf_Unsigned arange_count
= 0;
88 Dwarf_Arange arange
= 0;
90 /* Used to chain Dwarf_Aranges structs. */
91 Dwarf_Chain curr_chain
= NULL
;
92 Dwarf_Chain prev_chain
= NULL
;
93 Dwarf_Chain head_chain
= NULL
;
95 arange_ptr
= dbg
->de_debug_aranges
.dss_data
;
96 arange_ptr_start
= arange_ptr
;
98 /* Length of current set of aranges. */
99 Dwarf_Unsigned length
= 0;
100 Dwarf_Small remainder
= 0;
101 Dwarf_Small
*arange_ptr_past_end
= 0;
102 Dwarf_Unsigned range_entry_size
= 0;
104 int local_length_size
;
106 /*REFERENCED*/ /* Not used in this instance of the macro */
107 int local_extension_size
= 0;
109 header_ptr
= arange_ptr
;
111 /* READ_AREA_LENGTH updates arange_ptr for consumed bytes */
112 READ_AREA_LENGTH(dbg
, length
, Dwarf_Unsigned
,
113 arange_ptr
, local_length_size
,
114 local_extension_size
);
115 arange_ptr_past_end
= arange_ptr
+ length
;
118 READ_UNALIGNED(dbg
, version
, Dwarf_Half
,
119 arange_ptr
, sizeof(Dwarf_Half
));
120 arange_ptr
+= sizeof(Dwarf_Half
);
121 length
= length
- sizeof(Dwarf_Half
);
122 if (version
!= CURRENT_VERSION_STAMP
) {
123 _dwarf_error(dbg
, error
, DW_DLE_VERSION_STAMP_ERROR
);
124 return (DW_DLV_ERROR
);
127 READ_UNALIGNED(dbg
, info_offset
, Dwarf_Off
,
128 arange_ptr
, local_length_size
);
129 arange_ptr
+= local_length_size
;
130 length
= length
- local_length_size
;
131 if (info_offset
>= dbg
->de_debug_info
.dss_size
) {
132 FIX_UP_OFFSET_IRIX_BUG(dbg
, info_offset
,
133 "arange info offset.a");
134 if (info_offset
>= dbg
->de_debug_info
.dss_size
) {
135 _dwarf_error(dbg
, error
, DW_DLE_ARANGE_OFFSET_BAD
);
136 return (DW_DLV_ERROR
);
140 address_size
= *(Dwarf_Small
*) arange_ptr
;
141 /* It is not an error if the sizes differ.
142 Unusual, but not an error. */
143 arange_ptr
= arange_ptr
+ sizeof(Dwarf_Small
);
144 length
= length
- sizeof(Dwarf_Small
);
146 segment_size
= *(Dwarf_Small
*) arange_ptr
;
147 arange_ptr
= arange_ptr
+ sizeof(Dwarf_Small
);
148 length
= length
- sizeof(Dwarf_Small
);
149 if (segment_size
!= 0) {
150 _dwarf_error(dbg
, error
, DW_DLE_SEGMENT_SIZE_BAD
);
151 return (DW_DLV_ERROR
);
154 range_entry_size
= 2*address_size
+ segment_size
;
155 /* Round arange_ptr offset to next multiple of address_size. */
156 remainder
= (Dwarf_Unsigned
) (arange_ptr
- header_ptr
) %
158 if (remainder
!= 0) {
159 arange_ptr
= arange_ptr
+ (2 * address_size
) - remainder
;
160 length
= length
- ((2 * address_size
) - remainder
);
163 Dwarf_Addr range_address
= 0;
164 Dwarf_Unsigned segment_selector
= 0;
165 Dwarf_Unsigned range_length
= 0;
166 /* For segmented address spaces, the first field to
167 read is a segment selector (new in DWARF4) */
168 if(version
== 4 && segment_size
!= 0) {
169 READ_UNALIGNED(dbg
, segment_selector
, Dwarf_Unsigned
,
170 arange_ptr
, segment_size
);
171 arange_ptr
+= address_size
;
172 length
= length
- address_size
;
175 READ_UNALIGNED(dbg
, range_address
, Dwarf_Addr
,
176 arange_ptr
, address_size
);
177 arange_ptr
+= address_size
;
178 length
= length
- address_size
;
180 READ_UNALIGNED(dbg
, range_length
, Dwarf_Unsigned
,
181 arange_ptr
, address_size
);
182 arange_ptr
+= address_size
;
183 length
= length
- address_size
;
185 { /* We used to suppress all-zero entries, but
186 now we return all aranges entries so we show
187 the entire content. March 31, 2010. */
189 arange
= (Dwarf_Arange
)
190 _dwarf_get_alloc(dbg
, DW_DLA_ARANGE
, 1);
191 if (arange
== NULL
) {
192 _dwarf_error(dbg
, error
, DW_DLE_ALLOC_FAIL
);
193 return (DW_DLV_ERROR
);
196 arange
->ar_segment_selector
= segment_selector
;
197 arange
->ar_segment_selector_size
= segment_size
;
198 arange
->ar_address
= range_address
;
199 arange
->ar_length
= range_length
;
200 arange
->ar_info_offset
= info_offset
;
201 arange
->ar_dbg
= dbg
;
204 curr_chain
= (Dwarf_Chain
)
205 _dwarf_get_alloc(dbg
, DW_DLA_CHAIN
, 1);
206 if (curr_chain
== NULL
) {
207 _dwarf_error(dbg
, error
, DW_DLE_ALLOC_FAIL
);
208 return (DW_DLV_ERROR
);
211 curr_chain
->ch_item
= arange
;
212 if (head_chain
== NULL
)
213 head_chain
= prev_chain
= curr_chain
;
215 prev_chain
->ch_next
= curr_chain
;
216 prev_chain
= curr_chain
;
219 /* The current set of ranges is terminated by
220 range_address 0 and range_length 0, but that
221 does not necessarily terminate the ranges for this CU!
222 There can be multiple sets in that DWARF
223 does not explicitly forbid multiple sets.
224 DWARF2,3,4 section 7.20
225 We stop short to avoid overrun of the end of the CU.
228 } while (arange_ptr_past_end
>= (arange_ptr
+ range_entry_size
));
230 /* A compiler could emit some padding bytes here. dwarf2/3
231 (dwarf4 sec 7.20) does not clearly make extra padding
233 if (arange_ptr_past_end
< arange_ptr
) {
235 Dwarf_Unsigned pad_count
= arange_ptr
- arange_ptr_past_end
;
236 Dwarf_Unsigned offset
= arange_ptr
- arange_ptr_start
;
237 snprintf(buf
,sizeof(buf
),"DW_DLE_ARANGE_LENGTH_BAD."
239 " pad bytes at offset 0x%" DW_PR_DUx
240 " in .debug_aranges",
242 dwarf_insert_harmless_error(dbg
,buf
);
244 /* For most compilers, arange_ptr == arange_ptr_past_end at
245 this point. But not if there were padding bytes */
246 arange_ptr
= arange_ptr_past_end
;
247 } while (arange_ptr
<
248 dbg
->de_debug_aranges
.dss_data
+ dbg
->de_debug_aranges
.dss_size
);
251 dbg
->de_debug_aranges
.dss_data
+ dbg
->de_debug_aranges
.dss_size
) {
252 _dwarf_error(dbg
, error
, DW_DLE_ARANGE_DECODE_ERROR
);
253 return (DW_DLV_ERROR
);
255 *chain_out
= head_chain
;
256 *chain_count_out
= arange_count
;
261 This function returns the count of the number of
262 aranges in the .debug_aranges section. It sets
263 aranges to point to a block of Dwarf_Arange's
264 describing the arange's. It returns DW_DLV_ERROR
267 Must be identical in most aspects to
268 dwarf_get_aranges_addr_offsets!
272 dwarf_get_aranges(Dwarf_Debug dbg
,
273 Dwarf_Arange
** aranges
,
274 Dwarf_Signed
* returned_count
, Dwarf_Error
* error
)
276 /* Count of total number of aranges. */
277 Dwarf_Signed arange_count
= 0;
279 Dwarf_Arange
*arange_block
= 0;
281 /* Used to chain Dwarf_Aranges structs. */
282 Dwarf_Chain curr_chain
= NULL
;
283 Dwarf_Chain prev_chain
= NULL
;
284 Dwarf_Chain head_chain
= NULL
;
285 Dwarf_Unsigned i
= 0;
286 int res
= DW_DLV_ERROR
;
288 /* ***** BEGIN CODE ***** */
291 _dwarf_error(NULL
, error
, DW_DLE_DBG_NULL
);
292 return (DW_DLV_ERROR
);
295 res
= _dwarf_load_section(dbg
, &dbg
->de_debug_aranges
, error
);
296 if (res
!= DW_DLV_OK
) {
300 res
= dwarf_get_aranges_list(dbg
,&head_chain
,&arange_count
,error
);
301 if(res
!= DW_DLV_OK
) {
305 arange_block
= (Dwarf_Arange
*)
306 _dwarf_get_alloc(dbg
, DW_DLA_LIST
, arange_count
);
307 if (arange_block
== NULL
) {
308 _dwarf_error(dbg
, error
, DW_DLE_ALLOC_FAIL
);
309 return (DW_DLV_ERROR
);
312 curr_chain
= head_chain
;
313 for (i
= 0; i
< arange_count
; i
++) {
314 *(arange_block
+ i
) = curr_chain
->ch_item
;
315 prev_chain
= curr_chain
;
316 curr_chain
= curr_chain
->ch_next
;
317 dwarf_dealloc(dbg
, prev_chain
, DW_DLA_CHAIN
);
320 *aranges
= arange_block
;
321 *returned_count
= (arange_count
);
326 This function returns DW_DLV_OK if it succeeds
327 and DW_DLV_ERR or DW_DLV_OK otherwise.
328 count is set to the number of addresses in the
329 .debug_aranges section.
330 For each address, the corresponding element in
331 an array is set to the address itself(aranges) and
332 the section offset (offsets).
333 Must be identical in most aspects to
337 _dwarf_get_aranges_addr_offsets(Dwarf_Debug dbg
,
339 Dwarf_Off
** offsets
,
340 Dwarf_Signed
* count
,
343 Dwarf_Unsigned i
= 0;
345 /* Used to chain Dwarf_Aranges structs. */
346 Dwarf_Chain curr_chain
= NULL
;
347 Dwarf_Chain prev_chain
= NULL
;
348 Dwarf_Chain head_chain
= NULL
;
350 Dwarf_Signed arange_count
= 0;
351 Dwarf_Addr
*arange_addrs
= 0;
352 Dwarf_Off
*arange_offsets
= 0;
354 int res
= DW_DLV_ERROR
;
356 /* ***** BEGIN CODE ***** */
362 _dwarf_error(NULL
, error
, DW_DLE_DBG_NULL
);
363 return (DW_DLV_ERROR
);
366 res
= _dwarf_load_section(dbg
, &dbg
->de_debug_aranges
,error
);
367 if (res
!= DW_DLV_OK
) {
371 res
= dwarf_get_aranges_list(dbg
,&head_chain
,&arange_count
,error
);
372 if(res
!= DW_DLV_OK
) {
376 arange_addrs
= (Dwarf_Addr
*)
377 _dwarf_get_alloc(dbg
, DW_DLA_ADDR
, arange_count
);
378 if (arange_addrs
== NULL
) {
379 _dwarf_error(dbg
, error
, DW_DLE_ALLOC_FAIL
);
380 return (DW_DLV_ERROR
);
382 arange_offsets
= (Dwarf_Off
*)
383 _dwarf_get_alloc(dbg
, DW_DLA_ADDR
, arange_count
);
384 if (arange_offsets
== NULL
) {
385 _dwarf_error(dbg
, error
, DW_DLE_ALLOC_FAIL
);
386 return (DW_DLV_ERROR
);
389 curr_chain
= head_chain
;
390 for (i
= 0; i
< arange_count
; i
++) {
391 Dwarf_Arange ar
= curr_chain
->ch_item
;
393 arange_addrs
[i
] = ar
->ar_address
;
394 arange_offsets
[i
] = ar
->ar_info_offset
;
395 prev_chain
= curr_chain
;
396 curr_chain
= curr_chain
->ch_next
;
397 dwarf_dealloc(dbg
, ar
, DW_DLA_ARANGE
);
398 dwarf_dealloc(dbg
, prev_chain
, DW_DLA_CHAIN
);
400 *count
= arange_count
;
401 *offsets
= arange_offsets
;
402 *addrs
= arange_addrs
;
408 This function takes a pointer to a block
409 of Dwarf_Arange's, and a count of the
410 length of the block. It checks if the
411 given address is within the range of an
412 address range in the block. If yes, it
413 returns the appropriate Dwarf_Arange.
414 Otherwise, it returns DW_DLV_ERROR.
417 dwarf_get_arange(Dwarf_Arange
* aranges
,
418 Dwarf_Unsigned arange_count
,
420 Dwarf_Arange
* returned_arange
, Dwarf_Error
* error
)
422 Dwarf_Arange curr_arange
= 0;
423 Dwarf_Unsigned i
= 0;
425 if (aranges
== NULL
) {
426 _dwarf_error(NULL
, error
, DW_DLE_ARANGES_NULL
);
427 return (DW_DLV_ERROR
);
429 for (i
= 0; i
< arange_count
; i
++) {
430 curr_arange
= *(aranges
+ i
);
431 if (address
>= curr_arange
->ar_address
&&
433 curr_arange
->ar_address
+ curr_arange
->ar_length
) {
434 *returned_arange
= curr_arange
;
439 return (DW_DLV_NO_ENTRY
);
444 This function takes an Dwarf_Arange,
445 and returns the offset of the first
446 die in the compilation-unit that the
447 arange belongs to. Returns DW_DLV_ERROR
451 dwarf_get_cu_die_offset(Dwarf_Arange arange
,
452 Dwarf_Off
* returned_offset
,
456 Dwarf_Off offset
= 0;
458 if (arange
== NULL
) {
459 _dwarf_error(NULL
, error
, DW_DLE_ARANGE_NULL
);
460 return (DW_DLV_ERROR
);
462 dbg
= arange
->ar_dbg
;
463 offset
= arange
->ar_info_offset
;
464 if (!dbg
->de_debug_info
.dss_data
) {
465 int res
= _dwarf_load_debug_info(dbg
, error
);
467 if (res
!= DW_DLV_OK
) {
471 *returned_offset
= offset
+ _dwarf_length_of_cu_header(dbg
, offset
);
476 This function takes an Dwarf_Arange,
477 and returns the offset of the CU header
478 in the compilation-unit that the
479 arange belongs to. Returns DW_DLV_ERROR
481 Ensures .debug_info loaded so
482 the cu_offset is meaningful.
485 dwarf_get_arange_cu_header_offset(Dwarf_Arange arange
,
486 Dwarf_Off
* cu_header_offset_returned
,
490 if (arange
== NULL
) {
491 _dwarf_error(NULL
, error
, DW_DLE_ARANGE_NULL
);
492 return (DW_DLV_ERROR
);
494 dbg
= arange
->ar_dbg
;
495 /* Like dwarf_get_arange_info this ensures debug_info loaded:
496 the cu_header is in debug_info and will be used else
497 we would not call dwarf_get_arange_cu_header_offset. */
498 if (!dbg
->de_debug_info
.dss_data
) {
499 int res
= _dwarf_load_debug_info(dbg
, error
);
500 if (res
!= DW_DLV_OK
) {
504 *cu_header_offset_returned
= arange
->ar_info_offset
;
512 This function takes a Dwarf_Arange, and returns
513 true if it is not NULL. It also stores the start
514 address of the range in *start, the length of the
515 range in *length, and the offset of the first die
516 in the compilation-unit in *cu_die_offset. It
517 returns false on error.
518 If cu_die_offset returned ensures .debug_info loaded so
519 the cu_die_offset is meaningful.
522 dwarf_get_arange_info(Dwarf_Arange arange
,
524 Dwarf_Unsigned
* length
,
525 Dwarf_Off
* cu_die_offset
, Dwarf_Error
* error
)
527 if (arange
== NULL
) {
528 _dwarf_error(NULL
, error
, DW_DLE_ARANGE_NULL
);
529 return (DW_DLV_ERROR
);
533 *start
= arange
->ar_address
;
535 *length
= arange
->ar_length
;
536 if (cu_die_offset
!= NULL
) {
537 Dwarf_Debug dbg
= arange
->ar_dbg
;
538 Dwarf_Off offset
= arange
->ar_info_offset
;
540 if (!dbg
->de_debug_info
.dss_data
) {
541 int res
= _dwarf_load_debug_info(dbg
, error
);
542 if (res
!= DW_DLV_OK
) {
547 offset
+ _dwarf_length_of_cu_header(dbg
, offset
);
553 /* New for DWARF4, entries may have segment information.
554 *segment is only meaningful if *segment_entry_size is non-zero. */
556 dwarf_get_arange_info_b(Dwarf_Arange arange
,
557 Dwarf_Unsigned
* segment
,
558 Dwarf_Unsigned
* segment_entry_size
,
560 Dwarf_Unsigned
* length
,
561 Dwarf_Off
* cu_die_offset
,
564 if (arange
== NULL
) {
565 _dwarf_error(NULL
, error
, DW_DLE_ARANGE_NULL
);
566 return (DW_DLV_ERROR
);
569 if(segment
!= NULL
) {
570 *segment
= arange
->ar_segment_selector
;
572 if(segment_entry_size
!= NULL
) {
573 *segment_entry_size
= arange
->ar_segment_selector_size
;
576 *start
= arange
->ar_address
;
578 *length
= arange
->ar_length
;
579 if (cu_die_offset
!= NULL
) {
580 Dwarf_Debug dbg
= arange
->ar_dbg
;
581 Dwarf_Off offset
= arange
->ar_info_offset
;
583 if (!dbg
->de_debug_info
.dss_data
) {
584 int res
= _dwarf_load_debug_info(dbg
, error
);
585 if (res
!= DW_DLV_OK
) {
590 offset
+ _dwarf_length_of_cu_header(dbg
, offset
);