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