forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			224 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			224 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
// 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
 | 
						|
 | 
						|
 | 
						|
#include "virtualcall.h"
 | 
						|
 | 
						|
void clang_analyzer_warnIfReached();
 | 
						|
 | 
						|
class A {
 | 
						|
public:
 | 
						|
  A();
 | 
						|
 | 
						|
  ~A(){};
 | 
						|
 | 
						|
  virtual int foo() = 0;
 | 
						|
  virtual void bar() = 0;
 | 
						|
  void f() {
 | 
						|
    foo(); // pure-warning{{Call to pure virtual method 'A::foo' during construction has undefined behavior}}
 | 
						|
    clang_analyzer_warnIfReached(); // no-warning
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
A::A() {
 | 
						|
  f();
 | 
						|
}
 | 
						|
 | 
						|
class B {
 | 
						|
public:
 | 
						|
  B() {
 | 
						|
    foo(); // impure-warning {{Call to virtual method 'B::foo' during construction bypasses virtual dispatch}}
 | 
						|
  }
 | 
						|
  ~B();
 | 
						|
 | 
						|
  virtual int foo();
 | 
						|
  virtual void bar() {
 | 
						|
    foo(); // impure-warning {{Call to virtual method 'B::foo' during destruction bypasses virtual dispatch}}
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
B::~B() {
 | 
						|
  this->B::foo(); // no-warning
 | 
						|
  this->B::bar();
 | 
						|
  this->foo(); // impure-warning {{Call to virtual method 'B::foo' during destruction bypasses virtual dispatch}}
 | 
						|
}
 | 
						|
 | 
						|
class C : public B {
 | 
						|
public:
 | 
						|
  C();
 | 
						|
  ~C();
 | 
						|
 | 
						|
  virtual int foo();
 | 
						|
  void f(int i);
 | 
						|
};
 | 
						|
 | 
						|
C::C() {
 | 
						|
  f(foo()); // impure-warning {{Call to virtual method 'C::foo' during construction bypasses virtual dispatch}}
 | 
						|
}
 | 
						|
 | 
						|
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;
 | 
						|
};
 | 
						|
 | 
						|
class F {
 | 
						|
public:
 | 
						|
  F() {
 | 
						|
    void (F::*ptr)() = &F::foo;
 | 
						|
    (this->*ptr)();
 | 
						|
  }
 | 
						|
  void foo();
 | 
						|
};
 | 
						|
 | 
						|
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
 | 
						|
    f(); // impure-warning {{Call to virtual method 'H::f' during construction bypasses virtual dispatch}}
 | 
						|
    H &h = *this;
 | 
						|
    h.f(); // impure-warning {{Call to virtual method 'H::f' during construction bypasses virtual dispatch}}
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
class X {
 | 
						|
public:
 | 
						|
  X() {
 | 
						|
    g(); // impure-warning {{Call to virtual method 'X::g' during construction bypasses virtual dispatch}}
 | 
						|
  }
 | 
						|
  X(int i) {
 | 
						|
    if (i > 0) {
 | 
						|
      X x(i - 1);
 | 
						|
      x.g(); // no warning
 | 
						|
    }
 | 
						|
    g(); // impure-warning {{Call to virtual method 'X::g' during construction bypasses virtual dispatch}}
 | 
						|
  }
 | 
						|
  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) {
 | 
						|
  m->foo(); // impure-warning {{Call to virtual method 'M::foo' during construction bypasses virtual dispatch}}
 | 
						|
}
 | 
						|
 | 
						|
class Y {
 | 
						|
public:
 | 
						|
  virtual void foobar();
 | 
						|
  void fooY() {
 | 
						|
    F f1;
 | 
						|
    foobar(); // impure-warning {{Call to virtual method 'Y::foobar' during construction bypasses virtual dispatch}}
 | 
						|
  }
 | 
						|
  Y() { fooY(); }
 | 
						|
};
 | 
						|
 | 
						|
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;
 | 
						|
}
 | 
						|
 | 
						|
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(); }
 | 
						|
}
 |