//===--- SemaCXXScopeSpec.cpp - Semantic Analysis for C++ scope specifiers-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This file implements C++ semantic analysis for scope specifiers. // //===----------------------------------------------------------------------===// #include "Sema.h" #include "clang/AST/ASTContext.h" #include "clang/Parse/DeclSpec.h" #include "llvm/ADT/STLExtras.h" using namespace clang; /// ActOnCXXGlobalScopeSpecifier - Return the object that represents the /// global scope ('::'). Sema::CXXScopeTy *Sema::ActOnCXXGlobalScopeSpecifier(Scope *S, SourceLocation CCLoc) { return cast(Context.getTranslationUnitDecl()); } /// ActOnCXXNestedNameSpecifier - Called during parsing of a /// nested-name-specifier. e.g. for "foo::bar::" we parsed "foo::" and now /// we want to resolve "bar::". 'SS' is empty or the previously parsed /// nested-name part ("foo::"), 'IdLoc' is the source location of 'bar', /// 'CCLoc' is the location of '::' and 'II' is the identifier for 'bar'. /// Returns a CXXScopeTy* object representing the C++ scope. Sema::CXXScopeTy *Sema::ActOnCXXNestedNameSpecifier(Scope *S, const CXXScopeSpec &SS, SourceLocation IdLoc, SourceLocation CCLoc, IdentifierInfo &II) { Decl *SD = LookupParsedName(S, &SS, &II, LookupNestedNameSpecifierName); if (SD) { if (TypedefDecl *TD = dyn_cast(SD)) { if (const RecordType* Record = TD->getUnderlyingType()->getAsRecordType()) return cast(Record->getDecl()); } else if (isa(SD) || isa(SD)) { return cast(SD); } // FIXME: Template parameters and dependent types. // FIXME: C++0x scoped enums // Fall through to produce an error: we found something that isn't // a class or a namespace. } // If we didn't find anything during our lookup, try again with // ordinary name lookup, which can help us produce better error // messages. if (!SD) SD = LookupParsedName(S, &SS, &II, LookupOrdinaryName); unsigned DiagID; if (SD) DiagID = diag::err_expected_class_or_namespace; else if (SS.isSet()) DiagID = diag::err_typecheck_no_member; else DiagID = diag::err_undeclared_var_use; if (SS.isSet()) Diag(IdLoc, DiagID) << &II << SS.getRange(); else Diag(IdLoc, DiagID) << &II; return 0; } /// ActOnCXXEnterDeclaratorScope - Called when a C++ scope specifier (global /// scope or nested-name-specifier) is parsed, part of a declarator-id. /// After this method is called, according to [C++ 3.4.3p3], names should be /// looked up in the declarator-id's scope, until the declarator is parsed and /// ActOnCXXExitDeclaratorScope is called. /// The 'SS' should be a non-empty valid CXXScopeSpec. void Sema::ActOnCXXEnterDeclaratorScope(Scope *S, const CXXScopeSpec &SS) { assert(SS.isSet() && "Parser passed invalid CXXScopeSpec."); assert(PreDeclaratorDC == 0 && "Previous declarator context not popped?"); PreDeclaratorDC = static_cast(S->getEntity()); CurContext = static_cast(SS.getScopeRep()); S->setEntity(CurContext); } /// ActOnCXXExitDeclaratorScope - Called when a declarator that previously /// invoked ActOnCXXEnterDeclaratorScope(), is finished. 'SS' is the same /// CXXScopeSpec that was passed to ActOnCXXEnterDeclaratorScope as well. /// Used to indicate that names should revert to being looked up in the /// defining scope. void Sema::ActOnCXXExitDeclaratorScope(Scope *S, const CXXScopeSpec &SS) { assert(SS.isSet() && "Parser passed invalid CXXScopeSpec."); assert(S->getEntity() == SS.getScopeRep() && "Context imbalance!"); S->setEntity(PreDeclaratorDC); PreDeclaratorDC = 0; // Reset CurContext to the nearest enclosing context. while (!S->getEntity() && S->getParent()) S = S->getParent(); CurContext = static_cast(S->getEntity()); }