; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --check-attributes --check-globals all --version 5 ; RUN: opt -S -passes=early-cse < %s | FileCheck %s declare i8 @baz(i8, i8) declare i8 @baz_side_effects(i8, i8) declare i8 @buz(i8, i8) declare ptr @baz.ptr(i8, i8) declare i8 @buz.ptr(ptr, ptr) declare float @baz.fp(float, float) declare i8 @buz.fp(float, float) define i8 @same_parent_combine_diff_attrs(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1:[0-9]+]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C1]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call i8 @baz(i8 noundef %x, i8 noundef %y) readnone %c0 = call i8 @baz(i8 %x, i8 noundef %y) readnone %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_needs_intersect(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_needs_intersect( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call ptr @baz.ptr(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz.ptr(ptr [[C1]], ptr [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call nonnull ptr @baz.ptr(i8 noundef %x, i8 noundef %y) readnone %c0 = call ptr @baz.ptr(i8 %x, i8 noundef %y) readnone %r = call i8 @buz.ptr(ptr %c0, ptr %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_fmf(float %x, float %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_fmf( ; CHECK-SAME: float [[X:%.*]], float [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call nnan float @baz.fp(float [[X]], float noundef [[Y]]) #[[ATTR2:[0-9]+]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz.fp(float [[C1]], float [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call nnan nsz float @baz.fp(float noundef %x, float noundef %y) readonly %c0 = call nnan float @baz.fp(float %x, float noundef %y) readonly %r = call i8 @buz.fp(float %c0, float %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_fmf2(float %x, float %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_fmf2( ; CHECK-SAME: float [[X:%.*]], float [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call nnan float @baz.fp(float [[X]], float noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz.fp(float [[C1]], float [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call nnan float @baz.fp(float noundef %x, float noundef %y) readnone %c0 = call nnan nsz float @baz.fp(float %x, float noundef %y) readnone %r = call i8 @buz.fp(float %c0, float %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_needs_intersect2(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_needs_intersect2( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call ptr @baz.ptr(i8 [[X]], i8 noundef [[Y]]) #[[ATTR2]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz.ptr(ptr [[C1]], ptr [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call nonnull ptr @baz.ptr(i8 noundef %x, i8 noundef %y) readonly %c0 = call ptr @baz.ptr(i8 %x, i8 noundef %y) readonly %r = call i8 @buz.ptr(ptr %c0, ptr %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_really_needs_intersect(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_really_needs_intersect( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call ptr @baz.ptr(i8 [[X]], i8 noundef [[Y]]) #[[ATTR2]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz.ptr(ptr [[C1]], ptr noundef [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call nonnull ptr @baz.ptr(i8 noundef %x, i8 noundef %y) readonly %c0 = call ptr @baz.ptr(i8 %x, i8 noundef %y) readonly %r = call i8 @buz.ptr(ptr %c0, ptr noundef %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_fail_side_effects(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_fail_side_effects( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call i8 @baz(i8 noundef %x, i8 noundef %y) %c0 = call i8 @baz(i8 %x, i8 noundef %y) %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_quasi_side_effects2(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_quasi_side_effects2( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call i8 @baz(i8 noundef %x, i8 noundef %y) readnone %c0 = call i8 @baz(i8 %x, i8 noundef %y) %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r } define i8 @diff_parent_combine_diff_attrs(i1 %c, i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @diff_parent_combine_diff_attrs( ; CHECK-SAME: i1 [[C:%.*]], i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: br i1 [[C]], label %[[T:.*]], label %[[F:.*]] ; CHECK: [[T]]: ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) #[[ATTR2]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; CHECK: [[F]]: ; CHECK-NEXT: [[R2:%.*]] = add i8 [[C1]], 4 ; CHECK-NEXT: ret i8 [[R2]] ; %c1 = call i8 @baz(i8 %x, i8 noundef %y) readnone br i1 %c, label %T, label %F T: %c0 = call i8 @baz(i8 noundef %x, i8 noundef %y) readonly %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r F: %r2 = add i8 %c1, 4 ret i8 %r2 } define i8 @diff_parent_combine_diff_attrs_preserves_return_attrs(i1 %c, i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @diff_parent_combine_diff_attrs_preserves_return_attrs( ; CHECK-SAME: i1 [[C:%.*]], i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call nonnull ptr @baz.ptr(i8 [[X]], i8 noundef [[Y]]) #[[ATTR2]] ; CHECK-NEXT: br i1 [[C]], label %[[T:.*]], label %[[F:.*]] ; CHECK: [[T]]: ; CHECK-NEXT: [[R:%.*]] = call i8 @buz.ptr(ptr [[C1]], ptr noundef [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; CHECK: [[F]]: ; CHECK-NEXT: ret i8 9 ; %c1 = call nonnull ptr @baz.ptr(i8 %x, i8 noundef %y) readonly br i1 %c, label %T, label %F T: %c0 = call nonnull ptr @baz.ptr(i8 noundef %x, i8 noundef %y) readonly %r = call i8 @buz.ptr(ptr %c0, ptr noundef %c1) ret i8 %r F: ret i8 9 } define i8 @same_parent_combine_diff_attrs_todo(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_todo( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) #[[ATTR3:[0-9]+]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call i8 @baz(i8 %x, i8 noundef %y) readnone %c0 = call i8 @baz(i8 noundef %x, i8 noundef %y) readnone alwaysinline %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r } define i8 @same_parent_combine_diff_attrs_fail(i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @same_parent_combine_diff_attrs_fail( ; CHECK-SAME: i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) #[[ATTR4:[0-9]+]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; %c1 = call i8 @baz(i8 %x, i8 noundef %y) readnone %c0 = call i8 @baz(i8 noundef %x, i8 noundef %y) readnone strictfp %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r } define i8 @diff_parent_combine_diff_attrs_todo(i1 %c, i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @diff_parent_combine_diff_attrs_todo( ; CHECK-SAME: i1 [[C:%.*]], i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: br i1 [[C]], label %[[T:.*]], label %[[F:.*]] ; CHECK: [[T]]: ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) #[[ATTR5:[0-9]+]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; CHECK: [[F]]: ; CHECK-NEXT: [[R2:%.*]] = add i8 [[C1]], 4 ; CHECK-NEXT: ret i8 [[R2]] ; %c1 = call i8 @baz(i8 %x, i8 noundef %y) readnone br i1 %c, label %T, label %F T: %c0 = call i8 @baz(i8 noundef %x, i8 noundef %y) readnone optnone noinline %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r F: %r2 = add i8 %c1, 4 ret i8 %r2 } define i8 @diff_parent_combine_diff_attrs_fail(i1 %c, i8 %x, i8 %y) { ; CHECK-LABEL: define i8 @diff_parent_combine_diff_attrs_fail( ; CHECK-SAME: i1 [[C:%.*]], i8 [[X:%.*]], i8 [[Y:%.*]]) { ; CHECK-NEXT: [[C1:%.*]] = call i8 @baz(i8 [[X]], i8 noundef [[Y]]) #[[ATTR1]] ; CHECK-NEXT: br i1 [[C]], label %[[T:.*]], label %[[F:.*]] ; CHECK: [[T]]: ; CHECK-NEXT: [[C0:%.*]] = call i8 @baz(i8 noundef [[X]], i8 noundef [[Y]]) #[[ATTR4]] ; CHECK-NEXT: [[R:%.*]] = call i8 @buz(i8 [[C0]], i8 [[C1]]) ; CHECK-NEXT: ret i8 [[R]] ; CHECK: [[F]]: ; CHECK-NEXT: [[R2:%.*]] = add i8 [[C1]], 4 ; CHECK-NEXT: ret i8 [[R2]] ; %c1 = call i8 @baz(i8 %x, i8 noundef %y) readnone br i1 %c, label %T, label %F T: %c0 = call i8 @baz(i8 noundef %x, i8 noundef %y) readnone strictfp %r = call i8 @buz(i8 %c0, i8 %c1) ret i8 %r F: %r2 = add i8 %c1, 4 ret i8 %r2 } define i32 @commutative_intrinsic_intersection_failure(i32 %arg, i32 %arg1) { ; CHECK-LABEL: define i32 @commutative_intrinsic_intersection_failure( ; CHECK-SAME: i32 [[ARG:%.*]], i32 [[ARG1:%.*]]) { ; CHECK-NEXT: [[CALL:%.*]] = call i32 @llvm.smin.i32(i32 [[ARG]], i32 [[ARG1]]) #[[ATTR6:[0-9]+]] ; CHECK-NEXT: [[CALL2:%.*]] = call i32 @llvm.smin.i32(i32 [[ARG1]], i32 [[ARG]]) ; CHECK-NEXT: [[OR:%.*]] = or i32 [[CALL2]], [[CALL]] ; CHECK-NEXT: ret i32 [[OR]] ; %call = call i32 @llvm.smin.i32(i32 %arg, i32 %arg1) strictfp %call2 = call i32 @llvm.smin.i32(i32 %arg1, i32 %arg) %or = or i32 %call2, %call ret i32 %or } ;. ; CHECK: attributes #[[ATTR0:[0-9]+]] = { nocallback nofree nosync nounwind speculatable willreturn memory(none) } ; CHECK: attributes #[[ATTR1]] = { memory(none) } ; CHECK: attributes #[[ATTR2]] = { memory(read) } ; CHECK: attributes #[[ATTR3]] = { alwaysinline memory(none) } ; CHECK: attributes #[[ATTR4]] = { strictfp memory(none) } ; CHECK: attributes #[[ATTR5]] = { noinline optnone memory(none) } ; CHECK: attributes #[[ATTR6]] = { strictfp } ;.