blob: 0f155a95d60703bb456c838f6cf55f1c79cc64e5 [file] [log] [blame]
Eugene Zelenko8456b162017-06-29 00:05:44 +00001//===- DebugChecksumsSubsection.cpp ---------------------------------------===//
Zachary Turnerc37cb0c2017-04-27 16:12:16 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Zachary Turner8c099fe2017-05-30 16:36:15 +000010#include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
Eugene Zelenko8456b162017-06-29 00:05:44 +000011#include "llvm/ADT/ArrayRef.h"
12#include "llvm/DebugInfo/CodeView/CodeView.h"
Zachary Turner591312c2017-05-30 17:13:33 +000013#include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
Zachary Turnerc37cb0c2017-04-27 16:12:16 +000014#include "llvm/Support/BinaryStreamReader.h"
Eugene Zelenko8456b162017-06-29 00:05:44 +000015#include "llvm/Support/BinaryStreamWriter.h"
16#include "llvm/Support/Endian.h"
17#include "llvm/Support/Error.h"
18#include "llvm/Support/MathExtras.h"
19#include <cassert>
20#include <cstdint>
21#include <cstring>
Zachary Turnerc37cb0c2017-04-27 16:12:16 +000022
23using namespace llvm;
24using namespace llvm::codeview;
25
26struct FileChecksumEntryHeader {
27 using ulittle32_t = support::ulittle32_t;
28
29 ulittle32_t FileNameOffset; // Byte offset of filename in global string table.
30 uint8_t ChecksumSize; // Number of bytes of checksum.
31 uint8_t ChecksumKind; // FileChecksumKind
32 // Checksum bytes follow.
33};
34
Eugene Zelenko8456b162017-06-29 00:05:44 +000035Error VarStreamArrayExtractor<FileChecksumEntry>::
Zachary Turner7e62cd12017-06-09 17:54:36 +000036operator()(BinaryStreamRef Stream, uint32_t &Len, FileChecksumEntry &Item) {
Zachary Turnerc37cb0c2017-04-27 16:12:16 +000037 BinaryStreamReader Reader(Stream);
38
39 const FileChecksumEntryHeader *Header;
40 if (auto EC = Reader.readObject(Header))
41 return EC;
42
43 Item.FileNameOffset = Header->FileNameOffset;
44 Item.Kind = static_cast<FileChecksumKind>(Header->ChecksumKind);
45 if (auto EC = Reader.readBytes(Item.Checksum, Header->ChecksumSize))
46 return EC;
47
48 Len = alignTo(Header->ChecksumSize + sizeof(FileChecksumEntryHeader), 4);
49 return Error::success();
50}
51
Zachary Turner8c099fe2017-05-30 16:36:15 +000052Error DebugChecksumsSubsectionRef::initialize(BinaryStreamReader Reader) {
Zachary Turnerc37cb0c2017-04-27 16:12:16 +000053 if (auto EC = Reader.readArray(Checksums, Reader.bytesRemaining()))
54 return EC;
55
56 return Error::success();
57}
Eugene Zelenko8456b162017-06-29 00:05:44 +000058
Zachary Turner591312c2017-05-30 17:13:33 +000059Error DebugChecksumsSubsectionRef::initialize(BinaryStreamRef Section) {
60 BinaryStreamReader Reader(Section);
61 return initialize(Reader);
62}
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000063
Zachary Turner591312c2017-05-30 17:13:33 +000064DebugChecksumsSubsection::DebugChecksumsSubsection(
65 DebugStringTableSubsection &Strings)
Zachary Turner8c099fe2017-05-30 16:36:15 +000066 : DebugSubsection(DebugSubsectionKind::FileChecksums), Strings(Strings) {}
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000067
Zachary Turner8c099fe2017-05-30 16:36:15 +000068void DebugChecksumsSubsection::addChecksum(StringRef FileName,
69 FileChecksumKind Kind,
70 ArrayRef<uint8_t> Bytes) {
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000071 FileChecksumEntry Entry;
72 if (!Bytes.empty()) {
73 uint8_t *Copy = Storage.Allocate<uint8_t>(Bytes.size());
74 ::memcpy(Copy, Bytes.data(), Bytes.size());
75 Entry.Checksum = makeArrayRef(Copy, Bytes.size());
76 }
Zachary Turnercf468d82017-05-03 17:11:40 +000077
78 Entry.FileNameOffset = Strings.insert(FileName);
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000079 Entry.Kind = Kind;
80 Checksums.push_back(Entry);
81
82 // This maps the offset of this string in the string table to the offset
83 // of this checksum entry in the checksum buffer.
Zachary Turnercf468d82017-05-03 17:11:40 +000084 OffsetMap[Entry.FileNameOffset] = SerializedSize;
Zachary Turneredef1452017-05-02 16:56:09 +000085 assert(SerializedSize % 4 == 0);
86
87 uint32_t Len = alignTo(sizeof(FileChecksumEntryHeader) + Bytes.size(), 4);
88 SerializedSize += Len;
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000089}
90
Zachary Turner591312c2017-05-30 17:13:33 +000091uint32_t DebugChecksumsSubsection::calculateSerializedSize() const {
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000092 return SerializedSize;
93}
94
Zachary Turner591312c2017-05-30 17:13:33 +000095Error DebugChecksumsSubsection::commit(BinaryStreamWriter &Writer) const {
Zachary Turner8a2ebfb2017-05-01 23:27:42 +000096 for (const auto &FC : Checksums) {
97 FileChecksumEntryHeader Header;
98 Header.ChecksumKind = uint8_t(FC.Kind);
99 Header.ChecksumSize = FC.Checksum.size();
100 Header.FileNameOffset = FC.FileNameOffset;
101 if (auto EC = Writer.writeObject(Header))
102 return EC;
103 if (auto EC = Writer.writeArray(makeArrayRef(FC.Checksum)))
104 return EC;
Zachary Turneredef1452017-05-02 16:56:09 +0000105 if (auto EC = Writer.padToAlignment(4))
106 return EC;
Zachary Turner8a2ebfb2017-05-01 23:27:42 +0000107 }
108 return Error::success();
109}
110
Zachary Turner8c099fe2017-05-30 16:36:15 +0000111uint32_t DebugChecksumsSubsection::mapChecksumOffset(StringRef FileName) const {
Zachary Turner71d36ad2018-03-22 17:37:28 +0000112 uint32_t Offset = Strings.getIdForString(FileName);
Zachary Turnercf468d82017-05-03 17:11:40 +0000113 auto Iter = OffsetMap.find(Offset);
Zachary Turner8a2ebfb2017-05-01 23:27:42 +0000114 assert(Iter != OffsetMap.end());
115 return Iter->second;
116}