add iteration support to TinyPtrVector for clang's use.
[oota-llvm.git] / include / llvm / ADT / TinyPtrVector.h
1 //===- llvm/ADT/TinyPtrVector.h - 'Normally tiny' vectors -------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file defines the Type class.
11 //
12 //===----------------------------------------------------------------------===//
13
14 #ifndef LLVM_ADT_TINYPTRVECTOR_H
15 #define LLVM_ADT_TINYPTRVECTOR_H
16
17 #include "llvm/ADT/SmallVector.h"
18 #include "llvm/ADT/PointerUnion.h"
19
20 namespace llvm {
21   
22 /// TinyPtrVector - This class is specialized for cases where there are
23 /// normally 0 or 1 element in a vector, but is general enough to go beyond that
24 /// when required.
25 ///
26 /// NOTE: This container doesn't allow you to store a null pointer into it.
27 ///
28 template <typename EltTy>
29 class TinyPtrVector {
30 public:
31   typedef llvm::SmallVector<EltTy, 4> VecTy;
32   llvm::PointerUnion<EltTy, VecTy*> Val;
33   
34   TinyPtrVector() {}
35   TinyPtrVector(const TinyPtrVector &RHS) : Val(RHS.Val) {
36     if (VecTy *V = Val.template dyn_cast<VecTy*>())
37       Val = new VecTy(*V);
38   }
39   ~TinyPtrVector() {
40     if (VecTy *V = Val.template dyn_cast<VecTy*>())
41       delete V;
42   }
43   
44   /// empty() - This vector can be empty if it contains no element, or if it
45   /// contains a pointer to an empty vector.
46   bool empty() const {
47     if (Val.isNull()) return true;
48     if (VecTy *Vec = Val.template dyn_cast<VecTy*>())
49       return Vec->empty();
50     return false;
51   }
52   
53   unsigned size() const {
54     if (empty())
55       return 0;
56     if (Val. template is<EltTy>())
57       return 1;
58     return Val. template get<VecTy*>()->size();
59   }
60   
61   typedef const EltTy *iterator;
62   iterator begin() const {
63     if (empty())
64       return 0;
65     
66     if (Val.template is<EltTy>())
67       return Val.template getAddrOf<EltTy>();
68     
69     return Val.template get<VecTy *>()->begin();
70
71   }
72   iterator end() const {
73     if (empty())
74       return 0;
75     
76     if (Val.template is<EltTy>())
77       return begin() + 1;
78     
79     return Val.template get<VecTy *>()->end();
80   }
81
82   
83   EltTy operator[](unsigned i) const {
84     assert(!Val.isNull() && "can't index into an empty vector");
85     if (EltTy V = Val.template dyn_cast<EltTy>()) {
86       assert(i == 0 && "tinyvector index out of range");
87       return V;
88     }
89     
90     assert(i < Val. template get<VecTy*>()->size() && 
91            "tinyvector index out of range");
92     return (*Val. template get<VecTy*>())[i];
93   }
94   
95   EltTy front() const {
96     assert(!empty() && "vector empty");
97     if (EltTy V = Val.template dyn_cast<EltTy>())
98       return V;
99     return Val.template get<VecTy*>()->front();
100   }
101   
102   void push_back(EltTy NewVal) {
103     assert(NewVal != 0 && "Can't add a null value");
104     
105     // If we have nothing, add something.
106     if (Val.isNull()) {
107       Val = NewVal;
108       return;
109     }
110     
111     // If we have a single value, convert to a vector.
112     if (EltTy V = Val.template  dyn_cast<EltTy>()) {
113       Val = new VecTy();
114       Val.template get<VecTy*>()->push_back(V);
115     }
116     
117     // Add the new value, we know we have a vector.
118     Val.template get<VecTy*>()->push_back(NewVal);
119   }
120   
121   void clear() {
122     // If we have a single value, convert to empty.
123     if (Val.template is<EltTy>()) {
124       Val = (EltTy)0;
125     } else if (VecTy *Vec = Val.template dyn_cast<VecTy*>()) {
126       // If we have a vector form, just clear it.
127       Vec->clear();
128     }
129     // Otherwise, we're already empty.
130   }
131   
132 private:
133   void operator=(const TinyPtrVector&); // NOT IMPLEMENTED YET.
134 };
135 } // end namespace llvm
136
137 #endif