blob: 2be1656e06bbda2a7dab7acbe0418bdaa7c69d44 [file] [log] [blame]
Zachary Turnerc504ae32017-05-03 15:58:37 +00001//===- PDBStringTable.cpp - PDB String Table ---------------------*- C++-*-===//
Zachary Turner0eace0b2016-05-02 18:09:14 +00002//
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 Turner0eace0b2016-05-02 18:09:14 +00006//
7//===----------------------------------------------------------------------===//
8
Zachary Turnere204a6c2017-05-02 18:00:13 +00009#include "llvm/DebugInfo/PDB/Native/PDBStringTable.h"
Zachary Turner0eace0b2016-05-02 18:09:14 +000010
11#include "llvm/ADT/ArrayRef.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000012#include "llvm/DebugInfo/PDB/Native/Hash.h"
13#include "llvm/DebugInfo/PDB/Native/RawError.h"
14#include "llvm/DebugInfo/PDB/Native/RawTypes.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000015#include "llvm/Support/BinaryStreamReader.h"
Zachary Turner0eace0b2016-05-02 18:09:14 +000016#include "llvm/Support/Endian.h"
17
18using namespace llvm;
19using namespace llvm::support;
20using namespace llvm::pdb;
21
Zachary Turnerf3b4b2d2017-07-07 18:45:37 +000022uint32_t PDBStringTable::getByteSize() const { return Header->ByteSize; }
Zachary Turnerc504ae32017-05-03 15:58:37 +000023uint32_t PDBStringTable::getNameCount() const { return NameCount; }
24uint32_t PDBStringTable::getHashVersion() const { return Header->HashVersion; }
25uint32_t PDBStringTable::getSignature() const { return Header->Signature; }
Zachary Turner0eace0b2016-05-02 18:09:14 +000026
Zachary Turnerc504ae32017-05-03 15:58:37 +000027Error PDBStringTable::readHeader(BinaryStreamReader &Reader) {
28 if (auto EC = Reader.readObject(Header))
Zachary Turner819e77d2016-05-06 20:51:57 +000029 return EC;
30
Zachary Turnerc504ae32017-05-03 15:58:37 +000031 if (Header->Signature != PDBStringTableSignature)
Zachary Turner819e77d2016-05-06 20:51:57 +000032 return make_error<RawError>(raw_error_code::corrupt_file,
33 "Invalid hash table signature");
Zachary Turnerc504ae32017-05-03 15:58:37 +000034 if (Header->HashVersion != 1 && Header->HashVersion != 2)
Zachary Turner819e77d2016-05-06 20:51:57 +000035 return make_error<RawError>(raw_error_code::corrupt_file,
36 "Unsupported hash version");
Zachary Turner0eace0b2016-05-02 18:09:14 +000037
Zachary Turnerc504ae32017-05-03 15:58:37 +000038 assert(Reader.bytesRemaining() == 0);
39 return Error::success();
40}
41
42Error PDBStringTable::readStrings(BinaryStreamReader &Reader) {
Zachary Turner2d5c2cd2017-05-03 17:11:11 +000043 BinaryStreamRef Stream;
44 if (auto EC = Reader.readStreamRef(Stream))
45 return EC;
46
47 if (auto EC = Strings.initialize(Stream)) {
David Majnemer836937e2016-05-27 16:16:56 +000048 return joinErrors(std::move(EC),
49 make_error<RawError>(raw_error_code::corrupt_file,
50 "Invalid hash table byte length"));
Zachary Turnerc504ae32017-05-03 15:58:37 +000051 }
Zachary Turner0eace0b2016-05-02 18:09:14 +000052
Zachary Turnerc504ae32017-05-03 15:58:37 +000053 assert(Reader.bytesRemaining() == 0);
54 return Error::success();
55}
56
Zachary Turnera8cfc292017-06-14 15:59:27 +000057const codeview::DebugStringTableSubsectionRef &
58PDBStringTable::getStringTable() const {
Zachary Turner92dcdda2017-06-02 19:49:14 +000059 return Strings;
60}
61
Zachary Turnerc504ae32017-05-03 15:58:37 +000062Error PDBStringTable::readHashTable(BinaryStreamReader &Reader) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000063 const support::ulittle32_t *HashCount;
Zachary Turnerc504ae32017-05-03 15:58:37 +000064 if (auto EC = Reader.readObject(HashCount))
Zachary Turner819e77d2016-05-06 20:51:57 +000065 return EC;
66
Zachary Turnerc504ae32017-05-03 15:58:37 +000067 if (auto EC = Reader.readArray(IDs, *HashCount)) {
David Majnemer836937e2016-05-27 16:16:56 +000068 return joinErrors(std::move(EC),
69 make_error<RawError>(raw_error_code::corrupt_file,
70 "Could not read bucket array"));
Zachary Turnerc504ae32017-05-03 15:58:37 +000071 }
Zachary Turnerf1220082017-03-15 22:19:30 +000072
Zachary Turner819e77d2016-05-06 20:51:57 +000073 return Error::success();
Zachary Turner0eace0b2016-05-02 18:09:14 +000074}
75
Zachary Turnerc504ae32017-05-03 15:58:37 +000076Error PDBStringTable::readEpilogue(BinaryStreamReader &Reader) {
77 if (auto EC = Reader.readInteger(NameCount))
78 return EC;
79
80 assert(Reader.bytesRemaining() == 0);
81 return Error::success();
82}
83
84Error PDBStringTable::reload(BinaryStreamReader &Reader) {
85
86 BinaryStreamReader SectionReader;
87
88 std::tie(SectionReader, Reader) = Reader.split(sizeof(PDBStringTableHeader));
89 if (auto EC = readHeader(SectionReader))
90 return EC;
91
92 std::tie(SectionReader, Reader) = Reader.split(Header->ByteSize);
93 if (auto EC = readStrings(SectionReader))
94 return EC;
95
96 // We don't know how long the hash table is until we parse it, so let the
97 // function responsible for doing that figure it out.
98 if (auto EC = readHashTable(Reader))
99 return EC;
100
101 std::tie(SectionReader, Reader) = Reader.split(sizeof(uint32_t));
102 if (auto EC = readEpilogue(SectionReader))
103 return EC;
104
105 assert(Reader.bytesRemaining() == 0);
106 return Error::success();
107}
Zachary Turnerf1220082017-03-15 22:19:30 +0000108
Zachary Turner2d5c2cd2017-05-03 17:11:11 +0000109Expected<StringRef> PDBStringTable::getStringForID(uint32_t ID) const {
Zachary Turnerc504ae32017-05-03 15:58:37 +0000110 return Strings.getString(ID);
Zachary Turner0eace0b2016-05-02 18:09:14 +0000111}
112
Zachary Turner2d5c2cd2017-05-03 17:11:11 +0000113Expected<uint32_t> PDBStringTable::getIDForString(StringRef Str) const {
Zachary Turnerc504ae32017-05-03 15:58:37 +0000114 uint32_t Hash =
115 (Header->HashVersion == 1) ? hashStringV1(Str) : hashStringV2(Str);
Zachary Turner0eace0b2016-05-02 18:09:14 +0000116 size_t Count = IDs.size();
117 uint32_t Start = Hash % Count;
118 for (size_t I = 0; I < Count; ++I) {
119 // The hash is just a starting point for the search, but if it
120 // doesn't work we should find the string no matter what, because
121 // we iterate the entire array.
122 uint32_t Index = (Start + I) % Count;
123
Zachary Turnereb629992018-03-21 22:23:59 +0000124 // If we find 0, it means the item isn't in the hash table.
Zachary Turner0eace0b2016-05-02 18:09:14 +0000125 uint32_t ID = IDs[Index];
Zachary Turnereb629992018-03-21 22:23:59 +0000126 if (ID == 0)
127 return make_error<RawError>(raw_error_code::no_entry);
Zachary Turner2d5c2cd2017-05-03 17:11:11 +0000128 auto ExpectedStr = getStringForID(ID);
129 if (!ExpectedStr)
130 return ExpectedStr.takeError();
131
132 if (*ExpectedStr == Str)
Zachary Turner0eace0b2016-05-02 18:09:14 +0000133 return ID;
134 }
Zachary Turner2d5c2cd2017-05-03 17:11:11 +0000135 return make_error<RawError>(raw_error_code::no_entry);
Zachary Turner0eace0b2016-05-02 18:09:14 +0000136}
137
Zachary Turnere204a6c2017-05-02 18:00:13 +0000138FixedStreamArray<support::ulittle32_t> PDBStringTable::name_ids() const {
Zachary Turnerb393d952016-05-27 03:51:53 +0000139 return IDs;
140}