Fix #438 - check for __gshared and immutable redundancies
This commit is contained in:
parent
6ca59c71a2
commit
558c1bf6cf
|
@ -142,6 +142,7 @@ Note that the "--skipTests" option is the equivalent of changing each
|
||||||
* Asserts without an explanatory message. By default disabled.
|
* Asserts without an explanatory message. By default disabled.
|
||||||
* Indentation of if constraints
|
* Indentation of if constraints
|
||||||
* Check that `@trusted` is not applied to a whole scope. Trusting a whole scope can be a problem when new declarations are added and if they are not verified manually to be trustable.
|
* Check that `@trusted` is not applied to a whole scope. Trusting a whole scope can be a problem when new declarations are added and if they are not verified manually to be trustable.
|
||||||
|
* Redundant storage class attributes
|
||||||
|
|
||||||
#### Wishlist
|
#### Wishlist
|
||||||
|
|
||||||
|
@ -228,7 +229,7 @@ the given source file to standard output in XML format.
|
||||||
```sh
|
```sh
|
||||||
$ dscanner --ast helloworld.d
|
$ dscanner --ast helloworld.d
|
||||||
```
|
```
|
||||||
|
|
||||||
```xml
|
```xml
|
||||||
<module>
|
<module>
|
||||||
<declaration>
|
<declaration>
|
||||||
|
|
|
@ -197,6 +197,9 @@ struct StaticAnalysisConfig
|
||||||
@INI("Check for @trusted applied to a bigger scope than a single function")
|
@INI("Check for @trusted applied to a bigger scope than a single function")
|
||||||
string trust_too_much = Check.enabled;
|
string trust_too_much = Check.enabled;
|
||||||
|
|
||||||
|
@INI("Check for redundant storage classes on variable declarations")
|
||||||
|
string redundant_storage_classes = Check.enabled;
|
||||||
|
|
||||||
@INI("Module-specific filters")
|
@INI("Module-specific filters")
|
||||||
ModuleFilters filters;
|
ModuleFilters filters;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,102 @@
|
||||||
|
// Copyright (c) 2018, dlang-community
|
||||||
|
// 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 dscanner.analysis.redundant_storage_class;
|
||||||
|
|
||||||
|
import std.stdio;
|
||||||
|
import std.string;
|
||||||
|
import dparse.ast;
|
||||||
|
import dparse.lexer;
|
||||||
|
import dscanner.analysis.base;
|
||||||
|
import dscanner.analysis.helpers;
|
||||||
|
import dsymbol.scope_ : Scope;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks for redundant storage classes such immutable and __gshared, static and __gshared
|
||||||
|
*/
|
||||||
|
class RedundantStorageClassCheck : BaseAnalyzer
|
||||||
|
{
|
||||||
|
alias visit = BaseAnalyzer.visit;
|
||||||
|
enum string REDUNDANT_VARIABLE_ATTRIBUTES = "Variable declaration for `%s` has redundant attributes (%-(`%s`%|, %)).";
|
||||||
|
|
||||||
|
this(string fileName, bool skipTests = false)
|
||||||
|
{
|
||||||
|
super(fileName, null, skipTests);
|
||||||
|
}
|
||||||
|
|
||||||
|
override void visit(const Declaration node)
|
||||||
|
{
|
||||||
|
checkAttributes(node);
|
||||||
|
node.accept(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void checkAttributes(const Declaration node)
|
||||||
|
{
|
||||||
|
if (node.variableDeclaration !is null && node.attributes !is null)
|
||||||
|
checkVariableDeclaration(node.variableDeclaration, node.attributes);
|
||||||
|
}
|
||||||
|
|
||||||
|
void checkVariableDeclaration(const VariableDeclaration vd, const Attribute[] attributes)
|
||||||
|
{
|
||||||
|
import std.algorithm.comparison : among;
|
||||||
|
import std.algorithm.searching: all;
|
||||||
|
|
||||||
|
string[] globalAttributes;
|
||||||
|
foreach (attrib; attributes)
|
||||||
|
{
|
||||||
|
if (attrib.attribute.type.among(tok!"shared", tok!"static", tok!"__gshared", tok!"immutable"))
|
||||||
|
globalAttributes ~= attrib.attribute.type.str;
|
||||||
|
}
|
||||||
|
if (globalAttributes.length > 1)
|
||||||
|
{
|
||||||
|
if (globalAttributes.length == 2 && (
|
||||||
|
globalAttributes.all!(a => a.among("shared", "static")) ||
|
||||||
|
globalAttributes.all!(a => a.among("static", "immutable"))
|
||||||
|
))
|
||||||
|
return;
|
||||||
|
auto t = vd.declarators[0].name;
|
||||||
|
string message = REDUNDANT_VARIABLE_ATTRIBUTES.format(t.text, globalAttributes);
|
||||||
|
addErrorMessage(t.line, t.column, "dscanner.unnecessary.duplicate_attribute", message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest
|
||||||
|
{
|
||||||
|
import dscanner.analysis.config : StaticAnalysisConfig, Check, disabledConfig;
|
||||||
|
|
||||||
|
StaticAnalysisConfig sac = disabledConfig();
|
||||||
|
sac.redundant_storage_classes = Check.enabled;
|
||||||
|
|
||||||
|
// https://github.com/dlang-community/D-Scanner/issues/438
|
||||||
|
assertAnalyzerWarnings(q{
|
||||||
|
immutable int a;
|
||||||
|
|
||||||
|
immutable shared int a; // [warn]: %s
|
||||||
|
shared immutable int a; // [warn]: %s
|
||||||
|
|
||||||
|
immutable __gshared int a; // [warn]: %s
|
||||||
|
__gshared immutable int a; // [warn]: %s
|
||||||
|
|
||||||
|
__gshared static int a; // [warn]: %s
|
||||||
|
|
||||||
|
shared static int a;
|
||||||
|
static shared int a;
|
||||||
|
static immutable int a;
|
||||||
|
immutable static int a;
|
||||||
|
|
||||||
|
enum int a;
|
||||||
|
extern(C++) immutable int a;
|
||||||
|
immutable int function(immutable int, shared int) a;
|
||||||
|
}c.format(
|
||||||
|
RedundantStorageClassCheck.REDUNDANT_VARIABLE_ATTRIBUTES.format("a", ["immutable", "shared"]),
|
||||||
|
RedundantStorageClassCheck.REDUNDANT_VARIABLE_ATTRIBUTES.format("a", ["shared", "immutable"]),
|
||||||
|
RedundantStorageClassCheck.REDUNDANT_VARIABLE_ATTRIBUTES.format("a", ["immutable", "__gshared"]),
|
||||||
|
RedundantStorageClassCheck.REDUNDANT_VARIABLE_ATTRIBUTES.format("a", ["__gshared", "immutable"]),
|
||||||
|
RedundantStorageClassCheck.REDUNDANT_VARIABLE_ATTRIBUTES.format("a", ["__gshared", "static"]),
|
||||||
|
), sac);
|
||||||
|
|
||||||
|
stderr.writeln("Unittest for RedundantStorageClassCheck passed.");
|
||||||
|
}
|
|
@ -73,6 +73,7 @@ import dscanner.analysis.has_public_example;
|
||||||
import dscanner.analysis.assert_without_msg;
|
import dscanner.analysis.assert_without_msg;
|
||||||
import dscanner.analysis.if_constraints_indent;
|
import dscanner.analysis.if_constraints_indent;
|
||||||
import dscanner.analysis.trust_too_much;
|
import dscanner.analysis.trust_too_much;
|
||||||
|
import dscanner.analysis.redundant_storage_class;
|
||||||
|
|
||||||
import dsymbol.string_interning : internString;
|
import dsymbol.string_interning : internString;
|
||||||
import dsymbol.scope_;
|
import dsymbol.scope_;
|
||||||
|
@ -516,6 +517,10 @@ MessageSet analyze(string fileName, const Module m, const StaticAnalysisConfig a
|
||||||
checks ~= new TrustTooMuchCheck(fileName,
|
checks ~= new TrustTooMuchCheck(fileName,
|
||||||
analysisConfig.trust_too_much == Check.skipTests && !ut);
|
analysisConfig.trust_too_much == Check.skipTests && !ut);
|
||||||
|
|
||||||
|
if (moduleName.shouldRun!"redundant_storage_classes"(analysisConfig))
|
||||||
|
checks ~= new RedundantStorageClassCheck(fileName,
|
||||||
|
analysisConfig.redundant_storage_classes == Check.skipTests && !ut);
|
||||||
|
|
||||||
version (none)
|
version (none)
|
||||||
if (moduleName.shouldRun!"redundant_if_check"(analysisConfig))
|
if (moduleName.shouldRun!"redundant_if_check"(analysisConfig))
|
||||||
checks ~= new IfStatementCheck(fileName, moduleScope,
|
checks ~= new IfStatementCheck(fileName, moduleScope,
|
||||||
|
|
Loading…
Reference in New Issue