blob: e9979fe437b616739d61132c20faaf8ed0bd30b3 [file] [log] [blame]
// 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;
}
}
}