implement a check for the virtual calls inside constructors, close #423
This commit is contained in:
parent
3c3fbe6725
commit
b615d3e299
|
@ -118,6 +118,7 @@ Note that the "--skipTests" option is the equivalent of changing each
|
|||
* Some assertions that check conditions that will always be true.
|
||||
* Auto functions without return statement. The compiler doesn't see an omission and it infers 'void' as return type.
|
||||
* `final` attribute is used but in this context it's a noop.
|
||||
* Virtual calls inside classes constructors.
|
||||
|
||||
#### Wishlist
|
||||
|
||||
|
|
|
@ -159,4 +159,7 @@ struct StaticAnalysisConfig
|
|||
|
||||
@INI("Check for useless usage of the final attribute")
|
||||
string final_attribute_check = Check.disabled;
|
||||
|
||||
@INI("Check for virtual calls in the class constructors")
|
||||
string vcall_in_ctor = Check.disabled;
|
||||
}
|
||||
|
|
|
@ -64,6 +64,7 @@ import analysis.imports_sortedness;
|
|||
import analysis.explicitly_annotated_unittests;
|
||||
import analysis.properly_documented_public_functions;
|
||||
import analysis.final_attribute;
|
||||
import analysis.vcall_in_ctor;
|
||||
|
||||
import dsymbol.string_interning : internString;
|
||||
import dsymbol.scope_;
|
||||
|
@ -379,6 +380,10 @@ MessageSet analyze(string fileName, const Module m, const StaticAnalysisConfig a
|
|||
checks ~= new FinalAttributeChecker(fileName,
|
||||
analysisConfig.final_attribute_check == Check.skipTests && !ut);
|
||||
|
||||
if (analysisConfig.vcall_in_ctor != Check.disabled)
|
||||
checks ~= new VcallCtorChecker(fileName,
|
||||
analysisConfig.vcall_in_ctor == Check.skipTests && !ut);
|
||||
|
||||
version (none)
|
||||
if (analysisConfig.redundant_if_check != Check.disabled)
|
||||
checks ~= new IfStatementCheck(fileName, moduleScope,
|
||||
|
|
|
@ -0,0 +1,388 @@
|
|||
// Copyright Basile Burg 2017.
|
||||
// Distributed under the Boost Software License, Version 1.0.
|
||||
// (See accompanying file LICENSE_1_0.txt or copy at
|
||||
//
|
||||
module analysis.vcall_in_ctor;
|
||||
|
||||
import analysis.base;
|
||||
import dparse.ast, dparse.lexer;
|
||||
import std.algorithm: among;
|
||||
import std.algorithm.iteration : filter;
|
||||
import std.algorithm.searching : find;
|
||||
import std.range.primitives : empty;
|
||||
import std.range: retro;
|
||||
|
||||
/**
|
||||
* Checks virtual calls from the constructor to methods defined in the same class.
|
||||
*
|
||||
* When not used carefully, virtual calls from constructors can lead to a call
|
||||
* in a derived instance that's not yet constructed.
|
||||
*/
|
||||
class VcallCtorChecker : BaseAnalyzer
|
||||
{
|
||||
alias visit = BaseAnalyzer.visit;
|
||||
|
||||
private:
|
||||
|
||||
enum string KEY = "dscanner.vcall_ctor";
|
||||
enum string MSG = "a virtual call inside a constructor may lead to"
|
||||
~ " unexpected results in the derived classes";
|
||||
|
||||
// what's called in the ctor
|
||||
Token[][] _ctorCalls;
|
||||
// the virtual method in the classes
|
||||
Token[][] _virtualMethods;
|
||||
|
||||
|
||||
// The problem only happens in classes
|
||||
bool[] _inClass = [false];
|
||||
// The problem only happens in __ctor
|
||||
bool[] _inCtor = [false];
|
||||
// The problem only happens with call to virtual methods
|
||||
bool[] _isVirtual = [true];
|
||||
// The problem only happens with call to virtual methods
|
||||
bool[] _isNestedFun = [false];
|
||||
// The problem only happens in derived classes that override
|
||||
bool[] _isFinal = [false];
|
||||
|
||||
void pushVirtual(bool value)
|
||||
{
|
||||
_isVirtual ~= value;
|
||||
}
|
||||
|
||||
void pushInClass(bool value)
|
||||
{
|
||||
_inClass ~= value;
|
||||
_ctorCalls.length += 1;
|
||||
_virtualMethods.length += 1;
|
||||
}
|
||||
|
||||
void pushInCtor(bool value)
|
||||
{
|
||||
_inCtor ~= value;
|
||||
}
|
||||
|
||||
void pushNestedFunc(bool value)
|
||||
{
|
||||
_isNestedFun ~= value;
|
||||
}
|
||||
|
||||
void pushIsFinal(bool value)
|
||||
{
|
||||
_isFinal ~= value;
|
||||
}
|
||||
|
||||
void popVirtual()
|
||||
{
|
||||
_isVirtual.length -= 1;
|
||||
}
|
||||
|
||||
void popInClass()
|
||||
{
|
||||
_inClass.length -= 1;
|
||||
_ctorCalls.length -= 1;
|
||||
_virtualMethods.length -= 1;
|
||||
}
|
||||
|
||||
void popInCtor()
|
||||
{
|
||||
_inCtor.length -= 1;
|
||||
}
|
||||
|
||||
void popNestedFunc()
|
||||
{
|
||||
_isNestedFun.length -= 1;
|
||||
}
|
||||
|
||||
void popIsFinal()
|
||||
{
|
||||
_isFinal.length -= 1;
|
||||
}
|
||||
|
||||
void overwriteVirtual(bool value)
|
||||
{
|
||||
_isVirtual[$-1] = value;
|
||||
}
|
||||
|
||||
bool isVirtual()
|
||||
{
|
||||
return _isVirtual[$-1];
|
||||
}
|
||||
|
||||
bool isInClass()
|
||||
{
|
||||
return _inClass[$-1];
|
||||
}
|
||||
|
||||
bool isInCtor()
|
||||
{
|
||||
return _inCtor[$-1];
|
||||
}
|
||||
|
||||
bool isFinal()
|
||||
{
|
||||
return _isFinal[$-1];
|
||||
}
|
||||
|
||||
bool isInNestedFunc()
|
||||
{
|
||||
return _isNestedFun[$-1];
|
||||
}
|
||||
|
||||
void check()
|
||||
{
|
||||
foreach (call; _ctorCalls[$-1])
|
||||
foreach (vm; _virtualMethods[$-1])
|
||||
{
|
||||
if (call == vm)
|
||||
addErrorMessage(call.line, call.column, KEY, MSG);
|
||||
}
|
||||
}
|
||||
|
||||
public:
|
||||
|
||||
///
|
||||
this(string fileName, bool skipTests = false)
|
||||
{
|
||||
super(fileName, null, skipTests);
|
||||
}
|
||||
|
||||
override void visit(const(ClassDeclaration) decl)
|
||||
{
|
||||
pushVirtual(true);
|
||||
pushInClass(true);
|
||||
pushNestedFunc(false);
|
||||
decl.accept(this);
|
||||
check();
|
||||
popVirtual();
|
||||
popInClass();
|
||||
popNestedFunc();
|
||||
}
|
||||
|
||||
override void visit(const(Constructor) ctor)
|
||||
{
|
||||
pushInCtor(isInClass);
|
||||
ctor.accept(this);
|
||||
popInCtor();
|
||||
}
|
||||
|
||||
override void visit(const(Declaration) d)
|
||||
{
|
||||
// "<protection>:"
|
||||
if (d.attributeDeclaration && d.attributeDeclaration.attribute)
|
||||
{
|
||||
const tp = d.attributeDeclaration.attribute.attribute.type;
|
||||
overwriteVirtual(isProtection(tp) & (tp != tok!"private"));
|
||||
}
|
||||
|
||||
// "protection {}"
|
||||
bool pop;
|
||||
scope(exit) if (pop)
|
||||
popVirtual;
|
||||
if (d.attributes) foreach (attr; d.attributes.retro)
|
||||
{
|
||||
if (attr.attribute == tok!"public" || attr.attribute == tok!"protected" ||
|
||||
attr.attribute == tok!"package")
|
||||
{
|
||||
pushVirtual(true);
|
||||
pop = true;
|
||||
break;
|
||||
}
|
||||
else if (attr.attribute == tok!"private")
|
||||
{
|
||||
pushVirtual(false);
|
||||
pop = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// final class... final function
|
||||
const bool pf = !d.attributes.find!(a => a.attribute.type == tok!"final").empty;
|
||||
if ((d.classDeclaration || d.functionDeclaration) && pf)
|
||||
pushIsFinal(true);
|
||||
|
||||
d.accept(this);
|
||||
|
||||
if ((d.classDeclaration || d.functionDeclaration) && pf)
|
||||
popIsFinal;
|
||||
}
|
||||
|
||||
override void visit(const(FunctionCallExpression) exp)
|
||||
{
|
||||
// nested function are not virtual
|
||||
pushNestedFunc(true);
|
||||
exp.accept(this);
|
||||
popNestedFunc();
|
||||
}
|
||||
|
||||
override void visit(const(UnaryExpression) exp)
|
||||
{
|
||||
// get function identifier for a call, only for this member (so no ident chain)
|
||||
if (isInCtor && exp.functionCallExpression &&
|
||||
exp.functionCallExpression.unaryExpression &&
|
||||
exp.functionCallExpression.unaryExpression.primaryExpression &&
|
||||
exp.functionCallExpression.unaryExpression.primaryExpression
|
||||
.identifierOrTemplateInstance)
|
||||
{
|
||||
const Token t = exp.functionCallExpression.unaryExpression
|
||||
.primaryExpression.identifierOrTemplateInstance.identifier;
|
||||
|
||||
if (t != tok!"")
|
||||
{
|
||||
_ctorCalls[$-1] ~= t;
|
||||
}
|
||||
}
|
||||
exp.accept(this);
|
||||
}
|
||||
|
||||
override void visit(const(FunctionDeclaration) d)
|
||||
{
|
||||
if (isInClass() && !isInNestedFunc() && !isFinal() && !d.templateParameters)
|
||||
{
|
||||
bool virtualOnce;
|
||||
bool notVirtualOnce;
|
||||
|
||||
// handle "private", "public"... for this declaration
|
||||
if (d.attributes) foreach (attr; d.attributes.retro)
|
||||
{
|
||||
if (attr.attribute == tok!"public" || attr.attribute == tok!"protected" ||
|
||||
attr.attribute == tok!"package")
|
||||
{
|
||||
if (!isVirtual)
|
||||
{
|
||||
virtualOnce = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
else if (attr.attribute == tok!"private")
|
||||
{
|
||||
if (isVirtual)
|
||||
{
|
||||
notVirtualOnce = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!isVirtual && virtualOnce)
|
||||
_virtualMethods[$-1] ~= d.name;
|
||||
else if (isVirtual && !virtualOnce)
|
||||
_virtualMethods[$-1] ~= d.name;
|
||||
|
||||
}
|
||||
d.accept(this);
|
||||
}
|
||||
}
|
||||
|
||||
unittest
|
||||
{
|
||||
import analysis.config : StaticAnalysisConfig, Check;
|
||||
import analysis.helpers : assertAnalyzerWarnings;
|
||||
import std.stdio : stderr;
|
||||
import std.format : format;
|
||||
|
||||
StaticAnalysisConfig sac;
|
||||
sac.vcall_in_ctor = Check.enabled;
|
||||
|
||||
// fails
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this(){foo();} // [warn]: %s
|
||||
private:
|
||||
public
|
||||
void foo(){}
|
||||
|
||||
}
|
||||
}c.format(VcallCtorChecker.MSG), sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this()
|
||||
{
|
||||
foo(); // [warn]: %s
|
||||
foo(); // [warn]: %s
|
||||
bar();
|
||||
}
|
||||
private: void bar();
|
||||
public{void foo(){}}
|
||||
}
|
||||
}c.format(VcallCtorChecker.MSG, VcallCtorChecker.MSG), sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this()
|
||||
{
|
||||
foo();
|
||||
bar(); // [warn]: %s
|
||||
}
|
||||
private: public void bar();
|
||||
public private {void foo(){}}
|
||||
}
|
||||
}c.format(VcallCtorChecker.MSG), sac);
|
||||
|
||||
// passes
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this(){foo();}
|
||||
private void foo(){}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this(){foo();}
|
||||
private {void foo(){}}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this(){foo();}
|
||||
private public protected private void foo(){}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this(){foo();}
|
||||
final private public protected void foo(){}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
this(){foo();}
|
||||
final void foo(){}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
final class Bar
|
||||
{
|
||||
public:
|
||||
this(){foo();}
|
||||
void foo(){}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
assertAnalyzerWarnings(q{
|
||||
class Bar
|
||||
{
|
||||
public:
|
||||
this(){foo();}
|
||||
void foo(T)(){}
|
||||
}
|
||||
}, sac);
|
||||
|
||||
import std.stdio: writeln;
|
||||
writeln("Unittest for VcallCtorChecker passed");
|
||||
}
|
||||
|
Loading…
Reference in New Issue