xref: /aosp_15_r20/external/llvm/lib/CodeGen/FaultMaps.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
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