Path: blob/main/contrib/llvm-project/llvm/lib/DebugInfo/DWARF/DWARFDataExtractor.cpp
35266 views
//===- DWARFDataExtractor.cpp ---------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#include "llvm/DebugInfo/DWARF/DWARFDataExtractor.h"9#include "llvm/DebugInfo/DWARF/DWARFObject.h"10#include "llvm/DebugInfo/DWARF/DWARFRelocMap.h"11#include "llvm/Support/Errc.h"1213using namespace llvm;1415std::pair<uint64_t, dwarf::DwarfFormat>16DWARFDataExtractor::getInitialLength(uint64_t *Off, Error *Err) const {17ErrorAsOutParameter ErrAsOut(Err);18if (Err && *Err)19return {0, dwarf::DWARF32};2021Cursor C(*Off);22uint64_t Length = getRelocatedValue(C, 4);23dwarf::DwarfFormat Format = dwarf::DWARF32;24if (Length == dwarf::DW_LENGTH_DWARF64) {25Length = getRelocatedValue(C, 8);26Format = dwarf::DWARF64;27} else if (Length >= dwarf::DW_LENGTH_lo_reserved) {28cantFail(C.takeError());29if (Err)30*Err = createStringError(31errc::invalid_argument,32"unsupported reserved unit length of value 0x%8.8" PRIx64, Length);33return {0, dwarf::DWARF32};34}3536if (C) {37*Off = C.tell();38return {Length, Format};39}40if (Err)41*Err = C.takeError();42else43consumeError(C.takeError());44return {0, dwarf::DWARF32};45}4647uint64_t DWARFDataExtractor::getRelocatedValue(uint32_t Size, uint64_t *Off,48uint64_t *SecNdx,49Error *Err) const {50if (SecNdx)51*SecNdx = object::SectionedAddress::UndefSection;52if (!Section)53return getUnsigned(Off, Size, Err);5455ErrorAsOutParameter ErrAsOut(Err);56std::optional<RelocAddrEntry> E = Obj->find(*Section, *Off);57uint64_t LocData = getUnsigned(Off, Size, Err);58if (!E || (Err && *Err))59return LocData;60if (SecNdx)61*SecNdx = E->SectionIndex;6263uint64_t R =64object::resolveRelocation(E->Resolver, E->Reloc, E->SymbolValue, LocData);65if (E->Reloc2)66R = object::resolveRelocation(E->Resolver, *E->Reloc2, E->SymbolValue2, R);67return R;68}6970std::optional<uint64_t>71DWARFDataExtractor::getEncodedPointer(uint64_t *Offset, uint8_t Encoding,72uint64_t PCRelOffset) const {73if (Encoding == dwarf::DW_EH_PE_omit)74return std::nullopt;7576uint64_t Result = 0;77uint64_t OldOffset = *Offset;78// First get value79switch (Encoding & 0x0F) {80case dwarf::DW_EH_PE_absptr:81switch (getAddressSize()) {82case 2:83case 4:84case 8:85Result = getUnsigned(Offset, getAddressSize());86break;87default:88return std::nullopt;89}90break;91case dwarf::DW_EH_PE_uleb128:92Result = getULEB128(Offset);93break;94case dwarf::DW_EH_PE_sleb128:95Result = getSLEB128(Offset);96break;97case dwarf::DW_EH_PE_udata2:98Result = getUnsigned(Offset, 2);99break;100case dwarf::DW_EH_PE_udata4:101Result = getUnsigned(Offset, 4);102break;103case dwarf::DW_EH_PE_udata8:104Result = getUnsigned(Offset, 8);105break;106case dwarf::DW_EH_PE_sdata2:107Result = getSigned(Offset, 2);108break;109case dwarf::DW_EH_PE_sdata4:110Result = SignExtend64<32>(getRelocatedValue(4, Offset));111break;112case dwarf::DW_EH_PE_sdata8:113Result = getRelocatedValue(8, Offset);114break;115default:116return std::nullopt;117}118// Then add relative offset, if required119switch (Encoding & 0x70) {120case dwarf::DW_EH_PE_absptr:121// do nothing122break;123case dwarf::DW_EH_PE_pcrel:124Result += PCRelOffset;125break;126case dwarf::DW_EH_PE_datarel:127case dwarf::DW_EH_PE_textrel:128case dwarf::DW_EH_PE_funcrel:129case dwarf::DW_EH_PE_aligned:130default:131*Offset = OldOffset;132return std::nullopt;133}134135return Result;136}137138139