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 | 620961d | 2016-09-14 23:00:02 +0000 | [diff] [blame^] | 53 | Info = llvm::make_unique<InfoStreamBuilder>(*Msf); |
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 | 620961d | 2016-09-14 23:00:02 +0000 | [diff] [blame^] | 59 | Dbi = llvm::make_unique<DbiStreamBuilder>(*Msf); |
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) |
Zachary Turner | 620961d | 2016-09-14 23:00:02 +0000 | [diff] [blame^] | 65 | Tpi = llvm::make_unique<TpiStreamBuilder>(*Msf); |
Zachary Turner | c6d54da | 2016-09-09 17:46:17 +0000 | [diff] [blame] | 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 | 620961d | 2016-09-14 23:00:02 +0000 | [diff] [blame^] | 71 | if (auto EC = Info->finalizeMsfLayout()) |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 72 | return std::move(EC); |
| 73 | } |
| 74 | if (Dbi) { |
Zachary Turner | 620961d | 2016-09-14 23:00:02 +0000 | [diff] [blame^] | 75 | if (auto EC = Dbi->finalizeMsfLayout()) |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 76 | return std::move(EC); |
| 77 | } |
Zachary Turner | c6d54da | 2016-09-09 17:46:17 +0000 | [diff] [blame] | 78 | if (Tpi) { |
Zachary Turner | 620961d | 2016-09-14 23:00:02 +0000 | [diff] [blame^] | 79 | if (auto EC = Tpi->finalizeMsfLayout()) |
Zachary Turner | c6d54da | 2016-09-09 17:46:17 +0000 | [diff] [blame] | 80 | return std::move(EC); |
| 81 | } |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 82 | |
Zachary Turner | 199f48a | 2016-07-28 19:11:09 +0000 | [diff] [blame] | 83 | return Msf->build(); |
| 84 | } |
| 85 | |
| 86 | Expected<std::unique_ptr<PDBFile>> |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 87 | PDBFileBuilder::build(std::unique_ptr<msf::WritableStream> PdbFileBuffer) { |
Zachary Turner | 199f48a | 2016-07-28 19:11:09 +0000 | [diff] [blame] | 88 | auto ExpectedLayout = finalizeMsfLayout(); |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 89 | if (!ExpectedLayout) |
| 90 | return ExpectedLayout.takeError(); |
| 91 | |
Zachary Turner | e109dc6 | 2016-07-22 19:56:26 +0000 | [diff] [blame] | 92 | auto File = llvm::make_unique<PDBFile>(std::move(PdbFileBuffer), Allocator); |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 93 | File->ContainerLayout = *ExpectedLayout; |
Zachary Turner | faa554b | 2016-07-15 22:16:56 +0000 | [diff] [blame] | 94 | |
| 95 | if (Info) { |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 96 | auto ExpectedInfo = Info->build(*File, *PdbFileBuffer); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 97 | if (!ExpectedInfo) |
| 98 | return ExpectedInfo.takeError(); |
| 99 | File->Info = std::move(*ExpectedInfo); |
| 100 | } |
| 101 | |
| 102 | if (Dbi) { |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 103 | auto ExpectedDbi = Dbi->build(*File, *PdbFileBuffer); |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 104 | if (!ExpectedDbi) |
| 105 | return ExpectedDbi.takeError(); |
| 106 | File->Dbi = std::move(*ExpectedDbi); |
| 107 | } |
| 108 | |
Zachary Turner | c6d54da | 2016-09-09 17:46:17 +0000 | [diff] [blame] | 109 | if (Tpi) { |
| 110 | auto ExpectedTpi = Tpi->build(*File, *PdbFileBuffer); |
| 111 | if (!ExpectedTpi) |
| 112 | return ExpectedTpi.takeError(); |
| 113 | File->Tpi = std::move(*ExpectedTpi); |
| 114 | } |
| 115 | |
Zachary Turner | dbeaea7 | 2016-07-11 21:45:26 +0000 | [diff] [blame] | 116 | if (File->Info && File->Dbi && File->Info->getAge() != File->Dbi->getAge()) |
| 117 | return llvm::make_error<RawError>( |
| 118 | raw_error_code::corrupt_file, |
| 119 | "PDB Stream Age doesn't match Dbi Stream Age!"); |
| 120 | |
| 121 | return std::move(File); |
| 122 | } |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 123 | |
| 124 | Error PDBFileBuilder::commit(const msf::WritableStream &Buffer) { |
| 125 | StreamWriter Writer(Buffer); |
| 126 | auto ExpectedLayout = finalizeMsfLayout(); |
| 127 | if (!ExpectedLayout) |
| 128 | return ExpectedLayout.takeError(); |
| 129 | auto &Layout = *ExpectedLayout; |
| 130 | |
| 131 | if (auto EC = Writer.writeObject(*Layout.SB)) |
| 132 | return EC; |
| 133 | uint32_t BlockMapOffset = |
| 134 | msf::blockToOffset(Layout.SB->BlockMapAddr, Layout.SB->BlockSize); |
| 135 | Writer.setOffset(BlockMapOffset); |
| 136 | if (auto EC = Writer.writeArray(Layout.DirectoryBlocks)) |
| 137 | return EC; |
| 138 | |
| 139 | auto DirStream = |
| 140 | WritableMappedBlockStream::createDirectoryStream(Layout, Buffer); |
| 141 | StreamWriter DW(*DirStream); |
Zachary Turner | 9f73c20 | 2016-07-28 19:29:52 +0000 | [diff] [blame] | 142 | if (auto EC = |
| 143 | DW.writeInteger(static_cast<uint32_t>(Layout.StreamSizes.size()))) |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 144 | return EC; |
| 145 | |
| 146 | if (auto EC = DW.writeArray(Layout.StreamSizes)) |
| 147 | return EC; |
| 148 | |
| 149 | for (const auto &Blocks : Layout.StreamMap) { |
| 150 | if (auto EC = DW.writeArray(Blocks)) |
| 151 | return EC; |
| 152 | } |
| 153 | |
| 154 | if (Info) { |
| 155 | if (auto EC = Info->commit(Layout, Buffer)) |
| 156 | return EC; |
| 157 | } |
| 158 | |
| 159 | if (Dbi) { |
| 160 | if (auto EC = Dbi->commit(Layout, Buffer)) |
| 161 | return EC; |
| 162 | } |
| 163 | |
Zachary Turner | c6d54da | 2016-09-09 17:46:17 +0000 | [diff] [blame] | 164 | if (Tpi) { |
| 165 | if (auto EC = Tpi->commit(Layout, Buffer)) |
| 166 | return EC; |
| 167 | } |
| 168 | |
Zachary Turner | d66889c | 2016-07-28 19:12:28 +0000 | [diff] [blame] | 169 | return Buffer.commit(); |
| 170 | } |