blob: db2ea20ce14702c37dc73ae9dec4d17c71da1353 [file] [log] [blame]
Zachary Turnerf5c59652016-05-03 00:28:21 +00001//===- TpiStream.cpp - PDB Type Info (TPI) Stream 2 Access ----------------===//
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/PDB/Raw/TpiStream.h"
11
12#include "llvm/DebugInfo/CodeView/CodeView.h"
13#include "llvm/DebugInfo/CodeView/TypeRecord.h"
14#include "llvm/DebugInfo/PDB/Raw/MappedBlockStream.h"
15#include "llvm/DebugInfo/PDB/Raw/RawConstants.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000016#include "llvm/DebugInfo/PDB/Raw/RawError.h"
Zachary Turnerf5c59652016-05-03 00:28:21 +000017#include "llvm/DebugInfo/PDB/Raw/StreamReader.h"
18
19#include "llvm/Support/Endian.h"
20
21using namespace llvm;
22using namespace llvm::support;
23using namespace llvm::pdb;
24
25namespace {
Zachary Turnerf5c59652016-05-03 00:28:21 +000026const uint32_t MinHashBuckets = 0x1000;
27const uint32_t MaxHashBuckets = 0x40000;
28}
29
30static uint32_t HashBufferV8(uint8_t *buffer, uint32_t NumBuckets) {
31 // Not yet implemented, this is probably some variation of CRC32 but we need
32 // to be sure of the precise implementation otherwise we won't be able to work
33 // with persisted hash values.
34 return 0;
35}
36
37struct TpiStream::HeaderInfo {
38 struct EmbeddedBuf {
39 little32_t Off;
40 ulittle32_t Length;
41 };
42
43 ulittle32_t Version;
44 ulittle32_t HeaderSize;
45 ulittle32_t TypeIndexBegin;
46 ulittle32_t TypeIndexEnd;
47 ulittle32_t TypeRecordBytes;
48
49 ulittle16_t HashStreamIndex;
50 ulittle16_t HashAuxStreamIndex;
51 ulittle32_t HashKeySize;
52 ulittle32_t NumHashBuckets;
53
54 EmbeddedBuf HashValueBuffer;
55 EmbeddedBuf IndexOffsetBuffer;
56 EmbeddedBuf HashAdjBuffer;
57};
58
59TpiStream::TpiStream(PDBFile &File)
60 : Pdb(File), Stream(StreamTPI, File), HashFunction(nullptr) {}
61
62TpiStream::~TpiStream() {}
63
Zachary Turner819e77d2016-05-06 20:51:57 +000064Error TpiStream::reload() {
Zachary Turnerf5c59652016-05-03 00:28:21 +000065 StreamReader Reader(Stream);
66
67 if (Reader.bytesRemaining() < sizeof(HeaderInfo))
Zachary Turner819e77d2016-05-06 20:51:57 +000068 return make_error<RawError>(raw_error_code::corrupt_file,
69 "TPI Stream does not contain a header.");
Zachary Turnerf5c59652016-05-03 00:28:21 +000070
71 Header.reset(new HeaderInfo());
Zachary Turner819e77d2016-05-06 20:51:57 +000072 if (Reader.readObject(Header.get()))
73 return make_error<RawError>(raw_error_code::corrupt_file,
74 "TPI Stream does not contain a header.");
Zachary Turnerf5c59652016-05-03 00:28:21 +000075
76 if (Header->Version != PdbTpiV80)
Zachary Turner819e77d2016-05-06 20:51:57 +000077 return make_error<RawError>(raw_error_code::corrupt_file,
78 "Unsupported TPI Version.");
Zachary Turnerf5c59652016-05-03 00:28:21 +000079
80 if (Header->HeaderSize != sizeof(HeaderInfo))
Zachary Turner819e77d2016-05-06 20:51:57 +000081 return make_error<RawError>(raw_error_code::corrupt_file,
82 "Corrupt TPI Header size.");
Zachary Turnerf5c59652016-05-03 00:28:21 +000083
84 if (Header->HashKeySize != sizeof(ulittle32_t))
Zachary Turner819e77d2016-05-06 20:51:57 +000085 return make_error<RawError>(raw_error_code::corrupt_file,
86 "TPI Stream expected 4 byte hash key size.");
Zachary Turnerf5c59652016-05-03 00:28:21 +000087
88 if (Header->NumHashBuckets < MinHashBuckets ||
89 Header->NumHashBuckets > MaxHashBuckets)
Zachary Turner819e77d2016-05-06 20:51:57 +000090 return make_error<RawError>(raw_error_code::corrupt_file,
91 "TPI Stream Invalid number of hash buckets.");
Zachary Turnerf5c59652016-05-03 00:28:21 +000092
93 HashFunction = HashBufferV8;
94
95 // The actual type records themselves come from this stream
Zachary Turner819e77d2016-05-06 20:51:57 +000096 if (auto EC = RecordsBuffer.initialize(Reader, Header->TypeRecordBytes))
97 return EC;
Zachary Turnerf5c59652016-05-03 00:28:21 +000098
99 // Hash indices, hash values, etc come from the hash stream.
100 MappedBlockStream HS(Header->HashStreamIndex, Pdb);
101 StreamReader HSR(HS);
102 HSR.setOffset(Header->HashValueBuffer.Off);
Zachary Turner819e77d2016-05-06 20:51:57 +0000103 if (auto EC =
104 HashValuesBuffer.initialize(HSR, Header->HashValueBuffer.Length))
105 return EC;
Zachary Turnerf5c59652016-05-03 00:28:21 +0000106
107 HSR.setOffset(Header->HashAdjBuffer.Off);
Zachary Turner819e77d2016-05-06 20:51:57 +0000108 if (auto EC = HashAdjBuffer.initialize(HSR, Header->HashAdjBuffer.Length))
109 return EC;
Zachary Turnerf5c59652016-05-03 00:28:21 +0000110
111 HSR.setOffset(Header->IndexOffsetBuffer.Off);
Zachary Turner819e77d2016-05-06 20:51:57 +0000112 if (auto EC = TypeIndexOffsetBuffer.initialize(
113 HSR, Header->IndexOffsetBuffer.Length))
114 return EC;
Zachary Turnerf5c59652016-05-03 00:28:21 +0000115
Zachary Turner819e77d2016-05-06 20:51:57 +0000116 return Error::success();
Zachary Turnerf5c59652016-05-03 00:28:21 +0000117}
118
119PdbRaw_TpiVer TpiStream::getTpiVersion() const {
120 uint32_t Value = Header->Version;
121 return static_cast<PdbRaw_TpiVer>(Value);
122}
123
124uint32_t TpiStream::TypeIndexBegin() const { return Header->TypeIndexBegin; }
125
126uint32_t TpiStream::TypeIndexEnd() const { return Header->TypeIndexEnd; }
127
128uint32_t TpiStream::NumTypeRecords() const {
129 return TypeIndexEnd() - TypeIndexBegin();
130}
131
Reid Klecknerce5196e2016-05-12 23:26:23 +0000132iterator_range<codeview::TypeIterator> TpiStream::types(bool *HadError) const {
133 return codeview::makeTypeRange(RecordsBuffer.data(), /*HadError=*/HadError);
Zachary Turnerf5c59652016-05-03 00:28:21 +0000134}