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.
93 lines
2.0 KiB
C++
93 lines
2.0 KiB
C++
// RUN: %clang_analyze_cc1 -analyzer-checker=alpha.webkit.UncheckedCallArgsChecker -verify %s
|
|
|
|
#include "mock-types.h"
|
|
|
|
namespace call_args_const_checkedptr_member {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
private:
|
|
const CheckedPtr<CheckedObj> m_obj1;
|
|
CheckedPtr<CheckedObj> m_obj2;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is unchecked and unsafe}}
|
|
}
|
|
|
|
} // namespace call_args_const_checkedptr_member
|
|
|
|
namespace call_args_const_checkedref_member {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
private:
|
|
const CheckedRef<CheckedObj> m_obj1;
|
|
CheckedRef<CheckedObj> m_obj2;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is unchecked and unsafe}}
|
|
}
|
|
|
|
} // namespace call_args_const_checkedref_member
|
|
|
|
namespace call_args_const_unique_ptr {
|
|
|
|
class Foo {
|
|
public:
|
|
Foo();
|
|
void bar();
|
|
|
|
CheckedObj& ensureObj3() {
|
|
if (!m_obj3)
|
|
const_cast<std::unique_ptr<CheckedObj>&>(m_obj3) = new CheckedObj;
|
|
return *m_obj3;
|
|
}
|
|
|
|
CheckedObj& badEnsureObj4() {
|
|
if (!m_obj4)
|
|
const_cast<std::unique_ptr<CheckedObj>&>(m_obj4) = new CheckedObj;
|
|
if (auto* next = m_obj4->next())
|
|
return *next;
|
|
return *m_obj4;
|
|
}
|
|
|
|
CheckedObj* ensureObj5() {
|
|
if (!m_obj5)
|
|
const_cast<std::unique_ptr<CheckedObj>&>(m_obj5) = new CheckedObj;
|
|
if (m_obj5->next())
|
|
return nullptr;
|
|
return m_obj5.get();
|
|
}
|
|
|
|
private:
|
|
const std::unique_ptr<CheckedObj> m_obj1;
|
|
std::unique_ptr<CheckedObj> m_obj2;
|
|
const std::unique_ptr<CheckedObj> m_obj3;
|
|
const std::unique_ptr<CheckedObj> m_obj4;
|
|
const std::unique_ptr<CheckedObj> m_obj5;
|
|
};
|
|
|
|
void Foo::bar() {
|
|
m_obj1->method();
|
|
m_obj2->method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is unchecked and unsafe}}
|
|
ensureObj3().method();
|
|
badEnsureObj4().method();
|
|
// expected-warning@-1{{Call argument for 'this' parameter is unchecked and unsafe}}
|
|
ensureObj5()->method();
|
|
}
|
|
|
|
} // namespace call_args_const_unique_ptr
|