diff --git a/README.md b/README.md index ac4ca07..140a56f 100644 --- a/README.md +++ b/README.md @@ -142,6 +142,7 @@ Note that the "--skipTests" option is the equivalent of changing each * Asserts without an explanatory message. By default disabled. * 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. +* Redundant storage class attributes #### Wishlist @@ -228,7 +229,7 @@ the given source file to standard output in XML format. ```sh $ dscanner --ast helloworld.d ``` - + ```xml diff --git a/src/dscanner/analysis/config.d b/src/dscanner/analysis/config.d index 436a60b..b269a1c 100644 --- a/src/dscanner/analysis/config.d +++ b/src/dscanner/analysis/config.d @@ -197,6 +197,9 @@ struct StaticAnalysisConfig @INI("Check for @trusted applied to a bigger scope than a single function") 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") ModuleFilters filters; } diff --git a/src/dscanner/analysis/redundant_storage_class.d b/src/dscanner/analysis/redundant_storage_class.d new file mode 100644 index 0000000..f9bb2be --- /dev/null +++ b/src/dscanner/analysis/redundant_storage_class.d @@ -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."); +} diff --git a/src/dscanner/analysis/run.d b/src/dscanner/analysis/run.d index 653d775..01e586c 100644 --- a/src/dscanner/analysis/run.d +++ b/src/dscanner/analysis/run.d @@ -73,6 +73,7 @@ import dscanner.analysis.has_public_example; import dscanner.analysis.assert_without_msg; import dscanner.analysis.if_constraints_indent; import dscanner.analysis.trust_too_much; +import dscanner.analysis.redundant_storage_class; import dsymbol.string_interning : internString; import dsymbol.scope_; @@ -516,6 +517,10 @@ MessageSet analyze(string fileName, const Module m, const StaticAnalysisConfig a checks ~= new TrustTooMuchCheck(fileName, 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) if (moduleName.shouldRun!"redundant_if_check"(analysisConfig)) checks ~= new IfStatementCheck(fileName, moduleScope,