blob: 13106ccbc32c2596f081ff9d0b8387d9e859d625 [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"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000013#include "llvm/DebugInfo/PDB/Native/Hash.h"
14#include "llvm/DebugInfo/PDB/Native/RawTypes.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000015#include "llvm/Support/BinaryStreamWriter.h"
Rui Ueyamadcd32932017-01-15 00:36:02 +000016#include "llvm/Support/Endian.h"
17
18using namespace llvm;
Zachary Turnerc504ae32017-05-03 15:58:37 +000019using namespace llvm::msf;
Rui Ueyamadcd32932017-01-15 00:36:02 +000020using namespace llvm::support;
21using namespace llvm::support::endian;
22using namespace llvm::pdb;
23
Zachary Turner132d7a12018-03-20 17:06:39 +000024StringTableHashTraits::StringTableHashTraits(PDBStringTableBuilder &Table)
25 : Table(&Table) {}
26
27uint32_t StringTableHashTraits::hashLookupKey(StringRef S) const {
28 return Table->getIdForString(S);
29}
30
31StringRef StringTableHashTraits::storageKeyToLookupKey(uint32_t Offset) const {
32 return Table->getStringForId(Offset);
33}
34
35uint32_t StringTableHashTraits::lookupKeyToStorageKey(StringRef S) {
36 return Table->insert(S);
37}
38
Zachary Turnere204a6c2017-05-02 18:00:13 +000039uint32_t PDBStringTableBuilder::insert(StringRef S) {
Zachary Turnerc504ae32017-05-03 15:58:37 +000040 return Strings.insert(S);
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000041}
42
Zachary Turner132d7a12018-03-20 17:06:39 +000043uint32_t PDBStringTableBuilder::getIdForString(StringRef S) const {
44 return Strings.getIdForString(S);
45}
46
47StringRef PDBStringTableBuilder::getStringForId(uint32_t Id) const {
48 return Strings.getStringForId(Id);
49}
50
Rui Ueyamadcd32932017-01-15 00:36:02 +000051static uint32_t computeBucketCount(uint32_t NumStrings) {
52 // The /names stream is basically an on-disk open-addressing hash table.
53 // Hash collisions are resolved by linear probing. We cannot make
54 // utilization 100% because it will make the linear probing extremely
55 // slow. But lower utilization wastes disk space. As a reasonable
56 // load factor, we choose 80%. We need +1 because slot 0 is reserved.
57 return (NumStrings + 1) * 1.25;
58}
59
Zachary Turnerc504ae32017-05-03 15:58:37 +000060uint32_t PDBStringTableBuilder::calculateHashTableSize() const {
61 uint32_t Size = sizeof(uint32_t); // Hash table begins with 4-byte size field.
62 Size += sizeof(uint32_t) * computeBucketCount(Strings.size());
Daniel Jasperdff096f2017-05-03 07:29:25 +000063
Zachary Turner7dba20b2017-05-02 23:36:17 +000064 return Size;
65}
66
Zachary Turnerc504ae32017-05-03 15:58:37 +000067uint32_t PDBStringTableBuilder::calculateSerializedSize() const {
68 uint32_t Size = 0;
69 Size += sizeof(PDBStringTableHeader);
70 Size += Strings.calculateSerializedSize();
71 Size += calculateHashTableSize();
72 Size += sizeof(uint32_t); // The /names stream ends with the string count.
73 return Size;
74}
75
Zachary Turnera8cfc292017-06-14 15:59:27 +000076void PDBStringTableBuilder::setStrings(
77 const codeview::DebugStringTableSubsection &Strings) {
78 this->Strings = Strings;
79}
80
Zachary Turnerc504ae32017-05-03 15:58:37 +000081Error PDBStringTableBuilder::writeHeader(BinaryStreamWriter &Writer) const {
Rui Ueyamadcd32932017-01-15 00:36:02 +000082 // Write a header
Zachary Turnere204a6c2017-05-02 18:00:13 +000083 PDBStringTableHeader H;
84 H.Signature = PDBStringTableSignature;
Rui Ueyamadcd32932017-01-15 00:36:02 +000085 H.HashVersion = 1;
Zachary Turnerc504ae32017-05-03 15:58:37 +000086 H.ByteSize = Strings.calculateSerializedSize();
Rui Ueyamadcd32932017-01-15 00:36:02 +000087 if (auto EC = Writer.writeObject(H))
88 return EC;
Zachary Turnerc504ae32017-05-03 15:58:37 +000089 assert(Writer.bytesRemaining() == 0);
90 return Error::success();
91}
Rui Ueyamadcd32932017-01-15 00:36:02 +000092
Zachary Turnerc504ae32017-05-03 15:58:37 +000093Error PDBStringTableBuilder::writeStrings(BinaryStreamWriter &Writer) const {
94 if (auto EC = Strings.commit(Writer))
95 return EC;
Rui Ueyamadcd32932017-01-15 00:36:02 +000096
Zachary Turnerc504ae32017-05-03 15:58:37 +000097 assert(Writer.bytesRemaining() == 0);
98 return Error::success();
99}
100
101Error PDBStringTableBuilder::writeHashTable(BinaryStreamWriter &Writer) const {
Rui Ueyamadcd32932017-01-15 00:36:02 +0000102 // Write a hash table.
103 uint32_t BucketCount = computeBucketCount(Strings.size());
Zachary Turner695ed562017-02-28 00:04:07 +0000104 if (auto EC = Writer.writeInteger(BucketCount))
Rui Ueyamadcd32932017-01-15 00:36:02 +0000105 return EC;
106 std::vector<ulittle32_t> Buckets(BucketCount);
107
Zachary Turnerc504ae32017-05-03 15:58:37 +0000108 for (auto &Pair : Strings) {
109 StringRef S = Pair.getKey();
110 uint32_t Offset = Pair.getValue();
Rui Ueyamadcd32932017-01-15 00:36:02 +0000111 uint32_t Hash = hashStringV1(S);
112
113 for (uint32_t I = 0; I != BucketCount; ++I) {
114 uint32_t Slot = (Hash + I) % BucketCount;
115 if (Slot == 0)
116 continue; // Skip reserved slot
117 if (Buckets[Slot] != 0)
118 continue;
119 Buckets[Slot] = Offset;
120 break;
121 }
122 }
123
124 if (auto EC = Writer.writeArray(ArrayRef<ulittle32_t>(Buckets)))
125 return EC;
Zachary Turnerc504ae32017-05-03 15:58:37 +0000126
127 assert(Writer.bytesRemaining() == 0);
128 return Error::success();
129}
130
131Error PDBStringTableBuilder::writeEpilogue(BinaryStreamWriter &Writer) const {
132 if (auto EC = Writer.writeInteger<uint32_t>(Strings.size()))
Rui Ueyamadcd32932017-01-15 00:36:02 +0000133 return EC;
Zachary Turnerc504ae32017-05-03 15:58:37 +0000134 assert(Writer.bytesRemaining() == 0);
135 return Error::success();
136}
137
138Error PDBStringTableBuilder::commit(BinaryStreamWriter &Writer) const {
139 BinaryStreamWriter SectionWriter;
140
141 std::tie(SectionWriter, Writer) = Writer.split(sizeof(PDBStringTableHeader));
142 if (auto EC = writeHeader(SectionWriter))
143 return EC;
144
145 std::tie(SectionWriter, Writer) =
146 Writer.split(Strings.calculateSerializedSize());
147 if (auto EC = writeStrings(SectionWriter))
148 return EC;
149
150 std::tie(SectionWriter, Writer) = Writer.split(calculateHashTableSize());
151 if (auto EC = writeHashTable(SectionWriter))
152 return EC;
153
154 std::tie(SectionWriter, Writer) = Writer.split(sizeof(uint32_t));
155 if (auto EC = writeEpilogue(SectionWriter))
156 return EC;
157
Rui Ueyamadcd32932017-01-15 00:36:02 +0000158 return Error::success();
159}