• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- DeclContextInternals.h - DeclContext Representation -----*- 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 data structures used in the implementation
11 //  of DeclContext.
12 //
13 //===----------------------------------------------------------------------===//
14 #ifndef LLVM_CLANG_AST_DECLCONTEXTINTERNALS_H
15 #define LLVM_CLANG_AST_DECLCONTEXTINTERNALS_H
16 
17 #include "clang/AST/Decl.h"
18 #include "clang/AST/DeclCXX.h"
19 #include "clang/AST/DeclarationName.h"
20 #include "llvm/ADT/DenseMap.h"
21 #include "llvm/ADT/PointerUnion.h"
22 #include "llvm/ADT/SmallVector.h"
23 #include <algorithm>
24 
25 namespace clang {
26 
27 class DependentDiagnostic;
28 
29 /// StoredDeclsList - This is an array of decls optimized a common case of only
30 /// containing one entry.
31 struct StoredDeclsList {
32 
33   /// DeclsTy - When in vector form, this is what the Data pointer points to.
34   typedef SmallVector<NamedDecl *, 4> DeclsTy;
35 
36   /// \brief The stored data, which will be either a pointer to a NamedDecl,
37   /// or a pointer to a vector.
38   llvm::PointerUnion<NamedDecl *, DeclsTy *> Data;
39 
40 public:
StoredDeclsListStoredDeclsList41   StoredDeclsList() {}
42 
StoredDeclsListStoredDeclsList43   StoredDeclsList(const StoredDeclsList &RHS) : Data(RHS.Data) {
44     if (DeclsTy *RHSVec = RHS.getAsVector())
45       Data = new DeclsTy(*RHSVec);
46   }
47 
~StoredDeclsListStoredDeclsList48   ~StoredDeclsList() {
49     // If this is a vector-form, free the vector.
50     if (DeclsTy *Vector = getAsVector())
51       delete Vector;
52   }
53 
54   StoredDeclsList &operator=(const StoredDeclsList &RHS) {
55     if (DeclsTy *Vector = getAsVector())
56       delete Vector;
57     Data = RHS.Data;
58     if (DeclsTy *RHSVec = RHS.getAsVector())
59       Data = new DeclsTy(*RHSVec);
60     return *this;
61   }
62 
isNullStoredDeclsList63   bool isNull() const { return Data.isNull(); }
64 
getAsDeclStoredDeclsList65   NamedDecl *getAsDecl() const {
66     return Data.dyn_cast<NamedDecl *>();
67   }
68 
getAsVectorStoredDeclsList69   DeclsTy *getAsVector() const {
70     return Data.dyn_cast<DeclsTy *>();
71   }
72 
setOnlyValueStoredDeclsList73   void setOnlyValue(NamedDecl *ND) {
74     assert(!getAsVector() && "Not inline");
75     Data = ND;
76     // Make sure that Data is a plain NamedDecl* so we can use its address
77     // at getLookupResult.
78     assert(*(NamedDecl **)&Data == ND &&
79            "PointerUnion mangles the NamedDecl pointer!");
80   }
81 
removeStoredDeclsList82   void remove(NamedDecl *D) {
83     assert(!isNull() && "removing from empty list");
84     if (NamedDecl *Singleton = getAsDecl()) {
85       assert(Singleton == D && "list is different singleton");
86       (void)Singleton;
87       Data = (NamedDecl *)0;
88       return;
89     }
90 
91     DeclsTy &Vec = *getAsVector();
92     DeclsTy::iterator I = std::find(Vec.begin(), Vec.end(), D);
93     assert(I != Vec.end() && "list does not contain decl");
94     Vec.erase(I);
95 
96     assert(std::find(Vec.begin(), Vec.end(), D)
97              == Vec.end() && "list still contains decl");
98   }
99 
100   /// \brief Remove any declarations which were imported from an external
101   /// AST source.
removeExternalDeclsStoredDeclsList102   void removeExternalDecls() {
103     if (isNull()) {
104       // Nothing to do.
105     } else if (NamedDecl *Singleton = getAsDecl()) {
106       if (Singleton->isFromASTFile())
107         *this = StoredDeclsList();
108     } else {
109       DeclsTy &Vec = *getAsVector();
110       Vec.erase(std::remove_if(Vec.begin(), Vec.end(),
111                                std::mem_fun(&Decl::isFromASTFile)),
112                 Vec.end());
113     }
114   }
115 
116   /// getLookupResult - Return an array of all the decls that this list
117   /// represents.
getLookupResultStoredDeclsList118   DeclContext::lookup_result getLookupResult() {
119     if (isNull())
120       return DeclContext::lookup_result(DeclContext::lookup_iterator(0),
121                                         DeclContext::lookup_iterator(0));
122 
123     // If we have a single NamedDecl, return it.
124     if (getAsDecl()) {
125       assert(!isNull() && "Empty list isn't allowed");
126 
127       // Data is a raw pointer to a NamedDecl*, return it.
128       void *Ptr = &Data;
129       return DeclContext::lookup_result((NamedDecl**)Ptr, (NamedDecl**)Ptr+1);
130     }
131 
132     assert(getAsVector() && "Must have a vector at this point");
133     DeclsTy &Vector = *getAsVector();
134 
135     // Otherwise, we have a range result.
136     return DeclContext::lookup_result(Vector.begin(), Vector.end());
137   }
138 
139   /// HandleRedeclaration - If this is a redeclaration of an existing decl,
140   /// replace the old one with D and return true.  Otherwise return false.
HandleRedeclarationStoredDeclsList141   bool HandleRedeclaration(NamedDecl *D) {
142     // Most decls only have one entry in their list, special case it.
143     if (NamedDecl *OldD = getAsDecl()) {
144       if (!D->declarationReplaces(OldD))
145         return false;
146       setOnlyValue(D);
147       return true;
148     }
149 
150     // Determine if this declaration is actually a redeclaration.
151     DeclsTy &Vec = *getAsVector();
152     for (DeclsTy::iterator OD = Vec.begin(), ODEnd = Vec.end();
153          OD != ODEnd; ++OD) {
154       NamedDecl *OldD = *OD;
155       if (D->declarationReplaces(OldD)) {
156         *OD = D;
157         return true;
158       }
159     }
160 
161     return false;
162   }
163 
164   /// AddSubsequentDecl - This is called on the second and later decl when it is
165   /// not a redeclaration to merge it into the appropriate place in our list.
166   ///
AddSubsequentDeclStoredDeclsList167   void AddSubsequentDecl(NamedDecl *D) {
168     // If this is the second decl added to the list, convert this to vector
169     // form.
170     if (NamedDecl *OldD = getAsDecl()) {
171       DeclsTy *VT = new DeclsTy();
172       VT->push_back(OldD);
173       Data = VT;
174     }
175 
176     DeclsTy &Vec = *getAsVector();
177 
178     // Using directives end up in a special entry which contains only
179     // other using directives, so all this logic is wasted for them.
180     // But avoiding the logic wastes time in the far-more-common case
181     // that we're *not* adding a new using directive.
182 
183     // Tag declarations always go at the end of the list so that an
184     // iterator which points at the first tag will start a span of
185     // decls that only contains tags.
186     if (D->hasTagIdentifierNamespace())
187       Vec.push_back(D);
188 
189     // Resolved using declarations go at the front of the list so that
190     // they won't show up in other lookup results.  Unresolved using
191     // declarations (which are always in IDNS_Using | IDNS_Ordinary)
192     // follow that so that the using declarations will be contiguous.
193     else if (D->getIdentifierNamespace() & Decl::IDNS_Using) {
194       DeclsTy::iterator I = Vec.begin();
195       if (D->getIdentifierNamespace() != Decl::IDNS_Using) {
196         while (I != Vec.end() &&
197                (*I)->getIdentifierNamespace() == Decl::IDNS_Using)
198           ++I;
199       }
200       Vec.insert(I, D);
201 
202     // All other declarations go at the end of the list, but before any
203     // tag declarations.  But we can be clever about tag declarations
204     // because there can only ever be one in a scope.
205     } else if (!Vec.empty() && Vec.back()->hasTagIdentifierNamespace()) {
206       NamedDecl *TagD = Vec.back();
207       Vec.back() = D;
208       Vec.push_back(TagD);
209     } else
210       Vec.push_back(D);
211   }
212 };
213 
214 class StoredDeclsMap
215   : public llvm::SmallDenseMap<DeclarationName, StoredDeclsList, 4> {
216 
217 public:
218   static void DestroyAll(StoredDeclsMap *Map, bool Dependent);
219 
220 private:
221   friend class ASTContext; // walks the chain deleting these
222   friend class DeclContext;
223   llvm::PointerIntPair<StoredDeclsMap*, 1> Previous;
224 };
225 
226 class DependentStoredDeclsMap : public StoredDeclsMap {
227 public:
DependentStoredDeclsMap()228   DependentStoredDeclsMap() : FirstDiagnostic(0) {}
229 
230 private:
231   friend class DependentDiagnostic;
232   friend class DeclContext; // iterates over diagnostics
233 
234   DependentDiagnostic *FirstDiagnostic;
235 };
236 
237 } // end namespace clang
238 
239 #endif
240