From 189d41f625ebe2bfbb6dda5657cd4cb819dcd2b3 Mon Sep 17 00:00:00 2001 From: Steve Naroff Date: Wed, 4 Feb 2009 17:14:05 +0000 Subject: [PATCH] Fix error: redefinition of 'XCElementAnchorDelegate' as different kind of symbol. At first glance, this looked like a recent regression (possibly created by http://llvm.org/viewvc/llvm-project?view=rev&revision=63354, which was the only recent change to this section of Sema::ActOnStartClassInterface()). After more investigation, it looks like an edge case bug that we didn't cover in our tests. llvm-svn: 63738 --- clang/lib/Sema/SemaDeclObjC.cpp | 39 ++++++++++++++++++-------- clang/test/SemaObjC/class-def-test-1.m | 7 +++++ 2 files changed, 34 insertions(+), 12 deletions(-) diff --git a/clang/lib/Sema/SemaDeclObjC.cpp b/clang/lib/Sema/SemaDeclObjC.cpp index 3b0265e7033f..7c76f1627550 100644 --- a/clang/lib/Sema/SemaDeclObjC.cpp +++ b/clang/lib/Sema/SemaDeclObjC.cpp @@ -105,26 +105,41 @@ ActOnStartClassInterface(SourceLocation AtInterfaceLoc, } if (SuperName) { - ObjCInterfaceDecl* SuperClassEntry = 0; // Check if a different kind of symbol declared in this scope. PrevDecl = LookupName(TUScope, SuperName, LookupOrdinaryName); - if (PrevDecl && !isa(PrevDecl)) { - Diag(SuperLoc, diag::err_redefinition_different_kind) << SuperName; - Diag(PrevDecl->getLocation(), diag::note_previous_definition); - } - else { - // Check that super class is previously defined - SuperClassEntry = dyn_cast_or_null(PrevDecl); - if (!SuperClassEntry) + ObjCInterfaceDecl *SuperClassDecl = + dyn_cast_or_null(PrevDecl); + if (PrevDecl && SuperClassDecl == 0) { + // The previous declaration was not a class decl. Check if we have a + // typedef. If we do, get the underlying class type. + if (const TypedefDecl *TDecl = dyn_cast_or_null(PrevDecl)) { + QualType T = TDecl->getUnderlyingType(); + if (T->isObjCInterfaceType()) { + if (NamedDecl *IDecl = T->getAsObjCInterfaceType()->getDecl()) + SuperClassDecl = dyn_cast(IDecl); + } + } + // This handles the following case: + // + // typedef int SuperClass; + // @interface MyClass : SuperClass {} @end + // + if (!SuperClassDecl) { + Diag(SuperLoc, diag::err_redefinition_different_kind) << SuperName; + Diag(PrevDecl->getLocation(), diag::note_previous_definition); + } + } + if (!dyn_cast_or_null(PrevDecl)) { + if (!SuperClassDecl) Diag(SuperLoc, diag::err_undef_superclass) << SuperName << ClassName << SourceRange(AtInterfaceLoc, ClassLoc); - else if (SuperClassEntry->isForwardDecl()) + else if (SuperClassDecl->isForwardDecl()) Diag(SuperLoc, diag::err_undef_superclass) - << SuperClassEntry->getDeclName() << ClassName + << SuperClassDecl->getDeclName() << ClassName << SourceRange(AtInterfaceLoc, ClassLoc); } - IDecl->setSuperClass(SuperClassEntry); + IDecl->setSuperClass(SuperClassDecl); IDecl->setSuperClassLoc(SuperLoc); IDecl->setLocEnd(SuperLoc); } else { // we have a root class. diff --git a/clang/test/SemaObjC/class-def-test-1.m b/clang/test/SemaObjC/class-def-test-1.m index ef226a515f1c..50493f1b8651 100644 --- a/clang/test/SemaObjC/class-def-test-1.m +++ b/clang/test/SemaObjC/class-def-test-1.m @@ -24,3 +24,10 @@ typedef int OBJECT; // expected-note {{previous definition is here}} \ @interface INTF3 : PROTO @end // expected-error {{cannot find interface declaration for 'PROTO', superclass of 'INTF3'}} +// Make sure we allow the following (for GCC compatibility). +@interface NSObject @end +typedef NSObject TD_NSObject; +@interface XCElementUnit : TD_NSObject {} +@end + +