| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py |
| ; RUN: opt < %s -passes=instcombine -S | FileCheck %s |
| |
| define i64 @src_2add_2sext_sub_1(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_1( |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_2(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_2( |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %z, %x |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_3(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_3( |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %y, %x |
| %add2 = add nsw i32 %z, %x |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_4(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_4( |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %y, %x |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sextlike_sub(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sextlike_sub( |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[SEXT1]], [[SEXT2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = zext nneg i32 %add1 to i64 |
| %sext2 = zext nneg i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_nsw(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_nsw( |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[SEXT1]], [[SEXT2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub nsw i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_nuw(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_nuw( |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub nuw i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| declare void @use_i32(i32, i32) |
| declare void @use_i64(i64, i64) |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_1(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_1( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], [[Y:%.*]] |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[ADD1]] to i64 |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT1]], i64 [[SEXT1]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i64(i64 %sext1, i64 %sext1) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_2(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_2( |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X:%.*]], [[Z:%.*]] |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[ADD2]] to i64 |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT2]], i64 [[SEXT2]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i64(i64 %sext2, i64 %sext2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_3(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_3( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], [[Y:%.*]] |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X]], [[Z:%.*]] |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[ADD1]] to i64 |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[ADD2]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[SEXT1]], [[SEXT2]] |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT1]], i64 [[SEXT2]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i64(i64 %sext1, i64 %sext2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_4(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_4( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], [[Y:%.*]] |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X]], [[Z:%.*]] |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: call void @use_i32(i32 [[ADD1]], i32 [[ADD2]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i32(i32 %add1, i32 %add2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_5(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_5( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], [[Y:%.*]] |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[ADD1]] to i64 |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z:%.*]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: call void @use_i32(i32 [[ADD1]], i32 [[ADD1]]) |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT1]], i64 [[SEXT1]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i32(i32 %add1, i32 %add1) |
| call void @use_i64(i64 %sext1, i64 %sext1) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_6(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_6( |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X:%.*]], [[Z:%.*]] |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[ADD2]] to i64 |
| ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[Z]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[TMP1]], [[TMP2]] |
| ; CHECK-NEXT: call void @use_i32(i32 [[ADD2]], i32 [[ADD2]]) |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT2]], i64 [[SEXT2]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i32(i32 %add2, i32 %add2) |
| call void @use_i64(i64 %sext2, i64 %sext2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_7(i32 %x, i32 %y, i32 %z) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_7( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], [[Y:%.*]] |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X]], [[Z:%.*]] |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[ADD1]] to i64 |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[ADD2]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[SEXT1]], [[SEXT2]] |
| ; CHECK-NEXT: call void @use_i32(i32 [[ADD1]], i32 [[ADD2]]) |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT1]], i64 [[SEXT2]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, %z |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i32(i32 %add1, i32 %add2) |
| call void @use_i64(i64 %sext1, i64 %sext2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_8(i32 %x, i32 %y) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_8( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], [[Y:%.*]] |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X]], 1 |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[ADD1]] to i64 |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[ADD2]] to i64 |
| ; CHECK-NEXT: [[SUB:%.*]] = sub nsw i64 [[SEXT1]], [[SEXT2]] |
| ; CHECK-NEXT: call void @use_i32(i32 [[ADD1]], i32 [[ADD2]]) |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT1]], i64 [[SEXT2]]) |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %add2 = add nsw i32 %x, 1 |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i32(i32 %add1, i32 %add2) |
| call void @use_i64(i64 %sext1, i64 %sext2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_2add_2sext_sub_multiple_uses_9(i32 %x) { |
| ; CHECK-LABEL: @src_2add_2sext_sub_multiple_uses_9( |
| ; CHECK-NEXT: [[ADD1:%.*]] = add nsw i32 [[X:%.*]], 2 |
| ; CHECK-NEXT: [[ADD2:%.*]] = add nsw i32 [[X]], 1 |
| ; CHECK-NEXT: [[SEXT1:%.*]] = sext i32 [[ADD1]] to i64 |
| ; CHECK-NEXT: [[SEXT2:%.*]] = sext i32 [[ADD2]] to i64 |
| ; CHECK-NEXT: call void @use_i32(i32 [[ADD1]], i32 [[ADD2]]) |
| ; CHECK-NEXT: call void @use_i64(i64 [[SEXT1]], i64 [[SEXT2]]) |
| ; CHECK-NEXT: ret i64 1 |
| ; |
| %add1 = add nsw i32 %x, 2 |
| %add2 = add nsw i32 %x, 1 |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %add2 to i64 |
| %sub = sub i64 %sext1, %sext2 |
| call void @use_i32(i32 %add1, i32 %add2) |
| call void @use_i64(i64 %sext1, i64 %sext2) |
| ret i64 %sub |
| } |
| |
| define i64 @src_x_add_2sext_sub_1(i32 %x, i32 %y) { |
| ; CHECK-LABEL: @src_x_add_2sext_sub_1( |
| ; CHECK-NEXT: [[SUB:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %x to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_x_add_2sext_sub_2(i32 %x, i32 %y) { |
| ; CHECK-LABEL: @src_x_add_2sext_sub_2( |
| ; CHECK-NEXT: [[SUB:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %y, %x |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %x to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_x_add_2sextlike_sub(i32 %x, i32 %y) { |
| ; CHECK-LABEL: @src_x_add_2sextlike_sub( |
| ; CHECK-NEXT: [[SUB:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %sext1 = zext nneg i32 %add1 to i64 |
| %sext2 = zext nneg i32 %x to i64 |
| %sub = sub i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_x_add_2sext_sub_nsw(i32 %x, i32 %y) { |
| ; CHECK-LABEL: @src_x_add_2sext_sub_nsw( |
| ; CHECK-NEXT: [[SUB:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %x to i64 |
| %sub = sub nsw i64 %sext1, %sext2 |
| ret i64 %sub |
| } |
| |
| define i64 @src_x_add_2sext_sub_nuw(i32 %x, i32 %y) { |
| ; CHECK-LABEL: @src_x_add_2sext_sub_nuw( |
| ; CHECK-NEXT: [[SUB:%.*]] = sext i32 [[Y:%.*]] to i64 |
| ; CHECK-NEXT: ret i64 [[SUB]] |
| ; |
| %add1 = add nsw i32 %x, %y |
| %sext1 = sext i32 %add1 to i64 |
| %sext2 = sext i32 %x to i64 |
| %sub = sub nuw i64 %sext1, %sext2 |
| ret i64 %sub |
| } |