blob: fa1c52fd42820504224725b70ec2bdba62ddf4cd [file] [log] [blame]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt < %s -passes=instcombine -S | FileCheck %s
;
; Exercise folding of memcmp calls with addition expressions involving
; pointers into constant arrays of types larger than char and fractional
; offsets.
declare i32 @memcmp(i8*, i8*, i64)
@i32a = constant [2 x i16] [i16 4386, i16 13124]
@i32b = constant [2 x i16] [i16 4386, i16 13124]
define void @fold_memcmp_i32a_i32b_pIb(i32 %I, i32* %pcmp)
; CHECK-LABEL: @fold_memcmp_i32a_i32b_pIb(
; CHECK-NEXT: store i32 0, i32* [[PCMP:%.*]], align 4
; CHECK-NEXT: [[PST_1_1_2:%.*]] = getelementptr i32, i32* [[PCMP]], i64 1
; CHECK-NEXT: store i32 0, i32* [[PST_1_1_2]], align 4
; CHECK-NEXT: [[PST_1_1_3:%.*]] = getelementptr i32, i32* [[PCMP]], i64 2
; CHECK-NEXT: store i32 0, i32* [[PST_1_1_3]], align 4
; CHECK-NEXT: ret void
;
{
%pi32a = getelementptr [2 x i16], [2 x i16]* @i32a, i32 0, i32 0
%pi32b = getelementptr [2 x i16], [2 x i16]* @i32b, i32 0, i32 0
%pi8a = bitcast i16* %pi32a to i8*
%pi8b = bitcast i16* %pi32b to i8*
%pi8ap1 = getelementptr i8, i8* %pi8a, i32 1
%pi8bp1 = getelementptr i8, i8* %pi8b, i32 1
%pst_1_1_1 = getelementptr i32, i32* %pcmp, i32 0
%cmp_1_1_1 = call i32 @memcmp(i8* %pi8ap1, i8* %pi8ap1, i64 1)
store i32 %cmp_1_1_1, i32* %pst_1_1_1
%pst_1_1_2 = getelementptr i32, i32* %pcmp, i32 1
%cmp_1_1_2 = call i32 @memcmp(i8* %pi8ap1, i8* %pi8ap1, i64 2)
store i32 %cmp_1_1_2, i32* %pst_1_1_2
%pst_1_1_3 = getelementptr i32, i32* %pcmp, i32 2
%cmp_1_1_3 = call i32 @memcmp(i8* %pi8ap1, i8* %pi8ap1, i64 3)
store i32 %cmp_1_1_3, i32* %pst_1_1_3
ret void
}
%struct.A = type { [4 x i8] }
%struct.B = type { [2 x i8], [2 x i8] }
@a = constant [1 x %struct.A] [%struct.A { [4 x i8] [i8 1, i8 2, i8 3, i8 4] }]
@b = constant [1 x %struct.B] [%struct.B { [2 x i8] [i8 1, i8 2], [2 x i8] [i8 3, i8 4]}]
define void @fold_memcmp_A_B_pIb(i32 %I, i32* %pcmp) {
; CHECK-LABEL: @fold_memcmp_A_B_pIb(
; CHECK-NEXT: store i32 0, i32* [[PCMP:%.*]], align 4
; CHECK-NEXT: [[PST_0_0_2:%.*]] = getelementptr i32, i32* [[PCMP]], i64 1
; CHECK-NEXT: store i32 0, i32* [[PST_0_0_2]], align 4
; CHECK-NEXT: [[PST_0_0_3:%.*]] = getelementptr i32, i32* [[PCMP]], i64 2
; CHECK-NEXT: store i32 0, i32* [[PST_0_0_3]], align 4
; CHECK-NEXT: [[PST_0_0_4:%.*]] = getelementptr i32, i32* [[PCMP]], i64 3
; CHECK-NEXT: store i32 0, i32* [[PST_0_0_4]], align 4
; CHECK-NEXT: [[PST_0_1_1:%.*]] = getelementptr i32, i32* [[PCMP]], i64 4
; CHECK-NEXT: store i32 -1, i32* [[PST_0_1_1]], align 4
; CHECK-NEXT: [[PST_0_1_2:%.*]] = getelementptr i32, i32* [[PCMP]], i64 5
; CHECK-NEXT: store i32 -1, i32* [[PST_0_1_2]], align 4
; CHECK-NEXT: [[PST_0_1_3:%.*]] = getelementptr i32, i32* [[PCMP]], i64 6
; CHECK-NEXT: store i32 -1, i32* [[PST_0_1_3]], align 4
; CHECK-NEXT: [[PST_1_0_1:%.*]] = getelementptr i32, i32* [[PCMP]], i64 4
; CHECK-NEXT: store i32 1, i32* [[PST_1_0_1]], align 4
; CHECK-NEXT: [[PST_1_0_2:%.*]] = getelementptr i32, i32* [[PCMP]], i64 5
; CHECK-NEXT: store i32 1, i32* [[PST_1_0_2]], align 4
; CHECK-NEXT: [[PST_1_0_3:%.*]] = getelementptr i32, i32* [[PCMP]], i64 6
; CHECK-NEXT: store i32 1, i32* [[PST_1_0_3]], align 4
; CHECK-NEXT: ret void
;
%pa = getelementptr [1 x %struct.A], [1 x %struct.A]* @a, i64 0, i64 0
%pb = getelementptr [1 x %struct.B], [1 x %struct.B]* @b, i64 0, i64 0
%pi8a = bitcast %struct.A* %pa to i8*
%pi8b = bitcast %struct.B* %pb to i8*
%pi8ap0 = getelementptr i8, i8* %pi8a, i32 0
%pi8bp0 = getelementptr i8, i8* %pi8b, i32 0
; Fold memcmp(&a, &b, 1) to 0;
%pst_0_0_1 = getelementptr i32, i32* %pcmp, i32 0
%cmp_0_0_1 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp0, i64 1)
store i32 %cmp_0_0_1, i32* %pst_0_0_1
; Fold memcmp(&a, &b, 2) to 0;
%pst_0_0_2 = getelementptr i32, i32* %pcmp, i32 1
%cmp_0_0_2 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp0, i64 2)
store i32 %cmp_0_0_2, i32* %pst_0_0_2
; Fold memcmp(&a, &b, 3) to 0;
%pst_0_0_3 = getelementptr i32, i32* %pcmp, i32 2
%cmp_0_0_3 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp0, i64 3)
store i32 %cmp_0_0_3, i32* %pst_0_0_3
; Fold memcmp(&a, &b, 4) to 0;
%pst_0_0_4 = getelementptr i32, i32* %pcmp, i32 3
%cmp_0_0_4 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp0, i64 4)
store i32 %cmp_0_0_4, i32* %pst_0_0_4
%pi8bp1 = getelementptr i8, i8* %pi8b, i32 1
; Fold memcmp(&a, (char*)&b + 1, 1) to -1;
%pst_0_1_1 = getelementptr i32, i32* %pcmp, i32 4
%cmp_0_1_1 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp1, i64 1)
store i32 %cmp_0_1_1, i32* %pst_0_1_1
; Fold memcmp(&a, (char*)&b + 1, 2) to -1;
%pst_0_1_2 = getelementptr i32, i32* %pcmp, i32 5
%cmp_0_1_2 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp1, i64 2)
store i32 %cmp_0_1_2, i32* %pst_0_1_2
; Fold memcmp(&a, (char*)&b + 1, 3) to -1;
%pst_0_1_3 = getelementptr i32, i32* %pcmp, i32 6
%cmp_0_1_3 = call i32 @memcmp(i8* %pi8ap0, i8* %pi8bp1, i64 3)
store i32 %cmp_0_1_3, i32* %pst_0_1_3
%pi8ap1 = getelementptr i8, i8* %pi8a, i32 1
; Fold memcmp((char*)&a + 1, &b, 1) to +1;
%pst_1_0_1 = getelementptr i32, i32* %pcmp, i32 4
%cmp_1_0_1 = call i32 @memcmp(i8* %pi8ap1, i8* %pi8bp0, i64 1)
store i32 %cmp_1_0_1, i32* %pst_1_0_1
; Fold memcmp((char*)&a + 1, &b, 2) to +1;
%pst_1_0_2 = getelementptr i32, i32* %pcmp, i32 5
%cmp_1_0_2 = call i32 @memcmp(i8* %pi8ap1, i8* %pi8bp0, i64 2)
store i32 %cmp_1_0_2, i32* %pst_1_0_2
; Fold memcmp((char*)&a + 1, &b, 3) to +1;
%pst_1_0_3 = getelementptr i32, i32* %pcmp, i32 6
%cmp_1_0_3 = call i32 @memcmp(i8* %pi8ap1, i8* %pi8bp0, i64 3)
store i32 %cmp_1_0_3, i32* %pst_1_0_3
ret void
}