// RUN: %clang_cc1 -std=c2y -triple x86_64-apple-darwin -Wno-objc-root-class -emit-llvm -o - %s | FileCheck %s int g(id x); // CHECK-LABEL: define void @f1(ptr {{.*}} %y) // CHECK: entry: // CHECK: %y.addr = alloca ptr, align 8 // CHECK: %x1 = alloca ptr, align 8 // CHECK: %state.ptr = alloca %struct.__objcFastEnumerationState, align 8 // CHECK: %items.ptr = alloca [16 x ptr], align 8 // CHECK: store ptr %y, ptr %y.addr, align 8 // CHECK: br label %x // CHECK: x: // CHECK: call void @llvm.memset.p0.i64(ptr align 8 %state.ptr, i8 0, i64 64, i1 false) // CHECK: %0 = load ptr, ptr %y.addr, align 8 // CHECK: %1 = load ptr, ptr @OBJC_SELECTOR_REFERENCES_, align 8 // CHECK: %call = call i64 @objc_msgSend(ptr {{.*}} %0, ptr {{.*}} %1, ptr {{.*}} %state.ptr, ptr {{.*}} %items.ptr, i64 {{.*}} 16) // CHECK: %iszero = icmp eq i64 %call, 0 // CHECK: br i1 %iszero, label %forcoll.empty, label %forcoll.loopinit // CHECK: forcoll.loopinit: // CHECK: %mutationsptr.ptr = getelementptr inbounds nuw %struct.__objcFastEnumerationState, ptr %state.ptr, i32 0, i32 2 // CHECK: %mutationsptr = load ptr, ptr %mutationsptr.ptr, align 8 // CHECK: %forcoll.initial-mutations = load i64, ptr %mutationsptr, align 8 // CHECK: br label %forcoll.loopbody // CHECK: forcoll.loopbody: // CHECK: %forcoll.index = phi i64 [ 0, %forcoll.loopinit ], [ %6, %forcoll.next ], [ 0, %forcoll.refetch ] // CHECK: %forcoll.count = phi i64 [ %call, %forcoll.loopinit ], [ %forcoll.count, %forcoll.next ], [ %call8, %forcoll.refetch ] // CHECK: %mutationsptr2 = load ptr, ptr %mutationsptr.ptr, align 8 // CHECK: %statemutations = load i64, ptr %mutationsptr2, align 8 // CHECK: %2 = icmp eq i64 %statemutations, %forcoll.initial-mutations // CHECK: br i1 %2, label %forcoll.notmutated, label %forcoll.mutated // CHECK: forcoll.mutated: // CHECK: call void @objc_enumerationMutation(ptr {{.*}} %0) // CHECK: br label %forcoll.notmutated // CHECK: forcoll.notmutated: // CHECK: %stateitems.ptr = getelementptr inbounds nuw %struct.__objcFastEnumerationState, ptr %state.ptr, i32 0, i32 1 // CHECK: %stateitems = load ptr, ptr %stateitems.ptr, align 8 // CHECK: %currentitem.ptr = getelementptr inbounds ptr, ptr %stateitems, i64 %forcoll.index // CHECK: %3 = load ptr, ptr %currentitem.ptr, align 8 // CHECK: store ptr %3, ptr %x1, align 8 // CHECK: %4 = load ptr, ptr %x1, align 8 // CHECK: %call3 = call i32 @g(ptr {{.*}} %4) // CHECK: %tobool = icmp ne i32 %call3, 0 // CHECK: br i1 %tobool, label %if.then, label %if.end // CHECK: if.then: // CHECK: br label %forcoll.end // CHECK: if.end: // CHECK: %5 = load ptr, ptr %x1, align 8 // CHECK: %call4 = call i32 @g(ptr {{.*}} %5) // CHECK: %tobool5 = icmp ne i32 %call4, 0 // CHECK: br i1 %tobool5, label %if.then6, label %if.end7 // CHECK: if.then6: // CHECK: br label %forcoll.next // CHECK: if.end7: // CHECK: br label %forcoll.next // CHECK: forcoll.next: // CHECK: %6 = add nuw i64 %forcoll.index, 1 // CHECK: %7 = icmp ult i64 %6, %forcoll.count // CHECK: br i1 %7, label %forcoll.loopbody, label %forcoll.refetch // CHECK: forcoll.refetch: // CHECK: %8 = load ptr, ptr @OBJC_SELECTOR_REFERENCES_, align 8 // CHECK: %call8 = call i64 @objc_msgSend(ptr {{.*}} %0, ptr {{.*}} %8, ptr {{.*}} %state.ptr, ptr {{.*}} %items.ptr, i64 {{.*}} 16) // CHECK: %9 = icmp eq i64 %call8, 0 // CHECK: br i1 %9, label %forcoll.empty, label %forcoll.loopbody // CHECK: forcoll.empty: // CHECK: br label %forcoll.end // CHECK: forcoll.end: // CHECK: ret void void f1(id y) { x: for (id x in y) { if (g(x)) break x; if (g(x)) continue x; } } // CHECK-LABEL: define void @f2(ptr {{.*}} %y) // CHECK: entry: // CHECK: %y.addr = alloca ptr, align 8 // CHECK: %x = alloca ptr, align 8 // CHECK: %state.ptr = alloca %struct.__objcFastEnumerationState, align 8 // CHECK: %items.ptr = alloca [16 x ptr], align 8 // CHECK: store ptr %y, ptr %y.addr, align 8 // CHECK: br label %a // CHECK: a: // CHECK: br label %while.cond // CHECK: while.cond: // CHECK: %0 = load ptr, ptr %y.addr, align 8 // CHECK: %call = call i32 @g(ptr {{.*}} %0) // CHECK: %tobool = icmp ne i32 %call, 0 // CHECK: br i1 %tobool, label %while.body, label %while.end // CHECK: while.body: // CHECK: br label %b // CHECK: b: // CHECK: call void @llvm.memset.p0.i64(ptr align 8 %state.ptr, i8 0, i64 64, i1 false) // CHECK: %1 = load ptr, ptr %y.addr, align 8 // CHECK: %2 = load ptr, ptr @OBJC_SELECTOR_REFERENCES_, align 8 // CHECK: %call1 = call i64 @objc_msgSend(ptr {{.*}} %1, ptr {{.*}} %2, ptr {{.*}} %state.ptr, ptr {{.*}} %items.ptr, i64 {{.*}} 16) // CHECK: %iszero = icmp eq i64 %call1, 0 // CHECK: br i1 %iszero, label %forcoll.empty, label %forcoll.loopinit // CHECK: forcoll.loopinit: // CHECK: %mutationsptr.ptr = getelementptr inbounds nuw %struct.__objcFastEnumerationState, ptr %state.ptr, i32 0, i32 2 // CHECK: %mutationsptr = load ptr, ptr %mutationsptr.ptr, align 8 // CHECK: %forcoll.initial-mutations = load i64, ptr %mutationsptr, align 8 // CHECK: br label %forcoll.loopbody // CHECK: forcoll.loopbody: // CHECK: %forcoll.index = phi i64 [ 0, %forcoll.loopinit ], [ %9, %forcoll.next ], [ 0, %forcoll.refetch ] // CHECK: %forcoll.count = phi i64 [ %call1, %forcoll.loopinit ], [ %forcoll.count, %forcoll.next ], [ %call17, %forcoll.refetch ] // CHECK: %mutationsptr2 = load ptr, ptr %mutationsptr.ptr, align 8 // CHECK: %statemutations = load i64, ptr %mutationsptr2, align 8 // CHECK: %3 = icmp eq i64 %statemutations, %forcoll.initial-mutations // CHECK: br i1 %3, label %forcoll.notmutated, label %forcoll.mutated // CHECK: forcoll.mutated: // CHECK: call void @objc_enumerationMutation(ptr {{.*}} %1) // CHECK: br label %forcoll.notmutated // CHECK: forcoll.notmutated: // CHECK: %stateitems.ptr = getelementptr inbounds nuw %struct.__objcFastEnumerationState, ptr %state.ptr, i32 0, i32 1 // CHECK: %stateitems = load ptr, ptr %stateitems.ptr, align 8 // CHECK: %currentitem.ptr = getelementptr inbounds ptr, ptr %stateitems, i64 %forcoll.index // CHECK: %4 = load ptr, ptr %currentitem.ptr, align 8 // CHECK: store ptr %4, ptr %x, align 8 // CHECK: %5 = load ptr, ptr %x, align 8 // CHECK: %call3 = call i32 @g(ptr {{.*}} %5) // CHECK: %tobool4 = icmp ne i32 %call3, 0 // CHECK: br i1 %tobool4, label %if.then, label %if.end // CHECK: if.then: // CHECK: br label %while.end // CHECK: if.end: // CHECK: %6 = load ptr, ptr %x, align 8 // CHECK: %call5 = call i32 @g(ptr {{.*}} %6) // CHECK: %tobool6 = icmp ne i32 %call5, 0 // CHECK: br i1 %tobool6, label %if.then7, label %if.end8 // CHECK: if.then7: // CHECK: br label %while.cond // CHECK: if.end8: // CHECK: %7 = load ptr, ptr %x, align 8 // CHECK: %call9 = call i32 @g(ptr {{.*}} %7) // CHECK: %tobool10 = icmp ne i32 %call9, 0 // CHECK: br i1 %tobool10, label %if.then11, label %if.end12 // CHECK: if.then11: // CHECK: br label %forcoll.end // CHECK: if.end12: // CHECK: %8 = load ptr, ptr %x, align 8 // CHECK: %call13 = call i32 @g(ptr {{.*}} %8) // CHECK: %tobool14 = icmp ne i32 %call13, 0 // CHECK: br i1 %tobool14, label %if.then15, label %if.end16 // CHECK: if.then15: // CHECK: br label %forcoll.next // CHECK: if.end16: // CHECK: br label %forcoll.next // CHECK: forcoll.next: // CHECK: %9 = add nuw i64 %forcoll.index, 1 // CHECK: %10 = icmp ult i64 %9, %forcoll.count // CHECK: br i1 %10, label %forcoll.loopbody, label %forcoll.refetch // CHECK: forcoll.refetch: // CHECK: %11 = load ptr, ptr @OBJC_SELECTOR_REFERENCES_, align 8 // CHECK: %call17 = call i64 @objc_msgSend(ptr {{.*}} %1, ptr {{.*}} %11, ptr {{.*}} %state.ptr, ptr {{.*}} %items.ptr, i64 {{.*}} 16) // CHECK: %12 = icmp eq i64 %call17, 0 // CHECK: br i1 %12, label %forcoll.empty, label %forcoll.loopbody // CHECK: forcoll.empty: // CHECK: br label %forcoll.end // CHECK: forcoll.end: // CHECK: br label %while.cond // CHECK: while.end: // CHECK: ret void void f2(id y) { a: while (g(y)) { b: for (id x in y) { if (g(x)) break a; if (g(x)) continue a; if (g(x)) break b; if (g(x)) continue b; } } }