blob: 462c036d73ada4a188f5d4fc03d755241da435b2 [file] [log] [blame]
Wolfgang Pieb439801b2018-07-23 22:37:17 +00001//===- DWARFListTable.cpp ---------------------------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "llvm/DebugInfo/DWARF/DWARFListTable.h"
11#include "llvm/BinaryFormat/Dwarf.h"
Victor Leschukcba595d2018-08-20 09:59:08 +000012#include "llvm/Support/Errc.h"
Wolfgang Pieb439801b2018-07-23 22:37:17 +000013#include "llvm/Support/Error.h"
14#include "llvm/Support/Format.h"
15#include "llvm/Support/raw_ostream.h"
16
17using namespace llvm;
18
Wolfgang Pieb439801b2018-07-23 22:37:17 +000019Error DWARFListTableHeader::extract(DWARFDataExtractor Data,
20 uint32_t *OffsetPtr) {
21 HeaderOffset = *OffsetPtr;
22 // Read and verify the length field.
23 if (!Data.isValidOffsetForDataOfSize(*OffsetPtr, sizeof(uint32_t)))
Victor Leschukcba595d2018-08-20 09:59:08 +000024 return createStringError(errc::invalid_argument,
25 "section is not large enough to contain a "
Wolfgang Pieb439801b2018-07-23 22:37:17 +000026 "%s table length at offset 0x%" PRIx32,
27 SectionName.data(), *OffsetPtr);
28 // TODO: Add support for DWARF64.
29 HeaderData.Length = Data.getU32(OffsetPtr);
30 if (HeaderData.Length == 0xffffffffu)
Victor Leschukcba595d2018-08-20 09:59:08 +000031 return createStringError(errc::not_supported,
32 "DWARF64 is not supported in %s at offset 0x%" PRIx32,
Wolfgang Pieb439801b2018-07-23 22:37:17 +000033 SectionName.data(), HeaderOffset);
34 Format = dwarf::DwarfFormat::DWARF32;
35 if (HeaderData.Length + sizeof(uint32_t) < sizeof(Header))
Victor Leschukcba595d2018-08-20 09:59:08 +000036 return createStringError(errc::invalid_argument,
37 "%s table at offset 0x%" PRIx32
Wolfgang Pieb439801b2018-07-23 22:37:17 +000038 " has too small length (0x%" PRIx32
39 ") to contain a complete header",
40 SectionName.data(), HeaderOffset, length());
41 uint32_t End = HeaderOffset + length();
42 if (!Data.isValidOffsetForDataOfSize(HeaderOffset, End - HeaderOffset))
Victor Leschukcba595d2018-08-20 09:59:08 +000043 return createStringError(errc::invalid_argument,
44 "section is not large enough to contain a %s table "
Wolfgang Pieb439801b2018-07-23 22:37:17 +000045 "of length 0x%" PRIx32 " at offset 0x%" PRIx32,
46 SectionName.data(), length(), HeaderOffset);
47
48 HeaderData.Version = Data.getU16(OffsetPtr);
49 HeaderData.AddrSize = Data.getU8(OffsetPtr);
50 HeaderData.SegSize = Data.getU8(OffsetPtr);
51 HeaderData.OffsetEntryCount = Data.getU32(OffsetPtr);
52
53 // Perform basic validation of the remaining header fields.
54 if (HeaderData.Version != 5)
Victor Leschukcba595d2018-08-20 09:59:08 +000055 return createStringError(errc::invalid_argument,
56 "unrecognised %s table version %" PRIu16
Wolfgang Pieb439801b2018-07-23 22:37:17 +000057 " in table at offset 0x%" PRIx32,
58 SectionName.data(), HeaderData.Version, HeaderOffset);
59 if (HeaderData.AddrSize != 4 && HeaderData.AddrSize != 8)
Victor Leschukcba595d2018-08-20 09:59:08 +000060 return createStringError(errc::not_supported,
61 "%s table at offset 0x%" PRIx32
62 " has unsupported address size %" PRIu8,
Wolfgang Pieb439801b2018-07-23 22:37:17 +000063 SectionName.data(), HeaderOffset, HeaderData.AddrSize);
64 if (HeaderData.SegSize != 0)
Victor Leschukcba595d2018-08-20 09:59:08 +000065 return createStringError(errc::not_supported,
66 "%s table at offset 0x%" PRIx32
Wolfgang Pieb439801b2018-07-23 22:37:17 +000067 " has unsupported segment selector size %" PRIu8,
68 SectionName.data(), HeaderOffset, HeaderData.SegSize);
69 if (End < HeaderOffset + sizeof(HeaderData) +
70 HeaderData.OffsetEntryCount * sizeof(uint32_t))
Victor Leschukcba595d2018-08-20 09:59:08 +000071 return createStringError(errc::invalid_argument,
Wolfgang Pieb439801b2018-07-23 22:37:17 +000072 "%s table at offset 0x%" PRIx32 " has more offset entries (%" PRIu32
73 ") than there is space for",
74 SectionName.data(), HeaderOffset, HeaderData.OffsetEntryCount);
75 Data.setAddressSize(HeaderData.AddrSize);
76 for (uint32_t I = 0; I < HeaderData.OffsetEntryCount; ++I)
77 Offsets.push_back(Data.getU32(OffsetPtr));
78 return Error::success();
79}
80
81void DWARFListTableHeader::dump(raw_ostream &OS, DIDumpOptions DumpOpts) const {
82 if (DumpOpts.Verbose)
83 OS << format("0x%8.8" PRIx32 ": ", HeaderOffset);
84 OS << format(
85 "%s list header: length = 0x%8.8" PRIx32 ", version = 0x%4.4" PRIx16 ", "
86 "addr_size = 0x%2.2" PRIx8 ", seg_size = 0x%2.2" PRIx8
87 ", offset_entry_count = "
88 "0x%8.8" PRIx32 "\n",
89 ListTypeString.data(), HeaderData.Length, HeaderData.Version,
90 HeaderData.AddrSize, HeaderData.SegSize, HeaderData.OffsetEntryCount);
91
92 if (HeaderData.OffsetEntryCount > 0) {
93 OS << "offsets: [";
94 for (const auto &Off : Offsets) {
95 OS << format("\n0x%8.8" PRIx32, Off);
96 if (DumpOpts.Verbose)
97 OS << format(" => 0x%8.8" PRIx32,
98 Off + HeaderOffset + sizeof(HeaderData));
99 }
100 OS << "\n]\n";
101 }
102}
103
104uint32_t DWARFListTableHeader::length() const {
105 if (HeaderData.Length == 0)
106 return 0;
107 // TODO: DWARF64 support.
108 return HeaderData.Length + sizeof(uint32_t);
109}