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