manas added inline comments.
================ Comment at: clang/test/Analysis/constant-folding.c:280-281 + if (c < 0 && c != INT_MIN && d < 0) { + clang_analyzer_eval((c + d) == -1); // expected-warning{{FALSE}} + clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} + clang_analyzer_eval((c + d) <= -2); // expected-warning{{UNKNOWN}} ---------------- manas wrote: > vsavchenko wrote: > > manas wrote: > > > vsavchenko wrote: > > > > manas wrote: > > > > > When I pull these cases out to a separate function (say > > > > > testAdditionRules2) in constant-folding.c then algorithm is able to > > > > > reason about the range correctly, but inside testAdditionRules, it is > > > > > unable to figure stuff out. Does it have something to do with > > > > > constraints being propagated which we discussed below? > > > > > > > > > > @vsavchenko wrote: > > > > > > I have one note here. The fact that we don't have a testing > > > > > > framework and use this approach of inspecting the actual analysis > > > > > > has an interesting implication. > > > > > > > > > > > > ``` > > > > > > if (a == 10) { // here we split path into 2 paths: one where a == > > > > > > 10, and one where a != 10; > > > > > > // one goes inside of if, one doesn't > > > > > > . . . > > > > > > } > > > > > > if (a >= 5) { // here we split into 3 paths: a == 10, a < 5, and a > > > > > > in [5, 9] and [11, INT_MAX] (aka a >= 5 and a != 10). > > > > > > // 1 path was infeasible: a == 10 and a < 5 > > > > > > // Two of these paths go inside of the if, one doesn't > > > > > > . . . > > > > > > clang_analyzer_eval(a == 10); // it will produce two results: > > > > > > TRUE and FALSE > > > > > > } > > > > > > clang_analyzer_eval(a == 10); // it will produce three results: > > > > > > TRUE, FALSE, and FALSE > > > > > > ``` > > > > > > > > > > > > We don't want to test how path splitting works in these particular > > > > > > tests (they are about solver and constant folding after all), > > > > > > that's why we try to have only one path going through each > > > > > > `clang_analyzer_eval(...)` > > > > > > > > > > > > > > > > > > > > It might be that or it might be something different. Just by looking > > > > at this example, the previous `if` statement shouldn't add more paths > > > > that go inside of this `if`, so it shouldn't be the root cause. > > > > Whenever you encounter problems and you want to tell other people, > > > > **please, provide more detail**. Did you notice it when running the > > > > test case? What was the output? What did you try? How did you > > > > extract it into a separate function? > > > I put a new test function in `constant-folding.c` as: > > > > > > ``` > > > void testAdditionRulesNew(int c, int d) { > > > if (c < 0 && c != INT_MIN && d < 0) { > > > clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} > > > } > > > } > > > ``` > > > I tested this specific function as: > > > > > > ./build/bin/clang -cc1 -analyze > > > -analyzer-checker=core,debug.ExprInspection > > > -analyze-function=testAdditionRulesNew constant-folding.c > > > > > > And I got the following output: > > > > > > ../clang/test/Analysis/constant-folding.c:338:5: warning: FALSE > > > [debug.ExprInspection] > > > clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} > > > > > > which is correct. > > > > > > But when I ran the same test inside `testAdditionRules`, using: > > > ./build/bin/clang -cc1 -analyze > > > -analyzer-checker=core,debug.ExprInspection > > > -analyze-function=testAdditionRules constant-folding.c > > > then I got: > > > > > > ../clang/test/Analysis/constant-folding.c:281:5: warning: FALSE > > > [debug.ExprInspection] > > > clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} > > > > > > > > > ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ > > > > > > ../clang/test/Analysis/constant-folding.c:281:5: warning: TRUE > > > [debug.ExprInspection] > > > clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} > > > > > > > > > ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ > > > > > > Here, `c = [INT_MIN + 1, -1]` and `d = [INT_MIN, 0]`, so `c + d = > > > [INT_MIN, -2] U [1, INT_MAX]`. So `c + d == 0` should be false. But in > > > latter case, it is reasoning `c + d == 0` to be `UNKNOWN`. > > > > > > Also, the error arises in `c + d == 0` case only and not in `c + d == -1` > > > case. I mistakenly highlighted that case while commenting. > > Hmm, I don't know what can be the reason. > > > > There are three reasonable approaches: > > 1. Straightforward: use debugger. Find a place when we ask the solver > > about this assumption and dig in. > > 2. Print-based: use `clang_analyzer_printState` inside of this if. Among > > other things it will print you all constraints that we know about, you can > > check if they match your expectations. Also, if it is indeed because of > > residual paths, you will get another print from residual path. > > 3. Reduction: add the whole `testAdditionRules` into > > `testAdditionRulesNew` and start removing parts of it. At some point > > problem should disappear (based on your previous observations). That > > "minimal" reproducer can give you insight into what's going on. > > > > These are not mutually exclusive, so you can try them in combination. > I tried print the ProgramState as in both the cases like: > > if (c < 0 && c != INT_MIN && d < 0) { > clang_analyzer_printState(); > clang_analyzer_eval(...); > ... > } > > 1. During `testAdditionRulesNew`, when I added `clang_analyzer_printState` > just before doing any `clang_analyzer_eval` calls, I got the following > constraints (these are as expected): > > ``` > "constraints": [ > > > { "symbol": "reg_$0<int c>", "range": "{ [-2147483647, -1] }" }, > > > { "symbol": "reg_$1<int d>", "range": "{ [-2147483648, -1] }" } > > > ], > ``` > > 2. In `testAdditionRules`, adding `printState` before any evaluation led to: > > ``` > "constraints": [ > > > { "symbol": "reg_$2<int c>", "range": "{ [-2147483647, -1] }" }, > > > { "symbol": "reg_$3<int d>", "range": "{ [-2147483648, -1] }" }, > > > { "symbol": "(reg_$2<int c>) + (reg_$3<int d>)", "range": "{ [0, > 2147483647] }" } > > ], > > "constraints": [ > { "symbol": "reg_$2<int c>", "range": "{ [-2147483647, -1] }" }, > { "symbol": "reg_$3<int d>", "range": "{ [-2147483648, -1] }" }, > { "symbol": "(reg_$2<int c>) + (reg_$3<int d>)", "range": "{ > [-2147483648, -2] }" } > ], > ``` > A point to note is that this ProgramState is **before** any > `clang_analyzer_eval` calls have been made. So, the third constraint, in both > sets of constraints (for `c + d`) should not exist there, that is, > > a. { "symbol": "(reg_$2<int c>) + (reg_$3<int d>)", "range": "{ [0, > 2147483647] }" } > b. { "symbol": "(reg_$2<int c>) + (reg_$3<int d>)", "range": "{ > [-2147483648, -2] }" } > > should not be in the constraints set. > > And as obvious, the `UNKNOWN` reasoning for `c + d == 0` in > `testAdditionRules` is arising from constraint (a) above, which is a wrong > reasoning (it ideally should be `[1, INT_MAX]` and not `[0, INT_MAX]`). @vsavchenko wrote: > 3. Reduction: add the whole `testAdditionRules` into `testAdditionRulesNew` > and start removing parts of it. At some point problem should disappear > (based on your previous observations). That "minimal" reproducer can give > you insight into what's going on. > Trying this, I was able to deduce that the following snippet is causing `UNKNOWN` triggers: ``` void testAdditionRulesNew(unsigned int a, unsigned int b, int c, int d) { if (c < 0 && d < 0) { clang_analyzer_printState(); clang_analyzer_eval((c + d) < 0); // expected-warning{{UNKNOWN}} clang_analyzer_printState(); } if (c < 0 && c != INT_MIN && d < 0) { clang_analyzer_printState(); clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} clang_analyzer_printState(); } } ``` The output warnings are as follows: ``` <snipped-ProgramStates> ../clang/test/Analysis/constant-folding.c:341:5: warning: FALSE [debug.ExprInspection] clang_analyzer_eval((c + d) < 0); // expected-warning{{UNKNOWN}} ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ../clang/test/Analysis/constant-folding.c:341:5: warning: TRUE [debug.ExprInspection] clang_analyzer_eval((c + d) < 0); // expected-warning{{UNKNOWN}} ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ../clang/test/Analysis/constant-folding.c:347:5: warning: FALSE [debug.ExprInspection] clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ../clang/test/Analysis/constant-folding.c:347:5: warning: TRUE [debug.ExprInspection] clang_analyzer_eval((c + d) == 0); // expected-warning{{FALSE}} ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 4 warnings generated. ``` The second evaluation is leading to wrong reasoning. I think the constraints are being propagated somehow from first conditional (`c < 0 && d < 0`) to the second conditional (`c < 0 && c != INT_MIN && d < 0`). I have added a pastebin using `clang_analyzer_printState` here: https://pastebin.com/0E8hmTWh Repository: rG LLVM Github Monorepo CHANGES SINCE LAST ACTION https://reviews.llvm.org/D103440/new/ https://reviews.llvm.org/D103440 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits