2016-05-03 08:28:21 +08:00
|
|
|
//===- TpiStream.cpp - PDB Type Info (TPI) Stream 2 Access ----------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/DebugInfo/PDB/Raw/TpiStream.h"
|
|
|
|
|
|
|
|
#include "llvm/DebugInfo/CodeView/CodeView.h"
|
|
|
|
#include "llvm/DebugInfo/CodeView/TypeRecord.h"
|
|
|
|
#include "llvm/DebugInfo/PDB/Raw/MappedBlockStream.h"
|
|
|
|
#include "llvm/DebugInfo/PDB/Raw/RawConstants.h"
|
2016-05-07 04:51:57 +08:00
|
|
|
#include "llvm/DebugInfo/PDB/Raw/RawError.h"
|
2016-05-03 08:28:21 +08:00
|
|
|
#include "llvm/DebugInfo/PDB/Raw/StreamReader.h"
|
|
|
|
|
|
|
|
#include "llvm/Support/Endian.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace llvm::support;
|
|
|
|
using namespace llvm::pdb;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
const uint32_t MinHashBuckets = 0x1000;
|
|
|
|
const uint32_t MaxHashBuckets = 0x40000;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t HashBufferV8(uint8_t *buffer, uint32_t NumBuckets) {
|
|
|
|
// Not yet implemented, this is probably some variation of CRC32 but we need
|
|
|
|
// to be sure of the precise implementation otherwise we won't be able to work
|
|
|
|
// with persisted hash values.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct TpiStream::HeaderInfo {
|
|
|
|
struct EmbeddedBuf {
|
|
|
|
little32_t Off;
|
|
|
|
ulittle32_t Length;
|
|
|
|
};
|
|
|
|
|
|
|
|
ulittle32_t Version;
|
|
|
|
ulittle32_t HeaderSize;
|
|
|
|
ulittle32_t TypeIndexBegin;
|
|
|
|
ulittle32_t TypeIndexEnd;
|
|
|
|
ulittle32_t TypeRecordBytes;
|
|
|
|
|
|
|
|
ulittle16_t HashStreamIndex;
|
|
|
|
ulittle16_t HashAuxStreamIndex;
|
|
|
|
ulittle32_t HashKeySize;
|
|
|
|
ulittle32_t NumHashBuckets;
|
|
|
|
|
|
|
|
EmbeddedBuf HashValueBuffer;
|
|
|
|
EmbeddedBuf IndexOffsetBuffer;
|
|
|
|
EmbeddedBuf HashAdjBuffer;
|
|
|
|
};
|
|
|
|
|
2016-05-25 12:35:22 +08:00
|
|
|
TpiStream::TpiStream(PDBFile &File, uint32_t StreamIdx)
|
|
|
|
: Pdb(File), Stream(StreamIdx, File), HashFunction(nullptr) {}
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
TpiStream::~TpiStream() {}
|
|
|
|
|
2016-05-07 04:51:57 +08:00
|
|
|
Error TpiStream::reload() {
|
2016-05-03 08:28:21 +08:00
|
|
|
StreamReader Reader(Stream);
|
|
|
|
|
|
|
|
if (Reader.bytesRemaining() < sizeof(HeaderInfo))
|
2016-05-07 04:51:57 +08:00
|
|
|
return make_error<RawError>(raw_error_code::corrupt_file,
|
|
|
|
"TPI Stream does not contain a header.");
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
Header.reset(new HeaderInfo());
|
2016-05-07 04:51:57 +08:00
|
|
|
if (Reader.readObject(Header.get()))
|
|
|
|
return make_error<RawError>(raw_error_code::corrupt_file,
|
|
|
|
"TPI Stream does not contain a header.");
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
if (Header->Version != PdbTpiV80)
|
2016-05-07 04:51:57 +08:00
|
|
|
return make_error<RawError>(raw_error_code::corrupt_file,
|
|
|
|
"Unsupported TPI Version.");
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
if (Header->HeaderSize != sizeof(HeaderInfo))
|
2016-05-07 04:51:57 +08:00
|
|
|
return make_error<RawError>(raw_error_code::corrupt_file,
|
|
|
|
"Corrupt TPI Header size.");
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
if (Header->HashKeySize != sizeof(ulittle32_t))
|
2016-05-07 04:51:57 +08:00
|
|
|
return make_error<RawError>(raw_error_code::corrupt_file,
|
|
|
|
"TPI Stream expected 4 byte hash key size.");
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
if (Header->NumHashBuckets < MinHashBuckets ||
|
|
|
|
Header->NumHashBuckets > MaxHashBuckets)
|
2016-05-07 04:51:57 +08:00
|
|
|
return make_error<RawError>(raw_error_code::corrupt_file,
|
|
|
|
"TPI Stream Invalid number of hash buckets.");
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
HashFunction = HashBufferV8;
|
|
|
|
|
|
|
|
// The actual type records themselves come from this stream
|
2016-05-07 04:51:57 +08:00
|
|
|
if (auto EC = RecordsBuffer.initialize(Reader, Header->TypeRecordBytes))
|
|
|
|
return EC;
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
// Hash indices, hash values, etc come from the hash stream.
|
|
|
|
MappedBlockStream HS(Header->HashStreamIndex, Pdb);
|
|
|
|
StreamReader HSR(HS);
|
|
|
|
HSR.setOffset(Header->HashValueBuffer.Off);
|
2016-05-07 04:51:57 +08:00
|
|
|
if (auto EC =
|
|
|
|
HashValuesBuffer.initialize(HSR, Header->HashValueBuffer.Length))
|
|
|
|
return EC;
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
HSR.setOffset(Header->HashAdjBuffer.Off);
|
2016-05-07 04:51:57 +08:00
|
|
|
if (auto EC = HashAdjBuffer.initialize(HSR, Header->HashAdjBuffer.Length))
|
|
|
|
return EC;
|
2016-05-03 08:28:21 +08:00
|
|
|
|
|
|
|
HSR.setOffset(Header->IndexOffsetBuffer.Off);
|
2016-05-07 04:51:57 +08:00
|
|
|
if (auto EC = TypeIndexOffsetBuffer.initialize(
|
|
|
|
HSR, Header->IndexOffsetBuffer.Length))
|
|
|
|
return EC;
|
2016-05-03 08:28:21 +08:00
|
|
|
|
2016-05-07 04:51:57 +08:00
|
|
|
return Error::success();
|
2016-05-03 08:28:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
PdbRaw_TpiVer TpiStream::getTpiVersion() const {
|
|
|
|
uint32_t Value = Header->Version;
|
|
|
|
return static_cast<PdbRaw_TpiVer>(Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t TpiStream::TypeIndexBegin() const { return Header->TypeIndexBegin; }
|
|
|
|
|
|
|
|
uint32_t TpiStream::TypeIndexEnd() const { return Header->TypeIndexEnd; }
|
|
|
|
|
|
|
|
uint32_t TpiStream::NumTypeRecords() const {
|
|
|
|
return TypeIndexEnd() - TypeIndexBegin();
|
|
|
|
}
|
|
|
|
|
2016-05-25 11:43:17 +08:00
|
|
|
uint16_t TpiStream::getTypeHashStreamIndex() const {
|
|
|
|
return Header->HashStreamIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t TpiStream::getTypeHashStreamAuxIndex() const {
|
|
|
|
return Header->HashAuxStreamIndex;
|
|
|
|
}
|
|
|
|
|
2016-05-13 07:26:23 +08:00
|
|
|
iterator_range<codeview::TypeIterator> TpiStream::types(bool *HadError) const {
|
|
|
|
return codeview::makeTypeRange(RecordsBuffer.data(), /*HadError=*/HadError);
|
2016-05-03 08:28:21 +08:00
|
|
|
}
|