; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 2 ; RUN: opt -passes=instcombine -S < %s | FileCheck %s declare void @llvm.init.trampoline(ptr, ptr, ptr) declare ptr @llvm.adjust.trampoline(ptr) declare i32 @f(ptr nest, i32) ; Most common case define i32 @test0(i32 %n) !dbg !4 { ; CHECK-LABEL: define i32 @test0 ; CHECK-SAME: (i32 [[N:%.*]]) !dbg [[DBG4:![0-9]+]] { ; CHECK-NEXT: [[ALLOCA:%.*]] = alloca [10 x i8], align 16 ; CHECK-NEXT: call void @llvm.init.trampoline(ptr nonnull [[ALLOCA]], ptr nonnull @f, ptr null) ; CHECK-NEXT: [[RET:%.*]] = call i32 @f(ptr nest null, i32 [[N]]), !dbg [[DBG10:![0-9]+]] ; CHECK-NEXT: ret i32 [[RET]] ; %alloca = alloca [10 x i8], align 16 call void @llvm.init.trampoline(ptr %alloca, ptr @f, ptr null) %tramp = call ptr @llvm.adjust.trampoline(ptr %alloca) %ret = call i32 %tramp(i32 %n), !dbg !10 ret i32 %ret } define i32 @test1(i32 %n, ptr %trampmem) { ; CHECK-LABEL: define i32 @test1 ; CHECK-SAME: (i32 [[N:%.*]], ptr [[TRAMPMEM:%.*]]) { ; CHECK-NEXT: call void @llvm.init.trampoline(ptr [[TRAMPMEM]], ptr nonnull @f, ptr null) ; CHECK-NEXT: [[RET:%.*]] = call i32 @f(ptr nest null, i32 [[N]]) ; CHECK-NEXT: ret i32 [[RET]] ; call void @llvm.init.trampoline(ptr %trampmem, ptr @f, ptr null) %tramp = call ptr @llvm.adjust.trampoline(ptr %trampmem) %ret = call i32 %tramp(i32 %n) ret i32 %ret } define i32 @test2(i32 %n, ptr %trampmem) { ; CHECK-LABEL: define i32 @test2 ; CHECK-SAME: (i32 [[N:%.*]], ptr [[TRAMPMEM:%.*]]) { ; CHECK-NEXT: [[TRAMP:%.*]] = call ptr @llvm.adjust.trampoline(ptr [[TRAMPMEM]]) ; CHECK-NEXT: [[RET:%.*]] = call i32 [[TRAMP]](i32 [[N]]) ; CHECK-NEXT: ret i32 [[RET]] ; %tramp = call ptr @llvm.adjust.trampoline(ptr %trampmem) %ret = call i32 %tramp(i32 %n) ret i32 %ret } define i32 @test3(i32 %n, ptr %trampmem) { ; CHECK-LABEL: define i32 @test3 ; CHECK-SAME: (i32 [[N:%.*]], ptr [[TRAMPMEM:%.*]]) { ; CHECK-NEXT: call void @llvm.init.trampoline(ptr [[TRAMPMEM]], ptr nonnull @f, ptr null) ; CHECK-NEXT: [[RET0:%.*]] = call i32 @f(ptr nest null, i32 [[N]]) ; CHECK-NEXT: [[TRAMP1:%.*]] = call ptr @llvm.adjust.trampoline(ptr [[TRAMPMEM]]) ; CHECK-NEXT: [[RET1:%.*]] = call i32 [[TRAMP1]](i32 [[N]]) ; CHECK-NEXT: ret i32 [[RET1]] ; call void @llvm.init.trampoline(ptr %trampmem, ptr @f, ptr null) %tramp0 = call ptr @llvm.adjust.trampoline(ptr %trampmem) %ret0 = call i32 %tramp0(i32 %n) ;; Not optimized since previous call could be writing. %tramp1 = call ptr @llvm.adjust.trampoline(ptr %trampmem) %ret1 = call i32 %tramp1(i32 %n) ret i32 %ret1 } define i32 @test4(i32 %n) { ; CHECK-LABEL: define i32 @test4 ; CHECK-SAME: (i32 [[N:%.*]]) { ; CHECK-NEXT: [[ALLOCA:%.*]] = alloca [10 x i8], align 16 ; CHECK-NEXT: call void @llvm.init.trampoline(ptr nonnull [[ALLOCA]], ptr nonnull @f, ptr null) ; CHECK-NEXT: [[RET0:%.*]] = call i32 @f(ptr nest null, i32 [[N]]) ; CHECK-NEXT: [[RET1:%.*]] = call i32 @f(ptr nest null, i32 [[N]]) ; CHECK-NEXT: [[RET2:%.*]] = call i32 @f(ptr nest null, i32 [[N]]) ; CHECK-NEXT: ret i32 [[RET2]] ; %alloca = alloca [10 x i8], align 16 call void @llvm.init.trampoline(ptr %alloca, ptr @f, ptr null) %tramp0 = call ptr @llvm.adjust.trampoline(ptr %alloca) %ret0 = call i32 %tramp0(i32 %n) %tramp1 = call ptr @llvm.adjust.trampoline(ptr %alloca) %ret1 = call i32 %tramp0(i32 %n) %tramp2 = call ptr @llvm.adjust.trampoline(ptr %alloca) %ret2 = call i32 %tramp2(i32 %n) ret i32 %ret2 } !llvm.dbg.cu = !{!0} !llvm.module.flags = !{!3} !0 = distinct !DICompileUnit(language: DW_LANG_C99, file: !1, producer: "clang version 3.0 (trunk 127710)", isOptimized: true, runtimeVersion: 0, emissionKind: FullDebug, enums: !2, retainedTypes: !2) !1 = !DIFile(filename: "string.h", directory: "Game") !2 = !{} !3 = !{i32 1, !"Debug Info Version", i32 3} !4 = distinct !DISubprogram(name: "passthru", scope: !1, file: !1, line: 79, type: !5, isLocal: true, isDefinition: true, scopeLine: 79, virtualIndex: 6, flags: DIFlagPrototyped, isOptimized: true, unit: !0, retainedNodes: !8) !5 = !DISubroutineType(types: !6) !6 = !{!7} !7 = !DIDerivedType(tag: DW_TAG_pointer_type, scope: !0, baseType: null, size: 64, align: 64) !8 = !{!9} !9 = !DILocalVariable(name: "a", arg: 1, scope: !4, file: !1, line: 78, type: !7) !10 = !DILocation(line: 78, column: 28, scope: !4)