| // RUN: %clang_cc1 -Wno-int-to-pointer-cast -analyze -analyzer-store=region -analyzer-constraints=range -fblocks -analyzer-opt-analyze-nested-blocks -analyzer-checker=alpha.deadcode.IdempotentOperations -verify %s |
| |
| // Basic tests |
| |
| extern void test(int i); |
| extern void test_f(float f); |
| |
| unsigned basic() { |
| int x = 10, zero = 0, one = 1; |
| |
| // x op x |
| x = x; // expected-warning {{Assigned value is always the same as the existing value}} |
| test(x - x); // expected-warning {{Both operands to '-' always have the same value}} |
| x -= x; // expected-warning {{Both operands to '-=' always have the same value}} |
| x = 10; // no-warning |
| test(x / x); // expected-warning {{Both operands to '/' always have the same value}} |
| x /= x; // expected-warning {{Both operands to '/=' always have the same value}} |
| x = 10; // no-warning |
| test(x & x); // expected-warning {{Both operands to '&' always have the same value}} |
| x &= x; // expected-warning {{Both operands to '&=' always have the same value}} |
| test(x | x); // expected-warning {{Both operands to '|' always have the same value}} |
| x |= x; // expected-warning {{Both operands to '|=' always have the same value}} |
| |
| // x op 1 |
| test(x * one); // expected-warning {{The right operand to '*' is always 1}} |
| x *= one; // expected-warning {{The right operand to '*=' is always 1}} |
| test(x / one); // expected-warning {{The right operand to '/' is always 1}} |
| x /= one; // expected-warning {{The right operand to '/=' is always 1}} |
| |
| // 1 op x |
| test(one * x); // expected-warning {{The left operand to '*' is always 1}} |
| |
| // x op 0 |
| test(x + zero); // expected-warning {{The right operand to '+' is always 0}} |
| test(x - zero); // expected-warning {{The right operand to '-' is always 0}} |
| test(x * zero); // expected-warning {{The right operand to '*' is always 0}} |
| test(x & zero); // expected-warning {{The right operand to '&' is always 0}} |
| test(x | zero); // expected-warning {{The right operand to '|' is always 0}} |
| test(x ^ zero); // expected-warning {{The right operand to '^' is always 0}} |
| test(x << zero); // expected-warning {{The right operand to '<<' is always 0}} |
| test(x >> zero); // expected-warning {{The right operand to '>>' is always 0}} |
| |
| // 0 op x |
| test(zero + x); // expected-warning {{The left operand to '+' is always 0}} |
| test(zero - x); // expected-warning {{The left operand to '-' is always 0}} |
| test(zero / x); // expected-warning {{The left operand to '/' is always 0}} |
| test(zero * x); // expected-warning {{The left operand to '*' is always 0}} |
| test(zero & x); // expected-warning {{The left operand to '&' is always 0}} |
| test(zero | x); // expected-warning {{The left operand to '|' is always 0}} |
| test(zero ^ x); // expected-warning {{The left operand to '^' is always 0}} |
| test(zero << x); // expected-warning {{The left operand to '<<' is always 0}} |
| test(zero >> x); // expected-warning {{The left operand to '>>' is always 0}} |
| |
| // Overwrite the values so these aren't marked as Pseudoconstants |
| x = 1; |
| zero = 2; |
| one = 3; |
| |
| return x + zero + one; |
| } |
| |
| void floats(float x) { |
| test_f(x * 1.0); // no-warning |
| test_f(x * 1.0F); // no-warning |
| } |
| |
| // Ensure that we don't report false poitives in complex loops |
| void bailout() { |
| int unused = 0, result = 4; |
| result = result; // expected-warning {{Assigned value is always the same as the existing value}} |
| |
| for (unsigned bg = 0; bg < 1024; bg ++) { |
| result = bg * result; // no-warning |
| |
| for (int i = 0; i < 256; i++) { |
| unused *= i; // no-warning |
| } |
| } |
| } |
| |
| // Relaxed liveness - check that we don't kill liveness at assignments |
| typedef unsigned uintptr_t; |
| void kill_at_assign() { |
| short array[2]; |
| uintptr_t x = (uintptr_t) array; |
| short *p = (short *) x; |
| |
| // The following branch should be infeasible. |
| if (!(p = &array[0])) { // expected-warning{{Assigned value is always the same as the existing value}} |
| p = 0; |
| *p = 1; // no-warning |
| } |
| } |
| |
| // False positive tests |
| |
| unsigned false1() { |
| int a = 10; |
| return a * (5 - 2 - 3); // no-warning |
| } |
| |
| enum testenum { enum1 = 0, enum2 }; |
| unsigned false2() { |
| int a = 1234; |
| return enum1 + a; // no-warning |
| } |
| |
| // Self assignments of unused variables are common false positives |
| unsigned false3(int param, int param2) { |
| param = param; // no-warning |
| |
| // if a self assigned variable is used later, then it should be reported still |
| param2 = param2; // expected-warning{{Assigned value is always the same as the existing value}} |
| |
| unsigned nonparam = 5; |
| |
| nonparam = nonparam; // expected-warning{{Assigned value is always the same as the existing value}} |
| |
| return param2 + nonparam; |
| } |
| |
| // Pseudo-constants (vars only read) and constants should not be reported |
| unsigned false4() { |
| // Trivial constant |
| const int height = 1; |
| int c = 42; |
| test(height * c); // no-warning |
| |
| // Pseudo-constant (never changes after decl) |
| int width = height; |
| |
| return width * 10; // no-warning |
| } |
| |
| // Block pseudoconstants |
| void false4a() { |
| // Pseudo-constant |
| __block int a = 1; |
| int b = 10; |
| __block int c = 0; |
| b *= a; // no-warning |
| |
| ^{ |
| // Psuedoconstant block var |
| test(b * c); // no-warning |
| |
| // Non-pseudoconstant block var |
| int d = 0; |
| test(b * d); // expected-warning{{The right operand to '*' is always 0}} |
| d = 5; |
| test(d); |
| }(); |
| |
| test(a + b); |
| } |
| |
| // Static vars are common false positives |
| int false5() { |
| static int test = 0; |
| int a = 56; |
| a *= test; // no-warning |
| test++; |
| return a; |
| } |
| |
| // Non-local storage vars are considered false positives |
| int globalInt = 1; |
| int false6() { |
| int localInt = 23; |
| |
| localInt /= globalInt; |
| |
| return localInt; |
| } |
| |
| // Check that assignments filter out false positives correctly |
| int false7() { |
| int zero = 0; // pseudo-constant |
| int one = 1; |
| |
| int a = 55; |
| a = a; // expected-warning{{Assigned value is always the same as the existing value}} |
| a = enum1 * a; // no-warning |
| |
| int b = 123; |
| b = b; // no-warning |
| |
| return a; |
| } |
| |
| // Check truncations do not flag as self-assignments |
| void false8() { |
| int a = 10000000; |
| a = (short)a; // no-warning |
| test(a); |
| } |
| |
| // This test case previously flagged a warning at 'b == c' because the |
| // analyzer previously allowed 'UnknownVal' as the index for ElementRegions. |
| typedef struct RDar8431728_F { |
| int RDar8431728_A; |
| unsigned char *RDar8431728_B; |
| int RDar8431728_E[6]; |
| } RDar8431728_D; |
| static inline int RDar8431728_C(RDar8431728_D * s, int n, |
| unsigned char **RDar8431728_B_ptr) { |
| int xy, wrap, pred, a, b, c; |
| |
| xy = s->RDar8431728_E[n]; |
| wrap = s->RDar8431728_A; |
| |
| a = s->RDar8431728_B[xy - 1]; |
| b = s->RDar8431728_B[xy - 1 - wrap]; |
| c = s->RDar8431728_B[xy - wrap]; |
| |
| if (b == c) { // no-warning |
| pred = a; |
| } else { |
| pred = c; |
| } |
| |
| *RDar8431728_B_ptr = &s->RDar8431728_B[xy]; |
| |
| return pred; |
| } |
| |
| // <rdar://problem/8601243> - Don't warn on pointer arithmetic. This |
| // is often idiomatic. |
| unsigned rdar8601243_aux(unsigned n); |
| void rdar8601243() { |
| char arr[100]; |
| char *start = arr; |
| start = start + rdar8601243_aux(sizeof(arr) - (arr - start)); // no-warning |
| (void) start; |
| } |
| |
| |
| float testFloatCast(int i) { |
| float f = i; |
| |
| // Don't crash when trying to create a "zero" float. |
| return f - f; |
| } |
| |