// RUN: %clang_analyze_cc1 -triple i386-apple-darwin10 -analyzer-checker=core,debug.ExprInspection -analyzer-config ipa=none -verify %s void clang_analyzer_eval(bool); class A { public: virtual void f(){}; }; class B : public A{ public: int m; }; class C : public A{}; class BB: public B{}; // A lot of the tests below have the if statement in them, which forces the // analyzer to explore both path - when the result is 0 and not. This makes // sure that we definitely know that the result is non-0 (as the result of // the cast). int testDynCastFromRadar() { B aa; A *a = &aa; const int* res = 0; B *b = dynamic_cast(a); static const int i = 5; if(b) { res = &i; } else { res = 0; } return *res; // no warning } int testBaseToBase1() { B b; B *pb = &b; B *pbb = dynamic_cast(pb); const int* res = 0; static const int i = 5; if (pbb) { res = &i; } else { res = 0; } return *res; // no warning } int testMultipleLevelsOfSubclassing1() { BB bb; B *pb = &bb; A *pa = pb; B *b = dynamic_cast(pa); const int* res = 0; static const int i = 5; if (b) { res = &i; } else { res = 0; } return *res; // no warning } int testMultipleLevelsOfSubclassing2() { BB bb; A *pbb = &bb; B *b = dynamic_cast(pbb); BB *s = dynamic_cast(b); const int* res = 0; static const int i = 5; if (s) { res = &i; } else { res = 0; } return *res; // no warning } int testMultipleLevelsOfSubclassing3() { BB bb; A *pbb = &bb; B *b = dynamic_cast(pbb); return b->m; // no warning } int testLHS() { B aa; A *a = &aa; return (dynamic_cast(a))->m; } int testLHS2() { B aa; A *a = &aa; return (*dynamic_cast(a)).m; } int testDynCastUnknown2(class A *a) { B *b = dynamic_cast(a); return b->m; // no warning } int testDynCastUnknown(class A *a) { B *b = dynamic_cast(a); const int* res = 0; static const int i = 5; if (b) { res = &i; } else { res = 0; } return *res; // expected-warning {{Dereference of null pointer}} } int testDynCastFail2() { C c; A *pa = &c; B *b = dynamic_cast(pa); return b->m; // expected-warning {{dereference of a null pointer}} } int testLHSFail() { C c; A *a = &c; return (*dynamic_cast(a)).m; // expected-warning {{Dereference of null pointer}} } int testBaseToDerivedFail() { A a; B *b = dynamic_cast(&a); return b->m; // expected-warning {{dereference of a null pointer}} } int testConstZeroFail() { B *b = dynamic_cast((A *)0); return b->m; // expected-warning {{dereference of a null pointer}} } int testConstZeroFail2() { A *a = 0; B *b = dynamic_cast(a); return b->m; // expected-warning {{dereference of a null pointer}} } int testUpcast() { B b; A *a = dynamic_cast(&b); const int* res = 0; static const int i = 5; if (a) { res = &i; } else { res = 0; } return *res; // no warning } int testCastToVoidStar() { A a; void *b = dynamic_cast(&a); const int* res = 0; static const int i = 5; if (b) { res = &i; } else { res = 0; } return *res; // no warning } int testReferenceSuccessfulCast() { B rb; B &b = dynamic_cast(rb); int *x = 0; return *x; // expected-warning {{Dereference of null pointer}} } int testReferenceFailedCast() { A a; B &b = dynamic_cast(a); int *x = 0; return *x; // no warning (An exception is thrown by the cast.) } // Here we allow any outcome of the cast and this is good because there is a // situation where this will fail. So if the user has written the code in this // way, we assume they expect the cast to succeed. // Note, this might need special handling if we track types of symbolic casts // and use them for dynamic_cast handling. int testDynCastMostLikelyWillFail(C *c) { B *b = 0; b = dynamic_cast(c); const int* res = 0; static const int i = 5; if (b) { res = &i; } else { res = 0; } // Note: IPA is turned off for this test because the code below shows how the // dynamic_cast could succeed. return *res; // expected-warning{{Dereference of null pointer}} } class M : public B, public C {}; void callTestDynCastMostLikelyWillFail() { M m; testDynCastMostLikelyWillFail(&m); } void testDynCastToMiddleClass () { class BBB : public BB {}; BBB obj; A &ref = obj; // These didn't always correctly layer base regions. B *ptr = dynamic_cast(&ref); clang_analyzer_eval(ptr != 0); // expected-warning{{TRUE}} // This is actually statically resolved to be a DerivedToBase cast. ptr = dynamic_cast(&obj); clang_analyzer_eval(ptr != 0); // expected-warning{{TRUE}} } // ----------------------------- // False positives/negatives. // ----------------------------- // Due to symbolic regions not being typed. int testDynCastFalsePositive(BB *c) { B *b = 0; b = dynamic_cast(c); const int* res = 0; static const int i = 5; if (b) { res = &i; } else { res = 0; } return *res; // expected-warning{{Dereference of null pointer}} } // Does not work when we new an object. int testDynCastFail3() { A *a = new A(); B *b = dynamic_cast(a); return b->m; }