|  | //===- PDBFileBuilder.cpp - PDB File Creation -------------------*- C++ -*-===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "llvm/DebugInfo/PDB/Native/PDBFileBuilder.h" | 
|  | #include "llvm/DebugInfo/CodeView/CodeView.h" | 
|  | #include "llvm/DebugInfo/CodeView/GUID.h" | 
|  | #include "llvm/DebugInfo/MSF/MSFBuilder.h" | 
|  | #include "llvm/DebugInfo/MSF/MSFCommon.h" | 
|  | #include "llvm/DebugInfo/MSF/MappedBlockStream.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/DbiStreamBuilder.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/GSIStreamBuilder.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/InfoStreamBuilder.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/RawConstants.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/RawError.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/RawTypes.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/TpiStreamBuilder.h" | 
|  | #include "llvm/Support/BinaryStreamWriter.h" | 
|  | #include "llvm/Support/CRC.h" | 
|  | #include "llvm/Support/Path.h" | 
|  | #include "llvm/Support/xxhash.h" | 
|  |  | 
|  | #include <ctime> | 
|  |  | 
|  | using namespace llvm; | 
|  | using namespace llvm::codeview; | 
|  | using namespace llvm::msf; | 
|  | using namespace llvm::pdb; | 
|  | using namespace llvm::support; | 
|  |  | 
|  | namespace llvm { | 
|  | class WritableBinaryStream; | 
|  | } | 
|  |  | 
|  | PDBFileBuilder::PDBFileBuilder(BumpPtrAllocator &Allocator) | 
|  | : Allocator(Allocator), InjectedSourceHashTraits(Strings), | 
|  | InjectedSourceTable(2) {} | 
|  |  | 
|  | PDBFileBuilder::~PDBFileBuilder() = default; | 
|  |  | 
|  | Error PDBFileBuilder::initialize(uint32_t BlockSize) { | 
|  | auto ExpectedMsf = MSFBuilder::create(Allocator, BlockSize); | 
|  | if (!ExpectedMsf) | 
|  | return ExpectedMsf.takeError(); | 
|  | Msf = std::make_unique<MSFBuilder>(std::move(*ExpectedMsf)); | 
|  | return Error::success(); | 
|  | } | 
|  |  | 
|  | MSFBuilder &PDBFileBuilder::getMsfBuilder() { return *Msf; } | 
|  |  | 
|  | InfoStreamBuilder &PDBFileBuilder::getInfoBuilder() { | 
|  | if (!Info) | 
|  | Info = std::make_unique<InfoStreamBuilder>(*Msf, NamedStreams); | 
|  | return *Info; | 
|  | } | 
|  |  | 
|  | DbiStreamBuilder &PDBFileBuilder::getDbiBuilder() { | 
|  | if (!Dbi) | 
|  | Dbi = std::make_unique<DbiStreamBuilder>(*Msf); | 
|  | return *Dbi; | 
|  | } | 
|  |  | 
|  | TpiStreamBuilder &PDBFileBuilder::getTpiBuilder() { | 
|  | if (!Tpi) | 
|  | Tpi = std::make_unique<TpiStreamBuilder>(*Msf, StreamTPI); | 
|  | return *Tpi; | 
|  | } | 
|  |  | 
|  | TpiStreamBuilder &PDBFileBuilder::getIpiBuilder() { | 
|  | if (!Ipi) | 
|  | Ipi = std::make_unique<TpiStreamBuilder>(*Msf, StreamIPI); | 
|  | return *Ipi; | 
|  | } | 
|  |  | 
|  | PDBStringTableBuilder &PDBFileBuilder::getStringTableBuilder() { | 
|  | return Strings; | 
|  | } | 
|  |  | 
|  | GSIStreamBuilder &PDBFileBuilder::getGsiBuilder() { | 
|  | if (!Gsi) | 
|  | Gsi = std::make_unique<GSIStreamBuilder>(*Msf); | 
|  | return *Gsi; | 
|  | } | 
|  |  | 
|  | Expected<uint32_t> PDBFileBuilder::allocateNamedStream(StringRef Name, | 
|  | uint32_t Size) { | 
|  | auto ExpectedStream = Msf->addStream(Size); | 
|  | if (ExpectedStream) | 
|  | NamedStreams.set(Name, *ExpectedStream); | 
|  | return ExpectedStream; | 
|  | } | 
|  |  | 
|  | Error PDBFileBuilder::addNamedStream(StringRef Name, StringRef Data) { | 
|  | Expected<uint32_t> ExpectedIndex = allocateNamedStream(Name, Data.size()); | 
|  | if (!ExpectedIndex) | 
|  | return ExpectedIndex.takeError(); | 
|  | assert(NamedStreamData.count(*ExpectedIndex) == 0); | 
|  | NamedStreamData[*ExpectedIndex] = std::string(Data); | 
|  | return Error::success(); | 
|  | } | 
|  |  | 
|  | void PDBFileBuilder::addInjectedSource(StringRef Name, | 
|  | std::unique_ptr<MemoryBuffer> Buffer) { | 
|  | // Stream names must be exact matches, since they get looked up in a hash | 
|  | // table and the hash value is dependent on the exact contents of the string. | 
|  | // link.exe lowercases a path and converts / to \, so we must do the same. | 
|  | SmallString<64> VName; | 
|  | sys::path::native(Name.lower(), VName, sys::path::Style::windows_backslash); | 
|  |  | 
|  | uint32_t NI = getStringTableBuilder().insert(Name); | 
|  | uint32_t VNI = getStringTableBuilder().insert(VName); | 
|  |  | 
|  | InjectedSourceDescriptor Desc; | 
|  | Desc.Content = std::move(Buffer); | 
|  | Desc.NameIndex = NI; | 
|  | Desc.VNameIndex = VNI; | 
|  | Desc.StreamName = "/src/files/"; | 
|  |  | 
|  | Desc.StreamName += VName; | 
|  |  | 
|  | InjectedSources.push_back(std::move(Desc)); | 
|  | } | 
|  |  | 
|  | Error PDBFileBuilder::finalizeMsfLayout() { | 
|  |  | 
|  | if (Ipi && Ipi->getRecordCount() > 0) { | 
|  | // In theory newer PDBs always have an ID stream, but by saying that we're | 
|  | // only going to *really* have an ID stream if there is at least one ID | 
|  | // record, we leave open the opportunity to test older PDBs such as those | 
|  | // that don't have an ID stream. | 
|  | auto &Info = getInfoBuilder(); | 
|  | Info.addFeature(PdbRaw_FeatureSig::VC140); | 
|  | } | 
|  |  | 
|  | uint32_t StringsLen = Strings.calculateSerializedSize(); | 
|  |  | 
|  | Expected<uint32_t> SN = allocateNamedStream("/LinkInfo", 0); | 
|  | if (!SN) | 
|  | return SN.takeError(); | 
|  |  | 
|  | if (Gsi) { | 
|  | if (auto EC = Gsi->finalizeMsfLayout()) | 
|  | return EC; | 
|  | if (Dbi) { | 
|  | Dbi->setPublicsStreamIndex(Gsi->getPublicsStreamIndex()); | 
|  | Dbi->setGlobalsStreamIndex(Gsi->getGlobalsStreamIndex()); | 
|  | Dbi->setSymbolRecordStreamIndex(Gsi->getRecordStreamIndex()); | 
|  | } | 
|  | } | 
|  | if (Tpi) { | 
|  | if (auto EC = Tpi->finalizeMsfLayout()) | 
|  | return EC; | 
|  | } | 
|  | if (Dbi) { | 
|  | if (auto EC = Dbi->finalizeMsfLayout()) | 
|  | return EC; | 
|  | } | 
|  | SN = allocateNamedStream("/names", StringsLen); | 
|  | if (!SN) | 
|  | return SN.takeError(); | 
|  |  | 
|  | if (Ipi) { | 
|  | if (auto EC = Ipi->finalizeMsfLayout()) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | // Do this last, since it relies on the named stream map being complete, and | 
|  | // that can be updated by previous steps in the finalization. | 
|  | if (Info) { | 
|  | if (auto EC = Info->finalizeMsfLayout()) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | if (!InjectedSources.empty()) { | 
|  | for (const auto &IS : InjectedSources) { | 
|  | JamCRC CRC(0); | 
|  | CRC.update(arrayRefFromStringRef(IS.Content->getBuffer())); | 
|  |  | 
|  | SrcHeaderBlockEntry Entry; | 
|  | ::memset(&Entry, 0, sizeof(SrcHeaderBlockEntry)); | 
|  | Entry.Size = sizeof(SrcHeaderBlockEntry); | 
|  | Entry.FileSize = IS.Content->getBufferSize(); | 
|  | Entry.FileNI = IS.NameIndex; | 
|  | Entry.VFileNI = IS.VNameIndex; | 
|  | Entry.ObjNI = 1; | 
|  | Entry.IsVirtual = 0; | 
|  | Entry.Version = | 
|  | static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne); | 
|  | Entry.CRC = CRC.getCRC(); | 
|  | StringRef VName = getStringTableBuilder().getStringForId(IS.VNameIndex); | 
|  | InjectedSourceTable.set_as(VName, std::move(Entry), | 
|  | InjectedSourceHashTraits); | 
|  | } | 
|  |  | 
|  | uint32_t SrcHeaderBlockSize = | 
|  | sizeof(SrcHeaderBlockHeader) + | 
|  | InjectedSourceTable.calculateSerializedLength(); | 
|  | SN = allocateNamedStream("/src/headerblock", SrcHeaderBlockSize); | 
|  | if (!SN) | 
|  | return SN.takeError(); | 
|  | for (const auto &IS : InjectedSources) { | 
|  | SN = allocateNamedStream(IS.StreamName, IS.Content->getBufferSize()); | 
|  | if (!SN) | 
|  | return SN.takeError(); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Do this last, since it relies on the named stream map being complete, and | 
|  | // that can be updated by previous steps in the finalization. | 
|  | if (Info) { | 
|  | if (auto EC = Info->finalizeMsfLayout()) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | return Error::success(); | 
|  | } | 
|  |  | 
|  | Expected<uint32_t> PDBFileBuilder::getNamedStreamIndex(StringRef Name) const { | 
|  | uint32_t SN = 0; | 
|  | if (!NamedStreams.get(Name, SN)) | 
|  | return llvm::make_error<pdb::RawError>(raw_error_code::no_stream); | 
|  | return SN; | 
|  | } | 
|  |  | 
|  | void PDBFileBuilder::commitSrcHeaderBlock(WritableBinaryStream &MsfBuffer, | 
|  | const msf::MSFLayout &Layout) { | 
|  | assert(!InjectedSourceTable.empty()); | 
|  |  | 
|  | uint32_t SN = cantFail(getNamedStreamIndex("/src/headerblock")); | 
|  | auto Stream = WritableMappedBlockStream::createIndexedStream( | 
|  | Layout, MsfBuffer, SN, Allocator); | 
|  | BinaryStreamWriter Writer(*Stream); | 
|  |  | 
|  | SrcHeaderBlockHeader Header; | 
|  | ::memset(&Header, 0, sizeof(Header)); | 
|  | Header.Version = static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne); | 
|  | Header.Size = Writer.bytesRemaining(); | 
|  |  | 
|  | cantFail(Writer.writeObject(Header)); | 
|  | cantFail(InjectedSourceTable.commit(Writer)); | 
|  |  | 
|  | assert(Writer.bytesRemaining() == 0); | 
|  | } | 
|  |  | 
|  | void PDBFileBuilder::commitInjectedSources(WritableBinaryStream &MsfBuffer, | 
|  | const msf::MSFLayout &Layout) { | 
|  | if (InjectedSourceTable.empty()) | 
|  | return; | 
|  |  | 
|  | commitSrcHeaderBlock(MsfBuffer, Layout); | 
|  |  | 
|  | for (const auto &IS : InjectedSources) { | 
|  | uint32_t SN = cantFail(getNamedStreamIndex(IS.StreamName)); | 
|  |  | 
|  | auto SourceStream = WritableMappedBlockStream::createIndexedStream( | 
|  | Layout, MsfBuffer, SN, Allocator); | 
|  | BinaryStreamWriter SourceWriter(*SourceStream); | 
|  | assert(SourceWriter.bytesRemaining() == IS.Content->getBufferSize()); | 
|  | cantFail(SourceWriter.writeBytes( | 
|  | arrayRefFromStringRef(IS.Content->getBuffer()))); | 
|  | } | 
|  | } | 
|  |  | 
|  | Error PDBFileBuilder::commit(StringRef Filename, codeview::GUID *Guid) { | 
|  | assert(!Filename.empty()); | 
|  | if (auto EC = finalizeMsfLayout()) | 
|  | return EC; | 
|  |  | 
|  | MSFLayout Layout; | 
|  | Expected<FileBufferByteStream> ExpectedMsfBuffer = | 
|  | Msf->commit(Filename, Layout); | 
|  | if (!ExpectedMsfBuffer) | 
|  | return ExpectedMsfBuffer.takeError(); | 
|  | FileBufferByteStream Buffer = std::move(*ExpectedMsfBuffer); | 
|  |  | 
|  | auto ExpectedSN = getNamedStreamIndex("/names"); | 
|  | if (!ExpectedSN) | 
|  | return ExpectedSN.takeError(); | 
|  |  | 
|  | auto NS = WritableMappedBlockStream::createIndexedStream( | 
|  | Layout, Buffer, *ExpectedSN, Allocator); | 
|  | BinaryStreamWriter NSWriter(*NS); | 
|  | if (auto EC = Strings.commit(NSWriter)) | 
|  | return EC; | 
|  |  | 
|  | for (const auto &NSE : NamedStreamData) { | 
|  | if (NSE.second.empty()) | 
|  | continue; | 
|  |  | 
|  | auto NS = WritableMappedBlockStream::createIndexedStream( | 
|  | Layout, Buffer, NSE.first, Allocator); | 
|  | BinaryStreamWriter NSW(*NS); | 
|  | if (auto EC = NSW.writeBytes(arrayRefFromStringRef(NSE.second))) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | if (Info) { | 
|  | if (auto EC = Info->commit(Layout, Buffer)) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | if (Dbi) { | 
|  | if (auto EC = Dbi->commit(Layout, Buffer)) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | if (Tpi) { | 
|  | if (auto EC = Tpi->commit(Layout, Buffer)) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | if (Ipi) { | 
|  | if (auto EC = Ipi->commit(Layout, Buffer)) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | if (Gsi) { | 
|  | if (auto EC = Gsi->commit(Layout, Buffer)) | 
|  | return EC; | 
|  | } | 
|  |  | 
|  | auto InfoStreamBlocks = Layout.StreamMap[StreamPDB]; | 
|  | assert(!InfoStreamBlocks.empty()); | 
|  | uint64_t InfoStreamFileOffset = | 
|  | blockToOffset(InfoStreamBlocks.front(), Layout.SB->BlockSize); | 
|  | InfoStreamHeader *H = reinterpret_cast<InfoStreamHeader *>( | 
|  | Buffer.getBufferStart() + InfoStreamFileOffset); | 
|  |  | 
|  | commitInjectedSources(Buffer, Layout); | 
|  |  | 
|  | // Set the build id at the very end, after every other byte of the PDB | 
|  | // has been written. | 
|  | if (Info->hashPDBContentsToGUID()) { | 
|  | // Compute a hash of all sections of the output file. | 
|  | uint64_t Digest = | 
|  | xxHash64({Buffer.getBufferStart(), Buffer.getBufferEnd()}); | 
|  |  | 
|  | H->Age = 1; | 
|  |  | 
|  | memcpy(H->Guid.Guid, &Digest, 8); | 
|  | // xxhash only gives us 8 bytes, so put some fixed data in the other half. | 
|  | memcpy(H->Guid.Guid + 8, "LLD PDB.", 8); | 
|  |  | 
|  | // Put the hash in the Signature field too. | 
|  | H->Signature = static_cast<uint32_t>(Digest); | 
|  |  | 
|  | // Return GUID to caller. | 
|  | memcpy(Guid, H->Guid.Guid, 16); | 
|  | } else { | 
|  | H->Age = Info->getAge(); | 
|  | H->Guid = Info->getGuid(); | 
|  | Optional<uint32_t> Sig = Info->getSignature(); | 
|  | H->Signature = Sig ? *Sig : time(nullptr); | 
|  | } | 
|  |  | 
|  | return Buffer.commit(); | 
|  | } |