aboutsummaryrefslogtreecommitdiff
path: root/lib/Sema/SemaDeclCXX.cpp
diff options
context:
space:
mode:
authorSebastian Redl <sebastian.redl@getdesigned.at>2008-12-22 19:15:10 +0000
committerSebastian Redl <sebastian.redl@getdesigned.at>2008-12-22 19:15:10 +0000
commit4b07b2968f87f3cd5a3d8c76145f1cbfd718d42d (patch)
treec92a07839c5ff786350b17c99a1d7a68838a17fe /lib/Sema/SemaDeclCXX.cpp
parent804058ece0d8f692faac8518ce4d98975ba57ac2 (diff)
Partial AST and Sema support for C++ try-catch.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@61337 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Sema/SemaDeclCXX.cpp')
-rw-r--r--lib/Sema/SemaDeclCXX.cpp63
1 files changed, 63 insertions, 0 deletions
diff --git a/lib/Sema/SemaDeclCXX.cpp b/lib/Sema/SemaDeclCXX.cpp
index f2f02e2125..4ac7ecb8af 100644
--- a/lib/Sema/SemaDeclCXX.cpp
+++ b/lib/Sema/SemaDeclCXX.cpp
@@ -2026,3 +2026,66 @@ Sema::DeclTy *Sema::ActOnLinkageSpec(SourceLocation Loc,
return LinkageSpecDecl::Create(Context, Loc, Language, dcl);
}
+/// ActOnExceptionDeclarator - Parsed the exception-declarator in a C++ catch
+/// handler.
+Sema::DeclTy *Sema::ActOnExceptionDeclarator(Scope *S, Declarator &D)
+{
+ QualType ExDeclType = GetTypeForDeclarator(D, S);
+ SourceLocation Begin = D.getDeclSpec().getSourceRange().getBegin();
+
+ bool Invalid = false;
+
+ // Arrays and functions decay.
+ if (ExDeclType->isArrayType())
+ ExDeclType = Context.getArrayDecayedType(ExDeclType);
+ else if (ExDeclType->isFunctionType())
+ ExDeclType = Context.getPointerType(ExDeclType);
+
+ // C++ 15.3p1: The exception-declaration shall not denote an incomplete type.
+ // The exception-declaration shall not denote a pointer or reference to an
+ // incomplete type, other than [cv] void*.
+ QualType BaseType = ExDeclType;
+ int Mode = 0; // 0 for direct type, 1 for pointer, 2 for reference
+ if (const PointerType *Ptr = BaseType->getAsPointerType()) {
+ BaseType = Ptr->getPointeeType();
+ Mode = 1;
+ } else if(const ReferenceType *Ref = BaseType->getAsReferenceType()) {
+ BaseType = Ref->getPointeeType();
+ Mode = 2;
+ }
+ if ((Mode == 0 || !BaseType->isVoidType()) && BaseType->isIncompleteType()) {
+ Invalid = true;
+ Diag(Begin, diag::err_catch_incomplete) << BaseType << Mode;
+ }
+
+ IdentifierInfo *II = D.getIdentifier();
+ if (Decl *PrevDecl = LookupDecl(II, Decl::IDNS_Ordinary, S)) {
+ // The scope should be freshly made just for us. There is just no way
+ // it contains any previous declaration.
+ assert(!S->isDeclScope(PrevDecl));
+ if (PrevDecl->isTemplateParameter()) {
+ // Maybe we will complain about the shadowed template parameter.
+ DiagnoseTemplateParameterShadow(D.getIdentifierLoc(), PrevDecl);
+
+ }
+ }
+
+ VarDecl *ExDecl = VarDecl::Create(Context, CurContext, D.getIdentifierLoc(),
+ II, ExDeclType, VarDecl::None, 0, Begin);
+ if (D.getInvalidType() || Invalid)
+ ExDecl->setInvalidDecl();
+
+ if (D.getCXXScopeSpec().isSet()) {
+ Diag(D.getIdentifierLoc(), diag::err_qualified_catch_declarator)
+ << D.getCXXScopeSpec().getRange();
+ ExDecl->setInvalidDecl();
+ }
+
+ // Add the exception declaration into this scope.
+ S->AddDecl(ExDecl);
+ if (II)
+ IdResolver.AddDecl(ExDecl);
+
+ ProcessDeclAttributes(ExDecl, D);
+ return ExDecl;
+}