xref: /freebsd/contrib/llvm-project/llvm/lib/DebugInfo/PDB/Native/PDBFileBuilder.cpp (revision 2f513db72b034fd5ef7f080b11be5c711c15186a)
1 //===- PDBFileBuilder.cpp - PDB File Creation -------------------*- C++ -*-===//
2 //
3 // 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
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "llvm/DebugInfo/PDB/Native/PDBFileBuilder.h"
10 
11 #include "llvm/ADT/BitVector.h"
12 
13 #include "llvm/DebugInfo/MSF/MSFBuilder.h"
14 #include "llvm/DebugInfo/PDB/Native/DbiStream.h"
15 #include "llvm/DebugInfo/PDB/Native/DbiStreamBuilder.h"
16 #include "llvm/DebugInfo/PDB/Native/GSIStreamBuilder.h"
17 #include "llvm/DebugInfo/PDB/Native/InfoStream.h"
18 #include "llvm/DebugInfo/PDB/Native/InfoStreamBuilder.h"
19 #include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h"
20 #include "llvm/DebugInfo/PDB/Native/RawError.h"
21 #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
22 #include "llvm/DebugInfo/PDB/Native/TpiStreamBuilder.h"
23 #include "llvm/Support/BinaryStream.h"
24 #include "llvm/Support/BinaryStreamWriter.h"
25 #include "llvm/Support/JamCRC.h"
26 #include "llvm/Support/Path.h"
27 #include "llvm/Support/xxhash.h"
28 
29 using namespace llvm;
30 using namespace llvm::codeview;
31 using namespace llvm::msf;
32 using namespace llvm::pdb;
33 using namespace llvm::support;
34 
35 PDBFileBuilder::PDBFileBuilder(BumpPtrAllocator &Allocator)
36     : Allocator(Allocator), InjectedSourceHashTraits(Strings),
37       InjectedSourceTable(2) {}
38 
39 PDBFileBuilder::~PDBFileBuilder() {}
40 
41 Error PDBFileBuilder::initialize(uint32_t BlockSize) {
42   auto ExpectedMsf = MSFBuilder::create(Allocator, BlockSize);
43   if (!ExpectedMsf)
44     return ExpectedMsf.takeError();
45   Msf = llvm::make_unique<MSFBuilder>(std::move(*ExpectedMsf));
46   return Error::success();
47 }
48 
49 MSFBuilder &PDBFileBuilder::getMsfBuilder() { return *Msf; }
50 
51 InfoStreamBuilder &PDBFileBuilder::getInfoBuilder() {
52   if (!Info)
53     Info = llvm::make_unique<InfoStreamBuilder>(*Msf, NamedStreams);
54   return *Info;
55 }
56 
57 DbiStreamBuilder &PDBFileBuilder::getDbiBuilder() {
58   if (!Dbi)
59     Dbi = llvm::make_unique<DbiStreamBuilder>(*Msf);
60   return *Dbi;
61 }
62 
63 TpiStreamBuilder &PDBFileBuilder::getTpiBuilder() {
64   if (!Tpi)
65     Tpi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamTPI);
66   return *Tpi;
67 }
68 
69 TpiStreamBuilder &PDBFileBuilder::getIpiBuilder() {
70   if (!Ipi)
71     Ipi = llvm::make_unique<TpiStreamBuilder>(*Msf, StreamIPI);
72   return *Ipi;
73 }
74 
75 PDBStringTableBuilder &PDBFileBuilder::getStringTableBuilder() {
76   return Strings;
77 }
78 
79 GSIStreamBuilder &PDBFileBuilder::getGsiBuilder() {
80   if (!Gsi)
81     Gsi = llvm::make_unique<GSIStreamBuilder>(*Msf);
82   return *Gsi;
83 }
84 
85 Expected<uint32_t> PDBFileBuilder::allocateNamedStream(StringRef Name,
86                                                        uint32_t Size) {
87   auto ExpectedStream = Msf->addStream(Size);
88   if (ExpectedStream)
89     NamedStreams.set(Name, *ExpectedStream);
90   return ExpectedStream;
91 }
92 
93 Error 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;
99   return Error::success();
100 }
101 
102 void 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 
124 Error PDBFileBuilder::finalizeMsfLayout() {
125 
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 
135   uint32_t StringsLen = Strings.calculateSerializedSize();
136 
137   Expected<uint32_t> SN = allocateNamedStream("/LinkInfo", 0);
138   if (!SN)
139     return SN.takeError();
140 
141   if (Gsi) {
142     if (auto EC = Gsi->finalizeMsfLayout())
143       return EC;
144     if (Dbi) {
145       Dbi->setPublicsStreamIndex(Gsi->getPublicsStreamIndex());
146       Dbi->setGlobalsStreamIndex(Gsi->getGlobalsStreamIndex());
147       Dbi->setSymbolRecordStreamIndex(Gsi->getRecordStreamIdx());
148     }
149   }
150   if (Tpi) {
151     if (auto EC = Tpi->finalizeMsfLayout())
152       return EC;
153   }
154   if (Dbi) {
155     if (auto EC = Dbi->finalizeMsfLayout())
156       return EC;
157   }
158   SN = allocateNamedStream("/names", StringsLen);
159   if (!SN)
160     return SN.takeError();
161 
162   if (Ipi) {
163     if (auto EC = Ipi->finalizeMsfLayout())
164       return EC;
165   }
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())
171       return EC;
172   }
173 
174   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                                  InjectedSourceHashTraits);
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 EC;
214   }
215 
216   return Error::success();
217 }
218 
219 Expected<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 
226 void PDBFileBuilder::commitSrcHeaderBlock(WritableBinaryStream &MsfBuffer,
227                                           const msf::MSFLayout &Layout) {
228   assert(!InjectedSourceTable.empty());
229 
230   uint32_t SN = cantFail(getNamedStreamIndex("/src/headerblock"));
231   auto Stream = WritableMappedBlockStream::createIndexedStream(
232       Layout, MsfBuffer, SN, Allocator);
233   BinaryStreamWriter Writer(*Stream);
234 
235   SrcHeaderBlockHeader Header;
236   ::memset(&Header, 0, sizeof(Header));
237   Header.Version = static_cast<uint32_t>(PdbRaw_SrcHeaderBlockVer::SrcVerOne);
238   Header.Size = Writer.bytesRemaining();
239 
240   cantFail(Writer.writeObject(Header));
241   cantFail(InjectedSourceTable.commit(Writer));
242 
243   assert(Writer.bytesRemaining() == 0);
244 }
245 
246 void PDBFileBuilder::commitInjectedSources(WritableBinaryStream &MsfBuffer,
247                                            const msf::MSFLayout &Layout) {
248   if (InjectedSourceTable.empty())
249     return;
250 
251   commitSrcHeaderBlock(MsfBuffer, Layout);
252 
253   for (const auto &IS : InjectedSources) {
254     uint32_t SN = cantFail(getNamedStreamIndex(IS.StreamName));
255 
256     auto SourceStream = WritableMappedBlockStream::createIndexedStream(
257         Layout, MsfBuffer, SN, Allocator);
258     BinaryStreamWriter SourceWriter(*SourceStream);
259     assert(SourceWriter.bytesRemaining() == IS.Content->getBufferSize());
260     cantFail(SourceWriter.writeBytes(
261         arrayRefFromStringRef(IS.Content->getBuffer())));
262   }
263 }
264 
265 Error PDBFileBuilder::commit(StringRef Filename, codeview::GUID *Guid) {
266   assert(!Filename.empty());
267   if (auto EC = finalizeMsfLayout())
268     return EC;
269 
270   MSFLayout Layout;
271   Expected<FileBufferByteStream> ExpectedMsfBuffer =
272       Msf->commit(Filename, Layout);
273   if (!ExpectedMsfBuffer)
274     return ExpectedMsfBuffer.takeError();
275   FileBufferByteStream Buffer = std::move(*ExpectedMsfBuffer);
276 
277   auto ExpectedSN = getNamedStreamIndex("/names");
278   if (!ExpectedSN)
279     return ExpectedSN.takeError();
280 
281   auto NS = WritableMappedBlockStream::createIndexedStream(
282       Layout, Buffer, *ExpectedSN, Allocator);
283   BinaryStreamWriter NSWriter(*NS);
284   if (auto EC = Strings.commit(NSWriter))
285     return EC;
286 
287   for (const auto &NSE : NamedStreamData) {
288     if (NSE.second.empty())
289       continue;
290 
291     auto NS = WritableMappedBlockStream::createIndexedStream(
292         Layout, Buffer, NSE.first, Allocator);
293     BinaryStreamWriter NSW(*NS);
294     if (auto EC = NSW.writeBytes(arrayRefFromStringRef(NSE.second)))
295       return EC;
296   }
297 
298   if (Info) {
299     if (auto EC = Info->commit(Layout, Buffer))
300       return EC;
301   }
302 
303   if (Dbi) {
304     if (auto EC = Dbi->commit(Layout, Buffer))
305       return EC;
306   }
307 
308   if (Tpi) {
309     if (auto EC = Tpi->commit(Layout, Buffer))
310       return EC;
311   }
312 
313   if (Ipi) {
314     if (auto EC = Ipi->commit(Layout, Buffer))
315       return EC;
316   }
317 
318   if (Gsi) {
319     if (auto EC = Gsi->commit(Layout, Buffer))
320       return EC;
321   }
322 
323   auto InfoStreamBlocks = Layout.StreamMap[StreamPDB];
324   assert(!InfoStreamBlocks.empty());
325   uint64_t InfoStreamFileOffset =
326       blockToOffset(InfoStreamBlocks.front(), Layout.SB->BlockSize);
327   InfoStreamHeader *H = reinterpret_cast<InfoStreamHeader *>(
328       Buffer.getBufferStart() + InfoStreamFileOffset);
329 
330   commitInjectedSources(Buffer, Layout);
331 
332   // Set the build id at the very end, after every other byte of the PDB
333   // has been written.
334   if (Info->hashPDBContentsToGUID()) {
335     // Compute a hash of all sections of the output file.
336     uint64_t Digest =
337         xxHash64({Buffer.getBufferStart(), Buffer.getBufferEnd()});
338 
339     H->Age = 1;
340 
341     memcpy(H->Guid.Guid, &Digest, 8);
342     // xxhash only gives us 8 bytes, so put some fixed data in the other half.
343     memcpy(H->Guid.Guid + 8, "LLD PDB.", 8);
344 
345     // Put the hash in the Signature field too.
346     H->Signature = static_cast<uint32_t>(Digest);
347 
348     // Return GUID to caller.
349     memcpy(Guid, H->Guid.Guid, 16);
350   } else {
351     H->Age = Info->getAge();
352     H->Guid = Info->getGuid();
353     Optional<uint32_t> Sig = Info->getSignature();
354     H->Signature = Sig.hasValue() ? *Sig : time(nullptr);
355   }
356 
357   return Buffer.commit();
358 }
359