//===- GlobalTypeTableBuilder.cpp -----------------------------------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/DebugInfo/CodeView/GlobalTypeTableBuilder.h" #include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/DenseSet.h" #include "llvm/ADT/STLExtras.h" #include "llvm/DebugInfo/CodeView/CodeView.h" #include "llvm/DebugInfo/CodeView/ContinuationRecordBuilder.h" #include "llvm/DebugInfo/CodeView/RecordSerialization.h" #include "llvm/DebugInfo/CodeView/TypeIndex.h" #include "llvm/Support/Allocator.h" #include "llvm/Support/BinaryByteStream.h" #include "llvm/Support/BinaryStreamWriter.h" #include "llvm/Support/Endian.h" #include "llvm/Support/Error.h" #include #include #include #include using namespace llvm; using namespace llvm::codeview; TypeIndex GlobalTypeTableBuilder::nextTypeIndex() const { return TypeIndex::fromArrayIndex(SeenRecords.size()); } GlobalTypeTableBuilder::GlobalTypeTableBuilder(BumpPtrAllocator &Storage) : RecordStorage(Storage) { SeenRecords.reserve(4096); } GlobalTypeTableBuilder::~GlobalTypeTableBuilder() = default; Optional GlobalTypeTableBuilder::getFirst() { if (empty()) return None; return TypeIndex(TypeIndex::FirstNonSimpleIndex); } Optional GlobalTypeTableBuilder::getNext(TypeIndex Prev) { if (++Prev == nextTypeIndex()) return None; return Prev; } CVType GlobalTypeTableBuilder::getType(TypeIndex Index) { CVType Type; Type.RecordData = SeenRecords[Index.toArrayIndex()]; const RecordPrefix *P = reinterpret_cast(Type.RecordData.data()); Type.Type = static_cast(uint16_t(P->RecordKind)); return Type; } StringRef GlobalTypeTableBuilder::getTypeName(TypeIndex Index) { llvm_unreachable("Method not implemented"); } bool GlobalTypeTableBuilder::contains(TypeIndex Index) { if (Index.isSimple() || Index.isNoneType()) return false; return Index.toArrayIndex() < SeenRecords.size(); } uint32_t GlobalTypeTableBuilder::size() { return SeenRecords.size(); } uint32_t GlobalTypeTableBuilder::capacity() { return SeenRecords.size(); } ArrayRef> GlobalTypeTableBuilder::records() const { return SeenRecords; } ArrayRef GlobalTypeTableBuilder::hashes() const { return SeenHashes; } void GlobalTypeTableBuilder::reset() { HashedRecords.clear(); SeenRecords.clear(); } static inline ArrayRef stabilize(BumpPtrAllocator &Alloc, ArrayRef Data) { uint8_t *Stable = Alloc.Allocate(Data.size()); memcpy(Stable, Data.data(), Data.size()); return makeArrayRef(Stable, Data.size()); } TypeIndex GlobalTypeTableBuilder::insertRecordAs(GloballyHashedType Hash, CreateRecord Create) { auto Result = HashedRecords.try_emplace(Hash, nextTypeIndex()); if (Result.second) { ArrayRef RecordData = stabilize(RecordStorage, Create()); SeenRecords.push_back(RecordData); SeenHashes.push_back(Hash); } // Update the caller's copy of Record to point a stable copy. return Result.first->second; } TypeIndex GlobalTypeTableBuilder::insertRecordBytes(ArrayRef Record) { GloballyHashedType GHT = GloballyHashedType::hashType(Record, SeenHashes, SeenHashes); return insertRecordAs(GHT, [Record]() { return Record; }); } TypeIndex GlobalTypeTableBuilder::insertRecord(ContinuationRecordBuilder &Builder) { TypeIndex TI; auto Fragments = Builder.end(nextTypeIndex()); assert(!Fragments.empty()); for (auto C : Fragments) TI = insertRecordBytes(C.RecordData); return TI; }