This is an archive of the discontinued LLVM Phabricator instance.

[analyzer] Fix false negative on post-increment of uninitialized variable.
ClosedPublic

Authored by lebedev.ri on Nov 26 2017, 1:18 AM.

Diff Detail

Repository
rL LLVM

Event Timeline

lebedev.ri created this revision.Nov 26 2017, 1:18 AM

Fix typo in release notes.

JonasToth added inline comments.
docs/ReleaseNotes.rst
261

The end of the sentence should be either 'on uninitialized values' or 'on an uninitialized values'.

lebedev.ri added inline comments.Nov 26 2017, 3:08 AM
docs/ReleaseNotes.rst
261

The end of the sentence should be either 'on uninitialized values' or 'on an uninitialized values'.

But it is already on an uninitialized values. ?

JonasToth added inline comments.Nov 26 2017, 3:56 AM
docs/ReleaseNotes.rst
261

Sorry, my tablet autocompleted :(

With 'an' values needs to be singular 'an' -> 'value'. If you want 'values' drop the 'an'.

xazax.hun added a subscriber: NoQ.Nov 26 2017, 3:57 AM
xazax.hun added inline comments.
docs/ReleaseNotes.rst
261

Maybe Jonas was thinking if removing the plural form if you having an or of you want to keep it plural remove the an.

lib/StaticAnalyzer/Core/ExprEngineC.cpp
1048

Comments should be whole sentences starting with capital letters.

1051

This will trigger checkBind when unknown value is pre/post incremented. I wonder if this is the desired behavior. Couldn't you achieve the same on the checker side by having a checkPreStmt hook?

@NoQ, @dcoughlin what do you think about triggering checkBind here? In fact, there is a bind during pre/post increment/decrement. But do we want these events to be visible for checkers with undefined values?

test/Analysis/malloc-plist.c
2

In case you willing to replace the plist based test with -analyzer-output=text based, it would be awesome, though not required for this patch to be accepted.

lebedev.ri marked 5 inline comments as done.

Fix some more typos.

jordan_rose edited reviewers, added: NoQ; removed: dergachev.a.
dcoughlin edited edge metadata.Nov 28 2017, 12:02 PM

Thanks for tackling this! There is one major comment inline (you are generating an extra node that you shouldn't, which is causing the analyzer to explore code it shouldn't) and a suggestion for simpler diagnostic text.

lib/StaticAnalyzer/Checkers/UndefinedAssignmentChecker.cpp
64

"Unary operator" is probably not something we should expect users to know about. How about just "The expression is an uninitialized value. The computed value will also be garbage."

lib/StaticAnalyzer/Core/ExprEngineC.cpp
1046

Instead of generating a node node here, you'll want to generate a new state:

state = state->BindExpr(U, LCtx, V2_untested);

and use that state in the call to evalStore().

Otherwise, you'll end up exploring from both the new generated node and from *I.

Here is a test that demonstrates why this is a problem. You should add it to your tests.

void foo() {
  int x;

  x++; // expected-warning {{The expression of the unary operator is an uninitialized value. The computed value will also be garbage}}

  clang_analyzer_warnIfReached(); // no-warning

The undefined assignment checker generates what we call "fatal" errors. These errors are so bad that it decides not to explore further on that path to report additional errors. This means that the analyzer should treat any code that is dominated by such an error as not reachable. You'll need to add the debug.ExprInspection checker to your test RUN line and a prototype for clang_analyzer_warnIfReached() for this test to to work.

test/Analysis/malloc-plist.c
2

Let's hold off on replacing the plist test with text. We'll need to make sure we're still testing plist emission separately.

13

Once you change the core modeling to not generate an extra node, you'll want to initialize *p to 0 or something to preserve the intent of this test. For this test it is important that the increment not stop further exploration of the path.

111

Same point applies here.

test/Analysis/objc-for.m
131

You should just initialize j in order to preserve the test's intent. (And similarly for the other new warnings in this file.

test/Analysis/uninit-const.c
127

You'll need to split this test into multiple functions once you stop generating the extra node.

test/Analysis/uninit-const.cpp
11

I don't think this test is needed. The logic for C vs. C++ is shared for your new functionality, so I think it is sufficient to just test the C case.

lebedev.ri marked 5 inline comments as done.
lebedev.ri changed the repository for this revision from rL LLVM to rC Clang.

@dcoughlin thank you for the review! A few notes of my own below.

lib/StaticAnalyzer/Checkers/UndefinedAssignmentChecker.cpp
64

Yep, i did not like my wording either :)

lib/StaticAnalyzer/Core/ExprEngineC.cpp
1046

Instead of generating a node node here, you'll want to generate a new state:

state = state->BindExpr(U, LCtx, V2_untested);

and use that state in the call to evalStore().

Hm, so the only change needed here is

diff
- Bldr.generateNode(U, *I, state->BindExpr(U, LCtx, V2_untested));
+ state = state->BindExpr(U, LCtx, V2_untested);

?

1051

This will trigger checkBind when unknown value is pre/post incremented. I wonder if this is the desired behavior. Couldn't you achieve the same on the checker side by having a checkPreStmt hook?

The current approach is what was suggested in the https://bugs.llvm.org/show_bug.cgi?id=35419#c6, unless i have misread it of course.

test/Analysis/malloc-plist.c
13

Hmm, *this* test did not break after i changed ExprEngine::VisitIncrementDecrementOperator().
Did i change it wrong?

111

Same here.

xazax.hun added inline comments.Nov 29 2017, 7:14 AM
lib/StaticAnalyzer/Checkers/UndefinedAssignmentChecker.cpp
64

A value being undefined does nt mean uninitialized. E.g.: the result of a bad shift operation might be UndefVal as well.
Aren't these diagnostics misleading in those cases? Or we do not care about those?

dcoughlin added inline comments.Nov 29 2017, 10:05 AM
lib/StaticAnalyzer/Core/ExprEngineC.cpp
1046

Yes, that's right. What you have here looks good to me.

test/Analysis/malloc-plist.c
13

Aah, my mistake. I though this test was passing -verify (which verifies expected-warning{{}} contents). Instead, it is only checking the plists, so the missing expected leak would only be caught by the plist change.

I suggest changing the test to:

...
    if (in > 5) {
        int *p = malloc(12);
        *p = 0;
        (*p)++;
    }
...

So that the test doesn't generate a fatal error for the access to uninitialized memory and instead continues to check for the path of the leak.

112

Here you should stick the *p = 0; before the post increment so the rest of the code is exercised.

lebedev.ri marked 8 inline comments as done.

Address @dcoughlin's review notes.

lib/StaticAnalyzer/Checkers/UndefinedAssignmentChecker.cpp
64

I think this is for @dcoughlin to answer.
But shift operation is a binary operator, so this diff should not change that.

xazax.hun added inline comments.Nov 29 2017, 10:46 AM
lib/StaticAnalyzer/Checkers/UndefinedAssignmentChecker.cpp
64

I was wondering about examples like:

int x = 1 << -1;
++x;

In this particular case, it will not issue the misleading error message. The shift that results in an undefined value will generate an error node, so we will not warn for the pre increment.

But in general I am a bit uncomfortable about the assumption of this check: if a value is undefined the reason is that it is being uninitialized.

Note that, of course this problem is not specific to this patch but a general question about the checker.

dcoughlin accepted this revision.Nov 29 2017, 3:03 PM

Thanks, this looks good to me!

Do you have commit access or do you need someone to commit it for you?

lib/StaticAnalyzer/Checkers/UndefinedAssignmentChecker.cpp
64

It is a good point, but I think fixing it should wait until a later patch.

This revision is now accepted and ready to land.Nov 29 2017, 3:03 PM

Thanks, this looks good to me!

Awesome, thank you!

Do you have commit access or do you need someone to commit it for you?

I do have commit access.

This revision was automatically updated to reflect the committed changes.