1*8c099fe0SZachary Turner //===- DebugChecksumsSubsection.cpp ----------------------*- C++ -*-===//
2*8c099fe0SZachary Turner //
3*8c099fe0SZachary Turner //                     The LLVM Compiler Infrastructure
4*8c099fe0SZachary Turner //
5*8c099fe0SZachary Turner // This file is distributed under the University of Illinois Open Source
6*8c099fe0SZachary Turner // License. See LICENSE.TXT for details.
7*8c099fe0SZachary Turner //
8*8c099fe0SZachary Turner //===----------------------------------------------------------------------===//
9*8c099fe0SZachary Turner 
10*8c099fe0SZachary Turner #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
11*8c099fe0SZachary Turner 
12*8c099fe0SZachary Turner #include "llvm/DebugInfo/CodeView/CodeViewError.h"
13*8c099fe0SZachary Turner #include "llvm/DebugInfo/CodeView/StringTable.h"
14*8c099fe0SZachary Turner #include "llvm/Support/BinaryStreamReader.h"
15*8c099fe0SZachary Turner 
16*8c099fe0SZachary Turner using namespace llvm;
17*8c099fe0SZachary Turner using namespace llvm::codeview;
18*8c099fe0SZachary Turner 
19*8c099fe0SZachary Turner struct FileChecksumEntryHeader {
20*8c099fe0SZachary Turner   using ulittle32_t = support::ulittle32_t;
21*8c099fe0SZachary Turner 
22*8c099fe0SZachary Turner   ulittle32_t FileNameOffset; // Byte offset of filename in global string table.
23*8c099fe0SZachary Turner   uint8_t ChecksumSize;       // Number of bytes of checksum.
24*8c099fe0SZachary Turner   uint8_t ChecksumKind;       // FileChecksumKind
25*8c099fe0SZachary Turner                               // Checksum bytes follow.
26*8c099fe0SZachary Turner };
27*8c099fe0SZachary Turner 
28*8c099fe0SZachary Turner Error llvm::VarStreamArrayExtractor<FileChecksumEntry>::extract(
29*8c099fe0SZachary Turner     BinaryStreamRef Stream, uint32_t &Len, FileChecksumEntry &Item) {
30*8c099fe0SZachary Turner   BinaryStreamReader Reader(Stream);
31*8c099fe0SZachary Turner 
32*8c099fe0SZachary Turner   const FileChecksumEntryHeader *Header;
33*8c099fe0SZachary Turner   if (auto EC = Reader.readObject(Header))
34*8c099fe0SZachary Turner     return EC;
35*8c099fe0SZachary Turner 
36*8c099fe0SZachary Turner   Item.FileNameOffset = Header->FileNameOffset;
37*8c099fe0SZachary Turner   Item.Kind = static_cast<FileChecksumKind>(Header->ChecksumKind);
38*8c099fe0SZachary Turner   if (auto EC = Reader.readBytes(Item.Checksum, Header->ChecksumSize))
39*8c099fe0SZachary Turner     return EC;
40*8c099fe0SZachary Turner 
41*8c099fe0SZachary Turner   Len = alignTo(Header->ChecksumSize + sizeof(FileChecksumEntryHeader), 4);
42*8c099fe0SZachary Turner   return Error::success();
43*8c099fe0SZachary Turner }
44*8c099fe0SZachary Turner 
45*8c099fe0SZachary Turner Error DebugChecksumsSubsectionRef::initialize(BinaryStreamReader Reader) {
46*8c099fe0SZachary Turner   if (auto EC = Reader.readArray(Checksums, Reader.bytesRemaining()))
47*8c099fe0SZachary Turner     return EC;
48*8c099fe0SZachary Turner 
49*8c099fe0SZachary Turner   return Error::success();
50*8c099fe0SZachary Turner }
51*8c099fe0SZachary Turner 
52*8c099fe0SZachary Turner DebugChecksumsSubsection::DebugChecksumsSubsection(StringTable &Strings)
53*8c099fe0SZachary Turner     : DebugSubsection(DebugSubsectionKind::FileChecksums), Strings(Strings) {}
54*8c099fe0SZachary Turner 
55*8c099fe0SZachary Turner void DebugChecksumsSubsection::addChecksum(StringRef FileName,
56*8c099fe0SZachary Turner                                            FileChecksumKind Kind,
57*8c099fe0SZachary Turner                                            ArrayRef<uint8_t> Bytes) {
58*8c099fe0SZachary Turner   FileChecksumEntry Entry;
59*8c099fe0SZachary Turner   if (!Bytes.empty()) {
60*8c099fe0SZachary Turner     uint8_t *Copy = Storage.Allocate<uint8_t>(Bytes.size());
61*8c099fe0SZachary Turner     ::memcpy(Copy, Bytes.data(), Bytes.size());
62*8c099fe0SZachary Turner     Entry.Checksum = makeArrayRef(Copy, Bytes.size());
63*8c099fe0SZachary Turner   }
64*8c099fe0SZachary Turner 
65*8c099fe0SZachary Turner   Entry.FileNameOffset = Strings.insert(FileName);
66*8c099fe0SZachary Turner   Entry.Kind = Kind;
67*8c099fe0SZachary Turner   Checksums.push_back(Entry);
68*8c099fe0SZachary Turner 
69*8c099fe0SZachary Turner   // This maps the offset of this string in the string table to the offset
70*8c099fe0SZachary Turner   // of this checksum entry in the checksum buffer.
71*8c099fe0SZachary Turner   OffsetMap[Entry.FileNameOffset] = SerializedSize;
72*8c099fe0SZachary Turner   assert(SerializedSize % 4 == 0);
73*8c099fe0SZachary Turner 
74*8c099fe0SZachary Turner   uint32_t Len = alignTo(sizeof(FileChecksumEntryHeader) + Bytes.size(), 4);
75*8c099fe0SZachary Turner   SerializedSize += Len;
76*8c099fe0SZachary Turner }
77*8c099fe0SZachary Turner 
78*8c099fe0SZachary Turner uint32_t DebugChecksumsSubsection::calculateSerializedLength() {
79*8c099fe0SZachary Turner   return SerializedSize;
80*8c099fe0SZachary Turner }
81*8c099fe0SZachary Turner 
82*8c099fe0SZachary Turner Error DebugChecksumsSubsection::commit(BinaryStreamWriter &Writer) {
83*8c099fe0SZachary Turner   for (const auto &FC : Checksums) {
84*8c099fe0SZachary Turner     FileChecksumEntryHeader Header;
85*8c099fe0SZachary Turner     Header.ChecksumKind = uint8_t(FC.Kind);
86*8c099fe0SZachary Turner     Header.ChecksumSize = FC.Checksum.size();
87*8c099fe0SZachary Turner     Header.FileNameOffset = FC.FileNameOffset;
88*8c099fe0SZachary Turner     if (auto EC = Writer.writeObject(Header))
89*8c099fe0SZachary Turner       return EC;
90*8c099fe0SZachary Turner     if (auto EC = Writer.writeArray(makeArrayRef(FC.Checksum)))
91*8c099fe0SZachary Turner       return EC;
92*8c099fe0SZachary Turner     if (auto EC = Writer.padToAlignment(4))
93*8c099fe0SZachary Turner       return EC;
94*8c099fe0SZachary Turner   }
95*8c099fe0SZachary Turner   return Error::success();
96*8c099fe0SZachary Turner }
97*8c099fe0SZachary Turner 
98*8c099fe0SZachary Turner uint32_t DebugChecksumsSubsection::mapChecksumOffset(StringRef FileName) const {
99*8c099fe0SZachary Turner   uint32_t Offset = Strings.getStringId(FileName);
100*8c099fe0SZachary Turner   auto Iter = OffsetMap.find(Offset);
101*8c099fe0SZachary Turner   assert(Iter != OffsetMap.end());
102*8c099fe0SZachary Turner   return Iter->second;
103*8c099fe0SZachary Turner }
104