2017-03-04 02:02:02 +08:00
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=core -verify %s
|
2013-03-09 11:23:19 +08:00
|
|
|
// expected-no-diagnostics
|
|
|
|
|
2013-03-15 06:31:56 +08:00
|
|
|
extern void __assert_fail (__const char *__assertion, __const char *__file,
|
|
|
|
unsigned int __line, __const char *__function)
|
|
|
|
__attribute__ ((__noreturn__));
|
|
|
|
#define assert(expr) \
|
|
|
|
((expr) ? (void)(0) : __assert_fail (#expr, __FILE__, __LINE__, __func__))
|
|
|
|
|
2013-03-09 11:23:19 +08:00
|
|
|
class ButterFly {
|
|
|
|
private:
|
|
|
|
ButterFly() { }
|
|
|
|
public:
|
|
|
|
int triggerderef() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
ButterFly *getInP();
|
|
|
|
class X{
|
|
|
|
ButterFly *p;
|
|
|
|
void setP(ButterFly *inP) {
|
|
|
|
if(inP)
|
|
|
|
;
|
|
|
|
p = inP;
|
|
|
|
};
|
|
|
|
void subtest1() {
|
|
|
|
ButterFly *inP = getInP();
|
|
|
|
setP(inP);
|
|
|
|
}
|
|
|
|
int subtest2() {
|
|
|
|
int c = p->triggerderef(); // no-warning
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
int test() {
|
|
|
|
subtest1();
|
|
|
|
return subtest2();
|
|
|
|
}
|
2013-03-15 06:31:56 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef const int *Ty;
|
|
|
|
extern
|
|
|
|
Ty notNullArg(Ty cf) __attribute__((nonnull));
|
|
|
|
typedef const void *CFTypeRef;
|
|
|
|
extern Ty getTyVal();
|
|
|
|
inline void radar13224271_callee(Ty def, Ty& result ) {
|
|
|
|
result = def;
|
|
|
|
// Clearly indicates that result cannot be 0 if def is not NULL.
|
|
|
|
assert( (result != 0) || (def == 0) );
|
|
|
|
}
|
|
|
|
void radar13224271_caller()
|
|
|
|
{
|
|
|
|
Ty value;
|
|
|
|
radar13224271_callee(getTyVal(), value );
|
|
|
|
notNullArg(value); // no-warning
|
2013-04-06 07:50:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
struct Foo {
|
|
|
|
int *ptr;
|
|
|
|
Foo(int *p) {
|
|
|
|
*p = 1; // no-warning
|
|
|
|
}
|
|
|
|
};
|
|
|
|
void idc(int *p3) {
|
|
|
|
if (p3)
|
|
|
|
;
|
|
|
|
}
|
|
|
|
int *retNull() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
void test(int *p1, int *p2) {
|
|
|
|
idc(p1);
|
|
|
|
Foo f(p1);
|
2017-04-25 03:30:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
struct Bar {
|
|
|
|
int x;
|
|
|
|
};
|
|
|
|
void idcBar(Bar *b) {
|
|
|
|
if (b)
|
|
|
|
;
|
|
|
|
}
|
|
|
|
void testRefToField(Bar *b) {
|
|
|
|
idcBar(b);
|
|
|
|
int &x = b->x; // no-warning
|
|
|
|
x = 5;
|
|
|
|
}
|
2018-06-26 07:55:07 +08:00
|
|
|
|
|
|
|
namespace get_deref_expr_with_cleanups {
|
|
|
|
struct S {
|
|
|
|
~S();
|
|
|
|
};
|
|
|
|
S *conjure();
|
|
|
|
// The argument won't be used, but it'll cause cleanups
|
|
|
|
// to appear around the call site.
|
|
|
|
S *get_conjured(S _) {
|
|
|
|
S *s = conjure();
|
|
|
|
if (s) {}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
void test_conjured() {
|
|
|
|
S &s = *get_conjured(S()); // no-warning
|
|
|
|
}
|
|
|
|
} // namespace get_deref_expr_with_cleanups
|