| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --check-globals all --version 6 |
| ; RUN: opt -S -passes='cgscc(function-attrs),rpo-function-attrs' < %s | FileCheck %s |
| |
| define float @return_f32_extern(ptr %ptr) { |
| ; CHECK-LABEL: define float @return_f32_extern( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0:[0-9]+]] { |
| ; CHECK-NEXT: [[VAL:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: ret float [[VAL]] |
| ; |
| %val = load volatile float, ptr %ptr |
| ret float %val |
| } |
| |
| ; Deduce nofpclass(inf) on return |
| define internal float @only_noinf_ret_uses(ptr %ptr) { |
| ; CHECK-LABEL: define internal nofpclass(inf) float @only_noinf_ret_uses( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[VAL:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: ret float [[VAL]] |
| ; |
| %val = load volatile float, ptr %ptr |
| ret float %val |
| } |
| |
| define float @calls_no_inf_return(ptr %ptr) { |
| ; CHECK-LABEL: define float @calls_no_inf_return( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[CALL0:%.*]] = call float @return_f32_extern(ptr [[PTR]]) |
| ; CHECK-NEXT: [[CALL1:%.*]] = call nofpclass(inf) float @only_noinf_ret_uses(ptr [[PTR]]) |
| ; CHECK-NEXT: [[ADD:%.*]] = fadd float [[CALL0]], [[CALL1]] |
| ; CHECK-NEXT: ret float [[ADD]] |
| ; |
| %call0 = call float @return_f32_extern(ptr %ptr) |
| %call1 = call nofpclass(inf) float @only_noinf_ret_uses(ptr %ptr) |
| %add = fadd float %call0, %call1 |
| ret float %add |
| } |
| |
| ; Deduce nofpclass(nan) on return, not inf or zero |
| define internal float @merged_ret_uses(ptr %ptr) { |
| ; CHECK-LABEL: define internal nofpclass(nan) float @merged_ret_uses( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[VAL:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: ret float [[VAL]] |
| ; |
| %val = load volatile float, ptr %ptr |
| ret float %val |
| } |
| |
| define float @calls_merge_rets(ptr %ptr) { |
| ; CHECK-LABEL: define float @calls_merge_rets( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[CALL0:%.*]] = call nofpclass(nan inf) float @merged_ret_uses(ptr [[PTR]]) |
| ; CHECK-NEXT: [[CALL1:%.*]] = call nofpclass(nan zero) float @merged_ret_uses(ptr [[PTR]]) |
| ; CHECK-NEXT: [[ADD:%.*]] = fadd float [[CALL0]], [[CALL1]] |
| ; CHECK-NEXT: ret float [[ADD]] |
| ; |
| %call0 = call nofpclass(nan inf) float @merged_ret_uses(ptr %ptr) |
| %call1 = call nofpclass(nan zero) float @merged_ret_uses(ptr %ptr) |
| %add = fadd float %call0, %call1 |
| ret float %add |
| } |
| |
| ; Do not infer nofpclass on return |
| define internal float @called_with_wrong_ret_type(ptr %ptr) { |
| ; CHECK-LABEL: define internal float @called_with_wrong_ret_type( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[VAL:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: ret float [[VAL]] |
| ; |
| %val = load volatile float, ptr %ptr |
| ret float %val |
| } |
| |
| define <2 x half> @wrong_callee_ret_type(ptr %ptr) { |
| ; CHECK-LABEL: define <2 x half> @wrong_callee_ret_type( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR1:[0-9]+]] { |
| ; CHECK-NEXT: [[RET:%.*]] = call nofpclass(nan) <2 x half> @called_with_wrong_ret_type(ptr [[PTR]]) |
| ; CHECK-NEXT: ret <2 x half> [[RET]] |
| ; |
| %ret = call nofpclass(nan) <2 x half> @called_with_wrong_ret_type(ptr %ptr) |
| ret <2 x half> %ret |
| } |
| |
| ; Do not infer nofpclass on return |
| define internal float @non_callee_use_ret(ptr %ptr) { |
| ; CHECK-LABEL: define internal float @non_callee_use_ret( |
| ; CHECK-SAME: ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[VAL:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: ret float [[VAL]] |
| ; |
| %val = load volatile float, ptr %ptr |
| ret float %val |
| } |
| |
| declare float @uses_func_ptr(ptr) |
| |
| define float @caller_non_callee_use(ptr %ptr) { |
| ; CHECK-LABEL: define float @caller_non_callee_use( |
| ; CHECK-SAME: ptr readnone captures(none) [[PTR:%.*]]) { |
| ; CHECK-NEXT: [[RET:%.*]] = call nofpclass(nan) float @uses_func_ptr(ptr @non_callee_use_ret) |
| ; CHECK-NEXT: ret float [[RET]] |
| ; |
| %ret = call nofpclass(nan) float @uses_func_ptr(ptr @non_callee_use_ret) |
| ret float %ret |
| } |
| |
| define internal { double, double } @mixed_ret_and_args(float %scalar.arg, i32 %not.fp, <2 x double> %vec, { double, double } %struct) { |
| ; CHECK-LABEL: define internal nofpclass(sub) { double, double } @mixed_ret_and_args( |
| ; CHECK-SAME: float nofpclass(nan) [[SCALAR_ARG:%.*]], i32 [[NOT_FP:%.*]], <2 x double> nofpclass(inf) [[VEC:%.*]], { double, double } nofpclass(zero) [[STRUCT:%.*]]) #[[ATTR2:[0-9]+]] { |
| ; CHECK-NEXT: [[FPEXT:%.*]] = fpext float [[SCALAR_ARG]] to double |
| ; CHECK-NEXT: [[VEC_0:%.*]] = extractelement <2 x double> [[VEC]], i32 0 |
| ; CHECK-NEXT: [[INSERT_0:%.*]] = insertvalue { double, double } poison, double [[FPEXT]], 0 |
| ; CHECK-NEXT: [[INSERT_1:%.*]] = insertvalue { double, double } poison, double [[VEC_0]], 0 |
| ; CHECK-NEXT: ret { double, double } [[INSERT_1]] |
| ; |
| %fpext = fpext float %scalar.arg to double |
| %vec.0 = extractelement <2 x double> %vec, i32 0 |
| %insert.0 = insertvalue { double, double } poison, double %fpext, 0 |
| %insert.1 = insertvalue { double, double } poison, double %vec.0, 0 |
| ret { double, double } %insert.1 |
| } |
| |
| define void @call_mixed_ret_and_args_0(float %a, i32 %b, <2 x double> %c, { double, double } %d) { |
| ; CHECK-LABEL: define void @call_mixed_ret_and_args_0( |
| ; CHECK-SAME: float [[A:%.*]], i32 [[B:%.*]], <2 x double> [[C:%.*]], { double, double } [[D:%.*]]) #[[ATTR2]] { |
| ; CHECK-NEXT: [[RESULT:%.*]] = call nofpclass(sub) { double, double } @mixed_ret_and_args(float nofpclass(nan) [[A]], i32 [[B]], <2 x double> nofpclass(inf) [[C]], { double, double } nofpclass(zero) [[D]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %result = call nofpclass(sub) { double, double } @mixed_ret_and_args(float nofpclass(nan) %a, i32 %b, <2 x double> nofpclass(inf) %c, { double, double } nofpclass(zero) %d) |
| ret void |
| } |
| |
| define internal float @merged_arg_uses(float %arg0, half %arg1, half %arg2) { |
| ; CHECK-LABEL: define internal float @merged_arg_uses( |
| ; CHECK-SAME: float nofpclass(nan) [[ARG0:%.*]], half [[ARG1:%.*]], half [[ARG2:%.*]]) #[[ATTR2]] { |
| ; CHECK-NEXT: [[FPEXT0:%.*]] = fpext half [[ARG1]] to float |
| ; CHECK-NEXT: [[FPEXT1:%.*]] = fpext half [[ARG1]] to float |
| ; CHECK-NEXT: [[ADD0:%.*]] = fadd float [[ARG0]], [[FPEXT0]] |
| ; CHECK-NEXT: [[ADD1:%.*]] = fadd float [[ADD0]], [[FPEXT1]] |
| ; CHECK-NEXT: ret float [[ADD1]] |
| ; |
| %fpext0 = fpext half %arg1 to float |
| %fpext1 = fpext half %arg1 to float |
| %add0 = fadd float %arg0, %fpext0 |
| %add1 = fadd float %add0, %fpext1 |
| ret float %add1 |
| } |
| |
| define float @calls_merged_arg_uses(float %arg0, half %arg1, half %arg2) { |
| ; CHECK-LABEL: define float @calls_merged_arg_uses( |
| ; CHECK-SAME: float [[ARG0:%.*]], half [[ARG1:%.*]], half [[ARG2:%.*]]) #[[ATTR2]] { |
| ; CHECK-NEXT: [[CALL0:%.*]] = call float @merged_arg_uses(float nofpclass(nan inf) [[ARG0]], half nofpclass(zero) [[ARG1]], half nofpclass(zero) [[ARG2]]) |
| ; CHECK-NEXT: [[CALL1:%.*]] = call float @merged_arg_uses(float nofpclass(nan) [[ARG0]], half nofpclass(sub) [[ARG1]], half [[ARG2]]) |
| ; CHECK-NEXT: [[RET:%.*]] = fadd float [[CALL0]], [[CALL1]] |
| ; CHECK-NEXT: ret float [[RET]] |
| ; |
| %call0 = call float @merged_arg_uses(float nofpclass(inf nan) %arg0, half nofpclass(zero) %arg1, half nofpclass(zero) %arg2) |
| %call1 = call float @merged_arg_uses(float nofpclass(nan) %arg0, half nofpclass(sub) %arg1, half %arg2) |
| %ret = fadd float %call0, %call1 |
| ret float %ret |
| } |
| |
| define internal float @self_recursive_callsite_attrs(float %x) { |
| ; CHECK-LABEL: define internal float @self_recursive_callsite_attrs( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[RET:%.*]] = call nofpclass(inf) float @self_recursive_callsite_attrs(float nofpclass(nan) [[X]]) |
| ; CHECK-NEXT: ret float [[RET]] |
| ; |
| %ret = call nofpclass(inf) float @self_recursive_callsite_attrs(float nofpclass(nan) %x) |
| ret float %ret |
| } |
| |
| define internal float @mutually_recursive0(float %arg) { |
| ; CHECK-LABEL: define internal float @mutually_recursive0( |
| ; CHECK-SAME: float [[ARG:%.*]]) { |
| ; CHECK-NEXT: [[CALL:%.*]] = call nofpclass(nan) float @mutually_recursive1(float [[ARG]]) |
| ; CHECK-NEXT: ret float [[CALL]] |
| ; |
| %call = call nofpclass(nan) float @mutually_recursive1(float %arg) |
| ret float %call |
| } |
| |
| define internal float @mutually_recursive1(float %arg) { |
| ; CHECK-LABEL: define internal float @mutually_recursive1( |
| ; CHECK-SAME: float [[ARG:%.*]]) { |
| ; CHECK-NEXT: [[CALL:%.*]] = call float @mutually_recursive0(float [[ARG]]) |
| ; CHECK-NEXT: ret float [[CALL]] |
| ; |
| %call = call float @mutually_recursive0(float %arg) |
| ret float %call |
| } |
| |
| define internal void @infer_arg_from_constants(float %a, <2 x half> %b, float %c, float %d) { |
| ; CHECK-LABEL: define internal void @infer_arg_from_constants( |
| ; CHECK-SAME: float nofpclass(nan inf nzero sub norm) [[A:%.*]], <2 x half> [[B:%.*]], float [[C:%.*]], float nofpclass(snan inf zero sub norm) [[D:%.*]]) #[[ATTR2]] { |
| ; CHECK-NEXT: ret void |
| ; |
| ret void |
| } |
| |
| define void @call_infer_arg_from_constants() { |
| ; CHECK-LABEL: define void @call_infer_arg_from_constants( |
| ; CHECK-SAME: ) #[[ATTR2]] { |
| ; CHECK-NEXT: call void @infer_arg_from_constants(float 0.000000e+00, <2 x half> <half 0xH3C00, half 0xHBC00>, float poison, float 0x7FF8000000000000) |
| ; CHECK-NEXT: ret void |
| ; |
| call void @infer_arg_from_constants(float 0.0, <2 x half> <half 1.0, half -1.0>, float poison, float 0x7FF8000000000000) |
| ret void |
| } |
| |
| define internal void @infer_arg_from_load(float %arg) { |
| ; CHECK-LABEL: define internal void @infer_arg_from_load( |
| ; CHECK-SAME: float [[ARG:%.*]]) #[[ATTR2]] { |
| ; CHECK-NEXT: ret void |
| ; |
| ret void |
| } |
| |
| define void @call_infer_arg_from_load(ptr %ptr) { |
| ; CHECK-LABEL: define void @call_infer_arg_from_load( |
| ; CHECK-SAME: ptr readonly captures(none) [[PTR:%.*]]) #[[ATTR3:[0-9]+]] { |
| ; CHECK-NEXT: [[NOT_NAN:%.*]] = load float, ptr [[PTR]], align 4, !nofpclass [[META0:![0-9]+]] |
| ; CHECK-NEXT: call void @infer_arg_from_load(float [[NOT_NAN]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %not.nan = load float, ptr %ptr, !nofpclass !{i32 3} |
| call void @infer_arg_from_load(float %not.nan) |
| ret void |
| } |
| |
| ; Expand ret nofpclass(inf nan), arg to nofpclass(inf zero) |
| define internal nofpclass(nan) float @refine_existing_nofpclass(float nofpclass(inf) %arg, ptr %ptr) { |
| ; CHECK-LABEL: define internal nofpclass(nan inf) float @refine_existing_nofpclass( |
| ; CHECK-SAME: float nofpclass(inf zero) [[ARG:%.*]], ptr readonly captures(none) [[PTR:%.*]]) #[[ATTR3]] { |
| ; CHECK-NEXT: [[LOAD:%.*]] = load float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: ret float [[LOAD]] |
| ; |
| %load = load float, ptr %ptr |
| ret float %load |
| } |
| |
| define float @caller_refine_existing_nofpclass(float %arg, ptr %ptr) { |
| ; CHECK-LABEL: define float @caller_refine_existing_nofpclass( |
| ; CHECK-SAME: float [[ARG:%.*]], ptr readonly captures(none) [[PTR:%.*]]) #[[ATTR3]] { |
| ; CHECK-NEXT: [[CALL:%.*]] = call nofpclass(inf) float @refine_existing_nofpclass(float nofpclass(zero) [[ARG]], ptr [[PTR]]) |
| ; CHECK-NEXT: ret float [[CALL]] |
| ; |
| %call = call nofpclass(inf) float @refine_existing_nofpclass(float nofpclass(zero) %arg, ptr %ptr) |
| ret float %call |
| } |
| |
| ; Do not infer nofpclass |
| define internal float @nofpclass_non_call_user(float %arg, ptr %ptr) { |
| ; CHECK-LABEL: define internal float @nofpclass_non_call_user( |
| ; CHECK-SAME: float [[ARG:%.*]], ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[LOAD:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: [[ADD:%.*]] = fadd float [[ARG]], [[LOAD]] |
| ; CHECK-NEXT: ret float [[ADD]] |
| ; |
| %load = load volatile float, ptr %ptr |
| %add = fadd float %arg, %load |
| ret float %add |
| } |
| |
| define float @call_nofpclass_non_call_user(float %arg, ptr %ptr, ptr %fptr.ptr) { |
| ; CHECK-LABEL: define float @call_nofpclass_non_call_user( |
| ; CHECK-SAME: float [[ARG:%.*]], ptr [[PTR:%.*]], ptr writeonly captures(none) initializes((0, 8)) [[FPTR_PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[RET:%.*]] = call nofpclass(nan) float @nofpclass_non_call_user(float nofpclass(nan) [[ARG]], ptr [[PTR]]) |
| ; CHECK-NEXT: store ptr @nofpclass_non_call_user, ptr [[FPTR_PTR]], align 8 |
| ; CHECK-NEXT: ret float [[RET]] |
| ; |
| %ret = call nofpclass(nan) float @nofpclass_non_call_user(float nofpclass(nan) %arg, ptr %ptr) |
| store ptr @nofpclass_non_call_user, ptr %fptr.ptr |
| ret float %ret |
| } |
| |
| ; TODO: This case is missed |
| define internal float @transitive_nonan_callee0(float %arg, ptr %ptr) { |
| ; CHECK-LABEL: define internal float @transitive_nonan_callee0( |
| ; CHECK-SAME: float [[ARG:%.*]], ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[LOAD:%.*]] = load volatile float, ptr [[PTR]], align 4 |
| ; CHECK-NEXT: [[ADD:%.*]] = fadd float [[ARG]], [[LOAD]] |
| ; CHECK-NEXT: ret float [[ADD]] |
| ; |
| %load = load volatile float, ptr %ptr |
| %add = fadd float %arg, %load |
| ret float %add |
| } |
| |
| define internal float @transitive_nonan_callee1(float %arg, ptr %ptr) { |
| ; CHECK-LABEL: define internal nofpclass(nan) float @transitive_nonan_callee1( |
| ; CHECK-SAME: float nofpclass(nan) [[ARG:%.*]], ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[RET:%.*]] = call float @transitive_nonan_callee0(float [[ARG]], ptr [[PTR]]) |
| ; CHECK-NEXT: ret float [[RET]] |
| ; |
| %ret = call float @transitive_nonan_callee0(float %arg, ptr %ptr) |
| ret float %ret |
| } |
| |
| define float @caller_transitive_nonan(float %arg, ptr %ptr) { |
| ; CHECK-LABEL: define float @caller_transitive_nonan( |
| ; CHECK-SAME: float [[ARG:%.*]], ptr [[PTR:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: [[RET:%.*]] = call nofpclass(nan) float @transitive_nonan_callee1(float nofpclass(nan) [[ARG]], ptr [[PTR]]) |
| ; CHECK-NEXT: ret float [[RET]] |
| ; |
| %ret = call nofpclass(nan) float @transitive_nonan_callee1(float nofpclass(nan) %arg, ptr %ptr) |
| ret float %ret |
| } |
| ;. |
| ; CHECK: attributes #[[ATTR0]] = { mustprogress nofree norecurse nounwind willreturn memory(argmem: readwrite, inaccessiblemem: readwrite) } |
| ; CHECK: attributes #[[ATTR1]] = { mustprogress nofree nounwind willreturn memory(argmem: readwrite, inaccessiblemem: readwrite) } |
| ; CHECK: attributes #[[ATTR2]] = { mustprogress nofree norecurse nosync nounwind willreturn memory(none) } |
| ; CHECK: attributes #[[ATTR3]] = { mustprogress nofree norecurse nosync nounwind willreturn memory(argmem: read) } |
| ;. |
| ; CHECK: [[META0]] = !{i32 3} |
| ;. |