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 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