1; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2; Verify that calls to snprintf with members of constant structs as arguments
3; are folded to constants as expected.
4; RUN: opt < %s -passes=instcombine -S | FileCheck %s
5
6declare i32 @snprintf(i8*, i64, i8*, ...)
7
8%struct.A = type { [5 x i8], [6 x i8], [7 x i8] }
9
10@a = constant [2 x %struct.A] [%struct.A { [5 x i8] c"1\00\00\00\00", [6 x i8] c"12\00\00\00\00", [7 x i8] c"123\00\00\00\00" }, %struct.A { [5 x i8] c"1234\00", [6 x i8] c"12345\00", [7 x i8] c"123456\00" }]
11
12@pcnt_s = constant [3 x i8] c"%s\00"
13
14
15; Fold snprintf(0, 0, "%s", a[I].M + C) for constant I in [0, 1],
16; member M in [a, b, c], and C in a valid range to a constant.
17
18define void @fold_snprintf_member_pC(i32* %pi) {
19; CHECK-LABEL: @fold_snprintf_member_pC(
20; CHECK-NEXT:    store i32 1, i32* [[PI:%.*]], align 4
21; CHECK-NEXT:    [[PIA0AP1:%.*]] = getelementptr i32, i32* [[PI]], i64 1
22; CHECK-NEXT:    store i32 0, i32* [[PIA0AP1]], align 4
23; CHECK-NEXT:    [[PIA0B:%.*]] = getelementptr i32, i32* [[PI]], i64 2
24; CHECK-NEXT:    store i32 2, i32* [[PIA0B]], align 4
25; CHECK-NEXT:    [[PIA0BP1:%.*]] = getelementptr i32, i32* [[PI]], i64 3
26; CHECK-NEXT:    store i32 1, i32* [[PIA0BP1]], align 4
27; CHECK-NEXT:    [[PIA0BP2:%.*]] = getelementptr i32, i32* [[PI]], i64 4
28; CHECK-NEXT:    store i32 0, i32* [[PIA0BP2]], align 4
29; CHECK-NEXT:    [[PIA0C:%.*]] = getelementptr i32, i32* [[PI]], i64 5
30; CHECK-NEXT:    store i32 3, i32* [[PIA0C]], align 4
31; CHECK-NEXT:    [[PIA1A:%.*]] = getelementptr i32, i32* [[PI]], i64 6
32; CHECK-NEXT:    store i32 4, i32* [[PIA1A]], align 4
33; CHECK-NEXT:    [[PIA1B:%.*]] = getelementptr i32, i32* [[PI]], i64 7
34; CHECK-NEXT:    store i32 5, i32* [[PIA1B]], align 4
35; CHECK-NEXT:    [[PIA1C:%.*]] = getelementptr i32, i32* [[PI]], i64 8
36; CHECK-NEXT:    store i32 6, i32* [[PIA1C]], align 4
37; CHECK-NEXT:    ret void
38;
39  %fmt = getelementptr [3 x i8], [3 x i8]* @pcnt_s, i32 0, i32 0
40; Fold snprintf(0, 0, "%s", a[0].a) to 1.
41  %pa0a = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 0, i32 0, i64 0
42  %ia0a = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa0a)
43  %pia0a = getelementptr i32, i32* %pi, i32 0
44  store i32 %ia0a, i32* %pia0a
45
46; Fold snprintf(0, 0, "%s", a[0].a) to 0.
47  %pa0ap1 = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 0, i32 0, i64 1
48  %ia0ap1 = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa0ap1)
49  %pia0ap1 = getelementptr i32, i32* %pi, i32 1
50  store i32 %ia0ap1, i32* %pia0ap1
51
52; Fold snprintf(0, 0, "%s", a[0].b) to 2.
53  %pa0b = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 0, i32 1, i64 0
54  %ia0b = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa0b)
55  %pia0b = getelementptr i32, i32* %pi, i32 2
56  store i32 %ia0b, i32* %pia0b
57
58; Fold snprintf(0, 0, "%s", a[0].b + 1) to 1.
59  %pa0bp1 = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 0, i32 1, i64 1
60  %ia0bp1 = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa0bp1)
61  %pia0bp1 = getelementptr i32, i32* %pi, i32 3
62  store i32 %ia0bp1, i32* %pia0bp1
63
64; Fold snprintf(0, 0, "%s", a[0].b + 2) to 0.
65  %pa0bp2 = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 0, i32 1, i64 2
66  %ia0bp2 = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa0bp2)
67  %pia0bp2 = getelementptr i32, i32* %pi, i32 4
68  store i32 %ia0bp2, i32* %pia0bp2
69
70; Fold snprintf(0, 0, "%s", a[0].c) to 3.
71  %pa0c = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 0, i32 2, i64 0
72  %ia0c = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa0c)
73  %pia0c = getelementptr i32, i32* %pi, i32 5
74  store i32 %ia0c, i32* %pia0c
75
76; Fold snprintf(0, 0, "%s", a[1].a) to 4.
77  %pa1a = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 1, i32 0, i64 0
78  %ia1a = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa1a)
79  %pia1a = getelementptr i32, i32* %pi, i32 6
80  store i32 %ia1a, i32* %pia1a
81
82; Fold snprintf(0, 0, "%s", a[1].b) to 5.
83  %pa1b = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 1, i32 1, i64 0
84  %ia1b = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa1b)
85  %pia1b = getelementptr i32, i32* %pi, i32 7
86  store i32 %ia1b, i32* %pia1b
87
88; Fold snprintf(0, 0, "%s", a[1].c) to 6.
89  %pa1c = getelementptr [2 x %struct.A], [2 x %struct.A]* @a, i64 0, i64 1, i32 2, i64 0
90  %ia1c = call i32 (i8*, i64, i8*, ...) @snprintf(i8* null, i64 0, i8* %fmt, i8* %pa1c)
91  %pia1c = getelementptr i32, i32* %pi, i32 8
92  store i32 %ia1c, i32* %pia1c
93
94  ret void
95}
96