blob: 15c06eb421ee409d1ea3f1603ff8ae0f80ae9d5c [file] [log] [blame]
Ted Kremenekc00d5b92010-10-30 00:43:15 +00001// RUN: %clang_cc1 -fsyntax-only -Wall -Wuninitialized -verify %s
2
Chandler Carruthedcc04e2011-03-27 20:35:59 +00003int foo(int x);
4int bar(int* x);
5int boo(int& x);
6int far(const int& x);
7
8// Test self-references within initializers which are guaranteed to be
9// uninitialized.
Chandler Carruthb5d48312011-04-05 17:41:31 +000010int a = a; // no-warning: used to signal intended lack of initialization.
Chandler Carruthedcc04e2011-03-27 20:35:59 +000011int b = b + 1; // expected-warning {{variable 'b' is uninitialized when used within its own initialization}}
12int c = (c + c); // expected-warning 2 {{variable 'c' is uninitialized when used within its own initialization}}
13void test() {
Ted Kremenek37881932011-04-04 23:29:12 +000014 int d = ({ d + d ;}); // expected-warning {{variable 'd' is uninitialized when used within its own initialization}}
Chandler Carruthedcc04e2011-03-27 20:35:59 +000015}
16int e = static_cast<long>(e) + 1; // expected-warning {{variable 'e' is uninitialized when used within its own initialization}}
17int f = foo(f); // expected-warning {{variable 'f' is uninitialized when used within its own initialization}}
18
19// Thes don't warn as they don't require the value.
20int g = sizeof(g);
21void* ptr = &ptr;
22int h = bar(&h);
23int i = boo(i);
24int j = far(j);
25int k = __alignof__(k);
26
Richard Trieua04ad1a2011-09-01 21:44:13 +000027
28// Test self-references with record types.
29class A {
30 // Non-POD class.
31 public:
32 enum count { ONE, TWO, THREE };
33 int num;
34 static int count;
35 int get() const { return num; }
Richard Trieu978dfc02012-03-08 01:15:31 +000036 int get2() { return num; }
Richard Trieua04ad1a2011-09-01 21:44:13 +000037 void set(int x) { num = x; }
38 static int zero() { return 0; }
39
40 A() {}
41 A(A const &a) {}
42 A(int x) {}
43 A(int *x) {}
44 A(A *a) {}
Rafael Espindola7c23b082012-01-06 04:54:01 +000045 ~A();
Richard Trieua04ad1a2011-09-01 21:44:13 +000046};
47
48A getA() { return A(); }
49A getA(int x) { return A(); }
50A getA(A* a) { return A(); }
51
52void setupA() {
53 A a1;
54 a1.set(a1.get());
55 A a2(a1.get());
56 A a3(a1);
57 A a4(&a4);
58 A a5(a5.zero());
59 A a6(a6.ONE);
60 A a7 = getA();
61 A a8 = getA(a8.TWO);
62 A a9 = getA(&a9);
63 A a10(a10.count);
64
65 A a11(a11); // expected-warning {{variable 'a11' is uninitialized when used within its own initialization}}
66 A a12(a12.get()); // expected-warning {{variable 'a12' is uninitialized when used within its own initialization}}
67 A a13(a13.num); // expected-warning {{variable 'a13' is uninitialized when used within its own initialization}}
68 A a14 = A(a14); // expected-warning {{variable 'a14' is uninitialized when used within its own initialization}}
69 A a15 = getA(a15.num); // expected-warning {{variable 'a15' is uninitialized when used within its own initialization}}
70 A a16(&a16.num); // expected-warning {{variable 'a16' is uninitialized when used within its own initialization}}
Richard Trieu978dfc02012-03-08 01:15:31 +000071 A a17(a17.get2()); // expected-warning {{variable 'a17' is uninitialized when used within its own initialization}}
Richard Trieua04ad1a2011-09-01 21:44:13 +000072}
73
74struct B {
75 // POD struct.
76 int x;
77 int *y;
78};
79
80B getB() { return B(); };
81B getB(int x) { return B(); };
82B getB(int *x) { return B(); };
83B getB(B *b) { return B(); };
84
85void setupB() {
86 B b1;
87 B b2(b1);
88 B b3 = { 5, &b3.x };
89 B b4 = getB();
90 B b5 = getB(&b5);
91 B b6 = getB(&b6.x);
92
93 // Silence unused warning
94 (void) b2;
95 (void) b4;
96
97 B b7(b7); // expected-warning {{variable 'b7' is uninitialized when used within its own initialization}}
98 B b8 = getB(b8.x); // expected-warning {{variable 'b8' is uninitialized when used within its own initialization}}
99 B b9 = getB(b9.y); // expected-warning {{variable 'b9' is uninitialized when used within its own initialization}}
100}
101
Chandler Carruthedcc04e2011-03-27 20:35:59 +0000102// Also test similar constructs in a field's initializer.
103struct S {
104 int x;
105 void *ptr;
106
107 S(bool (*)[1]) : x(x) {} // expected-warning {{field is uninitialized when used here}}
108 S(bool (*)[2]) : x(x + 1) {} // expected-warning {{field is uninitialized when used here}}
109 S(bool (*)[3]) : x(x + x) {} // expected-warning {{field is uninitialized when used here}}
110 S(bool (*)[4]) : x(static_cast<long>(x) + 1) {} // expected-warning {{field is uninitialized when used here}}
111 S(bool (*)[5]) : x(foo(x)) {} // FIXME: This should warn!
112
113 // These don't actually require the value of x and so shouldn't warn.
114 S(char (*)[1]) : x(sizeof(x)) {} // rdar://8610363
115 S(char (*)[2]) : ptr(&ptr) {}
116 S(char (*)[3]) : x(__alignof__(x)) {}
117 S(char (*)[4]) : x(bar(&x)) {}
118 S(char (*)[5]) : x(boo(x)) {}
119 S(char (*)[6]) : x(far(x)) {}
Ted Kremenekc00d5b92010-10-30 00:43:15 +0000120};
Richard Trieuaa5e2562011-09-07 00:58:53 +0000121
122struct C { char a[100], *e; } car = { .e = car.a };
Douglas Gregor6c8f07f2011-11-15 15:29:30 +0000123
124// <rdar://problem/10398199>
125namespace rdar10398199 {
126 class FooBase { protected: ~FooBase() {} };
127 class Foo : public FooBase {
128 public:
129 operator int&() const;
130 };
131 void stuff();
132 template <typename T> class FooImpl : public Foo {
133 T val;
134 public:
135 FooImpl(const T &x) : val(x) {}
136 ~FooImpl() { stuff(); }
137 };
138
139 template <typename T> FooImpl<T> makeFoo(const T& x) {
140 return FooImpl<T>(x);
141 }
142
143 void test() {
144 const Foo &x = makeFoo(42);
145 const int&y = makeFoo(42u);
146 (void)x;
147 (void)y;
148 };
149}
Ted Kremenek213d0532012-03-22 05:57:43 +0000150
151// PR 12325 - this was a false uninitialized value warning due to
152// a broken CFG.
153int pr12325(int params) {
154 int x = ({
155 while (false)
156 ;
157 int _v = params;
158 if (false)
159 ;
160 _v; // no-warning
161 });
162 return x;
163}
164