diff options
Diffstat (limited to 'llvm/lib/DebugInfo')
| -rw-r--r-- | llvm/lib/DebugInfo/PDB/CMakeLists.txt | 1 | ||||
| -rw-r--r-- | llvm/lib/DebugInfo/PDB/Raw/NameHashTable.cpp | 11 | ||||
| -rw-r--r-- | llvm/lib/DebugInfo/PDB/Raw/NameHashTableBuilder.cpp | 101 | 
3 files changed, 105 insertions, 8 deletions
| diff --git a/llvm/lib/DebugInfo/PDB/CMakeLists.txt b/llvm/lib/DebugInfo/PDB/CMakeLists.txt index 599f01eaf74..a49c06b1e90 100644 --- a/llvm/lib/DebugInfo/PDB/CMakeLists.txt +++ b/llvm/lib/DebugInfo/PDB/CMakeLists.txt @@ -39,6 +39,7 @@ add_pdb_impl_folder(Raw    Raw/ModInfo.cpp    Raw/ModStream.cpp    Raw/NameHashTable.cpp +  Raw/NameHashTableBuilder.cpp    Raw/NameMap.cpp    Raw/NameMapBuilder.cpp    Raw/PDBFile.cpp diff --git a/llvm/lib/DebugInfo/PDB/Raw/NameHashTable.cpp b/llvm/lib/DebugInfo/PDB/Raw/NameHashTable.cpp index 84cccb354bd..86784a7aa75 100644 --- a/llvm/lib/DebugInfo/PDB/Raw/NameHashTable.cpp +++ b/llvm/lib/DebugInfo/PDB/Raw/NameHashTable.cpp @@ -13,6 +13,7 @@  #include "llvm/DebugInfo/MSF/StreamReader.h"  #include "llvm/DebugInfo/PDB/Raw/Hash.h"  #include "llvm/DebugInfo/PDB/Raw/RawError.h" +#include "llvm/DebugInfo/PDB/Raw/RawTypes.h"  #include "llvm/Support/Endian.h"  using namespace llvm; @@ -23,17 +24,11 @@ using namespace llvm::pdb;  NameHashTable::NameHashTable() : Signature(0), HashVersion(0), NameCount(0) {}  Error NameHashTable::load(StreamReader &Stream) { -  struct Header { -    support::ulittle32_t Signature; -    support::ulittle32_t HashVersion; -    support::ulittle32_t ByteSize; -  }; - -  const Header *H; +  const NameHashTableHeader *H;    if (auto EC = Stream.readObject(H))      return EC; -  if (H->Signature != 0xEFFEEFFE) +  if (H->Signature != NameHashTableSignature)      return make_error<RawError>(raw_error_code::corrupt_file,                                  "Invalid hash table signature");    if (H->HashVersion != 1 && H->HashVersion != 2) diff --git a/llvm/lib/DebugInfo/PDB/Raw/NameHashTableBuilder.cpp b/llvm/lib/DebugInfo/PDB/Raw/NameHashTableBuilder.cpp new file mode 100644 index 00000000000..2ef242515df --- /dev/null +++ b/llvm/lib/DebugInfo/PDB/Raw/NameHashTableBuilder.cpp @@ -0,0 +1,101 @@ +//===- NameHashTable.cpp - PDB Name Hash Table ------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "llvm/DebugInfo/PDB/Raw/NameHashTableBuilder.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/DebugInfo/MSF/StreamWriter.h" +#include "llvm/DebugInfo/PDB/Raw/Hash.h" +#include "llvm/DebugInfo/PDB/Raw/RawTypes.h" +#include "llvm/Support/Endian.h" + +using namespace llvm; +using namespace llvm::support; +using namespace llvm::support::endian; +using namespace llvm::pdb; + +uint32_t NameHashTableBuilder::insert(StringRef S) { +  auto P = Strings.insert({S, StringSize}); + +  // If a given string didn't exist in the string table, we want to increment +  // the string table size. +  if (P.second) +    StringSize += S.size() + 1; // +1 for '\0' +  return P.first->second; +} + +static uint32_t computeBucketCount(uint32_t NumStrings) { +  // The /names stream is basically an on-disk open-addressing hash table. +  // Hash collisions are resolved by linear probing. We cannot make +  // utilization 100% because it will make the linear probing extremely +  // slow. But lower utilization wastes disk space. As a reasonable +  // load factor, we choose 80%. We need +1 because slot 0 is reserved. +  return (NumStrings + 1) * 1.25; +} + +uint32_t NameHashTableBuilder::calculateSerializedLength() const { +  uint32_t Size = 0; +  Size += sizeof(NameHashTableHeader); +  Size += StringSize; +  Size += 4; // Hash table begins with 4-byte size field. + +  uint32_t BucketCount = computeBucketCount(Strings.size()); +  Size += BucketCount * 4; + +  Size += 4; // The /names stream ends with the number of strings. +  return Size; +} + +Error NameHashTableBuilder::commit(msf::StreamWriter &Writer) const { +  // Write a header +  NameHashTableHeader H; +  H.Signature = NameHashTableSignature; +  H.HashVersion = 1; +  H.ByteSize = StringSize; +  if (auto EC = Writer.writeObject(H)) +    return EC; + +  // Write a string table. +  uint32_t StringStart = Writer.getOffset(); +  for (auto Pair : Strings) { +    StringRef S = Pair.first; +    uint32_t Offset = Pair.second; +    Writer.setOffset(StringStart + Offset); +    if (auto EC = Writer.writeZeroString(S)) +      return EC; +  } +  Writer.setOffset(StringStart + StringSize); + +  // Write a hash table. +  uint32_t BucketCount = computeBucketCount(Strings.size()); +  if (auto EC = Writer.writeInteger(BucketCount)) +    return EC; +  std::vector<ulittle32_t> Buckets(BucketCount); + +  for (auto Pair : Strings) { +    StringRef S = Pair.first; +    uint32_t Offset = Pair.second; +    uint32_t Hash = hashStringV1(S); + +    for (uint32_t I = 0; I != BucketCount; ++I) { +      uint32_t Slot = (Hash + I) % BucketCount; +      if (Slot == 0) +        continue; // Skip reserved slot +      if (Buckets[Slot] != 0) +        continue; +      Buckets[Slot] = Offset; +      break; +    } +  } + +  if (auto EC = Writer.writeArray(ArrayRef<ulittle32_t>(Buckets))) +    return EC; +  if (auto EC = Writer.writeInteger(static_cast<uint32_t>(Strings.size()))) +    return EC; +  return Error::success(); +} | 

