зеркало из https://github.com/microsoft/clang-1.git
Fix <rdar://problem/6574319> clang issues error on 'readonly' property with a defaul setter attribute.
Needed to make isPropertyReadonly() non-const (for this fix to compile). I imagine there's a way to retain the const-ness, however I have more important fish to fry. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@65562 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Родитель
24a9f6e11d
Коммит
22dc0b0435
|
@ -887,7 +887,7 @@ public:
|
||||||
NamespaceDecl *GetStdNamespace();
|
NamespaceDecl *GetStdNamespace();
|
||||||
|
|
||||||
bool isPropertyReadonly(ObjCPropertyDecl *PropertyDecl,
|
bool isPropertyReadonly(ObjCPropertyDecl *PropertyDecl,
|
||||||
ObjCInterfaceDecl *IDecl) const;
|
ObjCInterfaceDecl *IDecl);
|
||||||
|
|
||||||
/// CheckProtocolMethodDefs - This routine checks unimplemented
|
/// CheckProtocolMethodDefs - This routine checks unimplemented
|
||||||
/// methods declared in protocol, and those referenced by it.
|
/// methods declared in protocol, and those referenced by it.
|
||||||
|
|
|
@ -720,7 +720,7 @@ void Sema::WarnConflictingTypedMethods(ObjCMethodDecl *ImpMethodDecl,
|
||||||
/// for the property in the class and in its categories and implementations
|
/// for the property in the class and in its categories and implementations
|
||||||
///
|
///
|
||||||
bool Sema::isPropertyReadonly(ObjCPropertyDecl *PDecl,
|
bool Sema::isPropertyReadonly(ObjCPropertyDecl *PDecl,
|
||||||
ObjCInterfaceDecl *IDecl) const {
|
ObjCInterfaceDecl *IDecl) {
|
||||||
// by far the most common case.
|
// by far the most common case.
|
||||||
if (!PDecl->isReadOnly())
|
if (!PDecl->isReadOnly())
|
||||||
return false;
|
return false;
|
||||||
|
@ -758,6 +758,11 @@ bool Sema::isPropertyReadonly(ObjCPropertyDecl *PDecl,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Lastly, look through the implementation (if one is in scope).
|
||||||
|
if (ObjCImplementationDecl *ImpDecl =
|
||||||
|
ObjCImplementations[IDecl->getIdentifier()])
|
||||||
|
if (ImpDecl->getInstanceMethod(PDecl->getSetterName()))
|
||||||
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,77 @@
|
||||||
|
// RUN: clang -fsyntax-only -verify %s
|
||||||
|
|
||||||
|
@interface NSObject
|
||||||
|
+ alloc;
|
||||||
|
- init;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@protocol Test
|
||||||
|
@property int required;
|
||||||
|
|
||||||
|
@optional
|
||||||
|
@property int optional;
|
||||||
|
@property int optional1;
|
||||||
|
@property int optional_preexisting_setter_getter;
|
||||||
|
@property (setter = setOptional_preexisting_setter_getter: ,
|
||||||
|
getter = optional_preexisting_setter_getter) int optional_with_setter_getter_attr;
|
||||||
|
@required
|
||||||
|
@property int required1;
|
||||||
|
@optional
|
||||||
|
@property int optional_to_be_defined;
|
||||||
|
@property (readonly, getter = optional_preexisting_setter_getter) int optional_getter_attr;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@interface Test : NSObject <Test> {
|
||||||
|
int ivar;
|
||||||
|
int ivar1;
|
||||||
|
int ivar2;
|
||||||
|
}
|
||||||
|
@property int required;
|
||||||
|
@property int optional_to_be_defined;
|
||||||
|
- (int) optional_preexisting_setter_getter;
|
||||||
|
- (void) setOptional_preexisting_setter_getter:(int)value;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation Test
|
||||||
|
@synthesize required = ivar;
|
||||||
|
@synthesize required1 = ivar1;
|
||||||
|
@synthesize optional_to_be_defined = ivar2;
|
||||||
|
- (int) optional_preexisting_setter_getter { return ivar; }
|
||||||
|
- (void) setOptional_preexisting_setter_getter:(int)value
|
||||||
|
{
|
||||||
|
ivar = value;
|
||||||
|
}
|
||||||
|
- (void) setOptional_getter_attr:(int)value { ivar = value; }
|
||||||
|
@end
|
||||||
|
|
||||||
|
int main ()
|
||||||
|
{
|
||||||
|
Test *x = [[Test alloc] init];
|
||||||
|
/* 1. Test of a requred property */
|
||||||
|
x.required1 = 100;
|
||||||
|
if (x.required1 != 100)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
/* 2. Test of a synthesize optional property */
|
||||||
|
x.optional_to_be_defined = 123;
|
||||||
|
if (x.optional_to_be_defined != 123)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
/* 3. Test of optional property with pre-sxisting defined setter/getter */
|
||||||
|
x.optional_preexisting_setter_getter = 200;
|
||||||
|
if (x.optional_preexisting_setter_getter != 200)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
/* 4. Test of optional property with setter/getter attribute */
|
||||||
|
if (x.optional_with_setter_getter_attr != 200)
|
||||||
|
abort ();
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* 5. Test of optional property with getter attribute and default setter method. */
|
||||||
|
x.optional_getter_attr = 1000;
|
||||||
|
if (x.optional_getter_attr != 1000)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
Загрузка…
Ссылка в новой задаче