| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py |
| ;RUN: opt < %s -passes='require<profile-summary>,function(chr,instcombine,simplifycfg)' -S | FileCheck %s |
| |
| declare void @foo() |
| declare void @bar() |
| |
| declare token @llvm.coro.id(i32, ptr, ptr, ptr) |
| declare i1 @llvm.coro.alloc(token) |
| declare ptr @llvm.coro.begin(token, ptr) |
| declare noalias ptr @malloc(i32) |
| |
| %f.Frame = type { ptr, ptr, i1 } |
| |
| ; resume part of the coroutine |
| define fastcc void @f.resume(ptr noalias nonnull align 8 dereferenceable(24) %FramePtr) { |
| tail call void @bar() |
| ret void |
| } |
| |
| ; destroy part of the coroutine |
| define fastcc void @f.destroy(ptr noalias nonnull align 8 dereferenceable(24) %FramePtr) { |
| tail call void @bar() |
| ret void |
| } |
| |
| ; cleanup part of the coroutine |
| define fastcc void @f.cleanup(ptr noalias nonnull align 8 dereferenceable(24) %FramePtr) { |
| tail call void @bar() |
| ret void |
| } |
| |
| @f.resumers = private constant [3 x ptr] [ptr @f.resume, ptr @f.destroy, ptr @f.cleanup] |
| |
| ; Test that chr will skip block containing llvm.coro.id. |
| define ptr @test_chr_with_coro_id(ptr %i) !prof !14 { |
| ; CHECK-LABEL: @test_chr_with_coro_id( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[TMP0:%.*]] = load i32, ptr [[I:%.*]], align 4 |
| ; CHECK-NEXT: [[DOTFR1:%.*]] = freeze i32 [[TMP0]] |
| ; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[DOTFR1]], 3 |
| ; CHECK-NEXT: [[TMP2:%.*]] = icmp eq i32 [[TMP1]], 3 |
| ; CHECK-NEXT: br i1 [[TMP2]], label [[BB0:%.*]], label [[ENTRY_SPLIT_NONCHR:%.*]], !prof [[PROF15:![0-9]+]] |
| ; CHECK: bb0: |
| ; CHECK-NEXT: call void @foo() |
| ; CHECK-NEXT: br label [[BB_CORO_ID:%.*]] |
| ; CHECK: entry.split.nonchr: |
| ; CHECK-NEXT: [[TMP3:%.*]] = and i32 [[DOTFR1]], 1 |
| ; CHECK-NEXT: [[DOTNOT:%.*]] = icmp eq i32 [[TMP3]], 0 |
| ; CHECK-NEXT: br i1 [[DOTNOT]], label [[BB1_NONCHR:%.*]], label [[BB0_NONCHR:%.*]], !prof [[PROF16:![0-9]+]] |
| ; CHECK: bb0.nonchr: |
| ; CHECK-NEXT: call void @foo() |
| ; CHECK-NEXT: br label [[BB1_NONCHR]] |
| ; CHECK: bb1.nonchr: |
| ; CHECK-NEXT: [[TMP4:%.*]] = and i32 [[DOTFR1]], 2 |
| ; CHECK-NEXT: [[TMP5:%.*]] = icmp eq i32 [[TMP4]], 0 |
| ; CHECK-NEXT: br i1 [[TMP5]], label [[BB2_NONCHR:%.*]], label [[BB_CORO_ID]], !prof [[PROF16]] |
| ; CHECK: bb2.nonchr: |
| ; CHECK-NEXT: call void @foo() |
| ; CHECK-NEXT: br label [[BB_CORO_ID]] |
| ; CHECK: bb.coro.id: |
| ; CHECK-NEXT: [[ID:%.*]] = call token @llvm.coro.id(i32 0, ptr null, ptr null, ptr nonnull @f.resumers) |
| ; CHECK-NEXT: [[NEED_DYN_ALLOC:%.*]] = call i1 @llvm.coro.alloc(token [[ID]]) |
| ; CHECK-NEXT: br i1 [[NEED_DYN_ALLOC]], label [[BB_CORO_DYN_ALLOC:%.*]], label [[BB_CORO_BEGIN:%.*]] |
| ; CHECK: bb.coro.dyn.alloc: |
| ; CHECK-NEXT: [[ALLOC:%.*]] = call ptr @malloc(i32 24) |
| ; CHECK-NEXT: br label [[BB_CORO_BEGIN]] |
| ; CHECK: bb.coro.begin: |
| ; CHECK-NEXT: [[PHI:%.*]] = phi ptr [ null, [[BB_CORO_ID]] ], [ [[ALLOC]], [[BB_CORO_DYN_ALLOC]] ] |
| ; CHECK-NEXT: [[HDL:%.*]] = call noalias nonnull ptr @llvm.coro.begin(token [[ID]], ptr [[PHI]]) |
| ; CHECK-NEXT: ret ptr [[HDL]] |
| ; |
| entry: |
| %0 = load i32, ptr %i |
| %1 = and i32 %0, 1 |
| %2 = icmp eq i32 %1, 0 |
| br i1 %2, label %bb1, label %bb0, !prof !15 |
| |
| bb0: |
| call void @foo() |
| br label %bb1 |
| |
| bb1: |
| %3 = and i32 %0, 2 |
| %4 = icmp eq i32 %3, 0 |
| br i1 %4, label %bb2, label %bb.coro.id, !prof !15 |
| |
| bb2: |
| call void @foo() |
| br label %bb.coro.id |
| |
| bb.coro.id: |
| %id = call token @llvm.coro.id(i32 0, ptr null, ptr null, ptr @f.resumers) |
| %need.dyn.alloc = call i1 @llvm.coro.alloc(token %id) |
| br i1 %need.dyn.alloc, label %bb.coro.dyn.alloc, label %bb.coro.begin |
| |
| bb.coro.dyn.alloc: |
| %alloc = call ptr @malloc(i32 24) |
| br label %bb.coro.begin |
| |
| bb.coro.begin: |
| %phi = phi ptr [ null, %bb.coro.id ], [ %alloc, %bb.coro.dyn.alloc ] |
| %hdl = call noalias nonnull ptr @llvm.coro.begin(token %id, ptr %phi) |
| ret ptr %hdl |
| } |
| |
| !llvm.module.flags = !{!0} |
| !0 = !{i32 1, !"ProfileSummary", !1} |
| !1 = !{!2, !3, !4, !5, !6, !7, !8, !9} |
| !2 = !{!"ProfileFormat", !"InstrProf"} |
| !3 = !{!"TotalCount", i64 10000} |
| !4 = !{!"MaxCount", i64 10} |
| !5 = !{!"MaxInternalCount", i64 1} |
| !6 = !{!"MaxFunctionCount", i64 1000} |
| !7 = !{!"NumCounts", i64 3} |
| !8 = !{!"NumFunctions", i64 3} |
| !9 = !{!"DetailedSummary", !10} |
| !10 = !{!11, !12, !13} |
| !11 = !{i32 10000, i64 100, i32 1} |
| !12 = !{i32 999000, i64 100, i32 1} |
| !13 = !{i32 999999, i64 1, i32 2} |
| |
| !14 = !{!"function_entry_count", i64 100} |
| !15 = !{!"branch_weights", i32 0, i32 1} |
| !16 = !{!"branch_weights", i32 1, i32 1} |
| !17 = !{!"branch_weights", i32 0, i32 0} |
| ; CHECK: !15 = !{!"branch_weights", i32 1000, i32 0} |
| ; CHECK: !16 = !{!"branch_weights", i32 0, i32 1} |