20
20
#include " llvm/ADT/DenseMap.h"
21
21
#include " llvm/ADT/PostOrderIterator.h"
22
22
#include " llvm/ADT/SmallPtrSet.h"
23
+ #include " llvm/ADT/SmallSet.h"
23
24
#include " llvm/ADT/SmallVector.h"
24
25
#include " llvm/ADT/SparseBitVector.h"
25
26
#include " llvm/ADT/Statistic.h"
57
58
#include < cstdint>
58
59
#include < functional>
59
60
#include < queue>
61
+ #include < tuple>
60
62
#include < utility>
61
63
#include < vector>
62
64
@@ -107,24 +109,63 @@ class LiveDebugValues : public MachineFunctionPass {
107
109
}
108
110
};
109
111
110
- // / Based on std::pair so it can be used as an index into a DenseMap.
111
- using DebugVariableBase =
112
- std::pair<const DILocalVariable *, const DILocation *>;
113
- // / A potentially inlined instance of a variable.
114
- struct DebugVariable : public DebugVariableBase {
115
- DebugVariable (const DILocalVariable *Var, const DILocation *InlinedAt)
116
- : DebugVariableBase(Var, InlinedAt) {}
117
-
118
- const DILocalVariable *getVar () const { return this ->first ; }
119
- const DILocation *getInlinedAt () const { return this ->second ; }
120
-
121
- bool operator <(const DebugVariable &DV) const {
122
- if (getVar () == DV.getVar ())
123
- return getInlinedAt () < DV.getInlinedAt ();
124
- return getVar () < DV.getVar ();
112
+ using FragmentInfo = DIExpression::FragmentInfo;
113
+ using OptFragmentInfo = Optional<DIExpression::FragmentInfo>;
114
+
115
+ // / Storage for identifying a potentially inlined instance of a variable,
116
+ // / or a fragment thereof.
117
+ class DebugVariable {
118
+ const DILocalVariable *Variable;
119
+ OptFragmentInfo Fragment;
120
+ const DILocation *InlinedAt;
121
+
122
+ // / Fragment that will overlap all other fragments. Used as default when
123
+ // / caller demands a fragment.
124
+ static const FragmentInfo DefaultFragment;
125
+
126
+ public:
127
+ DebugVariable (const DILocalVariable *Var, OptFragmentInfo &&FragmentInfo,
128
+ const DILocation *InlinedAt)
129
+ : Variable(Var), Fragment(FragmentInfo), InlinedAt(InlinedAt) {}
130
+
131
+ DebugVariable (const DILocalVariable *Var, OptFragmentInfo &FragmentInfo,
132
+ const DILocation *InlinedAt)
133
+ : Variable(Var), Fragment(FragmentInfo), InlinedAt(InlinedAt) {}
134
+
135
+ DebugVariable (const DILocalVariable *Var, const DIExpression *DIExpr,
136
+ const DILocation *InlinedAt)
137
+ : DebugVariable(Var, DIExpr->getFragmentInfo (), InlinedAt) {}
138
+
139
+ DebugVariable (const MachineInstr &MI)
140
+ : DebugVariable(MI.getDebugVariable(),
141
+ MI.getDebugExpression()->getFragmentInfo(),
142
+ MI.getDebugLoc()->getInlinedAt()) {}
143
+
144
+ const DILocalVariable *getVar () const { return Variable; }
145
+ const OptFragmentInfo &getFragment () const { return Fragment; }
146
+ const DILocation *getInlinedAt () const { return InlinedAt; }
147
+
148
+ const FragmentInfo getFragmentDefault () const {
149
+ return Fragment.getValueOr (DefaultFragment);
150
+ }
151
+
152
+ static bool isFragmentDefault (FragmentInfo &F) {
153
+ return F == DefaultFragment;
154
+ }
155
+
156
+ bool operator ==(const DebugVariable &Other) const {
157
+ return std::tie (Variable, Fragment, InlinedAt) ==
158
+ std::tie (Other.Variable , Other.Fragment , Other.InlinedAt );
159
+ }
160
+
161
+ bool operator <(const DebugVariable &Other) const {
162
+ return std::tie (Variable, Fragment, InlinedAt) <
163
+ std::tie (Other.Variable , Other.Fragment , Other.InlinedAt );
125
164
}
126
165
};
127
166
167
+ friend struct llvm ::DenseMapInfo<DebugVariable>;
168
+
128
169
// / A pair of debug variable and value location.
129
170
struct VarLoc {
130
171
// The location at which a spilled variable resides. It consists of a
@@ -159,8 +200,7 @@ class LiveDebugValues : public MachineFunctionPass {
159
200
} Loc;
160
201
161
202
VarLoc (const MachineInstr &MI, LexicalScopes &LS)
162
- : Var(MI.getDebugVariable(), MI.getDebugLoc()->getInlinedAt ()), MI(MI),
163
- UVS(MI.getDebugLoc(), LS) {
203
+ : Var(MI), MI(MI), UVS(MI.getDebugLoc(), LS) {
164
204
static_assert ((sizeof (Loc) == sizeof (uint64_t )),
165
205
" hash does not cover all members of Loc" );
166
206
assert (MI.isDebugValue () && " not a DBG_VALUE" );
@@ -183,8 +223,7 @@ class LiveDebugValues : public MachineFunctionPass {
183
223
// / The constructor for spill locations.
184
224
VarLoc (const MachineInstr &MI, unsigned SpillBase, int SpillOffset,
185
225
LexicalScopes &LS)
186
- : Var(MI.getDebugVariable(), MI.getDebugLoc()->getInlinedAt()), MI(MI),
187
- UVS(MI.getDebugLoc(), LS) {
226
+ : Var(MI), MI(MI), UVS(MI.getDebugLoc(), LS) {
188
227
assert (MI.isDebugValue () && " not a DBG_VALUE" );
189
228
assert (MI.getNumOperands () == 4 && " malformed DBG_VALUE" );
190
229
Kind = SpillLocKind;
@@ -232,26 +271,35 @@ class LiveDebugValues : public MachineFunctionPass {
232
271
};
233
272
using TransferMap = SmallVector<TransferDebugPair, 4 >;
234
273
274
+ // Types for recording sets of variable fragments that overlap. For a given
275
+ // local variable, we record all other fragments of that variable that could
276
+ // overlap it, to reduce search time.
277
+ using FragmentOfVar =
278
+ std::pair<const DILocalVariable *, DIExpression::FragmentInfo>;
279
+ using OverlapMap =
280
+ DenseMap<FragmentOfVar, SmallVector<DIExpression::FragmentInfo, 1 >>;
281
+
282
+ // Helper while building OverlapMap, a map of all fragments seen for a given
283
+ // DILocalVariable.
284
+ using VarToFragments =
285
+ DenseMap<const DILocalVariable *, SmallSet<FragmentInfo, 4 >>;
286
+
235
287
// / This holds the working set of currently open ranges. For fast
236
288
// / access, this is done both as a set of VarLocIDs, and a map of
237
289
// / DebugVariable to recent VarLocID. Note that a DBG_VALUE ends all
238
290
// / previous open ranges for the same variable.
239
291
class OpenRangesSet {
240
292
VarLocSet VarLocs;
241
- SmallDenseMap<DebugVariableBase, unsigned , 8 > Vars;
293
+ SmallDenseMap<DebugVariable, unsigned , 8 > Vars;
294
+ OverlapMap &OverlappingFragments;
242
295
243
296
public:
297
+ OpenRangesSet (OverlapMap &_OLapMap) : OverlappingFragments(_OLapMap) {}
298
+
244
299
const VarLocSet &getVarLocs () const { return VarLocs; }
245
300
246
301
// / Terminate all open ranges for Var by removing it from the set.
247
- void erase (DebugVariable Var) {
248
- auto It = Vars.find (Var);
249
- if (It != Vars.end ()) {
250
- unsigned ID = It->second ;
251
- VarLocs.reset (ID);
252
- Vars.erase (It);
253
- }
254
- }
302
+ void erase (DebugVariable Var);
255
303
256
304
// / Terminate all open ranges listed in \c KillSet by removing
257
305
// / them from the set.
@@ -262,7 +310,7 @@ class LiveDebugValues : public MachineFunctionPass {
262
310
}
263
311
264
312
// / Insert a new range into the set.
265
- void insert (unsigned VarLocID, DebugVariableBase Var) {
313
+ void insert (unsigned VarLocID, DebugVariable Var) {
266
314
VarLocs.set (VarLocID);
267
315
Vars.insert ({Var, VarLocID});
268
316
}
@@ -310,6 +358,9 @@ class LiveDebugValues : public MachineFunctionPass {
310
358
VarLocInMBB &OutLocs, VarLocMap &VarLocIDs,
311
359
TransferMap &Transfers, bool transferChanges);
312
360
361
+ void accumulateFragmentMap (MachineInstr &MI, VarToFragments &SeenFragments,
362
+ OverlapMap &OLapMap);
363
+
313
364
bool join (MachineBasicBlock &MBB, VarLocInMBB &OutLocs, VarLocInMBB &InLocs,
314
365
const VarLocMap &VarLocIDs,
315
366
SmallPtrSet<const MachineBasicBlock *, 16 > &Visited,
@@ -343,10 +394,46 @@ class LiveDebugValues : public MachineFunctionPass {
343
394
344
395
} // end anonymous namespace
345
396
397
+ namespace llvm {
398
+
399
+ template <> struct DenseMapInfo <LiveDebugValues::DebugVariable> {
400
+ using DV = LiveDebugValues::DebugVariable;
401
+ using OptFragmentInfo = LiveDebugValues::OptFragmentInfo;
402
+ using FragmentInfo = LiveDebugValues::FragmentInfo;
403
+
404
+ // Empty key: no key should be generated that has no DILocalVariable.
405
+ static inline DV getEmptyKey () {
406
+ return DV (nullptr , OptFragmentInfo (), nullptr );
407
+ }
408
+
409
+ // Difference in tombstone is that the Optional is meaningful
410
+ static inline DV getTombstoneKey () {
411
+ return DV (nullptr , OptFragmentInfo ({0 , 0 }), nullptr );
412
+ }
413
+
414
+ static unsigned getHashValue (const DV &D) {
415
+ unsigned HV = 0 ;
416
+ const OptFragmentInfo &Fragment = D.getFragment ();
417
+ if (Fragment)
418
+ HV = DenseMapInfo<FragmentInfo>::getHashValue (*Fragment);
419
+
420
+ return hash_combine (D.getVar (), HV, D.getInlinedAt ());
421
+ }
422
+
423
+ static bool isEqual (const DV &A, const DV &B) { return A == B; }
424
+ };
425
+
426
+ }; // namespace llvm
427
+
346
428
// ===----------------------------------------------------------------------===//
347
429
// Implementation
348
430
// ===----------------------------------------------------------------------===//
349
431
432
+ const DIExpression::FragmentInfo
433
+ LiveDebugValues::DebugVariable::DefaultFragment = {
434
+ std::numeric_limits<uint64_t >::max (),
435
+ std::numeric_limits<uint64_t >::min ()};
436
+
350
437
char LiveDebugValues::ID = 0 ;
351
438
352
439
char &llvm::LiveDebugValuesID = LiveDebugValues::ID;
@@ -366,6 +453,39 @@ void LiveDebugValues::getAnalysisUsage(AnalysisUsage &AU) const {
366
453
MachineFunctionPass::getAnalysisUsage (AU);
367
454
}
368
455
456
+ // / Erase a variable from the set of open ranges, and additionally erase any
457
+ // / fragments that may overlap it.
458
+ void LiveDebugValues::OpenRangesSet::erase (DebugVariable Var) {
459
+ // Erasure helper.
460
+ auto DoErase = [this ](DebugVariable VarToErase) {
461
+ auto It = Vars.find (VarToErase);
462
+ if (It != Vars.end ()) {
463
+ unsigned ID = It->second ;
464
+ VarLocs.reset (ID);
465
+ Vars.erase (It);
466
+ }
467
+ };
468
+
469
+ // Erase the variable/fragment that ends here.
470
+ DoErase (Var);
471
+
472
+ // Extract the fragment. Interpret an empty fragment as one that covers all
473
+ // possible bits.
474
+ FragmentInfo ThisFragment = Var.getFragmentDefault ();
475
+
476
+ // There may be fragments that overlap the designated fragment. Look them up
477
+ // in the pre-computed overlap map, and erase them too.
478
+ auto MapIt = OverlappingFragments.find ({Var.getVar (), ThisFragment});
479
+ if (MapIt != OverlappingFragments.end ()) {
480
+ for (auto Fragment : MapIt->second ) {
481
+ LiveDebugValues::OptFragmentInfo FragmentHolder;
482
+ if (!DebugVariable::isFragmentDefault (Fragment))
483
+ FragmentHolder = LiveDebugValues::OptFragmentInfo (Fragment);
484
+ DoErase ({Var.getVar (), FragmentHolder, Var.getInlinedAt ()});
485
+ }
486
+ }
487
+ }
488
+
369
489
// ===----------------------------------------------------------------------===//
370
490
// Debug Range Extension Implementation
371
491
// ===----------------------------------------------------------------------===//
@@ -416,13 +536,14 @@ void LiveDebugValues::transferDebugValue(const MachineInstr &MI,
416
536
if (!MI.isDebugValue ())
417
537
return ;
418
538
const DILocalVariable *Var = MI.getDebugVariable ();
539
+ const DIExpression *Expr = MI.getDebugExpression ();
419
540
const DILocation *DebugLoc = MI.getDebugLoc ();
420
541
const DILocation *InlinedAt = DebugLoc->getInlinedAt ();
421
542
assert (Var->isValidLocationForIntrinsic (DebugLoc) &&
422
543
" Expected inlined-at fields to agree" );
423
544
424
545
// End all previous ranges of Var.
425
- DebugVariable V (Var, InlinedAt);
546
+ DebugVariable V (Var, Expr, InlinedAt);
426
547
OpenRanges.erase (V);
427
548
428
549
// Add the VarLoc to OpenRanges from this DBG_VALUE.
@@ -464,8 +585,7 @@ void LiveDebugValues::insertTransferDebugPair(
464
585
unsigned LocId = VarLocIDs.insert (VL);
465
586
466
587
// Close this variable's previous location range.
467
- DebugVariable V (DebugInstr->getDebugVariable (),
468
- DebugInstr->getDebugLoc ()->getInlinedAt ());
588
+ DebugVariable V (*DebugInstr);
469
589
OpenRanges.erase (V);
470
590
471
591
OpenRanges.insert (LocId, VL.Var );
@@ -757,6 +877,66 @@ bool LiveDebugValues::transferTerminatorInst(MachineInstr &MI,
757
877
return Changed;
758
878
}
759
879
880
+ // / Accumulate a mapping between each DILocalVariable fragment and other
881
+ // / fragments of that DILocalVariable which overlap. This reduces work during
882
+ // / the data-flow stage from "Find any overlapping fragments" to "Check if the
883
+ // / known-to-overlap fragments are present".
884
+ // / \param MI A previously unprocessed DEBUG_VALUE instruction to analyze for
885
+ // / fragment usage.
886
+ // / \param SeenFragments Map from DILocalVariable to all fragments of that
887
+ // / Variable which are known to exist.
888
+ // / \param OverlappingFragments The overlap map being constructed, from one
889
+ // / Var/Fragment pair to a vector of fragments known to overlap.
890
+ void LiveDebugValues::accumulateFragmentMap (MachineInstr &MI,
891
+ VarToFragments &SeenFragments,
892
+ OverlapMap &OverlappingFragments) {
893
+ DebugVariable MIVar (MI);
894
+ FragmentInfo ThisFragment = MIVar.getFragmentDefault ();
895
+
896
+ // If this is the first sighting of this variable, then we are guaranteed
897
+ // there are currently no overlapping fragments either. Initialize the set
898
+ // of seen fragments, record no overlaps for the current one, and return.
899
+ auto SeenIt = SeenFragments.find (MIVar.getVar ());
900
+ if (SeenIt == SeenFragments.end ()) {
901
+ SmallSet<FragmentInfo, 4 > OneFragment;
902
+ OneFragment.insert (ThisFragment);
903
+ SeenFragments.insert ({MIVar.getVar (), OneFragment});
904
+
905
+ OverlappingFragments.insert ({{MIVar.getVar (), ThisFragment}, {}});
906
+ return ;
907
+ }
908
+
909
+ // If this particular Variable/Fragment pair already exists in the overlap
910
+ // map, it has already been accounted for.
911
+ auto IsInOLapMap =
912
+ OverlappingFragments.insert ({{MIVar.getVar (), ThisFragment}, {}});
913
+ if (!IsInOLapMap.second )
914
+ return ;
915
+
916
+ auto &ThisFragmentsOverlaps = IsInOLapMap.first ->second ;
917
+ auto &AllSeenFragments = SeenIt->second ;
918
+
919
+ // Otherwise, examine all other seen fragments for this variable, with "this"
920
+ // fragment being a previously unseen fragment. Record any pair of
921
+ // overlapping fragments.
922
+ for (auto &ASeenFragment : AllSeenFragments) {
923
+ // Does this previously seen fragment overlap?
924
+ if (DIExpression::fragmentsOverlap (ThisFragment, ASeenFragment)) {
925
+ // Yes: Mark the current fragment as being overlapped.
926
+ ThisFragmentsOverlaps.push_back (ASeenFragment);
927
+ // Mark the previously seen fragment as being overlapped by the current
928
+ // one.
929
+ auto ASeenFragmentsOverlaps =
930
+ OverlappingFragments.find ({MIVar.getVar (), ASeenFragment});
931
+ assert (ASeenFragmentsOverlaps != OverlappingFragments.end () &&
932
+ " Previously seen var fragment has no vector of overlaps" );
933
+ ASeenFragmentsOverlaps->second .push_back (ThisFragment);
934
+ }
935
+ }
936
+
937
+ AllSeenFragments.insert (ThisFragment);
938
+ }
939
+
760
940
// / This routine creates OpenRanges and OutLocs.
761
941
bool LiveDebugValues::process (MachineInstr &MI, OpenRangesSet &OpenRanges,
762
942
VarLocInMBB &OutLocs, VarLocMap &VarLocIDs,
@@ -888,11 +1068,15 @@ bool LiveDebugValues::ExtendRanges(MachineFunction &MF) {
888
1068
bool OLChanged = false ;
889
1069
bool MBBJoined = false ;
890
1070
891
- VarLocMap VarLocIDs; // Map VarLoc<>unique ID for use in bitvectors.
892
- OpenRangesSet OpenRanges; // Ranges that are open until end of bb.
893
- VarLocInMBB OutLocs; // Ranges that exist beyond bb.
894
- VarLocInMBB InLocs; // Ranges that are incoming after joining.
895
- TransferMap Transfers; // DBG_VALUEs associated with spills.
1071
+ VarLocMap VarLocIDs; // Map VarLoc<>unique ID for use in bitvectors.
1072
+ OverlapMap OverlapFragments; // Map of overlapping variable fragments
1073
+ OpenRangesSet OpenRanges (OverlapFragments);
1074
+ // Ranges that are open until end of bb.
1075
+ VarLocInMBB OutLocs; // Ranges that exist beyond bb.
1076
+ VarLocInMBB InLocs; // Ranges that are incoming after joining.
1077
+ TransferMap Transfers; // DBG_VALUEs associated with spills.
1078
+
1079
+ VarToFragments SeenFragments;
896
1080
897
1081
// Blocks which are artificial, i.e. blocks which exclusively contain
898
1082
// instructions without locations, or with line 0 locations.
@@ -914,10 +1098,14 @@ bool LiveDebugValues::ExtendRanges(MachineFunction &MF) {
914
1098
// over the BBs. The LiveDebugVariables pass has already created DBG_VALUE
915
1099
// instructions for spills of registers that are known to be user variables
916
1100
// within the BB in which the spill occurs.
917
- for (auto &MBB : MF)
918
- for (auto &MI : MBB)
1101
+ for (auto &MBB : MF) {
1102
+ for (auto &MI : MBB) {
919
1103
process (MI, OpenRanges, OutLocs, VarLocIDs, Transfers,
920
1104
dontTransferChanges);
1105
+ if (MI.isDebugValue ())
1106
+ accumulateFragmentMap (MI, SeenFragments, OverlapFragments);
1107
+ }
1108
+ }
921
1109
922
1110
auto hasNonArtificialLocation = [](const MachineInstr &MI) -> bool {
923
1111
if (const DebugLoc &DL = MI.getDebugLoc ())
0 commit comments