18456b16eSEugene Zelenko //===- DebugChecksumsSubsection.cpp ---------------------------------------===//
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"
118456b16eSEugene Zelenko #include "llvm/ADT/ArrayRef.h"
128456b16eSEugene Zelenko #include "llvm/DebugInfo/CodeView/CodeView.h"
13591312c5SZachary Turner #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
148c099fe0SZachary Turner #include "llvm/Support/BinaryStreamReader.h"
158456b16eSEugene Zelenko #include "llvm/Support/BinaryStreamWriter.h"
168456b16eSEugene Zelenko #include "llvm/Support/Endian.h"
178456b16eSEugene Zelenko #include "llvm/Support/Error.h"
188456b16eSEugene Zelenko #include "llvm/Support/MathExtras.h"
198456b16eSEugene Zelenko #include <cassert>
208456b16eSEugene Zelenko #include <cstdint>
218456b16eSEugene Zelenko #include <cstring>
228c099fe0SZachary Turner 
238c099fe0SZachary Turner using namespace llvm;
248c099fe0SZachary Turner using namespace llvm::codeview;
258c099fe0SZachary Turner 
268c099fe0SZachary Turner struct FileChecksumEntryHeader {
278c099fe0SZachary Turner   using ulittle32_t = support::ulittle32_t;
288c099fe0SZachary Turner 
298c099fe0SZachary Turner   ulittle32_t FileNameOffset; // Byte offset of filename in global string table.
308c099fe0SZachary Turner   uint8_t ChecksumSize;       // Number of bytes of checksum.
318c099fe0SZachary Turner   uint8_t ChecksumKind;       // FileChecksumKind
328c099fe0SZachary Turner                               // Checksum bytes follow.
338c099fe0SZachary Turner };
348c099fe0SZachary Turner 
358456b16eSEugene Zelenko Error VarStreamArrayExtractor<FileChecksumEntry>::
367e62cd17SZachary Turner operator()(BinaryStreamRef Stream, uint32_t &Len, FileChecksumEntry &Item) {
378c099fe0SZachary Turner   BinaryStreamReader Reader(Stream);
388c099fe0SZachary Turner 
398c099fe0SZachary Turner   const FileChecksumEntryHeader *Header;
408c099fe0SZachary Turner   if (auto EC = Reader.readObject(Header))
418c099fe0SZachary Turner     return EC;
428c099fe0SZachary Turner 
438c099fe0SZachary Turner   Item.FileNameOffset = Header->FileNameOffset;
448c099fe0SZachary Turner   Item.Kind = static_cast<FileChecksumKind>(Header->ChecksumKind);
458c099fe0SZachary Turner   if (auto EC = Reader.readBytes(Item.Checksum, Header->ChecksumSize))
468c099fe0SZachary Turner     return EC;
478c099fe0SZachary Turner 
488c099fe0SZachary Turner   Len = alignTo(Header->ChecksumSize + sizeof(FileChecksumEntryHeader), 4);
498c099fe0SZachary Turner   return Error::success();
508c099fe0SZachary Turner }
518c099fe0SZachary Turner 
528c099fe0SZachary Turner Error DebugChecksumsSubsectionRef::initialize(BinaryStreamReader Reader) {
538c099fe0SZachary Turner   if (auto EC = Reader.readArray(Checksums, Reader.bytesRemaining()))
548c099fe0SZachary Turner     return EC;
558c099fe0SZachary Turner 
568c099fe0SZachary Turner   return Error::success();
578c099fe0SZachary Turner }
588456b16eSEugene Zelenko 
59591312c5SZachary Turner Error DebugChecksumsSubsectionRef::initialize(BinaryStreamRef Section) {
60591312c5SZachary Turner   BinaryStreamReader Reader(Section);
61591312c5SZachary Turner   return initialize(Reader);
62591312c5SZachary Turner }
638c099fe0SZachary Turner 
64591312c5SZachary Turner DebugChecksumsSubsection::DebugChecksumsSubsection(
65591312c5SZachary Turner     DebugStringTableSubsection &Strings)
668c099fe0SZachary Turner     : DebugSubsection(DebugSubsectionKind::FileChecksums), Strings(Strings) {}
678c099fe0SZachary Turner 
688c099fe0SZachary Turner void DebugChecksumsSubsection::addChecksum(StringRef FileName,
698c099fe0SZachary Turner                                            FileChecksumKind Kind,
708c099fe0SZachary Turner                                            ArrayRef<uint8_t> Bytes) {
718c099fe0SZachary Turner   FileChecksumEntry Entry;
728c099fe0SZachary Turner   if (!Bytes.empty()) {
738c099fe0SZachary Turner     uint8_t *Copy = Storage.Allocate<uint8_t>(Bytes.size());
748c099fe0SZachary Turner     ::memcpy(Copy, Bytes.data(), Bytes.size());
758c099fe0SZachary Turner     Entry.Checksum = makeArrayRef(Copy, Bytes.size());
768c099fe0SZachary Turner   }
778c099fe0SZachary Turner 
788c099fe0SZachary Turner   Entry.FileNameOffset = Strings.insert(FileName);
798c099fe0SZachary Turner   Entry.Kind = Kind;
808c099fe0SZachary Turner   Checksums.push_back(Entry);
818c099fe0SZachary Turner 
828c099fe0SZachary Turner   // This maps the offset of this string in the string table to the offset
838c099fe0SZachary Turner   // of this checksum entry in the checksum buffer.
848c099fe0SZachary Turner   OffsetMap[Entry.FileNameOffset] = SerializedSize;
858c099fe0SZachary Turner   assert(SerializedSize % 4 == 0);
868c099fe0SZachary Turner 
878c099fe0SZachary Turner   uint32_t Len = alignTo(sizeof(FileChecksumEntryHeader) + Bytes.size(), 4);
888c099fe0SZachary Turner   SerializedSize += Len;
898c099fe0SZachary Turner }
908c099fe0SZachary Turner 
91591312c5SZachary Turner uint32_t DebugChecksumsSubsection::calculateSerializedSize() const {
928c099fe0SZachary Turner   return SerializedSize;
938c099fe0SZachary Turner }
948c099fe0SZachary Turner 
95591312c5SZachary Turner Error DebugChecksumsSubsection::commit(BinaryStreamWriter &Writer) const {
968c099fe0SZachary Turner   for (const auto &FC : Checksums) {
978c099fe0SZachary Turner     FileChecksumEntryHeader Header;
988c099fe0SZachary Turner     Header.ChecksumKind = uint8_t(FC.Kind);
998c099fe0SZachary Turner     Header.ChecksumSize = FC.Checksum.size();
1008c099fe0SZachary Turner     Header.FileNameOffset = FC.FileNameOffset;
1018c099fe0SZachary Turner     if (auto EC = Writer.writeObject(Header))
1028c099fe0SZachary Turner       return EC;
1038c099fe0SZachary Turner     if (auto EC = Writer.writeArray(makeArrayRef(FC.Checksum)))
1048c099fe0SZachary Turner       return EC;
1058c099fe0SZachary Turner     if (auto EC = Writer.padToAlignment(4))
1068c099fe0SZachary Turner       return EC;
1078c099fe0SZachary Turner   }
1088c099fe0SZachary Turner   return Error::success();
1098c099fe0SZachary Turner }
1108c099fe0SZachary Turner 
1118c099fe0SZachary Turner uint32_t DebugChecksumsSubsection::mapChecksumOffset(StringRef FileName) const {
112*71d36ad9SZachary Turner   uint32_t Offset = Strings.getIdForString(FileName);
1138c099fe0SZachary Turner   auto Iter = OffsetMap.find(Offset);
1148c099fe0SZachary Turner   assert(Iter != OffsetMap.end());
1158c099fe0SZachary Turner   return Iter->second;
1168c099fe0SZachary Turner }
117