| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --check-globals none --version 6 |
| ; RUN: opt -S < %s -p loop-vectorize -force-vector-width=4 -force-vector-interleave=1 | FileCheck %s |
| |
| declare void @init_mem(ptr, i64) |
| |
| @GlobA = external constant [1024 x i8] |
| |
| define i64 @two_early_exits_same_exit() { |
| ; CHECK-LABEL: define i64 @two_early_exits_same_exit() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ 0, %[[ENTRY]] ], [ [[INDEX_NEXT:%.*]], %[[LOOP_LATCH:.*]] ] |
| ; CHECK-NEXT: [[GEP_A:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD1:%.*]] = load i8, ptr [[GEP_A]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[LD1]], 42 |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[ARRAYIDX1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD2:%.*]] = load i8, ptr [[ARRAYIDX1]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD1]], [[LD2]] |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[INDEX_NEXT]] = add i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EXITCOND:%.*]] = icmp ne i64 [[INDEX_NEXT]], 67 |
| ; CHECK-NEXT: br i1 [[EXITCOND]], label %[[LOOP_HEADER]], label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ], [ 100, %[[EARLY_EXIT_0]] ], [ 43, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ 0, %entry ], [ %iv.next, %loop.latch ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 42 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp2, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 67 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ 100, %early.exit.0 ], [ 43, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define i64 @two_early_exits_different_exits() { |
| ; CHECK-LABEL: define i64 @two_early_exits_different_exits() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ 0, %[[ENTRY]] ], [ [[INDEX_NEXT:%.*]], %[[LOOP_LATCH:.*]] ] |
| ; CHECK-NEXT: [[GEP_A:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD1:%.*]] = load i8, ptr [[GEP_A]], align 1 |
| ; CHECK-NEXT: [[ARRAYIDX1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD2:%.*]] = load i8, ptr [[ARRAYIDX1]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[LD1]], [[LD2]] |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EXIT1:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[LD1]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT2:.*]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[INDEX_NEXT]] = add i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EXITCOND:%.*]] = icmp ne i64 [[INDEX_NEXT]], 67 |
| ; CHECK-NEXT: br i1 [[EXITCOND]], label %[[LOOP_HEADER]], label %[[EXIT:.*]] |
| ; CHECK: [[EXIT1]]: |
| ; CHECK-NEXT: [[RET2:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ] |
| ; CHECK-NEXT: ret i64 [[RET2]] |
| ; CHECK: [[EXIT2]]: |
| ; CHECK-NEXT: [[RET3:%.*]] = phi i64 [ 100, %[[EARLY_EXIT_0]] ] |
| ; CHECK-NEXT: ret i64 [[RET3]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: ret i64 43 |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ 0, %entry ], [ %iv.next, %loop.latch ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit1, label %early.exit.0 |
| |
| early.exit.0: |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit2, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 67 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit1: |
| %ret1 = phi i64 [ %iv, %loop.header ] |
| ret i64 %ret1 |
| |
| exit2: |
| %ret2 = phi i64 [ 100, %early.exit.0 ] |
| ret i64 %ret2 |
| |
| exit: |
| ret i64 43 |
| } |
| |
| define i64 @two_early_exits_different_exits_second_load_not_deref(ptr %B) { |
| ; CHECK-LABEL: define i64 @two_early_exits_different_exits_second_load_not_deref( |
| ; CHECK-SAME: ptr [[B:%.*]]) { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[A:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[A]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ 0, %[[ENTRY]] ], [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ] |
| ; CHECK-NEXT: [[GEP_A:%.*]] = getelementptr inbounds i8, ptr [[A]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[GEP_A]], align 1 |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[B]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EXIT1:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[LD_A]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT2:.*]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp ne i64 [[IV_NEXT]], 67 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LOOP_HEADER]], label %[[EXIT:.*]] |
| ; CHECK: [[EXIT1]]: |
| ; CHECK-NEXT: [[RET1:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ] |
| ; CHECK-NEXT: ret i64 [[RET1]] |
| ; CHECK: [[EXIT2]]: |
| ; CHECK-NEXT: [[RET2:%.*]] = phi i64 [ 100, %[[EARLY_EXIT_0]] ] |
| ; CHECK-NEXT: ret i64 [[RET2]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: ret i64 43 |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ 0, %entry ], [ %iv.next, %loop.latch ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit1, label %early.exit.0 |
| |
| early.exit.0: |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit2, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 67 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit1: |
| %ret1 = phi i64 [ %iv, %loop.header ] |
| ret i64 %ret1 |
| |
| exit2: |
| %ret2 = phi i64 [ 100, %early.exit.0 ] |
| ret i64 %ret2 |
| |
| exit: |
| ret i64 43 |
| } |
| |
| |
| define i64 @three_early_exits_same_exit() { |
| ; CHECK-LABEL: define i64 @three_early_exits_same_exit() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[OFFSET_IDX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 3, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP8:%.*]] = icmp eq i8 [[LD_A]], 42 |
| ; CHECK-NEXT: br i1 [[TMP8]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[CMP3:%.*]] = icmp ugt i8 [[LD_B]], 100 |
| ; CHECK-NEXT: br i1 [[CMP3]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[OFFSET_IDX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp ne i64 [[IV_NEXT]], 67 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LOOP_HEADER]], label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[OFFSET_IDX]], %[[LOOP_HEADER]] ], [ 100, %[[EARLY_EXIT_0]] ], [ 200, %[[EARLY_EXIT_1]] ], [ 43, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 3, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 42 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp2, label %exit, label %early.exit.1 |
| |
| early.exit.1: |
| %cmp3 = icmp ugt i8 %ld.B, 100 |
| br i1 %cmp3, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 67 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ 100, %early.exit.0 ], [ 200, %early.exit.1 ], [ 43, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define i64 @four_early_exits_same_exit() { |
| ; CHECK-LABEL: define i64 @four_early_exits_same_exit() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P3:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P3]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[OFFSET_IDX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 3, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP11:%.*]] = icmp eq i8 [[LD_A]], 42 |
| ; CHECK-NEXT: br i1 [[TMP11]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[GEP_C:%.*]] = getelementptr inbounds i8, ptr [[P3]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_C:%.*]] = load i8, ptr [[GEP_C]], align 1 |
| ; CHECK-NEXT: [[CMP3:%.*]] = icmp eq i8 [[LD_B]], [[LD_C]] |
| ; CHECK-NEXT: br i1 [[CMP3]], label %[[EXIT]], label %[[EARLY_EXIT_2:.*]] |
| ; CHECK: [[EARLY_EXIT_2]]: |
| ; CHECK-NEXT: [[CMP4:%.*]] = icmp ugt i8 [[LD_C]], 100 |
| ; CHECK-NEXT: br i1 [[CMP4]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[OFFSET_IDX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp ne i64 [[IV_NEXT]], 67 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LOOP_HEADER]], label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[OFFSET_IDX]], %[[LOOP_HEADER]] ], [ 100, %[[EARLY_EXIT_0]] ], [ 200, %[[EARLY_EXIT_1]] ], [ 300, %[[EARLY_EXIT_2]] ], [ 43, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| %C = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| call void @init_mem(ptr %C, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 3, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 42 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp2, label %exit, label %early.exit.1 |
| |
| early.exit.1: |
| %gep.C = getelementptr inbounds i8, ptr %C, i64 %iv |
| %ld.C = load i8, ptr %gep.C, align 1 |
| %cmp3 = icmp eq i8 %ld.B, %ld.C |
| br i1 %cmp3, label %exit, label %early.exit.2 |
| |
| early.exit.2: |
| %cmp4 = icmp ugt i8 %ld.C, 100 |
| br i1 %cmp4, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 67 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ 100, %early.exit.0 ], [ 200, %early.exit.1 ], [ 300, %early.exit.2 ], [ 43, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define i64 @two_early_exits_with_live_out_values() { |
| ; CHECK-LABEL: define i64 @two_early_exits_with_live_out_values() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[OFFSET_IDX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 3, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[TMP7:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[TMP7]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[SUM:%.*]] = add i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[SUM]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[OFFSET_IDX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp ne i64 [[IV_NEXT]], 67 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LOOP_HEADER]], label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[OFFSET_IDX]], %[[LOOP_HEADER]] ], [ [[OFFSET_IDX]], %[[EARLY_EXIT_0]] ], [ 99, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: [[RETVAL2:%.*]] = phi i8 [ 0, %[[LOOP_HEADER]] ], [ [[SUM]], %[[EARLY_EXIT_0]] ], [ 0, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: [[EXT:%.*]] = zext i8 [[RETVAL2]] to i64 |
| ; CHECK-NEXT: [[RET:%.*]] = add i64 [[RETVAL]], [[EXT]] |
| ; CHECK-NEXT: ret i64 [[RET]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 3, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %sum = add i8 %ld.A, %ld.B |
| %cmp2 = icmp ult i8 %sum, 34 |
| br i1 %cmp2, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 67 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ 99, %loop.latch ] |
| %retval2 = phi i8 [ 0, %loop.header ], [ %sum, %early.exit.0 ], [ 0, %loop.latch ] |
| %ext = zext i8 %retval2 to i64 |
| %ret = add i64 %retval, %ext |
| ret i64 %ret |
| } |
| |
| define i64 @two_early_exits_negated_condition() { |
| ; CHECK-LABEL: define i64 @two_early_exits_negated_condition() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 3, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[GEP_A:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[GEP_A]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp ne i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EARLY_EXIT_0:.*]], label %[[EXIT:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp uge i8 [[LD_A]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[LOOP_LATCH]], label %[[EXIT]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp ne i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LOOP_HEADER]], label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ], [ 100, %[[EARLY_EXIT_0]] ], [ 43, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 3, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp ne i8 %ld.A, %ld.B |
| br i1 %cmp1, label %early.exit.0, label %exit |
| |
| early.exit.0: |
| %cmp2 = icmp uge i8 %ld.A, 34 |
| br i1 %cmp2, label %loop.latch, label %exit |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 128 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ 100, %early.exit.0 ], [ 43, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| ; Three early exits to three different exit blocks. |
| define i64 @three_early_exits_three_exit_blocks() { |
| ; CHECK-LABEL: define i64 @three_early_exits_three_exit_blocks() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[OFFSET_IDX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 3, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[OFFSET_IDX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[TMP8:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[TMP8]], label %[[EXIT1:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[TMP10:%.*]] = icmp ult i8 [[LD_A]], 34 |
| ; CHECK-NEXT: br i1 [[TMP10]], label %[[EXIT2:.*]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[TMP11:%.*]] = icmp ugt i8 [[LD_B]], 100 |
| ; CHECK-NEXT: br i1 [[TMP11]], label %[[EXIT3:.*]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[OFFSET_IDX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp ne i64 [[IV_NEXT]], 99 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LOOP_HEADER]], label %[[EXIT:.*]] |
| ; CHECK: [[EXIT1]]: |
| ; CHECK-NEXT: [[RET4:%.*]] = phi i64 [ [[OFFSET_IDX]], %[[LOOP_HEADER]] ] |
| ; CHECK-NEXT: ret i64 [[RET4]] |
| ; CHECK: [[EXIT2]]: |
| ; CHECK-NEXT: [[RET2:%.*]] = phi i64 [ 100, %[[EARLY_EXIT_0]] ] |
| ; CHECK-NEXT: ret i64 [[RET2]] |
| ; CHECK: [[EXIT3]]: |
| ; CHECK-NEXT: [[RET3:%.*]] = phi i64 [ 200, %[[EARLY_EXIT_1]] ] |
| ; CHECK-NEXT: ret i64 [[RET3]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: ret i64 43 |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 3, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit1, label %early.exit.0 |
| |
| early.exit.0: |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit2, label %early.exit.1 |
| |
| early.exit.1: |
| %cmp3 = icmp ugt i8 %ld.B, 100 |
| br i1 %cmp3, label %exit3, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, 99 |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit1: |
| %ret1 = phi i64 [ %iv, %loop.header ] |
| ret i64 %ret1 |
| |
| exit2: |
| %ret2 = phi i64 [ 100, %early.exit.0 ] |
| ret i64 %ret2 |
| |
| exit3: |
| %ret3 = phi i64 [ 200, %early.exit.1 ] |
| ret i64 %ret3 |
| |
| exit: |
| ret i64 43 |
| } |
| |
| ; Two early exits with IV used as live-out from both exits. |
| define i64 @two_early_exits_iv_live_out() { |
| ; CHECK-LABEL: define i64 @two_early_exits_iv_live_out() { |
| ; CHECK-NEXT: [[ENTRY:.*:]] |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[VECTOR_PH:.*]] |
| ; CHECK: [[VECTOR_PH]]: |
| ; CHECK-NEXT: br label %[[VECTOR_BODY:.*]] |
| ; CHECK: [[VECTOR_BODY]]: |
| ; CHECK-NEXT: [[INDEX:%.*]] = phi i64 [ 0, %[[VECTOR_PH]] ], [ [[INDEX_NEXT:%.*]], %[[VECTOR_BODY_INTERIM:.*]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[WIDE_LOAD:%.*]] = load <4 x i8>, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[WIDE_LOAD1:%.*]] = load <4 x i8>, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[TMP2:%.*]] = icmp eq <4 x i8> [[WIDE_LOAD]], [[WIDE_LOAD1]] |
| ; CHECK-NEXT: [[TMP3:%.*]] = icmp ult <4 x i8> [[WIDE_LOAD]], splat (i8 34) |
| ; CHECK-NEXT: [[TMP4:%.*]] = or <4 x i1> [[TMP2]], [[TMP3]] |
| ; CHECK-NEXT: [[INDEX_NEXT]] = add nuw i64 [[INDEX]], 4 |
| ; CHECK-NEXT: [[TMP5:%.*]] = freeze <4 x i1> [[TMP4]] |
| ; CHECK-NEXT: [[TMP6:%.*]] = call i1 @llvm.vector.reduce.or.v4i1(<4 x i1> [[TMP5]]) |
| ; CHECK-NEXT: [[TMP7:%.*]] = icmp eq i64 [[INDEX_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[TMP6]], label %[[VECTOR_EARLY_EXIT:.*]], label %[[VECTOR_BODY_INTERIM]] |
| ; CHECK: [[VECTOR_BODY_INTERIM]]: |
| ; CHECK-NEXT: br i1 [[TMP7]], label %[[MIDDLE_BLOCK:.*]], label %[[VECTOR_BODY]], !llvm.loop [[LOOP0:![0-9]+]] |
| ; CHECK: [[MIDDLE_BLOCK]]: |
| ; CHECK-NEXT: br label %[[EXIT:.*]] |
| ; CHECK: [[VECTOR_EARLY_EXIT]]: |
| ; CHECK-NEXT: [[TMP8:%.*]] = call i64 @llvm.experimental.cttz.elts.i64.v4i1(<4 x i1> [[TMP4]], i1 false) |
| ; CHECK-NEXT: [[TMP9:%.*]] = add i64 [[INDEX]], [[TMP8]] |
| ; CHECK-NEXT: br label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[TMP9]], %[[VECTOR_EARLY_EXIT]] ], [ 128, %[[MIDDLE_BLOCK]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ 128, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define i64 @three_early_exits_iv_and_load_live_out() { |
| ; CHECK-LABEL: define i64 @three_early_exits_iv_and_load_live_out() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[INDEX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[LD_A]], 34 |
| ; CHECK-NEXT: [[TMP8:%.*]] = or i1 [[CMP1]], [[CMP2]] |
| ; CHECK-NEXT: br i1 [[TMP8]], label %[[EXIT:.*]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[CMP3:%.*]] = icmp ugt i8 [[LD_B]], 100 |
| ; CHECK-NEXT: br i1 [[CMP3]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[INDEX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[EXIT]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL_IV:%.*]] = phi i64 [ [[INDEX]], %[[LOOP_HEADER]] ], [ 128, %[[LOOP_LATCH]] ], [ [[INDEX]], %[[EARLY_EXIT_1]] ] |
| ; CHECK-NEXT: [[RETVAL_LD:%.*]] = phi i8 [ [[LD_A]], %[[LOOP_HEADER]] ], [ 0, %[[LOOP_LATCH]] ], [ [[LD_B]], %[[EARLY_EXIT_1]] ] |
| ; CHECK-NEXT: [[EXT:%.*]] = zext i8 [[RETVAL_LD]] to i64 |
| ; CHECK-NEXT: [[RET:%.*]] = add i64 [[RETVAL_IV]], [[EXT]] |
| ; CHECK-NEXT: ret i64 [[RET]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit, label %early.exit.1 |
| |
| early.exit.1: |
| %cmp3 = icmp ugt i8 %ld.B, 100 |
| br i1 %cmp3, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval.iv = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ %iv, %early.exit.1 ], [ 128, %loop.latch ] |
| %retval.ld = phi i8 [ %ld.A, %loop.header ], [ %ld.A, %early.exit.0 ], [ %ld.B, %early.exit.1 ], [ 0, %loop.latch ] |
| %ext = zext i8 %retval.ld to i64 |
| %ret = add i64 %retval.iv, %ext |
| ret i64 %ret |
| } |
| |
| ; Two early exits with different live-out values from each exit. |
| define i64 @two_early_exits_different_live_outs() { |
| ; CHECK-LABEL: define i64 @two_early_exits_different_live_outs() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[SUM:%.*]] = phi i64 [ [[SUM_NEXT:%.*]], %[[LOOP_LATCH]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[ARRAYIDX:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD1:%.*]] = load i8, ptr [[ARRAYIDX]], align 1 |
| ; CHECK-NEXT: [[ARRAYIDX1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD2:%.*]] = load i8, ptr [[ARRAYIDX1]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[LD1]], [[LD2]] |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EXIT1:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[LD1]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT2:.*]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[IV]], 1 |
| ; CHECK-NEXT: [[LD1_EXT:%.*]] = zext i8 [[LD1]] to i64 |
| ; CHECK-NEXT: [[SUM_NEXT]] = add i64 [[SUM]], [[LD1_EXT]] |
| ; CHECK-NEXT: [[EXITCOND:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EXITCOND]], label %[[EXIT_LATCH:.*]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT1]]: |
| ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ] |
| ; CHECK-NEXT: ret i64 [[IV_LCSSA]] |
| ; CHECK: [[EXIT2]]: |
| ; CHECK-NEXT: [[LD1_LCSSA1:%.*]] = phi i8 [ [[LD1]], %[[EARLY_EXIT_0]] ] |
| ; CHECK-NEXT: [[LD1_EXT2:%.*]] = zext i8 [[LD1_LCSSA1]] to i64 |
| ; CHECK-NEXT: ret i64 [[LD1_EXT2]] |
| ; CHECK: [[EXIT_LATCH]]: |
| ; CHECK-NEXT: [[SUM_NEXT_LCSSA:%.*]] = phi i64 [ [[SUM_NEXT]], %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[SUM_NEXT_LCSSA]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %sum = phi i64 [ %sum.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit1, label %early.exit.0 |
| |
| early.exit.0: |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit2, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ld.A.ext = zext i8 %ld.A to i64 |
| %sum.next = add i64 %sum, %ld.A.ext |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit.latch, label %loop.header |
| |
| exit1: |
| ret i64 %iv |
| |
| exit2: |
| %ld.A.ext2 = zext i8 %ld.A to i64 |
| ret i64 %ld.A.ext2 |
| |
| exit.latch: |
| ret i64 %sum.next |
| } |
| |
| ; Two early exits to same block, IV live-out with different incoming values. |
| define i64 @two_early_exits_iv_diff_incoming() { |
| ; CHECK-LABEL: define i64 @two_early_exits_iv_diff_incoming() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[INDEX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[TMP14:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[TMP14]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[IV_PLUS1:%.*]] = add i64 [[INDEX]], 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[LD_A]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[INDEX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[EXIT]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[INDEX]], %[[LOOP_HEADER]] ], [ [[IV_PLUS1]], %[[EARLY_EXIT_0]] ], [ 200, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %iv.plus1 = add i64 %iv, 1 |
| %cmp2 = icmp ult i8 %ld.A, 34 |
| br i1 %cmp2, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ %iv.plus1, %early.exit.0 ], [ 200, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define { i64, i64 } @three_early_exits_multiple_live_outs() { |
| ; CHECK-LABEL: define { i64, i64 } @three_early_exits_multiple_live_outs() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[INDEX:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[TMP0:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[TMP0]], align 1 |
| ; CHECK-NEXT: [[TMP1:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[INDEX]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[TMP1]], align 1 |
| ; CHECK-NEXT: [[TMP10:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[TMP10]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[SUM:%.*]] = add i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i8 [[SUM]], 34 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[DIFF:%.*]] = sub i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: [[CMP3:%.*]] = icmp ugt i8 [[DIFF]], 100 |
| ; CHECK-NEXT: br i1 [[CMP3]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[INDEX]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[EXIT]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL_IV:%.*]] = phi i64 [ [[INDEX]], %[[LOOP_HEADER]] ], [ [[INDEX]], %[[EARLY_EXIT_0]] ], [ [[INDEX]], %[[EARLY_EXIT_1]] ], [ 128, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: [[RETVAL_VAL:%.*]] = phi i8 [ [[LD_A]], %[[LOOP_HEADER]] ], [ [[SUM]], %[[EARLY_EXIT_0]] ], [ [[DIFF]], %[[EARLY_EXIT_1]] ], [ 0, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: [[EXT:%.*]] = zext i8 [[RETVAL_VAL]] to i64 |
| ; CHECK-NEXT: [[R1:%.*]] = insertvalue { i64, i64 } undef, i64 [[RETVAL_IV]], 0 |
| ; CHECK-NEXT: [[R2:%.*]] = insertvalue { i64, i64 } [[R1]], i64 [[EXT]], 1 |
| ; CHECK-NEXT: ret { i64, i64 } [[R2]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp1 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %sum = add i8 %ld.A, %ld.B |
| %cmp2 = icmp ult i8 %sum, 34 |
| br i1 %cmp2, label %exit, label %early.exit.1 |
| |
| early.exit.1: |
| %diff = sub i8 %ld.A, %ld.B |
| %cmp3 = icmp ugt i8 %diff, 100 |
| br i1 %cmp3, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval.iv = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ %iv, %early.exit.1 ], [ 128, %loop.latch ] |
| %retval.val = phi i8 [ %ld.A, %loop.header ], [ %sum, %early.exit.0 ], [ %diff, %early.exit.1 ], [ 0, %loop.latch ] |
| %ext = zext i8 %retval.val to i64 |
| %r1 = insertvalue { i64, i64 } undef, i64 %retval.iv, 0 |
| %r2 = insertvalue { i64, i64 } %r1, i64 %ext, 1 |
| ret { i64, i64 } %r2 |
| } |
| |
| ; Two early exits with second load only executed conditionally (after first exit check). |
| define i64 @two_early_exits_load_in_early_exit_block() { |
| ; CHECK-LABEL: define i64 @two_early_exits_load_in_early_exit_block() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[ARRAYIDX:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[ARRAYIDX]], align 1 |
| ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i8 [[LD_A]], 42 |
| ; CHECK-NEXT: br i1 [[TMP6]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[EXIT]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ], [ [[IV]], %[[EARLY_EXIT_0]] ], [ 128, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 42 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp2, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ 128, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define i64 @three_early_exits_loads_in_different_blocks() { |
| ; CHECK-LABEL: define i64 @three_early_exits_loads_in_different_blocks() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P3:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P3]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[ARRAYIDX1:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[ARRAYIDX1]], align 1 |
| ; CHECK-NEXT: [[TMP9:%.*]] = icmp eq i8 [[LD_A]], 42 |
| ; CHECK-NEXT: br i1 [[TMP9]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD_A]], [[LD_B]] |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[GEP_C:%.*]] = getelementptr inbounds i8, ptr [[P3]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_C:%.*]] = load i8, ptr [[GEP_C]], align 1 |
| ; CHECK-NEXT: [[CMP3:%.*]] = icmp eq i8 [[LD_B]], [[LD_C]] |
| ; CHECK-NEXT: br i1 [[CMP3]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[EXIT]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ], [ [[IV]], %[[EARLY_EXIT_0]] ], [ [[IV]], %[[EARLY_EXIT_1]] ], [ 128, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| %C = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| call void @init_mem(ptr %C, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 42 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp2, label %exit, label %early.exit.1 |
| |
| early.exit.1: |
| %gep.C = getelementptr inbounds i8, ptr %C, i64 %iv |
| %ld.C = load i8, ptr %gep.C, align 1 |
| %cmp3 = icmp eq i8 %ld.B, %ld.C |
| br i1 %cmp3, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ %iv, %early.exit.1 ], [ 128, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define { i64, i8 } @four_early_exits_with_conditional_loads() { |
| ; CHECK-LABEL: define { i64, i8 } @four_early_exits_with_conditional_loads() { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[P1:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P2:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P3:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[P4:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[P1]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P2]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P3]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[P4]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ], [ 0, %[[ENTRY]] ] |
| ; CHECK-NEXT: [[ARRAYIDX1:%.*]] = getelementptr inbounds i8, ptr [[P1]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_A:%.*]] = load i8, ptr [[ARRAYIDX1]], align 1 |
| ; CHECK-NEXT: [[TMP12:%.*]] = icmp eq i8 [[LD_A]], 10 |
| ; CHECK-NEXT: br i1 [[TMP12]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[P2]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_B:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD_B]], 20 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[EARLY_EXIT_1:.*]] |
| ; CHECK: [[EARLY_EXIT_1]]: |
| ; CHECK-NEXT: [[GEP_C:%.*]] = getelementptr inbounds i8, ptr [[P3]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_C:%.*]] = load i8, ptr [[GEP_C]], align 1 |
| ; CHECK-NEXT: [[CMP3:%.*]] = icmp eq i8 [[LD_C]], 30 |
| ; CHECK-NEXT: br i1 [[CMP3]], label %[[EXIT]], label %[[EARLY_EXIT_2:.*]] |
| ; CHECK: [[EARLY_EXIT_2]]: |
| ; CHECK-NEXT: [[GEP_D:%.*]] = getelementptr inbounds i8, ptr [[P4]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD_D:%.*]] = load i8, ptr [[GEP_D]], align 1 |
| ; CHECK-NEXT: [[CMP4:%.*]] = icmp eq i8 [[LD_D]], 40 |
| ; CHECK-NEXT: br i1 [[CMP4]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV_NEXT]], 128 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[EXIT]], label %[[LOOP_HEADER]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL_IV:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ], [ [[IV]], %[[EARLY_EXIT_0]] ], [ [[IV]], %[[EARLY_EXIT_1]] ], [ [[IV]], %[[EARLY_EXIT_2]] ], [ 128, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: [[RETVAL_VAL:%.*]] = phi i8 [ [[LD_A]], %[[LOOP_HEADER]] ], [ [[LD_B]], %[[EARLY_EXIT_0]] ], [ [[LD_C]], %[[EARLY_EXIT_1]] ], [ [[LD_D]], %[[EARLY_EXIT_2]] ], [ 0, %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: [[R1:%.*]] = insertvalue { i64, i8 } undef, i64 [[RETVAL_IV]], 0 |
| ; CHECK-NEXT: [[R2:%.*]] = insertvalue { i64, i8 } [[R1]], i8 [[RETVAL_VAL]], 1 |
| ; CHECK-NEXT: ret { i64, i8 } [[R2]] |
| ; |
| entry: |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| %C = alloca [1024 x i8] |
| %D = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| call void @init_mem(ptr %C, i64 1024) |
| call void @init_mem(ptr %D, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ %iv.next, %loop.latch ], [ 0, %entry ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 10 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.B, 20 |
| br i1 %cmp2, label %exit, label %early.exit.1 |
| |
| early.exit.1: |
| %gep.C = getelementptr inbounds i8, ptr %C, i64 %iv |
| %ld.C = load i8, ptr %gep.C, align 1 |
| %cmp3 = icmp eq i8 %ld.C, 30 |
| br i1 %cmp3, label %exit, label %early.exit.2 |
| |
| early.exit.2: |
| %gep.D = getelementptr inbounds i8, ptr %D, i64 %iv |
| %ld.D = load i8, ptr %gep.D, align 1 |
| %cmp4 = icmp eq i8 %ld.D, 40 |
| br i1 %cmp4, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw i64 %iv, 1 |
| %ec = icmp eq i64 %iv.next, 128 |
| br i1 %ec, label %exit, label %loop.header |
| |
| exit: |
| %retval.iv = phi i64 [ %iv, %loop.header ], [ %iv, %early.exit.0 ], [ %iv, %early.exit.1 ], [ %iv, %early.exit.2 ], [ 128, %loop.latch ] |
| %retval.val = phi i8 [ %ld.A, %loop.header ], [ %ld.B, %early.exit.0 ], [ %ld.C, %early.exit.1 ], [ %ld.D, %early.exit.2 ], [ 0, %loop.latch ] |
| %r1 = insertvalue { i64, i8 } undef, i64 %retval.iv, 0 |
| %r2 = insertvalue { i64, i8 } %r1, i8 %retval.val, 1 |
| ret { i64, i8 } %r2 |
| } |
| |
| define i64 @two_early_exits_variable_trip_count(i64 %n) { |
| ; CHECK-LABEL: define i64 @two_early_exits_variable_trip_count( |
| ; CHECK-SAME: i64 [[N:%.*]]) { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp ult i64 [[N]], 1024 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: [[A:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: [[B:%.*]] = alloca [1024 x i8], align 1 |
| ; CHECK-NEXT: call void @init_mem(ptr [[A]], i64 1024) |
| ; CHECK-NEXT: call void @init_mem(ptr [[B]], i64 1024) |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ 0, %[[ENTRY]] ], [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ] |
| ; CHECK-NEXT: [[GEP_A:%.*]] = getelementptr inbounds i8, ptr [[A]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD1:%.*]] = load i8, ptr [[GEP_A]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[LD1]], 42 |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EXIT:.*]], label %[[EARLY_EXIT_0:.*]] |
| ; CHECK: [[EARLY_EXIT_0]]: |
| ; CHECK-NEXT: [[GEP_B:%.*]] = getelementptr inbounds i8, ptr [[B]], i64 [[IV]] |
| ; CHECK-NEXT: [[LD2:%.*]] = load i8, ptr [[GEP_B]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[LD1]], [[LD2]] |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EXITCOND:%.*]] = icmp ne i64 [[IV_NEXT]], [[N]] |
| ; CHECK-NEXT: br i1 [[EXITCOND]], label %[[LOOP_HEADER]], label %[[EXIT]] |
| ; CHECK: [[EXIT]]: |
| ; CHECK-NEXT: [[RETVAL:%.*]] = phi i64 [ [[IV]], %[[LOOP_HEADER]] ], [ 100, %[[EARLY_EXIT_0]] ], [ [[N]], %[[LOOP_LATCH]] ] |
| ; CHECK-NEXT: ret i64 [[RETVAL]] |
| ; |
| entry: |
| %cmp = icmp ult i64 %n, 1024 |
| call void @llvm.assume(i1 %cmp) |
| %A = alloca [1024 x i8] |
| %B = alloca [1024 x i8] |
| call void @init_mem(ptr %A, i64 1024) |
| call void @init_mem(ptr %B, i64 1024) |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ 0, %entry ], [ %iv.next, %loop.latch ] |
| %gep.A = getelementptr inbounds i8, ptr %A, i64 %iv |
| %ld.A = load i8, ptr %gep.A, align 1 |
| %cmp1 = icmp eq i8 %ld.A, 42 |
| br i1 %cmp1, label %exit, label %early.exit.0 |
| |
| early.exit.0: |
| %gep.B = getelementptr inbounds i8, ptr %B, i64 %iv |
| %ld.B = load i8, ptr %gep.B, align 1 |
| %cmp2 = icmp eq i8 %ld.A, %ld.B |
| br i1 %cmp2, label %exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add i64 %iv, 1 |
| %ec = icmp ne i64 %iv.next, %n |
| br i1 %ec, label %loop.header, label %exit |
| |
| exit: |
| %retval = phi i64 [ %iv, %loop.header ], [ 100, %early.exit.0 ], [ %n, %loop.latch ] |
| ret i64 %retval |
| } |
| |
| define i64 @three_early_exits_same_block_no_latch_exit(i1 %cmp0) { |
| ; CHECK-LABEL: define i64 @three_early_exits_same_block_no_latch_exit( |
| ; CHECK-SAME: i1 [[CMP0:%.*]]) { |
| ; CHECK-NEXT: [[ENTRY:.*]]: |
| ; CHECK-NEXT: br label %[[LOOP_HEADER:.*]] |
| ; CHECK: [[LOOP_HEADER]]: |
| ; CHECK-NEXT: [[IV:%.*]] = phi i64 [ 0, %[[ENTRY]] ], [ [[IV_NEXT:%.*]], %[[LOOP_LATCH:.*]] ] |
| ; CHECK-NEXT: br i1 [[CMP0]], label %[[EARLY_EXIT:.*]], label %[[LOOP_BODY_0:.*]] |
| ; CHECK: [[LOOP_BODY_0]]: |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr [24 x i8], ptr @GlobA, i64 [[IV]] |
| ; CHECK-NEXT: [[GEP_1:%.*]] = getelementptr i8, ptr [[GEP]], i64 4 |
| ; CHECK-NEXT: [[L_0:%.*]] = load i8, ptr [[GEP_1]], align 1 |
| ; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8 [[L_0]], 0 |
| ; CHECK-NEXT: br i1 [[CMP1]], label %[[EARLY_EXIT]], label %[[LOOP_BODY_1:.*]] |
| ; CHECK: [[LOOP_BODY_1]]: |
| ; CHECK-NEXT: [[L_1:%.*]] = load i8, ptr [[GEP]], align 1 |
| ; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i8 [[L_1]], 0 |
| ; CHECK-NEXT: br i1 [[CMP2]], label %[[EARLY_EXIT]], label %[[LOOP_LATCH]] |
| ; CHECK: [[LOOP_LATCH]]: |
| ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i64 [[IV]], 1 |
| ; CHECK-NEXT: [[EC:%.*]] = icmp eq i64 [[IV]], 0 |
| ; CHECK-NEXT: br i1 [[EC]], label %[[LATCH_EXIT:.*]], label %[[LOOP_HEADER]] |
| ; CHECK: [[LATCH_EXIT]]: |
| ; CHECK-NEXT: ret i64 0 |
| ; CHECK: [[EARLY_EXIT]]: |
| ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i64 [ [[IV]], %[[LOOP_BODY_1]] ], [ [[IV]], %[[LOOP_BODY_0]] ], [ [[IV]], %[[LOOP_HEADER]] ] |
| ; CHECK-NEXT: ret i64 [[IV_LCSSA]] |
| ; |
| entry: |
| br label %loop.header |
| |
| loop.header: |
| %iv = phi i64 [ 0, %entry ], [ %iv.next, %loop.latch ] |
| br i1 %cmp0, label %early.exit, label %loop.body.0 |
| |
| loop.body.0: |
| %gep = getelementptr [24 x i8], ptr @GlobA, i64 %iv |
| %gep.1 = getelementptr i8, ptr %gep, i64 4 |
| %l.0 = load i8, ptr %gep.1, align 1 |
| %cmp1 = icmp eq i8 %l.0, 0 |
| br i1 %cmp1, label %early.exit, label %loop.body.1 |
| |
| loop.body.1: |
| %l.1 = load i8, ptr %gep, align 1 |
| %cmp2 = icmp eq i8 %l.1, 0 |
| br i1 %cmp2, label %early.exit, label %loop.latch |
| |
| loop.latch: |
| %iv.next = add nuw nsw i64 %iv, 1 |
| %ec = icmp eq i64 %iv, 0 |
| br i1 %ec, label %latch.exit, label %loop.header |
| |
| latch.exit: |
| ret i64 0 |
| |
| early.exit: |
| ret i64 %iv |
| } |