|
| 1 | +// Check that when Objective-C is first to touch a Swift class, it gives the |
| 2 | +// Swift runtime a chance to update instance size and ivar offset metadata. |
| 3 | + |
| 4 | +// RUN: %empty-directory(%t) |
| 5 | +// RUN: %target-build-swift -emit-library -emit-module -o %t/libResilient.dylib %S/Inputs/class-layout-from-objc/Resilient.swift -Xlinker -install_name -Xlinker @executable_path/libResilient.dylib -Xfrontend -enable-resilience -DSMALL |
| 6 | + |
| 7 | +// RUN: %target-clang -c %S/Inputs/class-layout-from-objc/OneWordSuperclass.m -fmodules -fobjc-arc -o %t/OneWordSuperclass.o |
| 8 | +// RUN: %target-build-swift -emit-library -o %t/libClasses.dylib -emit-objc-header-path %t/Classes.h -I %t -I %S/Inputs/class-layout-from-objc/ %S/Inputs/class-layout-from-objc/Classes.swift %t/OneWordSuperclass.o -Xlinker -install_name -Xlinker @executable_path/libClasses.dylib -lResilient -L %t |
| 9 | +// RUN: %target-clang %s -I %S/Inputs/class-layout-from-objc/ -I %t -fmodules -fobjc-arc -o %t/main -lResilient -lClasses -L %t |
| 10 | +// RUN: %target-codesign %t/main %t/libResilient.dylib %t/libClasses.dylib |
| 11 | +// RUN: %target-run %t/main OLD %t/libResilient.dylib %t/libClasses.dylib |
| 12 | + |
| 13 | +// RUN: %target-build-swift -emit-library -emit-module -o %t/libResilient.dylib %S/Inputs/class-layout-from-objc/Resilient.swift -Xlinker -install_name -Xlinker @executable_path/libResilient.dylib -Xfrontend -enable-resilience -DBIG |
| 14 | +// RUN: %target-codesign %t/libResilient.dylib |
| 15 | +// RUN: %target-run %t/main NEW %t/libResilient.dylib %t/libClasses.dylib |
| 16 | + |
| 17 | +// Try again when the class itself is also resilient. |
| 18 | +// RUN: %target-build-swift -emit-library -o %t/libClasses.dylib -emit-objc-header-path %t/Classes.h -I %S/Inputs/class-layout-from-objc/ -I %t %S/Inputs/class-layout-from-objc/Classes.swift %t/OneWordSuperclass.o -Xlinker -install_name -Xlinker @executable_path/libClasses.dylib -lResilient -L %t |
| 19 | +// RUN: %target-codesign %t/libClasses.dylib |
| 20 | +// RUN: %target-run %t/main OLD %t/libResilient.dylib %t/libClasses.dylib |
| 21 | + |
| 22 | +// RUN: %target-build-swift -emit-library -emit-module -o %t/libResilient.dylib %S/Inputs/class-layout-from-objc/Resilient.swift -Xlinker -install_name -Xlinker @executable_path/libResilient.dylib -Xfrontend -enable-resilience -DSMALL |
| 23 | +// RUN: %target-codesign %t/libResilient.dylib |
| 24 | +// RUN: %target-run %t/main NEW %t/libResilient.dylib %t/libClasses.dylib |
| 25 | + |
| 26 | +// REQUIRES: executable_test |
| 27 | +// REQUIRES: objc_interop |
| 28 | + |
| 29 | +#import <objc/runtime.h> |
| 30 | +#import <assert.h> |
| 31 | +#import <dlfcn.h> |
| 32 | +#import <stdbool.h> |
| 33 | +#import <string.h> |
| 34 | + |
| 35 | +#import "Classes.h" |
| 36 | + |
| 37 | +void check(Class c) { |
| 38 | + assert(c); |
| 39 | + |
| 40 | + size_t expectedSize = [c totalSize]; |
| 41 | + size_t actualSize = class_getInstanceSize([c class]); |
| 42 | + NSLog(@"%@: expected size %zd, actual size %zd", c, expectedSize, actualSize); |
| 43 | + assert(expectedSize == actualSize); |
| 44 | + |
| 45 | + size_t expectedOffsetOfFirst = [c offsetOfFirst]; |
| 46 | + size_t offsetOfFirst = ivar_getOffset(class_getInstanceVariable(c, "first")); |
| 47 | + NSLog(@"expected offset of 'first' %zd, actual %zd", |
| 48 | + expectedOffsetOfFirst, offsetOfFirst); |
| 49 | + assert(offsetOfFirst == expectedOffsetOfFirst); |
| 50 | + |
| 51 | + size_t offsetOfLast = ivar_getOffset(class_getInstanceVariable(c, "last")); |
| 52 | + NSLog(@"offset of 'last' %zd", offsetOfLast); |
| 53 | + assert(offsetOfLast == actualSize - sizeof(intptr_t)); |
| 54 | +} |
| 55 | + |
| 56 | +int main(int argc, const char * const argv[]) { |
| 57 | + assert(argc > 1); |
| 58 | + |
| 59 | + if (!strcmp(argv[1], "OLD")) { |
| 60 | + ; |
| 61 | + } else if (!strcmp(argv[1], "NEW")) { |
| 62 | + // Only test the new behavior on a new enough libobjc. |
| 63 | + if (!dlsym(RTLD_NEXT, "_objc_realizeClassFromSwift")) { |
| 64 | + fprintf(stderr, "skipping evolution tests; OS too old\n"); |
| 65 | + return EXIT_SUCCESS; |
| 66 | + } |
| 67 | + } else { |
| 68 | + fprintf(stderr, "usage: %s (OLD|NEW)\n", argv[0]); |
| 69 | + return EXIT_FAILURE; |
| 70 | + } |
| 71 | + |
| 72 | + @autoreleasepool { |
| 73 | + NSLog(@"%zd", class_getInstanceSize([OneWordSuperclass class])); |
| 74 | + check([StaticClass class]); |
| 75 | + check(objc_getClass("Classes.DynamicClass")); |
| 76 | + check(objc_getClass("Classes.PureSwiftClass")); |
| 77 | + } |
| 78 | +} |
0 commit comments