; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 4 ; RUN: opt < %s -passes='function-attrs' -S | FileCheck %s define i32 @test_ret_constant() { ; CHECK-LABEL: define noundef i32 @test_ret_constant( ; CHECK-SAME: ) #[[ATTR0:[0-9]+]] { ; CHECK-NEXT: ret i32 0 ; ret i32 0 } define i32 @test_ret_poison() { ; CHECK-LABEL: define i32 @test_ret_poison( ; CHECK-SAME: ) #[[ATTR0]] { ; CHECK-NEXT: ret i32 poison ; ret i32 poison } define i32 @test_ret_undef() { ; CHECK-LABEL: define i32 @test_ret_undef( ; CHECK-SAME: ) #[[ATTR0]] { ; CHECK-NEXT: ret i32 undef ; ret i32 undef } define i32 @test_ret_param(i32 %x) { ; CHECK-LABEL: define i32 @test_ret_param( ; CHECK-SAME: i32 returned [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: ret i32 [[X]] ; ret i32 %x } define i32 @test_ret_noundef_param(i32 noundef %x) { ; CHECK-LABEL: define noundef i32 @test_ret_noundef_param( ; CHECK-SAME: i32 noundef returned [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: ret i32 [[X]] ; ret i32 %x } define i32 @test_ret_noundef_expr(i32 noundef %x) { ; CHECK-LABEL: define noundef i32 @test_ret_noundef_expr( ; CHECK-SAME: i32 noundef [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: [[Y:%.*]] = add i32 [[X]], 1 ; CHECK-NEXT: ret i32 [[Y]] ; %y = add i32 %x, 1 ret i32 %y } define i32 @test_ret_create_poison_expr(i32 noundef %x) { ; CHECK-LABEL: define i32 @test_ret_create_poison_expr( ; CHECK-SAME: i32 noundef [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: [[Y:%.*]] = add nsw i32 [[X]], 1 ; CHECK-NEXT: ret i32 [[Y]] ; %y = add nsw i32 %x, 1 ret i32 %y } define i32 @test_ret_freezed(i32 noundef %x) { ; CHECK-LABEL: define noundef i32 @test_ret_freezed( ; CHECK-SAME: i32 noundef [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: [[Y:%.*]] = add nsw i32 [[X]], 1 ; CHECK-NEXT: [[Z:%.*]] = freeze i32 [[Y]] ; CHECK-NEXT: ret i32 [[Z]] ; %y = add nsw i32 %x, 1 %z = freeze i32 %y ret i32 %z } define i32 @test_ret_control_flow(i32 noundef %x) { ; CHECK-LABEL: define noundef i32 @test_ret_control_flow( ; CHECK-SAME: i32 noundef [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: [[COND:%.*]] = icmp eq i32 [[X]], 0 ; CHECK-NEXT: br i1 [[COND]], label [[IF_THEN:%.*]], label [[IF_ELSE:%.*]] ; CHECK: if.then: ; CHECK-NEXT: ret i32 2 ; CHECK: if.else: ; CHECK-NEXT: [[RET:%.*]] = add i32 [[X]], 1 ; CHECK-NEXT: ret i32 [[RET]] ; %cond = icmp eq i32 %x, 0 br i1 %cond, label %if.then, label %if.else if.then: ret i32 2 if.else: %ret = add i32 %x, 1 ret i32 %ret } define i32 @test_ret_control_flow_may_poison(i32 noundef %x) { ; CHECK-LABEL: define i32 @test_ret_control_flow_may_poison( ; CHECK-SAME: i32 noundef [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: [[COND:%.*]] = icmp eq i32 [[X]], 0 ; CHECK-NEXT: br i1 [[COND]], label [[IF_THEN:%.*]], label [[IF_ELSE:%.*]] ; CHECK: if.then: ; CHECK-NEXT: ret i32 2 ; CHECK: if.else: ; CHECK-NEXT: [[RET:%.*]] = add nsw i32 [[X]], 1 ; CHECK-NEXT: ret i32 [[RET]] ; %cond = icmp eq i32 %x, 0 br i1 %cond, label %if.then, label %if.else if.then: ret i32 2 if.else: %ret = add nsw i32 %x, 1 ret i32 %ret } ; TODO: use context-sensitive analysis define i32 @test_ret_control_flow_never_poison(i32 noundef %x) { ; CHECK-LABEL: define i32 @test_ret_control_flow_never_poison( ; CHECK-SAME: i32 noundef [[X:%.*]]) #[[ATTR0]] { ; CHECK-NEXT: [[COND:%.*]] = icmp eq i32 [[X]], 2147483647 ; CHECK-NEXT: br i1 [[COND]], label [[IF_THEN:%.*]], label [[IF_ELSE:%.*]] ; CHECK: if.then: ; CHECK-NEXT: ret i32 2 ; CHECK: if.else: ; CHECK-NEXT: [[RET:%.*]] = add nsw i32 [[X]], 1 ; CHECK-NEXT: ret i32 [[RET]] ; %cond = icmp eq i32 %x, 2147483647 br i1 %cond, label %if.then, label %if.else if.then: ret i32 2 if.else: %ret = add nsw i32 %x, 1 ret i32 %ret } define i32 @test_noundef_prop() { ; CHECK-LABEL: define noundef i32 @test_noundef_prop( ; CHECK-SAME: ) #[[ATTR0]] { ; CHECK-NEXT: [[RET:%.*]] = call i32 @test_ret_constant() ; CHECK-NEXT: ret i32 [[RET]] ; %ret = call i32 @test_ret_constant() ret i32 %ret } ; Don't deduce noundef for functions with sanitize_memory. define i32 @test_ret_constant_msan() sanitize_memory { ; CHECK-LABEL: define i32 @test_ret_constant_msan( ; CHECK-SAME: ) #[[ATTR1:[0-9]+]] { ; CHECK-NEXT: ret i32 0 ; ret i32 0 }