/* APPLE LOCAL file ObjC GC */ /* A run-time test for insertion of write barriers. */ /* { dg-do run { target *-*-darwin* } } */ /* { dg-options "-fnext-runtime -fobjc-gc" } */ #include #include #include typedef const struct __CFDictionary * CFDictionaryRef; // callouts to these are generated with cc -fobjc-gc int GlobalAssigns; int IvarAssigns; int StrongCastAssigns; id objc_assign_global(id value, id *dest) { ++GlobalAssigns; return (*dest = value); } id objc_assign_ivar(id value, id dest, unsigned int offset) { id *slot = (id*) ((char *)dest + offset); ++IvarAssigns; return (*slot = value); } id objc_assign_strongCast(id value, id *dest) { id base; ++StrongCastAssigns ; return (*dest = value); } // The test case elements; @class NSObject; @class NSString; typedef struct { id element; id elementArray[10]; __strong CFDictionaryRef cfElement; __strong CFDictionaryRef cfElementArray[10]; } struct_with_ids_t; @interface Foo { @public // assignments to any/all of these fields should generate objc_assign_ivar __strong CFDictionaryRef dict; __strong CFDictionaryRef dictArray[3]; id ivar; id array[10]; NSObject *nsobject; NSString *stringArray[10]; struct_with_ids_t inner; } @end // assignments to these should generate objc_assign_global id GlobalId; id GlobalArray[20]; NSObject *GlobalObject; NSObject *GlobalObjectArray[20]; __strong CFDictionaryRef Gdict; __strong CFDictionaryRef Gdictarray[10]; struct_with_ids_t GlobalStruct; struct_with_ids_t GlobalStructArray[10]; // The test cases void *rhs = 0; #define ASSIGNTEST(expr, global) expr = rhs; if (!global) { printf(# expr " is busted\n"); ++counter; } global = 0; int testGlobals() { // Everything in this function generates assign_global intercepts int counter = 0; static id staticGlobalId; static id staticGlobalArray[20]; static NSObject *staticGlobalObject; static NSObject *staticGlobalObjectArray[20]; static __strong CFDictionaryRef staticGdict; static __strong CFDictionaryRef staticGdictarray[10]; static struct_with_ids_t staticGlobalStruct; static struct_with_ids_t staticGlobalStructArray[10]; ASSIGNTEST(GlobalId, GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalObject, GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalObjectArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(Gdict, GlobalAssigns) // objc_assign_global ASSIGNTEST(Gdictarray[1], GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalStruct.element, GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalStruct.elementArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalStruct.cfElement, GlobalAssigns) // objc_assign_global ASSIGNTEST(GlobalStruct.cfElementArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalId, GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalObject, GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalObjectArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGdict, GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGdictarray[1], GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalStruct.element, GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalStruct.elementArray[0], GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalStruct.cfElement, GlobalAssigns) // objc_assign_global ASSIGNTEST(staticGlobalStruct.cfElementArray[0], GlobalAssigns) // objc_assign_global return counter; } int testIvars() { Foo *foo = (Foo *)malloc(sizeof(Foo)); // don't call in ObjC int counter = 0; ASSIGNTEST(foo->ivar, IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->dict, IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->dictArray[0], IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->array[0], IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->nsobject, IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->stringArray[0], IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->inner.element, IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->inner.elementArray[0], IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->inner.cfElement, IvarAssigns) // objc_assign_ivar ASSIGNTEST(foo->inner.cfElementArray[0], IvarAssigns) // objc_assign_ivar return counter; } int testStrongCasts() { id x = nil; int counter = 0; typedef struct { @defs(Foo) } Foo_defs; Foo_defs *foo = (Foo_defs *)malloc(sizeof(Foo)); // strong casts should always be issued, even if the compiler could know better ASSIGNTEST((__strong id)foo->ivar, StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->dict, StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->dictArray[0], StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->array[0], StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->nsobject, StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->stringArray[0], StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->inner.element, StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->inner.elementArray[0], StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->inner.cfElement, StrongCastAssigns) // objc_assign_strongCast ASSIGNTEST((__strong id)foo->inner.cfElementArray[0], StrongCastAssigns) // objc_assign_strongCast // assignments to declared __strong on plain structure elements shouldn't work return counter; } @implementation Foo @end int main(int argc, char *argv[]) { int errors = 0; errors += testGlobals(); errors += testIvars(); errors += testStrongCasts(); return(errors != 0); }