xref: /src/contrib/llvm-project/llvm/lib/DebugInfo/CodeView/DebugChecksumsSubsection.cpp (revision bdd1243df58e60e85101c09001d9812a789b6bc4)
19df3605dSDimitry Andric //===- DebugChecksumsSubsection.cpp ---------------------------------------===//
2a303c417SDimitry Andric //
3e6d15924SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e6d15924SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5e6d15924SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6a303c417SDimitry Andric //
7a303c417SDimitry Andric //===----------------------------------------------------------------------===//
8a303c417SDimitry Andric 
9ee2f195dSDimitry Andric #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
109df3605dSDimitry Andric #include "llvm/ADT/ArrayRef.h"
119df3605dSDimitry Andric #include "llvm/DebugInfo/CodeView/CodeView.h"
12ee2f195dSDimitry Andric #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
13a303c417SDimitry Andric #include "llvm/Support/BinaryStreamReader.h"
149df3605dSDimitry Andric #include "llvm/Support/BinaryStreamWriter.h"
159df3605dSDimitry Andric #include "llvm/Support/Endian.h"
169df3605dSDimitry Andric #include "llvm/Support/Error.h"
179df3605dSDimitry Andric #include "llvm/Support/MathExtras.h"
189df3605dSDimitry Andric #include <cassert>
199df3605dSDimitry Andric #include <cstdint>
209df3605dSDimitry Andric #include <cstring>
21a303c417SDimitry Andric 
22a303c417SDimitry Andric using namespace llvm;
23a303c417SDimitry Andric using namespace llvm::codeview;
24a303c417SDimitry Andric 
25a303c417SDimitry Andric struct FileChecksumEntryHeader {
26a303c417SDimitry Andric   using ulittle32_t = support::ulittle32_t;
27a303c417SDimitry Andric 
28a303c417SDimitry Andric   ulittle32_t FileNameOffset; // Byte offset of filename in global string table.
29a303c417SDimitry Andric   uint8_t ChecksumSize;       // Number of bytes of checksum.
30a303c417SDimitry Andric   uint8_t ChecksumKind;       // FileChecksumKind
31a303c417SDimitry Andric                               // Checksum bytes follow.
32a303c417SDimitry Andric };
33a303c417SDimitry Andric 
349df3605dSDimitry Andric Error VarStreamArrayExtractor<FileChecksumEntry>::
operator ()(BinaryStreamRef Stream,uint32_t & Len,FileChecksumEntry & Item)357ab83427SDimitry Andric operator()(BinaryStreamRef Stream, uint32_t &Len, FileChecksumEntry &Item) {
36a303c417SDimitry Andric   BinaryStreamReader Reader(Stream);
37a303c417SDimitry Andric 
38a303c417SDimitry Andric   const FileChecksumEntryHeader *Header;
39a303c417SDimitry Andric   if (auto EC = Reader.readObject(Header))
40a303c417SDimitry Andric     return EC;
41a303c417SDimitry Andric 
42a303c417SDimitry Andric   Item.FileNameOffset = Header->FileNameOffset;
43a303c417SDimitry Andric   Item.Kind = static_cast<FileChecksumKind>(Header->ChecksumKind);
44a303c417SDimitry Andric   if (auto EC = Reader.readBytes(Item.Checksum, Header->ChecksumSize))
45a303c417SDimitry Andric     return EC;
46a303c417SDimitry Andric 
47a303c417SDimitry Andric   Len = alignTo(Header->ChecksumSize + sizeof(FileChecksumEntryHeader), 4);
48a303c417SDimitry Andric   return Error::success();
49a303c417SDimitry Andric }
50a303c417SDimitry Andric 
initialize(BinaryStreamReader Reader)51ee2f195dSDimitry Andric Error DebugChecksumsSubsectionRef::initialize(BinaryStreamReader Reader) {
52a303c417SDimitry Andric   if (auto EC = Reader.readArray(Checksums, Reader.bytesRemaining()))
53a303c417SDimitry Andric     return EC;
54a303c417SDimitry Andric 
55a303c417SDimitry Andric   return Error::success();
56a303c417SDimitry Andric }
579df3605dSDimitry Andric 
initialize(BinaryStreamRef Section)58ee2f195dSDimitry Andric Error DebugChecksumsSubsectionRef::initialize(BinaryStreamRef Section) {
59ee2f195dSDimitry Andric   BinaryStreamReader Reader(Section);
60ee2f195dSDimitry Andric   return initialize(Reader);
61ee2f195dSDimitry Andric }
62a303c417SDimitry Andric 
DebugChecksumsSubsection(DebugStringTableSubsection & Strings)63ee2f195dSDimitry Andric DebugChecksumsSubsection::DebugChecksumsSubsection(
64ee2f195dSDimitry Andric     DebugStringTableSubsection &Strings)
65ee2f195dSDimitry Andric     : DebugSubsection(DebugSubsectionKind::FileChecksums), Strings(Strings) {}
66a303c417SDimitry Andric 
addChecksum(StringRef FileName,FileChecksumKind Kind,ArrayRef<uint8_t> Bytes)67ee2f195dSDimitry Andric void DebugChecksumsSubsection::addChecksum(StringRef FileName,
68a303c417SDimitry Andric                                            FileChecksumKind Kind,
69a303c417SDimitry Andric                                            ArrayRef<uint8_t> Bytes) {
70a303c417SDimitry Andric   FileChecksumEntry Entry;
71a303c417SDimitry Andric   if (!Bytes.empty()) {
72a303c417SDimitry Andric     uint8_t *Copy = Storage.Allocate<uint8_t>(Bytes.size());
73a303c417SDimitry Andric     ::memcpy(Copy, Bytes.data(), Bytes.size());
74e3b55780SDimitry Andric     Entry.Checksum = ArrayRef(Copy, Bytes.size());
75a303c417SDimitry Andric   }
76148779dfSDimitry Andric 
77148779dfSDimitry Andric   Entry.FileNameOffset = Strings.insert(FileName);
78a303c417SDimitry Andric   Entry.Kind = Kind;
79a303c417SDimitry Andric   Checksums.push_back(Entry);
80a303c417SDimitry Andric 
81a303c417SDimitry Andric   // This maps the offset of this string in the string table to the offset
82a303c417SDimitry Andric   // of this checksum entry in the checksum buffer.
83148779dfSDimitry Andric   OffsetMap[Entry.FileNameOffset] = SerializedSize;
84a303c417SDimitry Andric   assert(SerializedSize % 4 == 0);
85a303c417SDimitry Andric 
86a303c417SDimitry Andric   uint32_t Len = alignTo(sizeof(FileChecksumEntryHeader) + Bytes.size(), 4);
87a303c417SDimitry Andric   SerializedSize += Len;
88a303c417SDimitry Andric }
89a303c417SDimitry Andric 
calculateSerializedSize() const90ee2f195dSDimitry Andric uint32_t DebugChecksumsSubsection::calculateSerializedSize() const {
91a303c417SDimitry Andric   return SerializedSize;
92a303c417SDimitry Andric }
93a303c417SDimitry Andric 
commit(BinaryStreamWriter & Writer) const94ee2f195dSDimitry Andric Error DebugChecksumsSubsection::commit(BinaryStreamWriter &Writer) const {
95a303c417SDimitry Andric   for (const auto &FC : Checksums) {
96a303c417SDimitry Andric     FileChecksumEntryHeader Header;
97a303c417SDimitry Andric     Header.ChecksumKind = uint8_t(FC.Kind);
98a303c417SDimitry Andric     Header.ChecksumSize = FC.Checksum.size();
99a303c417SDimitry Andric     Header.FileNameOffset = FC.FileNameOffset;
100a303c417SDimitry Andric     if (auto EC = Writer.writeObject(Header))
101a303c417SDimitry Andric       return EC;
102e3b55780SDimitry Andric     if (auto EC = Writer.writeArray(ArrayRef(FC.Checksum)))
103a303c417SDimitry Andric       return EC;
104a303c417SDimitry Andric     if (auto EC = Writer.padToAlignment(4))
105a303c417SDimitry Andric       return EC;
106a303c417SDimitry Andric   }
107a303c417SDimitry Andric   return Error::success();
108a303c417SDimitry Andric }
109a303c417SDimitry Andric 
mapChecksumOffset(StringRef FileName) const110ee2f195dSDimitry Andric uint32_t DebugChecksumsSubsection::mapChecksumOffset(StringRef FileName) const {
111eb11fae6SDimitry Andric   uint32_t Offset = Strings.getIdForString(FileName);
112148779dfSDimitry Andric   auto Iter = OffsetMap.find(Offset);
113a303c417SDimitry Andric   assert(Iter != OffsetMap.end());
114a303c417SDimitry Andric   return Iter->second;
115a303c417SDimitry Andric }
116