forked from OSchip/llvm-project
189 lines
5.4 KiB
C++
189 lines
5.4 KiB
C++
//===- PDBFileBuilder.cpp - PDB File Creation -------------------*- C++ -*-===//
|
|
//
|
|
// 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/Native/PDBFileBuilder.h"
|
|
|
|
#include "llvm/ADT/BitVector.h"
|
|
|
|
#include "llvm/DebugInfo/MSF/MSFBuilder.h"
|
|
#include "llvm/DebugInfo/PDB/GenericError.h"
|
|
#include "llvm/DebugInfo/PDB/Native/DbiStream.h"
|
|
#include "llvm/DebugInfo/PDB/Native/DbiStreamBuilder.h"
|
|
#include "llvm/DebugInfo/PDB/Native/InfoStream.h"
|
|
#include "llvm/DebugInfo/PDB/Native/InfoStreamBuilder.h"
|
|
#include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h"
|
|
#include "llvm/DebugInfo/PDB/Native/RawError.h"
|
|
#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
|
|
#include "llvm/DebugInfo/PDB/Native/TpiStreamBuilder.h"
|
|
#include "llvm/Support/BinaryStream.h"
|
|
#include "llvm/Support/BinaryStreamWriter.h"
|
|
|
|
using namespace llvm;
|
|
using namespace llvm::codeview;
|
|
using namespace llvm::msf;
|
|
using namespace llvm::pdb;
|
|
using namespace llvm::support;
|
|
|
|
PDBFileBuilder::PDBFileBuilder(BumpPtrAllocator &Allocator)
|
|
: Allocator(Allocator) {}
|
|
|
|
Error PDBFileBuilder::initialize(uint32_t BlockSize) {
|
|
auto ExpectedMsf = MSFBuilder::create(Allocator, BlockSize);
|
|
if (!ExpectedMsf)
|
|
return ExpectedMsf.takeError();
|
|
Msf = llvm::make_unique<MSFBuilder>(std::move(*ExpectedMsf));
|
|
return Error::success();
|
|
}
|
|
|
|
MSFBuilder &PDBFileBuilder::getMsfBuilder() { return *Msf; }
|
|
|
|
InfoStreamBuilder &PDBFileBuilder::getInfoBuilder() {
|
|
if (!Info)
|
|
Info = llvm::make_unique<InfoStreamBuilder>(*Msf, NamedStreams);
|
|
return *Info;
|
|
}
|
|
|
|
DbiStreamBuilder &PDBFileBuilder::getDbiBuilder() {
|
|
if (!Dbi)
|
|
Dbi = llvm::make_unique<DbiStreamBuilder>(*Msf);
|
|
return *Dbi;
|
|
}
|
|
|
|
TpiStreamBuilder &PDBFileBuilder::getTpiBuilder() {
|
|
if (!Tpi)
|
|
Tpi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamTPI);
|
|
return *Tpi;
|
|
}
|
|
|
|
TpiStreamBuilder &PDBFileBuilder::getIpiBuilder() {
|
|
if (!Ipi)
|
|
Ipi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamIPI);
|
|
return *Ipi;
|
|
}
|
|
|
|
PDBStringTableBuilder &PDBFileBuilder::getStringTableBuilder() {
|
|
return Strings;
|
|
}
|
|
|
|
Error PDBFileBuilder::addNamedStream(StringRef Name, uint32_t Size) {
|
|
auto ExpectedStream = Msf->addStream(Size);
|
|
if (!ExpectedStream)
|
|
return ExpectedStream.takeError();
|
|
NamedStreams.set(Name, *ExpectedStream);
|
|
return Error::success();
|
|
}
|
|
|
|
Expected<msf::MSFLayout> PDBFileBuilder::finalizeMsfLayout() {
|
|
uint32_t StringsLen = Strings.calculateSerializedSize();
|
|
|
|
if (auto EC = addNamedStream("/names", StringsLen))
|
|
return std::move(EC);
|
|
if (auto EC = addNamedStream("/LinkInfo", 0))
|
|
return std::move(EC);
|
|
if (auto EC = addNamedStream("/src/headerblock", 0))
|
|
return std::move(EC);
|
|
|
|
if (Info) {
|
|
if (auto EC = Info->finalizeMsfLayout())
|
|
return std::move(EC);
|
|
}
|
|
if (Dbi) {
|
|
if (auto EC = Dbi->finalizeMsfLayout())
|
|
return std::move(EC);
|
|
}
|
|
if (Tpi) {
|
|
if (auto EC = Tpi->finalizeMsfLayout())
|
|
return std::move(EC);
|
|
}
|
|
if (Ipi) {
|
|
if (auto EC = Ipi->finalizeMsfLayout())
|
|
return std::move(EC);
|
|
}
|
|
|
|
return Msf->build();
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
Error PDBFileBuilder::commit(StringRef Filename) {
|
|
assert(!Filename.empty());
|
|
auto ExpectedLayout = finalizeMsfLayout();
|
|
if (!ExpectedLayout)
|
|
return ExpectedLayout.takeError();
|
|
auto &Layout = *ExpectedLayout;
|
|
|
|
uint64_t Filesize = Layout.SB->BlockSize * Layout.SB->NumBlocks;
|
|
auto OutFileOrError = FileOutputBuffer::create(Filename, Filesize);
|
|
if (OutFileOrError.getError())
|
|
return llvm::make_error<pdb::GenericError>(generic_error_code::invalid_path,
|
|
Filename);
|
|
FileBufferByteStream Buffer(std::move(*OutFileOrError),
|
|
llvm::support::little);
|
|
BinaryStreamWriter Writer(Buffer);
|
|
|
|
if (auto EC = Writer.writeObject(*Layout.SB))
|
|
return EC;
|
|
uint32_t BlockMapOffset =
|
|
msf::blockToOffset(Layout.SB->BlockMapAddr, Layout.SB->BlockSize);
|
|
Writer.setOffset(BlockMapOffset);
|
|
if (auto EC = Writer.writeArray(Layout.DirectoryBlocks))
|
|
return EC;
|
|
|
|
auto DirStream = WritableMappedBlockStream::createDirectoryStream(
|
|
Layout, Buffer, Allocator);
|
|
BinaryStreamWriter DW(*DirStream);
|
|
if (auto EC = DW.writeInteger<uint32_t>(Layout.StreamSizes.size()))
|
|
return EC;
|
|
|
|
if (auto EC = DW.writeArray(Layout.StreamSizes))
|
|
return EC;
|
|
|
|
for (const auto &Blocks : Layout.StreamMap) {
|
|
if (auto EC = DW.writeArray(Blocks))
|
|
return EC;
|
|
}
|
|
|
|
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;
|
|
|
|
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;
|
|
}
|
|
|
|
return Buffer.commit();
|
|
}
|