objc: allow typedef'ing an id to a pointer to a c-struct only.

// rdar://11356439

llvm-svn: 156788
This commit is contained in:
Fariborz Jahanian 2012-05-14 22:48:56 +00:00
parent c10948d02b
commit 16d71bb834
2 changed files with 27 additions and 7 deletions

View File

@ -1518,12 +1518,22 @@ void Sema::MergeTypedefNameDecl(TypedefNameDecl *New, LookupResult &OldDecls) {
switch (TypeID->getLength()) {
default: break;
case 2:
if (!TypeID->isStr("id"))
break;
Context.setObjCIdRedefinitionType(New->getUnderlyingType());
// Install the built-in type for 'id', ignoring the current definition.
New->setTypeForDecl(Context.getObjCIdType().getTypePtr());
return;
{
if (!TypeID->isStr("id"))
break;
QualType T = New->getUnderlyingType();
if (!T->isPointerType())
break;
if (!T->isVoidPointerType()) {
QualType PT = T->getAs<PointerType>()->getPointeeType();
if (!PT->isStructureType())
break;
}
Context.setObjCIdRedefinitionType(T);
// Install the built-in type for 'id', ignoring the current definition.
New->setTypeForDecl(Context.getObjCIdType().getTypePtr());
return;
}
case 5:
if (!TypeID->isStr("Class"))
break;

View File

@ -16,6 +16,16 @@ void foo() {
}
// Test attempt to redefine 'id' in an incompatible fashion.
typedef int id; // FIXME: Decide how we want to deal with this (now that 'id' is more of a built-in type).
// rdar://11356439
typedef int id; // expected-error {{typedef redefinition with different types ('int' vs 'id')}}
id b;
typedef double id; // expected-error {{typedef redefinition with different types ('double' vs 'id')}}
typedef char *id; // expected-error {{typedef redefinition with different types ('char *' vs 'id')}}
typedef union U{ int iu; } *id; // expected-error {{typedef redefinition with different types ('union U *' vs 'id')}}
void test11356439(id o) {
o->x; // expected-error {{member reference base type 'id' is not a structure or union}}
}