aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFariborz Jahanian <fjahanian@apple.com>2008-12-16 01:08:35 +0000
committerFariborz Jahanian <fjahanian@apple.com>2008-12-16 01:08:35 +0000
commit3281eff4c1fbf7e2e59cc5381d5270deaf06839e (patch)
tree82ef3bb2eaf10c2d5ad6148532a811556ac1a7ff
parent0a59acb9ae36077ce46fb2807956c5e84f0f6837 (diff)
Diagnose that ivars in current and super class may not
be duplicates and a test case. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@61068 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r--lib/AST/DeclObjC.cpp6
-rw-r--r--lib/Sema/SemaDecl.cpp19
-rw-r--r--test/SemaObjC/duplicate-ivar-check.m20
3 files changed, 43 insertions, 2 deletions
diff --git a/lib/AST/DeclObjC.cpp b/lib/AST/DeclObjC.cpp
index 7e2a99fb26..9e803b71e1 100644
--- a/lib/AST/DeclObjC.cpp
+++ b/lib/AST/DeclObjC.cpp
@@ -345,7 +345,8 @@ void ObjCInterfaceDecl::CollectObjCIvars(std::vector<FieldDecl*> &Fields) {
for (ObjCInterfaceDecl::ivar_iterator I = ivar_begin(),
E = ivar_end(); I != E; ++I) {
ObjCIvarDecl *IVDecl = (*I);
- Fields.push_back(cast<FieldDecl>(IVDecl));
+ if (!IVDecl->isInvalidDecl())
+ Fields.push_back(cast<FieldDecl>(IVDecl));
}
}
@@ -390,7 +391,8 @@ void ObjCInterfaceDecl::addRecordToClass(ASTContext &Context)
/// FIXME! Can do collection of ivars and adding to the record while
/// doing it.
for (unsigned int i = 0; i != RecFields.size(); i++) {
- FieldDecl *Field = FieldDecl::Create(Context, RD, SourceLocation(),
+ FieldDecl *Field = FieldDecl::Create(Context, RD,
+ RecFields[i]->getLocation(),
RecFields[i]->getIdentifier(),
RecFields[i]->getType(),
RecFields[i]->getBitWidth(), false, 0);
diff --git a/lib/Sema/SemaDecl.cpp b/lib/Sema/SemaDecl.cpp
index 5e0edcd1ec..0d48c62612 100644
--- a/lib/Sema/SemaDecl.cpp
+++ b/lib/Sema/SemaDecl.cpp
@@ -3091,6 +3091,25 @@ void Sema::ActOnFields(Scope* S,
if (ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(EnclosingDecl)) {
ID->addInstanceVariablesToClass(ClsFields, RecFields.size(), RBrac);
ID->addRecordToClass(Context);
+ // Must enforce the rule that ivars in the base classes may not be
+ // duplicates.
+ FieldIDs.clear();
+ RecordDecl *RD = ID->getRecordForDecl();
+ for (RecordDecl::field_iterator i = RD->field_begin(),
+ e = RD->field_end(); i != e; ++i) {
+ FieldDecl *FD = *i;
+ if (IdentifierInfo *II = FD->getIdentifier())
+ if (!FieldIDs.insert(II)) {
+ Diag(FD->getLocation(), diag::err_duplicate_member) << II;
+ FD->setInvalidDecl();
+ for (RecordDecl::field_iterator j = RD->field_begin(),
+ e1 = RD->field_end(); j != e1; ++j)
+ if (II == (*j)->getIdentifier()) {
+ Diag((*j)->getLocation(), diag::note_previous_definition);
+ break;
+ }
+ }
+ }
}
else if (ObjCImplementationDecl *IMPDecl =
dyn_cast<ObjCImplementationDecl>(EnclosingDecl)) {
diff --git a/test/SemaObjC/duplicate-ivar-check.m b/test/SemaObjC/duplicate-ivar-check.m
new file mode 100644
index 0000000000..ea42e9e277
--- /dev/null
+++ b/test/SemaObjC/duplicate-ivar-check.m
@@ -0,0 +1,20 @@
+// RUN: clang -fsyntax-only -verify %s
+
+@interface B1 {
+@public
+ double fill_B; // expected-note {{previous definition is here}}
+}
+@end
+
+@interface B : B1 {
+@public
+ int one; // expected-note {{previous definition is here}}
+ int one; // expected-error {{duplicate member 'one'}}
+}
+@end
+
+@interface A : B {
+@public
+ int fill_B; // expected-error {{duplicate member 'fill_B'}}
+}
+@end