1*9880d681SAndroid Build Coastguard Worker //===---------------------------- FaultMaps.cpp ---------------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker
10*9880d681SAndroid Build Coastguard Worker #include "llvm/CodeGen/FaultMaps.h"
11*9880d681SAndroid Build Coastguard Worker
12*9880d681SAndroid Build Coastguard Worker #include "llvm/CodeGen/AsmPrinter.h"
13*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCContext.h"
14*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCExpr.h"
15*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCObjectFileInfo.h"
16*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCStreamer.h"
17*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Debug.h"
18*9880d681SAndroid Build Coastguard Worker
19*9880d681SAndroid Build Coastguard Worker using namespace llvm;
20*9880d681SAndroid Build Coastguard Worker
21*9880d681SAndroid Build Coastguard Worker #define DEBUG_TYPE "faultmaps"
22*9880d681SAndroid Build Coastguard Worker
23*9880d681SAndroid Build Coastguard Worker static const int FaultMapVersion = 1;
24*9880d681SAndroid Build Coastguard Worker const char *FaultMaps::WFMP = "Fault Maps: ";
25*9880d681SAndroid Build Coastguard Worker
FaultMaps(AsmPrinter & AP)26*9880d681SAndroid Build Coastguard Worker FaultMaps::FaultMaps(AsmPrinter &AP) : AP(AP) {}
27*9880d681SAndroid Build Coastguard Worker
recordFaultingOp(FaultKind FaultTy,const MCSymbol * HandlerLabel)28*9880d681SAndroid Build Coastguard Worker void FaultMaps::recordFaultingOp(FaultKind FaultTy,
29*9880d681SAndroid Build Coastguard Worker const MCSymbol *HandlerLabel) {
30*9880d681SAndroid Build Coastguard Worker MCContext &OutContext = AP.OutStreamer->getContext();
31*9880d681SAndroid Build Coastguard Worker MCSymbol *FaultingLabel = OutContext.createTempSymbol();
32*9880d681SAndroid Build Coastguard Worker
33*9880d681SAndroid Build Coastguard Worker AP.OutStreamer->EmitLabel(FaultingLabel);
34*9880d681SAndroid Build Coastguard Worker
35*9880d681SAndroid Build Coastguard Worker const MCExpr *FaultingOffset = MCBinaryExpr::createSub(
36*9880d681SAndroid Build Coastguard Worker MCSymbolRefExpr::create(FaultingLabel, OutContext),
37*9880d681SAndroid Build Coastguard Worker MCSymbolRefExpr::create(AP.CurrentFnSymForSize, OutContext), OutContext);
38*9880d681SAndroid Build Coastguard Worker
39*9880d681SAndroid Build Coastguard Worker const MCExpr *HandlerOffset = MCBinaryExpr::createSub(
40*9880d681SAndroid Build Coastguard Worker MCSymbolRefExpr::create(HandlerLabel, OutContext),
41*9880d681SAndroid Build Coastguard Worker MCSymbolRefExpr::create(AP.CurrentFnSymForSize, OutContext), OutContext);
42*9880d681SAndroid Build Coastguard Worker
43*9880d681SAndroid Build Coastguard Worker FunctionInfos[AP.CurrentFnSym].emplace_back(FaultTy, FaultingOffset,
44*9880d681SAndroid Build Coastguard Worker HandlerOffset);
45*9880d681SAndroid Build Coastguard Worker }
46*9880d681SAndroid Build Coastguard Worker
serializeToFaultMapSection()47*9880d681SAndroid Build Coastguard Worker void FaultMaps::serializeToFaultMapSection() {
48*9880d681SAndroid Build Coastguard Worker if (FunctionInfos.empty())
49*9880d681SAndroid Build Coastguard Worker return;
50*9880d681SAndroid Build Coastguard Worker
51*9880d681SAndroid Build Coastguard Worker MCContext &OutContext = AP.OutStreamer->getContext();
52*9880d681SAndroid Build Coastguard Worker MCStreamer &OS = *AP.OutStreamer;
53*9880d681SAndroid Build Coastguard Worker
54*9880d681SAndroid Build Coastguard Worker // Create the section.
55*9880d681SAndroid Build Coastguard Worker MCSection *FaultMapSection =
56*9880d681SAndroid Build Coastguard Worker OutContext.getObjectFileInfo()->getFaultMapSection();
57*9880d681SAndroid Build Coastguard Worker OS.SwitchSection(FaultMapSection);
58*9880d681SAndroid Build Coastguard Worker
59*9880d681SAndroid Build Coastguard Worker // Emit a dummy symbol to force section inclusion.
60*9880d681SAndroid Build Coastguard Worker OS.EmitLabel(OutContext.getOrCreateSymbol(Twine("__LLVM_FaultMaps")));
61*9880d681SAndroid Build Coastguard Worker
62*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "********** Fault Map Output **********\n");
63*9880d681SAndroid Build Coastguard Worker
64*9880d681SAndroid Build Coastguard Worker // Header
65*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(FaultMapVersion, 1); // Version.
66*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(0, 1); // Reserved.
67*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(0, 2); // Reserved.
68*9880d681SAndroid Build Coastguard Worker
69*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << "#functions = " << FunctionInfos.size() << "\n");
70*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(FunctionInfos.size(), 4);
71*9880d681SAndroid Build Coastguard Worker
72*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << "functions:\n");
73*9880d681SAndroid Build Coastguard Worker
74*9880d681SAndroid Build Coastguard Worker for (const auto &FFI : FunctionInfos)
75*9880d681SAndroid Build Coastguard Worker emitFunctionInfo(FFI.first, FFI.second);
76*9880d681SAndroid Build Coastguard Worker }
77*9880d681SAndroid Build Coastguard Worker
emitFunctionInfo(const MCSymbol * FnLabel,const FunctionFaultInfos & FFI)78*9880d681SAndroid Build Coastguard Worker void FaultMaps::emitFunctionInfo(const MCSymbol *FnLabel,
79*9880d681SAndroid Build Coastguard Worker const FunctionFaultInfos &FFI) {
80*9880d681SAndroid Build Coastguard Worker MCStreamer &OS = *AP.OutStreamer;
81*9880d681SAndroid Build Coastguard Worker
82*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << " function addr: " << *FnLabel << "\n");
83*9880d681SAndroid Build Coastguard Worker OS.EmitSymbolValue(FnLabel, 8);
84*9880d681SAndroid Build Coastguard Worker
85*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << " #faulting PCs: " << FFI.size() << "\n");
86*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(FFI.size(), 4);
87*9880d681SAndroid Build Coastguard Worker
88*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(0, 4); // Reserved
89*9880d681SAndroid Build Coastguard Worker
90*9880d681SAndroid Build Coastguard Worker for (auto &Fault : FFI) {
91*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << " fault type: "
92*9880d681SAndroid Build Coastguard Worker << faultTypeToString(Fault.Kind) << "\n");
93*9880d681SAndroid Build Coastguard Worker OS.EmitIntValue(Fault.Kind, 4);
94*9880d681SAndroid Build Coastguard Worker
95*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << " faulting PC offset: "
96*9880d681SAndroid Build Coastguard Worker << *Fault.FaultingOffsetExpr << "\n");
97*9880d681SAndroid Build Coastguard Worker OS.EmitValue(Fault.FaultingOffsetExpr, 4);
98*9880d681SAndroid Build Coastguard Worker
99*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << WFMP << " fault handler PC offset: "
100*9880d681SAndroid Build Coastguard Worker << *Fault.HandlerOffsetExpr << "\n");
101*9880d681SAndroid Build Coastguard Worker OS.EmitValue(Fault.HandlerOffsetExpr, 4);
102*9880d681SAndroid Build Coastguard Worker }
103*9880d681SAndroid Build Coastguard Worker }
104*9880d681SAndroid Build Coastguard Worker
105*9880d681SAndroid Build Coastguard Worker
faultTypeToString(FaultMaps::FaultKind FT)106*9880d681SAndroid Build Coastguard Worker const char *FaultMaps::faultTypeToString(FaultMaps::FaultKind FT) {
107*9880d681SAndroid Build Coastguard Worker switch (FT) {
108*9880d681SAndroid Build Coastguard Worker default:
109*9880d681SAndroid Build Coastguard Worker llvm_unreachable("unhandled fault type!");
110*9880d681SAndroid Build Coastguard Worker
111*9880d681SAndroid Build Coastguard Worker case FaultMaps::FaultingLoad:
112*9880d681SAndroid Build Coastguard Worker return "FaultingLoad";
113*9880d681SAndroid Build Coastguard Worker }
114*9880d681SAndroid Build Coastguard Worker }
115*9880d681SAndroid Build Coastguard Worker
116*9880d681SAndroid Build Coastguard Worker raw_ostream &llvm::
operator <<(raw_ostream & OS,const FaultMapParser::FunctionFaultInfoAccessor & FFI)117*9880d681SAndroid Build Coastguard Worker operator<<(raw_ostream &OS,
118*9880d681SAndroid Build Coastguard Worker const FaultMapParser::FunctionFaultInfoAccessor &FFI) {
119*9880d681SAndroid Build Coastguard Worker OS << "Fault kind: "
120*9880d681SAndroid Build Coastguard Worker << FaultMaps::faultTypeToString((FaultMaps::FaultKind)FFI.getFaultKind())
121*9880d681SAndroid Build Coastguard Worker << ", faulting PC offset: " << FFI.getFaultingPCOffset()
122*9880d681SAndroid Build Coastguard Worker << ", handling PC offset: " << FFI.getHandlerPCOffset();
123*9880d681SAndroid Build Coastguard Worker return OS;
124*9880d681SAndroid Build Coastguard Worker }
125*9880d681SAndroid Build Coastguard Worker
126*9880d681SAndroid Build Coastguard Worker raw_ostream &llvm::
operator <<(raw_ostream & OS,const FaultMapParser::FunctionInfoAccessor & FI)127*9880d681SAndroid Build Coastguard Worker operator<<(raw_ostream &OS, const FaultMapParser::FunctionInfoAccessor &FI) {
128*9880d681SAndroid Build Coastguard Worker OS << "FunctionAddress: " << format_hex(FI.getFunctionAddr(), 8)
129*9880d681SAndroid Build Coastguard Worker << ", NumFaultingPCs: " << FI.getNumFaultingPCs() << "\n";
130*9880d681SAndroid Build Coastguard Worker for (unsigned i = 0, e = FI.getNumFaultingPCs(); i != e; ++i)
131*9880d681SAndroid Build Coastguard Worker OS << FI.getFunctionFaultInfoAt(i) << "\n";
132*9880d681SAndroid Build Coastguard Worker return OS;
133*9880d681SAndroid Build Coastguard Worker }
134*9880d681SAndroid Build Coastguard Worker
operator <<(raw_ostream & OS,const FaultMapParser & FMP)135*9880d681SAndroid Build Coastguard Worker raw_ostream &llvm::operator<<(raw_ostream &OS, const FaultMapParser &FMP) {
136*9880d681SAndroid Build Coastguard Worker OS << "Version: " << format_hex(FMP.getFaultMapVersion(), 2) << "\n";
137*9880d681SAndroid Build Coastguard Worker OS << "NumFunctions: " << FMP.getNumFunctions() << "\n";
138*9880d681SAndroid Build Coastguard Worker
139*9880d681SAndroid Build Coastguard Worker if (FMP.getNumFunctions() == 0)
140*9880d681SAndroid Build Coastguard Worker return OS;
141*9880d681SAndroid Build Coastguard Worker
142*9880d681SAndroid Build Coastguard Worker FaultMapParser::FunctionInfoAccessor FI;
143*9880d681SAndroid Build Coastguard Worker
144*9880d681SAndroid Build Coastguard Worker for (unsigned i = 0, e = FMP.getNumFunctions(); i != e; ++i) {
145*9880d681SAndroid Build Coastguard Worker FI = (i == 0) ? FMP.getFirstFunctionInfo() : FI.getNextFunctionInfo();
146*9880d681SAndroid Build Coastguard Worker OS << FI;
147*9880d681SAndroid Build Coastguard Worker }
148*9880d681SAndroid Build Coastguard Worker
149*9880d681SAndroid Build Coastguard Worker return OS;
150*9880d681SAndroid Build Coastguard Worker }
151