145 lines
5.6 KiB
LLVM
145 lines
5.6 KiB
LLVM
; 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
|
|
}
|