blob: bf9dd7c8686276ca2b7d1455129df3e958c28ede [file] [log] [blame]
Eugene Zelenko8456b162017-06-29 00:05:44 +00001//===- DebugCrossImpSubsection.cpp ----------------------------------------===//
Zachary Turner349c18f2017-06-05 21:40:33 +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
10#include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h"
Eugene Zelenko8456b162017-06-29 00:05:44 +000011#include "llvm/ADT/ArrayRef.h"
Zachary Turner349c18f2017-06-05 21:40:33 +000012#include "llvm/DebugInfo/CodeView/CodeViewError.h"
13#include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
Eugene Zelenko8456b162017-06-29 00:05:44 +000014#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>
Zachary Turner349c18f2017-06-05 21:40:33 +000022
23using namespace llvm;
24using namespace llvm::codeview;
25
Zachary Turner7e62cd12017-06-09 17:54:36 +000026Error VarStreamArrayExtractor<CrossModuleImportItem>::
27operator()(BinaryStreamRef Stream, uint32_t &Len,
28 codeview::CrossModuleImportItem &Item) {
Zachary Turner349c18f2017-06-05 21:40:33 +000029 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}
Zachary Turner349c18f2017-06-05 21:40:33 +000044
45Error DebugCrossModuleImportsSubsectionRef::initialize(
46 BinaryStreamReader Reader) {
47 return Reader.readArray(References, Reader.bytesRemaining());
48}
49
50Error DebugCrossModuleImportsSubsectionRef::initialize(BinaryStreamRef Stream) {
51 BinaryStreamReader Reader(Stream);
52 return initialize(Reader);
53}
54
55void 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
64uint32_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
73Error 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
Mandeep Singh Grangfe1d28e2018-04-01 16:18:49 +000082 llvm::sort(Ids.begin(), Ids.end(), [this](const T &L1, const T &L2) {
Zachary Turner71d36ad2018-03-22 17:37:28 +000083 return Strings.getIdForString(L1->getKey()) <
84 Strings.getIdForString(L2->getKey());
Zachary Turner349c18f2017-06-05 21:40:33 +000085 });
86
87 for (const auto &Item : Ids) {
88 CrossModuleImport Imp;
Zachary Turner71d36ad2018-03-22 17:37:28 +000089 Imp.ModuleNameOffset = Strings.getIdForString(Item->getKey());
Zachary Turner349c18f2017-06-05 21:40:33 +000090 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}