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