xref: /titanic_44/usr/src/tools/ctf/dwarf/common/dwarf_ranges.c (revision 07dc1947c362e187fb955d283b692f8769dd5def)
1*07dc1947SRichard Lowe /*
2*07dc1947SRichard Lowe 
3*07dc1947SRichard Lowe   Copyright (C) 2008-2010 David Anderson. All Rights Reserved.
4*07dc1947SRichard Lowe 
5*07dc1947SRichard Lowe   This program is free software; you can redistribute it and/or modify it
6*07dc1947SRichard Lowe   under the terms of version 2.1 of the GNU Lesser General Public License
7*07dc1947SRichard Lowe   as published by the Free Software Foundation.
8*07dc1947SRichard Lowe 
9*07dc1947SRichard Lowe   This program is distributed in the hope that it would be useful, but
10*07dc1947SRichard Lowe   WITHOUT ANY WARRANTY; without even the implied warranty of
11*07dc1947SRichard Lowe   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12*07dc1947SRichard Lowe 
13*07dc1947SRichard Lowe   Further, this software is distributed without any warranty that it is
14*07dc1947SRichard Lowe   free of the rightful claim of any third person regarding infringement
15*07dc1947SRichard Lowe   or the like.  Any license provided herein, whether implied or
16*07dc1947SRichard Lowe   otherwise, applies only to this software file.  Patent licenses, if
17*07dc1947SRichard Lowe   any, provided herein do not apply to combinations of this program with
18*07dc1947SRichard Lowe   other software, or any other product whatsoever.
19*07dc1947SRichard Lowe 
20*07dc1947SRichard Lowe   You should have received a copy of the GNU Lesser General Public
21*07dc1947SRichard Lowe   License along with this program; if not, write the Free Software
22*07dc1947SRichard Lowe   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
23*07dc1947SRichard Lowe   USA.
24*07dc1947SRichard Lowe 
25*07dc1947SRichard Lowe   Contact information:  Silicon Graphics, Inc., 1500 Crittenden Lane,
26*07dc1947SRichard Lowe   Mountain View, CA 94043, or:
27*07dc1947SRichard Lowe 
28*07dc1947SRichard Lowe   http://www.sgi.com
29*07dc1947SRichard Lowe 
30*07dc1947SRichard Lowe   For further information regarding this notice, see:
31*07dc1947SRichard Lowe 
32*07dc1947SRichard Lowe   http://oss.sgi.com/projects/GenInfo/NoticeExplan
33*07dc1947SRichard Lowe 
34*07dc1947SRichard Lowe */
35*07dc1947SRichard Lowe /* The address of the Free Software Foundation is
36*07dc1947SRichard Lowe    Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
37*07dc1947SRichard Lowe    Boston, MA 02110-1301, USA.
38*07dc1947SRichard Lowe    SGI has moved from the Crittenden Lane address.
39*07dc1947SRichard Lowe */
40*07dc1947SRichard Lowe 
41*07dc1947SRichard Lowe 
42*07dc1947SRichard Lowe 
43*07dc1947SRichard Lowe #include "config.h"
44*07dc1947SRichard Lowe #include <stdlib.h>
45*07dc1947SRichard Lowe #include "dwarf_incl.h"
46*07dc1947SRichard Lowe 
47*07dc1947SRichard Lowe struct ranges_entry {
48*07dc1947SRichard Lowe    struct ranges_entry *next;
49*07dc1947SRichard Lowe    Dwarf_Ranges cur;
50*07dc1947SRichard Lowe };
51*07dc1947SRichard Lowe 
52*07dc1947SRichard Lowe 
53*07dc1947SRichard Lowe #define MAX_ADDR ((address_size == 8)?0xffffffffffffffffULL:0xffffffff)
dwarf_get_ranges_a(Dwarf_Debug dbg,Dwarf_Off rangesoffset,Dwarf_Die die,Dwarf_Ranges ** rangesbuf,Dwarf_Signed * listlen,Dwarf_Unsigned * bytecount,Dwarf_Error * error)54*07dc1947SRichard Lowe int dwarf_get_ranges_a(Dwarf_Debug dbg,
55*07dc1947SRichard Lowe     Dwarf_Off rangesoffset,
56*07dc1947SRichard Lowe     Dwarf_Die die,
57*07dc1947SRichard Lowe     Dwarf_Ranges ** rangesbuf,
58*07dc1947SRichard Lowe     Dwarf_Signed * listlen,
59*07dc1947SRichard Lowe     Dwarf_Unsigned * bytecount,
60*07dc1947SRichard Lowe     Dwarf_Error * error)
61*07dc1947SRichard Lowe {
62*07dc1947SRichard Lowe     Dwarf_Small *rangeptr = 0;
63*07dc1947SRichard Lowe     Dwarf_Small *beginrangeptr = 0;
64*07dc1947SRichard Lowe     Dwarf_Small *section_end = 0;
65*07dc1947SRichard Lowe     unsigned entry_count = 0;
66*07dc1947SRichard Lowe     struct ranges_entry *base = 0;
67*07dc1947SRichard Lowe     struct ranges_entry *last = 0;
68*07dc1947SRichard Lowe     struct ranges_entry *curre = 0;
69*07dc1947SRichard Lowe     Dwarf_Ranges * ranges_data_out = 0;
70*07dc1947SRichard Lowe     unsigned copyindex = 0;
71*07dc1947SRichard Lowe     Dwarf_Half address_size = 0;
72*07dc1947SRichard Lowe     int res = DW_DLV_ERROR;
73*07dc1947SRichard Lowe 
74*07dc1947SRichard Lowe     res = _dwarf_load_section(dbg, &dbg->de_debug_ranges,error);
75*07dc1947SRichard Lowe     if (res != DW_DLV_OK) {
76*07dc1947SRichard Lowe         return res;
77*07dc1947SRichard Lowe     }
78*07dc1947SRichard Lowe     if(rangesoffset >= dbg->de_debug_ranges.dss_size) {
79*07dc1947SRichard Lowe         _dwarf_error(dbg, error, DW_DLE_DEBUG_RANGES_OFFSET_BAD);
80*07dc1947SRichard Lowe         return (DW_DLV_ERROR);
81*07dc1947SRichard Lowe 
82*07dc1947SRichard Lowe     }
83*07dc1947SRichard Lowe     address_size = _dwarf_get_address_size(dbg, die);
84*07dc1947SRichard Lowe     section_end = dbg->de_debug_ranges.dss_data +
85*07dc1947SRichard Lowe         dbg->de_debug_ranges.dss_size;
86*07dc1947SRichard Lowe     rangeptr = dbg->de_debug_ranges.dss_data + rangesoffset;
87*07dc1947SRichard Lowe     beginrangeptr = rangeptr;
88*07dc1947SRichard Lowe 
89*07dc1947SRichard Lowe     for(;;) {
90*07dc1947SRichard Lowe         struct ranges_entry * re = calloc(sizeof(struct ranges_entry),1);
91*07dc1947SRichard Lowe         if(!re) {
92*07dc1947SRichard Lowe             _dwarf_error(dbg, error, DW_DLE_DEBUG_RANGES_OUT_OF_MEM);
93*07dc1947SRichard Lowe             return (DW_DLV_ERROR);
94*07dc1947SRichard Lowe         }
95*07dc1947SRichard Lowe         if(rangeptr  >= section_end) {
96*07dc1947SRichard Lowe             return (DW_DLV_NO_ENTRY);
97*07dc1947SRichard Lowe         }
98*07dc1947SRichard Lowe         if((rangeptr + (2*address_size)) > section_end) {
99*07dc1947SRichard Lowe             _dwarf_error(dbg, error, DW_DLE_DEBUG_RANGES_OFFSET_BAD);
100*07dc1947SRichard Lowe             return (DW_DLV_ERROR);
101*07dc1947SRichard Lowe         }
102*07dc1947SRichard Lowe         entry_count++;
103*07dc1947SRichard Lowe         READ_UNALIGNED(dbg,re->cur.dwr_addr1,
104*07dc1947SRichard Lowe                        Dwarf_Addr, rangeptr,
105*07dc1947SRichard Lowe                        address_size);
106*07dc1947SRichard Lowe         rangeptr +=  address_size;
107*07dc1947SRichard Lowe         READ_UNALIGNED(dbg,re->cur.dwr_addr2 ,
108*07dc1947SRichard Lowe                        Dwarf_Addr, rangeptr,
109*07dc1947SRichard Lowe                        address_size);
110*07dc1947SRichard Lowe         rangeptr +=  address_size;
111*07dc1947SRichard Lowe         if(!base) {
112*07dc1947SRichard Lowe            base = re;
113*07dc1947SRichard Lowe            last = re;
114*07dc1947SRichard Lowe         } else {
115*07dc1947SRichard Lowe            last->next = re;
116*07dc1947SRichard Lowe            last = re;
117*07dc1947SRichard Lowe         }
118*07dc1947SRichard Lowe         if(re->cur.dwr_addr1 == 0 && re->cur.dwr_addr2 == 0) {
119*07dc1947SRichard Lowe             re->cur.dwr_type =  DW_RANGES_END;
120*07dc1947SRichard Lowe             break;
121*07dc1947SRichard Lowe         } else if ( re->cur.dwr_addr1 == MAX_ADDR) {
122*07dc1947SRichard Lowe             re->cur.dwr_type =  DW_RANGES_ADDRESS_SELECTION;
123*07dc1947SRichard Lowe         } else {
124*07dc1947SRichard Lowe             re->cur.dwr_type =  DW_RANGES_ENTRY;
125*07dc1947SRichard Lowe         }
126*07dc1947SRichard Lowe     }
127*07dc1947SRichard Lowe 
128*07dc1947SRichard Lowe     ranges_data_out =   (Dwarf_Ranges *)
129*07dc1947SRichard Lowe     _dwarf_get_alloc(dbg,DW_DLA_RANGES,entry_count);
130*07dc1947SRichard Lowe     if(!ranges_data_out) {
131*07dc1947SRichard Lowe             _dwarf_error(dbg, error, DW_DLE_DEBUG_RANGES_OUT_OF_MEM);
132*07dc1947SRichard Lowe             return (DW_DLV_ERROR);
133*07dc1947SRichard Lowe     }
134*07dc1947SRichard Lowe     curre = base;
135*07dc1947SRichard Lowe     *rangesbuf = ranges_data_out;
136*07dc1947SRichard Lowe     *listlen = entry_count;
137*07dc1947SRichard Lowe     for( copyindex = 0; curre && (copyindex < entry_count);
138*07dc1947SRichard Lowe         ++copyindex,++ranges_data_out) {
139*07dc1947SRichard Lowe 
140*07dc1947SRichard Lowe         struct ranges_entry *r = curre;
141*07dc1947SRichard Lowe         *ranges_data_out = curre->cur;
142*07dc1947SRichard Lowe         curre = curre->next;
143*07dc1947SRichard Lowe         free(r);
144*07dc1947SRichard Lowe     }
145*07dc1947SRichard Lowe     /* Callers will often not care about the bytes used. */
146*07dc1947SRichard Lowe     if(bytecount) {
147*07dc1947SRichard Lowe         *bytecount = rangeptr - beginrangeptr;
148*07dc1947SRichard Lowe     }
149*07dc1947SRichard Lowe     return DW_DLV_OK;
150*07dc1947SRichard Lowe }
dwarf_get_ranges(Dwarf_Debug dbg,Dwarf_Off rangesoffset,Dwarf_Ranges ** rangesbuf,Dwarf_Signed * listlen,Dwarf_Unsigned * bytecount,Dwarf_Error * error)151*07dc1947SRichard Lowe int dwarf_get_ranges(Dwarf_Debug dbg,
152*07dc1947SRichard Lowe     Dwarf_Off rangesoffset,
153*07dc1947SRichard Lowe     Dwarf_Ranges ** rangesbuf,
154*07dc1947SRichard Lowe     Dwarf_Signed * listlen,
155*07dc1947SRichard Lowe     Dwarf_Unsigned * bytecount,
156*07dc1947SRichard Lowe     Dwarf_Error * error)
157*07dc1947SRichard Lowe {
158*07dc1947SRichard Lowe     Dwarf_Die die = 0;
159*07dc1947SRichard Lowe     int res = dwarf_get_ranges_a(dbg,rangesoffset,die,
160*07dc1947SRichard Lowe         rangesbuf,listlen,bytecount,error);
161*07dc1947SRichard Lowe     return res;
162*07dc1947SRichard Lowe }
163*07dc1947SRichard Lowe 
164*07dc1947SRichard Lowe void
dwarf_ranges_dealloc(Dwarf_Debug dbg,Dwarf_Ranges * rangesbuf,Dwarf_Signed rangecount)165*07dc1947SRichard Lowe dwarf_ranges_dealloc(Dwarf_Debug dbg, Dwarf_Ranges * rangesbuf,
166*07dc1947SRichard Lowe     Dwarf_Signed rangecount)
167*07dc1947SRichard Lowe {
168*07dc1947SRichard Lowe     dwarf_dealloc(dbg,rangesbuf, DW_DLA_RANGES);
169*07dc1947SRichard Lowe 
170*07dc1947SRichard Lowe }
171*07dc1947SRichard Lowe 
172