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