Index: include/clang/Analysis/Analyses/Consumed.h =================================================================== --- /dev/null +++ include/clang/Analysis/Analyses/Consumed.h @@ -0,0 +1,193 @@ +//===- Consumed.h ----------------------------------------------*- C++ --*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// A intra-procedural analysis for checking consumed properties. This is based, +// in part, on research on linear types. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CLANG_UNIQUENESS_H +#define LLVM_CLANG_UNIQUENESS_H + +#include "clang/AST/DeclCXX.h" +#include "clang/AST/ExprCXX.h" +#include "clang/AST/StmtCXX.h" +#include "clang/Analysis/AnalysisContext.h" +#include "clang/Basic/SourceLocation.h" +#include "clang/Sema/Sema.h" +#include "llvm/ADT/StringRef.h" + +namespace clang { +namespace consumed { + + typedef SmallVector OptionalNotes; + typedef std::pair DelayedDiag; + typedef std::list DiagList; + + enum ConsumedState { + // No state information for the given variable. + CS_None, + + CS_Unknown, + CS_Unconsumed, + CS_Consumed + }; + + class ConsumedStateMap { + + typedef llvm::DenseMap MapType; + typedef std::pair PairType; + + protected: + + MapType Map; + + public: + /// \brief Get the consumed state of a given variable. + ConsumedState getState(const VarDecl *Var); + + /// \brief Merge this state map with another map. + void intersect(const ConsumedStateMap *Other); + + /// \brief Mark all variables as unknown. + void makeUnknown(); + + /// \brief Set the consumed state of a given variable. + void setState(const VarDecl *Var, ConsumedState State); + }; + + class ConsumedBlockInfo { + + typedef llvm::DenseMap MapType; + typedef std::pair PairType; + + typedef llvm::DenseSet SetType; + + MapType Map; + SetType VisitedBlocks; + + public: + + bool addInfo(const CFGBlock *Block, ConsumedStateMap *StateMap, + bool AlreadyOwned); + + ConsumedStateMap* getInfo(const CFGBlock *Block); + + void markVisited(const CFGBlock *Block); + }; + + struct VarTestResult { + const VarDecl *Var; + SourceLocation Loc; + bool UnconsumedInTrueBranch; + + VarTestResult() : Var(NULL), Loc(), UnconsumedInTrueBranch(true) {} + + VarTestResult(const VarDecl *Var, SourceLocation Loc, + bool UnconsumedInTrueBranch) + : Var(Var), Loc(Loc), UnconsumedInTrueBranch(UnconsumedInTrueBranch) {} + }; + + /// A class that handles the analysis of uniqueness violations. + class ConsumedAnalyzer { + + typedef llvm::DenseMap CacheMapType; + typedef std::pair CachePairType; + + Sema &S; + + DiagList Warnings; + SourceLocation FL, FEL; + + ConsumedBlockInfo BlockInfo; + ConsumedStateMap *CurrStates; + + CacheMapType ConsumableTypeCache; + + /// \brief Emit the warnings and notes left by the analysis. + void emitDiagnostics(); + bool hasConsumableAttributes(const CXXRecordDecl *RD); + void splitState(const CFGBlock *CurrBlock, const IfStmt *Terminator); + + public: + + ConsumedAnalyzer(Sema &S) : S(S) {} + + /// \brief Get a constant reference to the Sema object. + const Sema & getSema(); + + /// \brief Check to see if the type is a consumable type. + bool isConsumableType(QualType Type); + + /// \brief Check a function's CFG for consumed violations. + /// + /// We traverse the blocks in the CFG, keeping track of the state of each + /// value who's type has uniquness annotations. If methods are invoked in + /// the wrong state a warning is issued. Each block in the CFG is traversed + /// exactly once. + void run(AnalysisDeclContext &AC); + + /// Warn about unnecessary-test errors. + /// \param VariableName -- The name of the variable that holds the unique + /// value. + /// + /// \param Loc -- The SourceLocation of the unnecessary test. + void warnUnnecessaryTest(StringRef VariableName, StringRef VariableState, + SourceLocation Loc); + + /// Warn about use-while-consumed errors. + /// \param MethodName -- The name of the method that was incorrectly + /// invoked. + /// + /// \param VariableName -- The name of the variable that holds the unique + /// value. + /// + /// \param Loc -- The SourceLocation of the method invocation. + void warnUseOfTempWhileConsumed(StringRef MethodName, SourceLocation Loc); + + /// Warn about use-in-unknown-state errors. + /// \param MethodName -- The name of the method that was incorrectly + /// invoked. + /// + /// \param VariableName -- The name of the variable that holds the unique + /// value. + /// + /// \param Loc -- The SourceLocation of the method invocation. + void warnUseOfTempInUnknownState(StringRef MethodName, SourceLocation Loc); + + /// Warn about use-while-consumed errors. + /// \param MethodName -- The name of the method that was incorrectly + /// invoked. + /// + /// \param VariableName -- The name of the variable that holds the unique + /// value. + /// + /// \param Loc -- The SourceLocation of the method invocation. + void warnUseWhileConsumed(StringRef MethodName, StringRef VariableName, + SourceLocation Loc); + + /// Warn about use-in-unknown-state errors. + /// \param MethodName -- The name of the method that was incorrectly + /// invoked. + /// + /// \param VariableName -- The name of the variable that holds the unique + /// value. + /// + /// \param Loc -- The SourceLocation of the method invocation. + void warnUseInUnknownState(StringRef MethodName, StringRef VariableName, + SourceLocation Loc); + }; + + unsigned checkEnabled(DiagnosticsEngine &D); + /// \brief Check to see if a function tests an object's validity. + bool isTestingFunction(const CXXMethodDecl *MethodDecl); + +}} // end namespace clang::consumed + +#endif Index: include/clang/Basic/Attr.td =================================================================== --- include/clang/Basic/Attr.td +++ include/clang/Basic/Attr.td @@ -918,6 +918,33 @@ let TemplateDependent = 1; } +// C/C++ consumed attributes. + +def CallableWhenUnconsumed : InheritableAttr { + let Spellings = [GNU<"callable_when_unconsumed">]; + let Subjects = [CXXMethod]; +} + +def TestsUnconsumed : InheritableAttr { + let Spellings = [GNU<"tests_unconsumed">]; + let Subjects = [CXXMethod]; +} + +def Consumes : InheritableAttr { + let Spellings = [GNU<"consumes">]; + let Subjects = [CXXMethod]; +} + +def CallableAlways : InheritableAttr { + let Spellings = [GNU<"callable_always">]; + let Subjects = [CXXMethod]; +} + +def TestsConsumed : InheritableAttr { + let Spellings = [GNU<"tests_consumed">]; + let Subjects = [CXXMethod]; +} + // Type safety attributes for `void *' pointers and type tags. def ArgumentWithTypeTag : InheritableAttr { Index: include/clang/Basic/DiagnosticGroups.td =================================================================== --- include/clang/Basic/DiagnosticGroups.td +++ include/clang/Basic/DiagnosticGroups.td @@ -476,6 +476,10 @@ ThreadSafetyPrecise]>; def ThreadSafetyBeta : DiagGroup<"thread-safety-beta">; +// Uniqueness Analysis warnings +def Consumed : DiagGroup<"consumed">; +def ConsumedStrict : DiagGroup<"consumed-strict", [Consumed]>; + // Note that putting warnings in -Wall will not disable them by default. If a // warning should be active _only_ when -Wall is passed in, mark it as // DefaultIgnore in addition to putting it here. Index: include/clang/Basic/DiagnosticSemaKinds.td =================================================================== --- include/clang/Basic/DiagnosticSemaKinds.td +++ include/clang/Basic/DiagnosticSemaKinds.td @@ -2178,6 +2178,28 @@ def warn_thread_safety_beta : Warning< "Thread safety beta warning.">, InGroup, DefaultIgnore; +// Consumed warnings +def warn_use_while_consumed : Warning< + "invocation of method '%0' on object '%1' while it is in the 'consumed' " + "state">, InGroup, DefaultIgnore; +def warn_use_of_temp_while_consumed : Warning< + "invocation of method '%0' on a temporary object while it is in the " + "'consumed' state">, InGroup, DefaultIgnore; +def warn_uniqueness_attribute_wrong_decl_type : Warning< + "%0 attribute only applies to methods">, + InGroup, DefaultIgnore; + +// ConsumedStrict warnings +def warn_use_in_unknown_state : Warning< + "invocation of method '%0' on object '%1' while it is in an unknown state">, + InGroup, DefaultIgnore; +def warn_use_of_temp_in_unknown_state : Warning< + "invocation of method '%0' on a temporary object while it is in an unknown " + "state">, InGroup, DefaultIgnore; +def warn_unnecessary_test : Warning< + "unnecessary test. Variable '%0' is known to be in the '%1' state">, + InGroup, DefaultIgnore; + def warn_impcast_vector_scalar : Warning< "implicit conversion turns vector to scalar: %0 to %1">, InGroup, DefaultIgnore; Index: include/clang/Sema/AnalysisBasedWarnings.h =================================================================== --- include/clang/Sema/AnalysisBasedWarnings.h +++ include/clang/Sema/AnalysisBasedWarnings.h @@ -38,6 +38,7 @@ unsigned enableCheckFallThrough : 1; unsigned enableCheckUnreachable : 1; unsigned enableThreadSafetyAnalysis : 1; + unsigned enableConsumedAnalysis : 1; public: Policy(); void disableCheckFallThrough() { enableCheckFallThrough = 0; } Index: lib/Analysis/CMakeLists.txt =================================================================== --- lib/Analysis/CMakeLists.txt +++ lib/Analysis/CMakeLists.txt @@ -6,6 +6,7 @@ CFGStmtMap.cpp CallGraph.cpp CocoaConventions.cpp + Consumed.cpp Dominators.cpp FormatString.cpp LiveVariables.cpp Index: lib/Analysis/Consumed.cpp =================================================================== --- /dev/null +++ lib/Analysis/Consumed.cpp @@ -0,0 +1,905 @@ +//===- Consumed.cpp --------------------------------------------*- C++ --*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// A intra-procedural analysis for checking consumed properties. This is based, +// in part, on research on linear types. +// +//===----------------------------------------------------------------------===// + +#include "clang/AST/ASTContext.h" +#include "clang/AST/Attr.h" +#include "clang/AST/DeclCXX.h" +#include "clang/AST/ExprCXX.h" +#include "clang/AST/RecursiveASTVisitor.h" +#include "clang/AST/StmtCXX.h" +#include "clang/AST/Type.h" +#include "clang/Analysis/Analyses/PostOrderCFGView.h" +#include "clang/Analysis/AnalysisContext.h" +#include "clang/Analysis/CFG.h" +#include "clang/Analysis/Analyses/Consumed.h" +#include "clang/Basic/OperatorKinds.h" +#include "clang/Basic/SourceLocation.h" +#include "clang/Basic/SourceManager.h" +#include "clang/Sema/SemaDiagnostic.h" +#include "llvm/ADT/DenseMap.h" +#include "llvm/ADT/SmallVector.h" +#include "llvm/Support/raw_ostream.h" + +using namespace clang; +using namespace consumed; + +// TODO: Mark variables as Consumed going into while- or for-loops only if they +// are referenced inside that block. (Deferred) +// TODO: Support CONSUME methods. (Deferred) +// TODO: Add a method(s) to identify which method calls perform what state +// transitions. (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) +// TODO: Test IsFalseVisitor with values in the unknown state. (Deferred) +// TODO: Look into combining IsFalseVisitor and TestedVarsVisitor. (Deferred) + +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"; + } +} + +namespace { +class ConsumedStmtVisitor + : public RecursiveASTVisitor { + + union PropagationUnion { + ConsumedState State; + VarDecl *Var; + }; + + class PropagationInfo { + PropagationUnion StateOrVar; + + public: + bool IsVar; + + PropagationInfo() : IsVar(false) { + StateOrVar.State = consumed::CS_None; + } + + PropagationInfo(ConsumedState State) : IsVar(false) { + StateOrVar.State = State; + } + + PropagationInfo(VarDecl *Var) : IsVar(true) { + StateOrVar.Var = Var; + } + + ConsumedState getState() { return StateOrVar.State; }; + + VarDecl * getVar() { return IsVar ? StateOrVar.Var : NULL; }; + }; + + typedef llvm::DenseMap MapType; + typedef std::pair PairType; + typedef MapType::iterator InfoEntry; + + ConsumedAnalyzer &Analyzer; + ConsumedStateMap *StateMap; + MapType PropagationMap; + + void forwardInfo(Stmt *From, Stmt *To); + bool isLikeMoveAssignment(const CXXMethodDecl *MethodDecl); + +public: + + bool TraverseStmt(Stmt *StmtNode); + + bool VisitBinaryOperator(BinaryOperator *BinOp); + bool VisitCallExpr(CallExpr *Call); + bool VisitCastExpr(CastExpr *Cast); + bool VisitCXXConstructExpr(CXXConstructExpr *Call); + bool VisitCXXMemberCallExpr(CXXMemberCallExpr *Call); + bool VisitCXXOperatorCallExpr(CXXOperatorCallExpr *Call); + bool VisitDeclRefExpr(DeclRefExpr *DeclRef); + bool VisitDeclStmt(DeclStmt *DelcS); + bool VisitMaterializeTemporaryExpr(MaterializeTemporaryExpr *Temp); + bool VisitMemberExpr(MemberExpr *MExpr); + bool VisitUnaryOperator(UnaryOperator *UOp); + bool VisitVarDecl(VarDecl *Var); + + ConsumedStmtVisitor(ConsumedAnalyzer &Analyzer, ConsumedStateMap *StateMap) : + Analyzer(Analyzer), StateMap(StateMap) {} + + void reset() { + PropagationMap.clear(); + } +}; + +void ConsumedStmtVisitor::forwardInfo(Stmt *From, Stmt *To) { + InfoEntry Entry = PropagationMap.find(From); + + if (Entry != PropagationMap.end()) { + PropagationMap.insert(PairType(To, PropagationInfo(Entry->second))); + } +} + +bool ConsumedStmtVisitor::isLikeMoveAssignment( + const CXXMethodDecl *MethodDecl) { + + return MethodDecl->isMoveAssignmentOperator() || + (MethodDecl->getNameAsString() == "operator=" && + MethodDecl->getNumParams() == 1 && + MethodDecl->getParamDecl(0)->getType()->isRValueReferenceType()); +} + +bool ConsumedStmtVisitor::VisitBinaryOperator(BinaryOperator *BinOp) { + switch (BinOp->getOpcode()) { + case BO_PtrMemD: + case BO_PtrMemI: + forwardInfo(BinOp->getLHS(), BinOp); + break; + + default: + break; + } + + return false; +} + +bool ConsumedStmtVisitor::TraverseStmt(Stmt *StmtNode) { + bool RetVal = RecursiveASTVisitor::TraverseStmt(StmtNode); + + for (Stmt::child_iterator CI = StmtNode->child_begin(), + CE = StmtNode->child_end(); CI != CE; ++CI) { + + PropagationMap.erase(*CI); + } + + return RetVal; +} + +bool ConsumedStmtVisitor::VisitCallExpr(CallExpr *Call) { + if (const FunctionDecl *FunDecl = + dyn_cast_or_null(Call->getDirectCallee())) { + + if (const CXXMethodDecl *MethodDecl = + dyn_cast_or_null(FunDecl)) { + + if (isLikeMoveAssignment(MethodDecl)) { + + InfoEntry LEntry = PropagationMap.find(Call->getArg(0)); + InfoEntry REntry = PropagationMap.find(Call->getArg(1)); + + PropagationInfo LPState, RPState; + + if (LEntry != PropagationMap.end() && + REntry != PropagationMap.end()) { + + LPState = LEntry->second; + RPState = REntry->second; + + if (LPState.IsVar && RPState.IsVar) { + StateMap->setState(LPState.getVar(), + StateMap->getState(RPState.getVar())); + + StateMap->setState(RPState.getVar(), consumed::CS_Consumed); + + PropagationMap.insert(PairType(Call, LPState)); + + } else if (LPState.IsVar && !RPState.IsVar) { + StateMap->setState(LPState.getVar(), RPState.getState()); + + PropagationMap.insert(PairType(Call, LPState)); + + } else if (!LPState.IsVar && RPState.IsVar) { + PropagationMap.insert(PairType(Call, + PropagationInfo(StateMap->getState(RPState.getVar())))); + + StateMap->setState(RPState.getVar(), consumed::CS_Consumed); + + } else { + PropagationMap.insert(PairType(Call, RPState)); + } + + } else if (LEntry != PropagationMap.end() && + REntry == PropagationMap.end()) { + + LPState = LEntry->second; + + if (LPState.IsVar) { + StateMap->setState(LPState.getVar(), consumed::CS_Unknown); + + PropagationMap.insert(PairType(Call, LPState)); + + } else { + PropagationMap.insert(PairType(Call, + PropagationInfo(consumed::CS_Unknown))); + } + + } else if (LEntry == PropagationMap.end() && + REntry != PropagationMap.end()) { + + RPState = REntry->second; + + if (RPState.IsVar) { + VarDecl *Var = RPState.getVar(); + + PropagationMap.insert(PairType(Call, + PropagationInfo(StateMap->getState(Var)))); + + StateMap->setState(Var, consumed::CS_Consumed); + + } else { + PropagationMap.insert(PairType(Call, RPState)); + } + } + + return false; + } + } + + // Special case for the std::move function. + // TODO: Make this more specific. (Deferred) + if (FunDecl->getNameAsString() == "move") { + InfoEntry Entry = PropagationMap.find(Call->getArg(0)); + + if (Entry != PropagationMap.end()) { + PropagationMap.insert(PairType(Call, Entry->second)); + } + + return false; + } + + unsigned Offset = !(Call->getNumArgs() == FunDecl->getNumParams()); + + for (unsigned Index = Offset; Index < Call->getNumArgs(); ++Index) { + QualType ParamType = FunDecl->getParamDecl(Index - Offset)->getType(); + + InfoEntry Entry = PropagationMap.find(Call->getArg(Index)); + + if (Entry == PropagationMap.end() || !Entry->second.IsVar) { + continue; + } + + PropagationInfo PState = Entry->second; + + if (ParamType->isRValueReferenceType() || + (ParamType->isLValueReferenceType() && + !cast(*ParamType).isSpelledAsLValue())) { + + StateMap->setState(PState.getVar(), consumed::CS_Consumed); + + } else if (!(ParamType.isConstQualified() || + ((ParamType->isReferenceType() || + ParamType->isPointerType()) && + ParamType->getPointeeType().isConstQualified()))) { + + StateMap->setState(PState.getVar(), consumed::CS_Unknown); + } + } + } + + return true; +} + +bool ConsumedStmtVisitor::VisitCastExpr(CastExpr *Cast) { + InfoEntry Entry = PropagationMap.find(Cast->getSubExpr()); + + if (Entry != PropagationMap.end()) { + PropagationMap.insert(PairType(Cast, Entry->second)); + } + + return false; +} + +bool ConsumedStmtVisitor::VisitCXXConstructExpr(CXXConstructExpr *Call) { + CXXConstructorDecl *Constructor = Call->getConstructor(); + + ASTContext &CurrContext = Analyzer.getSema().getASTContext(); + QualType ThisType = Constructor->getThisType(CurrContext)->getPointeeType(); + + if (Analyzer.isConsumableType(ThisType)) { + + if (Constructor->isDefaultConstructor()) { + + PropagationMap.insert(PairType(Call, + PropagationInfo(consumed::CS_Consumed))); + + } else if (Constructor->isMoveConstructor()) { + + PropagationInfo PState = + PropagationMap.find(Call->getArg(0))->second; + + if (PState.IsVar) { + VarDecl* Var = PState.getVar(); + + PropagationMap.insert(PairType(Call, + PropagationInfo(StateMap->getState(Var)))); + + StateMap->setState(Var, consumed::CS_Consumed); + + } else { + + PropagationMap.insert(PairType(Call, PState)); + } + + } else if (Constructor->isCopyConstructor()) { + MapType::iterator Entry = PropagationMap.find(Call->getArg(0)); + + if (Entry != PropagationMap.end()) { + PropagationMap.insert(PairType(Call, Entry->second)); + } + + } else { + + PropagationMap.insert(PairType(Call, + PropagationInfo(consumed::CS_Unconsumed))); + } + } + + return false; +} + +bool ConsumedStmtVisitor::VisitCXXMemberCallExpr(CXXMemberCallExpr *Call) { + if (!Call->getMethodDecl()->isConst()) { + InfoEntry Entry = PropagationMap.find(Call->getCallee()->IgnoreParens()); + + if (Entry != PropagationMap.end()) { + PropagationInfo PState = Entry->second; + + if (PState.IsVar) { + StateMap->setState(PState.getVar(), consumed::CS_Unknown); + } + } + } + + return false; +} + +bool ConsumedStmtVisitor::VisitCXXOperatorCallExpr(CXXOperatorCallExpr *Call) { + + if (const FunctionDecl *FunDecl = + dyn_cast_or_null(Call->getDirectCallee())) { + + InfoEntry Entry = PropagationMap.find(Call->getArg(0)); + + if (Entry != PropagationMap.end()) { + + PropagationInfo PState = Entry->second; + + // TODO: When we support CallableAlways (or CallableWhenConsumed) this + // will have to check for the different attributes and change the + // behavior bellow. (Deferred) + if (FunDecl->hasAttr()) { + if (PState.IsVar) { + VarDecl *Var = PState.getVar(); + + switch (StateMap->getState(Var)) { + case CS_Consumed: + Analyzer.warnUseWhileConsumed(FunDecl->getNameAsString(), + Var->getNameAsString(), Call->getExprLoc()); + break; + + case CS_Unknown: + Analyzer.warnUseInUnknownState(FunDecl->getNameAsString(), + Var->getNameAsString(), Call->getExprLoc()); + break; + + default: + break; + } + + } else { + switch (PState.getState()) { + case CS_Consumed: + Analyzer.warnUseOfTempWhileConsumed(FunDecl->getNameAsString(), + Call->getExprLoc()); + break; + + case CS_Unknown: + Analyzer.warnUseOfTempInUnknownState(FunDecl->getNameAsString(), + Call->getExprLoc()); + break; + + default: + break; + } + } + } + + // Handle non-constant member operators. + if (const CXXMethodDecl *MethodDecl = + dyn_cast_or_null(FunDecl)) { + + if (!MethodDecl->isConst() && PState.IsVar) { + StateMap->setState(PState.getVar(), consumed::CS_Unknown); + } + } + } + } + + return false; +} + +bool ConsumedStmtVisitor::VisitDeclRefExpr(DeclRefExpr *DeclRef) { + if (VarDecl *Var = dyn_cast_or_null(DeclRef->getDecl())) { + if (StateMap->getState(Var) != consumed::CS_None) { + PropagationMap.insert(PairType(DeclRef, PropagationInfo(Var))); + } + } + + return false; +} + +bool ConsumedStmtVisitor::VisitDeclStmt(DeclStmt *DeclS) { + for (DeclStmt::decl_iterator DI = DeclS->decl_begin(), DE = DeclS->decl_end(); + DI != DE; ++DI) { + + TraverseDecl(*DI); + } + + if (DeclS->isSingleDecl()) { + if (VarDecl *Var = dyn_cast_or_null(DeclS->getSingleDecl())) { + PropagationMap.insert(PairType(DeclS, PropagationInfo(Var))); + } + } + + return false; +} + +bool ConsumedStmtVisitor::VisitMaterializeTemporaryExpr( + MaterializeTemporaryExpr *Temp) { + + InfoEntry Entry = PropagationMap.find(Temp->GetTemporaryExpr()); + + if (Entry != PropagationMap.end()) { + PropagationMap.insert(PairType(Temp, Entry->second)); + } + + return false; +} + +bool ConsumedStmtVisitor::VisitMemberExpr(MemberExpr *MExpr) { + forwardInfo(MExpr->getBase(), MExpr); + + return false; +} + +bool ConsumedStmtVisitor::VisitUnaryOperator(UnaryOperator *UOp) { + + if (UOp->getOpcode() == UO_AddrOf) { + InfoEntry Entry = PropagationMap.find(UOp->getSubExpr()); + + if (Entry != PropagationMap.end()) { + PropagationMap.insert(PairType(UOp, Entry->second)); + } + } + + return false; +} + +bool ConsumedStmtVisitor::VisitVarDecl(VarDecl *Var) { + if (Analyzer.isConsumableType(Var->getType())) { + PropagationInfo PState = + PropagationMap.find(Var->getInit())->second; + + StateMap->setState(Var, PState.IsVar ? + StateMap->getState(PState.getVar()) : PState.getState()); + } + + return false; +} +} + +namespace { +struct SortDiagBySourceLocation { + SourceManager &SM; + SortDiagBySourceLocation(SourceManager &SM) : SM(SM) {} + + bool operator()(const DelayedDiag &Left, const DelayedDiag &Right); +}; + +bool SortDiagBySourceLocation::operator()(const DelayedDiag &Left, + const DelayedDiag &Right) { + + // Although this call will be slow, this is only called when outputting + // multiple warnings. + return SM.isBeforeInTranslationUnit(Left.first.first, Right.first.first); +} +} + +namespace { + +// TODO: Handle variable definitions, e.g. bool valid = x.isValid(); +// if (valid) ...; (Deferred) +class TestedVarsVisitor : public RecursiveASTVisitor { + + bool Invert; + SourceLocation CurrTestLoc; + + ConsumedStateMap *StateMap; + +public: + bool IsUsefulConditional; + VarTestResult Test; + + TestedVarsVisitor(ConsumedStateMap *StateMap) : Invert(false), + StateMap(StateMap), IsUsefulConditional(false) {} + + bool VisitCallExpr(CallExpr *Call); + bool VisitDeclRefExpr(DeclRefExpr *DeclRef); + bool VisitUnaryOperator(UnaryOperator *UnaryOp); +}; + +bool TestedVarsVisitor::VisitCallExpr(CallExpr *Call) { + if (const CXXMethodDecl *Method = + dyn_cast_or_null(Call->getDirectCallee())) { + + if (isTestingFunction(Method)) { + CurrTestLoc = Call->getExprLoc(); + IsUsefulConditional = true; + return true; + } + + IsUsefulConditional = false; + } + + return false; +} + +bool TestedVarsVisitor::VisitDeclRefExpr(DeclRefExpr *DeclRef) { + if (const VarDecl *Var = dyn_cast_or_null(DeclRef->getDecl())) { + if (StateMap->getState(Var) != consumed::CS_None) { + Test = VarTestResult(Var, CurrTestLoc, !Invert); + } + + } else { + IsUsefulConditional = false; + } + + return IsUsefulConditional; +} + +bool TestedVarsVisitor::VisitUnaryOperator(UnaryOperator *UnaryOp) { + if (UnaryOp->getOpcode() == UO_LNot) { + Invert = true; + TraverseStmt(UnaryOp->getSubExpr()); + + } else { + IsUsefulConditional = false; + } + + return false; +} +} + +namespace clang { +namespace consumed { + +bool ConsumedBlockInfo::addInfo(const CFGBlock *Block, + ConsumedStateMap *StateMap, + bool AlreadyOwned) { + + if (VisitedBlocks.find(Block) != VisitedBlocks.end()) { + return false; + } + + std::pair Entry = Map.insert(PairType(Block, NULL)); + + if (!Entry.second) { + Entry.first->second->intersect(StateMap); + + } else if (AlreadyOwned) { + Entry.first->second = new ConsumedStateMap(*StateMap); + + } else { + Entry.first->second = StateMap; + return true; + } + + return false; +} + +ConsumedStateMap* ConsumedBlockInfo::getInfo(const CFGBlock *Block) { + ConsumedStateMap *RetValue = Map.find(Block)->second; + Map.erase(Block); + return RetValue; +} + +void ConsumedBlockInfo::markVisited(const CFGBlock *Block) { + VisitedBlocks.insert(Block); +} + +ConsumedState ConsumedStateMap::getState(const VarDecl *Var) { + MapType::const_iterator Entry = Map.find(Var); + + if (Entry != Map.end()) { + return Entry->second; + + } else { + return CS_None; + } +} + +void ConsumedStateMap::intersect(const ConsumedStateMap *Other) { + ConsumedState LocalState; + + for (MapType::const_iterator DMI = Other->Map.begin(), + DME = Other->Map.end(); DMI != DME; ++DMI) { + + LocalState = this->getState(DMI->first); + + if (LocalState != CS_None && LocalState != DMI->second) + setState(DMI->first, CS_Unknown); + } +} + +void ConsumedStateMap::makeUnknown() { + PairType Pair; + + for (MapType::const_iterator DMI = Map.begin(), DME = Map.end(); DMI != DME; + ++DMI) { + + Pair = *DMI; + + Map.erase(Pair.first); + Map.insert(PairType(Pair.first, CS_Unknown)); + } +} + +void ConsumedStateMap::setState(const VarDecl *Var, ConsumedState State) { + Map[Var] = State; +} + +const Sema & ConsumedAnalyzer::getSema() { + return S; +} + + +bool ConsumedAnalyzer::isConsumableType(QualType Type) { + const CXXRecordDecl *RD = + dyn_cast_or_null(Type->getAsCXXRecordDecl()); + + if (!RD) return false; + + std::pair Entry = + ConsumableTypeCache.insert(std::make_pair(RD, false)); + + if (Entry.second) + Entry.first->second = hasConsumableAttributes(RD); + + return Entry.first->second; +} + +// TODO: Walk the base classes to see if any of them are unique types. +// (Deferred) +bool ConsumedAnalyzer::hasConsumableAttributes(const CXXRecordDecl *RD) { + for (CXXRecordDecl::method_iterator MI = RD->method_begin(), + ME = RD->method_end(); MI != ME; ++MI) { + + for (Decl::attr_iterator AI = (*MI)->attr_begin(), AE = (*MI)->attr_end(); + AI != AE; ++AI) { + + switch ((*AI)->getKind()) { + case attr::CallableWhenUnconsumed: + case attr::TestsUnconsumed: + return true; + + default: + break; + } + } + } + + return false; +} + +// TODO: Handle other forms of branching with precision, including while- and +// for-loops. (Deferred) +void ConsumedAnalyzer::splitState(const CFGBlock *CurrBlock, + const IfStmt *Terminator) { + + TestedVarsVisitor Visitor(CurrStates); + Visitor.TraverseStmt(const_cast(Terminator->getCond())); + + bool HasElse = Terminator->getElse() != NULL; + + ConsumedStateMap *ElseOrMergeStates = new ConsumedStateMap(*CurrStates); + + if (Visitor.IsUsefulConditional) { + ConsumedState VarState = CurrStates->getState(Visitor.Test.Var); + + if (VarState != CS_Unknown) { + // FIXME: Make this not warn if the test is from a macro expansion. + // (Deferred) + warnUnnecessaryTest(Visitor.Test.Var->getNameAsString(), + stateToString(VarState), Visitor.Test.Loc); + } + + if (Visitor.Test.UnconsumedInTrueBranch) { + CurrStates->setState(Visitor.Test.Var, CS_Unconsumed); + if (HasElse) ElseOrMergeStates->setState(Visitor.Test.Var, CS_Consumed); + + } else { + CurrStates->setState(Visitor.Test.Var, CS_Consumed); + if (HasElse) ElseOrMergeStates->setState(Visitor.Test.Var, CS_Unconsumed); + } + } + + CFGBlock::const_succ_iterator SI = CurrBlock->succ_begin(); + + BlockInfo.addInfo(*SI, CurrStates, false); + BlockInfo.addInfo(*++SI, ElseOrMergeStates, false); +} + +void ConsumedAnalyzer::run(AnalysisDeclContext &AC) { + const FunctionDecl *D = dyn_cast_or_null(AC.getDecl()); + + if (!D) return; + + FL = AC.getDecl()->getLocation(); + FEL = AC.getDecl()->getLocEnd(); + + PostOrderCFGView *SortedGraph = AC.getAnalysis(); + + CurrStates = new ConsumedStateMap(); + + // Visit all of the function's basic blocks. + for (PostOrderCFGView::iterator I = SortedGraph->begin(), + E = SortedGraph->end(); I != E; ++I) { + + const CFGBlock *CurrBlock = *I; + BlockInfo.markVisited(CurrBlock); + + if (CurrStates == NULL) { + CurrStates = BlockInfo.getInfo(CurrBlock); + } + + ConsumedStmtVisitor Visitor(*this, CurrStates); + + // Visit all of the basic block's statements. + for (CFGBlock::const_iterator BI = CurrBlock->begin(), + BE = CurrBlock->end(); BI != BE; ++BI) { + + if (BI->getKind() == CFGElement::Statement) { + Stmt *StmtNode = const_cast(BI->castAs().getStmt()); + + Visitor.TraverseStmt(StmtNode); + } + } + + // TODO: Remove any variables that have reached the end of their + // lifetimes from the state map. (Deferred) + + if (const IfStmt *Terminator = + dyn_cast_or_null(CurrBlock->getTerminator().getStmt())) { + + splitState(CurrBlock, Terminator); + CurrStates = NULL; + + } else if (CurrBlock->succ_size() > 1) { + CurrStates->makeUnknown(); + + bool OwnershipTaken = false; + + for (CFGBlock::const_succ_iterator SI = CurrBlock->succ_begin(), + SE = CurrBlock->succ_end(); SI != SE; ++SI) { + + OwnershipTaken = BlockInfo.addInfo(*SI, CurrStates, OwnershipTaken); + } + + if (!OwnershipTaken) delete CurrStates; + + CurrStates = NULL; + + } else if (CurrBlock->succ_size() == 1 && + (*CurrBlock->succ_begin())->pred_size() > 1) { + + if (!BlockInfo.addInfo(*CurrBlock->succ_begin(), CurrStates, false)) + delete CurrStates; + + CurrStates = NULL; + } + + Visitor.reset(); + } // End of block iterator. + + // Delete the last existing state map. + delete CurrStates; + + emitDiagnostics(); +} + +/// \brief Emit all buffered diagnostics in order of sourcelocation. +/// We need to output diagnostics produced while checking operations on +/// unique types in deterministic order, so this function orders diagnostics +/// and outputs them. +void ConsumedAnalyzer::emitDiagnostics() { + Warnings.sort(SortDiagBySourceLocation(S.getSourceManager())); + + for (DiagList::iterator I = Warnings.begin(), E = Warnings.end(); + I != E; ++I) { + + const OptionalNotes &Notes = I->second; + S.Diag(I->first.first, I->first.second); + + for (unsigned NoteI = 0, NoteN = Notes.size(); NoteI != NoteN; ++NoteI) { + S.Diag(Notes[NoteI].first, Notes[NoteI].second); + } + } +} + +void ConsumedAnalyzer::warnUnnecessaryTest(StringRef VariableName, + StringRef VariableState, + SourceLocation Loc) { + + PartialDiagnosticAt Warning(Loc, S.PDiag(diag::warn_unnecessary_test) << + VariableName << VariableState); + + Warnings.push_back(DelayedDiag(Warning, OptionalNotes())); +} + +void ConsumedAnalyzer::warnUseOfTempWhileConsumed(StringRef MethodName, + SourceLocation Loc) { + + PartialDiagnosticAt Warning(Loc, S.PDiag( + diag::warn_use_of_temp_while_consumed) << MethodName); + + Warnings.push_back(DelayedDiag(Warning, OptionalNotes())); +} + +void ConsumedAnalyzer::warnUseOfTempInUnknownState(StringRef MethodName, + SourceLocation Loc) { + + PartialDiagnosticAt Warning(Loc, S.PDiag( + diag::warn_use_of_temp_in_unknown_state) << MethodName); + + Warnings.push_back(DelayedDiag(Warning, OptionalNotes())); +} + +void ConsumedAnalyzer::warnUseWhileConsumed(StringRef MethodName, + StringRef VariableName, + SourceLocation Loc) { + + PartialDiagnosticAt Warning(Loc, S.PDiag(diag::warn_use_while_consumed) << + MethodName << VariableName); + + Warnings.push_back(DelayedDiag(Warning, OptionalNotes())); +} + +void ConsumedAnalyzer::warnUseInUnknownState(StringRef MethodName, + StringRef VariableName, + SourceLocation Loc) { + + PartialDiagnosticAt Warning(Loc, S.PDiag(diag::warn_use_in_unknown_state) << + MethodName << VariableName); + + Warnings.push_back(DelayedDiag(Warning, OptionalNotes())); +} + +unsigned checkEnabled(DiagnosticsEngine &D) { + return (unsigned) + (D.getDiagnosticLevel(diag::warn_use_while_consumed, SourceLocation()) != + DiagnosticsEngine::Ignored); +} + +bool isTestingFunction(const CXXMethodDecl *Method) { + return Method->hasAttr(); +} + +}} // end namespace clang::consumed Index: lib/Sema/AnalysisBasedWarnings.cpp =================================================================== --- lib/Sema/AnalysisBasedWarnings.cpp +++ lib/Sema/AnalysisBasedWarnings.cpp @@ -25,6 +25,7 @@ #include "clang/AST/StmtObjC.h" #include "clang/AST/StmtVisitor.h" #include "clang/Analysis/Analyses/CFGReachabilityAnalysis.h" +#include "clang/Analysis/Analyses/Consumed.h" #include "clang/Analysis/Analyses/ReachableCode.h" #include "clang/Analysis/Analyses/ThreadSafety.h" #include "clang/Analysis/Analyses/UninitializedValues.h" @@ -1420,6 +1421,7 @@ enableCheckFallThrough = 1; enableCheckUnreachable = 0; enableThreadSafetyAnalysis = 0; + enableConsumedAnalysis = 0; } clang::sema::AnalysisBasedWarnings::AnalysisBasedWarnings(Sema &s) @@ -1440,6 +1442,7 @@ DefaultPolicy.enableThreadSafetyAnalysis = (unsigned) (D.getDiagnosticLevel(diag::warn_double_lock, SourceLocation()) != DiagnosticsEngine::Ignored); + DefaultPolicy.enableConsumedAnalysis = consumed::checkEnabled(D); } @@ -1501,7 +1504,8 @@ // prototyping, but we need a way for analyses to say what expressions they // expect to always be CFGElements and then fill in the BuildOptions // appropriately. This is essentially a layering violation. - if (P.enableCheckUnreachable || P.enableThreadSafetyAnalysis) { + if (P.enableCheckUnreachable || P.enableThreadSafetyAnalysis || + P.enableConsumedAnalysis) { // Unreachable code analysis and thread safety require a linearized CFG. AC.getCFGBuildOptions().setAllAlwaysAdd(); } @@ -1605,6 +1609,12 @@ Reporter.emitDiagnostics(); } + // Check for violations of consumed properties. + if (P.enableConsumedAnalysis) { + consumed::ConsumedAnalyzer Analyzer(S); + Analyzer.run(AC); + } + if (Diags.getDiagnosticLevel(diag::warn_uninit_var, D->getLocStart()) != DiagnosticsEngine::Ignored || Diags.getDiagnosticLevel(diag::warn_sometimes_uninit_var,D->getLocStart()) Index: lib/Sema/SemaDeclAttr.cpp =================================================================== --- lib/Sema/SemaDeclAttr.cpp +++ lib/Sema/SemaDeclAttr.cpp @@ -997,6 +997,85 @@ Attr.getAttributeSpellingListIndex())); } +static void handleConsumesAttr(Sema &S, Decl *D, + const AttributeList &Attr) { + assert(!Attr.isInvalid()); + if (!checkAttributeNumArgs(S, Attr, 0)) return; + + if (!isa(D)) { + S.Diag(Attr.getLoc(), diag::warn_uniqueness_attribute_wrong_decl_type) << + Attr.getName(); + return; + } + + D->addAttr(::new (S.Context) + ConsumesAttr(Attr.getRange(), S.Context, + Attr.getAttributeSpellingListIndex())); +} + +static void handleCallableAlwaysAttr(Sema &S, Decl *D, + const AttributeList &Attr) { + assert(!Attr.isInvalid()); + if (!checkAttributeNumArgs(S, Attr, 0)) return; + + if (!isa(D)) { + S.Diag(Attr.getLoc(), diag::warn_uniqueness_attribute_wrong_decl_type) << + Attr.getName(); + return; + } + + D->addAttr(::new (S.Context) + CallableAlwaysAttr(Attr.getRange(), S.Context, + Attr.getAttributeSpellingListIndex())); +} + +static void handleCallableWhenUnconsumedAttr(Sema &S, Decl *D, + const AttributeList &Attr) { + assert(!Attr.isInvalid()); + if (!checkAttributeNumArgs(S, Attr, 0)) return; + + if (!isa(D)) { + S.Diag(Attr.getLoc(), diag::warn_uniqueness_attribute_wrong_decl_type) << + Attr.getName(); + return; + } + + D->addAttr(::new (S.Context) + CallableWhenUnconsumedAttr(Attr.getRange(), S.Context, + Attr.getAttributeSpellingListIndex())); +} + +static void handleTestsConsumedAttr(Sema &S, Decl *D, + const AttributeList &Attr) { + assert(!Attr.isInvalid()); + if (!checkAttributeNumArgs(S, Attr, 0)) return; + + if (!isa(D)) { + S.Diag(Attr.getLoc(), diag::warn_uniqueness_attribute_wrong_decl_type) << + Attr.getName(); + return; + } + + D->addAttr(::new (S.Context) + TestsConsumedAttr(Attr.getRange(), S.Context, + Attr.getAttributeSpellingListIndex())); +} + +static void handleTestsUnconsumedAttr(Sema &S, Decl *D, + const AttributeList &Attr) { + assert(!Attr.isInvalid()); + if (!checkAttributeNumArgs(S, Attr, 0)) return; + + if (!isa(D)) { + S.Diag(Attr.getLoc(), diag::warn_uniqueness_attribute_wrong_decl_type) << + Attr.getName(); + return; + } + + D->addAttr(::new (S.Context) + TestsUnconsumedAttr(Attr.getRange(), S.Context, + Attr.getAttributeSpellingListIndex())); +} static void handleExtVectorTypeAttr(Sema &S, Scope *scope, Decl *D, const AttributeList &Attr) { @@ -4963,6 +5042,23 @@ handleAcquiredAfterAttr(S, D, Attr); break; + // Uniqueness analysis attributes. + case AttributeList::AT_Consumes: + handleConsumesAttr(S, D, Attr); + break; + case AttributeList::AT_CallableAlways: + handleCallableAlwaysAttr(S, D, Attr); + break; + case AttributeList::AT_CallableWhenUnconsumed: + handleCallableWhenUnconsumedAttr(S, D, Attr); + break; + case AttributeList::AT_TestsConsumed: + handleTestsConsumedAttr(S, D, Attr); + break; + case AttributeList::AT_TestsUnconsumed: + handleTestsUnconsumedAttr(S, D, Attr); + break; + // Type safety attributes. case AttributeList::AT_ArgumentWithTypeTag: handleArgumentWithTypeTagAttr(S, D, Attr); Index: test/SemaCXX/warn-consumed-analysis-strict.cpp =================================================================== --- /dev/null +++ test/SemaCXX/warn-consumed-analysis-strict.cpp @@ -0,0 +1,120 @@ +// RUN: %clang_cc1 -fsyntax-only -verify -Wconsumed-strict -std=c++11 %s + +#define CALLABLE_WHEN_UNCONSUMED __attribute__ ((callable_when_unconsumed)) +#define TESTS_UNCONSUMED __attribute__ ((tests_unconsumed)) + +typedef decltype(nullptr) nullptr_t; + +template +class Bar { + T var; + + public: + Bar(void); + Bar(T val); + Bar(Bar &other); + Bar(Bar &&other); + + Bar& operator=(Bar &other); + Bar& operator=(Bar &&other); + Bar& operator=(nullptr_t); + + template + Bar& operator=(Bar &other); + + template + Bar& operator=(Bar &&other); + + void operator*() const CALLABLE_WHEN_UNCONSUMED; + + bool isValid() const TESTS_UNCONSUMED; + + void constCall() const; + void nonconstCall(); +}; + +void baf0(Bar &var); +void baf1(Bar *var); + +void testIfStmt() { + Bar var; + + if (var.isValid()) { // expected-warning {{unnecessary test. Variable 'var' is known to be in the 'consumed' state}} + + // Empty + + } else { + // Empty + } +} + +void testConditionalMerge() { + Bar var; + + if (var.isValid()) {// expected-warning {{unnecessary test. Variable 'var' is known to be in the 'consumed' state}} + + // Empty + } + + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} + + if (var.isValid()) { + // Empty + + } else { + // Empty + } + + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} +} + +void testCallingConventions() { + Bar var(42); + + baf0(var); + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} + + var = Bar(42); + baf1(&var); + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} +} + +void testMoveAsignmentish() { + Bar var; + + var = nullptr; + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} +} + +void testConstAndNonConstMemberFunctions() { + Bar var(42); + + var.constCall(); + *var; + + var.nonconstCall(); + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} +} + +void testSimpleForLoop() { + Bar var; + + for (int i = 0; i < 10; ++i) { + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} + } + + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} +} + +void testSimpleWhileLoop() { + int i = 0; + + Bar var; + + while (i < 10) { + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} + ++i; + } + + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in an unknown state}} +} Index: test/SemaCXX/warn-consumed-analysis.cpp =================================================================== --- /dev/null +++ test/SemaCXX/warn-consumed-analysis.cpp @@ -0,0 +1,160 @@ +// RUN: %clang_cc1 -fsyntax-only -verify -Wconsumed -std=c++11 %s + +#define CALLABLE_WHEN_UNCONSUMED __attribute__ ((callable_when_unconsumed)) +#define TESTS_UNCONSUMED __attribute__ ((tests_unconsumed)) + +typedef decltype(nullptr) nullptr_t; + +template +class Bar { + T var; + + public: + Bar(void); + Bar(T val); + Bar(Bar &other); + Bar(Bar &&other); + + Bar& operator=(Bar &other); + Bar& operator=(Bar &&other); + Bar& operator=(nullptr_t); + + template + Bar& operator=(Bar &other); + + template + Bar& operator=(Bar &&other); + + void operator*() const CALLABLE_WHEN_UNCONSUMED; + + bool isValid() const TESTS_UNCONSUMED; + + void constCall() const; + void nonconstCall(); +}; + +void baf0(const Bar var); +void baf1(const Bar &var); +void baf2(const Bar *var); + +void baf3(Bar &&var); + +void testInitialization() { + Bar var0; + Bar var1 = Bar(); + + var0 = Bar(); + + *var0; // expected-warning {{invocation of method 'operator*' on object 'var0' while it is in the 'consumed' state}} + *var1; // expected-warning {{invocation of method 'operator*' on object 'var1' while it is in the 'consumed' state}} + + if (var0.isValid()) { + *var0; + *var1; // expected-warning {{invocation of method 'operator*' on object 'var1' while it is in the 'consumed' state}} + + } else { + *var0; // expected-warning {{invocation of method 'operator*' on object 'var0' while it is in the 'consumed' state}} + } +} + +void testSimpleRValueRefs() { + Bar var0; + Bar var1(42); + + *var0; // expected-warning {{invocation of method 'operator*' on object 'var0' while it is in the 'consumed' state}} + *var1; + + var0 = static_cast&&>(var1); + + *var0; + *var1; // expected-warning {{invocation of method 'operator*' on object 'var1' while it is in the 'consumed' state}} +} + +void testIfStmt() { + Bar var; + + if (var.isValid()) { + // Empty + + } else { + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in the 'consumed' state}} + } + + if (!var.isValid()) { + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in the 'consumed' state}} + + } else { + *var; + } +} + +void testCallingConventions() { + Bar var(42); + + baf0(var); + *var; + + baf1(var); + *var; + + baf2(&var); + *var; + + baf3(static_cast&&>(var)); + *var; // expected-warning {{invocation of method 'operator*' on object 'var' while it is in the 'consumed' state}} +} + +void testMoveAsignmentish() { + Bar var0; + Bar var1(42); + + *var0; // expected-warning {{invocation of method 'operator*' on object 'var0' while it is in the 'consumed' state}} + *var1; + + var0 = static_cast&&>(var1); + + *var0; + *var1; // expected-warning {{invocation of method 'operator*' on object 'var1' while it is in the 'consumed' state}} +} + +void testConditionalMerge() { + Bar var; + + if (var.isValid()) { + // Empty + } + + *var; + + if (var.isValid()) { + // Empty + + } else { + // Empty + } + + *var; +} + +void testSimpleForLoop() { + Bar var; + + for (int i = 0; i < 10; ++i) { + *var; + } + + *var; +} + +void testSimpleWhileLoop() { + int i = 0; + + Bar var; + + while (i < 10) { + *var; + ++i; + } + + *var; +} Index: test/SemaCXX/warn-consumed-parsing.cpp =================================================================== --- /dev/null +++ test/SemaCXX/warn-consumed-parsing.cpp @@ -0,0 +1,46 @@ +// RUN: %clang_cc1 -fsyntax-only -verify -Wconsumed -std=c++11 %s + +#define CONSUMES __attribute__ ((consumes)) +#define TESTS_UNCONSUMED __attribute__ ((tests_unconsumed)) +#define CALLABLE_ALWAYS __attribute__ ((callable_always)) +#define CALLABLE_WHEN_UNCONSUMED __attribute__ ((callable_when_unconsumed)) + +class AttrTester0 { + void Consumes(void) __attribute__ ((consumes(42))); // \ + // expected-error {{attribute takes no arguments}} + bool TestsUnconsumed(void) __attribute__ ((tests_unconsumed(42))); // \ + // expected-error {{attribute takes no arguments}} + void CallableAlways(void) __attribute__ ((callable_always(42))); // \ + // expected-error {{attribute takes no arguments}} + void CallableWhenUnconsumed(void) + __attribute__ ((callable_when_unconsumed(42))); // \ + // expected-error {{attribute takes no arguments}} +}; + +int var0 CONSUMES; // \ +\\ expected-warning {{'consumes' attribute only applies to methods}} +int var1 TESTS_UNCONSUMED; // \ +\\ expected-warning {{'tests_unconsumed' attribute only applies to methods}} +int var2 CALLABLE_ALWAYS; // \ +\\ expected-warning {{'callable_always' attribute only applies to methods}} +int var3 CALLABLE_WHEN_UNCONSUMED; // \ +\\ expected-warning {{'callable_when_unconsumed' attribute only applies to methods}} + +void function0(void) CONSUMES; // \ +\\ expected-warning {{'consumes' attribute only applies to methods}} +void function0(void) TESTS_UNCONSUMED; // \ +\\ expected-warning {{'tests_unconsumed' attribute only applies to methods}} +void function0(void) CALLABLE_ALWAYS; // \ +\\ expected-warning {{'callable_always' attribute only applies to methods}} +void function0(void) CALLABLE_WHEN_UNCONSUMED; // \ +\\ expected-warning {{'callable_when_unconsumed' attribute only applies to methods}} + +class AttrTester1 { + void Consumes(void) CONSUMES; + bool TestsUnconsumed(void) TESTS_UNCONSUMED; + void CallableAlways(void) CALLABLE_ALWAYS; +}; + +class AttrTester2 { + void CallableWhenUnconsumed(void) CALLABLE_WHEN_UNCONSUMED; +};