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.
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000023 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,
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000027 SectionName.data(), *OffsetPtr);
Wolfgang Pieb439801b2018-07-23 22:37:17 +000028 // TODO: Add support for DWARF64.
29 HeaderData.Length = Data.getU32(OffsetPtr);
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000030 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;
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000035 if (HeaderData.Length + sizeof(uint32_t) < sizeof(Header))
36 return createStringError(errc::invalid_argument,
37 "%s table at offset 0x%" PRIx32
38 " 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))
43 return createStringError(errc::invalid_argument,
44 "section is not large enough to contain a %s table "
45 "of length 0x%" PRIx32 " at offset 0x%" PRIx32,
46 SectionName.data(), length(), HeaderOffset);
Wolfgang Pieb439801b2018-07-23 22:37:17 +000047
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.
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000054 if (HeaderData.Version != 5)
Victor Leschukcba595d2018-08-20 09:59:08 +000055 return createStringError(errc::invalid_argument,
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000056 "unrecognised %s table version %" PRIu16
57 " 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);
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000064 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) +
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +000070 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
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +0000104uint32_t DWARFListTableHeader::length() const {
Wolfgang Pieb439801b2018-07-23 22:37:17 +0000105 if (HeaderData.Length == 0)
106 return 0;
107 // TODO: DWARF64 support.
Alexander Kornienkoe74e0f12018-09-17 15:40:01 +0000108 return HeaderData.Length + sizeof(uint32_t);
Wolfgang Pieb439801b2018-07-23 22:37:17 +0000109}