reference, declaration → definition definition → references, declarations, derived classes, virtual overrides reference to multiple definitions → definitions unreferenced |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 | //===- DebugStringTableSubsection.cpp - CodeView String Table -------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h" #include "llvm/ADT/StringRef.h" #include "llvm/DebugInfo/CodeView/CodeView.h" #include "llvm/Support/BinaryStreamReader.h" #include "llvm/Support/BinaryStreamWriter.h" #include "llvm/Support/Error.h" #include <algorithm> #include <cassert> #include <cstdint> using namespace llvm; using namespace llvm::codeview; DebugStringTableSubsectionRef::DebugStringTableSubsectionRef() : DebugSubsectionRef(DebugSubsectionKind::StringTable) {} Error DebugStringTableSubsectionRef::initialize(BinaryStreamRef Contents) { Stream = Contents; return Error::success(); } Error DebugStringTableSubsectionRef::initialize(BinaryStreamReader &Reader) { return Reader.readStreamRef(Stream); } Expected<StringRef> DebugStringTableSubsectionRef::getString(uint32_t Offset) const { BinaryStreamReader Reader(Stream); Reader.setOffset(Offset); StringRef Result; if (auto EC = Reader.readCString(Result)) return std::move(EC); return Result; } DebugStringTableSubsection::DebugStringTableSubsection() : DebugSubsection(DebugSubsectionKind::StringTable) {} uint32_t DebugStringTableSubsection::insert(StringRef S) { auto P = StringToId.insert({S, StringSize}); // If a given string didn't exist in the string table, we want to increment // the string table size and insert it into the reverse lookup. if (P.second) { IdToString.insert({P.first->getValue(), P.first->getKey()}); StringSize += S.size() + 1; // +1 for '\0' } return P.first->second; } uint32_t DebugStringTableSubsection::calculateSerializedSize() const { return StringSize; } Error DebugStringTableSubsection::commit(BinaryStreamWriter &Writer) const { uint32_t Begin = Writer.getOffset(); uint32_t End = Begin + StringSize; // Write a null string at the beginning. if (auto EC = Writer.writeCString(StringRef())) return EC; for (auto &Pair : StringToId) { StringRef S = Pair.getKey(); uint32_t Offset = Begin + Pair.getValue(); Writer.setOffset(Offset); if (auto EC = Writer.writeCString(S)) return EC; assert(Writer.getOffset() <= End); } Writer.setOffset(End); assert((End - Begin) == StringSize); return Error::success(); } uint32_t DebugStringTableSubsection::size() const { return StringToId.size(); } std::vector<uint32_t> DebugStringTableSubsection::sortedIds() const { std::vector<uint32_t> Result; Result.reserve(IdToString.size()); for (const auto &Entry : IdToString) Result.push_back(Entry.first); llvm::sort(Result); return Result; } uint32_t DebugStringTableSubsection::getIdForString(StringRef S) const { auto Iter = StringToId.find(S); assert(Iter != StringToId.end()); return Iter->second; } StringRef DebugStringTableSubsection::getStringForId(uint32_t Id) const { auto Iter = IdToString.find(Id); assert(Iter != IdToString.end()); return Iter->second; } |