1 //===--- Entity.cpp - Cross-translation-unit "token" for decls ------------===//
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 // Entity is a ASTContext-independent way to refer to declarations that are
11 // visible across translation units.
12 //
13 //===----------------------------------------------------------------------===//
14
15 #include "EntityImpl.h"
16 #include "ProgramImpl.h"
17 #include "clang/Index/Program.h"
18 #include "clang/Index/GlobalSelector.h"
19 #include "clang/AST/Decl.h"
20 #include "clang/AST/ASTContext.h"
21 #include "clang/AST/DeclVisitor.h"
22 using namespace clang;
23 using namespace idx;
24
25 // FIXME: Entity is really really basic currently, mostly written to work
26 // on variables and functions. Should support types and other decls eventually..
27
28
29 //===----------------------------------------------------------------------===//
30 // EntityGetter
31 //===----------------------------------------------------------------------===//
32
33 namespace clang {
34 namespace idx {
35
36 /// \brief Gets the Entity associated with a Decl.
37 class EntityGetter : public DeclVisitor<EntityGetter, Entity> {
38 Program &Prog;
39 ProgramImpl &ProgImpl;
40
41 public:
EntityGetter(Program & prog,ProgramImpl & progImpl)42 EntityGetter(Program &prog, ProgramImpl &progImpl)
43 : Prog(prog), ProgImpl(progImpl) { }
44
45 // Get an Entity.
46 Entity getEntity(Entity Parent, DeclarationName Name,
47 unsigned IdNS, bool isObjCInstanceMethod);
48
49 // Get an Entity associated with the name in the global namespace.
50 Entity getGlobalEntity(llvm::StringRef Name);
51
52 Entity VisitNamedDecl(NamedDecl *D);
53 Entity VisitVarDecl(VarDecl *D);
54 Entity VisitFieldDecl(FieldDecl *D);
55 Entity VisitFunctionDecl(FunctionDecl *D);
56 Entity VisitTypeDecl(TypeDecl *D);
57 };
58
59 }
60 }
61
getEntity(Entity Parent,DeclarationName Name,unsigned IdNS,bool isObjCInstanceMethod)62 Entity EntityGetter::getEntity(Entity Parent, DeclarationName Name,
63 unsigned IdNS, bool isObjCInstanceMethod) {
64 llvm::FoldingSetNodeID ID;
65 EntityImpl::Profile(ID, Parent, Name, IdNS, isObjCInstanceMethod);
66
67 ProgramImpl::EntitySetTy &Entities = ProgImpl.getEntities();
68 void *InsertPos = 0;
69 if (EntityImpl *Ent = Entities.FindNodeOrInsertPos(ID, InsertPos))
70 return Entity(Ent);
71
72 void *Buf = ProgImpl.Allocate(sizeof(EntityImpl));
73 EntityImpl *New =
74 new (Buf) EntityImpl(Parent, Name, IdNS, isObjCInstanceMethod);
75 Entities.InsertNode(New, InsertPos);
76
77 return Entity(New);
78 }
79
getGlobalEntity(llvm::StringRef Name)80 Entity EntityGetter::getGlobalEntity(llvm::StringRef Name) {
81 IdentifierInfo *II = &ProgImpl.getIdents().get(Name);
82 DeclarationName GlobName(II);
83 unsigned IdNS = Decl::IDNS_Ordinary;
84 return getEntity(Entity(), GlobName, IdNS, false);
85 }
86
VisitNamedDecl(NamedDecl * D)87 Entity EntityGetter::VisitNamedDecl(NamedDecl *D) {
88 Entity Parent;
89 if (!D->getDeclContext()->isTranslationUnit()) {
90 Parent = Visit(cast<Decl>(D->getDeclContext()));
91 // FIXME: Anonymous structs ?
92 if (Parent.isInvalid())
93 return Entity();
94 }
95 if (Parent.isValid() && Parent.isInternalToTU())
96 return Entity(D);
97
98 // FIXME: Only works for DeclarationNames that are identifiers and selectors.
99 // Treats other DeclarationNames as internal Decls for now..
100
101 DeclarationName LocalName = D->getDeclName();
102 if (!LocalName)
103 return Entity(D);
104
105 DeclarationName GlobName;
106
107 if (IdentifierInfo *II = LocalName.getAsIdentifierInfo()) {
108 IdentifierInfo *GlobII = &ProgImpl.getIdents().get(II->getName());
109 GlobName = DeclarationName(GlobII);
110 } else {
111 Selector LocalSel = LocalName.getObjCSelector();
112
113 // Treats other DeclarationNames as internal Decls for now..
114 if (LocalSel.isNull())
115 return Entity(D);
116
117 Selector GlobSel =
118 (uintptr_t)GlobalSelector::get(LocalSel, Prog).getAsOpaquePtr();
119 GlobName = DeclarationName(GlobSel);
120 }
121
122 assert(GlobName);
123
124 unsigned IdNS = D->getIdentifierNamespace();
125
126 ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(D);
127 bool isObjCInstanceMethod = MD && MD->isInstanceMethod();
128 return getEntity(Parent, GlobName, IdNS, isObjCInstanceMethod);
129 }
130
VisitVarDecl(VarDecl * D)131 Entity EntityGetter::VisitVarDecl(VarDecl *D) {
132 // Local variables have no linkage, make invalid Entities.
133 if (D->hasLocalStorage())
134 return Entity();
135
136 // If it's static it cannot be referred to by another translation unit.
137 if (D->getStorageClass() == SC_Static)
138 return Entity(D);
139
140 return VisitNamedDecl(D);
141 }
142
VisitFunctionDecl(FunctionDecl * D)143 Entity EntityGetter::VisitFunctionDecl(FunctionDecl *D) {
144 // If it's static it cannot be referred to by another translation unit.
145 if (D->getStorageClass() == SC_Static)
146 return Entity(D);
147
148 return VisitNamedDecl(D);
149 }
150
VisitFieldDecl(FieldDecl * D)151 Entity EntityGetter::VisitFieldDecl(FieldDecl *D) {
152 // Make FieldDecl an invalid Entity since it has no linkage.
153 return Entity();
154 }
155
VisitTypeDecl(TypeDecl * D)156 Entity EntityGetter::VisitTypeDecl(TypeDecl *D) {
157 // Although in C++ class name has external linkage, usually the definition of
158 // the class is available in the same translation unit when it's needed. So we
159 // make all of them invalid Entity.
160 return Entity();
161 }
162
163 //===----------------------------------------------------------------------===//
164 // EntityImpl Implementation
165 //===----------------------------------------------------------------------===//
166
getDecl(ASTContext & AST)167 Decl *EntityImpl::getDecl(ASTContext &AST) {
168 DeclContext *DC =
169 Parent.isInvalid() ? AST.getTranslationUnitDecl()
170 : cast<DeclContext>(Parent.getDecl(AST));
171 if (!DC)
172 return 0; // Couldn't get the parent context.
173
174 DeclarationName LocalName;
175
176 if (IdentifierInfo *GlobII = Name.getAsIdentifierInfo()) {
177 IdentifierInfo &II = AST.Idents.get(GlobII->getName());
178 LocalName = DeclarationName(&II);
179 } else {
180 Selector GlobSel = Name.getObjCSelector();
181 assert(!GlobSel.isNull() && "A not handled yet declaration name");
182 GlobalSelector GSel =
183 GlobalSelector::getFromOpaquePtr(GlobSel.getAsOpaquePtr());
184 LocalName = GSel.getSelector(AST);
185 }
186
187 assert(LocalName);
188
189 DeclContext::lookup_result Res = DC->lookup(LocalName);
190 for (DeclContext::lookup_iterator I = Res.first, E = Res.second; I!=E; ++I) {
191 Decl *D = *I;
192 if (D->getIdentifierNamespace() == IdNS) {
193 if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(D)) {
194 if (MD->isInstanceMethod() == IsObjCInstanceMethod)
195 return MD;
196 } else
197 return D;
198 }
199 }
200
201 return 0; // Failed to find a decl using this Entity.
202 }
203
204 /// \brief Get an Entity associated with the given Decl.
205 /// \returns Null if an Entity cannot refer to this Decl.
get(Decl * D,Program & Prog,ProgramImpl & ProgImpl)206 Entity EntityImpl::get(Decl *D, Program &Prog, ProgramImpl &ProgImpl) {
207 assert(D && "Passed null Decl");
208 return EntityGetter(Prog, ProgImpl).Visit(D);
209 }
210
211 /// \brief Get an Entity associated with a global name.
get(llvm::StringRef Name,Program & Prog,ProgramImpl & ProgImpl)212 Entity EntityImpl::get(llvm::StringRef Name, Program &Prog,
213 ProgramImpl &ProgImpl) {
214 return EntityGetter(Prog, ProgImpl).getGlobalEntity(Name);
215 }
216
getPrintableName()217 std::string EntityImpl::getPrintableName() {
218 return Name.getAsString();
219 }
220
221 //===----------------------------------------------------------------------===//
222 // Entity Implementation
223 //===----------------------------------------------------------------------===//
224
Entity(Decl * D)225 Entity::Entity(Decl *D) : Val(D->getCanonicalDecl()) { }
226
227 /// \brief Find the Decl that can be referred to by this entity.
getDecl(ASTContext & AST) const228 Decl *Entity::getDecl(ASTContext &AST) const {
229 if (isInvalid())
230 return 0;
231
232 if (Decl *D = Val.dyn_cast<Decl *>())
233 // Check that the passed AST is actually the one that this Decl belongs to.
234 return (&D->getASTContext() == &AST) ? D : 0;
235
236 return Val.get<EntityImpl *>()->getDecl(AST);
237 }
238
getPrintableName() const239 std::string Entity::getPrintableName() const {
240 if (isInvalid())
241 return "<< Invalid >>";
242
243 if (Decl *D = Val.dyn_cast<Decl *>()) {
244 if (NamedDecl *ND = dyn_cast<NamedDecl>(D))
245 return ND->getNameAsString();
246 else
247 return std::string();
248 }
249
250 return Val.get<EntityImpl *>()->getPrintableName();
251 }
252
253 /// \brief Get an Entity associated with the given Decl.
254 /// \returns Null if an Entity cannot refer to this Decl.
get(Decl * D,Program & Prog)255 Entity Entity::get(Decl *D, Program &Prog) {
256 if (D == 0)
257 return Entity();
258 ProgramImpl &ProgImpl = *static_cast<ProgramImpl*>(Prog.Impl);
259 return EntityImpl::get(D, Prog, ProgImpl);
260 }
261
get(llvm::StringRef Name,Program & Prog)262 Entity Entity::get(llvm::StringRef Name, Program &Prog) {
263 ProgramImpl &ProgImpl = *static_cast<ProgramImpl*>(Prog.Impl);
264 return EntityImpl::get(Name, Prog, ProgImpl);
265 }
266
267 unsigned
getHashValue(Entity E)268 llvm::DenseMapInfo<Entity>::getHashValue(Entity E) {
269 return DenseMapInfo<void*>::getHashValue(E.getAsOpaquePtr());
270 }
271