| Eugene Zelenko | 8456b16 | 2017-06-29 00:05:44 +0000 | [diff] [blame] | 1 | //===- DebugStringTableSubsection.cpp - CodeView String Table -------------===// |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 2 | // |
| Chandler Carruth | 2946cd7 | 2019-01-19 08:50:56 +0000 | [diff] [blame] | 3 | // 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 Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 9 | #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h" |
| Eugene Zelenko | 8456b16 | 2017-06-29 00:05:44 +0000 | [diff] [blame] | 10 | #include "llvm/ADT/StringRef.h" |
| 11 | #include "llvm/DebugInfo/CodeView/CodeView.h" |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 12 | #include "llvm/Support/BinaryStreamReader.h" |
| 13 | #include "llvm/Support/BinaryStreamWriter.h" |
| Eugene Zelenko | 8456b16 | 2017-06-29 00:05:44 +0000 | [diff] [blame] | 14 | #include "llvm/Support/Error.h" |
| 15 | #include <algorithm> |
| 16 | #include <cassert> |
| 17 | #include <cstdint> |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 18 | |
| 19 | using namespace llvm; |
| 20 | using namespace llvm::codeview; |
| 21 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 22 | DebugStringTableSubsectionRef::DebugStringTableSubsectionRef() |
| 23 | : DebugSubsectionRef(DebugSubsectionKind::StringTable) {} |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 24 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 25 | Error DebugStringTableSubsectionRef::initialize(BinaryStreamRef Contents) { |
| Zachary Turner | 2d5c2cd | 2017-05-03 17:11:11 +0000 | [diff] [blame] | 26 | Stream = Contents; |
| 27 | return Error::success(); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 28 | } |
| Eugene Zelenko | 8456b16 | 2017-06-29 00:05:44 +0000 | [diff] [blame] | 29 | |
| Zachary Turner | 1bf7762 | 2017-06-08 23:49:01 +0000 | [diff] [blame] | 30 | Error DebugStringTableSubsectionRef::initialize(BinaryStreamReader &Reader) { |
| Zachary Turner | deb3913 | 2017-06-09 00:28:08 +0000 | [diff] [blame] | 31 | return Reader.readStreamRef(Stream); |
| Zachary Turner | 1bf7762 | 2017-06-08 23:49:01 +0000 | [diff] [blame] | 32 | } |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 33 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 34 | Expected<StringRef> |
| 35 | DebugStringTableSubsectionRef::getString(uint32_t Offset) const { |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 36 | BinaryStreamReader Reader(Stream); |
| 37 | Reader.setOffset(Offset); |
| 38 | StringRef Result; |
| Zachary Turner | 2d5c2cd | 2017-05-03 17:11:11 +0000 | [diff] [blame] | 39 | if (auto EC = Reader.readCString(Result)) |
| 40 | return std::move(EC); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 41 | return Result; |
| 42 | } |
| 43 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 44 | DebugStringTableSubsection::DebugStringTableSubsection() |
| 45 | : DebugSubsection(DebugSubsectionKind::StringTable) {} |
| 46 | |
| 47 | uint32_t DebugStringTableSubsection::insert(StringRef S) { |
| Zachary Turner | 71d36ad | 2018-03-22 17:37:28 +0000 | [diff] [blame] | 48 | auto P = StringToId.insert({S, StringSize}); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 49 | |
| 50 | // If a given string didn't exist in the string table, we want to increment |
| Zachary Turner | 71d36ad | 2018-03-22 17:37:28 +0000 | [diff] [blame] | 51 | // the string table size and insert it into the reverse lookup. |
| 52 | if (P.second) { |
| 53 | IdToString.insert({P.first->getValue(), P.first->getKey()}); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 54 | StringSize += S.size() + 1; // +1 for '\0' |
| Zachary Turner | 71d36ad | 2018-03-22 17:37:28 +0000 | [diff] [blame] | 55 | } |
| 56 | |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 57 | return P.first->second; |
| 58 | } |
| 59 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 60 | uint32_t DebugStringTableSubsection::calculateSerializedSize() const { |
| 61 | return StringSize; |
| 62 | } |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 63 | |
| Zachary Turner | 591312c | 2017-05-30 17:13:33 +0000 | [diff] [blame] | 64 | Error DebugStringTableSubsection::commit(BinaryStreamWriter &Writer) const { |
| Zachary Turner | deb3913 | 2017-06-09 00:28:08 +0000 | [diff] [blame] | 65 | uint32_t Begin = Writer.getOffset(); |
| 66 | uint32_t End = Begin + StringSize; |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 67 | |
| Zachary Turner | a8cfc29 | 2017-06-14 15:59:27 +0000 | [diff] [blame] | 68 | // Write a null string at the beginning. |
| 69 | if (auto EC = Writer.writeCString(StringRef())) |
| 70 | return EC; |
| 71 | |
| Zachary Turner | 71d36ad | 2018-03-22 17:37:28 +0000 | [diff] [blame] | 72 | for (auto &Pair : StringToId) { |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 73 | StringRef S = Pair.getKey(); |
| Zachary Turner | deb3913 | 2017-06-09 00:28:08 +0000 | [diff] [blame] | 74 | uint32_t Offset = Begin + Pair.getValue(); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 75 | Writer.setOffset(Offset); |
| 76 | if (auto EC = Writer.writeCString(S)) |
| 77 | return EC; |
| Zachary Turner | deb3913 | 2017-06-09 00:28:08 +0000 | [diff] [blame] | 78 | assert(Writer.getOffset() <= End); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 79 | } |
| 80 | |
| Zachary Turner | deb3913 | 2017-06-09 00:28:08 +0000 | [diff] [blame] | 81 | Writer.setOffset(End); |
| Zachary Turner | a8cfc29 | 2017-06-14 15:59:27 +0000 | [diff] [blame] | 82 | assert((End - Begin) == StringSize); |
| Zachary Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 83 | return Error::success(); |
| 84 | } |
| 85 | |
| Zachary Turner | 71d36ad | 2018-03-22 17:37:28 +0000 | [diff] [blame] | 86 | uint32_t DebugStringTableSubsection::size() const { return StringToId.size(); } |
| Zachary Turner | cf468d8 | 2017-05-03 17:11:40 +0000 | [diff] [blame] | 87 | |
| Zachary Turner | a6fb536 | 2018-03-23 18:43:39 +0000 | [diff] [blame] | 88 | std::vector<uint32_t> DebugStringTableSubsection::sortedIds() const { |
| 89 | std::vector<uint32_t> Result; |
| 90 | Result.reserve(IdToString.size()); |
| 91 | for (const auto &Entry : IdToString) |
| 92 | Result.push_back(Entry.first); |
| Fangrui Song | 0cac726 | 2018-09-27 02:13:45 +0000 | [diff] [blame] | 93 | llvm::sort(Result); |
| Zachary Turner | a6fb536 | 2018-03-23 18:43:39 +0000 | [diff] [blame] | 94 | return Result; |
| 95 | } |
| 96 | |
| Zachary Turner | 71d36ad | 2018-03-22 17:37:28 +0000 | [diff] [blame] | 97 | uint32_t DebugStringTableSubsection::getIdForString(StringRef S) const { |
| 98 | auto Iter = StringToId.find(S); |
| 99 | assert(Iter != StringToId.end()); |
| 100 | return Iter->second; |
| 101 | } |
| 102 | |
| 103 | StringRef DebugStringTableSubsection::getStringForId(uint32_t Id) const { |
| 104 | auto Iter = IdToString.find(Id); |
| 105 | assert(Iter != IdToString.end()); |
| Zachary Turner | 92dcdda | 2017-06-02 19:49:14 +0000 | [diff] [blame] | 106 | return Iter->second; |
| Zachary Turner | cf468d8 | 2017-05-03 17:11:40 +0000 | [diff] [blame] | 107 | } |