1 //===- DebugCrossImpSubsection.cpp ----------------------------------------===//
2 //
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 
10 #include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h"
11 #include "llvm/ADT/ArrayRef.h"
12 #include "llvm/DebugInfo/CodeView/CodeViewError.h"
13 #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.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"
18 #include <algorithm>
19 #include <cstdint>
20 #include <utility>
21 #include <vector>
22 
23 using namespace llvm;
24 using namespace llvm::codeview;
25 
26 Error VarStreamArrayExtractor<CrossModuleImportItem>::
operator ()(BinaryStreamRef Stream,uint32_t & Len,codeview::CrossModuleImportItem & Item)27 operator()(BinaryStreamRef Stream, uint32_t &Len,
28            codeview::CrossModuleImportItem &Item) {
29   BinaryStreamReader Reader(Stream);
30   if (Reader.bytesRemaining() < sizeof(CrossModuleImport))
31     return make_error<CodeViewError>(
32         cv_error_code::insufficient_buffer,
33         "Not enough bytes for a Cross Module Import Header!");
34   if (auto EC = Reader.readObject(Item.Header))
35     return EC;
36   if (Reader.bytesRemaining() < Item.Header->Count * sizeof(uint32_t))
37     return make_error<CodeViewError>(
38         cv_error_code::insufficient_buffer,
39         "Not enough to read specified number of Cross Module References!");
40   if (auto EC = Reader.readArray(Item.Imports, Item.Header->Count))
41     return EC;
42   return Error::success();
43 }
44 
initialize(BinaryStreamReader Reader)45 Error DebugCrossModuleImportsSubsectionRef::initialize(
46     BinaryStreamReader Reader) {
47   return Reader.readArray(References, Reader.bytesRemaining());
48 }
49 
initialize(BinaryStreamRef Stream)50 Error DebugCrossModuleImportsSubsectionRef::initialize(BinaryStreamRef Stream) {
51   BinaryStreamReader Reader(Stream);
52   return initialize(Reader);
53 }
54 
addImport(StringRef Module,uint32_t ImportId)55 void DebugCrossModuleImportsSubsection::addImport(StringRef Module,
56                                                   uint32_t ImportId) {
57   Strings.insert(Module);
58   std::vector<support::ulittle32_t> Targets = {support::ulittle32_t(ImportId)};
59   auto Result = Mappings.insert(std::make_pair(Module, Targets));
60   if (!Result.second)
61     Result.first->getValue().push_back(Targets[0]);
62 }
63 
calculateSerializedSize() const64 uint32_t DebugCrossModuleImportsSubsection::calculateSerializedSize() const {
65   uint32_t Size = 0;
66   for (const auto &Item : Mappings) {
67     Size += sizeof(CrossModuleImport);
68     Size += sizeof(support::ulittle32_t) * Item.second.size();
69   }
70   return Size;
71 }
72 
commit(BinaryStreamWriter & Writer) const73 Error DebugCrossModuleImportsSubsection::commit(
74     BinaryStreamWriter &Writer) const {
75   using T = decltype(&*Mappings.begin());
76   std::vector<T> Ids;
77   Ids.reserve(Mappings.size());
78 
79   for (const auto &M : Mappings)
80     Ids.push_back(&M);
81 
82   llvm::sort(Ids.begin(), Ids.end(), [this](const T &L1, const T &L2) {
83     return Strings.getIdForString(L1->getKey()) <
84            Strings.getIdForString(L2->getKey());
85   });
86 
87   for (const auto &Item : Ids) {
88     CrossModuleImport Imp;
89     Imp.ModuleNameOffset = Strings.getIdForString(Item->getKey());
90     Imp.Count = Item->getValue().size();
91     if (auto EC = Writer.writeObject(Imp))
92       return EC;
93     if (auto EC = Writer.writeArray(makeArrayRef(Item->getValue())))
94       return EC;
95   }
96   return Error::success();
97 }
98