Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 1 | //===- PDBFileBuilder.cpp - PDB File Creation -------------------*- C++ -*-===// |
| 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/PDBFileBuilder.h" |
| 11 | |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 12 | #include "llvm/ADT/BitVector.h" |
| 13 | |
Zachary Turner | a3225b0 | 2016-07-29 20:56:36 +0000 | [diff] [blame^] | 14 | #include "llvm/DebugInfo/MSF/MSFBuilder.h" |
| 15 | #include "llvm/DebugInfo/MSF/StreamInterface.h" |
| 16 | #include "llvm/DebugInfo/MSF/StreamWriter.h" |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 17 | #include "llvm/DebugInfo/PDB/Raw/DbiStream.h" |
| 18 | #include "llvm/DebugInfo/PDB/Raw/DbiStreamBuilder.h" |
| 19 | #include "llvm/DebugInfo/PDB/Raw/InfoStream.h" |
| 20 | #include "llvm/DebugInfo/PDB/Raw/InfoStreamBuilder.h" |
| 21 | #include "llvm/DebugInfo/PDB/Raw/RawError.h" |
| 22 | |
| 23 | using namespace llvm; |
| 24 | using namespace llvm::codeview; |
Zachary Turner | bac69d3 | 2016-07-22 19:56:05 +0000 | [diff] [blame] | 25 | using namespace llvm::msf; |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 26 | using namespace llvm::pdb; |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 27 | using namespace llvm::support; |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 28 | |
Zachary Turner | e109dc6 | 2016-07-22 19:56:26 +0000 | [diff] [blame] | 29 | PDBFileBuilder::PDBFileBuilder(BumpPtrAllocator &Allocator) |
| 30 | : Allocator(Allocator) {} |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 31 | |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 32 | Error PDBFileBuilder::initialize(const msf::SuperBlock &Super) { |
| 33 | auto ExpectedMsf = |
Zachary Turner | a3225b0 | 2016-07-29 20:56:36 +0000 | [diff] [blame^] | 34 | MSFBuilder::create(Allocator, Super.BlockSize, Super.NumBlocks); |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 35 | if (!ExpectedMsf) |
| 36 | return ExpectedMsf.takeError(); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 37 | |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 38 | auto &MsfResult = *ExpectedMsf; |
| 39 | if (auto EC = MsfResult.setBlockMapAddr(Super.BlockMapAddr)) |
| 40 | return EC; |
Zachary Turner | a3225b0 | 2016-07-29 20:56:36 +0000 | [diff] [blame^] | 41 | Msf = llvm::make_unique<MSFBuilder>(std::move(MsfResult)); |
Zachary Turner | b927e02 | 2016-07-15 22:17:19 +0000 | [diff] [blame] | 42 | Msf->setFreePageMap(Super.FreeBlockMapBlock); |
| 43 | Msf->setUnknown1(Super.Unknown1); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 44 | return Error::success(); |
| 45 | } |
| 46 | |
Zachary Turner | a3225b0 | 2016-07-29 20:56:36 +0000 | [diff] [blame^] | 47 | MSFBuilder &PDBFileBuilder::getMsfBuilder() { return *Msf; } |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 48 | |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 49 | InfoStreamBuilder &PDBFileBuilder::getInfoBuilder() { |
| 50 | if (!Info) |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 51 | Info = llvm::make_unique<InfoStreamBuilder>(); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 52 | return *Info; |
| 53 | } |
| 54 | |
| 55 | DbiStreamBuilder &PDBFileBuilder::getDbiBuilder() { |
| 56 | if (!Dbi) |
Zachary Turner | e109dc6 | 2016-07-22 19:56:26 +0000 | [diff] [blame] | 57 | Dbi = llvm::make_unique<DbiStreamBuilder>(Allocator); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 58 | return *Dbi; |
| 59 | } |
| 60 | |
Zachary Turner | a3225b0 | 2016-07-29 20:56:36 +0000 | [diff] [blame^] | 61 | Expected<msf::MSFLayout> PDBFileBuilder::finalizeMsfLayout() const { |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 62 | if (Info) { |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 63 | uint32_t Length = Info->calculateSerializedLength(); |
| 64 | if (auto EC = Msf->setStreamSize(StreamPDB, Length)) |
| 65 | return std::move(EC); |
| 66 | } |
| 67 | if (Dbi) { |
| 68 | uint32_t Length = Dbi->calculateSerializedLength(); |
| 69 | if (auto EC = Msf->setStreamSize(StreamDBI, Length)) |
| 70 | return std::move(EC); |
| 71 | } |
| 72 | |
Zachary Turner | 199f48a | 2016-07-28 19:11:09 +0000 | [diff] [blame] | 73 | return Msf->build(); |
| 74 | } |
| 75 | |
| 76 | Expected<std::unique_ptr<PDBFile>> |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 77 | PDBFileBuilder::build(std::unique_ptr<msf::WritableStream> PdbFileBuffer) { |
Zachary Turner | 199f48a | 2016-07-28 19:11:09 +0000 | [diff] [blame] | 78 | auto ExpectedLayout = finalizeMsfLayout(); |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 79 | if (!ExpectedLayout) |
| 80 | return ExpectedLayout.takeError(); |
| 81 | |
Zachary Turner | e109dc6 | 2016-07-22 19:56:26 +0000 | [diff] [blame] | 82 | auto File = llvm::make_unique<PDBFile>(std::move(PdbFileBuffer), Allocator); |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 83 | File->ContainerLayout = *ExpectedLayout; |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 84 | |
| 85 | if (Info) { |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 86 | auto ExpectedInfo = Info->build(*File, *PdbFileBuffer); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 87 | if (!ExpectedInfo) |
| 88 | return ExpectedInfo.takeError(); |
| 89 | File->Info = std::move(*ExpectedInfo); |
| 90 | } |
| 91 | |
| 92 | if (Dbi) { |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 93 | auto ExpectedDbi = Dbi->build(*File, *PdbFileBuffer); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 94 | if (!ExpectedDbi) |
| 95 | return ExpectedDbi.takeError(); |
| 96 | File->Dbi = std::move(*ExpectedDbi); |
| 97 | } |
| 98 | |
| 99 | if (File->Info && File->Dbi && File->Info->getAge() != File->Dbi->getAge()) |
| 100 | return llvm::make_error<RawError>( |
| 101 | raw_error_code::corrupt_file, |
| 102 | "PDB Stream Age doesn't match Dbi Stream Age!"); |
| 103 | |
| 104 | return std::move(File); |
| 105 | } |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 106 | |
| 107 | Error PDBFileBuilder::commit(const msf::WritableStream &Buffer) { |
| 108 | StreamWriter Writer(Buffer); |
| 109 | auto ExpectedLayout = finalizeMsfLayout(); |
| 110 | if (!ExpectedLayout) |
| 111 | return ExpectedLayout.takeError(); |
| 112 | auto &Layout = *ExpectedLayout; |
| 113 | |
| 114 | if (auto EC = Writer.writeObject(*Layout.SB)) |
| 115 | return EC; |
| 116 | uint32_t BlockMapOffset = |
| 117 | msf::blockToOffset(Layout.SB->BlockMapAddr, Layout.SB->BlockSize); |
| 118 | Writer.setOffset(BlockMapOffset); |
| 119 | if (auto EC = Writer.writeArray(Layout.DirectoryBlocks)) |
| 120 | return EC; |
| 121 | |
| 122 | auto DirStream = |
| 123 | WritableMappedBlockStream::createDirectoryStream(Layout, Buffer); |
| 124 | StreamWriter DW(*DirStream); |
Zachary Turner | 9f73c20 | 2016-07-28 19:29:52 +0000 | [diff] [blame] | 125 | if (auto EC = |
| 126 | DW.writeInteger(static_cast<uint32_t>(Layout.StreamSizes.size()))) |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 127 | return EC; |
| 128 | |
| 129 | if (auto EC = DW.writeArray(Layout.StreamSizes)) |
| 130 | return EC; |
| 131 | |
| 132 | for (const auto &Blocks : Layout.StreamMap) { |
| 133 | if (auto EC = DW.writeArray(Blocks)) |
| 134 | return EC; |
| 135 | } |
| 136 | |
| 137 | if (Info) { |
| 138 | if (auto EC = Info->commit(Layout, Buffer)) |
| 139 | return EC; |
| 140 | } |
| 141 | |
| 142 | if (Dbi) { |
| 143 | if (auto EC = Dbi->commit(Layout, Buffer)) |
| 144 | return EC; |
| 145 | } |
| 146 | |
| 147 | return Buffer.commit(); |
| 148 | } |