blob: 90acfadd311ffdd11e15dad15177021d9d0e7ffb [file] [log] [blame]
Zachary Turnere204a6c2017-05-02 18:00:13 +00001//===- PDBStringTableBuilder.cpp - PDB String Table -------------*- C++ -*-===//
Rui Ueyamadcd32932017-01-15 00:36:02 +00002//
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
Zachary Turnere204a6c2017-05-02 18:00:13 +000010#include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h"
Zachary Turnerc504ae32017-05-03 15:58:37 +000011
Rui Ueyamadcd32932017-01-15 00:36:02 +000012#include "llvm/ADT/ArrayRef.h"
Zachary Turnerc504ae32017-05-03 15:58:37 +000013#include "llvm/DebugInfo/MSF/MappedBlockStream.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000014#include "llvm/DebugInfo/PDB/Native/Hash.h"
Zachary Turnerc504ae32017-05-03 15:58:37 +000015#include "llvm/DebugInfo/PDB/Native/PDBFileBuilder.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000016#include "llvm/DebugInfo/PDB/Native/RawTypes.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000017#include "llvm/Support/BinaryStreamWriter.h"
Rui Ueyamadcd32932017-01-15 00:36:02 +000018#include "llvm/Support/Endian.h"
19
20using namespace llvm;
Zachary Turnerc504ae32017-05-03 15:58:37 +000021using namespace llvm::msf;
Rui Ueyamadcd32932017-01-15 00:36:02 +000022using namespace llvm::support;
23using namespace llvm::support::endian;
24using namespace llvm::pdb;
25
Zachary Turnere204a6c2017-05-02 18:00:13 +000026uint32_t PDBStringTableBuilder::insert(StringRef S) {
Zachary Turnerc504ae32017-05-03 15:58:37 +000027 return Strings.insert(S);
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000028}
29
Rui Ueyamadcd32932017-01-15 00:36:02 +000030static uint32_t computeBucketCount(uint32_t NumStrings) {
31 // The /names stream is basically an on-disk open-addressing hash table.
32 // Hash collisions are resolved by linear probing. We cannot make
33 // utilization 100% because it will make the linear probing extremely
34 // slow. But lower utilization wastes disk space. As a reasonable
35 // load factor, we choose 80%. We need +1 because slot 0 is reserved.
36 return (NumStrings + 1) * 1.25;
37}
38
Zachary Turnerc504ae32017-05-03 15:58:37 +000039uint32_t PDBStringTableBuilder::calculateHashTableSize() const {
40 uint32_t Size = sizeof(uint32_t); // Hash table begins with 4-byte size field.
41 Size += sizeof(uint32_t) * computeBucketCount(Strings.size());
Daniel Jasperdff096f2017-05-03 07:29:25 +000042
Zachary Turner7dba20b2017-05-02 23:36:17 +000043 return Size;
44}
45
Zachary Turnerc504ae32017-05-03 15:58:37 +000046uint32_t PDBStringTableBuilder::calculateSerializedSize() const {
47 uint32_t Size = 0;
48 Size += sizeof(PDBStringTableHeader);
49 Size += Strings.calculateSerializedSize();
50 Size += calculateHashTableSize();
51 Size += sizeof(uint32_t); // The /names stream ends with the string count.
52 return Size;
53}
54
Zachary Turnera8cfc292017-06-14 15:59:27 +000055void PDBStringTableBuilder::setStrings(
56 const codeview::DebugStringTableSubsection &Strings) {
57 this->Strings = Strings;
58}
59
Zachary Turnerc504ae32017-05-03 15:58:37 +000060Error PDBStringTableBuilder::writeHeader(BinaryStreamWriter &Writer) const {
Rui Ueyamadcd32932017-01-15 00:36:02 +000061 // Write a header
Zachary Turnere204a6c2017-05-02 18:00:13 +000062 PDBStringTableHeader H;
63 H.Signature = PDBStringTableSignature;
Rui Ueyamadcd32932017-01-15 00:36:02 +000064 H.HashVersion = 1;
Zachary Turnerc504ae32017-05-03 15:58:37 +000065 H.ByteSize = Strings.calculateSerializedSize();
Rui Ueyamadcd32932017-01-15 00:36:02 +000066 if (auto EC = Writer.writeObject(H))
67 return EC;
Zachary Turnerc504ae32017-05-03 15:58:37 +000068 assert(Writer.bytesRemaining() == 0);
69 return Error::success();
70}
Rui Ueyamadcd32932017-01-15 00:36:02 +000071
Zachary Turnerc504ae32017-05-03 15:58:37 +000072Error PDBStringTableBuilder::writeStrings(BinaryStreamWriter &Writer) const {
73 if (auto EC = Strings.commit(Writer))
74 return EC;
Rui Ueyamadcd32932017-01-15 00:36:02 +000075
Zachary Turnerc504ae32017-05-03 15:58:37 +000076 assert(Writer.bytesRemaining() == 0);
77 return Error::success();
78}
79
80Error PDBStringTableBuilder::writeHashTable(BinaryStreamWriter &Writer) const {
Rui Ueyamadcd32932017-01-15 00:36:02 +000081 // Write a hash table.
82 uint32_t BucketCount = computeBucketCount(Strings.size());
Zachary Turner695ed562017-02-28 00:04:07 +000083 if (auto EC = Writer.writeInteger(BucketCount))
Rui Ueyamadcd32932017-01-15 00:36:02 +000084 return EC;
85 std::vector<ulittle32_t> Buckets(BucketCount);
86
Zachary Turnerc504ae32017-05-03 15:58:37 +000087 for (auto &Pair : Strings) {
88 StringRef S = Pair.getKey();
89 uint32_t Offset = Pair.getValue();
Rui Ueyamadcd32932017-01-15 00:36:02 +000090 uint32_t Hash = hashStringV1(S);
91
92 for (uint32_t I = 0; I != BucketCount; ++I) {
93 uint32_t Slot = (Hash + I) % BucketCount;
94 if (Slot == 0)
95 continue; // Skip reserved slot
96 if (Buckets[Slot] != 0)
97 continue;
98 Buckets[Slot] = Offset;
99 break;
100 }
101 }
102
103 if (auto EC = Writer.writeArray(ArrayRef<ulittle32_t>(Buckets)))
104 return EC;
Zachary Turnerc504ae32017-05-03 15:58:37 +0000105
106 assert(Writer.bytesRemaining() == 0);
107 return Error::success();
108}
109
110Error PDBStringTableBuilder::writeEpilogue(BinaryStreamWriter &Writer) const {
111 if (auto EC = Writer.writeInteger<uint32_t>(Strings.size()))
Rui Ueyamadcd32932017-01-15 00:36:02 +0000112 return EC;
Zachary Turnerc504ae32017-05-03 15:58:37 +0000113 assert(Writer.bytesRemaining() == 0);
114 return Error::success();
115}
116
117Error PDBStringTableBuilder::commit(BinaryStreamWriter &Writer) const {
118 BinaryStreamWriter SectionWriter;
119
120 std::tie(SectionWriter, Writer) = Writer.split(sizeof(PDBStringTableHeader));
121 if (auto EC = writeHeader(SectionWriter))
122 return EC;
123
124 std::tie(SectionWriter, Writer) =
125 Writer.split(Strings.calculateSerializedSize());
126 if (auto EC = writeStrings(SectionWriter))
127 return EC;
128
129 std::tie(SectionWriter, Writer) = Writer.split(calculateHashTableSize());
130 if (auto EC = writeHashTable(SectionWriter))
131 return EC;
132
133 std::tie(SectionWriter, Writer) = Writer.split(sizeof(uint32_t));
134 if (auto EC = writeEpilogue(SectionWriter))
135 return EC;
136
Rui Ueyamadcd32932017-01-15 00:36:02 +0000137 return Error::success();
138}