blob: 84eb4fbbfa631f4d1593d42646dd32e8456fe534 [file] [log] [blame]
Zachary Turnerdbeaea72016-07-11 21:45:26 +00001//===- PDBFileBuilder.cpp - PDB File Creation -------------------*- C++ -*-===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Zachary Turnerdbeaea72016-07-11 21:45:26 +00006//
7//===----------------------------------------------------------------------===//
8
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +00009#include "llvm/DebugInfo/PDB/Native/PDBFileBuilder.h"
Zachary Turnerdbeaea72016-07-11 21:45:26 +000010
Zachary Turnerfaa554b2016-07-15 22:16:56 +000011#include "llvm/ADT/BitVector.h"
12
Zachary Turnera3225b02016-07-29 20:56:36 +000013#include "llvm/DebugInfo/MSF/MSFBuilder.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000014#include "llvm/DebugInfo/PDB/Native/DbiStream.h"
15#include "llvm/DebugInfo/PDB/Native/DbiStreamBuilder.h"
Zachary Turner946204c2017-08-09 04:23:25 +000016#include "llvm/DebugInfo/PDB/Native/GSIStreamBuilder.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000017#include "llvm/DebugInfo/PDB/Native/InfoStream.h"
18#include "llvm/DebugInfo/PDB/Native/InfoStreamBuilder.h"
Zachary Turnere204a6c2017-05-02 18:00:13 +000019#include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000020#include "llvm/DebugInfo/PDB/Native/RawError.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000021#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
22#include "llvm/DebugInfo/PDB/Native/TpiStreamBuilder.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000023#include "llvm/Support/BinaryStream.h"
24#include "llvm/Support/BinaryStreamWriter.h"
Zachary Turnerf2282762018-03-23 19:57:25 +000025#include "llvm/Support/JamCRC.h"
26#include "llvm/Support/Path.h"
Nico Weber205ca682018-09-15 18:35:51 +000027#include "llvm/Support/xxhash.h"
Zachary Turnerdbeaea72016-07-11 21:45:26 +000028
29using namespace llvm;
30using namespace llvm::codeview;
Zachary Turnerbac69d32016-07-22 19:56:05 +000031using namespace llvm::msf;
Zachary Turnerdbeaea72016-07-11 21:45:26 +000032using namespace llvm::pdb;
Zachary Turnerfaa554b2016-07-15 22:16:56 +000033using namespace llvm::support;
Zachary Turnerdbeaea72016-07-11 21:45:26 +000034
Zachary Turnere109dc62016-07-22 19:56:26 +000035PDBFileBuilder::PDBFileBuilder(BumpPtrAllocator &Allocator)
Zachary Turnerf2282762018-03-23 19:57:25 +000036 : Allocator(Allocator), InjectedSourceHashTraits(Strings),
37 InjectedSourceTable(2, InjectedSourceHashTraits) {}
Zachary Turnerdbeaea72016-07-11 21:45:26 +000038
Zachary Turner7eaf1d92017-07-10 22:40:20 +000039PDBFileBuilder::~PDBFileBuilder() {}
40
Rui Ueyama5d6714e2016-09-30 20:52:12 +000041Error PDBFileBuilder::initialize(uint32_t BlockSize) {
42 auto ExpectedMsf = MSFBuilder::create(Allocator, BlockSize);
Zachary Turnerfaa554b2016-07-15 22:16:56 +000043 if (!ExpectedMsf)
44 return ExpectedMsf.takeError();
Rui Ueyama5d6714e2016-09-30 20:52:12 +000045 Msf = llvm::make_unique<MSFBuilder>(std::move(*ExpectedMsf));
Zachary Turnerdbeaea72016-07-11 21:45:26 +000046 return Error::success();
47}
48
Zachary Turnera3225b02016-07-29 20:56:36 +000049MSFBuilder &PDBFileBuilder::getMsfBuilder() { return *Msf; }
Zachary Turnerfaa554b2016-07-15 22:16:56 +000050
Zachary Turnerdbeaea72016-07-11 21:45:26 +000051InfoStreamBuilder &PDBFileBuilder::getInfoBuilder() {
52 if (!Info)
Zachary Turner760ad4d2017-01-20 22:42:09 +000053 Info = llvm::make_unique<InfoStreamBuilder>(*Msf, NamedStreams);
Zachary Turnerdbeaea72016-07-11 21:45:26 +000054 return *Info;
55}
56
57DbiStreamBuilder &PDBFileBuilder::getDbiBuilder() {
58 if (!Dbi)
Zachary Turner620961d2016-09-14 23:00:02 +000059 Dbi = llvm::make_unique<DbiStreamBuilder>(*Msf);
Zachary Turnerdbeaea72016-07-11 21:45:26 +000060 return *Dbi;
61}
62
Zachary Turnerc6d54da2016-09-09 17:46:17 +000063TpiStreamBuilder &PDBFileBuilder::getTpiBuilder() {
64 if (!Tpi)
Zachary Turnerde9ba152016-09-15 18:22:31 +000065 Tpi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamTPI);
Zachary Turnerc6d54da2016-09-09 17:46:17 +000066 return *Tpi;
67}
68
Zachary Turnerde9ba152016-09-15 18:22:31 +000069TpiStreamBuilder &PDBFileBuilder::getIpiBuilder() {
70 if (!Ipi)
71 Ipi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamIPI);
72 return *Ipi;
73}
74
Zachary Turnere204a6c2017-05-02 18:00:13 +000075PDBStringTableBuilder &PDBFileBuilder::getStringTableBuilder() {
76 return Strings;
77}
Zachary Turner760ad4d2017-01-20 22:42:09 +000078
Zachary Turner946204c2017-08-09 04:23:25 +000079GSIStreamBuilder &PDBFileBuilder::getGsiBuilder() {
80 if (!Gsi)
81 Gsi = llvm::make_unique<GSIStreamBuilder>(*Msf);
82 return *Gsi;
Zachary Turner8d927b62017-07-31 19:36:08 +000083}
84
Zachary Turnera6fb5362018-03-23 18:43:39 +000085Expected<uint32_t> PDBFileBuilder::allocateNamedStream(StringRef Name,
86 uint32_t Size) {
Zachary Turner760ad4d2017-01-20 22:42:09 +000087 auto ExpectedStream = Msf->addStream(Size);
Zachary Turnera6fb5362018-03-23 18:43:39 +000088 if (ExpectedStream)
89 NamedStreams.set(Name, *ExpectedStream);
90 return ExpectedStream;
91}
92
93Error PDBFileBuilder::addNamedStream(StringRef Name, StringRef Data) {
94 Expected<uint32_t> ExpectedIndex = allocateNamedStream(Name, Data.size());
95 if (!ExpectedIndex)
96 return ExpectedIndex.takeError();
97 assert(NamedStreamData.count(*ExpectedIndex) == 0);
98 NamedStreamData[*ExpectedIndex] = Data;
Zachary Turner760ad4d2017-01-20 22:42:09 +000099 return Error::success();
100}
101
Zachary Turnerf2282762018-03-23 19:57:25 +0000102void PDBFileBuilder::addInjectedSource(StringRef Name,
103 std::unique_ptr<MemoryBuffer> Buffer) {
104 // Stream names must be exact matches, since they get looked up in a hash
105 // table and the hash value is dependent on the exact contents of the string.
106 // link.exe lowercases a path and converts / to \, so we must do the same.
107 SmallString<64> VName;
108 sys::path::native(Name.lower(), VName);
109
110 uint32_t NI = getStringTableBuilder().insert(Name);
111 uint32_t VNI = getStringTableBuilder().insert(VName);
112
113 InjectedSourceDescriptor Desc;
114 Desc.Content = std::move(Buffer);
115 Desc.NameIndex = NI;
116 Desc.VNameIndex = VNI;
117 Desc.StreamName = "/src/files/";
118
119 Desc.StreamName += VName;
120
121 InjectedSources.push_back(std::move(Desc));
122}
123
Zachary Turneree8010a2018-06-27 21:18:15 +0000124Error PDBFileBuilder::finalizeMsfLayout() {
Zachary Turner68ea80d2017-06-12 21:46:51 +0000125
126 if (Ipi && Ipi->getRecordCount() > 0) {
127 // In theory newer PDBs always have an ID stream, but by saying that we're
128 // only going to *really* have an ID stream if there is at least one ID
129 // record, we leave open the opportunity to test older PDBs such as those
130 // that don't have an ID stream.
131 auto &Info = getInfoBuilder();
132 Info.addFeature(PdbRaw_FeatureSig::VC140);
133 }
134
Zachary Turnerc504ae32017-05-03 15:58:37 +0000135 uint32_t StringsLen = Strings.calculateSerializedSize();
Zachary Turner760ad4d2017-01-20 22:42:09 +0000136
Zachary Turnera6fb5362018-03-23 18:43:39 +0000137 Expected<uint32_t> SN = allocateNamedStream("/LinkInfo", 0);
138 if (!SN)
139 return SN.takeError();
Zachary Turner760ad4d2017-01-20 22:42:09 +0000140
Zachary Turner946204c2017-08-09 04:23:25 +0000141 if (Gsi) {
142 if (auto EC = Gsi->finalizeMsfLayout())
Zachary Turneree8010a2018-06-27 21:18:15 +0000143 return EC;
Zachary Turner7eaf1d92017-07-10 22:40:20 +0000144 if (Dbi) {
Zachary Turner946204c2017-08-09 04:23:25 +0000145 Dbi->setPublicsStreamIndex(Gsi->getPublicsStreamIndex());
146 Dbi->setGlobalsStreamIndex(Gsi->getGlobalsStreamIndex());
147 Dbi->setSymbolRecordStreamIndex(Gsi->getRecordStreamIdx());
Zachary Turner7eaf1d92017-07-10 22:40:20 +0000148 }
149 }
Zachary Turnera6fb5362018-03-23 18:43:39 +0000150 if (Tpi) {
151 if (auto EC = Tpi->finalizeMsfLayout())
Zachary Turneree8010a2018-06-27 21:18:15 +0000152 return EC;
Zachary Turnera6fb5362018-03-23 18:43:39 +0000153 }
154 if (Dbi) {
155 if (auto EC = Dbi->finalizeMsfLayout())
Zachary Turneree8010a2018-06-27 21:18:15 +0000156 return EC;
Zachary Turnera6fb5362018-03-23 18:43:39 +0000157 }
158 SN = allocateNamedStream("/names", StringsLen);
159 if (!SN)
160 return SN.takeError();
161
162 if (Ipi) {
163 if (auto EC = Ipi->finalizeMsfLayout())
Zachary Turneree8010a2018-06-27 21:18:15 +0000164 return EC;
Zachary Turnera6fb5362018-03-23 18:43:39 +0000165 }
166
167 // Do this last, since it relies on the named stream map being complete, and
168 // that can be updated by previous steps in the finalization.
169 if (Info) {
170 if (auto EC = Info->finalizeMsfLayout())
Zachary Turneree8010a2018-06-27 21:18:15 +0000171 return EC;
Zachary Turnera6fb5362018-03-23 18:43:39 +0000172 }
Zachary Turnerfaa554b2016-07-15 22:16:56 +0000173
Zachary Turnerf2282762018-03-23 19:57:25 +0000174 if (!InjectedSources.empty()) {
175 for (const auto &IS : InjectedSources) {
176 JamCRC CRC(0);
177 CRC.update(makeArrayRef(IS.Content->getBufferStart(),
178 IS.Content->getBufferSize()));
179
180 SrcHeaderBlockEntry Entry;
181 ::memset(&Entry, 0, sizeof(SrcHeaderBlockEntry));
182 Entry.Size = sizeof(SrcHeaderBlockEntry);
183 Entry.FileSize = IS.Content->getBufferSize();
184 Entry.FileNI = IS.NameIndex;
185 Entry.VFileNI = IS.VNameIndex;
186 Entry.ObjNI = 1;
187 Entry.IsVirtual = 0;
188 Entry.Version =
189 static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne);
190 Entry.CRC = CRC.getCRC();
191 StringRef VName = getStringTableBuilder().getStringForId(IS.VNameIndex);
192 InjectedSourceTable.set_as(VName, std::move(Entry));
193 }
194
195 uint32_t SrcHeaderBlockSize =
196 sizeof(SrcHeaderBlockHeader) +
197 InjectedSourceTable.calculateSerializedLength();
198 SN = allocateNamedStream("/src/headerblock", SrcHeaderBlockSize);
199 if (!SN)
200 return SN.takeError();
201 for (const auto &IS : InjectedSources) {
202 SN = allocateNamedStream(IS.StreamName, IS.Content->getBufferSize());
203 if (!SN)
204 return SN.takeError();
205 }
206 }
207
208 // Do this last, since it relies on the named stream map being complete, and
209 // that can be updated by previous steps in the finalization.
210 if (Info) {
211 if (auto EC = Info->finalizeMsfLayout())
Zachary Turneree8010a2018-06-27 21:18:15 +0000212 return EC;
Zachary Turnerf2282762018-03-23 19:57:25 +0000213 }
214
Zachary Turneree8010a2018-06-27 21:18:15 +0000215 return Error::success();
Zachary Turner199f48a2016-07-28 19:11:09 +0000216}
217
Zachary Turnerc504ae32017-05-03 15:58:37 +0000218Expected<uint32_t> PDBFileBuilder::getNamedStreamIndex(StringRef Name) const {
219 uint32_t SN = 0;
220 if (!NamedStreams.get(Name, SN))
221 return llvm::make_error<pdb::RawError>(raw_error_code::no_stream);
222 return SN;
223}
224
Zachary Turnerf2282762018-03-23 19:57:25 +0000225void PDBFileBuilder::commitSrcHeaderBlock(WritableBinaryStream &MsfBuffer,
226 const msf::MSFLayout &Layout) {
227 assert(!InjectedSourceTable.empty());
228
229 uint32_t SN = cantFail(getNamedStreamIndex("/src/headerblock"));
230 auto Stream = WritableMappedBlockStream::createIndexedStream(
231 Layout, MsfBuffer, SN, Allocator);
232 BinaryStreamWriter Writer(*Stream);
233
234 SrcHeaderBlockHeader Header;
235 ::memset(&Header, 0, sizeof(Header));
236 Header.Version = static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne);
237 Header.Size = Writer.bytesRemaining();
238
239 cantFail(Writer.writeObject(Header));
240 cantFail(InjectedSourceTable.commit(Writer));
241
242 assert(Writer.bytesRemaining() == 0);
243}
244
245void PDBFileBuilder::commitInjectedSources(WritableBinaryStream &MsfBuffer,
246 const msf::MSFLayout &Layout) {
247 if (InjectedSourceTable.empty())
248 return;
249
250 commitSrcHeaderBlock(MsfBuffer, Layout);
251
252 for (const auto &IS : InjectedSources) {
253 uint32_t SN = cantFail(getNamedStreamIndex(IS.StreamName));
254
255 auto SourceStream = WritableMappedBlockStream::createIndexedStream(
256 Layout, MsfBuffer, SN, Allocator);
257 BinaryStreamWriter SourceWriter(*SourceStream);
258 assert(SourceWriter.bytesRemaining() == IS.Content->getBufferSize());
259 cantFail(SourceWriter.writeBytes(
260 arrayRefFromStringRef(IS.Content->getBuffer())));
261 }
262}
263
Nico Weber205ca682018-09-15 18:35:51 +0000264Error PDBFileBuilder::commit(StringRef Filename, codeview::GUID *Guid) {
Bob Haarmande33a6372017-05-17 20:46:48 +0000265 assert(!Filename.empty());
Zachary Turneree8010a2018-06-27 21:18:15 +0000266 if (auto EC = finalizeMsfLayout())
Zachary Turnerd66889c2016-07-28 19:12:28 +0000267 return EC;
Zachary Turner9fb9d712017-08-02 22:31:39 +0000268
Zachary Turneree8010a2018-06-27 21:18:15 +0000269 MSFLayout Layout;
Nico Weber205ca682018-09-15 18:35:51 +0000270 Expected<FileBufferByteStream> ExpectedMsfBuffer =
271 Msf->commit(Filename, Layout);
Zachary Turneree8010a2018-06-27 21:18:15 +0000272 if (!ExpectedMsfBuffer)
273 return ExpectedMsfBuffer.takeError();
274 FileBufferByteStream Buffer = std::move(*ExpectedMsfBuffer);
Zachary Turnerd66889c2016-07-28 19:12:28 +0000275
Zachary Turnerc504ae32017-05-03 15:58:37 +0000276 auto ExpectedSN = getNamedStreamIndex("/names");
277 if (!ExpectedSN)
278 return ExpectedSN.takeError();
Zachary Turner760ad4d2017-01-20 22:42:09 +0000279
Zachary Turner5b74ff32017-06-03 00:33:35 +0000280 auto NS = WritableMappedBlockStream::createIndexedStream(
281 Layout, Buffer, *ExpectedSN, Allocator);
Zachary Turner120faca2017-02-27 22:11:43 +0000282 BinaryStreamWriter NSWriter(*NS);
Zachary Turner760ad4d2017-01-20 22:42:09 +0000283 if (auto EC = Strings.commit(NSWriter))
284 return EC;
285
Zachary Turnera6fb5362018-03-23 18:43:39 +0000286 for (const auto &NSE : NamedStreamData) {
287 if (NSE.second.empty())
288 continue;
289
290 auto NS = WritableMappedBlockStream::createIndexedStream(
291 Layout, Buffer, NSE.first, Allocator);
292 BinaryStreamWriter NSW(*NS);
293 if (auto EC = NSW.writeBytes(arrayRefFromStringRef(NSE.second)))
294 return EC;
295 }
296
Zachary Turnerd66889c2016-07-28 19:12:28 +0000297 if (Info) {
298 if (auto EC = Info->commit(Layout, Buffer))
299 return EC;
300 }
301
302 if (Dbi) {
303 if (auto EC = Dbi->commit(Layout, Buffer))
304 return EC;
305 }
306
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000307 if (Tpi) {
308 if (auto EC = Tpi->commit(Layout, Buffer))
309 return EC;
310 }
311
Zachary Turnerde9ba152016-09-15 18:22:31 +0000312 if (Ipi) {
313 if (auto EC = Ipi->commit(Layout, Buffer))
314 return EC;
315 }
316
Zachary Turner946204c2017-08-09 04:23:25 +0000317 if (Gsi) {
318 if (auto EC = Gsi->commit(Layout, Buffer))
Zachary Turner8d927b62017-07-31 19:36:08 +0000319 return EC;
320 }
321
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000322 auto InfoStreamBlocks = Layout.StreamMap[StreamPDB];
323 assert(!InfoStreamBlocks.empty());
324 uint64_t InfoStreamFileOffset =
325 blockToOffset(InfoStreamBlocks.front(), Layout.SB->BlockSize);
326 InfoStreamHeader *H = reinterpret_cast<InfoStreamHeader *>(
Zachary Turneree8010a2018-06-27 21:18:15 +0000327 Buffer.getBufferStart() + InfoStreamFileOffset);
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000328
Zachary Turnerf2282762018-03-23 19:57:25 +0000329 commitInjectedSources(Buffer, Layout);
330
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000331 // Set the build id at the very end, after every other byte of the PDB
332 // has been written.
Nico Weber205ca682018-09-15 18:35:51 +0000333 if (Info->hashPDBContentsToGUID()) {
334 // Compute a hash of all sections of the output file.
335 uint64_t Digest =
336 xxHash64({Buffer.getBufferStart(), Buffer.getBufferEnd()});
337
338 H->Age = 1;
339
340 memcpy(H->Guid.Guid, &Digest, 8);
341 // xxhash only gives us 8 bytes, so put some fixed data in the other half.
342 memcpy(H->Guid.Guid + 8, "LLD PDB.", 8);
343
344 // Put the hash in the Signature field too.
345 H->Signature = static_cast<uint32_t>(Digest);
346
347 // Return GUID to caller.
348 memcpy(Guid, H->Guid.Guid, 16);
349 } else {
350 H->Age = Info->getAge();
351 H->Guid = Info->getGuid();
352 Optional<uint32_t> Sig = Info->getSignature();
353 H->Signature = Sig.hasValue() ? *Sig : time(nullptr);
354 }
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000355
Zachary Turnerd66889c2016-07-28 19:12:28 +0000356 return Buffer.commit();
Rui Ueyamafc22cef2016-09-30 20:34:44 +0000357}