1 //===- DebugInlineeLinesSubsection.cpp ------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h"
10 #include "llvm/ADT/ArrayRef.h"
11 #include "llvm/DebugInfo/CodeView/CodeView.h"
12 #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
13 #include "llvm/DebugInfo/CodeView/RecordSerialization.h"
14 #include "llvm/Support/BinaryStreamReader.h"
15 #include "llvm/Support/BinaryStreamWriter.h"
16 #include "llvm/Support/Endian.h"
17 #include "llvm/Support/Error.h"
22 using namespace llvm::codeview
;
24 Error VarStreamArrayExtractor
<InlineeSourceLine
>::
25 operator()(BinaryStreamRef Stream
, uint32_t &Len
, InlineeSourceLine
&Item
) {
26 BinaryStreamReader
Reader(Stream
);
28 if (auto EC
= Reader
.readObject(Item
.Header
))
32 uint32_t ExtraFileCount
;
33 if (auto EC
= Reader
.readInteger(ExtraFileCount
))
35 if (auto EC
= Reader
.readArray(Item
.ExtraFiles
, ExtraFileCount
))
39 Len
= Reader
.getOffset();
40 return Error::success();
43 DebugInlineeLinesSubsectionRef::DebugInlineeLinesSubsectionRef()
44 : DebugSubsectionRef(DebugSubsectionKind::InlineeLines
) {}
46 Error
DebugInlineeLinesSubsectionRef::initialize(BinaryStreamReader Reader
) {
47 if (auto EC
= Reader
.readEnum(Signature
))
50 Lines
.getExtractor().HasExtraFiles
= hasExtraFiles();
51 if (auto EC
= Reader
.readArray(Lines
, Reader
.bytesRemaining()))
54 assert(Reader
.bytesRemaining() == 0);
55 return Error::success();
58 bool DebugInlineeLinesSubsectionRef::hasExtraFiles() const {
59 return Signature
== InlineeLinesSignature::ExtraFiles
;
62 DebugInlineeLinesSubsection::DebugInlineeLinesSubsection(
63 DebugChecksumsSubsection
&Checksums
, bool HasExtraFiles
)
64 : DebugSubsection(DebugSubsectionKind::InlineeLines
), Checksums(Checksums
),
65 HasExtraFiles(HasExtraFiles
) {}
67 uint32_t DebugInlineeLinesSubsection::calculateSerializedSize() const {
68 // 4 bytes for the signature
69 uint32_t Size
= sizeof(InlineeLinesSignature
);
71 // one header for each entry.
72 Size
+= Entries
.size() * sizeof(InlineeSourceLineHeader
);
74 // If extra files are enabled, one count for each entry.
75 Size
+= Entries
.size() * sizeof(uint32_t);
77 // And one file id for each file.
78 Size
+= ExtraFileCount
* sizeof(uint32_t);
80 assert(Size
% 4 == 0);
84 Error
DebugInlineeLinesSubsection::commit(BinaryStreamWriter
&Writer
) const {
85 InlineeLinesSignature Sig
= InlineeLinesSignature::Normal
;
87 Sig
= InlineeLinesSignature::ExtraFiles
;
89 if (auto EC
= Writer
.writeEnum(Sig
))
92 for (const auto &E
: Entries
) {
93 if (auto EC
= Writer
.writeObject(E
.Header
))
99 if (auto EC
= Writer
.writeInteger
<uint32_t>(E
.ExtraFiles
.size()))
101 if (auto EC
= Writer
.writeArray(ArrayRef(E
.ExtraFiles
)))
105 return Error::success();
108 void DebugInlineeLinesSubsection::addExtraFile(StringRef FileName
) {
109 uint32_t Offset
= Checksums
.mapChecksumOffset(FileName
);
111 auto &Entry
= Entries
.back();
112 Entry
.ExtraFiles
.push_back(ulittle32_t(Offset
));
116 void DebugInlineeLinesSubsection::addInlineSite(TypeIndex FuncId
,
118 uint32_t SourceLine
) {
119 uint32_t Offset
= Checksums
.mapChecksumOffset(FileName
);
121 Entries
.emplace_back();
122 auto &Entry
= Entries
.back();
123 Entry
.Header
.FileID
= Offset
;
124 Entry
.Header
.SourceLineNum
= SourceLine
;
125 Entry
.Header
.Inlinee
= FuncId
;