blob: 77048589e9fb7c2dbb81d93b3404aba521d40cb5 [file] [log] [blame]
Zachary Turnerdbeaea72016-07-11 21:45:26 +00001//===- 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
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000010#include "llvm/DebugInfo/PDB/Native/PDBFileBuilder.h"
Zachary Turnerdbeaea72016-07-11 21:45:26 +000011
Zachary Turnerfaa554b2016-07-15 22:16:56 +000012#include "llvm/ADT/BitVector.h"
13
Zachary Turnera3225b02016-07-29 20:56:36 +000014#include "llvm/DebugInfo/MSF/MSFBuilder.h"
Rui Ueyamafc22cef2016-09-30 20:34:44 +000015#include "llvm/DebugInfo/PDB/GenericError.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000016#include "llvm/DebugInfo/PDB/Native/DbiStream.h"
17#include "llvm/DebugInfo/PDB/Native/DbiStreamBuilder.h"
Zachary Turner946204c2017-08-09 04:23:25 +000018#include "llvm/DebugInfo/PDB/Native/GSIStreamBuilder.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000019#include "llvm/DebugInfo/PDB/Native/InfoStream.h"
20#include "llvm/DebugInfo/PDB/Native/InfoStreamBuilder.h"
Zachary Turnere204a6c2017-05-02 18:00:13 +000021#include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000022#include "llvm/DebugInfo/PDB/Native/RawError.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000023#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
24#include "llvm/DebugInfo/PDB/Native/TpiStreamBuilder.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000025#include "llvm/Support/BinaryStream.h"
26#include "llvm/Support/BinaryStreamWriter.h"
Zachary Turnerf2282762018-03-23 19:57:25 +000027#include "llvm/Support/JamCRC.h"
28#include "llvm/Support/Path.h"
Zachary Turnerdbeaea72016-07-11 21:45:26 +000029
30using namespace llvm;
31using namespace llvm::codeview;
Zachary Turnerbac69d32016-07-22 19:56:05 +000032using namespace llvm::msf;
Zachary Turnerdbeaea72016-07-11 21:45:26 +000033using namespace llvm::pdb;
Zachary Turnerfaa554b2016-07-15 22:16:56 +000034using namespace llvm::support;
Zachary Turnerdbeaea72016-07-11 21:45:26 +000035
Zachary Turnere109dc62016-07-22 19:56:26 +000036PDBFileBuilder::PDBFileBuilder(BumpPtrAllocator &Allocator)
Zachary Turnerf2282762018-03-23 19:57:25 +000037 : Allocator(Allocator), InjectedSourceHashTraits(Strings),
38 InjectedSourceTable(2, InjectedSourceHashTraits) {}
Zachary Turnerdbeaea72016-07-11 21:45:26 +000039
Zachary Turner7eaf1d92017-07-10 22:40:20 +000040PDBFileBuilder::~PDBFileBuilder() {}
41
Rui Ueyama5d6714e2016-09-30 20:52:12 +000042Error PDBFileBuilder::initialize(uint32_t BlockSize) {
43 auto ExpectedMsf = MSFBuilder::create(Allocator, BlockSize);
Zachary Turnerfaa554b2016-07-15 22:16:56 +000044 if (!ExpectedMsf)
45 return ExpectedMsf.takeError();
Rui Ueyama5d6714e2016-09-30 20:52:12 +000046 Msf = llvm::make_unique<MSFBuilder>(std::move(*ExpectedMsf));
Zachary Turnerdbeaea72016-07-11 21:45:26 +000047 return Error::success();
48}
49
Zachary Turnera3225b02016-07-29 20:56:36 +000050MSFBuilder &PDBFileBuilder::getMsfBuilder() { return *Msf; }
Zachary Turnerfaa554b2016-07-15 22:16:56 +000051
Zachary Turnerdbeaea72016-07-11 21:45:26 +000052InfoStreamBuilder &PDBFileBuilder::getInfoBuilder() {
53 if (!Info)
Zachary Turner760ad4d2017-01-20 22:42:09 +000054 Info = llvm::make_unique<InfoStreamBuilder>(*Msf, NamedStreams);
Zachary Turnerdbeaea72016-07-11 21:45:26 +000055 return *Info;
56}
57
58DbiStreamBuilder &PDBFileBuilder::getDbiBuilder() {
59 if (!Dbi)
Zachary Turner620961d2016-09-14 23:00:02 +000060 Dbi = llvm::make_unique<DbiStreamBuilder>(*Msf);
Zachary Turnerdbeaea72016-07-11 21:45:26 +000061 return *Dbi;
62}
63
Zachary Turnerc6d54da2016-09-09 17:46:17 +000064TpiStreamBuilder &PDBFileBuilder::getTpiBuilder() {
65 if (!Tpi)
Zachary Turnerde9ba152016-09-15 18:22:31 +000066 Tpi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamTPI);
Zachary Turnerc6d54da2016-09-09 17:46:17 +000067 return *Tpi;
68}
69
Zachary Turnerde9ba152016-09-15 18:22:31 +000070TpiStreamBuilder &PDBFileBuilder::getIpiBuilder() {
71 if (!Ipi)
72 Ipi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamIPI);
73 return *Ipi;
74}
75
Zachary Turnere204a6c2017-05-02 18:00:13 +000076PDBStringTableBuilder &PDBFileBuilder::getStringTableBuilder() {
77 return Strings;
78}
Zachary Turner760ad4d2017-01-20 22:42:09 +000079
Zachary Turner946204c2017-08-09 04:23:25 +000080GSIStreamBuilder &PDBFileBuilder::getGsiBuilder() {
81 if (!Gsi)
82 Gsi = llvm::make_unique<GSIStreamBuilder>(*Msf);
83 return *Gsi;
Zachary Turner8d927b62017-07-31 19:36:08 +000084}
85
Zachary Turnera6fb5362018-03-23 18:43:39 +000086Expected<uint32_t> PDBFileBuilder::allocateNamedStream(StringRef Name,
87 uint32_t Size) {
Zachary Turner760ad4d2017-01-20 22:42:09 +000088 auto ExpectedStream = Msf->addStream(Size);
Zachary Turnera6fb5362018-03-23 18:43:39 +000089 if (ExpectedStream)
90 NamedStreams.set(Name, *ExpectedStream);
91 return ExpectedStream;
92}
93
94Error PDBFileBuilder::addNamedStream(StringRef Name, StringRef Data) {
95 Expected<uint32_t> ExpectedIndex = allocateNamedStream(Name, Data.size());
96 if (!ExpectedIndex)
97 return ExpectedIndex.takeError();
98 assert(NamedStreamData.count(*ExpectedIndex) == 0);
99 NamedStreamData[*ExpectedIndex] = Data;
Zachary Turner760ad4d2017-01-20 22:42:09 +0000100 return Error::success();
101}
102
Zachary Turnerf2282762018-03-23 19:57:25 +0000103void PDBFileBuilder::addInjectedSource(StringRef Name,
104 std::unique_ptr<MemoryBuffer> Buffer) {
105 // Stream names must be exact matches, since they get looked up in a hash
106 // table and the hash value is dependent on the exact contents of the string.
107 // link.exe lowercases a path and converts / to \, so we must do the same.
108 SmallString<64> VName;
109 sys::path::native(Name.lower(), VName);
110
111 uint32_t NI = getStringTableBuilder().insert(Name);
112 uint32_t VNI = getStringTableBuilder().insert(VName);
113
114 InjectedSourceDescriptor Desc;
115 Desc.Content = std::move(Buffer);
116 Desc.NameIndex = NI;
117 Desc.VNameIndex = VNI;
118 Desc.StreamName = "/src/files/";
119
120 Desc.StreamName += VName;
121
122 InjectedSources.push_back(std::move(Desc));
123}
124
Zachary Turner760ad4d2017-01-20 22:42:09 +0000125Expected<msf::MSFLayout> PDBFileBuilder::finalizeMsfLayout() {
Zachary Turner68ea80d2017-06-12 21:46:51 +0000126
127 if (Ipi && Ipi->getRecordCount() > 0) {
128 // In theory newer PDBs always have an ID stream, but by saying that we're
129 // only going to *really* have an ID stream if there is at least one ID
130 // record, we leave open the opportunity to test older PDBs such as those
131 // that don't have an ID stream.
132 auto &Info = getInfoBuilder();
133 Info.addFeature(PdbRaw_FeatureSig::VC140);
134 }
135
Zachary Turnerc504ae32017-05-03 15:58:37 +0000136 uint32_t StringsLen = Strings.calculateSerializedSize();
Zachary Turner760ad4d2017-01-20 22:42:09 +0000137
Zachary Turnera6fb5362018-03-23 18:43:39 +0000138 Expected<uint32_t> SN = allocateNamedStream("/LinkInfo", 0);
139 if (!SN)
140 return SN.takeError();
Zachary Turner760ad4d2017-01-20 22:42:09 +0000141
Zachary Turner946204c2017-08-09 04:23:25 +0000142 if (Gsi) {
143 if (auto EC = Gsi->finalizeMsfLayout())
Zachary Turner7eaf1d92017-07-10 22:40:20 +0000144 return std::move(EC);
145 if (Dbi) {
Zachary Turner946204c2017-08-09 04:23:25 +0000146 Dbi->setPublicsStreamIndex(Gsi->getPublicsStreamIndex());
147 Dbi->setGlobalsStreamIndex(Gsi->getGlobalsStreamIndex());
148 Dbi->setSymbolRecordStreamIndex(Gsi->getRecordStreamIdx());
Zachary Turner7eaf1d92017-07-10 22:40:20 +0000149 }
150 }
Zachary Turnera6fb5362018-03-23 18:43:39 +0000151 if (Tpi) {
152 if (auto EC = Tpi->finalizeMsfLayout())
153 return std::move(EC);
154 }
155 if (Dbi) {
156 if (auto EC = Dbi->finalizeMsfLayout())
157 return std::move(EC);
158 }
159 SN = allocateNamedStream("/names", StringsLen);
160 if (!SN)
161 return SN.takeError();
162
163 if (Ipi) {
164 if (auto EC = Ipi->finalizeMsfLayout())
165 return std::move(EC);
166 }
167
168 // Do this last, since it relies on the named stream map being complete, and
169 // that can be updated by previous steps in the finalization.
170 if (Info) {
171 if (auto EC = Info->finalizeMsfLayout())
172 return std::move(EC);
173 }
Zachary Turnerfaa554b2016-07-15 22:16:56 +0000174
Zachary Turnerf2282762018-03-23 19:57:25 +0000175 if (!InjectedSources.empty()) {
176 for (const auto &IS : InjectedSources) {
177 JamCRC CRC(0);
178 CRC.update(makeArrayRef(IS.Content->getBufferStart(),
179 IS.Content->getBufferSize()));
180
181 SrcHeaderBlockEntry Entry;
182 ::memset(&Entry, 0, sizeof(SrcHeaderBlockEntry));
183 Entry.Size = sizeof(SrcHeaderBlockEntry);
184 Entry.FileSize = IS.Content->getBufferSize();
185 Entry.FileNI = IS.NameIndex;
186 Entry.VFileNI = IS.VNameIndex;
187 Entry.ObjNI = 1;
188 Entry.IsVirtual = 0;
189 Entry.Version =
190 static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne);
191 Entry.CRC = CRC.getCRC();
192 StringRef VName = getStringTableBuilder().getStringForId(IS.VNameIndex);
193 InjectedSourceTable.set_as(VName, std::move(Entry));
194 }
195
196 uint32_t SrcHeaderBlockSize =
197 sizeof(SrcHeaderBlockHeader) +
198 InjectedSourceTable.calculateSerializedLength();
199 SN = allocateNamedStream("/src/headerblock", SrcHeaderBlockSize);
200 if (!SN)
201 return SN.takeError();
202 for (const auto &IS : InjectedSources) {
203 SN = allocateNamedStream(IS.StreamName, IS.Content->getBufferSize());
204 if (!SN)
205 return SN.takeError();
206 }
207 }
208
209 // Do this last, since it relies on the named stream map being complete, and
210 // that can be updated by previous steps in the finalization.
211 if (Info) {
212 if (auto EC = Info->finalizeMsfLayout())
213 return std::move(EC);
214 }
215
Zachary Turner199f48a2016-07-28 19:11:09 +0000216 return Msf->build();
217}
218
Zachary Turnerc504ae32017-05-03 15:58:37 +0000219Expected<uint32_t> PDBFileBuilder::getNamedStreamIndex(StringRef Name) const {
220 uint32_t SN = 0;
221 if (!NamedStreams.get(Name, SN))
222 return llvm::make_error<pdb::RawError>(raw_error_code::no_stream);
223 return SN;
224}
225
Zachary Turner9fb9d712017-08-02 22:31:39 +0000226void PDBFileBuilder::commitFpm(WritableBinaryStream &MsfBuffer,
227 const MSFLayout &Layout) {
228 auto FpmStream =
229 WritableMappedBlockStream::createFpmStream(Layout, MsfBuffer, Allocator);
230
231 // We only need to create the alt fpm stream so that it gets initialized.
232 WritableMappedBlockStream::createFpmStream(Layout, MsfBuffer, Allocator,
233 true);
234
235 uint32_t BI = 0;
236 BinaryStreamWriter FpmWriter(*FpmStream);
237 while (BI < Layout.SB->NumBlocks) {
238 uint8_t ThisByte = 0;
239 for (uint32_t I = 0; I < 8; ++I) {
240 bool IsFree =
241 (BI < Layout.SB->NumBlocks) ? Layout.FreePageMap.test(BI) : true;
242 uint8_t Mask = uint8_t(IsFree) << I;
243 ThisByte |= Mask;
244 ++BI;
245 }
246 cantFail(FpmWriter.writeObject(ThisByte));
247 }
248 assert(FpmWriter.bytesRemaining() == 0);
249}
250
Zachary Turnerf2282762018-03-23 19:57:25 +0000251void PDBFileBuilder::commitSrcHeaderBlock(WritableBinaryStream &MsfBuffer,
252 const msf::MSFLayout &Layout) {
253 assert(!InjectedSourceTable.empty());
254
255 uint32_t SN = cantFail(getNamedStreamIndex("/src/headerblock"));
256 auto Stream = WritableMappedBlockStream::createIndexedStream(
257 Layout, MsfBuffer, SN, Allocator);
258 BinaryStreamWriter Writer(*Stream);
259
260 SrcHeaderBlockHeader Header;
261 ::memset(&Header, 0, sizeof(Header));
262 Header.Version = static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne);
263 Header.Size = Writer.bytesRemaining();
264
265 cantFail(Writer.writeObject(Header));
266 cantFail(InjectedSourceTable.commit(Writer));
267
268 assert(Writer.bytesRemaining() == 0);
269}
270
271void PDBFileBuilder::commitInjectedSources(WritableBinaryStream &MsfBuffer,
272 const msf::MSFLayout &Layout) {
273 if (InjectedSourceTable.empty())
274 return;
275
276 commitSrcHeaderBlock(MsfBuffer, Layout);
277
278 for (const auto &IS : InjectedSources) {
279 uint32_t SN = cantFail(getNamedStreamIndex(IS.StreamName));
280
281 auto SourceStream = WritableMappedBlockStream::createIndexedStream(
282 Layout, MsfBuffer, SN, Allocator);
283 BinaryStreamWriter SourceWriter(*SourceStream);
284 assert(SourceWriter.bytesRemaining() == IS.Content->getBufferSize());
285 cantFail(SourceWriter.writeBytes(
286 arrayRefFromStringRef(IS.Content->getBuffer())));
287 }
288}
289
Rui Ueyamafc22cef2016-09-30 20:34:44 +0000290Error PDBFileBuilder::commit(StringRef Filename) {
Bob Haarmande33a6372017-05-17 20:46:48 +0000291 assert(!Filename.empty());
Zachary Turnerd66889c2016-07-28 19:12:28 +0000292 auto ExpectedLayout = finalizeMsfLayout();
293 if (!ExpectedLayout)
294 return ExpectedLayout.takeError();
295 auto &Layout = *ExpectedLayout;
296
Rui Ueyamafc22cef2016-09-30 20:34:44 +0000297 uint64_t Filesize = Layout.SB->BlockSize * Layout.SB->NumBlocks;
298 auto OutFileOrError = FileOutputBuffer::create(Filename, Filesize);
Rafael Espindolae0df3572017-11-08 01:05:44 +0000299 if (auto E = OutFileOrError.takeError())
300 return E;
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000301 FileOutputBuffer *FOB = OutFileOrError->get();
302
Zachary Turner695ed562017-02-28 00:04:07 +0000303 FileBufferByteStream Buffer(std::move(*OutFileOrError),
304 llvm::support::little);
Zachary Turner120faca2017-02-27 22:11:43 +0000305 BinaryStreamWriter Writer(Buffer);
Rui Ueyamafc22cef2016-09-30 20:34:44 +0000306
Zachary Turnerd66889c2016-07-28 19:12:28 +0000307 if (auto EC = Writer.writeObject(*Layout.SB))
308 return EC;
Zachary Turner9fb9d712017-08-02 22:31:39 +0000309
310 commitFpm(Buffer, Layout);
311
Zachary Turnerd66889c2016-07-28 19:12:28 +0000312 uint32_t BlockMapOffset =
313 msf::blockToOffset(Layout.SB->BlockMapAddr, Layout.SB->BlockSize);
314 Writer.setOffset(BlockMapOffset);
315 if (auto EC = Writer.writeArray(Layout.DirectoryBlocks))
316 return EC;
317
Zachary Turner5b74ff32017-06-03 00:33:35 +0000318 auto DirStream = WritableMappedBlockStream::createDirectoryStream(
319 Layout, Buffer, Allocator);
Zachary Turner120faca2017-02-27 22:11:43 +0000320 BinaryStreamWriter DW(*DirStream);
Zachary Turner695ed562017-02-28 00:04:07 +0000321 if (auto EC = DW.writeInteger<uint32_t>(Layout.StreamSizes.size()))
Zachary Turnerd66889c2016-07-28 19:12:28 +0000322 return EC;
323
324 if (auto EC = DW.writeArray(Layout.StreamSizes))
325 return EC;
326
327 for (const auto &Blocks : Layout.StreamMap) {
328 if (auto EC = DW.writeArray(Blocks))
329 return EC;
330 }
331
Zachary Turnerc504ae32017-05-03 15:58:37 +0000332 auto ExpectedSN = getNamedStreamIndex("/names");
333 if (!ExpectedSN)
334 return ExpectedSN.takeError();
Zachary Turner760ad4d2017-01-20 22:42:09 +0000335
Zachary Turner5b74ff32017-06-03 00:33:35 +0000336 auto NS = WritableMappedBlockStream::createIndexedStream(
337 Layout, Buffer, *ExpectedSN, Allocator);
Zachary Turner120faca2017-02-27 22:11:43 +0000338 BinaryStreamWriter NSWriter(*NS);
Zachary Turner760ad4d2017-01-20 22:42:09 +0000339 if (auto EC = Strings.commit(NSWriter))
340 return EC;
341
Zachary Turnera6fb5362018-03-23 18:43:39 +0000342 for (const auto &NSE : NamedStreamData) {
343 if (NSE.second.empty())
344 continue;
345
346 auto NS = WritableMappedBlockStream::createIndexedStream(
347 Layout, Buffer, NSE.first, Allocator);
348 BinaryStreamWriter NSW(*NS);
349 if (auto EC = NSW.writeBytes(arrayRefFromStringRef(NSE.second)))
350 return EC;
351 }
352
Zachary Turnerd66889c2016-07-28 19:12:28 +0000353 if (Info) {
354 if (auto EC = Info->commit(Layout, Buffer))
355 return EC;
356 }
357
358 if (Dbi) {
359 if (auto EC = Dbi->commit(Layout, Buffer))
360 return EC;
361 }
362
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000363 if (Tpi) {
364 if (auto EC = Tpi->commit(Layout, Buffer))
365 return EC;
366 }
367
Zachary Turnerde9ba152016-09-15 18:22:31 +0000368 if (Ipi) {
369 if (auto EC = Ipi->commit(Layout, Buffer))
370 return EC;
371 }
372
Zachary Turner946204c2017-08-09 04:23:25 +0000373 if (Gsi) {
374 if (auto EC = Gsi->commit(Layout, Buffer))
Zachary Turner8d927b62017-07-31 19:36:08 +0000375 return EC;
376 }
377
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000378 auto InfoStreamBlocks = Layout.StreamMap[StreamPDB];
379 assert(!InfoStreamBlocks.empty());
380 uint64_t InfoStreamFileOffset =
381 blockToOffset(InfoStreamBlocks.front(), Layout.SB->BlockSize);
382 InfoStreamHeader *H = reinterpret_cast<InfoStreamHeader *>(
383 FOB->getBufferStart() + InfoStreamFileOffset);
384
Zachary Turnerf2282762018-03-23 19:57:25 +0000385 commitInjectedSources(Buffer, Layout);
386
Zachary Turnerc6a75a62018-03-01 18:00:29 +0000387 // Set the build id at the very end, after every other byte of the PDB
388 // has been written.
389 // FIXME: Use a hash of the PDB rather than time(nullptr) for the signature.
390 H->Age = Info->getAge();
391 H->Guid = Info->getGuid();
392 Optional<uint32_t> Sig = Info->getSignature();
393 H->Signature = Sig.hasValue() ? *Sig : time(nullptr);
394
Zachary Turnerd66889c2016-07-28 19:12:28 +0000395 return Buffer.commit();
Rui Ueyamafc22cef2016-09-30 20:34:44 +0000396}