2019-03-09 00:00:42 +08:00
|
|
|
// RUN: %clang_analyze_cc1 -verify %s \
|
|
|
|
// RUN: -analyzer-checker=alpha.clone.CloneChecker \
|
|
|
|
// RUN: -analyzer-config alpha.clone.CloneChecker:ReportNormalClones=false \
|
|
|
|
// RUN: -analyzer-config alpha.clone.CloneChecker:MinimumCloneComplexity=10
|
2016-08-18 20:29:41 +08:00
|
|
|
|
|
|
|
// Tests finding a suspicious clone that references local variables.
|
|
|
|
|
|
|
|
void log();
|
|
|
|
|
|
|
|
int max(int a, int b) {
|
|
|
|
log();
|
|
|
|
if (a > b)
|
|
|
|
return a;
|
2016-10-08 18:54:30 +08:00
|
|
|
return b; // expected-note{{Similar code using 'b' here}}
|
2016-08-18 20:29:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int maxClone(int x, int y, int z) {
|
|
|
|
log();
|
|
|
|
if (x > y)
|
|
|
|
return x;
|
2016-10-08 18:54:30 +08:00
|
|
|
return z; // expected-warning{{Potential copy-paste error; did you really mean to use 'z' here?}}
|
2016-08-18 20:29:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Tests finding a suspicious clone that references global variables.
|
|
|
|
|
|
|
|
struct mutex {
|
|
|
|
bool try_lock();
|
|
|
|
void unlock();
|
|
|
|
};
|
|
|
|
|
|
|
|
mutex m1;
|
|
|
|
mutex m2;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
void busyIncrement() {
|
|
|
|
while (true) {
|
|
|
|
if (m1.try_lock()) {
|
|
|
|
++i;
|
2016-10-08 18:54:30 +08:00
|
|
|
m1.unlock(); // expected-note{{Similar code using 'm1' here}}
|
2016-08-18 20:29:41 +08:00
|
|
|
if (i > 1000) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void faultyBusyIncrement() {
|
|
|
|
while (true) {
|
|
|
|
if (m1.try_lock()) {
|
|
|
|
++i;
|
2016-10-08 18:54:30 +08:00
|
|
|
m2.unlock(); // expected-warning{{Potential copy-paste error; did you really mean to use 'm2' here?}}
|
2016-08-18 20:29:41 +08:00
|
|
|
if (i > 1000) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tests that we provide two suggestions in cases where two fixes are possible.
|
|
|
|
|
|
|
|
int foo(int a, int b, int c) {
|
|
|
|
a += b + c;
|
|
|
|
b /= a + b;
|
2016-10-08 18:54:30 +08:00
|
|
|
c -= b * a; // expected-warning{{Potential copy-paste error; did you really mean to use 'b' here?}}
|
2016-08-18 20:29:41 +08:00
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
int fooClone(int a, int b, int c) {
|
|
|
|
a += b + c;
|
|
|
|
b /= a + b;
|
2016-10-08 18:54:30 +08:00
|
|
|
c -= a * a; // expected-note{{Similar code using 'a' here}}
|
2016-08-18 20:29:41 +08:00
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Tests that for clone groups with a many possible suspicious clone pairs, at
|
|
|
|
// most one warning per clone group is generated and every relevant clone is
|
|
|
|
// reported through either a warning or a note.
|
|
|
|
|
|
|
|
long bar1(long a, long b, long c, long d) {
|
|
|
|
c = a - b;
|
|
|
|
c = c / d * a;
|
2016-10-08 18:54:30 +08:00
|
|
|
d = b * b - c; // expected-warning{{Potential copy-paste error; did you really mean to use 'b' here?}}
|
2016-08-18 20:29:41 +08:00
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
long bar2(long a, long b, long c, long d) {
|
|
|
|
c = a - b;
|
|
|
|
c = c / d * a;
|
2016-10-08 18:54:30 +08:00
|
|
|
d = c * b - c; // expected-note{{Similar code using 'c' here}} \
|
|
|
|
// expected-warning{{Potential copy-paste error; did you really mean to use 'c' here?}}
|
2016-08-18 20:29:41 +08:00
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
long bar3(long a, long b, long c, long d) {
|
|
|
|
c = a - b;
|
|
|
|
c = c / d * a;
|
2016-10-08 18:54:30 +08:00
|
|
|
d = a * b - c; // expected-note{{Similar code using 'a' here}}
|
2016-08-18 20:29:41 +08:00
|
|
|
return d;
|
|
|
|
}
|