18c099fe0SZachary Turner //===- DebugChecksumsSubsection.cpp ----------------------*- C++ -*-===//
28c099fe0SZachary Turner //
38c099fe0SZachary Turner //                     The LLVM Compiler Infrastructure
48c099fe0SZachary Turner //
58c099fe0SZachary Turner // This file is distributed under the University of Illinois Open Source
68c099fe0SZachary Turner // License. See LICENSE.TXT for details.
78c099fe0SZachary Turner //
88c099fe0SZachary Turner //===----------------------------------------------------------------------===//
98c099fe0SZachary Turner 
108c099fe0SZachary Turner #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
118c099fe0SZachary Turner 
128c099fe0SZachary Turner #include "llvm/DebugInfo/CodeView/CodeViewError.h"
13591312c5SZachary Turner #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
148c099fe0SZachary Turner #include "llvm/Support/BinaryStreamReader.h"
158c099fe0SZachary Turner 
168c099fe0SZachary Turner using namespace llvm;
178c099fe0SZachary Turner using namespace llvm::codeview;
188c099fe0SZachary Turner 
198c099fe0SZachary Turner struct FileChecksumEntryHeader {
208c099fe0SZachary Turner   using ulittle32_t = support::ulittle32_t;
218c099fe0SZachary Turner 
228c099fe0SZachary Turner   ulittle32_t FileNameOffset; // Byte offset of filename in global string table.
238c099fe0SZachary Turner   uint8_t ChecksumSize;       // Number of bytes of checksum.
248c099fe0SZachary Turner   uint8_t ChecksumKind;       // FileChecksumKind
258c099fe0SZachary Turner                               // Checksum bytes follow.
268c099fe0SZachary Turner };
278c099fe0SZachary Turner 
28*7e62cd17SZachary Turner Error llvm::VarStreamArrayExtractor<FileChecksumEntry>::
29*7e62cd17SZachary Turner operator()(BinaryStreamRef Stream, uint32_t &Len, FileChecksumEntry &Item) {
308c099fe0SZachary Turner   BinaryStreamReader Reader(Stream);
318c099fe0SZachary Turner 
328c099fe0SZachary Turner   const FileChecksumEntryHeader *Header;
338c099fe0SZachary Turner   if (auto EC = Reader.readObject(Header))
348c099fe0SZachary Turner     return EC;
358c099fe0SZachary Turner 
368c099fe0SZachary Turner   Item.FileNameOffset = Header->FileNameOffset;
378c099fe0SZachary Turner   Item.Kind = static_cast<FileChecksumKind>(Header->ChecksumKind);
388c099fe0SZachary Turner   if (auto EC = Reader.readBytes(Item.Checksum, Header->ChecksumSize))
398c099fe0SZachary Turner     return EC;
408c099fe0SZachary Turner 
418c099fe0SZachary Turner   Len = alignTo(Header->ChecksumSize + sizeof(FileChecksumEntryHeader), 4);
428c099fe0SZachary Turner   return Error::success();
438c099fe0SZachary Turner }
448c099fe0SZachary Turner 
458c099fe0SZachary Turner Error DebugChecksumsSubsectionRef::initialize(BinaryStreamReader Reader) {
468c099fe0SZachary Turner   if (auto EC = Reader.readArray(Checksums, Reader.bytesRemaining()))
478c099fe0SZachary Turner     return EC;
488c099fe0SZachary Turner 
498c099fe0SZachary Turner   return Error::success();
508c099fe0SZachary Turner }
51591312c5SZachary Turner Error DebugChecksumsSubsectionRef::initialize(BinaryStreamRef Section) {
52591312c5SZachary Turner   BinaryStreamReader Reader(Section);
53591312c5SZachary Turner   return initialize(Reader);
54591312c5SZachary Turner }
558c099fe0SZachary Turner 
56591312c5SZachary Turner DebugChecksumsSubsection::DebugChecksumsSubsection(
57591312c5SZachary Turner     DebugStringTableSubsection &Strings)
588c099fe0SZachary Turner     : DebugSubsection(DebugSubsectionKind::FileChecksums), Strings(Strings) {}
598c099fe0SZachary Turner 
608c099fe0SZachary Turner void DebugChecksumsSubsection::addChecksum(StringRef FileName,
618c099fe0SZachary Turner                                            FileChecksumKind Kind,
628c099fe0SZachary Turner                                            ArrayRef<uint8_t> Bytes) {
638c099fe0SZachary Turner   FileChecksumEntry Entry;
648c099fe0SZachary Turner   if (!Bytes.empty()) {
658c099fe0SZachary Turner     uint8_t *Copy = Storage.Allocate<uint8_t>(Bytes.size());
668c099fe0SZachary Turner     ::memcpy(Copy, Bytes.data(), Bytes.size());
678c099fe0SZachary Turner     Entry.Checksum = makeArrayRef(Copy, Bytes.size());
688c099fe0SZachary Turner   }
698c099fe0SZachary Turner 
708c099fe0SZachary Turner   Entry.FileNameOffset = Strings.insert(FileName);
718c099fe0SZachary Turner   Entry.Kind = Kind;
728c099fe0SZachary Turner   Checksums.push_back(Entry);
738c099fe0SZachary Turner 
748c099fe0SZachary Turner   // This maps the offset of this string in the string table to the offset
758c099fe0SZachary Turner   // of this checksum entry in the checksum buffer.
768c099fe0SZachary Turner   OffsetMap[Entry.FileNameOffset] = SerializedSize;
778c099fe0SZachary Turner   assert(SerializedSize % 4 == 0);
788c099fe0SZachary Turner 
798c099fe0SZachary Turner   uint32_t Len = alignTo(sizeof(FileChecksumEntryHeader) + Bytes.size(), 4);
808c099fe0SZachary Turner   SerializedSize += Len;
818c099fe0SZachary Turner }
828c099fe0SZachary Turner 
83591312c5SZachary Turner uint32_t DebugChecksumsSubsection::calculateSerializedSize() const {
848c099fe0SZachary Turner   return SerializedSize;
858c099fe0SZachary Turner }
868c099fe0SZachary Turner 
87591312c5SZachary Turner Error DebugChecksumsSubsection::commit(BinaryStreamWriter &Writer) const {
888c099fe0SZachary Turner   for (const auto &FC : Checksums) {
898c099fe0SZachary Turner     FileChecksumEntryHeader Header;
908c099fe0SZachary Turner     Header.ChecksumKind = uint8_t(FC.Kind);
918c099fe0SZachary Turner     Header.ChecksumSize = FC.Checksum.size();
928c099fe0SZachary Turner     Header.FileNameOffset = FC.FileNameOffset;
938c099fe0SZachary Turner     if (auto EC = Writer.writeObject(Header))
948c099fe0SZachary Turner       return EC;
958c099fe0SZachary Turner     if (auto EC = Writer.writeArray(makeArrayRef(FC.Checksum)))
968c099fe0SZachary Turner       return EC;
978c099fe0SZachary Turner     if (auto EC = Writer.padToAlignment(4))
988c099fe0SZachary Turner       return EC;
998c099fe0SZachary Turner   }
1008c099fe0SZachary Turner   return Error::success();
1018c099fe0SZachary Turner }
1028c099fe0SZachary Turner 
1038c099fe0SZachary Turner uint32_t DebugChecksumsSubsection::mapChecksumOffset(StringRef FileName) const {
1048c099fe0SZachary Turner   uint32_t Offset = Strings.getStringId(FileName);
1058c099fe0SZachary Turner   auto Iter = OffsetMap.find(Offset);
1068c099fe0SZachary Turner   assert(Iter != OffsetMap.end());
1078c099fe0SZachary Turner   return Iter->second;
1088c099fe0SZachary Turner }
109