From 34074326449e1e6b7de77a4c9a0d78ea4b54ca64 Mon Sep 17 00:00:00 2001 From: Douglas Gregor Date: Wed, 14 Jan 2009 22:20:51 +0000 Subject: Refactor name lookup. This change refactors and cleans up our handling of name lookup with LookupDecl. There are several aspects to this refactoring: - The criteria for name lookup is now encapsulated into the class LookupCriteria, which replaces the hideous set of boolean values that LookupDecl currently has. - The results of name lookup are returned in a new class LookupResult, which can lazily build OverloadedFunctionDecls for overloaded function sets (and, eventually, eliminate the need to allocate member for OverloadedFunctionDecls) and contains a placeholder for handling ambiguous name lookup (for C++). - The primary entry points for name lookup are now LookupName (for unqualified name lookup) and LookupQualifiedName (for qualified name lookup). There is also a convenience function LookupParsedName that handles qualified/unqualified name lookup when given a scope specifier. Together, these routines are meant to gradually replace the kludgy LookupDecl, but this won't happen until after we have base class lookup (which forces us to cope with ambiguities). - Documented the heck out of name lookup. Experimenting a little with using Doxygen's member groups to make some sense of the Sema class. Feedback welcome! - Fixes some lingering issues with name lookup for nested-name-specifiers, which now goes through LookupName/LookupQualifiedName. llvm-svn: 62245 --- clang/lib/Sema/SemaCXXScopeSpec.cpp | 78 ++++++++----------------------------- 1 file changed, 17 insertions(+), 61 deletions(-) (limited to 'clang/lib/Sema/SemaCXXScopeSpec.cpp') diff --git a/clang/lib/Sema/SemaCXXScopeSpec.cpp b/clang/lib/Sema/SemaCXXScopeSpec.cpp index dcb2a5fba55..0c0fa35a5e7 100644 --- a/clang/lib/Sema/SemaCXXScopeSpec.cpp +++ b/clang/lib/Sema/SemaCXXScopeSpec.cpp @@ -19,54 +19,6 @@ using namespace clang; -namespace { - Decl *LookupNestedName(DeclContext *LookupCtx, bool LookInParentCtx, - DeclarationName Name, bool &IdIsUndeclared, - ASTContext &Context) { - if (LookupCtx && !LookInParentCtx) { - IdIsUndeclared = true; - DeclContext::lookup_const_iterator I, E; - for (llvm::tie(I, E) = LookupCtx->lookup(Name); I != E; ++I) { - IdIsUndeclared = false; - if (((*I)->isInIdentifierNamespace(Decl::IDNS_Tag)) || - isa(*I)) - return *I; - } - - return 0; - } - - // FIXME: Decouple this from the IdentifierResolver so that we can - // deal with lookups into the semantic parent contexts that aren't - // lexical parent contexts. - - IdentifierResolver::iterator - I = IdentifierResolver::begin(Name, LookupCtx, LookInParentCtx), - E = IdentifierResolver::end(); - - if (I == E) { - IdIsUndeclared = true; - return 0; - } - IdIsUndeclared = false; - - // C++ 3.4.3p1 : - // During the lookup for a name preceding the :: scope resolution operator, - // object, function, and enumerator names are ignored. If the name found is - // not a class-name or namespace-name, the program is ill-formed. - - for (; I != E; ++I) { - if (isa(*I)) { - break; - } - if (((*I)->getIdentifierNamespace() & Decl::IDNS_Tag)) - break; - } - - return (I != E ? *I : 0); - } -} // anonymous namespace - /// ActOnCXXGlobalScopeSpecifier - Return the object that represents the /// global scope ('::'). Sema::CXXScopeTy *Sema::ActOnCXXGlobalScopeSpecifier(Scope *S, @@ -85,16 +37,10 @@ Sema::CXXScopeTy *Sema::ActOnCXXNestedNameSpecifier(Scope *S, SourceLocation IdLoc, SourceLocation CCLoc, IdentifierInfo &II) { - DeclContext *DC = static_cast(SS.getScopeRep()); - Decl *SD; - bool IdIsUndeclared; - - if (DC) - SD = LookupNestedName(DC, false/*LookInParentCtx*/, &II, IdIsUndeclared, - Context); - else - SD = LookupNestedName(CurContext, true/*LookInParent*/, &II, - IdIsUndeclared, Context); + Decl *SD = LookupParsedName(S, SS, &II, + LookupCriteria(LookupCriteria::NestedNameSpecifier, + /*RedeclarationOnly=*/false, + /*CPlusPlus=*/true)); if (SD) { if (TypedefDecl *TD = dyn_cast(SD)) { @@ -104,19 +50,29 @@ Sema::CXXScopeTy *Sema::ActOnCXXNestedNameSpecifier(Scope *S, return cast(SD); } + // 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, + LookupCriteria(LookupCriteria::Ordinary, + /*RedeclarationOnly=*/false, + /*CPlusPlus=*/true)); unsigned DiagID; - if (!IdIsUndeclared) + if (SD) DiagID = diag::err_expected_class_or_namespace; - else if (DC) + else if (SS.isSet()) DiagID = diag::err_typecheck_no_member; else DiagID = diag::err_undeclared_var_use; - if (DC) + if (SS.isSet()) Diag(IdLoc, DiagID) << &II << SS.getRange(); else Diag(IdLoc, DiagID) << &II; -- cgit v1.2.3