xref: /freebsd/contrib/llvm-project/llvm/lib/Target/WebAssembly/WebAssemblyDebugValueManager.cpp (revision 2e3507c25e42292b45a5482e116d278f5515d04d)
1 //===-- WebAssemblyDebugValueManager.cpp - WebAssembly DebugValue Manager -===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 ///
9 /// \file
10 /// This file implements the manager for MachineInstr DebugValues.
11 ///
12 //===----------------------------------------------------------------------===//
13 
14 #include "WebAssemblyDebugValueManager.h"
15 #include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
16 #include "WebAssembly.h"
17 #include "WebAssemblyMachineFunctionInfo.h"
18 #include "llvm/CodeGen/MachineInstr.h"
19 #include "llvm/IR/DebugInfoMetadata.h"
20 
21 using namespace llvm;
22 
23 WebAssemblyDebugValueManager::WebAssemblyDebugValueManager(MachineInstr *Def)
24     : Def(Def) {
25   // This code differs from MachineInstr::collectDebugValues in that it scans
26   // the whole BB, not just contiguous DBG_VALUEs, until another definition to
27   // the same register is encountered.
28   if (!Def->getOperand(0).isReg())
29     return;
30   CurrentReg = Def->getOperand(0).getReg();
31 
32   for (MachineBasicBlock::iterator MI = std::next(Def->getIterator()),
33                                    ME = Def->getParent()->end();
34        MI != ME; ++MI) {
35     // If another definition appears, stop
36     if (MI->definesRegister(CurrentReg))
37       break;
38     if (MI->isDebugValue() && MI->hasDebugOperandForReg(CurrentReg))
39       DbgValues.push_back(&*MI);
40   }
41 }
42 
43 // Returns true if both A and B are the same CONST_I32/I64/F32/F64 instructions.
44 // Doesn't include CONST_V128.
45 static bool isSameScalarConst(const MachineInstr *A, const MachineInstr *B) {
46   if (A->getOpcode() != B->getOpcode() ||
47       !WebAssembly::isScalarConst(A->getOpcode()) ||
48       !WebAssembly::isScalarConst(B->getOpcode()))
49     return false;
50   const MachineOperand &OpA = A->getOperand(1), &OpB = B->getOperand(1);
51   if ((OpA.isImm() && OpB.isImm() && OpA.getImm() == OpB.getImm()) ||
52       (OpA.isFPImm() && OpB.isFPImm() && OpA.getFPImm() == OpB.getFPImm()) ||
53       (OpA.isGlobal() && OpB.isGlobal() && OpA.getGlobal() == OpB.getGlobal()))
54     return true;
55   return false;
56 }
57 
58 SmallVector<MachineInstr *, 1>
59 WebAssemblyDebugValueManager::getSinkableDebugValues(
60     MachineInstr *Insert) const {
61   if (DbgValues.empty())
62     return {};
63   // DBG_VALUEs between Def and Insert
64   SmallVector<MachineInstr *, 8> DbgValuesInBetween;
65 
66   if (Def->getParent() == Insert->getParent()) {
67     // When Def and Insert are within the same BB, check if Insert comes after
68     // Def, because we only support sinking.
69     bool DefFirst = false;
70     for (MachineBasicBlock::iterator MI = std::next(Def->getIterator()),
71                                      ME = Def->getParent()->end();
72          MI != ME; ++MI) {
73       if (&*MI == Insert) {
74         DefFirst = true;
75         break;
76       }
77       if (MI->isDebugValue())
78         DbgValuesInBetween.push_back(&*MI);
79     }
80     if (!DefFirst) // Not a sink
81       return {};
82 
83   } else { // Def and Insert are in different BBs
84     // If Def and Insert are in different BBs, we only handle a simple case in
85     // which Insert's BB is a successor of Def's BB.
86     if (!Def->getParent()->isSuccessor(Insert->getParent()))
87       return {};
88 
89     // Gather DBG_VALUEs between 'Def~Def BB's end' and
90     // 'Insert BB's begin~Insert'
91     for (MachineBasicBlock::iterator MI = std::next(Def->getIterator()),
92                                      ME = Def->getParent()->end();
93          MI != ME; ++MI) {
94       if (MI->isDebugValue())
95         DbgValuesInBetween.push_back(&*MI);
96     }
97     for (MachineBasicBlock::iterator MI = Insert->getParent()->begin(),
98                                      ME = Insert->getIterator();
99          MI != ME; ++MI) {
100       if (MI->isDebugValue())
101         DbgValuesInBetween.push_back(&*MI);
102     }
103   }
104 
105   // Gather DebugVariables that are seen between Def and Insert, excluding our
106   // own DBG_VALUEs in DbgValues.
107   SmallDenseMap<DebugVariable, SmallVector<MachineInstr *, 2>>
108       SeenDbgVarToDbgValues;
109   for (auto *DV : DbgValuesInBetween) {
110     if (!llvm::is_contained(DbgValues, DV)) {
111       DebugVariable Var(DV->getDebugVariable(), DV->getDebugExpression(),
112                         DV->getDebugLoc()->getInlinedAt());
113       SeenDbgVarToDbgValues[Var].push_back(DV);
114     }
115   }
116 
117   // Gather sinkable DBG_VALUEs. We should not sink a DBG_VALUE if there is
118   // another DBG_VALUE between Def and Insert referring to the same
119   // DebugVariable. For example,
120   //   %0 = someinst
121   //   DBG_VALUE %0, !"a", !DIExpression() // Should not sink with %0
122   //   %1 = anotherinst
123   //   DBG_VALUE %1, !"a", !DIExpression()
124   // Where if %0 were to sink, the DBG_VAUE should not sink with it, as that
125   // would re-order assignments.
126   SmallVector<MachineInstr *, 1> SinkableDbgValues;
127   MachineRegisterInfo &MRI = Def->getParent()->getParent()->getRegInfo();
128   for (auto *DV : DbgValues) {
129     DebugVariable Var(DV->getDebugVariable(), DV->getDebugExpression(),
130                       DV->getDebugLoc()->getInlinedAt());
131     auto It = SeenDbgVarToDbgValues.find(Var);
132     if (It == SeenDbgVarToDbgValues.end()) {
133       SinkableDbgValues.push_back(DV);
134       continue;
135     }
136     if (!WebAssembly::isScalarConst(Def->getOpcode()))
137       continue;
138     auto &OverlappingDbgValues = It->second;
139     bool Sinkable = true;
140     for (auto *OverlappingDV : OverlappingDbgValues) {
141       MachineOperand &DbgOp = OverlappingDV->getDebugOperand(0);
142       if (!DbgOp.isReg()) {
143         Sinkable = false;
144         break;
145       }
146       Register OtherReg = DbgOp.getReg();
147       MachineInstr *OtherDef = MRI.getUniqueVRegDef(OtherReg);
148       // We have an exception to allow encoutering other DBG_VALUEs with the
149       // smae DebugVariables, only when they are referring to the same scalar
150       // CONST instruction. For example,
151       //   %0 = CONST_I32 1
152       //   DBG_VALUE %0, !"a", !DIExpression() // Can sink with %0
153       //   %1 = CONST_I32 1
154       //   DBG_VALUE %1, !"a", !DIExpression()
155       // When %0 were to be sunk/cloneed, the DBG_VALUE can be sunk/cloned with
156       // it because even though the second DBG_VALUE refers to the same
157       // DebugVariable, its value in effect is the same CONST instruction.
158       //
159       // This is to allow a case that can happen with RegStackify's
160       // "rematerializeCheapDef". For example, we have this program with two
161       // BBs:
162       // bb0:
163       //   %0 = CONST_I32 1
164       //   DBG_VALUE %0, !"a", ...
165       //   ...
166       //   INST0 ..., $0 ...
167       //  bb1:
168       //   INST1 ..., $0 ...
169       //   INST2 ..., $0 ...
170       //
171       // We process bb0 first. Because %0 is used multiple times, %0 is cloned
172       // before INST0:
173       // bb0:
174       //   %0 = CONST_I32 1
175       //   DBG_VALUE %0, !"a", ...
176       //   ...
177       //   %1 = CONST_I32 1
178       //   DBG_VALUE %1, !"a", ...
179       //   INST0 ..., $1 ...
180       //
181       // And when we process bb1, we clone %0 and its DBG_VALUE again:
182       // bb0:
183       //   %0 = CONST_I32 1
184       //   DBG_VALUE %0, !"a", ...
185       //   ...
186       //   %1 = CONST_I32 1
187       //   DBG_VALUE %1, !"a", ...
188       //   INST0 ..., $1 ...
189       //  bb1:
190       //   %2 = CONST_I32 1
191       //   DBG_VALUE %2, !"a", ... // !!!
192       //   INST1 ..., $2 ...
193       //   %3 = CONST_I32 1
194       //   DBG_VALUE %3, !"a", ... // !!!
195       //   INST2 ..., $3 ...
196       //
197       // But (without this exception) the cloned DBG_VALUEs marked with !!! are
198       // not possible to be cloned, because there is a previously cloned
199       // 'DBG_VALUE %1, !"a"' at the end of bb0 referring to the same
200       // DebugVariable "a". But in this case they are OK to be cloned, because
201       // the interfering DBG_VALUE is pointing to the same 'CONST_I32 1',
202       // because it was cloned from the same instruction.
203       if (!OtherDef || !isSameScalarConst(Def, OtherDef)) {
204         Sinkable = false;
205         break;
206       }
207     }
208     if (Sinkable)
209       SinkableDbgValues.push_back(DV);
210   }
211   return SinkableDbgValues;
212 }
213 
214 // Returns true if the insertion point is the same as the current place.
215 // Following DBG_VALUEs for 'Def' are ignored.
216 bool WebAssemblyDebugValueManager::isInsertSamePlace(
217     MachineInstr *Insert) const {
218   if (Def->getParent() != Insert->getParent())
219     return false;
220   for (MachineBasicBlock::iterator MI = std::next(Def->getIterator()),
221                                    ME = Insert;
222        MI != ME; ++MI) {
223     if (!llvm::is_contained(DbgValues, MI)) {
224       return false;
225     }
226   }
227   return true;
228 }
229 
230 // Returns true if any instruction in MBB has the same debug location as DL.
231 // Also returns true if DL is an empty location.
232 static bool hasSameDebugLoc(const MachineBasicBlock *MBB, DebugLoc DL) {
233   for (const auto &MI : *MBB)
234     if (MI.getDebugLoc() == DL)
235       return true;
236   return false;
237 }
238 
239 // Sink 'Def', and also sink its eligible DBG_VALUEs to the place before
240 // 'Insert'. Convert the original DBG_VALUEs into undefs.
241 //
242 // For DBG_VALUEs to sink properly, if 'Def' and 'Insert' are within the same
243 // BB, 'Insert' should be below 'Def'; if they are in different BBs, 'Insert'
244 // should be in one of 'Def's BBs successors. Def will be sunk regardless of the
245 // location.
246 //
247 // This DebugValueManager's new Def and DbgValues will be updated to the newly
248 // sinked Def + DBG_VALUEs.
249 void WebAssemblyDebugValueManager::sink(MachineInstr *Insert) {
250   // In case Def is requested to be sunk to
251   // the same place, we don't need to do anything. If we actually do the sink,
252   // it will create unnecessary undef DBG_VALUEs. For example, if the original
253   // code is:
254   //   %0 = someinst           // Def
255   //   DBG_VALUE %0, ...
256   //   %1 = anotherinst        // Insert
257   //
258   // If we actually sink %0 and the following DBG_VALUE and setting the original
259   // DBG_VALUE undef, the result will be:
260   //   DBG_VALUE %noreg, ...   // Unnecessary!
261   //   %0 = someinst           // Def
262   //   DBG_VALUE %0, ...
263   //   %1 = anotherinst        // Insert
264   if (isInsertSamePlace(Insert))
265     return;
266 
267   MachineBasicBlock *MBB = Insert->getParent();
268   MachineFunction *MF = MBB->getParent();
269 
270   // Get the list of sinkable DBG_VALUEs. This should be done before sinking
271   // Def, because we need to examine instructions between Def and Insert.
272   SmallVector<MachineInstr *, 1> SinkableDbgValues =
273       getSinkableDebugValues(Insert);
274 
275   // Sink Def first.
276   //
277   // When moving to a different BB, we preserve the debug loc only if the
278   // destination BB contains the same location. See
279   // https://llvm.org/docs/HowToUpdateDebugInfo.html#when-to-preserve-an-instruction-location.
280   if (Def->getParent() != MBB && !hasSameDebugLoc(MBB, Def->getDebugLoc()))
281       Def->setDebugLoc(DebugLoc());
282   MBB->splice(Insert, Def->getParent(), Def);
283 
284   if (DbgValues.empty())
285     return;
286 
287   // Clone sinkable DBG_VALUEs and insert them.
288   SmallVector<MachineInstr *, 1> NewDbgValues;
289   for (MachineInstr *DV : SinkableDbgValues) {
290     MachineInstr *Clone = MF->CloneMachineInstr(DV);
291     MBB->insert(Insert, Clone);
292     NewDbgValues.push_back(Clone);
293   }
294 
295   // When sinking a Def and its DBG_VALUEs, we shouldn't just remove the
296   // original DBG_VALUE instructions; we should set them to undef not to create
297   // an impossible combination of variable assignments in the original program.
298   // For example, this is the original program in order:
299   //   %0 = CONST_I32 0
300   //   DBG_VALUE %0, !"a", !DIExpression()  // a = 0, b = ?
301   //   %1 = CONST_I32 1
302   //   DBG_VALUE %1, !"b", !DIExpression()  // a = 0, b = 1
303   //   %2 = CONST_I32 2
304   //   DBG_VALUE %2, !"a", !DIExpression()  // a = 2, b = 1
305   //   %3 = CONST_I32 3
306   //   DBG_VALUE %3, !"b", !DIExpression()  // a = 2, b = 3
307   //
308   // If %2 were to sink below %3, if we just sink DBG_VALUE %1 with it, the
309   // debug info will show the variable "b" is updated to 2, creating the
310   // variable assignment combination of (a = 0, b = 3), which is not possible in
311   // the original program:
312   //   %0 = CONST_I32 0
313   //   DBG_VALUE %0, !"a", !DIExpression()  // a = 0, b = ?
314   //   %1 = CONST_I32 1
315   //   DBG_VALUE %1, !"b", !DIExpression()  // a = 0, b = 1
316   //   %3 = CONST_I32 3
317   //   DBG_VALUE %3, !"b", !DIExpression()  // a = 0, b = 3 (Incorrect!)
318   //   %2 = CONST_I32 2
319   //   DBG_VALUE %2, !"a", !DIExpression()  // a = 2, b = 3
320   //
321   // To fix this,we leave an undef DBG_VALUE in its original place, so that the
322   // result will be
323   //   %0 = CONST_I32 0
324   //   DBG_VALUE %0, !"a", !DIExpression()      // a = 0, b = ?
325   //   %1 = CONST_I32 1
326   //   DBG_VALUE %1, !"b", !DIExpression()      // a = 0, b = 1
327   //   DBG_VALUE $noreg, !"a", !DIExpression()  // a = ?, b = 1
328   //   %3 = CONST_I32 3
329   //   DBG_VALUE %3, !"b", !DIExpression()      // a = ?, b = 3
330   //   %2 = CONST_I32 2
331   //   DBG_VALUE %2, !"a", !DIExpression()      // a = 2, b = 3
332   // Now in the middle "a" will be shown as "optimized out", but it wouldn't
333   // show the impossible combination of (a = 0, b = 3).
334   for (MachineInstr *DV : DbgValues)
335     DV->setDebugValueUndef();
336 
337   DbgValues.swap(NewDbgValues);
338 }
339 
340 // Clone 'Def', and also clone its eligible DBG_VALUEs to the place before
341 // 'Insert'.
342 //
343 // For DBG_VALUEs to be cloned properly, if 'Def' and 'Insert' are within the
344 // same BB, 'Insert' should be below 'Def'; if they are in different BBs,
345 // 'Insert' should be in one of 'Def's BBs successors. Def will be cloned
346 // regardless of the location.
347 //
348 // If NewReg is not $noreg, the newly cloned DBG_VALUEs will have the new
349 // register as its operand.
350 void WebAssemblyDebugValueManager::cloneSink(MachineInstr *Insert,
351                                              Register NewReg,
352                                              bool CloneDef) const {
353   MachineBasicBlock *MBB = Insert->getParent();
354   MachineFunction *MF = MBB->getParent();
355 
356   SmallVector<MachineInstr *> SinkableDbgValues =
357       getSinkableDebugValues(Insert);
358 
359   // Clone Def first.
360   if (CloneDef) {
361     MachineInstr *Clone = MF->CloneMachineInstr(Def);
362     // When cloning to a different BB, we preserve the debug loc only if the
363     // destination BB contains the same location. See
364     // https://llvm.org/docs/HowToUpdateDebugInfo.html#when-to-preserve-an-instruction-location.
365     if (Def->getParent() != MBB && !hasSameDebugLoc(MBB, Def->getDebugLoc()))
366       Clone->setDebugLoc(DebugLoc());
367     if (NewReg != CurrentReg && NewReg.isValid())
368       Clone->getOperand(0).setReg(NewReg);
369     MBB->insert(Insert, Clone);
370   }
371 
372   if (DbgValues.empty())
373     return;
374 
375   // Clone sinkable DBG_VALUEs and insert them.
376   SmallVector<MachineInstr *, 1> NewDbgValues;
377   for (MachineInstr *DV : SinkableDbgValues) {
378     MachineInstr *Clone = MF->CloneMachineInstr(DV);
379     MBB->insert(Insert, Clone);
380     NewDbgValues.push_back(Clone);
381   }
382 
383   if (NewReg != CurrentReg && NewReg.isValid())
384     for (auto *DBI : NewDbgValues)
385       for (auto &MO : DBI->getDebugOperandsForReg(CurrentReg))
386         MO.setReg(NewReg);
387 }
388 
389 // Update the register for Def and DBG_VALUEs.
390 void WebAssemblyDebugValueManager::updateReg(Register Reg) {
391   if (Reg != CurrentReg && Reg.isValid()) {
392     for (auto *DBI : DbgValues)
393       for (auto &MO : DBI->getDebugOperandsForReg(CurrentReg))
394         MO.setReg(Reg);
395     CurrentReg = Reg;
396     Def->getOperand(0).setReg(Reg);
397   }
398 }
399 
400 void WebAssemblyDebugValueManager::replaceWithLocal(unsigned LocalId) {
401   for (auto *DBI : DbgValues) {
402     auto IndexType = DBI->isIndirectDebugValue()
403                          ? llvm::WebAssembly::TI_LOCAL_INDIRECT
404                          : llvm::WebAssembly::TI_LOCAL;
405     for (auto &MO : DBI->getDebugOperandsForReg(CurrentReg))
406       MO.ChangeToTargetIndex(IndexType, LocalId);
407   }
408 }
409 
410 // Remove Def, and set its DBG_VALUEs to undef.
411 void WebAssemblyDebugValueManager::removeDef() {
412   Def->removeFromParent();
413   for (MachineInstr *DV : DbgValues)
414     DV->setDebugValueUndef();
415 }
416