diff options
author | Richard Smith <richard-llvm@metafoo.co.uk> | 2012-06-16 23:34:14 +0000 |
---|---|---|
committer | Richard Smith <richard-llvm@metafoo.co.uk> | 2012-06-16 23:34:14 +0000 |
commit | a8d4f229a645a232f7226a843c977f0772bbf9d8 (patch) | |
tree | b5b7bba8e0d0127fe8b8fbef4fc97f92288902d4 | |
parent | d0b767f849ded7197bb9fa7b9a4e23f51e43b005 (diff) | |
download | bcm5719-llvm-a8d4f229a645a232f7226a843c977f0772bbf9d8.tar.gz bcm5719-llvm-a8d4f229a645a232f7226a843c977f0772bbf9d8.zip |
-Wuninitialized bugfix: when entering the scope of a variable with no
initializer, it is uninitialized, even if we may be coming from somewhere where
it was initialized.
llvm-svn: 158611
-rw-r--r-- | clang/lib/Analysis/UninitializedValues.cpp | 12 | ||||
-rw-r--r-- | clang/test/Sema/uninit-variables.c | 25 |
2 files changed, 37 insertions, 0 deletions
diff --git a/clang/lib/Analysis/UninitializedValues.cpp b/clang/lib/Analysis/UninitializedValues.cpp index 57dfab36fdd..fbfa46c443c 100644 --- a/clang/lib/Analysis/UninitializedValues.cpp +++ b/clang/lib/Analysis/UninitializedValues.cpp @@ -625,6 +625,18 @@ void TransferFunctions::VisitDeclStmt(DeclStmt *ds) { // the use of the uninitialized value (which visiting the // initializer). vals[vd] = Initialized; + } else { + // No initializer: the variable is now uninitialized. This matters + // for cases like: + // while (...) { + // int n; + // use(n); + // n = 0; + // } + // FIXME: Mark the variable as uninitialized whenever its scope is + // left, since its scope could be re-entered by a jump over the + // declaration. + vals[vd] = Uninitialized; } } } diff --git a/clang/test/Sema/uninit-variables.c b/clang/test/Sema/uninit-variables.c index ef03d44d97b..180d60cbce7 100644 --- a/clang/test/Sema/uninit-variables.c +++ b/clang/test/Sema/uninit-variables.c @@ -437,3 +437,28 @@ void test54() { int c; // expected-note {{initialize the variable 'c' to silence this warning}} ASSIGN(int, c, d); // expected-warning {{variable 'c' is uninitialized when used here}} } + +void uninit_in_loop() { + int produce(void); + void consume(int); + for (int n = 0; n < 100; ++n) { + int k; // expected-note {{initialize}} + consume(k); // expected-warning {{variable 'k' is uninitialized}} + k = produce(); + } +} + +void uninit_in_loop_goto() { + int produce(void); + void consume(int); + for (int n = 0; n < 100; ++n) { + goto skip_decl; + int k; // expected-note {{initialize}} +skip_decl: + // FIXME: This should produce the 'is uninitialized' diagnostic, but we + // don't have enough information in the CFG to easily tell that the + // variable's scope has been left and re-entered. + consume(k); // expected-warning {{variable 'k' may be uninitialized}} + k = produce(); + } +} |