xref: /freebsd-src/contrib/llvm-project/llvm/lib/Target/WebAssembly/WebAssemblyDebugFixup.cpp (revision 5ffd83dbcc34f10e07f6d3e968ae6365869615f4)
1*5ffd83dbSDimitry Andric //===-- WebAssemblyDebugFixup.cpp - Debug Fixup ------------------===//
2*5ffd83dbSDimitry Andric //
3*5ffd83dbSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*5ffd83dbSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5*5ffd83dbSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*5ffd83dbSDimitry Andric //
7*5ffd83dbSDimitry Andric //===----------------------------------------------------------------------===//
8*5ffd83dbSDimitry Andric ///
9*5ffd83dbSDimitry Andric /// \file
10*5ffd83dbSDimitry Andric /// Several prior passes may "stackify" registers, here we ensure any references
11*5ffd83dbSDimitry Andric /// in such registers in debug_value instructions become stack relative also.
12*5ffd83dbSDimitry Andric /// This is done in a separate pass such that not all previous passes need to
13*5ffd83dbSDimitry Andric /// track stack depth when values get stackified.
14*5ffd83dbSDimitry Andric ///
15*5ffd83dbSDimitry Andric //===----------------------------------------------------------------------===//
16*5ffd83dbSDimitry Andric 
17*5ffd83dbSDimitry Andric #include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
18*5ffd83dbSDimitry Andric #include "WebAssembly.h"
19*5ffd83dbSDimitry Andric #include "WebAssemblyMachineFunctionInfo.h"
20*5ffd83dbSDimitry Andric #include "WebAssemblySubtarget.h"
21*5ffd83dbSDimitry Andric #include "WebAssemblyUtilities.h"
22*5ffd83dbSDimitry Andric #include "llvm/ADT/SCCIterator.h"
23*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineFrameInfo.h"
24*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineFunction.h"
25*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h"
26*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineLoopInfo.h"
27*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineRegisterInfo.h"
28*5ffd83dbSDimitry Andric #include "llvm/CodeGen/Passes.h"
29*5ffd83dbSDimitry Andric #include "llvm/Support/Debug.h"
30*5ffd83dbSDimitry Andric #include "llvm/Support/raw_ostream.h"
31*5ffd83dbSDimitry Andric using namespace llvm;
32*5ffd83dbSDimitry Andric 
33*5ffd83dbSDimitry Andric #define DEBUG_TYPE "wasm-debug-fixup"
34*5ffd83dbSDimitry Andric 
35*5ffd83dbSDimitry Andric namespace {
36*5ffd83dbSDimitry Andric class WebAssemblyDebugFixup final : public MachineFunctionPass {
37*5ffd83dbSDimitry Andric   StringRef getPassName() const override { return "WebAssembly Debug Fixup"; }
38*5ffd83dbSDimitry Andric 
39*5ffd83dbSDimitry Andric   void getAnalysisUsage(AnalysisUsage &AU) const override {
40*5ffd83dbSDimitry Andric     AU.setPreservesCFG();
41*5ffd83dbSDimitry Andric     MachineFunctionPass::getAnalysisUsage(AU);
42*5ffd83dbSDimitry Andric   }
43*5ffd83dbSDimitry Andric 
44*5ffd83dbSDimitry Andric   bool runOnMachineFunction(MachineFunction &MF) override;
45*5ffd83dbSDimitry Andric 
46*5ffd83dbSDimitry Andric public:
47*5ffd83dbSDimitry Andric   static char ID; // Pass identification, replacement for typeid
48*5ffd83dbSDimitry Andric   WebAssemblyDebugFixup() : MachineFunctionPass(ID) {}
49*5ffd83dbSDimitry Andric };
50*5ffd83dbSDimitry Andric } // end anonymous namespace
51*5ffd83dbSDimitry Andric 
52*5ffd83dbSDimitry Andric char WebAssemblyDebugFixup::ID = 0;
53*5ffd83dbSDimitry Andric INITIALIZE_PASS(
54*5ffd83dbSDimitry Andric     WebAssemblyDebugFixup, DEBUG_TYPE,
55*5ffd83dbSDimitry Andric     "Ensures debug_value's that have been stackified become stack relative",
56*5ffd83dbSDimitry Andric     false, false)
57*5ffd83dbSDimitry Andric 
58*5ffd83dbSDimitry Andric FunctionPass *llvm::createWebAssemblyDebugFixup() {
59*5ffd83dbSDimitry Andric   return new WebAssemblyDebugFixup();
60*5ffd83dbSDimitry Andric }
61*5ffd83dbSDimitry Andric 
62*5ffd83dbSDimitry Andric bool WebAssemblyDebugFixup::runOnMachineFunction(MachineFunction &MF) {
63*5ffd83dbSDimitry Andric   LLVM_DEBUG(dbgs() << "********** Debug Fixup **********\n"
64*5ffd83dbSDimitry Andric                        "********** Function: "
65*5ffd83dbSDimitry Andric                     << MF.getName() << '\n');
66*5ffd83dbSDimitry Andric 
67*5ffd83dbSDimitry Andric   WebAssemblyFunctionInfo &MFI = *MF.getInfo<WebAssemblyFunctionInfo>();
68*5ffd83dbSDimitry Andric   const auto *TII = MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
69*5ffd83dbSDimitry Andric 
70*5ffd83dbSDimitry Andric   struct StackElem {
71*5ffd83dbSDimitry Andric     unsigned Reg;
72*5ffd83dbSDimitry Andric     MachineInstr *DebugValue;
73*5ffd83dbSDimitry Andric   };
74*5ffd83dbSDimitry Andric   std::vector<StackElem> Stack;
75*5ffd83dbSDimitry Andric   for (MachineBasicBlock &MBB : MF) {
76*5ffd83dbSDimitry Andric     // We may insert into this list.
77*5ffd83dbSDimitry Andric     for (auto MII = MBB.begin(); MII != MBB.end(); ++MII) {
78*5ffd83dbSDimitry Andric       MachineInstr &MI = *MII;
79*5ffd83dbSDimitry Andric       if (MI.isDebugValue()) {
80*5ffd83dbSDimitry Andric         auto &MO = MI.getOperand(0);
81*5ffd83dbSDimitry Andric         // Also check if not a $noreg: likely a DBG_VALUE we just inserted.
82*5ffd83dbSDimitry Andric         if (MO.isReg() && MO.getReg().isValid() &&
83*5ffd83dbSDimitry Andric             MFI.isVRegStackified(MO.getReg())) {
84*5ffd83dbSDimitry Andric           // Found a DBG_VALUE with a stackified register we will
85*5ffd83dbSDimitry Andric           // change into a stack operand.
86*5ffd83dbSDimitry Andric           // Search for register rather than assume it is on top (which it
87*5ffd83dbSDimitry Andric           // typically is if it appears right after the def), since
88*5ffd83dbSDimitry Andric           // DBG_VALUE's may shift under some circumstances.
89*5ffd83dbSDimitry Andric           for (auto &Elem : reverse(Stack)) {
90*5ffd83dbSDimitry Andric             if (MO.getReg() == Elem.Reg) {
91*5ffd83dbSDimitry Andric               auto Depth = static_cast<unsigned>(&Elem - &Stack[0]);
92*5ffd83dbSDimitry Andric               LLVM_DEBUG(dbgs() << "Debug Value VReg " << MO.getReg()
93*5ffd83dbSDimitry Andric                                 << " -> Stack Relative " << Depth << "\n");
94*5ffd83dbSDimitry Andric               MO.ChangeToTargetIndex(WebAssembly::TI_OPERAND_STACK, Depth);
95*5ffd83dbSDimitry Andric               // Save the DBG_VALUE instruction that defined this stackified
96*5ffd83dbSDimitry Andric               // variable since later we need it to construct another one on
97*5ffd83dbSDimitry Andric               // pop.
98*5ffd83dbSDimitry Andric               Elem.DebugValue = &MI;
99*5ffd83dbSDimitry Andric               break;
100*5ffd83dbSDimitry Andric             }
101*5ffd83dbSDimitry Andric           }
102*5ffd83dbSDimitry Andric           // If the Reg was not found, we have a DBG_VALUE outside of its
103*5ffd83dbSDimitry Andric           // def-use range, and we leave it unmodified as reg, which means
104*5ffd83dbSDimitry Andric           // it will be culled later.
105*5ffd83dbSDimitry Andric         }
106*5ffd83dbSDimitry Andric       } else {
107*5ffd83dbSDimitry Andric         // Track stack depth.
108*5ffd83dbSDimitry Andric         for (MachineOperand &MO : reverse(MI.explicit_uses())) {
109*5ffd83dbSDimitry Andric           if (MO.isReg() && MFI.isVRegStackified(MO.getReg())) {
110*5ffd83dbSDimitry Andric             auto Prev = Stack.back();
111*5ffd83dbSDimitry Andric             Stack.pop_back();
112*5ffd83dbSDimitry Andric             assert(Prev.Reg == MO.getReg() &&
113*5ffd83dbSDimitry Andric                    "WebAssemblyDebugFixup: Pop: Register not matched!");
114*5ffd83dbSDimitry Andric             if (Prev.DebugValue) {
115*5ffd83dbSDimitry Andric               // This stackified reg is a variable that started life at
116*5ffd83dbSDimitry Andric               // Prev.DebugValue, so now that we're popping it we must insert
117*5ffd83dbSDimitry Andric               // a $noreg DBG_VALUE for the variable to end it, right after
118*5ffd83dbSDimitry Andric               // the current instruction.
119*5ffd83dbSDimitry Andric               BuildMI(*Prev.DebugValue->getParent(), std::next(MII),
120*5ffd83dbSDimitry Andric                       Prev.DebugValue->getDebugLoc(), TII->get(WebAssembly::DBG_VALUE), false,
121*5ffd83dbSDimitry Andric                       Register(), Prev.DebugValue->getOperand(2).getMetadata(),
122*5ffd83dbSDimitry Andric                       Prev.DebugValue->getOperand(3).getMetadata());
123*5ffd83dbSDimitry Andric             }
124*5ffd83dbSDimitry Andric           }
125*5ffd83dbSDimitry Andric         }
126*5ffd83dbSDimitry Andric         for (MachineOperand &MO : MI.defs()) {
127*5ffd83dbSDimitry Andric           if (MO.isReg() && MFI.isVRegStackified(MO.getReg())) {
128*5ffd83dbSDimitry Andric             Stack.push_back({MO.getReg(), nullptr});
129*5ffd83dbSDimitry Andric           }
130*5ffd83dbSDimitry Andric         }
131*5ffd83dbSDimitry Andric       }
132*5ffd83dbSDimitry Andric     }
133*5ffd83dbSDimitry Andric     assert(Stack.empty() &&
134*5ffd83dbSDimitry Andric            "WebAssemblyDebugFixup: Stack not empty at end of basic block!");
135*5ffd83dbSDimitry Andric   }
136*5ffd83dbSDimitry Andric 
137*5ffd83dbSDimitry Andric   return true;
138*5ffd83dbSDimitry Andric }
139