xref: /llvm-project/clang/test/Analysis/inlining/inline-defensive-checks.cpp (revision 57790c56853d57b675f650194dbf7a11d71bc5a5)
1184c6242SDominic Chen // RUN: %clang_analyze_cc1 -analyzer-checker=core -verify %s
25497d1a5SAnna Zaks // expected-no-diagnostics
35497d1a5SAnna Zaks 
42672a4ccSAnna Zaks extern void __assert_fail (__const char *__assertion, __const char *__file,
52672a4ccSAnna Zaks                            unsigned int __line, __const char *__function)
62672a4ccSAnna Zaks __attribute__ ((__noreturn__));
72672a4ccSAnna Zaks #define assert(expr) \
82672a4ccSAnna Zaks ((expr)  ? (void)(0)  : __assert_fail (#expr, __FILE__, __LINE__, __func__))
92672a4ccSAnna Zaks 
105497d1a5SAnna Zaks class ButterFly {
115497d1a5SAnna Zaks private:
ButterFly()125497d1a5SAnna Zaks   ButterFly() { }
135497d1a5SAnna Zaks public:
triggerderef()145497d1a5SAnna Zaks 	int triggerderef() {
155497d1a5SAnna Zaks 		return 0;
165497d1a5SAnna Zaks 	}
175497d1a5SAnna Zaks };
185497d1a5SAnna Zaks ButterFly *getInP();
195497d1a5SAnna Zaks class X{
205497d1a5SAnna Zaks 	ButterFly *p;
setP(ButterFly * inP)215497d1a5SAnna Zaks 	void setP(ButterFly *inP) {
225497d1a5SAnna Zaks 		if(inP)
235497d1a5SAnna Zaks       ;
245497d1a5SAnna Zaks 		p = inP;
255497d1a5SAnna Zaks 	};
subtest1()265497d1a5SAnna Zaks 	void subtest1() {
275497d1a5SAnna Zaks 		ButterFly *inP = getInP();
285497d1a5SAnna Zaks 		setP(inP);
295497d1a5SAnna Zaks 	}
subtest2()305497d1a5SAnna Zaks 	int subtest2() {
315497d1a5SAnna Zaks 		int c = p->triggerderef(); // no-warning
325497d1a5SAnna Zaks 		return c;
335497d1a5SAnna Zaks 	}
test()345497d1a5SAnna Zaks 	int test() {
355497d1a5SAnna Zaks 		subtest1();
365497d1a5SAnna Zaks 		return subtest2();
375497d1a5SAnna Zaks 	}
385497d1a5SAnna Zaks };
392672a4ccSAnna Zaks 
402672a4ccSAnna Zaks typedef const int *Ty;
412672a4ccSAnna Zaks extern
422672a4ccSAnna Zaks Ty notNullArg(Ty cf) __attribute__((nonnull));
432672a4ccSAnna Zaks typedef const void *CFTypeRef;
442672a4ccSAnna Zaks extern Ty getTyVal();
radar13224271_callee(Ty def,Ty & result)452672a4ccSAnna Zaks inline void radar13224271_callee(Ty def, Ty& result ) {
462672a4ccSAnna Zaks 	result = def;
472672a4ccSAnna Zaks   // Clearly indicates that result cannot be 0 if def is not NULL.
482672a4ccSAnna Zaks 	assert( (result != 0) || (def == 0) );
492672a4ccSAnna Zaks }
radar13224271_caller()502672a4ccSAnna Zaks void radar13224271_caller()
512672a4ccSAnna Zaks {
522672a4ccSAnna Zaks 	Ty value;
532672a4ccSAnna Zaks 	radar13224271_callee(getTyVal(), value );
542672a4ccSAnna Zaks 	notNullArg(value); // no-warning
552672a4ccSAnna Zaks }
5694b48bdbSAnna Zaks 
5794b48bdbSAnna Zaks struct Foo {
5894b48bdbSAnna Zaks 	int *ptr;
FooFoo5994b48bdbSAnna Zaks 	Foo(int *p)  {
6094b48bdbSAnna Zaks 		*p = 1; // no-warning
6194b48bdbSAnna Zaks 	}
6294b48bdbSAnna Zaks };
idc(int * p3)6394b48bdbSAnna Zaks void idc(int *p3) {
6494b48bdbSAnna Zaks   if (p3)
6594b48bdbSAnna Zaks     ;
6694b48bdbSAnna Zaks }
retNull()6794b48bdbSAnna Zaks int *retNull() {
6894b48bdbSAnna Zaks   return 0;
6994b48bdbSAnna Zaks }
test(int * p1,int * p2)7094b48bdbSAnna Zaks void test(int *p1, int *p2) {
7194b48bdbSAnna Zaks   idc(p1);
7294b48bdbSAnna Zaks 	Foo f(p1);
7394b48bdbSAnna Zaks }
7437de8888SArtem Dergachev 
7537de8888SArtem Dergachev struct Bar {
7637de8888SArtem Dergachev   int x;
7737de8888SArtem Dergachev };
idcBar(Bar * b)7837de8888SArtem Dergachev void idcBar(Bar *b) {
7937de8888SArtem Dergachev   if (b)
8037de8888SArtem Dergachev     ;
8137de8888SArtem Dergachev }
testRefToField(Bar * b)8237de8888SArtem Dergachev void testRefToField(Bar *b) {
8337de8888SArtem Dergachev   idcBar(b);
8437de8888SArtem Dergachev   int &x = b->x; // no-warning
8537de8888SArtem Dergachev   x = 5;
8637de8888SArtem Dergachev }
87*57790c56SArtem Dergachev 
88*57790c56SArtem Dergachev namespace get_deref_expr_with_cleanups {
89*57790c56SArtem Dergachev struct S {
90*57790c56SArtem Dergachev ~S();
91*57790c56SArtem Dergachev };
92*57790c56SArtem Dergachev S *conjure();
93*57790c56SArtem Dergachev // The argument won't be used, but it'll cause cleanups
94*57790c56SArtem Dergachev // to appear around the call site.
get_conjured(S _)95*57790c56SArtem Dergachev S *get_conjured(S _) {
96*57790c56SArtem Dergachev   S *s = conjure();
97*57790c56SArtem Dergachev   if (s) {}
98*57790c56SArtem Dergachev   return s;
99*57790c56SArtem Dergachev }
test_conjured()100*57790c56SArtem Dergachev void test_conjured() {
101*57790c56SArtem Dergachev   S &s = *get_conjured(S()); // no-warning
102*57790c56SArtem Dergachev }
103*57790c56SArtem Dergachev } // namespace get_deref_expr_with_cleanups
104