1da0c48c4Sopenharmony_ci/* Return DWARF attribute associated with a location expression op. 2da0c48c4Sopenharmony_ci Copyright (C) 2013, 2014, 2017, 2018 Red Hat, Inc. 3da0c48c4Sopenharmony_ci This file is part of elfutils. 4da0c48c4Sopenharmony_ci 5da0c48c4Sopenharmony_ci This file is free software; you can redistribute it and/or modify 6da0c48c4Sopenharmony_ci it under the terms of either 7da0c48c4Sopenharmony_ci 8da0c48c4Sopenharmony_ci * the GNU Lesser General Public License as published by the Free 9da0c48c4Sopenharmony_ci Software Foundation; either version 3 of the License, or (at 10da0c48c4Sopenharmony_ci your option) any later version 11da0c48c4Sopenharmony_ci 12da0c48c4Sopenharmony_ci or 13da0c48c4Sopenharmony_ci 14da0c48c4Sopenharmony_ci * the GNU General Public License as published by the Free 15da0c48c4Sopenharmony_ci Software Foundation; either version 2 of the License, or (at 16da0c48c4Sopenharmony_ci your option) any later version 17da0c48c4Sopenharmony_ci 18da0c48c4Sopenharmony_ci or both in parallel, as here. 19da0c48c4Sopenharmony_ci 20da0c48c4Sopenharmony_ci elfutils is distributed in the hope that it will be useful, but 21da0c48c4Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 22da0c48c4Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 23da0c48c4Sopenharmony_ci General Public License for more details. 24da0c48c4Sopenharmony_ci 25da0c48c4Sopenharmony_ci You should have received copies of the GNU General Public License and 26da0c48c4Sopenharmony_ci the GNU Lesser General Public License along with this program. If 27da0c48c4Sopenharmony_ci not, see <http://www.gnu.org/licenses/>. */ 28da0c48c4Sopenharmony_ci 29da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H 30da0c48c4Sopenharmony_ci# include <config.h> 31da0c48c4Sopenharmony_ci#endif 32da0c48c4Sopenharmony_ci 33da0c48c4Sopenharmony_ci#include <dwarf.h> 34da0c48c4Sopenharmony_ci#include <libdwP.h> 35da0c48c4Sopenharmony_ci 36da0c48c4Sopenharmony_cistatic Dwarf_CU * 37da0c48c4Sopenharmony_ciattr_form_cu (Dwarf_Attribute *attr) 38da0c48c4Sopenharmony_ci{ 39da0c48c4Sopenharmony_ci /* If the attribute has block/expr form the data comes from the 40da0c48c4Sopenharmony_ci .debug_info from the same cu as the attr. Otherwise it comes from 41da0c48c4Sopenharmony_ci the .debug_loc or .debug_loclists data section. */ 42da0c48c4Sopenharmony_ci switch (attr->form) 43da0c48c4Sopenharmony_ci { 44da0c48c4Sopenharmony_ci case DW_FORM_block1: 45da0c48c4Sopenharmony_ci case DW_FORM_block2: 46da0c48c4Sopenharmony_ci case DW_FORM_block4: 47da0c48c4Sopenharmony_ci case DW_FORM_block: 48da0c48c4Sopenharmony_ci case DW_FORM_exprloc: 49da0c48c4Sopenharmony_ci return attr->cu; 50da0c48c4Sopenharmony_ci default: 51da0c48c4Sopenharmony_ci return (attr->cu->version < 5 52da0c48c4Sopenharmony_ci ? attr->cu->dbg->fake_loc_cu 53da0c48c4Sopenharmony_ci : attr->cu->dbg->fake_loclists_cu); 54da0c48c4Sopenharmony_ci } 55da0c48c4Sopenharmony_ci} 56da0c48c4Sopenharmony_ci 57da0c48c4Sopenharmony_cistatic unsigned char * 58da0c48c4Sopenharmony_ciaddr_valp (Dwarf_CU *cu, Dwarf_Word index) 59da0c48c4Sopenharmony_ci{ 60da0c48c4Sopenharmony_ci Elf_Data *debug_addr = cu->dbg->sectiondata[IDX_debug_addr]; 61da0c48c4Sopenharmony_ci if (debug_addr == NULL) 62da0c48c4Sopenharmony_ci { 63da0c48c4Sopenharmony_ci __libdw_seterrno (DWARF_E_NO_DEBUG_ADDR); 64da0c48c4Sopenharmony_ci return NULL; 65da0c48c4Sopenharmony_ci } 66da0c48c4Sopenharmony_ci 67da0c48c4Sopenharmony_ci Dwarf_Word offset = __libdw_cu_addr_base (cu) + (index * cu->address_size); 68da0c48c4Sopenharmony_ci return (unsigned char *) debug_addr->d_buf + offset; 69da0c48c4Sopenharmony_ci} 70da0c48c4Sopenharmony_ci 71da0c48c4Sopenharmony_ciint 72da0c48c4Sopenharmony_cidwarf_getlocation_attr (Dwarf_Attribute *attr, const Dwarf_Op *op, Dwarf_Attribute *result) 73da0c48c4Sopenharmony_ci{ 74da0c48c4Sopenharmony_ci if (attr == NULL) 75da0c48c4Sopenharmony_ci return -1; 76da0c48c4Sopenharmony_ci 77da0c48c4Sopenharmony_ci switch (op->atom) 78da0c48c4Sopenharmony_ci { 79da0c48c4Sopenharmony_ci case DW_OP_implicit_value: 80da0c48c4Sopenharmony_ci result->code = DW_AT_const_value; 81da0c48c4Sopenharmony_ci result->form = DW_FORM_block; 82da0c48c4Sopenharmony_ci result->valp = (unsigned char *) (uintptr_t) op->number2; 83da0c48c4Sopenharmony_ci result->cu = attr_form_cu (attr); 84da0c48c4Sopenharmony_ci break; 85da0c48c4Sopenharmony_ci 86da0c48c4Sopenharmony_ci case DW_OP_entry_value: 87da0c48c4Sopenharmony_ci case DW_OP_GNU_entry_value: 88da0c48c4Sopenharmony_ci result->code = DW_AT_location; 89da0c48c4Sopenharmony_ci result->form = DW_FORM_exprloc; 90da0c48c4Sopenharmony_ci result->valp = (unsigned char *) (uintptr_t) op->number2; 91da0c48c4Sopenharmony_ci result->cu = attr_form_cu (attr); 92da0c48c4Sopenharmony_ci break; 93da0c48c4Sopenharmony_ci 94da0c48c4Sopenharmony_ci case DW_OP_const_type: 95da0c48c4Sopenharmony_ci case DW_OP_GNU_const_type: 96da0c48c4Sopenharmony_ci result->code = DW_AT_const_value; 97da0c48c4Sopenharmony_ci result->form = DW_FORM_block1; 98da0c48c4Sopenharmony_ci result->valp = (unsigned char *) (uintptr_t) op->number2; 99da0c48c4Sopenharmony_ci result->cu = attr_form_cu (attr); 100da0c48c4Sopenharmony_ci break; 101da0c48c4Sopenharmony_ci 102da0c48c4Sopenharmony_ci case DW_OP_GNU_const_index: 103da0c48c4Sopenharmony_ci case DW_OP_constx: 104da0c48c4Sopenharmony_ci result->code = DW_AT_const_value; 105da0c48c4Sopenharmony_ci if (attr->cu->address_size == 4) 106da0c48c4Sopenharmony_ci result->form = DW_FORM_data4; 107da0c48c4Sopenharmony_ci else 108da0c48c4Sopenharmony_ci result->form = DW_FORM_data8; 109da0c48c4Sopenharmony_ci result->valp = addr_valp (attr->cu, op->number); 110da0c48c4Sopenharmony_ci if (result->valp == NULL) 111da0c48c4Sopenharmony_ci return -1; 112da0c48c4Sopenharmony_ci result->cu = attr->cu->dbg->fake_addr_cu; 113da0c48c4Sopenharmony_ci break; 114da0c48c4Sopenharmony_ci 115da0c48c4Sopenharmony_ci case DW_OP_GNU_addr_index: 116da0c48c4Sopenharmony_ci case DW_OP_addrx: 117da0c48c4Sopenharmony_ci result->code = DW_AT_low_pc; 118da0c48c4Sopenharmony_ci result->form = DW_FORM_addr; 119da0c48c4Sopenharmony_ci result->valp = addr_valp (attr->cu, op->number); 120da0c48c4Sopenharmony_ci if (result->valp == NULL) 121da0c48c4Sopenharmony_ci return -1; 122da0c48c4Sopenharmony_ci result->cu = attr->cu->dbg->fake_addr_cu; 123da0c48c4Sopenharmony_ci break; 124da0c48c4Sopenharmony_ci 125da0c48c4Sopenharmony_ci case DW_OP_call2: 126da0c48c4Sopenharmony_ci case DW_OP_call4: 127da0c48c4Sopenharmony_ci case DW_OP_call_ref: 128da0c48c4Sopenharmony_ci { 129da0c48c4Sopenharmony_ci Dwarf_Die die; 130da0c48c4Sopenharmony_ci if (INTUSE(dwarf_getlocation_die) (attr, op, &die) != 0) 131da0c48c4Sopenharmony_ci return -1; 132da0c48c4Sopenharmony_ci if (INTUSE(dwarf_attr) (&die, DW_AT_location, result) == NULL) 133da0c48c4Sopenharmony_ci { 134da0c48c4Sopenharmony_ci __libdw_empty_loc_attr (result); 135da0c48c4Sopenharmony_ci return 0; 136da0c48c4Sopenharmony_ci } 137da0c48c4Sopenharmony_ci } 138da0c48c4Sopenharmony_ci break; 139da0c48c4Sopenharmony_ci 140da0c48c4Sopenharmony_ci case DW_OP_implicit_pointer: 141da0c48c4Sopenharmony_ci case DW_OP_GNU_implicit_pointer: 142da0c48c4Sopenharmony_ci case DW_OP_GNU_variable_value: 143da0c48c4Sopenharmony_ci { 144da0c48c4Sopenharmony_ci Dwarf_Die die; 145da0c48c4Sopenharmony_ci if (INTUSE(dwarf_getlocation_die) (attr, op, &die) != 0) 146da0c48c4Sopenharmony_ci return -1; 147da0c48c4Sopenharmony_ci if (INTUSE(dwarf_attr) (&die, DW_AT_location, result) == NULL 148da0c48c4Sopenharmony_ci && INTUSE(dwarf_attr) (&die, DW_AT_const_value, result) == NULL) 149da0c48c4Sopenharmony_ci { 150da0c48c4Sopenharmony_ci __libdw_empty_loc_attr (result); 151da0c48c4Sopenharmony_ci return 0; 152da0c48c4Sopenharmony_ci } 153da0c48c4Sopenharmony_ci } 154da0c48c4Sopenharmony_ci break; 155da0c48c4Sopenharmony_ci 156da0c48c4Sopenharmony_ci default: 157da0c48c4Sopenharmony_ci __libdw_seterrno (DWARF_E_INVALID_ACCESS); 158da0c48c4Sopenharmony_ci return -1; 159da0c48c4Sopenharmony_ci } 160da0c48c4Sopenharmony_ci 161da0c48c4Sopenharmony_ci return 0; 162da0c48c4Sopenharmony_ci} 163