diff --git a/README.md b/README.md index 4438442..32cdcc5 100644 --- a/README.md +++ b/README.md @@ -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 diff --git a/src/analysis/config.d b/src/analysis/config.d index fd1c9f2..9478245 100644 --- a/src/analysis/config.d +++ b/src/analysis/config.d @@ -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; } diff --git a/src/analysis/run.d b/src/analysis/run.d index 2c8b0b7..7a8c491 100644 --- a/src/analysis/run.d +++ b/src/analysis/run.d @@ -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, diff --git a/src/analysis/vcall_in_ctor.d b/src/analysis/vcall_in_ctor.d new file mode 100644 index 0000000..794745d --- /dev/null +++ b/src/analysis/vcall_in_ctor.d @@ -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) + { + // ":" + 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"); +} +