Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
#include "clang/ASTMatchers/ASTMatchers.h"
#include "clang/Analysis/CFG.h"
#include "clang/Analysis/FlowSensitive/CFGMatchSwitch.h"
#include "clang/Analysis/FlowSensitive/CachedConstAccessorsLattice.h"
#include "clang/Analysis/FlowSensitive/DataflowAnalysis.h"
#include "clang/Analysis/FlowSensitive/DataflowEnvironment.h"
#include "clang/Analysis/FlowSensitive/MatchSwitch.h"
Expand Down Expand Up @@ -69,7 +70,8 @@ struct UncheckedStatusOrAccessModelOptions {};

// Dataflow analysis that discovers unsafe uses of StatusOr values.
class UncheckedStatusOrAccessModel
: public DataflowAnalysis<UncheckedStatusOrAccessModel, NoopLattice> {
: public DataflowAnalysis<UncheckedStatusOrAccessModel,
CachedConstAccessorsLattice<NoopLattice>> {
public:
explicit UncheckedStatusOrAccessModel(ASTContext &Ctx, Environment &Env);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -237,6 +237,49 @@ static auto isAsStatusCallWithStatusOr() {
hasArgument(0, hasType(statusOrType())));
}

static auto possiblyReferencedStatusOrType() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return anyOf(statusOrType(), referenceType(pointee(statusOrType())));
}

static auto isConstStatusOrAccessorMemberCall() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return cxxMemberCallExpr(callee(
cxxMethodDecl(parameterCountIs(0), isConst(),
returns(qualType(possiblyReferencedStatusOrType())))));
}

static auto isConstStatusOrAccessorMemberOperatorCall() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return cxxOperatorCallExpr(
callee(cxxMethodDecl(parameterCountIs(0), isConst(),
returns(possiblyReferencedStatusOrType()))));
}

static auto isConstStatusOrPointerAccessorMemberCall() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return cxxMemberCallExpr(callee(cxxMethodDecl(
parameterCountIs(0), isConst(),
returns(pointerType(pointee(possiblyReferencedStatusOrType()))))));
}

static auto isConstStatusOrPointerAccessorMemberOperatorCall() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return cxxOperatorCallExpr(callee(cxxMethodDecl(
parameterCountIs(0), isConst(),
returns(pointerType(pointee(possiblyReferencedStatusOrType()))))));
}

static auto isNonConstMemberCall() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return cxxMemberCallExpr(callee(cxxMethodDecl(unless(isConst()))));
}

static auto isNonConstMemberOperatorCall() {
using namespace ::clang::ast_matchers; // NOLINT: Too many names
return cxxOperatorCallExpr(callee(cxxMethodDecl(unless(isConst()))));
}

static auto
buildDiagnoseMatchSwitch(const UncheckedStatusOrAccessModelOptions &Options) {
return CFGMatchSwitchBuilder<const Environment,
Expand Down Expand Up @@ -698,6 +741,114 @@ static void transferPointerToBoolean(const ImplicitCastExpr *Expr,
State.Env.setValue(*Expr, *SubExprVal);
}

static void transferStatusOrReturningCall(const CallExpr *Expr,
LatticeTransferState &State) {
RecordStorageLocation *StatusOrLoc =
Expr->isPRValue() ? &State.Env.getResultObjectLocation(*Expr)
: State.Env.get<RecordStorageLocation>(*Expr);
if (StatusOrLoc != nullptr &&
State.Env.getValue(locForOk(locForStatus(*StatusOrLoc))) == nullptr)
initializeStatusOr(*StatusOrLoc, State.Env);
}

static bool doHandleConstStatusOrAccessorMemberCall(
const CallExpr *Expr, RecordStorageLocation *RecordLoc,
const MatchFinder::MatchResult &Result, LatticeTransferState &State) {
assert(isStatusOrType(Expr->getType()));
if (RecordLoc == nullptr)
return false;
const FunctionDecl *DirectCallee = Expr->getDirectCallee();
if (DirectCallee == nullptr)
return false;
StorageLocation &Loc =
State.Lattice.getOrCreateConstMethodReturnStorageLocation(
*RecordLoc, DirectCallee, State.Env, [&](StorageLocation &Loc) {
initializeStatusOr(cast<RecordStorageLocation>(Loc), State.Env);
});
if (Expr->isPRValue()) {
auto &ResultLoc = State.Env.getResultObjectLocation(*Expr);
copyRecord(cast<RecordStorageLocation>(Loc), ResultLoc, State.Env);
} else {
State.Env.setStorageLocation(*Expr, Loc);
}
return true;
}

static void handleConstStatusOrAccessorMemberCall(
const CallExpr *Expr, RecordStorageLocation *RecordLoc,
const MatchFinder::MatchResult &Result, LatticeTransferState &State) {
if (!doHandleConstStatusOrAccessorMemberCall(Expr, RecordLoc, Result, State))
transferStatusOrReturningCall(Expr, State);
}
static void handleConstStatusOrPointerAccessorMemberCall(
const CallExpr *Expr, RecordStorageLocation *RecordLoc,
const MatchFinder::MatchResult &Result, LatticeTransferState &State) {
if (RecordLoc == nullptr)
return;
auto *Val = State.Lattice.getOrCreateConstMethodReturnValue(*RecordLoc, Expr,
State.Env);
State.Env.setValue(*Expr, *Val);
}

static void
transferConstStatusOrAccessorMemberCall(const CXXMemberCallExpr *Expr,
const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
handleConstStatusOrAccessorMemberCall(
Expr, getImplicitObjectLocation(*Expr, State.Env), Result, State);
}

static void transferConstStatusOrAccessorMemberOperatorCall(
const CXXOperatorCallExpr *Expr, const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
auto *RecordLoc = cast_or_null<RecordStorageLocation>(
State.Env.getStorageLocation(*Expr->getArg(0)));
handleConstStatusOrAccessorMemberCall(Expr, RecordLoc, Result, State);
}

static void transferConstStatusOrPointerAccessorMemberCall(
const CXXMemberCallExpr *Expr, const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
handleConstStatusOrPointerAccessorMemberCall(
Expr, getImplicitObjectLocation(*Expr, State.Env), Result, State);
}

static void transferConstStatusOrPointerAccessorMemberOperatorCall(
const CXXOperatorCallExpr *Expr, const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
auto *RecordLoc = cast_or_null<RecordStorageLocation>(
State.Env.getStorageLocation(*Expr->getArg(0)));
handleConstStatusOrPointerAccessorMemberCall(Expr, RecordLoc, Result, State);
}

static void handleNonConstMemberCall(const CallExpr *Expr,
RecordStorageLocation *RecordLoc,
const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
if (RecordLoc) {
State.Lattice.clearConstMethodReturnValues(*RecordLoc);
State.Lattice.clearConstMethodReturnStorageLocations(*RecordLoc);
}
if (isStatusOrType(Expr->getType()))
transferStatusOrReturningCall(Expr, State);
}

static void transferNonConstMemberCall(const CXXMemberCallExpr *Expr,
const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
handleNonConstMemberCall(Expr, getImplicitObjectLocation(*Expr, State.Env),
Result, State);
}

static void
transferNonConstMemberOperatorCall(const CXXOperatorCallExpr *Expr,
const MatchFinder::MatchResult &Result,
LatticeTransferState &State) {
auto *RecordLoc = cast_or_null<RecordStorageLocation>(
State.Env.getStorageLocation(*Expr->getArg(0)));
handleNonConstMemberCall(Expr, RecordLoc, Result, State);
}

CFGMatchSwitch<LatticeTransferState>
buildTransferMatchSwitch(ASTContext &Ctx,
CFGMatchSwitchBuilder<LatticeTransferState> Builder) {
Expand Down Expand Up @@ -755,6 +906,23 @@ buildTransferMatchSwitch(ASTContext &Ctx,
transferLoggingGetReferenceableValueCall)
.CaseOfCFGStmt<CallExpr>(isLoggingCheckEqImpl(),
transferLoggingCheckEqImpl)
// const accessor calls
.CaseOfCFGStmt<CXXMemberCallExpr>(isConstStatusOrAccessorMemberCall(),
transferConstStatusOrAccessorMemberCall)
.CaseOfCFGStmt<CXXOperatorCallExpr>(
isConstStatusOrAccessorMemberOperatorCall(),
transferConstStatusOrAccessorMemberOperatorCall)
.CaseOfCFGStmt<CXXMemberCallExpr>(
isConstStatusOrPointerAccessorMemberCall(),
transferConstStatusOrPointerAccessorMemberCall)
.CaseOfCFGStmt<CXXOperatorCallExpr>(
isConstStatusOrPointerAccessorMemberOperatorCall(),
transferConstStatusOrPointerAccessorMemberOperatorCall)
// non-const member calls that may modify the state of an object.
.CaseOfCFGStmt<CXXMemberCallExpr>(isNonConstMemberCall(),
transferNonConstMemberCall)
.CaseOfCFGStmt<CXXOperatorCallExpr>(isNonConstMemberOperatorCall(),
transferNonConstMemberOperatorCall)
// N.B. These need to come after all other CXXConstructExpr.
// These are there to make sure that every Status and StatusOr object
// have their ok boolean initialized when constructed. If we were to
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3270,6 +3270,179 @@ TEST_P(UncheckedStatusOrAccessModelTest, ConstructStatusFromReference) {
)cc");
}

TEST_P(UncheckedStatusOrAccessModelTest, AccessorCall) {
// Accessor returns reference.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

const STATUSOR_INT& sor() const { return sor_; }
};

void target(Foo foo) {
if (foo.sor().ok()) foo.sor().value();
}
)cc");

// Uses an operator
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

const STATUSOR_INT& operator()() const { return sor_; }
};

void target(Foo foo) {
if (foo().ok()) foo().value();
}
)cc");

// Calls nonconst method in between.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

void invalidate() {}

const STATUSOR_INT& sor() const { return sor_; }
};

void target(Foo foo) {
if (foo.sor().ok()) {
foo.invalidate();
foo.sor().value(); // [[unsafe]]
}
}
)cc");

// Calls nonconst operator in between.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

void operator()() {}

const STATUSOR_INT& sor() const { return sor_; }
};

void target(Foo foo) {
if (foo.sor().ok()) {
foo();
foo.sor().value(); // [[unsafe]]
}
}
)cc");

// Accessor returns copy.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

STATUSOR_INT sor() const { return sor_; }
};

void target(Foo foo) {
if (foo.sor().ok()) foo.sor().value();
}
)cc");

// Non-const accessor.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

const STATUSOR_INT& sor() { return sor_; }
};

void target(Foo foo) {
if (foo.sor().ok()) foo.sor().value(); // [[unsafe]]
}
)cc");

// Non-const rvalue accessor.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

STATUSOR_INT&& sor() { return std::move(sor_); }
};

void target(Foo foo) {
if (foo.sor().ok()) foo.sor().value(); // [[unsafe]]
}
)cc");

// const pointer accessor.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

const STATUSOR_INT* sor() const { return &sor_; }
};

void target(Foo foo) {
if (foo.sor()->ok()) foo.sor()->value();
}
)cc");

// const pointer operator.
ExpectDiagnosticsFor(
R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

const STATUSOR_INT* operator->() const { return &sor_; }
};

void target(Foo foo) {
if (foo->ok()) foo->value();
}
)cc");

// We copy the result of the accessor.
ExpectDiagnosticsFor(R"cc(
#include "unchecked_statusor_access_test_defs.h"

struct Foo {
STATUSOR_INT sor_;

const STATUSOR_INT& sor() const { return sor_; }
};
void target() {
Foo foo;
if (!foo.sor().ok()) return;
const auto sor = foo.sor();
sor.value();
}
)cc");
}

} // namespace

std::string
Expand Down
Loading