1*7fd79137SRobert Mustacchi /*
2*7fd79137SRobert Mustacchi
3*7fd79137SRobert Mustacchi Copyright (C) 2000-2006 Silicon Graphics, Inc. All Rights Reserved.
4*7fd79137SRobert Mustacchi Portions Copyright (C) 2009-2010 David Anderson. All Rights Reserved.
5*7fd79137SRobert Mustacchi
6*7fd79137SRobert Mustacchi This program is free software; you can redistribute it and/or modify it
7*7fd79137SRobert Mustacchi under the terms of version 2.1 of the GNU Lesser General Public License
8*7fd79137SRobert Mustacchi as published by the Free Software Foundation.
9*7fd79137SRobert Mustacchi
10*7fd79137SRobert Mustacchi This program is distributed in the hope that it would be useful, but
11*7fd79137SRobert Mustacchi WITHOUT ANY WARRANTY; without even the implied warranty of
12*7fd79137SRobert Mustacchi MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13*7fd79137SRobert Mustacchi
14*7fd79137SRobert Mustacchi Further, this software is distributed without any warranty that it is
15*7fd79137SRobert Mustacchi free of the rightful claim of any third person regarding infringement
16*7fd79137SRobert Mustacchi or the like. Any license provided herein, whether implied or
17*7fd79137SRobert Mustacchi otherwise, applies only to this software file. Patent licenses, if
18*7fd79137SRobert Mustacchi any, provided herein do not apply to combinations of this program with
19*7fd79137SRobert Mustacchi other software, or any other product whatsoever.
20*7fd79137SRobert Mustacchi
21*7fd79137SRobert Mustacchi You should have received a copy of the GNU Lesser General Public
22*7fd79137SRobert Mustacchi License along with this program; if not, write the Free Software
23*7fd79137SRobert Mustacchi Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
24*7fd79137SRobert Mustacchi USA.
25*7fd79137SRobert Mustacchi
26*7fd79137SRobert Mustacchi Contact information: Silicon Graphics, Inc., 1500 Crittenden Lane,
27*7fd79137SRobert Mustacchi Mountain View, CA 94043, or:
28*7fd79137SRobert Mustacchi
29*7fd79137SRobert Mustacchi http://www.sgi.com
30*7fd79137SRobert Mustacchi
31*7fd79137SRobert Mustacchi For further information regarding this notice, see:
32*7fd79137SRobert Mustacchi
33*7fd79137SRobert Mustacchi http://oss.sgi.com/projects/GenInfo/NoticeExplan
34*7fd79137SRobert Mustacchi
35*7fd79137SRobert Mustacchi */
36*7fd79137SRobert Mustacchi
37*7fd79137SRobert Mustacchi
38*7fd79137SRobert Mustacchi
39*7fd79137SRobert Mustacchi #include "config.h"
40*7fd79137SRobert Mustacchi #include "dwarf_incl.h"
41*7fd79137SRobert Mustacchi #include <stdio.h>
42*7fd79137SRobert Mustacchi #include <stdlib.h>
43*7fd79137SRobert Mustacchi #include "dwarf_frame.h"
44*7fd79137SRobert Mustacchi #include "dwarf_arange.h" /* using Arange as a way to build a list */
45*7fd79137SRobert Mustacchi
46*7fd79137SRobert Mustacchi /*
47*7fd79137SRobert Mustacchi Used by rqs (an IRIX application).
48*7fd79137SRobert Mustacchi Not needed except for that one application.
49*7fd79137SRobert Mustacchi Should be moved to its own source file since
50*7fd79137SRobert Mustacchi it is so rarely needed.
51*7fd79137SRobert Mustacchi Returns DW_DLV_OK if returns the arrays.
52*7fd79137SRobert Mustacchi Returns DW_DLV_NO_ENTRY if no section. ?? (How do I tell?)
53*7fd79137SRobert Mustacchi Returns DW_DLV_ERROR if there is an error.
54*7fd79137SRobert Mustacchi
55*7fd79137SRobert Mustacchi Uses DW_FRAME_CFA_COL because IRIX is only DWARF2
56*7fd79137SRobert Mustacchi and that is what IRIX compilers and compatible
57*7fd79137SRobert Mustacchi compilers support on IRIX.
58*7fd79137SRobert Mustacchi */
59*7fd79137SRobert Mustacchi int
_dwarf_frame_address_offsets(Dwarf_Debug dbg,Dwarf_Addr ** addrlist,Dwarf_Off ** offsetlist,Dwarf_Signed * returncount,Dwarf_Error * err)60*7fd79137SRobert Mustacchi _dwarf_frame_address_offsets(Dwarf_Debug dbg, Dwarf_Addr ** addrlist,
61*7fd79137SRobert Mustacchi Dwarf_Off ** offsetlist,
62*7fd79137SRobert Mustacchi Dwarf_Signed * returncount,
63*7fd79137SRobert Mustacchi Dwarf_Error * err)
64*7fd79137SRobert Mustacchi {
65*7fd79137SRobert Mustacchi int retval = DW_DLV_OK;
66*7fd79137SRobert Mustacchi int res = DW_DLV_ERROR;
67*7fd79137SRobert Mustacchi Dwarf_Cie *cie_data;
68*7fd79137SRobert Mustacchi Dwarf_Signed cie_count;
69*7fd79137SRobert Mustacchi Dwarf_Fde *fde_data;
70*7fd79137SRobert Mustacchi Dwarf_Signed fde_count;
71*7fd79137SRobert Mustacchi Dwarf_Signed i;
72*7fd79137SRobert Mustacchi Dwarf_Frame_Op *frame_inst;
73*7fd79137SRobert Mustacchi Dwarf_Fde fdep;
74*7fd79137SRobert Mustacchi Dwarf_Cie ciep;
75*7fd79137SRobert Mustacchi Dwarf_Chain curr_chain = 0;
76*7fd79137SRobert Mustacchi Dwarf_Chain head_chain = 0;
77*7fd79137SRobert Mustacchi Dwarf_Chain prev_chain = 0;
78*7fd79137SRobert Mustacchi Dwarf_Arange arange;
79*7fd79137SRobert Mustacchi Dwarf_Unsigned arange_count = 0;
80*7fd79137SRobert Mustacchi Dwarf_Addr *arange_addrs = 0;
81*7fd79137SRobert Mustacchi Dwarf_Off *arange_offsets = 0;
82*7fd79137SRobert Mustacchi
83*7fd79137SRobert Mustacchi res = dwarf_get_fde_list(dbg, &cie_data, &cie_count,
84*7fd79137SRobert Mustacchi &fde_data, &fde_count, err);
85*7fd79137SRobert Mustacchi if (res != DW_DLV_OK) {
86*7fd79137SRobert Mustacchi return res;
87*7fd79137SRobert Mustacchi }
88*7fd79137SRobert Mustacchi
89*7fd79137SRobert Mustacchi res = _dwarf_load_section(dbg, &dbg->de_debug_frame, err);
90*7fd79137SRobert Mustacchi if (res != DW_DLV_OK) {
91*7fd79137SRobert Mustacchi return res;
92*7fd79137SRobert Mustacchi }
93*7fd79137SRobert Mustacchi
94*7fd79137SRobert Mustacchi for (i = 0; i < cie_count; i++) {
95*7fd79137SRobert Mustacchi Dwarf_Off instoff = 0;
96*7fd79137SRobert Mustacchi Dwarf_Signed initial_instructions_length = 0;
97*7fd79137SRobert Mustacchi Dwarf_Small *instr_end = 0;
98*7fd79137SRobert Mustacchi Dwarf_Sword icount = 0;
99*7fd79137SRobert Mustacchi int j = 0;
100*7fd79137SRobert Mustacchi int dw_err;
101*7fd79137SRobert Mustacchi
102*7fd79137SRobert Mustacchi ciep = cie_data[i];
103*7fd79137SRobert Mustacchi instoff = ciep->ci_cie_instr_start - dbg->de_debug_frame.dss_data;
104*7fd79137SRobert Mustacchi initial_instructions_length = ciep->ci_length +
105*7fd79137SRobert Mustacchi ciep->ci_length_size + ciep->ci_extension_size -
106*7fd79137SRobert Mustacchi (ciep->ci_cie_instr_start - ciep->ci_cie_start);
107*7fd79137SRobert Mustacchi instr_end = ciep->ci_cie_instr_start +
108*7fd79137SRobert Mustacchi initial_instructions_length;
109*7fd79137SRobert Mustacchi res = _dwarf_exec_frame_instr( /* make_instr */ true,
110*7fd79137SRobert Mustacchi &frame_inst,
111*7fd79137SRobert Mustacchi /* search_pc= */ false,
112*7fd79137SRobert Mustacchi /* search_pc_val= */ 0,
113*7fd79137SRobert Mustacchi /* location */ 0,
114*7fd79137SRobert Mustacchi ciep->ci_cie_instr_start,
115*7fd79137SRobert Mustacchi instr_end,
116*7fd79137SRobert Mustacchi /* Dwarf_frame= */ 0,
117*7fd79137SRobert Mustacchi /* cie= */ 0,
118*7fd79137SRobert Mustacchi dbg,
119*7fd79137SRobert Mustacchi DW_FRAME_CFA_COL,
120*7fd79137SRobert Mustacchi &icount, &dw_err);
121*7fd79137SRobert Mustacchi if (res == DW_DLV_ERROR) {
122*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, dw_err);
123*7fd79137SRobert Mustacchi return (res);
124*7fd79137SRobert Mustacchi } else if (res == DW_DLV_NO_ENTRY) {
125*7fd79137SRobert Mustacchi continue;
126*7fd79137SRobert Mustacchi }
127*7fd79137SRobert Mustacchi
128*7fd79137SRobert Mustacchi for (j = 0; j < icount; ++j) {
129*7fd79137SRobert Mustacchi Dwarf_Frame_Op *finst = frame_inst + j;
130*7fd79137SRobert Mustacchi
131*7fd79137SRobert Mustacchi if (finst->fp_base_op == 0 && finst->fp_extended_op == 1) {
132*7fd79137SRobert Mustacchi /* is DW_CFA_set_loc */
133*7fd79137SRobert Mustacchi Dwarf_Addr add = (Dwarf_Addr) finst->fp_offset;
134*7fd79137SRobert Mustacchi Dwarf_Off off = finst->fp_instr_offset + instoff;
135*7fd79137SRobert Mustacchi
136*7fd79137SRobert Mustacchi arange = (Dwarf_Arange)
137*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_ARANGE, 1);
138*7fd79137SRobert Mustacchi if (arange == NULL) {
139*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
140*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
141*7fd79137SRobert Mustacchi }
142*7fd79137SRobert Mustacchi arange->ar_address = add;
143*7fd79137SRobert Mustacchi arange->ar_info_offset = off;
144*7fd79137SRobert Mustacchi arange_count++;
145*7fd79137SRobert Mustacchi curr_chain = (Dwarf_Chain)
146*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_CHAIN, 1);
147*7fd79137SRobert Mustacchi if (curr_chain == NULL) {
148*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
149*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
150*7fd79137SRobert Mustacchi }
151*7fd79137SRobert Mustacchi curr_chain->ch_item = arange;
152*7fd79137SRobert Mustacchi if (head_chain == NULL)
153*7fd79137SRobert Mustacchi head_chain = prev_chain = curr_chain;
154*7fd79137SRobert Mustacchi else {
155*7fd79137SRobert Mustacchi prev_chain->ch_next = curr_chain;
156*7fd79137SRobert Mustacchi prev_chain = curr_chain;
157*7fd79137SRobert Mustacchi }
158*7fd79137SRobert Mustacchi }
159*7fd79137SRobert Mustacchi }
160*7fd79137SRobert Mustacchi dwarf_dealloc(dbg, frame_inst, DW_DLA_FRAME_BLOCK);
161*7fd79137SRobert Mustacchi
162*7fd79137SRobert Mustacchi }
163*7fd79137SRobert Mustacchi for (i = 0; i < fde_count; i++) {
164*7fd79137SRobert Mustacchi Dwarf_Small *instr_end = 0;
165*7fd79137SRobert Mustacchi Dwarf_Sword icount = 0;
166*7fd79137SRobert Mustacchi Dwarf_Signed instructions_length = 0;
167*7fd79137SRobert Mustacchi Dwarf_Off instoff = 0;
168*7fd79137SRobert Mustacchi Dwarf_Off off = 0;
169*7fd79137SRobert Mustacchi Dwarf_Addr addr = 0;
170*7fd79137SRobert Mustacchi int j = 0;
171*7fd79137SRobert Mustacchi int dw_err;
172*7fd79137SRobert Mustacchi
173*7fd79137SRobert Mustacchi fdep = fde_data[i];
174*7fd79137SRobert Mustacchi off = fdep->fd_initial_loc_pos - dbg->de_debug_frame.dss_data;
175*7fd79137SRobert Mustacchi addr = fdep->fd_initial_location;
176*7fd79137SRobert Mustacchi arange = (Dwarf_Arange)
177*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_ARANGE, 1);
178*7fd79137SRobert Mustacchi if (arange == NULL) {
179*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
180*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
181*7fd79137SRobert Mustacchi }
182*7fd79137SRobert Mustacchi arange->ar_address = addr;
183*7fd79137SRobert Mustacchi arange->ar_info_offset = off;
184*7fd79137SRobert Mustacchi arange_count++;
185*7fd79137SRobert Mustacchi curr_chain = (Dwarf_Chain)
186*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_CHAIN, 1);
187*7fd79137SRobert Mustacchi if (curr_chain == NULL) {
188*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
189*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
190*7fd79137SRobert Mustacchi }
191*7fd79137SRobert Mustacchi curr_chain->ch_item = arange;
192*7fd79137SRobert Mustacchi if (head_chain == NULL)
193*7fd79137SRobert Mustacchi head_chain = prev_chain = curr_chain;
194*7fd79137SRobert Mustacchi else {
195*7fd79137SRobert Mustacchi prev_chain->ch_next = curr_chain;
196*7fd79137SRobert Mustacchi prev_chain = curr_chain;
197*7fd79137SRobert Mustacchi }
198*7fd79137SRobert Mustacchi
199*7fd79137SRobert Mustacchi
200*7fd79137SRobert Mustacchi instoff = fdep->fd_fde_instr_start - dbg->de_debug_frame.dss_data;
201*7fd79137SRobert Mustacchi instructions_length = fdep->fd_length +
202*7fd79137SRobert Mustacchi fdep->fd_length_size + fdep->fd_extension_size -
203*7fd79137SRobert Mustacchi (fdep->fd_fde_instr_start - fdep->fd_fde_start);
204*7fd79137SRobert Mustacchi instr_end = fdep->fd_fde_instr_start + instructions_length;
205*7fd79137SRobert Mustacchi res = _dwarf_exec_frame_instr( /* make_instr */ true,
206*7fd79137SRobert Mustacchi &frame_inst,
207*7fd79137SRobert Mustacchi /* search_pc= */ false,
208*7fd79137SRobert Mustacchi /* search_pc_val= */ 0,
209*7fd79137SRobert Mustacchi /* location */ 0,
210*7fd79137SRobert Mustacchi fdep->fd_fde_instr_start,
211*7fd79137SRobert Mustacchi instr_end,
212*7fd79137SRobert Mustacchi /* Dwarf_frame= */ 0,
213*7fd79137SRobert Mustacchi /* cie= */ 0,
214*7fd79137SRobert Mustacchi dbg,
215*7fd79137SRobert Mustacchi DW_FRAME_CFA_COL,
216*7fd79137SRobert Mustacchi &icount, &dw_err);
217*7fd79137SRobert Mustacchi if (res == DW_DLV_ERROR) {
218*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, dw_err);
219*7fd79137SRobert Mustacchi return (res);
220*7fd79137SRobert Mustacchi } else if (res == DW_DLV_NO_ENTRY) {
221*7fd79137SRobert Mustacchi continue;
222*7fd79137SRobert Mustacchi }
223*7fd79137SRobert Mustacchi
224*7fd79137SRobert Mustacchi for (j = 0; j < icount; ++j) {
225*7fd79137SRobert Mustacchi Dwarf_Frame_Op *finst2 = frame_inst + j;
226*7fd79137SRobert Mustacchi
227*7fd79137SRobert Mustacchi if (finst2->fp_base_op == 0 && finst2->fp_extended_op == 1) {
228*7fd79137SRobert Mustacchi /* is DW_CFA_set_loc */
229*7fd79137SRobert Mustacchi Dwarf_Addr add = (Dwarf_Addr) finst2->fp_offset;
230*7fd79137SRobert Mustacchi Dwarf_Off off = finst2->fp_instr_offset + instoff;
231*7fd79137SRobert Mustacchi
232*7fd79137SRobert Mustacchi arange = (Dwarf_Arange)
233*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_ARANGE, 1);
234*7fd79137SRobert Mustacchi if (arange == NULL) {
235*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
236*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
237*7fd79137SRobert Mustacchi }
238*7fd79137SRobert Mustacchi arange->ar_address = add;
239*7fd79137SRobert Mustacchi arange->ar_info_offset = off;
240*7fd79137SRobert Mustacchi arange_count++;
241*7fd79137SRobert Mustacchi curr_chain = (Dwarf_Chain)
242*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_CHAIN, 1);
243*7fd79137SRobert Mustacchi if (curr_chain == NULL) {
244*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
245*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
246*7fd79137SRobert Mustacchi }
247*7fd79137SRobert Mustacchi curr_chain->ch_item = arange;
248*7fd79137SRobert Mustacchi if (head_chain == NULL)
249*7fd79137SRobert Mustacchi head_chain = prev_chain = curr_chain;
250*7fd79137SRobert Mustacchi else {
251*7fd79137SRobert Mustacchi prev_chain->ch_next = curr_chain;
252*7fd79137SRobert Mustacchi prev_chain = curr_chain;
253*7fd79137SRobert Mustacchi }
254*7fd79137SRobert Mustacchi
255*7fd79137SRobert Mustacchi }
256*7fd79137SRobert Mustacchi }
257*7fd79137SRobert Mustacchi dwarf_dealloc(dbg, frame_inst, DW_DLA_FRAME_BLOCK);
258*7fd79137SRobert Mustacchi
259*7fd79137SRobert Mustacchi }
260*7fd79137SRobert Mustacchi dwarf_dealloc(dbg, fde_data, DW_DLA_LIST);
261*7fd79137SRobert Mustacchi dwarf_dealloc(dbg, cie_data, DW_DLA_LIST);
262*7fd79137SRobert Mustacchi arange_addrs = (Dwarf_Addr *)
263*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_ADDR, arange_count);
264*7fd79137SRobert Mustacchi if (arange_addrs == NULL) {
265*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
266*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
267*7fd79137SRobert Mustacchi }
268*7fd79137SRobert Mustacchi arange_offsets = (Dwarf_Off *)
269*7fd79137SRobert Mustacchi _dwarf_get_alloc(dbg, DW_DLA_ADDR, arange_count);
270*7fd79137SRobert Mustacchi if (arange_offsets == NULL) {
271*7fd79137SRobert Mustacchi _dwarf_error(dbg, err, DW_DLE_ALLOC_FAIL);
272*7fd79137SRobert Mustacchi return (DW_DLV_ERROR);
273*7fd79137SRobert Mustacchi }
274*7fd79137SRobert Mustacchi
275*7fd79137SRobert Mustacchi curr_chain = head_chain;
276*7fd79137SRobert Mustacchi for (i = 0; i < arange_count; i++) {
277*7fd79137SRobert Mustacchi Dwarf_Arange ar = curr_chain->ch_item;
278*7fd79137SRobert Mustacchi
279*7fd79137SRobert Mustacchi arange_addrs[i] = ar->ar_address;
280*7fd79137SRobert Mustacchi arange_offsets[i] = ar->ar_info_offset;
281*7fd79137SRobert Mustacchi prev_chain = curr_chain;
282*7fd79137SRobert Mustacchi curr_chain = curr_chain->ch_next;
283*7fd79137SRobert Mustacchi dwarf_dealloc(dbg, ar, DW_DLA_ARANGE);
284*7fd79137SRobert Mustacchi dwarf_dealloc(dbg, prev_chain, DW_DLA_CHAIN);
285*7fd79137SRobert Mustacchi }
286*7fd79137SRobert Mustacchi *returncount = arange_count;
287*7fd79137SRobert Mustacchi *offsetlist = arange_offsets;
288*7fd79137SRobert Mustacchi *addrlist = arange_addrs;
289*7fd79137SRobert Mustacchi return retval;
290*7fd79137SRobert Mustacchi }
291