// RUN: %clang_cc1 %s -verify -o /dev/null // RUN: %clang_cc1 %s -triple x86_64-apple-darwin -emit-llvm -fsanitize=objc-cast -o - | FileCheck %s void p(const char*, ...); @interface NSArray +(NSArray*) arrayWithObjects: (id) first, ...; -(unsigned) count; @end @interface NSString -(const char*) cString; @end #define S(n) @#n #define L1(n) S(n+0),S(n+1) #define L2(n) L1(n+0),L1(n+2) #define L3(n) L2(n+0),L2(n+4) #define L4(n) L3(n+0),L3(n+8) #define L5(n) L4(n+0),L4(n+16) #define L6(n) L5(n+0),L5(n+32) // CHECK-LABEL: define{{.*}} void @t0 void t0(void) { NSArray *array = [NSArray arrayWithObjects: L1(0), (void*)0]; p("array.length: %d\n", [array count]); unsigned index = 0; for (NSString *i in array) { // expected-warning {{collection expression type 'NSArray *' may not respond}} // CHECK: [[expectedCls:%.*]] = load ptr, {{.*}}, !nosanitize // CHECK-NEXT: [[kindOfClassSel:%.*]] = load ptr, ptr @OBJC_SELECTOR_REFERENCES{{.*}}, !nosanitize // CHECK-NEXT: [[isCls:%.*]] = call zeroext i1 @objc_msgSend(ptr noundef [[theItem:%.*]], ptr noundef [[kindOfClassSel]], ptr noundef [[expectedCls]]), !nosanitize // CHECK: br i1 [[isCls]] // CHECK: ptrtoint ptr [[theItem]] to i64, !nosanitize // CHECK-NEXT: call void @__ubsan_handle_invalid_objc_cast // CHECK-NEXT: unreachable, !nosanitize p("element %d: %s\n", index++, [i cString]); } } void t1(void) { NSArray *array = [NSArray arrayWithObjects: L6(0), (void*)0]; p("array.length: %d\n", [array count]); unsigned index = 0; for (NSString *i in array) { // expected-warning {{collection expression type 'NSArray *' may not respond}} index++; if (index == 10) continue; p("element %d: %s\n", index, [i cString]); if (index == 55) break; } } void t2(NSArray *array) { for (NSArray *array in array) { // expected-warning {{collection expression type 'NSArray *' may not respond}} } }