xref: /aosp_15_r20/external/llvm/lib/IR/Use.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===-- Use.cpp - Implement the Use class ---------------------------------===//
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/IR/Use.h"
11*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/User.h"
12*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/Value.h"
13*9880d681SAndroid Build Coastguard Worker #include <new>
14*9880d681SAndroid Build Coastguard Worker 
15*9880d681SAndroid Build Coastguard Worker namespace llvm {
16*9880d681SAndroid Build Coastguard Worker 
swap(Use & RHS)17*9880d681SAndroid Build Coastguard Worker void Use::swap(Use &RHS) {
18*9880d681SAndroid Build Coastguard Worker   if (Val == RHS.Val)
19*9880d681SAndroid Build Coastguard Worker     return;
20*9880d681SAndroid Build Coastguard Worker 
21*9880d681SAndroid Build Coastguard Worker   if (Val)
22*9880d681SAndroid Build Coastguard Worker     removeFromList();
23*9880d681SAndroid Build Coastguard Worker 
24*9880d681SAndroid Build Coastguard Worker   Value *OldVal = Val;
25*9880d681SAndroid Build Coastguard Worker   if (RHS.Val) {
26*9880d681SAndroid Build Coastguard Worker     RHS.removeFromList();
27*9880d681SAndroid Build Coastguard Worker     Val = RHS.Val;
28*9880d681SAndroid Build Coastguard Worker     Val->addUse(*this);
29*9880d681SAndroid Build Coastguard Worker   } else {
30*9880d681SAndroid Build Coastguard Worker     Val = nullptr;
31*9880d681SAndroid Build Coastguard Worker   }
32*9880d681SAndroid Build Coastguard Worker 
33*9880d681SAndroid Build Coastguard Worker   if (OldVal) {
34*9880d681SAndroid Build Coastguard Worker     RHS.Val = OldVal;
35*9880d681SAndroid Build Coastguard Worker     RHS.Val->addUse(RHS);
36*9880d681SAndroid Build Coastguard Worker   } else {
37*9880d681SAndroid Build Coastguard Worker     RHS.Val = nullptr;
38*9880d681SAndroid Build Coastguard Worker   }
39*9880d681SAndroid Build Coastguard Worker }
40*9880d681SAndroid Build Coastguard Worker 
getUser() const41*9880d681SAndroid Build Coastguard Worker User *Use::getUser() const {
42*9880d681SAndroid Build Coastguard Worker   const Use *End = getImpliedUser();
43*9880d681SAndroid Build Coastguard Worker   const UserRef *ref = reinterpret_cast<const UserRef *>(End);
44*9880d681SAndroid Build Coastguard Worker   return ref->getInt() ? ref->getPointer()
45*9880d681SAndroid Build Coastguard Worker                        : reinterpret_cast<User *>(const_cast<Use *>(End));
46*9880d681SAndroid Build Coastguard Worker }
47*9880d681SAndroid Build Coastguard Worker 
getOperandNo() const48*9880d681SAndroid Build Coastguard Worker unsigned Use::getOperandNo() const {
49*9880d681SAndroid Build Coastguard Worker   return this - getUser()->op_begin();
50*9880d681SAndroid Build Coastguard Worker }
51*9880d681SAndroid Build Coastguard Worker 
52*9880d681SAndroid Build Coastguard Worker // Sets up the waymarking algorithm's tags for a series of Uses. See the
53*9880d681SAndroid Build Coastguard Worker // algorithm details here:
54*9880d681SAndroid Build Coastguard Worker //
55*9880d681SAndroid Build Coastguard Worker //   http://www.llvm.org/docs/ProgrammersManual.html#the-waymarking-algorithm
56*9880d681SAndroid Build Coastguard Worker //
initTags(Use * const Start,Use * Stop)57*9880d681SAndroid Build Coastguard Worker Use *Use::initTags(Use *const Start, Use *Stop) {
58*9880d681SAndroid Build Coastguard Worker   ptrdiff_t Done = 0;
59*9880d681SAndroid Build Coastguard Worker   while (Done < 20) {
60*9880d681SAndroid Build Coastguard Worker     if (Start == Stop--)
61*9880d681SAndroid Build Coastguard Worker       return Start;
62*9880d681SAndroid Build Coastguard Worker     static const PrevPtrTag tags[20] = {
63*9880d681SAndroid Build Coastguard Worker         fullStopTag,  oneDigitTag,  stopTag,      oneDigitTag, oneDigitTag,
64*9880d681SAndroid Build Coastguard Worker         stopTag,      zeroDigitTag, oneDigitTag,  oneDigitTag, stopTag,
65*9880d681SAndroid Build Coastguard Worker         zeroDigitTag, oneDigitTag,  zeroDigitTag, oneDigitTag, stopTag,
66*9880d681SAndroid Build Coastguard Worker         oneDigitTag,  oneDigitTag,  oneDigitTag,  oneDigitTag, stopTag};
67*9880d681SAndroid Build Coastguard Worker     new (Stop) Use(tags[Done++]);
68*9880d681SAndroid Build Coastguard Worker   }
69*9880d681SAndroid Build Coastguard Worker 
70*9880d681SAndroid Build Coastguard Worker   ptrdiff_t Count = Done;
71*9880d681SAndroid Build Coastguard Worker   while (Start != Stop) {
72*9880d681SAndroid Build Coastguard Worker     --Stop;
73*9880d681SAndroid Build Coastguard Worker     if (!Count) {
74*9880d681SAndroid Build Coastguard Worker       new (Stop) Use(stopTag);
75*9880d681SAndroid Build Coastguard Worker       ++Done;
76*9880d681SAndroid Build Coastguard Worker       Count = Done;
77*9880d681SAndroid Build Coastguard Worker     } else {
78*9880d681SAndroid Build Coastguard Worker       new (Stop) Use(PrevPtrTag(Count & 1));
79*9880d681SAndroid Build Coastguard Worker       Count >>= 1;
80*9880d681SAndroid Build Coastguard Worker       ++Done;
81*9880d681SAndroid Build Coastguard Worker     }
82*9880d681SAndroid Build Coastguard Worker   }
83*9880d681SAndroid Build Coastguard Worker 
84*9880d681SAndroid Build Coastguard Worker   return Start;
85*9880d681SAndroid Build Coastguard Worker }
86*9880d681SAndroid Build Coastguard Worker 
zap(Use * Start,const Use * Stop,bool del)87*9880d681SAndroid Build Coastguard Worker void Use::zap(Use *Start, const Use *Stop, bool del) {
88*9880d681SAndroid Build Coastguard Worker   while (Start != Stop)
89*9880d681SAndroid Build Coastguard Worker     (--Stop)->~Use();
90*9880d681SAndroid Build Coastguard Worker   if (del)
91*9880d681SAndroid Build Coastguard Worker     ::operator delete(Start);
92*9880d681SAndroid Build Coastguard Worker }
93*9880d681SAndroid Build Coastguard Worker 
getImpliedUser() const94*9880d681SAndroid Build Coastguard Worker const Use *Use::getImpliedUser() const {
95*9880d681SAndroid Build Coastguard Worker   const Use *Current = this;
96*9880d681SAndroid Build Coastguard Worker 
97*9880d681SAndroid Build Coastguard Worker   while (true) {
98*9880d681SAndroid Build Coastguard Worker     unsigned Tag = (Current++)->Prev.getInt();
99*9880d681SAndroid Build Coastguard Worker     switch (Tag) {
100*9880d681SAndroid Build Coastguard Worker     case zeroDigitTag:
101*9880d681SAndroid Build Coastguard Worker     case oneDigitTag:
102*9880d681SAndroid Build Coastguard Worker       continue;
103*9880d681SAndroid Build Coastguard Worker 
104*9880d681SAndroid Build Coastguard Worker     case stopTag: {
105*9880d681SAndroid Build Coastguard Worker       ++Current;
106*9880d681SAndroid Build Coastguard Worker       ptrdiff_t Offset = 1;
107*9880d681SAndroid Build Coastguard Worker       while (true) {
108*9880d681SAndroid Build Coastguard Worker         unsigned Tag = Current->Prev.getInt();
109*9880d681SAndroid Build Coastguard Worker         switch (Tag) {
110*9880d681SAndroid Build Coastguard Worker         case zeroDigitTag:
111*9880d681SAndroid Build Coastguard Worker         case oneDigitTag:
112*9880d681SAndroid Build Coastguard Worker           ++Current;
113*9880d681SAndroid Build Coastguard Worker           Offset = (Offset << 1) + Tag;
114*9880d681SAndroid Build Coastguard Worker           continue;
115*9880d681SAndroid Build Coastguard Worker         default:
116*9880d681SAndroid Build Coastguard Worker           return Current + Offset;
117*9880d681SAndroid Build Coastguard Worker         }
118*9880d681SAndroid Build Coastguard Worker       }
119*9880d681SAndroid Build Coastguard Worker     }
120*9880d681SAndroid Build Coastguard Worker 
121*9880d681SAndroid Build Coastguard Worker     case fullStopTag:
122*9880d681SAndroid Build Coastguard Worker       return Current;
123*9880d681SAndroid Build Coastguard Worker     }
124*9880d681SAndroid Build Coastguard Worker   }
125*9880d681SAndroid Build Coastguard Worker }
126*9880d681SAndroid Build Coastguard Worker 
127*9880d681SAndroid Build Coastguard Worker } // End llvm namespace
128