2019-08-21 05:41:14 +08:00
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=core,optin.cplusplus.VirtualCall \
|
|
|
|
// RUN: -analyzer-checker=debug.ExprInspection \
|
|
|
|
// RUN: -std=c++11 -verify=impure %s
|
|
|
|
|
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=core,cplusplus.PureVirtualCall \
|
|
|
|
// RUN: -analyzer-checker=debug.ExprInspection \
|
|
|
|
// RUN: -std=c++11 -verify=pure -std=c++11 %s
|
|
|
|
|
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=core,optin.cplusplus.VirtualCall \
|
|
|
|
// RUN: -analyzer-config \
|
|
|
|
// RUN: optin.cplusplus.VirtualCall:PureOnly=true \
|
|
|
|
// RUN: -analyzer-checker=debug.ExprInspection \
|
|
|
|
// RUN: -std=c++11 -verify=none %s
|
|
|
|
|
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=core,cplusplus.PureVirtualCall \
|
|
|
|
// RUN: -analyzer-checker=optin.cplusplus.VirtualCall \
|
|
|
|
// RUN: -analyzer-checker=debug.ExprInspection \
|
|
|
|
// RUN: -std=c++11 -verify=pure,impure -std=c++11 %s
|
|
|
|
|
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=core,cplusplus.PureVirtualCall \
|
|
|
|
// RUN: -analyzer-checker=optin.cplusplus.VirtualCall \
|
|
|
|
// RUN: -analyzer-config \
|
|
|
|
// RUN: optin.cplusplus.VirtualCall:PureOnly=true \
|
|
|
|
// RUN: -analyzer-checker=debug.ExprInspection \
|
|
|
|
// RUN: -std=c++11 -verify=pure %s
|
|
|
|
|
|
|
|
|
|
|
|
// We expect no diagnostics when all checks are disabled.
|
|
|
|
// none-no-diagnostics
|
2016-12-10 09:16:09 +08:00
|
|
|
|
|
|
|
|
2017-08-28 16:44:43 +08:00
|
|
|
#include "virtualcall.h"
|
2012-01-04 07:18:57 +08:00
|
|
|
|
2019-08-21 05:41:14 +08:00
|
|
|
void clang_analyzer_warnIfReached();
|
|
|
|
|
2012-01-04 07:18:57 +08:00
|
|
|
class A {
|
|
|
|
public:
|
|
|
|
A();
|
2016-12-10 09:16:09 +08:00
|
|
|
|
2017-08-28 16:44:43 +08:00
|
|
|
~A(){};
|
|
|
|
|
|
|
|
virtual int foo() = 0;
|
2012-01-04 07:18:57 +08:00
|
|
|
virtual void bar() = 0;
|
|
|
|
void f() {
|
2019-08-21 05:41:14 +08:00
|
|
|
foo(); // pure-warning{{Call to pure virtual method 'A::foo' during construction has undefined behavior}}
|
|
|
|
clang_analyzer_warnIfReached(); // no-warning
|
2012-01-04 07:18:57 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-08-21 05:41:14 +08:00
|
|
|
A::A() {
|
|
|
|
f();
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
2012-01-04 07:18:57 +08:00
|
|
|
public:
|
2019-08-21 05:41:14 +08:00
|
|
|
B() {
|
|
|
|
foo(); // impure-warning {{Call to virtual method 'B::foo' during construction bypasses virtual dispatch}}
|
2012-01-04 07:18:57 +08:00
|
|
|
}
|
|
|
|
~B();
|
2017-08-28 16:44:43 +08:00
|
|
|
|
2012-01-04 07:18:57 +08:00
|
|
|
virtual int foo();
|
2017-08-28 16:44:43 +08:00
|
|
|
virtual void bar() {
|
2019-08-21 05:41:14 +08:00
|
|
|
foo(); // impure-warning {{Call to virtual method 'B::foo' during destruction bypasses virtual dispatch}}
|
|
|
|
}
|
2012-01-04 07:18:57 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
B::~B() {
|
|
|
|
this->B::foo(); // no-warning
|
|
|
|
this->B::bar();
|
2019-08-21 05:41:14 +08:00
|
|
|
this->foo(); // impure-warning {{Call to virtual method 'B::foo' during destruction bypasses virtual dispatch}}
|
2012-01-04 07:18:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
class C : public B {
|
|
|
|
public:
|
|
|
|
C();
|
|
|
|
~C();
|
2017-08-28 16:44:43 +08:00
|
|
|
|
2012-01-04 07:18:57 +08:00
|
|
|
virtual int foo();
|
|
|
|
void f(int i);
|
|
|
|
};
|
|
|
|
|
|
|
|
C::C() {
|
2019-08-21 05:41:14 +08:00
|
|
|
f(foo()); // impure-warning {{Call to virtual method 'C::foo' during construction bypasses virtual dispatch}}
|
2012-01-04 07:18:57 +08:00
|
|
|
}
|
|
|
|
|
2014-08-21 18:25:03 +08:00
|
|
|
class D : public B {
|
|
|
|
public:
|
|
|
|
D() {
|
|
|
|
foo(); // no-warning
|
|
|
|
}
|
|
|
|
~D() { bar(); }
|
|
|
|
int foo() final;
|
|
|
|
void bar() final { foo(); } // no-warning
|
|
|
|
};
|
|
|
|
|
|
|
|
class E final : public B {
|
|
|
|
public:
|
|
|
|
E() {
|
|
|
|
foo(); // no-warning
|
|
|
|
}
|
|
|
|
~E() { bar(); }
|
|
|
|
int foo() override;
|
|
|
|
};
|
|
|
|
|
2017-01-31 13:23:20 +08:00
|
|
|
class F {
|
|
|
|
public:
|
|
|
|
F() {
|
2017-08-28 16:44:43 +08:00
|
|
|
void (F::*ptr)() = &F::foo;
|
2017-01-31 13:23:20 +08:00
|
|
|
(this->*ptr)();
|
|
|
|
}
|
|
|
|
void foo();
|
|
|
|
};
|
|
|
|
|
2017-08-28 16:44:43 +08:00
|
|
|
class G {
|
|
|
|
public:
|
|
|
|
G() {}
|
|
|
|
virtual void bar();
|
|
|
|
void foo() {
|
|
|
|
bar(); // no warning
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class H {
|
|
|
|
public:
|
|
|
|
H() : initState(0) { init(); }
|
|
|
|
int initState;
|
|
|
|
virtual void f() const;
|
|
|
|
void init() {
|
|
|
|
if (initState)
|
|
|
|
f(); // no warning
|
|
|
|
}
|
|
|
|
|
|
|
|
H(int i) {
|
|
|
|
G g;
|
|
|
|
g.foo();
|
|
|
|
g.bar(); // no warning
|
2019-08-21 05:41:14 +08:00
|
|
|
f(); // impure-warning {{Call to virtual method 'H::f' during construction bypasses virtual dispatch}}
|
2017-08-28 16:44:43 +08:00
|
|
|
H &h = *this;
|
2019-08-21 05:41:14 +08:00
|
|
|
h.f(); // impure-warning {{Call to virtual method 'H::f' during construction bypasses virtual dispatch}}
|
2017-08-28 16:44:43 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class X {
|
|
|
|
public:
|
|
|
|
X() {
|
2019-08-21 05:41:14 +08:00
|
|
|
g(); // impure-warning {{Call to virtual method 'X::g' during construction bypasses virtual dispatch}}
|
2017-08-28 16:44:43 +08:00
|
|
|
}
|
|
|
|
X(int i) {
|
|
|
|
if (i > 0) {
|
|
|
|
X x(i - 1);
|
|
|
|
x.g(); // no warning
|
|
|
|
}
|
2019-08-21 05:41:14 +08:00
|
|
|
g(); // impure-warning {{Call to virtual method 'X::g' during construction bypasses virtual dispatch}}
|
2017-08-28 16:44:43 +08:00
|
|
|
}
|
|
|
|
virtual void g();
|
|
|
|
};
|
|
|
|
|
|
|
|
class M;
|
|
|
|
class N {
|
|
|
|
public:
|
|
|
|
virtual void virtualMethod();
|
|
|
|
void callFooOfM(M *);
|
|
|
|
};
|
|
|
|
class M {
|
|
|
|
public:
|
|
|
|
M() {
|
|
|
|
N n;
|
|
|
|
n.virtualMethod(); // no warning
|
|
|
|
n.callFooOfM(this);
|
|
|
|
}
|
|
|
|
virtual void foo();
|
|
|
|
};
|
|
|
|
void N::callFooOfM(M *m) {
|
2019-08-21 05:41:14 +08:00
|
|
|
m->foo(); // impure-warning {{Call to virtual method 'M::foo' during construction bypasses virtual dispatch}}
|
2012-01-04 07:18:57 +08:00
|
|
|
}
|
2012-10-11 01:55:40 +08:00
|
|
|
|
2017-08-28 16:44:43 +08:00
|
|
|
class Y {
|
|
|
|
public:
|
|
|
|
virtual void foobar();
|
|
|
|
void fooY() {
|
|
|
|
F f1;
|
2019-08-21 05:41:14 +08:00
|
|
|
foobar(); // impure-warning {{Call to virtual method 'Y::foobar' during construction bypasses virtual dispatch}}
|
2017-08-28 16:44:43 +08:00
|
|
|
}
|
|
|
|
Y() { fooY(); }
|
|
|
|
};
|
2012-10-11 01:55:40 +08:00
|
|
|
|
2017-08-28 16:44:43 +08:00
|
|
|
int main() {
|
|
|
|
B b;
|
|
|
|
C c;
|
|
|
|
D d;
|
|
|
|
E e;
|
|
|
|
F f;
|
|
|
|
G g;
|
|
|
|
H h;
|
|
|
|
H h1(1);
|
|
|
|
X x;
|
|
|
|
X x1(1);
|
|
|
|
M m;
|
|
|
|
Y *y = new Y;
|
|
|
|
delete y;
|
|
|
|
header::Z z;
|
|
|
|
}
|
2017-09-21 16:18:59 +08:00
|
|
|
|
|
|
|
namespace PR34451 {
|
|
|
|
struct a {
|
|
|
|
void b() {
|
|
|
|
a c[1];
|
|
|
|
c->b();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class e {
|
|
|
|
public:
|
|
|
|
void b() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
class c {
|
|
|
|
void m_fn2() const;
|
|
|
|
e d[];
|
|
|
|
};
|
|
|
|
|
|
|
|
void c::m_fn2() const { d->b(); }
|
|
|
|
}
|