Implement Sema support for C++ nested-name-specifiers.

git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@58916 91177308-0d34-0410-b5e6-96231b3b80d8
diff --git a/lib/Sema/SemaCXXScopeSpec.cpp b/lib/Sema/SemaCXXScopeSpec.cpp
new file mode 100644
index 0000000..1480527
--- /dev/null
+++ b/lib/Sema/SemaCXXScopeSpec.cpp
@@ -0,0 +1,131 @@
+//===--- 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 "clang/Basic/Diagnostic.h"

+using namespace clang;

+

+

+namespace {

+  Decl *LookupNestedName(DeclContext *LookupCtx, bool LookInParentCtx,

+                         const IdentifierInfo &II, bool &IdIsUndeclared) {

+    IdentifierResolver::iterator

+      I = IdentifierResolver::begin(&II, 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 (TypedefDecl *TD = dyn_cast<TypedefDecl>(*I)) {

+        if (TD->getUnderlyingType()->isRecordType())

+          break;

+        continue;

+      }

+      if (((*I)->getIdentifierNamespace()&Decl::IDNS_Tag) && !isa<EnumDecl>(*I))

+        break;    

+    }

+

+    return (I != E ? *I : 0);

+  }

+} // anonymous namespace

+

+/// ActOnCXXGlobalScopeSpecifier - Return the object that represents the

+/// global scope ('::').

+Sema::CXXScopeTy *Sema::ActOnCXXGlobalScopeSpecifier(Scope *S,

+                                                     SourceLocation CCLoc) {

+  return cast<DeclContext>(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,

+                                                    const IdentifierInfo &II) {

+  DeclContext *DC = static_cast<DeclContext*>(SS.getScopeRep());

+  Decl *SD;

+  bool IdIsUndeclared;

+

+  if (DC)

+    SD = LookupNestedName(DC, false/*LookInParentCtx*/, II, IdIsUndeclared);

+  else

+    SD = LookupNestedName(CurContext, true/*LookInParent*/, II, IdIsUndeclared);

+

+  if (SD) {

+    if (TypedefDecl *TD = dyn_cast<TypedefDecl>(SD)) {

+      assert(TD->getUnderlyingType()->isRecordType() &&"Invalid Scope Decl!");

+      SD = TD->getUnderlyingType()->getAsRecordType()->getDecl();

+    }

+

+    assert((isa<NamespaceDecl>(SD) || isa<RecordDecl>(SD)) &&

+           "Invalid Scope Decl!");

+    return cast<DeclContext>(SD);

+  }

+

+  unsigned DiagID;

+  if (!IdIsUndeclared)

+    DiagID = diag::err_expected_class_or_namespace;

+  else if (DC)

+    DiagID = diag::err_typecheck_no_member;

+  else

+    DiagID = diag::err_undeclared_var_use;

+

+  if (DC)

+    Diag(IdLoc, DiagID, II.getName(), SS.getRange());

+  else

+    Diag(IdLoc, DiagID, II.getName());

+

+  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 = CurContext;

+  CurContext = static_cast<DeclContext*>(SS.getScopeRep());

+}

+

+/// 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(const CXXScopeSpec &SS) {

+  assert(SS.isSet() && "Parser passed invalid CXXScopeSpec.");

+  assert(CurContext == static_cast<DeclContext*>(SS.getScopeRep()) &&

+         "Context imbalance!");

+  CurContext = PreDeclaratorDC;

+  PreDeclaratorDC = 0;

+}