implement a check for the virtual calls inside constructors, close #423

This commit is contained in:
Basile Burg 2017-04-16 03:20:17 +02:00
parent 3c3fbe6725
commit b615d3e299
No known key found for this signature in database
GPG Key ID: 1868039F415CB8CF
4 changed files with 397 additions and 0 deletions

View File

@ -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

View File

@ -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;
}

View File

@ -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,

View File

@ -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");
}