* Add a check for useless initializers, close #285 # Conflicts: # src/analysis/config.d * remove duplicated oror condition * remove useless protection * also handle several case of assignation to init * assign to init, forgot to test for ".init" * update with new configuration initialization
This commit is contained in:
parent
28945dec81
commit
6e9e841a9a
|
@ -122,6 +122,7 @@ Note that the "--skipTests" option is the equivalent of changing each
|
||||||
* Check for explicitly annotated unittests (_@system_ or _@safe_). Initially implemented to lint Phobos. By default disabled.
|
* Check for explicitly annotated unittests (_@system_ or _@safe_). Initially implemented to lint Phobos. By default disabled.
|
||||||
* Check for that imports are sorted. Initially implemented to lint Phobos. By default disabled.
|
* Check for that imports are sorted. Initially implemented to lint Phobos. By default disabled.
|
||||||
* Virtual calls inside classes constructors.
|
* Virtual calls inside classes constructors.
|
||||||
|
* Useless initializers.
|
||||||
|
|
||||||
#### Wishlist
|
#### Wishlist
|
||||||
|
|
||||||
|
|
|
@ -176,4 +176,7 @@ struct StaticAnalysisConfig
|
||||||
|
|
||||||
@INI("Check for virtual calls in the class constructors")
|
@INI("Check for virtual calls in the class constructors")
|
||||||
string vcall_in_ctor = Check.enabled;
|
string vcall_in_ctor = Check.enabled;
|
||||||
|
|
||||||
|
@INI("Check for useless user defined initializers")
|
||||||
|
string useless_initializer = Check.enabled;
|
||||||
}
|
}
|
||||||
|
|
|
@ -65,6 +65,7 @@ import analysis.explicitly_annotated_unittests;
|
||||||
import analysis.properly_documented_public_functions;
|
import analysis.properly_documented_public_functions;
|
||||||
import analysis.final_attribute;
|
import analysis.final_attribute;
|
||||||
import analysis.vcall_in_ctor;
|
import analysis.vcall_in_ctor;
|
||||||
|
import analysis.useless_initializer;
|
||||||
|
|
||||||
import dsymbol.string_interning : internString;
|
import dsymbol.string_interning : internString;
|
||||||
import dsymbol.scope_;
|
import dsymbol.scope_;
|
||||||
|
@ -384,6 +385,10 @@ MessageSet analyze(string fileName, const Module m, const StaticAnalysisConfig a
|
||||||
checks ~= new VcallCtorChecker(fileName,
|
checks ~= new VcallCtorChecker(fileName,
|
||||||
analysisConfig.vcall_in_ctor == Check.skipTests && !ut);
|
analysisConfig.vcall_in_ctor == Check.skipTests && !ut);
|
||||||
|
|
||||||
|
if (analysisConfig.useless_initializer != Check.disabled)
|
||||||
|
checks ~= new UselessInitializerChecker(fileName,
|
||||||
|
analysisConfig.useless_initializer == Check.skipTests && !ut);
|
||||||
|
|
||||||
version (none)
|
version (none)
|
||||||
if (analysisConfig.redundant_if_check != Check.disabled)
|
if (analysisConfig.redundant_if_check != Check.disabled)
|
||||||
checks ~= new IfStatementCheck(fileName, moduleScope,
|
checks ~= new IfStatementCheck(fileName, moduleScope,
|
||||||
|
|
|
@ -0,0 +1,228 @@
|
||||||
|
// Copyright Basile Burg 2017.
|
||||||
|
// Distributed under the Boost Software License, Version 1.0.
|
||||||
|
// (See accompanying file LICENSE_1_0.txt or copy at
|
||||||
|
// http://www.boost.org/LICENSE_1_0.txt)
|
||||||
|
module analysis.useless_initializer;
|
||||||
|
|
||||||
|
import analysis.base;
|
||||||
|
import dparse.ast;
|
||||||
|
import dparse.lexer;
|
||||||
|
import std.stdio;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Limitations:
|
||||||
|
- Stuff = Stuff.init doesnot work with type with * []
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check that detects the initializers that are
|
||||||
|
* not different from the implcit initializer.
|
||||||
|
*/
|
||||||
|
final class UselessInitializerChecker : BaseAnalyzer
|
||||||
|
{
|
||||||
|
alias visit = BaseAnalyzer.visit;
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
enum key = "dscanner.useless-initializer";
|
||||||
|
version(unittest)
|
||||||
|
enum msg = "X";
|
||||||
|
else
|
||||||
|
enum msg = `Variable %s initializer is useless because it does not differ from the default value`;
|
||||||
|
|
||||||
|
static immutable strDefs = [`""`, `""c`, `""w`, `""d`, "``", "``c", "``w", "``d", "q{}"];
|
||||||
|
static immutable intDefs = ["0", "0L", "0UL", "0uL", "0U", "0x0", "0b0"];
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
///
|
||||||
|
this(string fileName, bool skipTests = false)
|
||||||
|
{
|
||||||
|
super(fileName, null, skipTests);
|
||||||
|
}
|
||||||
|
|
||||||
|
override void visit(const(VariableDeclaration) decl)
|
||||||
|
{
|
||||||
|
foreach (declarator; decl.declarators)
|
||||||
|
{
|
||||||
|
if (!decl.type || !decl.type.type2)
|
||||||
|
continue;
|
||||||
|
if (!declarator.initializer || !declarator.initializer.nonVoidInitializer)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
import std.format : format;
|
||||||
|
|
||||||
|
version(unittest)
|
||||||
|
enum warn = q{addErrorMessage(declarator.name.line, declarator.name.column,
|
||||||
|
key, msg);};
|
||||||
|
else
|
||||||
|
enum warn = q{addErrorMessage(declarator.name.line, declarator.name.column,
|
||||||
|
key, msg.format(declarator.name.text));};
|
||||||
|
|
||||||
|
// --- Info about the declaration type --- //
|
||||||
|
import std.algorithm : among, canFind;
|
||||||
|
import std.algorithm.iteration : filter;
|
||||||
|
import std.range : empty;
|
||||||
|
|
||||||
|
const bool isPtr = decl.type.typeSuffixes && !decl.type.typeSuffixes
|
||||||
|
.filter!(a => a.star != tok!"").empty;
|
||||||
|
const bool isArr = decl.type.typeSuffixes && !decl.type.typeSuffixes
|
||||||
|
.filter!(a => a.array).empty;
|
||||||
|
|
||||||
|
bool isStr, isSzInt;
|
||||||
|
Token customType;
|
||||||
|
|
||||||
|
if (decl.type.type2.symbol && decl.type.type2.symbol.identifierOrTemplateChain &&
|
||||||
|
decl.type.type2.symbol.identifierOrTemplateChain.identifiersOrTemplateInstances.length == 1)
|
||||||
|
{
|
||||||
|
const IdentifierOrTemplateInstance idt =
|
||||||
|
decl.type.type2.symbol.identifierOrTemplateChain.identifiersOrTemplateInstances[0];
|
||||||
|
|
||||||
|
customType = idt.identifier;
|
||||||
|
isStr = customType.text.among("string", "wstring", "dstring") != 0;
|
||||||
|
isSzInt = customType.text.among("size_t", "ptrdiff_t") != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- 'BasicType/Symbol AssignExpression' ---//
|
||||||
|
const NonVoidInitializer nvi = declarator.initializer.nonVoidInitializer;
|
||||||
|
const UnaryExpression ue = cast(UnaryExpression) nvi.assignExpression;
|
||||||
|
if (ue && ue.primaryExpression)
|
||||||
|
{
|
||||||
|
const Token value = ue.primaryExpression.primary;
|
||||||
|
|
||||||
|
if (!isPtr && !isArr && !isStr && decl.type.type2.builtinType != tok!"")
|
||||||
|
{
|
||||||
|
switch(decl.type.type2.builtinType)
|
||||||
|
{
|
||||||
|
// check for common cases of default values
|
||||||
|
case tok!"byte", tok!"ubyte":
|
||||||
|
case tok!"short", tok!"ushort":
|
||||||
|
case tok!"int", tok!"uint":
|
||||||
|
case tok!"long", tok!"ulong":
|
||||||
|
case tok!"cent", tok!"ucent":
|
||||||
|
if (intDefs.canFind(value.text))
|
||||||
|
mixin(warn);
|
||||||
|
goto default;
|
||||||
|
default:
|
||||||
|
// check for BasicType.init
|
||||||
|
if (ue.primaryExpression.basicType.type == decl.type.type2.builtinType &&
|
||||||
|
ue.primaryExpression.primary.text == "init" &&
|
||||||
|
!ue.primaryExpression.expression)
|
||||||
|
mixin(warn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (isSzInt)
|
||||||
|
{
|
||||||
|
if (intDefs.canFind(value.text))
|
||||||
|
mixin(warn);
|
||||||
|
}
|
||||||
|
else if (isPtr)
|
||||||
|
{
|
||||||
|
if (str(value.type) == "null")
|
||||||
|
mixin(warn);
|
||||||
|
}
|
||||||
|
else if (isArr)
|
||||||
|
{
|
||||||
|
if (str(value.type) == "null")
|
||||||
|
mixin(warn);
|
||||||
|
else if (nvi.arrayInitializer && nvi.arrayInitializer.arrayMemberInitializations.length == 0)
|
||||||
|
mixin(warn);
|
||||||
|
else if (decl.type.type2.builtinType != tok!"")
|
||||||
|
{
|
||||||
|
switch(decl.type.type2.builtinType)
|
||||||
|
{
|
||||||
|
case tok!"char", tok!"wchar", tok!"dchar":
|
||||||
|
if (strDefs.canFind(value.text))
|
||||||
|
mixin(warn);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (isStr)
|
||||||
|
{
|
||||||
|
if (strDefs.canFind(value.text))
|
||||||
|
mixin(warn);
|
||||||
|
else if (nvi.arrayInitializer && nvi.arrayInitializer.arrayMemberInitializations.length == 0)
|
||||||
|
mixin(warn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Symbol s = Symbol.init
|
||||||
|
else if (ue && customType != tok!"" && ue.unaryExpression && ue.unaryExpression.primaryExpression &&
|
||||||
|
ue.unaryExpression.primaryExpression.identifierOrTemplateInstance &&
|
||||||
|
ue.unaryExpression.primaryExpression.identifierOrTemplateInstance.identifier == customType &&
|
||||||
|
ue.identifierOrTemplateInstance && ue.identifierOrTemplateInstance.identifier.text == "init")
|
||||||
|
{
|
||||||
|
mixin(warn);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 'Symbol ArrayInitializer' : assumes Symbol is an array b/c of the Init
|
||||||
|
else if (nvi.arrayInitializer && (isArr || isStr))
|
||||||
|
{
|
||||||
|
if (nvi.arrayInitializer.arrayMemberInitializations.length == 0)
|
||||||
|
mixin(warn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
decl.accept(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@system unittest
|
||||||
|
{
|
||||||
|
import analysis.config : Check, disabledConfig, StaticAnalysisConfig;
|
||||||
|
import analysis.helpers: assertAnalyzerWarnings;
|
||||||
|
import std.stdio : stderr;
|
||||||
|
|
||||||
|
StaticAnalysisConfig sac = disabledConfig;
|
||||||
|
sac.useless_initializer = Check.enabled;
|
||||||
|
|
||||||
|
// fails
|
||||||
|
assertAnalyzerWarnings(q{
|
||||||
|
ubyte a = 0x0; // [warn]: X
|
||||||
|
int a = 0; // [warn]: X
|
||||||
|
ulong a = 0; // [warn]: X
|
||||||
|
int* a = null; // [warn]: X
|
||||||
|
Foo* a = null; // [warn]: X
|
||||||
|
int[] a = null; // [warn]: X
|
||||||
|
int[] a = []; // [warn]: X
|
||||||
|
string a = ""; // [warn]: X
|
||||||
|
string a = ""c; // [warn]: X
|
||||||
|
wstring a = ""w; // [warn]: X
|
||||||
|
dstring a = ""d; // [warn]: X
|
||||||
|
string a = q{}; // [warn]: X
|
||||||
|
size_t a = 0; // [warn]: X
|
||||||
|
ptrdiff_t a = 0; // [warn]: X
|
||||||
|
string a = []; // [warn]: X
|
||||||
|
char[] a = ""; // [warn]: X
|
||||||
|
int a = int.init; // [warn]: X
|
||||||
|
char a = char.init; // [warn]: X
|
||||||
|
S s = S.init; // [warn]: X
|
||||||
|
}, sac);
|
||||||
|
|
||||||
|
// passes
|
||||||
|
assertAnalyzerWarnings(q{
|
||||||
|
ubyte a = 0xFE;
|
||||||
|
int a = 1;
|
||||||
|
ulong a = 1;
|
||||||
|
int* a = &a;
|
||||||
|
Foo* a = &a;
|
||||||
|
int[] a = &a;
|
||||||
|
int[] a = [0];
|
||||||
|
string a = "sdf";
|
||||||
|
string a = "sdg"c;
|
||||||
|
wstring a = "sdg"w;
|
||||||
|
dstring a = "fgh"d;
|
||||||
|
string a = q{int a;};
|
||||||
|
size_t a = 1;
|
||||||
|
ptrdiff_t a = 1;
|
||||||
|
string a = ['a'];
|
||||||
|
char[] a = "ze";
|
||||||
|
S s = S(0,1);
|
||||||
|
S s = s.call();
|
||||||
|
}, sac);
|
||||||
|
|
||||||
|
stderr.writeln("Unittest for UselessInitializerChecker passed.");
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue