1416 lines
43 KiB
C++
1416 lines
43 KiB
C++
//===- Consumed.cpp -------------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// A intra-procedural analysis for checking consumed properties. This is based,
|
|
// in part, on research on linear types.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/Analysis/Analyses/Consumed.h"
|
|
#include "clang/AST/Attr.h"
|
|
#include "clang/AST/Decl.h"
|
|
#include "clang/AST/DeclCXX.h"
|
|
#include "clang/AST/Expr.h"
|
|
#include "clang/AST/ExprCXX.h"
|
|
#include "clang/AST/Stmt.h"
|
|
#include "clang/AST/StmtVisitor.h"
|
|
#include "clang/AST/Type.h"
|
|
#include "clang/Analysis/Analyses/PostOrderCFGView.h"
|
|
#include "clang/Analysis/AnalysisDeclContext.h"
|
|
#include "clang/Analysis/CFG.h"
|
|
#include "clang/Basic/LLVM.h"
|
|
#include "clang/Basic/OperatorKinds.h"
|
|
#include "clang/Basic/SourceLocation.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/ADT/Optional.h"
|
|
#include "llvm/ADT/STLExtras.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/Support/Casting.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include <cassert>
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
// TODO: Adjust states of args to constructors in the same way that arguments to
|
|
// function calls are handled.
|
|
// TODO: Use information from tests in for- and while-loop conditional.
|
|
// TODO: Add notes about the actual and expected state for
|
|
// TODO: Correctly identify unreachable blocks when chaining boolean operators.
|
|
// TODO: Adjust the parser and AttributesList class to support lists of
|
|
// identifiers.
|
|
// TODO: Warn about unreachable code.
|
|
// TODO: Switch to using a bitmap to track unreachable blocks.
|
|
// TODO: Handle variable definitions, e.g. bool valid = x.isValid();
|
|
// if (valid) ...; (Deferred)
|
|
// TODO: Take notes on state transitions to provide better warning messages.
|
|
// (Deferred)
|
|
// TODO: Test nested conditionals: A) Checking the same value multiple times,
|
|
// and 2) Checking different values. (Deferred)
|
|
|
|
using namespace clang;
|
|
using namespace consumed;
|
|
|
|
// Key method definition
|
|
ConsumedWarningsHandlerBase::~ConsumedWarningsHandlerBase() = default;
|
|
|
|
static SourceLocation getFirstStmtLoc(const CFGBlock *Block) {
|
|
// Find the source location of the first statement in the block, if the block
|
|
// is not empty.
|
|
for (const auto &B : *Block)
|
|
if (Optional<CFGStmt> CS = B.getAs<CFGStmt>())
|
|
return CS->getStmt()->getBeginLoc();
|
|
|
|
// Block is empty.
|
|
// If we have one successor, return the first statement in that block
|
|
if (Block->succ_size() == 1 && *Block->succ_begin())
|
|
return getFirstStmtLoc(*Block->succ_begin());
|
|
|
|
return {};
|
|
}
|
|
|
|
static SourceLocation getLastStmtLoc(const CFGBlock *Block) {
|
|
// Find the source location of the last statement in the block, if the block
|
|
// is not empty.
|
|
if (const Stmt *StmtNode = Block->getTerminatorStmt()) {
|
|
return StmtNode->getBeginLoc();
|
|
} else {
|
|
for (CFGBlock::const_reverse_iterator BI = Block->rbegin(),
|
|
BE = Block->rend(); BI != BE; ++BI) {
|
|
if (Optional<CFGStmt> CS = BI->getAs<CFGStmt>())
|
|
return CS->getStmt()->getBeginLoc();
|
|
}
|
|
}
|
|
|
|
// If we have one successor, return the first statement in that block
|
|
SourceLocation Loc;
|
|
if (Block->succ_size() == 1 && *Block->succ_begin())
|
|
Loc = getFirstStmtLoc(*Block->succ_begin());
|
|
if (Loc.isValid())
|
|
return Loc;
|
|
|
|
// If we have one predecessor, return the last statement in that block
|
|
if (Block->pred_size() == 1 && *Block->pred_begin())
|
|
return getLastStmtLoc(*Block->pred_begin());
|
|
|
|
return Loc;
|
|
}
|
|
|
|
static ConsumedState invertConsumedUnconsumed(ConsumedState State) {
|
|
switch (State) {
|
|
case CS_Unconsumed:
|
|
return CS_Consumed;
|
|
case CS_Consumed:
|
|
return CS_Unconsumed;
|
|
case CS_None:
|
|
return CS_None;
|
|
case CS_Unknown:
|
|
return CS_Unknown;
|
|
}
|
|
llvm_unreachable("invalid enum");
|
|
}
|
|
|
|
static bool isCallableInState(const CallableWhenAttr *CWAttr,
|
|
ConsumedState State) {
|
|
for (const auto &S : CWAttr->callableStates()) {
|
|
ConsumedState MappedAttrState = CS_None;
|
|
|
|
switch (S) {
|
|
case CallableWhenAttr::Unknown:
|
|
MappedAttrState = CS_Unknown;
|
|
break;
|
|
|
|
case CallableWhenAttr::Unconsumed:
|
|
MappedAttrState = CS_Unconsumed;
|
|
break;
|
|
|
|
case CallableWhenAttr::Consumed:
|
|
MappedAttrState = CS_Consumed;
|
|
break;
|
|
}
|
|
|
|
if (MappedAttrState == State)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static bool isConsumableType(const QualType &QT) {
|
|
if (QT->isPointerType() || QT->isReferenceType())
|
|
return false;
|
|
|
|
if (const CXXRecordDecl *RD = QT->getAsCXXRecordDecl())
|
|
return RD->hasAttr<ConsumableAttr>();
|
|
|
|
return false;
|
|
}
|
|
|
|
static bool isAutoCastType(const QualType &QT) {
|
|
if (QT->isPointerType() || QT->isReferenceType())
|
|
return false;
|
|
|
|
if (const CXXRecordDecl *RD = QT->getAsCXXRecordDecl())
|
|
return RD->hasAttr<ConsumableAutoCastAttr>();
|
|
|
|
return false;
|
|
}
|
|
|
|
static bool isSetOnReadPtrType(const QualType &QT) {
|
|
if (const CXXRecordDecl *RD = QT->getPointeeCXXRecordDecl())
|
|
return RD->hasAttr<ConsumableSetOnReadAttr>();
|
|
return false;
|
|
}
|
|
|
|
static bool isKnownState(ConsumedState State) {
|
|
switch (State) {
|
|
case CS_Unconsumed:
|
|
case CS_Consumed:
|
|
return true;
|
|
case CS_None:
|
|
case CS_Unknown:
|
|
return false;
|
|
}
|
|
llvm_unreachable("invalid enum");
|
|
}
|
|
|
|
static bool isRValueRef(QualType ParamType) {
|
|
return ParamType->isRValueReferenceType();
|
|
}
|
|
|
|
static bool isTestingFunction(const FunctionDecl *FunDecl) {
|
|
return FunDecl->hasAttr<TestTypestateAttr>();
|
|
}
|
|
|
|
static bool isPointerOrRef(QualType ParamType) {
|
|
return ParamType->isPointerType() || ParamType->isReferenceType();
|
|
}
|
|
|
|
static ConsumedState mapConsumableAttrState(const QualType QT) {
|
|
assert(isConsumableType(QT));
|
|
|
|
const ConsumableAttr *CAttr =
|
|
QT->getAsCXXRecordDecl()->getAttr<ConsumableAttr>();
|
|
|
|
switch (CAttr->getDefaultState()) {
|
|
case ConsumableAttr::Unknown:
|
|
return CS_Unknown;
|
|
case ConsumableAttr::Unconsumed:
|
|
return CS_Unconsumed;
|
|
case ConsumableAttr::Consumed:
|
|
return CS_Consumed;
|
|
}
|
|
llvm_unreachable("invalid enum");
|
|
}
|
|
|
|
static ConsumedState
|
|
mapParamTypestateAttrState(const ParamTypestateAttr *PTAttr) {
|
|
switch (PTAttr->getParamState()) {
|
|
case ParamTypestateAttr::Unknown:
|
|
return CS_Unknown;
|
|
case ParamTypestateAttr::Unconsumed:
|
|
return CS_Unconsumed;
|
|
case ParamTypestateAttr::Consumed:
|
|
return CS_Consumed;
|
|
}
|
|
llvm_unreachable("invalid_enum");
|
|
}
|
|
|
|
static ConsumedState
|
|
mapReturnTypestateAttrState(const ReturnTypestateAttr *RTSAttr) {
|
|
switch (RTSAttr->getState()) {
|
|
case ReturnTypestateAttr::Unknown:
|
|
return CS_Unknown;
|
|
case ReturnTypestateAttr::Unconsumed:
|
|
return CS_Unconsumed;
|
|
case ReturnTypestateAttr::Consumed:
|
|
return CS_Consumed;
|
|
}
|
|
llvm_unreachable("invalid enum");
|
|
}
|
|
|
|
static ConsumedState mapSetTypestateAttrState(const SetTypestateAttr *STAttr) {
|
|
switch (STAttr->getNewState()) {
|
|
case SetTypestateAttr::Unknown:
|
|
return CS_Unknown;
|
|
case SetTypestateAttr::Unconsumed:
|
|
return CS_Unconsumed;
|
|
case SetTypestateAttr::Consumed:
|
|
return CS_Consumed;
|
|
}
|
|
llvm_unreachable("invalid_enum");
|
|
}
|
|
|
|
static StringRef stateToString(ConsumedState State) {
|
|
switch (State) {
|
|
case consumed::CS_None:
|
|
return "none";
|
|
|
|
case consumed::CS_Unknown:
|
|
return "unknown";
|
|
|
|
case consumed::CS_Unconsumed:
|
|
return "unconsumed";
|
|
|
|
case consumed::CS_Consumed:
|
|
return "consumed";
|
|
}
|
|
llvm_unreachable("invalid enum");
|
|
}
|
|
|
|
static ConsumedState testsFor(const FunctionDecl *FunDecl) {
|
|
assert(isTestingFunction(FunDecl));
|
|
switch (FunDecl->getAttr<TestTypestateAttr>()->getTestState()) {
|
|
case TestTypestateAttr::Unconsumed:
|
|
return CS_Unconsumed;
|
|
case TestTypestateAttr::Consumed:
|
|
return CS_Consumed;
|
|
}
|
|
llvm_unreachable("invalid enum");
|
|
}
|
|
|
|
namespace {
|
|
|
|
struct VarTestResult {
|
|
const VarDecl *Var;
|
|
ConsumedState TestsFor;
|
|
};
|
|
|
|
} // namespace
|
|
|
|
namespace clang {
|
|
namespace consumed {
|
|
|
|
enum EffectiveOp {
|
|
EO_And,
|
|
EO_Or
|
|
};
|
|
|
|
class PropagationInfo {
|
|
enum {
|
|
IT_None,
|
|
IT_State,
|
|
IT_VarTest,
|
|
IT_BinTest,
|
|
IT_Var,
|
|
IT_Tmp
|
|
} InfoType = IT_None;
|
|
|
|
struct BinTestTy {
|
|
const BinaryOperator *Source;
|
|
EffectiveOp EOp;
|
|
VarTestResult LTest;
|
|
VarTestResult RTest;
|
|
};
|
|
|
|
union {
|
|
ConsumedState State;
|
|
VarTestResult VarTest;
|
|
const VarDecl *Var;
|
|
const CXXBindTemporaryExpr *Tmp;
|
|
BinTestTy BinTest;
|
|
};
|
|
|
|
public:
|
|
PropagationInfo() = default;
|
|
PropagationInfo(const VarTestResult &VarTest)
|
|
: InfoType(IT_VarTest), VarTest(VarTest) {}
|
|
|
|
PropagationInfo(const VarDecl *Var, ConsumedState TestsFor)
|
|
: InfoType(IT_VarTest) {
|
|
VarTest.Var = Var;
|
|
VarTest.TestsFor = TestsFor;
|
|
}
|
|
|
|
PropagationInfo(const BinaryOperator *Source, EffectiveOp EOp,
|
|
const VarTestResult <est, const VarTestResult &RTest)
|
|
: InfoType(IT_BinTest) {
|
|
BinTest.Source = Source;
|
|
BinTest.EOp = EOp;
|
|
BinTest.LTest = LTest;
|
|
BinTest.RTest = RTest;
|
|
}
|
|
|
|
PropagationInfo(const BinaryOperator *Source, EffectiveOp EOp,
|
|
const VarDecl *LVar, ConsumedState LTestsFor,
|
|
const VarDecl *RVar, ConsumedState RTestsFor)
|
|
: InfoType(IT_BinTest) {
|
|
BinTest.Source = Source;
|
|
BinTest.EOp = EOp;
|
|
BinTest.LTest.Var = LVar;
|
|
BinTest.LTest.TestsFor = LTestsFor;
|
|
BinTest.RTest.Var = RVar;
|
|
BinTest.RTest.TestsFor = RTestsFor;
|
|
}
|
|
|
|
PropagationInfo(ConsumedState State)
|
|
: InfoType(IT_State), State(State) {}
|
|
PropagationInfo(const VarDecl *Var) : InfoType(IT_Var), Var(Var) {}
|
|
PropagationInfo(const CXXBindTemporaryExpr *Tmp)
|
|
: InfoType(IT_Tmp), Tmp(Tmp) {}
|
|
|
|
const ConsumedState &getState() const {
|
|
assert(InfoType == IT_State);
|
|
return State;
|
|
}
|
|
|
|
const VarTestResult &getVarTest() const {
|
|
assert(InfoType == IT_VarTest);
|
|
return VarTest;
|
|
}
|
|
|
|
const VarTestResult &getLTest() const {
|
|
assert(InfoType == IT_BinTest);
|
|
return BinTest.LTest;
|
|
}
|
|
|
|
const VarTestResult &getRTest() const {
|
|
assert(InfoType == IT_BinTest);
|
|
return BinTest.RTest;
|
|
}
|
|
|
|
const VarDecl *getVar() const {
|
|
assert(InfoType == IT_Var);
|
|
return Var;
|
|
}
|
|
|
|
const CXXBindTemporaryExpr *getTmp() const {
|
|
assert(InfoType == IT_Tmp);
|
|
return Tmp;
|
|
}
|
|
|
|
ConsumedState getAsState(const ConsumedStateMap *StateMap) const {
|
|
assert(isVar() || isTmp() || isState());
|
|
|
|
if (isVar())
|
|
return StateMap->getState(Var);
|
|
else if (isTmp())
|
|
return StateMap->getState(Tmp);
|
|
else if (isState())
|
|
return State;
|
|
else
|
|
return CS_None;
|
|
}
|
|
|
|
EffectiveOp testEffectiveOp() const {
|
|
assert(InfoType == IT_BinTest);
|
|
return BinTest.EOp;
|
|
}
|
|
|
|
const BinaryOperator * testSourceNode() const {
|
|
assert(InfoType == IT_BinTest);
|
|
return BinTest.Source;
|
|
}
|
|
|
|
bool isValid() const { return InfoType != IT_None; }
|
|
bool isState() const { return InfoType == IT_State; }
|
|
bool isVarTest() const { return InfoType == IT_VarTest; }
|
|
bool isBinTest() const { return InfoType == IT_BinTest; }
|
|
bool isVar() const { return InfoType == IT_Var; }
|
|
bool isTmp() const { return InfoType == IT_Tmp; }
|
|
|
|
bool isTest() const {
|
|
return InfoType == IT_VarTest || InfoType == IT_BinTest;
|
|
}
|
|
|
|
bool isPointerToValue() const {
|
|
return InfoType == IT_Var || InfoType == IT_Tmp;
|
|
}
|
|
|
|
PropagationInfo invertTest() const {
|
|
assert(InfoType == IT_VarTest || InfoType == IT_BinTest);
|
|
|
|
if (InfoType == IT_VarTest) {
|
|
return PropagationInfo(VarTest.Var,
|
|
invertConsumedUnconsumed(VarTest.TestsFor));
|
|
|
|
} else if (InfoType == IT_BinTest) {
|
|
return PropagationInfo(BinTest.Source,
|
|
BinTest.EOp == EO_And ? EO_Or : EO_And,
|
|
BinTest.LTest.Var, invertConsumedUnconsumed(BinTest.LTest.TestsFor),
|
|
BinTest.RTest.Var, invertConsumedUnconsumed(BinTest.RTest.TestsFor));
|
|
} else {
|
|
return {};
|
|
}
|
|
}
|
|
};
|
|
|
|
} // namespace consumed
|
|
} // namespace clang
|
|
|
|
static void
|
|
setStateForVarOrTmp(ConsumedStateMap *StateMap, const PropagationInfo &PInfo,
|
|
ConsumedState State) {
|
|
assert(PInfo.isVar() || PInfo.isTmp());
|
|
|
|
if (PInfo.isVar())
|
|
StateMap->setState(PInfo.getVar(), State);
|
|
else
|
|
StateMap->setState(PInfo.getTmp(), State);
|
|
}
|
|
|
|
namespace clang {
|
|
namespace consumed {
|
|
|
|
class ConsumedStmtVisitor : public ConstStmtVisitor<ConsumedStmtVisitor> {
|
|
using MapType = llvm::DenseMap<const Stmt *, PropagationInfo>;
|
|
using PairType= std::pair<const Stmt *, PropagationInfo>;
|
|
using InfoEntry = MapType::iterator;
|
|
using ConstInfoEntry = MapType::const_iterator;
|
|
|
|
ConsumedAnalyzer &Analyzer;
|
|
ConsumedStateMap *StateMap;
|
|
MapType PropagationMap;
|
|
|
|
InfoEntry findInfo(const Expr *E) {
|
|
if (const auto Cleanups = dyn_cast<ExprWithCleanups>(E))
|
|
if (!Cleanups->cleanupsHaveSideEffects())
|
|
E = Cleanups->getSubExpr();
|
|
return PropagationMap.find(E->IgnoreParens());
|
|
}
|
|
|
|
ConstInfoEntry findInfo(const Expr *E) const {
|
|
if (const auto Cleanups = dyn_cast<ExprWithCleanups>(E))
|
|
if (!Cleanups->cleanupsHaveSideEffects())
|
|
E = Cleanups->getSubExpr();
|
|
return PropagationMap.find(E->IgnoreParens());
|
|
}
|
|
|
|
void insertInfo(const Expr *E, const PropagationInfo &PI) {
|
|
PropagationMap.insert(PairType(E->IgnoreParens(), PI));
|
|
}
|
|
|
|
void forwardInfo(const Expr *From, const Expr *To);
|
|
void copyInfo(const Expr *From, const Expr *To, ConsumedState CS);
|
|
ConsumedState getInfo(const Expr *From);
|
|
void setInfo(const Expr *To, ConsumedState NS);
|
|
void propagateReturnType(const Expr *Call, const FunctionDecl *Fun);
|
|
|
|
public:
|
|
void checkCallability(const PropagationInfo &PInfo,
|
|
const FunctionDecl *FunDecl,
|
|
SourceLocation BlameLoc);
|
|
bool handleCall(const CallExpr *Call, const Expr *ObjArg,
|
|
const FunctionDecl *FunD);
|
|
|
|
void VisitBinaryOperator(const BinaryOperator *BinOp);
|
|
void VisitCallExpr(const CallExpr *Call);
|
|
void VisitCastExpr(const CastExpr *Cast);
|
|
void VisitCXXBindTemporaryExpr(const CXXBindTemporaryExpr *Temp);
|
|
void VisitCXXConstructExpr(const CXXConstructExpr *Call);
|
|
void VisitCXXMemberCallExpr(const CXXMemberCallExpr *Call);
|
|
void VisitCXXOperatorCallExpr(const CXXOperatorCallExpr *Call);
|
|
void VisitDeclRefExpr(const DeclRefExpr *DeclRef);
|
|
void VisitDeclStmt(const DeclStmt *DelcS);
|
|
void VisitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *Temp);
|
|
void VisitMemberExpr(const MemberExpr *MExpr);
|
|
void VisitParmVarDecl(const ParmVarDecl *Param);
|
|
void VisitReturnStmt(const ReturnStmt *Ret);
|
|
void VisitUnaryOperator(const UnaryOperator *UOp);
|
|
void VisitVarDecl(const VarDecl *Var);
|
|
|
|
ConsumedStmtVisitor(ConsumedAnalyzer &Analyzer, ConsumedStateMap *StateMap)
|
|
: Analyzer(Analyzer), StateMap(StateMap) {}
|
|
|
|
PropagationInfo getInfo(const Expr *StmtNode) const {
|
|
ConstInfoEntry Entry = findInfo(StmtNode);
|
|
|
|
if (Entry != PropagationMap.end())
|
|
return Entry->second;
|
|
else
|
|
return {};
|
|
}
|
|
|
|
void reset(ConsumedStateMap *NewStateMap) {
|
|
StateMap = NewStateMap;
|
|
}
|
|
};
|
|
|
|
} // namespace consumed
|
|
} // namespace clang
|
|
|
|
void ConsumedStmtVisitor::forwardInfo(const Expr *From, const Expr *To) {
|
|
InfoEntry Entry = findInfo(From);
|
|
if (Entry != PropagationMap.end())
|
|
insertInfo(To, Entry->second);
|
|
}
|
|
|
|
// Create a new state for To, which is initialized to the state of From.
|
|
// If NS is not CS_None, sets the state of From to NS.
|
|
void ConsumedStmtVisitor::copyInfo(const Expr *From, const Expr *To,
|
|
ConsumedState NS) {
|
|
InfoEntry Entry = findInfo(From);
|
|
if (Entry != PropagationMap.end()) {
|
|
PropagationInfo& PInfo = Entry->second;
|
|
ConsumedState CS = PInfo.getAsState(StateMap);
|
|
if (CS != CS_None)
|
|
insertInfo(To, PropagationInfo(CS));
|
|
if (NS != CS_None && PInfo.isPointerToValue())
|
|
setStateForVarOrTmp(StateMap, PInfo, NS);
|
|
}
|
|
}
|
|
|
|
// Get the ConsumedState for From
|
|
ConsumedState ConsumedStmtVisitor::getInfo(const Expr *From) {
|
|
InfoEntry Entry = findInfo(From);
|
|
if (Entry != PropagationMap.end()) {
|
|
PropagationInfo& PInfo = Entry->second;
|
|
return PInfo.getAsState(StateMap);
|
|
}
|
|
return CS_None;
|
|
}
|
|
|
|
// If we already have info for To then update it, otherwise create a new entry.
|
|
void ConsumedStmtVisitor::setInfo(const Expr *To, ConsumedState NS) {
|
|
InfoEntry Entry = findInfo(To);
|
|
if (Entry != PropagationMap.end()) {
|
|
PropagationInfo& PInfo = Entry->second;
|
|
if (PInfo.isPointerToValue())
|
|
setStateForVarOrTmp(StateMap, PInfo, NS);
|
|
} else if (NS != CS_None) {
|
|
insertInfo(To, PropagationInfo(NS));
|
|
}
|
|
}
|
|
|
|
void ConsumedStmtVisitor::checkCallability(const PropagationInfo &PInfo,
|
|
const FunctionDecl *FunDecl,
|
|
SourceLocation BlameLoc) {
|
|
assert(!PInfo.isTest());
|
|
|
|
const CallableWhenAttr *CWAttr = FunDecl->getAttr<CallableWhenAttr>();
|
|
if (!CWAttr)
|
|
return;
|
|
|
|
if (PInfo.isVar()) {
|
|
ConsumedState VarState = StateMap->getState(PInfo.getVar());
|
|
|
|
if (VarState == CS_None || isCallableInState(CWAttr, VarState))
|
|
return;
|
|
|
|
Analyzer.WarningsHandler.warnUseInInvalidState(
|
|
FunDecl->getNameAsString(), PInfo.getVar()->getNameAsString(),
|
|
stateToString(VarState), BlameLoc);
|
|
} else {
|
|
ConsumedState TmpState = PInfo.getAsState(StateMap);
|
|
|
|
if (TmpState == CS_None || isCallableInState(CWAttr, TmpState))
|
|
return;
|
|
|
|
Analyzer.WarningsHandler.warnUseOfTempInInvalidState(
|
|
FunDecl->getNameAsString(), stateToString(TmpState), BlameLoc);
|
|
}
|
|
}
|
|
|
|
// Factors out common behavior for function, method, and operator calls.
|
|
// Check parameters and set parameter state if necessary.
|
|
// Returns true if the state of ObjArg is set, or false otherwise.
|
|
bool ConsumedStmtVisitor::handleCall(const CallExpr *Call, const Expr *ObjArg,
|
|
const FunctionDecl *FunD) {
|
|
unsigned Offset = 0;
|
|
if (isa<CXXOperatorCallExpr>(Call) && isa<CXXMethodDecl>(FunD))
|
|
Offset = 1; // first argument is 'this'
|
|
|
|
// check explicit parameters
|
|
for (unsigned Index = Offset; Index < Call->getNumArgs(); ++Index) {
|
|
// Skip variable argument lists.
|
|
if (Index - Offset >= FunD->getNumParams())
|
|
break;
|
|
|
|
const ParmVarDecl *Param = FunD->getParamDecl(Index - Offset);
|
|
QualType ParamType = Param->getType();
|
|
|
|
InfoEntry Entry = findInfo(Call->getArg(Index));
|
|
|
|
if (Entry == PropagationMap.end() || Entry->second.isTest())
|
|
continue;
|
|
PropagationInfo PInfo = Entry->second;
|
|
|
|
// Check that the parameter is in the correct state.
|
|
if (ParamTypestateAttr *PTA = Param->getAttr<ParamTypestateAttr>()) {
|
|
ConsumedState ParamState = PInfo.getAsState(StateMap);
|
|
ConsumedState ExpectedState = mapParamTypestateAttrState(PTA);
|
|
|
|
if (ParamState != ExpectedState)
|
|
Analyzer.WarningsHandler.warnParamTypestateMismatch(
|
|
Call->getArg(Index)->getExprLoc(),
|
|
stateToString(ExpectedState), stateToString(ParamState));
|
|
}
|
|
|
|
if (!(Entry->second.isVar() || Entry->second.isTmp()))
|
|
continue;
|
|
|
|
// Adjust state on the caller side.
|
|
if (ReturnTypestateAttr *RT = Param->getAttr<ReturnTypestateAttr>())
|
|
setStateForVarOrTmp(StateMap, PInfo, mapReturnTypestateAttrState(RT));
|
|
else if (isRValueRef(ParamType) || isConsumableType(ParamType))
|
|
setStateForVarOrTmp(StateMap, PInfo, consumed::CS_Consumed);
|
|
else if (isPointerOrRef(ParamType) &&
|
|
(!ParamType->getPointeeType().isConstQualified() ||
|
|
isSetOnReadPtrType(ParamType)))
|
|
setStateForVarOrTmp(StateMap, PInfo, consumed::CS_Unknown);
|
|
}
|
|
|
|
if (!ObjArg)
|
|
return false;
|
|
|
|
// check implicit 'self' parameter, if present
|
|
InfoEntry Entry = findInfo(ObjArg);
|
|
if (Entry != PropagationMap.end()) {
|
|
PropagationInfo PInfo = Entry->second;
|
|
checkCallability(PInfo, FunD, Call->getExprLoc());
|
|
|
|
if (SetTypestateAttr *STA = FunD->getAttr<SetTypestateAttr>()) {
|
|
if (PInfo.isVar()) {
|
|
StateMap->setState(PInfo.getVar(), mapSetTypestateAttrState(STA));
|
|
return true;
|
|
}
|
|
else if (PInfo.isTmp()) {
|
|
StateMap->setState(PInfo.getTmp(), mapSetTypestateAttrState(STA));
|
|
return true;
|
|
}
|
|
}
|
|
else if (isTestingFunction(FunD) && PInfo.isVar()) {
|
|
PropagationMap.insert(PairType(Call,
|
|
PropagationInfo(PInfo.getVar(), testsFor(FunD))));
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void ConsumedStmtVisitor::propagateReturnType(const Expr *Call,
|
|
const FunctionDecl *Fun) {
|
|
QualType RetType = Fun->getCallResultType();
|
|
if (RetType->isReferenceType())
|
|
RetType = RetType->getPointeeType();
|
|
|
|
if (isConsumableType(RetType)) {
|
|
ConsumedState ReturnState;
|
|
if (ReturnTypestateAttr *RTA = Fun->getAttr<ReturnTypestateAttr>())
|
|
ReturnState = mapReturnTypestateAttrState(RTA);
|
|
else
|
|
ReturnState = mapConsumableAttrState(RetType);
|
|
|
|
PropagationMap.insert(PairType(Call, PropagationInfo(ReturnState)));
|
|
}
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitBinaryOperator(const BinaryOperator *BinOp) {
|
|
switch (BinOp->getOpcode()) {
|
|
case BO_LAnd:
|
|
case BO_LOr : {
|
|
InfoEntry LEntry = findInfo(BinOp->getLHS()),
|
|
REntry = findInfo(BinOp->getRHS());
|
|
|
|
VarTestResult LTest, RTest;
|
|
|
|
if (LEntry != PropagationMap.end() && LEntry->second.isVarTest()) {
|
|
LTest = LEntry->second.getVarTest();
|
|
} else {
|
|
LTest.Var = nullptr;
|
|
LTest.TestsFor = CS_None;
|
|
}
|
|
|
|
if (REntry != PropagationMap.end() && REntry->second.isVarTest()) {
|
|
RTest = REntry->second.getVarTest();
|
|
} else {
|
|
RTest.Var = nullptr;
|
|
RTest.TestsFor = CS_None;
|
|
}
|
|
|
|
if (!(LTest.Var == nullptr && RTest.Var == nullptr))
|
|
PropagationMap.insert(PairType(BinOp, PropagationInfo(BinOp,
|
|
static_cast<EffectiveOp>(BinOp->getOpcode() == BO_LOr), LTest, RTest)));
|
|
break;
|
|
}
|
|
|
|
case BO_PtrMemD:
|
|
case BO_PtrMemI:
|
|
forwardInfo(BinOp->getLHS(), BinOp);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitCallExpr(const CallExpr *Call) {
|
|
const FunctionDecl *FunDecl = Call->getDirectCallee();
|
|
if (!FunDecl)
|
|
return;
|
|
|
|
// Special case for the std::move function.
|
|
// TODO: Make this more specific. (Deferred)
|
|
if (Call->isCallToStdMove()) {
|
|
copyInfo(Call->getArg(0), Call, CS_Consumed);
|
|
return;
|
|
}
|
|
|
|
handleCall(Call, nullptr, FunDecl);
|
|
propagateReturnType(Call, FunDecl);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitCastExpr(const CastExpr *Cast) {
|
|
forwardInfo(Cast->getSubExpr(), Cast);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitCXXBindTemporaryExpr(
|
|
const CXXBindTemporaryExpr *Temp) {
|
|
|
|
InfoEntry Entry = findInfo(Temp->getSubExpr());
|
|
|
|
if (Entry != PropagationMap.end() && !Entry->second.isTest()) {
|
|
StateMap->setState(Temp, Entry->second.getAsState(StateMap));
|
|
PropagationMap.insert(PairType(Temp, PropagationInfo(Temp)));
|
|
}
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitCXXConstructExpr(const CXXConstructExpr *Call) {
|
|
CXXConstructorDecl *Constructor = Call->getConstructor();
|
|
|
|
QualType ThisType = Constructor->getThisType()->getPointeeType();
|
|
|
|
if (!isConsumableType(ThisType))
|
|
return;
|
|
|
|
// FIXME: What should happen if someone annotates the move constructor?
|
|
if (ReturnTypestateAttr *RTA = Constructor->getAttr<ReturnTypestateAttr>()) {
|
|
// TODO: Adjust state of args appropriately.
|
|
ConsumedState RetState = mapReturnTypestateAttrState(RTA);
|
|
PropagationMap.insert(PairType(Call, PropagationInfo(RetState)));
|
|
} else if (Constructor->isDefaultConstructor()) {
|
|
PropagationMap.insert(PairType(Call,
|
|
PropagationInfo(consumed::CS_Consumed)));
|
|
} else if (Constructor->isMoveConstructor()) {
|
|
copyInfo(Call->getArg(0), Call, CS_Consumed);
|
|
} else if (Constructor->isCopyConstructor()) {
|
|
// Copy state from arg. If setStateOnRead then set arg to CS_Unknown.
|
|
ConsumedState NS =
|
|
isSetOnReadPtrType(Constructor->getThisType()) ?
|
|
CS_Unknown : CS_None;
|
|
copyInfo(Call->getArg(0), Call, NS);
|
|
} else {
|
|
// TODO: Adjust state of args appropriately.
|
|
ConsumedState RetState = mapConsumableAttrState(ThisType);
|
|
PropagationMap.insert(PairType(Call, PropagationInfo(RetState)));
|
|
}
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitCXXMemberCallExpr(
|
|
const CXXMemberCallExpr *Call) {
|
|
CXXMethodDecl* MD = Call->getMethodDecl();
|
|
if (!MD)
|
|
return;
|
|
|
|
handleCall(Call, Call->getImplicitObjectArgument(), MD);
|
|
propagateReturnType(Call, MD);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitCXXOperatorCallExpr(
|
|
const CXXOperatorCallExpr *Call) {
|
|
const auto *FunDecl = dyn_cast_or_null<FunctionDecl>(Call->getDirectCallee());
|
|
if (!FunDecl) return;
|
|
|
|
if (Call->getOperator() == OO_Equal) {
|
|
ConsumedState CS = getInfo(Call->getArg(1));
|
|
if (!handleCall(Call, Call->getArg(0), FunDecl))
|
|
setInfo(Call->getArg(0), CS);
|
|
return;
|
|
}
|
|
|
|
if (const auto *MCall = dyn_cast<CXXMemberCallExpr>(Call))
|
|
handleCall(MCall, MCall->getImplicitObjectArgument(), FunDecl);
|
|
else
|
|
handleCall(Call, Call->getArg(0), FunDecl);
|
|
|
|
propagateReturnType(Call, FunDecl);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitDeclRefExpr(const DeclRefExpr *DeclRef) {
|
|
if (const auto *Var = dyn_cast_or_null<VarDecl>(DeclRef->getDecl()))
|
|
if (StateMap->getState(Var) != consumed::CS_None)
|
|
PropagationMap.insert(PairType(DeclRef, PropagationInfo(Var)));
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitDeclStmt(const DeclStmt *DeclS) {
|
|
for (const auto *DI : DeclS->decls())
|
|
if (isa<VarDecl>(DI))
|
|
VisitVarDecl(cast<VarDecl>(DI));
|
|
|
|
if (DeclS->isSingleDecl())
|
|
if (const auto *Var = dyn_cast_or_null<VarDecl>(DeclS->getSingleDecl()))
|
|
PropagationMap.insert(PairType(DeclS, PropagationInfo(Var)));
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitMaterializeTemporaryExpr(
|
|
const MaterializeTemporaryExpr *Temp) {
|
|
forwardInfo(Temp->getSubExpr(), Temp);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitMemberExpr(const MemberExpr *MExpr) {
|
|
forwardInfo(MExpr->getBase(), MExpr);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitParmVarDecl(const ParmVarDecl *Param) {
|
|
QualType ParamType = Param->getType();
|
|
ConsumedState ParamState = consumed::CS_None;
|
|
|
|
if (const ParamTypestateAttr *PTA = Param->getAttr<ParamTypestateAttr>())
|
|
ParamState = mapParamTypestateAttrState(PTA);
|
|
else if (isConsumableType(ParamType))
|
|
ParamState = mapConsumableAttrState(ParamType);
|
|
else if (isRValueRef(ParamType) &&
|
|
isConsumableType(ParamType->getPointeeType()))
|
|
ParamState = mapConsumableAttrState(ParamType->getPointeeType());
|
|
else if (ParamType->isReferenceType() &&
|
|
isConsumableType(ParamType->getPointeeType()))
|
|
ParamState = consumed::CS_Unknown;
|
|
|
|
if (ParamState != CS_None)
|
|
StateMap->setState(Param, ParamState);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitReturnStmt(const ReturnStmt *Ret) {
|
|
ConsumedState ExpectedState = Analyzer.getExpectedReturnState();
|
|
|
|
if (ExpectedState != CS_None) {
|
|
InfoEntry Entry = findInfo(Ret->getRetValue());
|
|
|
|
if (Entry != PropagationMap.end()) {
|
|
ConsumedState RetState = Entry->second.getAsState(StateMap);
|
|
|
|
if (RetState != ExpectedState)
|
|
Analyzer.WarningsHandler.warnReturnTypestateMismatch(
|
|
Ret->getReturnLoc(), stateToString(ExpectedState),
|
|
stateToString(RetState));
|
|
}
|
|
}
|
|
|
|
StateMap->checkParamsForReturnTypestate(Ret->getBeginLoc(),
|
|
Analyzer.WarningsHandler);
|
|
}
|
|
|
|
void ConsumedStmtVisitor::VisitUnaryOperator(const UnaryOperator *UOp) {
|
|
InfoEntry Entry = findInfo(UOp->getSubExpr());
|
|
if (Entry == PropagationMap.end()) return;
|
|
|
|
switch (UOp->getOpcode()) {
|
|
case UO_AddrOf:
|
|
PropagationMap.insert(PairType(UOp, Entry->second));
|
|
break;
|
|
|
|
case UO_LNot:
|
|
if (Entry->second.isTest())
|
|
PropagationMap.insert(PairType(UOp, Entry->second.invertTest()));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// TODO: See if I need to check for reference types here.
|
|
void ConsumedStmtVisitor::VisitVarDecl(const VarDecl *Var) {
|
|
if (isConsumableType(Var->getType())) {
|
|
if (Var->hasInit()) {
|
|
MapType::iterator VIT = findInfo(Var->getInit()->IgnoreImplicit());
|
|
if (VIT != PropagationMap.end()) {
|
|
PropagationInfo PInfo = VIT->second;
|
|
ConsumedState St = PInfo.getAsState(StateMap);
|
|
|
|
if (St != consumed::CS_None) {
|
|
StateMap->setState(Var, St);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
// Otherwise
|
|
StateMap->setState(Var, consumed::CS_Unknown);
|
|
}
|
|
}
|
|
|
|
static void splitVarStateForIf(const IfStmt *IfNode, const VarTestResult &Test,
|
|
ConsumedStateMap *ThenStates,
|
|
ConsumedStateMap *ElseStates) {
|
|
ConsumedState VarState = ThenStates->getState(Test.Var);
|
|
|
|
if (VarState == CS_Unknown) {
|
|
ThenStates->setState(Test.Var, Test.TestsFor);
|
|
ElseStates->setState(Test.Var, invertConsumedUnconsumed(Test.TestsFor));
|
|
} else if (VarState == invertConsumedUnconsumed(Test.TestsFor)) {
|
|
ThenStates->markUnreachable();
|
|
} else if (VarState == Test.TestsFor) {
|
|
ElseStates->markUnreachable();
|
|
}
|
|
}
|
|
|
|
static void splitVarStateForIfBinOp(const PropagationInfo &PInfo,
|
|
ConsumedStateMap *ThenStates,
|
|
ConsumedStateMap *ElseStates) {
|
|
const VarTestResult <est = PInfo.getLTest(),
|
|
&RTest = PInfo.getRTest();
|
|
|
|
ConsumedState LState = LTest.Var ? ThenStates->getState(LTest.Var) : CS_None,
|
|
RState = RTest.Var ? ThenStates->getState(RTest.Var) : CS_None;
|
|
|
|
if (LTest.Var) {
|
|
if (PInfo.testEffectiveOp() == EO_And) {
|
|
if (LState == CS_Unknown) {
|
|
ThenStates->setState(LTest.Var, LTest.TestsFor);
|
|
} else if (LState == invertConsumedUnconsumed(LTest.TestsFor)) {
|
|
ThenStates->markUnreachable();
|
|
} else if (LState == LTest.TestsFor && isKnownState(RState)) {
|
|
if (RState == RTest.TestsFor)
|
|
ElseStates->markUnreachable();
|
|
else
|
|
ThenStates->markUnreachable();
|
|
}
|
|
} else {
|
|
if (LState == CS_Unknown) {
|
|
ElseStates->setState(LTest.Var,
|
|
invertConsumedUnconsumed(LTest.TestsFor));
|
|
} else if (LState == LTest.TestsFor) {
|
|
ElseStates->markUnreachable();
|
|
} else if (LState == invertConsumedUnconsumed(LTest.TestsFor) &&
|
|
isKnownState(RState)) {
|
|
if (RState == RTest.TestsFor)
|
|
ElseStates->markUnreachable();
|
|
else
|
|
ThenStates->markUnreachable();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (RTest.Var) {
|
|
if (PInfo.testEffectiveOp() == EO_And) {
|
|
if (RState == CS_Unknown)
|
|
ThenStates->setState(RTest.Var, RTest.TestsFor);
|
|
else if (RState == invertConsumedUnconsumed(RTest.TestsFor))
|
|
ThenStates->markUnreachable();
|
|
} else {
|
|
if (RState == CS_Unknown)
|
|
ElseStates->setState(RTest.Var,
|
|
invertConsumedUnconsumed(RTest.TestsFor));
|
|
else if (RState == RTest.TestsFor)
|
|
ElseStates->markUnreachable();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool ConsumedBlockInfo::allBackEdgesVisited(const CFGBlock *CurrBlock,
|
|
const CFGBlock *TargetBlock) {
|
|
assert(CurrBlock && "Block pointer must not be NULL");
|
|
assert(TargetBlock && "TargetBlock pointer must not be NULL");
|
|
|
|
unsigned int CurrBlockOrder = VisitOrder[CurrBlock->getBlockID()];
|
|
for (CFGBlock::const_pred_iterator PI = TargetBlock->pred_begin(),
|
|
PE = TargetBlock->pred_end(); PI != PE; ++PI) {
|
|
if (*PI && CurrBlockOrder < VisitOrder[(*PI)->getBlockID()] )
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void ConsumedBlockInfo::addInfo(
|
|
const CFGBlock *Block, ConsumedStateMap *StateMap,
|
|
std::unique_ptr<ConsumedStateMap> &OwnedStateMap) {
|
|
assert(Block && "Block pointer must not be NULL");
|
|
|
|
auto &Entry = StateMapsArray[Block->getBlockID()];
|
|
|
|
if (Entry) {
|
|
Entry->intersect(*StateMap);
|
|
} else if (OwnedStateMap)
|
|
Entry = std::move(OwnedStateMap);
|
|
else
|
|
Entry = std::make_unique<ConsumedStateMap>(*StateMap);
|
|
}
|
|
|
|
void ConsumedBlockInfo::addInfo(const CFGBlock *Block,
|
|
std::unique_ptr<ConsumedStateMap> StateMap) {
|
|
assert(Block && "Block pointer must not be NULL");
|
|
|
|
auto &Entry = StateMapsArray[Block->getBlockID()];
|
|
|
|
if (Entry) {
|
|
Entry->intersect(*StateMap);
|
|
} else {
|
|
Entry = std::move(StateMap);
|
|
}
|
|
}
|
|
|
|
ConsumedStateMap* ConsumedBlockInfo::borrowInfo(const CFGBlock *Block) {
|
|
assert(Block && "Block pointer must not be NULL");
|
|
assert(StateMapsArray[Block->getBlockID()] && "Block has no block info");
|
|
|
|
return StateMapsArray[Block->getBlockID()].get();
|
|
}
|
|
|
|
void ConsumedBlockInfo::discardInfo(const CFGBlock *Block) {
|
|
StateMapsArray[Block->getBlockID()] = nullptr;
|
|
}
|
|
|
|
std::unique_ptr<ConsumedStateMap>
|
|
ConsumedBlockInfo::getInfo(const CFGBlock *Block) {
|
|
assert(Block && "Block pointer must not be NULL");
|
|
|
|
auto &Entry = StateMapsArray[Block->getBlockID()];
|
|
return isBackEdgeTarget(Block) ? std::make_unique<ConsumedStateMap>(*Entry)
|
|
: std::move(Entry);
|
|
}
|
|
|
|
bool ConsumedBlockInfo::isBackEdge(const CFGBlock *From, const CFGBlock *To) {
|
|
assert(From && "From block must not be NULL");
|
|
assert(To && "From block must not be NULL");
|
|
|
|
return VisitOrder[From->getBlockID()] > VisitOrder[To->getBlockID()];
|
|
}
|
|
|
|
bool ConsumedBlockInfo::isBackEdgeTarget(const CFGBlock *Block) {
|
|
assert(Block && "Block pointer must not be NULL");
|
|
|
|
// Anything with less than two predecessors can't be the target of a back
|
|
// edge.
|
|
if (Block->pred_size() < 2)
|
|
return false;
|
|
|
|
unsigned int BlockVisitOrder = VisitOrder[Block->getBlockID()];
|
|
for (CFGBlock::const_pred_iterator PI = Block->pred_begin(),
|
|
PE = Block->pred_end(); PI != PE; ++PI) {
|
|
if (*PI && BlockVisitOrder < VisitOrder[(*PI)->getBlockID()])
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void ConsumedStateMap::checkParamsForReturnTypestate(SourceLocation BlameLoc,
|
|
ConsumedWarningsHandlerBase &WarningsHandler) const {
|
|
|
|
for (const auto &DM : VarMap) {
|
|
if (isa<ParmVarDecl>(DM.first)) {
|
|
const auto *Param = cast<ParmVarDecl>(DM.first);
|
|
const ReturnTypestateAttr *RTA = Param->getAttr<ReturnTypestateAttr>();
|
|
|
|
if (!RTA)
|
|
continue;
|
|
|
|
ConsumedState ExpectedState = mapReturnTypestateAttrState(RTA);
|
|
if (DM.second != ExpectedState)
|
|
WarningsHandler.warnParamReturnTypestateMismatch(BlameLoc,
|
|
Param->getNameAsString(), stateToString(ExpectedState),
|
|
stateToString(DM.second));
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConsumedStateMap::clearTemporaries() {
|
|
TmpMap.clear();
|
|
}
|
|
|
|
ConsumedState ConsumedStateMap::getState(const VarDecl *Var) const {
|
|
VarMapType::const_iterator Entry = VarMap.find(Var);
|
|
|
|
if (Entry != VarMap.end())
|
|
return Entry->second;
|
|
|
|
return CS_None;
|
|
}
|
|
|
|
ConsumedState
|
|
ConsumedStateMap::getState(const CXXBindTemporaryExpr *Tmp) const {
|
|
TmpMapType::const_iterator Entry = TmpMap.find(Tmp);
|
|
|
|
if (Entry != TmpMap.end())
|
|
return Entry->second;
|
|
|
|
return CS_None;
|
|
}
|
|
|
|
void ConsumedStateMap::intersect(const ConsumedStateMap &Other) {
|
|
ConsumedState LocalState;
|
|
|
|
if (this->From && this->From == Other.From && !Other.Reachable) {
|
|
this->markUnreachable();
|
|
return;
|
|
}
|
|
|
|
for (const auto &DM : Other.VarMap) {
|
|
LocalState = this->getState(DM.first);
|
|
|
|
if (LocalState == CS_None)
|
|
continue;
|
|
|
|
if (LocalState != DM.second)
|
|
VarMap[DM.first] = CS_Unknown;
|
|
}
|
|
}
|
|
|
|
void ConsumedStateMap::intersectAtLoopHead(const CFGBlock *LoopHead,
|
|
const CFGBlock *LoopBack, const ConsumedStateMap *LoopBackStates,
|
|
ConsumedWarningsHandlerBase &WarningsHandler) {
|
|
|
|
ConsumedState LocalState;
|
|
SourceLocation BlameLoc = getLastStmtLoc(LoopBack);
|
|
|
|
for (const auto &DM : LoopBackStates->VarMap) {
|
|
LocalState = this->getState(DM.first);
|
|
|
|
if (LocalState == CS_None)
|
|
continue;
|
|
|
|
if (LocalState != DM.second) {
|
|
VarMap[DM.first] = CS_Unknown;
|
|
WarningsHandler.warnLoopStateMismatch(BlameLoc,
|
|
DM.first->getNameAsString());
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConsumedStateMap::markUnreachable() {
|
|
this->Reachable = false;
|
|
VarMap.clear();
|
|
TmpMap.clear();
|
|
}
|
|
|
|
void ConsumedStateMap::setState(const VarDecl *Var, ConsumedState State) {
|
|
VarMap[Var] = State;
|
|
}
|
|
|
|
void ConsumedStateMap::setState(const CXXBindTemporaryExpr *Tmp,
|
|
ConsumedState State) {
|
|
TmpMap[Tmp] = State;
|
|
}
|
|
|
|
void ConsumedStateMap::remove(const CXXBindTemporaryExpr *Tmp) {
|
|
TmpMap.erase(Tmp);
|
|
}
|
|
|
|
bool ConsumedStateMap::operator!=(const ConsumedStateMap *Other) const {
|
|
for (const auto &DM : Other->VarMap)
|
|
if (this->getState(DM.first) != DM.second)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
void ConsumedAnalyzer::determineExpectedReturnState(AnalysisDeclContext &AC,
|
|
const FunctionDecl *D) {
|
|
QualType ReturnType;
|
|
if (const auto *Constructor = dyn_cast<CXXConstructorDecl>(D)) {
|
|
ReturnType = Constructor->getThisType()->getPointeeType();
|
|
} else
|
|
ReturnType = D->getCallResultType();
|
|
|
|
if (const ReturnTypestateAttr *RTSAttr = D->getAttr<ReturnTypestateAttr>()) {
|
|
const CXXRecordDecl *RD = ReturnType->getAsCXXRecordDecl();
|
|
if (!RD || !RD->hasAttr<ConsumableAttr>()) {
|
|
// FIXME: This should be removed when template instantiation propagates
|
|
// attributes at template specialization definition, not
|
|
// declaration. When it is removed the test needs to be enabled
|
|
// in SemaDeclAttr.cpp.
|
|
WarningsHandler.warnReturnTypestateForUnconsumableType(
|
|
RTSAttr->getLocation(), ReturnType.getAsString());
|
|
ExpectedReturnState = CS_None;
|
|
} else
|
|
ExpectedReturnState = mapReturnTypestateAttrState(RTSAttr);
|
|
} else if (isConsumableType(ReturnType)) {
|
|
if (isAutoCastType(ReturnType)) // We can auto-cast the state to the
|
|
ExpectedReturnState = CS_None; // expected state.
|
|
else
|
|
ExpectedReturnState = mapConsumableAttrState(ReturnType);
|
|
}
|
|
else
|
|
ExpectedReturnState = CS_None;
|
|
}
|
|
|
|
bool ConsumedAnalyzer::splitState(const CFGBlock *CurrBlock,
|
|
const ConsumedStmtVisitor &Visitor) {
|
|
std::unique_ptr<ConsumedStateMap> FalseStates(
|
|
new ConsumedStateMap(*CurrStates));
|
|
PropagationInfo PInfo;
|
|
|
|
if (const auto *IfNode =
|
|
dyn_cast_or_null<IfStmt>(CurrBlock->getTerminator().getStmt())) {
|
|
const Expr *Cond = IfNode->getCond();
|
|
|
|
PInfo = Visitor.getInfo(Cond);
|
|
if (!PInfo.isValid() && isa<BinaryOperator>(Cond))
|
|
PInfo = Visitor.getInfo(cast<BinaryOperator>(Cond)->getRHS());
|
|
|
|
if (PInfo.isVarTest()) {
|
|
CurrStates->setSource(Cond);
|
|
FalseStates->setSource(Cond);
|
|
splitVarStateForIf(IfNode, PInfo.getVarTest(), CurrStates.get(),
|
|
FalseStates.get());
|
|
} else if (PInfo.isBinTest()) {
|
|
CurrStates->setSource(PInfo.testSourceNode());
|
|
FalseStates->setSource(PInfo.testSourceNode());
|
|
splitVarStateForIfBinOp(PInfo, CurrStates.get(), FalseStates.get());
|
|
} else {
|
|
return false;
|
|
}
|
|
} else if (const auto *BinOp =
|
|
dyn_cast_or_null<BinaryOperator>(CurrBlock->getTerminator().getStmt())) {
|
|
PInfo = Visitor.getInfo(BinOp->getLHS());
|
|
if (!PInfo.isVarTest()) {
|
|
if ((BinOp = dyn_cast_or_null<BinaryOperator>(BinOp->getLHS()))) {
|
|
PInfo = Visitor.getInfo(BinOp->getRHS());
|
|
|
|
if (!PInfo.isVarTest())
|
|
return false;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
CurrStates->setSource(BinOp);
|
|
FalseStates->setSource(BinOp);
|
|
|
|
const VarTestResult &Test = PInfo.getVarTest();
|
|
ConsumedState VarState = CurrStates->getState(Test.Var);
|
|
|
|
if (BinOp->getOpcode() == BO_LAnd) {
|
|
if (VarState == CS_Unknown)
|
|
CurrStates->setState(Test.Var, Test.TestsFor);
|
|
else if (VarState == invertConsumedUnconsumed(Test.TestsFor))
|
|
CurrStates->markUnreachable();
|
|
|
|
} else if (BinOp->getOpcode() == BO_LOr) {
|
|
if (VarState == CS_Unknown)
|
|
FalseStates->setState(Test.Var,
|
|
invertConsumedUnconsumed(Test.TestsFor));
|
|
else if (VarState == Test.TestsFor)
|
|
FalseStates->markUnreachable();
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
CFGBlock::const_succ_iterator SI = CurrBlock->succ_begin();
|
|
|
|
if (*SI)
|
|
BlockInfo.addInfo(*SI, std::move(CurrStates));
|
|
else
|
|
CurrStates = nullptr;
|
|
|
|
if (*++SI)
|
|
BlockInfo.addInfo(*SI, std::move(FalseStates));
|
|
|
|
return true;
|
|
}
|
|
|
|
void ConsumedAnalyzer::run(AnalysisDeclContext &AC) {
|
|
const auto *D = dyn_cast_or_null<FunctionDecl>(AC.getDecl());
|
|
if (!D)
|
|
return;
|
|
|
|
CFG *CFGraph = AC.getCFG();
|
|
if (!CFGraph)
|
|
return;
|
|
|
|
determineExpectedReturnState(AC, D);
|
|
|
|
PostOrderCFGView *SortedGraph = AC.getAnalysis<PostOrderCFGView>();
|
|
// AC.getCFG()->viewCFG(LangOptions());
|
|
|
|
BlockInfo = ConsumedBlockInfo(CFGraph->getNumBlockIDs(), SortedGraph);
|
|
|
|
CurrStates = std::make_unique<ConsumedStateMap>();
|
|
ConsumedStmtVisitor Visitor(*this, CurrStates.get());
|
|
|
|
// Add all trackable parameters to the state map.
|
|
for (const auto *PI : D->parameters())
|
|
Visitor.VisitParmVarDecl(PI);
|
|
|
|
// Visit all of the function's basic blocks.
|
|
for (const auto *CurrBlock : *SortedGraph) {
|
|
if (!CurrStates)
|
|
CurrStates = BlockInfo.getInfo(CurrBlock);
|
|
|
|
if (!CurrStates) {
|
|
continue;
|
|
} else if (!CurrStates->isReachable()) {
|
|
CurrStates = nullptr;
|
|
continue;
|
|
}
|
|
|
|
Visitor.reset(CurrStates.get());
|
|
|
|
// Visit all of the basic block's statements.
|
|
for (const auto &B : *CurrBlock) {
|
|
switch (B.getKind()) {
|
|
case CFGElement::Statement:
|
|
Visitor.Visit(B.castAs<CFGStmt>().getStmt());
|
|
break;
|
|
|
|
case CFGElement::TemporaryDtor: {
|
|
const CFGTemporaryDtor &DTor = B.castAs<CFGTemporaryDtor>();
|
|
const CXXBindTemporaryExpr *BTE = DTor.getBindTemporaryExpr();
|
|
|
|
Visitor.checkCallability(PropagationInfo(BTE),
|
|
DTor.getDestructorDecl(AC.getASTContext()),
|
|
BTE->getExprLoc());
|
|
CurrStates->remove(BTE);
|
|
break;
|
|
}
|
|
|
|
case CFGElement::AutomaticObjectDtor: {
|
|
const CFGAutomaticObjDtor &DTor = B.castAs<CFGAutomaticObjDtor>();
|
|
SourceLocation Loc = DTor.getTriggerStmt()->getEndLoc();
|
|
const VarDecl *Var = DTor.getVarDecl();
|
|
|
|
Visitor.checkCallability(PropagationInfo(Var),
|
|
DTor.getDestructorDecl(AC.getASTContext()),
|
|
Loc);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// TODO: Handle other forms of branching with precision, including while-
|
|
// and for-loops. (Deferred)
|
|
if (!splitState(CurrBlock, Visitor)) {
|
|
CurrStates->setSource(nullptr);
|
|
|
|
if (CurrBlock->succ_size() > 1 ||
|
|
(CurrBlock->succ_size() == 1 &&
|
|
(*CurrBlock->succ_begin())->pred_size() > 1)) {
|
|
|
|
auto *RawState = CurrStates.get();
|
|
|
|
for (CFGBlock::const_succ_iterator SI = CurrBlock->succ_begin(),
|
|
SE = CurrBlock->succ_end(); SI != SE; ++SI) {
|
|
if (*SI == nullptr) continue;
|
|
|
|
if (BlockInfo.isBackEdge(CurrBlock, *SI)) {
|
|
BlockInfo.borrowInfo(*SI)->intersectAtLoopHead(
|
|
*SI, CurrBlock, RawState, WarningsHandler);
|
|
|
|
if (BlockInfo.allBackEdgesVisited(CurrBlock, *SI))
|
|
BlockInfo.discardInfo(*SI);
|
|
} else {
|
|
BlockInfo.addInfo(*SI, RawState, CurrStates);
|
|
}
|
|
}
|
|
|
|
CurrStates = nullptr;
|
|
}
|
|
}
|
|
|
|
if (CurrBlock == &AC.getCFG()->getExit() &&
|
|
D->getCallResultType()->isVoidType())
|
|
CurrStates->checkParamsForReturnTypestate(D->getLocation(),
|
|
WarningsHandler);
|
|
} // End of block iterator.
|
|
|
|
// Delete the last existing state map.
|
|
CurrStates = nullptr;
|
|
|
|
WarningsHandler.emitDiagnostics();
|
|
}
|