2018-08-01 06:19:19 +08:00
|
|
|
//===- DWARFDebugAddr.cpp -------------------------------------------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-08-01 06:19:19 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFDebugAddr.h"
|
|
|
|
#include "llvm/BinaryFormat/Dwarf.h"
|
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFUnit.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
2020-02-06 15:14:12 +08:00
|
|
|
Error DWARFDebugAddrTable::extractAddresses(const DWARFDataExtractor &Data,
|
|
|
|
uint64_t *OffsetPtr,
|
|
|
|
uint64_t EndOffset) {
|
|
|
|
assert(EndOffset >= *OffsetPtr);
|
|
|
|
uint64_t DataSize = EndOffset - *OffsetPtr;
|
|
|
|
assert(Data.isValidOffsetForDataOfSize(*OffsetPtr, DataSize));
|
|
|
|
if (AddrSize != 4 && AddrSize != 8)
|
|
|
|
return createStringError(errc::not_supported,
|
|
|
|
"address table at offset 0x%" PRIx64
|
|
|
|
" has unsupported address size %" PRIu8
|
|
|
|
" (4 and 8 are supported)",
|
|
|
|
Offset, AddrSize);
|
|
|
|
if (DataSize % AddrSize != 0) {
|
|
|
|
invalidateLength();
|
|
|
|
return createStringError(errc::invalid_argument,
|
|
|
|
"address table at offset 0x%" PRIx64
|
|
|
|
" contains data of size 0x%" PRIx64
|
|
|
|
" which is not a multiple of addr size %" PRIu8,
|
|
|
|
Offset, DataSize, AddrSize);
|
|
|
|
}
|
2018-08-01 06:19:19 +08:00
|
|
|
Addrs.clear();
|
2020-02-06 15:14:12 +08:00
|
|
|
size_t Count = DataSize / AddrSize;
|
|
|
|
Addrs.reserve(Count);
|
|
|
|
while (Count--)
|
|
|
|
Addrs.push_back(Data.getRelocatedValue(AddrSize, OffsetPtr));
|
|
|
|
return Error::success();
|
2018-08-01 06:19:19 +08:00
|
|
|
}
|
|
|
|
|
2020-02-06 15:14:12 +08:00
|
|
|
Error DWARFDebugAddrTable::extractV5(const DWARFDataExtractor &Data,
|
|
|
|
uint64_t *OffsetPtr, uint8_t CUAddrSize,
|
|
|
|
std::function<void(Error)> WarnCallback) {
|
|
|
|
Offset = *OffsetPtr;
|
|
|
|
// Check that we can read the unit length field.
|
|
|
|
if (!Data.isValidOffsetForDataOfSize(Offset, 4))
|
2018-08-01 06:19:19 +08:00
|
|
|
return createStringError(errc::invalid_argument,
|
2020-02-06 15:14:12 +08:00
|
|
|
"section is not large enough to contain an "
|
|
|
|
"address table length at offset 0x%" PRIx64,
|
|
|
|
Offset);
|
2018-08-01 06:19:19 +08:00
|
|
|
Format = dwarf::DwarfFormat::DWARF32;
|
2020-02-06 15:14:12 +08:00
|
|
|
Length = Data.getU32(OffsetPtr);
|
|
|
|
if (Length == dwarf::DW_LENGTH_DWARF64) {
|
2019-07-08 22:55:36 +08:00
|
|
|
// Check that we can read the extended unit length field.
|
|
|
|
if (!Data.isValidOffsetForDataOfSize(*OffsetPtr, 8)) {
|
|
|
|
invalidateLength();
|
|
|
|
return createStringError(
|
|
|
|
errc::invalid_argument,
|
|
|
|
"section is not large enough to contain an extended length field "
|
|
|
|
"of the address table at offset 0x%" PRIx64,
|
|
|
|
Offset);
|
|
|
|
}
|
|
|
|
Format = dwarf::DwarfFormat::DWARF64;
|
|
|
|
Length = Data.getU64(OffsetPtr);
|
|
|
|
} else if (Length >= dwarf::DW_LENGTH_lo_reserved) {
|
|
|
|
uint64_t DiagnosticLength = Length;
|
2020-02-06 15:14:12 +08:00
|
|
|
invalidateLength();
|
|
|
|
return createStringError(
|
|
|
|
errc::not_supported,
|
2019-07-08 22:55:36 +08:00
|
|
|
"address table at offset 0x%" PRIx64
|
|
|
|
" has unsupported reserved unit length of value 0x%" PRIx64,
|
|
|
|
Offset, DiagnosticLength);
|
2020-02-06 15:14:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!Data.isValidOffsetForDataOfSize(*OffsetPtr, Length)) {
|
2019-07-08 22:55:36 +08:00
|
|
|
uint64_t DiagnosticLength = Length;
|
2020-02-06 15:14:12 +08:00
|
|
|
invalidateLength();
|
|
|
|
return createStringError(
|
|
|
|
errc::invalid_argument,
|
|
|
|
"section is not large enough to contain an address table "
|
2019-07-08 22:55:36 +08:00
|
|
|
"at offset 0x%" PRIx64 " with a unit_length value of 0x%" PRIx64,
|
2020-02-06 15:14:12 +08:00
|
|
|
Offset, DiagnosticLength);
|
|
|
|
}
|
|
|
|
uint64_t EndOffset = *OffsetPtr + Length;
|
|
|
|
// Ensure that we can read the remaining header fields.
|
|
|
|
if (Length < 4) {
|
2019-07-08 22:55:36 +08:00
|
|
|
uint64_t DiagnosticLength = Length;
|
2020-02-06 15:14:12 +08:00
|
|
|
invalidateLength();
|
|
|
|
return createStringError(
|
|
|
|
errc::invalid_argument,
|
|
|
|
"address table at offset 0x%" PRIx64
|
2019-07-08 22:55:36 +08:00
|
|
|
" has a unit_length value of 0x%" PRIx64
|
2020-02-06 15:14:12 +08:00
|
|
|
", which is too small to contain a complete header",
|
|
|
|
Offset, DiagnosticLength);
|
2018-08-01 06:19:19 +08:00
|
|
|
}
|
|
|
|
|
2020-02-06 15:14:12 +08:00
|
|
|
Version = Data.getU16(OffsetPtr);
|
|
|
|
AddrSize = Data.getU8(OffsetPtr);
|
|
|
|
SegSize = Data.getU8(OffsetPtr);
|
2018-08-01 06:19:19 +08:00
|
|
|
|
2020-02-06 15:14:12 +08:00
|
|
|
// Perform a basic validation of the header fields.
|
|
|
|
if (Version != 5)
|
2020-02-06 17:30:30 +08:00
|
|
|
return createStringError(errc::not_supported,
|
|
|
|
"address table at offset 0x%" PRIx64
|
|
|
|
" has unsupported version %" PRIu16,
|
2020-02-06 15:14:12 +08:00
|
|
|
Offset, Version);
|
|
|
|
// TODO: add support for non-zero segment selector size.
|
|
|
|
if (SegSize != 0)
|
2018-08-01 06:19:19 +08:00
|
|
|
return createStringError(errc::not_supported,
|
2020-02-06 15:14:12 +08:00
|
|
|
"address table at offset 0x%" PRIx64
|
|
|
|
" has unsupported segment selector size %" PRIu8,
|
|
|
|
Offset, SegSize);
|
|
|
|
|
|
|
|
if (Error Err = extractAddresses(Data, OffsetPtr, EndOffset))
|
|
|
|
return Err;
|
|
|
|
if (CUAddrSize && AddrSize != CUAddrSize) {
|
2020-02-06 18:08:20 +08:00
|
|
|
WarnCallback(createStringError(
|
|
|
|
errc::invalid_argument,
|
2020-02-11 16:55:12 +08:00
|
|
|
"address table at offset 0x%" PRIx64 " has address size %" PRIu8
|
2020-02-06 18:08:20 +08:00
|
|
|
" which is different from CU address size %" PRIu8,
|
2020-02-06 15:14:12 +08:00
|
|
|
Offset, AddrSize, CUAddrSize));
|
2018-08-01 06:19:19 +08:00
|
|
|
}
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
2020-02-06 15:14:12 +08:00
|
|
|
Error DWARFDebugAddrTable::extractPreStandard(const DWARFDataExtractor &Data,
|
|
|
|
uint64_t *OffsetPtr,
|
|
|
|
uint16_t CUVersion,
|
|
|
|
uint8_t CUAddrSize) {
|
|
|
|
assert(CUVersion > 0 && CUVersion < 5);
|
|
|
|
|
|
|
|
Offset = *OffsetPtr;
|
|
|
|
Length = 0;
|
|
|
|
Version = CUVersion;
|
|
|
|
AddrSize = CUAddrSize;
|
|
|
|
SegSize = 0;
|
|
|
|
|
|
|
|
return extractAddresses(Data, OffsetPtr, Data.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
Error DWARFDebugAddrTable::extract(const DWARFDataExtractor &Data,
|
|
|
|
uint64_t *OffsetPtr,
|
|
|
|
uint16_t CUVersion,
|
|
|
|
uint8_t CUAddrSize,
|
|
|
|
std::function<void(Error)> WarnCallback) {
|
|
|
|
if (CUVersion > 0 && CUVersion < 5)
|
|
|
|
return extractPreStandard(Data, OffsetPtr, CUVersion, CUAddrSize);
|
|
|
|
if (CUVersion == 0)
|
|
|
|
WarnCallback(createStringError(errc::invalid_argument,
|
|
|
|
"DWARF version is not defined in CU,"
|
|
|
|
" assuming version 5"));
|
|
|
|
return extractV5(Data, OffsetPtr, CUAddrSize, WarnCallback);
|
|
|
|
}
|
|
|
|
|
2018-08-01 06:19:19 +08:00
|
|
|
void DWARFDebugAddrTable::dump(raw_ostream &OS, DIDumpOptions DumpOpts) const {
|
|
|
|
if (DumpOpts.Verbose)
|
2019-07-08 22:55:36 +08:00
|
|
|
OS << format("0x%8.8" PRIx64 ": ", Offset);
|
|
|
|
if (Length) {
|
|
|
|
int LengthFieldWidth = (Format == dwarf::DwarfFormat::DWARF64) ? 16 : 8;
|
|
|
|
OS << format("Address table header: length = 0x%0*" PRIx64
|
2020-02-06 15:14:12 +08:00
|
|
|
", version = 0x%4.4" PRIx16 ", addr_size = 0x%2.2" PRIx8
|
|
|
|
", seg_size = 0x%2.2" PRIx8 "\n",
|
2019-07-08 22:55:36 +08:00
|
|
|
LengthFieldWidth, Length, Version, AddrSize, SegSize);
|
|
|
|
}
|
2018-08-01 06:19:19 +08:00
|
|
|
|
|
|
|
if (Addrs.size() > 0) {
|
2020-02-06 15:14:12 +08:00
|
|
|
const char *AddrFmt =
|
|
|
|
(AddrSize == 4) ? "0x%8.8" PRIx64 "\n" : "0x%16.16" PRIx64 "\n";
|
2019-07-02 17:57:28 +08:00
|
|
|
OS << "Addrs: [\n";
|
|
|
|
for (uint64_t Addr : Addrs)
|
|
|
|
OS << format(AddrFmt, Addr);
|
|
|
|
OS << "]\n";
|
2018-08-01 06:19:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Expected<uint64_t> DWARFDebugAddrTable::getAddrEntry(uint32_t Index) const {
|
|
|
|
if (Index < Addrs.size())
|
|
|
|
return Addrs[Index];
|
|
|
|
return createStringError(errc::invalid_argument,
|
2020-02-06 15:14:12 +08:00
|
|
|
"Index %" PRIu32 " is out of range of the "
|
2020-02-11 16:55:12 +08:00
|
|
|
"address table at offset 0x%" PRIx64,
|
2020-02-06 15:14:12 +08:00
|
|
|
Index, Offset);
|
2018-08-01 06:19:19 +08:00
|
|
|
}
|
|
|
|
|
2020-02-06 15:14:12 +08:00
|
|
|
Optional<uint64_t> DWARFDebugAddrTable::getFullLength() const {
|
|
|
|
if (Length == 0)
|
|
|
|
return None;
|
2019-07-08 22:55:36 +08:00
|
|
|
return Length + dwarf::getUnitLengthFieldByteSize(Format);
|
2018-08-01 06:19:19 +08:00
|
|
|
}
|
|
|
|
|