In WebKit, we often write Foo::ensureBar function which lazily initializes m_bar and returns a raw pointer or a raw reference to m_bar. Such a return value is safe to use for the duration of a member function call in Foo so long as m_bar is const so that it never gets unset or updated with a new value once it's initialized. This PR adds support for recognizing these types of functions and treating its return value as a safe origin of a function argument (including "this") or a local variable.
116 lines
2.4 KiB
C++
116 lines
2.4 KiB
C++
// RUN: %clang_analyze_cc1 -analyzer-checker=alpha.webkit.UncountedCallArgsChecker -verify %s
|
|
|
|
#include "mock-types.h"
|
|
|
|
namespace std {
|
|
}
|
|
|
|
namespace call_args_const_refptr_member {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
private:
|
|
const RefPtr<RefCountable> m_obj1;
|
|
RefPtr<RefCountable> m_obj2;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is uncounted and unsafe}}
|
|
}
|
|
|
|
} // namespace call_args_const_refptr_member
|
|
|
|
namespace call_args_const_ref_member {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
private:
|
|
const Ref<RefCountable> m_obj1;
|
|
Ref<RefCountable> m_obj2;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is uncounted and unsafe}}
|
|
}
|
|
|
|
} // namespace call_args_const_ref_member
|
|
|
|
namespace call_args_const_unique_ptr {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
RefCountable& ensureObj3() {
|
|
if (!m_obj3)
|
|
const_cast<std::unique_ptr<RefCountable>&>(m_obj3) = RefCountable::makeUnique();
|
|
return *m_obj3;
|
|
}
|
|
|
|
RefCountable& badEnsureObj4() {
|
|
if (!m_obj4)
|
|
const_cast<std::unique_ptr<RefCountable>&>(m_obj4) = RefCountable::makeUnique();
|
|
if (auto* next = m_obj4->next())
|
|
return *next;
|
|
return *m_obj4;
|
|
}
|
|
|
|
RefCountable* ensureObj5() {
|
|
if (!m_obj5)
|
|
const_cast<std::unique_ptr<RefCountable>&>(m_obj5) = RefCountable::makeUnique();
|
|
if (m_obj5->next())
|
|
return nullptr;
|
|
return m_obj5.get();
|
|
}
|
|
|
|
private:
|
|
const std::unique_ptr<RefCountable> m_obj1;
|
|
std::unique_ptr<RefCountable> m_obj2;
|
|
const std::unique_ptr<RefCountable> m_obj3;
|
|
const std::unique_ptr<RefCountable> m_obj4;
|
|
const std::unique_ptr<RefCountable> m_obj5;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is uncounted and unsafe}}
|
|
ensureObj3().method();
|
|
badEnsureObj4().method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is uncounted and unsafe}}
|
|
ensureObj5()->method();
|
|
}
|
|
|
|
} // namespace call_args_const_unique_ptr
|
|
|
|
namespace call_args_const_unique_ref {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
private:
|
|
const UniqueRef<RefCountable> m_obj1;
|
|
UniqueRef<RefCountable> m_obj2;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is uncounted and unsafe}}
|
|
}
|
|
|
|
} // namespace call_args_const_unique_ref
|