DeclBase.cpp revision 4111024be81e7c0525e42dadcc126d27e5bf2425
1//===--- DeclBase.cpp - Declaration AST Node Implementation ---------------===//
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 implements the Decl and DeclContext classes.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/AST/DeclBase.h"
15#include "clang/AST/DeclCXX.h"
16#include "clang/AST/ASTContext.h"
17#include "llvm/ADT/DenseMap.h"
18using namespace clang;
19
20//===----------------------------------------------------------------------===//
21//  Statistics
22//===----------------------------------------------------------------------===//
23
24// temporary statistics gathering
25static unsigned nFuncs = 0;
26static unsigned nVars = 0;
27static unsigned nParmVars = 0;
28static unsigned nSUC = 0;
29static unsigned nEnumConst = 0;
30static unsigned nEnumDecls = 0;
31static unsigned nNamespaces = 0;
32static unsigned nTypedef = 0;
33static unsigned nFieldDecls = 0;
34static unsigned nInterfaceDecls = 0;
35static unsigned nClassDecls = 0;
36static unsigned nMethodDecls = 0;
37static unsigned nProtocolDecls = 0;
38static unsigned nForwardProtocolDecls = 0;
39static unsigned nCategoryDecls = 0;
40static unsigned nIvarDecls = 0;
41static unsigned nObjCImplementationDecls = 0;
42static unsigned nObjCCategoryImpl = 0;
43static unsigned nObjCCompatibleAlias = 0;
44static unsigned nObjCPropertyDecl = 0;
45static unsigned nObjCPropertyImplDecl = 0;
46static unsigned nLinkageSpecDecl = 0;
47static unsigned nFileScopeAsmDecl = 0;
48
49static bool StatSwitch = false;
50
51// This keeps track of all decl attributes. Since so few decls have attrs, we
52// keep them in a hash map instead of wasting space in the Decl class.
53typedef llvm::DenseMap<const Decl*, Attr*> DeclAttrMapTy;
54
55static DeclAttrMapTy *DeclAttrs = 0;
56
57const char *Decl::getDeclKindName() const {
58  switch (DeclKind) {
59  default: assert(0 && "Unknown decl kind!");
60  case Namespace:           return "Namespace";
61  case Typedef:             return "Typedef";
62  case Function:            return "Function";
63  case Var:                 return "Var";
64  case ParmVar:             return "ParmVar";
65  case EnumConstant:        return "EnumConstant";
66  case ObjCIvar:            return "ObjCIvar";
67  case ObjCInterface:       return "ObjCInterface";
68  case ObjCClass:           return "ObjCClass";
69  case ObjCMethod:          return "ObjCMethod";
70  case ObjCProtocol:        return "ObjCProtocol";
71  case ObjCForwardProtocol: return "ObjCForwardProtocol";
72  case Struct:              return "Struct";
73  case Union:               return "Union";
74  case Class:               return "Class";
75  case Enum:                return "Enum";
76  }
77}
78
79bool Decl::CollectingStats(bool Enable) {
80  if (Enable)
81    StatSwitch = true;
82  return StatSwitch;
83}
84
85void Decl::PrintStats() {
86  fprintf(stderr, "*** Decl Stats:\n");
87  fprintf(stderr, "  %d decls total.\n",
88          int(nFuncs+nVars+nParmVars+nFieldDecls+nSUC+
89              nEnumDecls+nEnumConst+nTypedef+nInterfaceDecls+nClassDecls+
90              nMethodDecls+nProtocolDecls+nCategoryDecls+nIvarDecls+
91              nNamespaces));
92  fprintf(stderr, "    %d namespace decls, %d each (%d bytes)\n",
93          nNamespaces, (int)sizeof(NamespaceDecl),
94          int(nNamespaces*sizeof(NamespaceDecl)));
95  fprintf(stderr, "    %d function decls, %d each (%d bytes)\n",
96          nFuncs, (int)sizeof(FunctionDecl), int(nFuncs*sizeof(FunctionDecl)));
97  fprintf(stderr, "    %d variable decls, %d each (%d bytes)\n",
98          nVars, (int)sizeof(VarDecl),
99          int(nVars*sizeof(VarDecl)));
100  fprintf(stderr, "    %d parameter variable decls, %d each (%d bytes)\n",
101          nParmVars, (int)sizeof(ParmVarDecl),
102          int(nParmVars*sizeof(ParmVarDecl)));
103  fprintf(stderr, "    %d field decls, %d each (%d bytes)\n",
104          nFieldDecls, (int)sizeof(FieldDecl),
105          int(nFieldDecls*sizeof(FieldDecl)));
106  fprintf(stderr, "    %d struct/union/class decls, %d each (%d bytes)\n",
107          nSUC, (int)sizeof(RecordDecl),
108          int(nSUC*sizeof(RecordDecl)));
109  fprintf(stderr, "    %d enum decls, %d each (%d bytes)\n",
110          nEnumDecls, (int)sizeof(EnumDecl),
111          int(nEnumDecls*sizeof(EnumDecl)));
112  fprintf(stderr, "    %d enum constant decls, %d each (%d bytes)\n",
113          nEnumConst, (int)sizeof(EnumConstantDecl),
114          int(nEnumConst*sizeof(EnumConstantDecl)));
115  fprintf(stderr, "    %d typedef decls, %d each (%d bytes)\n",
116          nTypedef, (int)sizeof(TypedefDecl),int(nTypedef*sizeof(TypedefDecl)));
117  // Objective-C decls...
118  fprintf(stderr, "    %d interface decls, %d each (%d bytes)\n",
119          nInterfaceDecls, (int)sizeof(ObjCInterfaceDecl),
120          int(nInterfaceDecls*sizeof(ObjCInterfaceDecl)));
121  fprintf(stderr, "    %d instance variable decls, %d each (%d bytes)\n",
122          nIvarDecls, (int)sizeof(ObjCIvarDecl),
123          int(nIvarDecls*sizeof(ObjCIvarDecl)));
124  fprintf(stderr, "    %d class decls, %d each (%d bytes)\n",
125          nClassDecls, (int)sizeof(ObjCClassDecl),
126          int(nClassDecls*sizeof(ObjCClassDecl)));
127  fprintf(stderr, "    %d method decls, %d each (%d bytes)\n",
128          nMethodDecls, (int)sizeof(ObjCMethodDecl),
129          int(nMethodDecls*sizeof(ObjCMethodDecl)));
130  fprintf(stderr, "    %d protocol decls, %d each (%d bytes)\n",
131          nProtocolDecls, (int)sizeof(ObjCProtocolDecl),
132          int(nProtocolDecls*sizeof(ObjCProtocolDecl)));
133  fprintf(stderr, "    %d forward protocol decls, %d each (%d bytes)\n",
134          nForwardProtocolDecls, (int)sizeof(ObjCForwardProtocolDecl),
135          int(nForwardProtocolDecls*sizeof(ObjCForwardProtocolDecl)));
136  fprintf(stderr, "    %d category decls, %d each (%d bytes)\n",
137          nCategoryDecls, (int)sizeof(ObjCCategoryDecl),
138          int(nCategoryDecls*sizeof(ObjCCategoryDecl)));
139
140  fprintf(stderr, "    %d class implementation decls, %d each (%d bytes)\n",
141          nObjCImplementationDecls, (int)sizeof(ObjCImplementationDecl),
142          int(nObjCImplementationDecls*sizeof(ObjCImplementationDecl)));
143
144  fprintf(stderr, "    %d class implementation decls, %d each (%d bytes)\n",
145          nObjCCategoryImpl, (int)sizeof(ObjCCategoryImplDecl),
146          int(nObjCCategoryImpl*sizeof(ObjCCategoryImplDecl)));
147
148  fprintf(stderr, "    %d compatibility alias decls, %d each (%d bytes)\n",
149          nObjCCompatibleAlias, (int)sizeof(ObjCCompatibleAliasDecl),
150          int(nObjCCompatibleAlias*sizeof(ObjCCompatibleAliasDecl)));
151
152  fprintf(stderr, "    %d property decls, %d each (%d bytes)\n",
153          nObjCPropertyDecl, (int)sizeof(ObjCPropertyDecl),
154          int(nObjCPropertyDecl*sizeof(ObjCPropertyDecl)));
155
156  fprintf(stderr, "    %d property implementation decls, %d each (%d bytes)\n",
157          nObjCPropertyImplDecl, (int)sizeof(ObjCPropertyImplDecl),
158          int(nObjCPropertyImplDecl*sizeof(ObjCPropertyImplDecl)));
159
160  fprintf(stderr, "Total bytes = %d\n",
161          int(nFuncs*sizeof(FunctionDecl)+
162              nVars*sizeof(VarDecl)+nParmVars*sizeof(ParmVarDecl)+
163              nFieldDecls*sizeof(FieldDecl)+nSUC*sizeof(RecordDecl)+
164              nEnumDecls*sizeof(EnumDecl)+nEnumConst*sizeof(EnumConstantDecl)+
165              nTypedef*sizeof(TypedefDecl)+
166              nInterfaceDecls*sizeof(ObjCInterfaceDecl)+
167              nIvarDecls*sizeof(ObjCIvarDecl)+
168              nClassDecls*sizeof(ObjCClassDecl)+
169              nMethodDecls*sizeof(ObjCMethodDecl)+
170              nProtocolDecls*sizeof(ObjCProtocolDecl)+
171              nForwardProtocolDecls*sizeof(ObjCForwardProtocolDecl)+
172              nCategoryDecls*sizeof(ObjCCategoryDecl)+
173              nObjCImplementationDecls*sizeof(ObjCImplementationDecl)+
174              nObjCCategoryImpl*sizeof(ObjCCategoryImplDecl)+
175              nObjCCompatibleAlias*sizeof(ObjCCompatibleAliasDecl)+
176              nObjCPropertyDecl*sizeof(ObjCPropertyDecl)+
177              nObjCPropertyImplDecl*sizeof(ObjCPropertyImplDecl)+
178              nLinkageSpecDecl*sizeof(LinkageSpecDecl)+
179              nFileScopeAsmDecl*sizeof(FileScopeAsmDecl)+
180              nNamespaces*sizeof(NamespaceDecl)));
181
182}
183
184void Decl::addDeclKind(Kind k) {
185  switch (k) {
186  case Namespace:           nNamespaces++; break;
187  case Typedef:             nTypedef++; break;
188  case Function:            nFuncs++; break;
189  case Var:                 nVars++; break;
190  case ParmVar:             nParmVars++; break;
191  case EnumConstant:        nEnumConst++; break;
192  case Field:               nFieldDecls++; break;
193  case Struct: case Union: case Class: nSUC++; break;
194  case Enum:                nEnumDecls++; break;
195  case ObjCInterface:       nInterfaceDecls++; break;
196  case ObjCClass:           nClassDecls++; break;
197  case ObjCMethod:          nMethodDecls++; break;
198  case ObjCProtocol:        nProtocolDecls++; break;
199  case ObjCForwardProtocol: nForwardProtocolDecls++; break;
200  case ObjCCategory:        nCategoryDecls++; break;
201  case ObjCIvar:            nIvarDecls++; break;
202  case ObjCImplementation:  nObjCImplementationDecls++; break;
203  case ObjCCategoryImpl:    nObjCCategoryImpl++; break;
204  case ObjCCompatibleAlias: nObjCCompatibleAlias++; break;
205  case ObjCProperty:        nObjCPropertyDecl++; break;
206  case ObjCPropertyImpl:    nObjCPropertyImplDecl++; break;
207  case LinkageSpec:         nLinkageSpecDecl++; break;
208  case FileScopeAsm:        nFileScopeAsmDecl++; break;
209  case ImplicitParam:
210  case TranslationUnit:     break;
211
212  // FIXME: Statistics for C++ decls.
213  case CXXField:
214  case CXXStruct:  case CXXUnion:  case CXXClass:
215  case CXXMethod:
216  case CXXClassVar:
217    break;
218  }
219}
220
221//===----------------------------------------------------------------------===//
222// Decl Implementation
223//===----------------------------------------------------------------------===//
224
225// Out-of-line virtual method providing a home for Decl.
226Decl::~Decl() {
227  if (!HasAttrs)
228    return;
229
230  DeclAttrMapTy::iterator it = DeclAttrs->find(this);
231  assert(it != DeclAttrs->end() && "No attrs found but HasAttrs is true!");
232
233  // release attributes.
234  delete it->second;
235  invalidateAttrs();
236}
237
238void Decl::addAttr(Attr *NewAttr) {
239  if (!DeclAttrs)
240    DeclAttrs = new DeclAttrMapTy();
241
242  Attr *&ExistingAttr = (*DeclAttrs)[this];
243
244  NewAttr->setNext(ExistingAttr);
245  ExistingAttr = NewAttr;
246
247  HasAttrs = true;
248}
249
250void Decl::invalidateAttrs() {
251  if (!HasAttrs) return;
252
253  HasAttrs = false;
254  (*DeclAttrs)[this] = 0;
255  DeclAttrs->erase(this);
256
257  if (DeclAttrs->empty()) {
258    delete DeclAttrs;
259    DeclAttrs = 0;
260  }
261}
262
263const Attr *Decl::getAttrs() const {
264  if (!HasAttrs)
265    return 0;
266
267  return (*DeclAttrs)[this];
268}
269
270void Decl::swapAttrs(Decl *RHS) {
271  bool HasLHSAttr = this->HasAttrs;
272  bool HasRHSAttr = RHS->HasAttrs;
273
274  // Usually, neither decl has attrs, nothing to do.
275  if (!HasLHSAttr && !HasRHSAttr) return;
276
277  // If 'this' has no attrs, swap the other way.
278  if (!HasLHSAttr)
279    return RHS->swapAttrs(this);
280
281  // Handle the case when both decls have attrs.
282  if (HasRHSAttr) {
283    std::swap((*DeclAttrs)[this], (*DeclAttrs)[RHS]);
284    return;
285  }
286
287  // Otherwise, LHS has an attr and RHS doesn't.
288  (*DeclAttrs)[RHS] = (*DeclAttrs)[this];
289  (*DeclAttrs).erase(this);
290  this->HasAttrs = false;
291  RHS->HasAttrs = true;
292}
293
294
295void Decl::Destroy(ASTContext& C) {
296
297  if (ScopedDecl* SD = dyn_cast<ScopedDecl>(this)) {
298
299    // Observe the unrolled recursion.  By setting N->NextDeclarator = 0x0
300    // within the loop, only the Destroy method for the first ScopedDecl
301    // will deallocate all of the ScopedDecls in a chain.
302
303    ScopedDecl* N = SD->getNextDeclarator();
304
305    while (N) {
306      ScopedDecl* Tmp = N->getNextDeclarator();
307      N->NextDeclarator = 0x0;
308      N->Destroy(C);
309      N = Tmp;
310    }
311  }
312
313  this->~Decl();
314  C.getAllocator().Deallocate((void *)this);
315}
316
317//===----------------------------------------------------------------------===//
318// DeclContext Implementation
319//===----------------------------------------------------------------------===//
320
321DeclContext *DeclContext::getParent() const {
322  if (ScopedDecl *SD = dyn_cast<ScopedDecl>(this))
323    return SD->getDeclContext();
324  else
325    return NULL;
326}
327
328Decl *DeclContext::ToDecl (const DeclContext *D) {
329  return CastTo<Decl>(D);
330}
331
332DeclContext *DeclContext::FromDecl (const Decl *D) {
333  return CastTo<DeclContext>(D);
334}
335