blob: 4f42fa8daedcf8aaf8a0fe5964346a0d39591f92 [file] [log] [blame]
// Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'package:analyzer/dart/analysis/features.dart';
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/ast/token.dart';
import 'package:analyzer/dart/ast/visitor.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/dart/element/type.dart';
import 'package:analyzer/error/error.dart';
import 'package:analyzer/error/listener.dart';
import 'package:analyzer/source/source_range.dart';
import 'package:analyzer/src/dart/element/type_system.dart';
import 'package:analyzer/src/dart/resolver/flow_analysis_visitor.dart';
import 'package:analyzer/src/dart/resolver/scope.dart';
import 'package:analyzer/src/error/codes.dart';
typedef _CatchClausesVerifierReporter = void Function(
CatchClause first,
CatchClause last,
ErrorCode,
List<Object> arguments,
);
/// A visitor that finds dead code, other than unreachable code that is
/// handled in [NullSafetyDeadCodeVerifier].
class DeadCodeVerifier extends RecursiveAstVisitor<void> {
/// The error reporter by which errors will be reported.
final ErrorReporter _errorReporter;
/// The object used to track the usage of labels within a given label scope.
_LabelTracker? _labelTracker;
/// Whether the `wildcard_variables` feature is enabled.
final bool _wildCardVariablesEnabled;
DeadCodeVerifier(this._errorReporter, LibraryElement library)
: _wildCardVariablesEnabled =
library.featureSet.isEnabled(Feature.wildcard_variables);
@override
void visitBreakStatement(BreakStatement node) {
_labelTracker?.recordUsage(node.label?.name);
}
@override
void visitContinueStatement(ContinueStatement node) {
_labelTracker?.recordUsage(node.label?.name);
}
@override
void visitExportDirective(ExportDirective node) {
var exportElement = node.element;
if (exportElement != null) {
// The element is null when the URI is invalid.
LibraryElement? library = exportElement.exportedLibrary;
if (library != null && !library.isSynthetic) {
for (Combinator combinator in node.combinators) {
_checkCombinator(library, combinator);
}
}
}
super.visitExportDirective(node);
}
@override
void visitFunctionDeclaration(FunctionDeclaration node) {
var element = node.declaredElement;
// TODO(pq): ask the FunctionElement once implemented
if (_wildCardVariablesEnabled &&
element is FunctionElement &&
element.isLocal &&
element.name == '_') {
_errorReporter.atNode(node, WarningCode.DEAD_CODE);
}
super.visitFunctionDeclaration(node);
}
@override
void visitImportDirective(ImportDirective node) {
var importElement = node.element;
if (importElement != null) {
// The element is null when the URI is invalid, but not when the URI is
// valid but refers to a nonexistent file.
LibraryElement? library = importElement.importedLibrary;
if (library != null && !library.isSynthetic) {
for (Combinator combinator in node.combinators) {
_checkCombinator(library, combinator);
}
}
}
super.visitImportDirective(node);
}
@override
void visitLabeledStatement(LabeledStatement node) {
_withLabelTracker(node.labels, () {
super.visitLabeledStatement(node);
});
}
@override
void visitSwitchStatement(SwitchStatement node) {
List<Label> labels = <Label>[];
for (SwitchMember member in node.members) {
labels.addAll(member.labels);
}
_withLabelTracker(labels, () {
super.visitSwitchStatement(node);
});
}
/// Resolve the names in the given [combinator] in the scope of the given
/// [library].
void _checkCombinator(LibraryElement library, Combinator combinator) {
Namespace namespace =
NamespaceBuilder().createExportNamespaceForLibrary(library);
NodeList<SimpleIdentifier> names;
ErrorCode warningCode;
if (combinator is HideCombinator) {
names = combinator.hiddenNames;
warningCode = WarningCode.UNDEFINED_HIDDEN_NAME;
} else {
names = (combinator as ShowCombinator).shownNames;
warningCode = WarningCode.UNDEFINED_SHOWN_NAME;
}
for (SimpleIdentifier name in names) {
String nameStr = name.name;
Element? element = namespace.get(nameStr);
element ??= namespace.get("$nameStr=");
if (element == null) {
_errorReporter.atNode(
name,
warningCode,
arguments: [library.identifier, nameStr],
);
}
}
}
void _withLabelTracker(List<Label> labels, void Function() f) {
var labelTracker = _LabelTracker(_labelTracker, labels);
try {
_labelTracker = labelTracker;
f();
} finally {
for (Label label in labelTracker.unusedLabels()) {
_errorReporter.atNode(
label,
WarningCode.UNUSED_LABEL,
arguments: [label.label.name],
);
}
_labelTracker = labelTracker.outerTracker;
}
}
}
/// Helper for tracking dead code - [CatchClause]s and unreachable code.
///
/// [CatchClause]s are checked separately, as we visit AST we may make some
/// of them as dead, and record [_deadCatchClauseRanges].
///
/// When an unreachable node is found, and [_firstDeadNode] is `null`, we
/// set [_firstDeadNode], so start a new dead nodes interval. The dead code
/// interval ends when [flowEnd] is invoked with a node that is the start
/// node, or contains it. So, we end the end of the covering control flow.
class NullSafetyDeadCodeVerifier {
final TypeSystemImpl _typeSystem;
final ErrorReporter _errorReporter;
final FlowAnalysisHelper? _flowAnalysis;
/// The stack of verifiers of (potentially nested) try statements.
final List<_CatchClausesVerifier> _catchClausesVerifiers = [];
/// When a sequence [CatchClause]s is found to be dead, we don't want to
/// report additional dead code inside of already dead code.
final List<SourceRange> _deadCatchClauseRanges = [];
/// When this field is `null`, we are in reachable code.
/// Once we find the first unreachable node, we store it here.
///
/// When this field is not `null`, and we see an unreachable node, this new
/// node is ignored, because it continues the same dead code range.
AstNode? _firstDeadNode;
NullSafetyDeadCodeVerifier(
this._typeSystem,
this._errorReporter,
this._flowAnalysis,
);
/// The [node] ends a basic block in the control flow. If [_firstDeadNode] is
/// not `null`, and is covered by the [node], then we reached the end of
/// the current dead code interval.
void flowEnd(AstNode node) {
var firstDeadNode = _firstDeadNode;
if (firstDeadNode == null) {
return;
}
if (!_containsFirstDeadNode(node)) {
return;
}
if (node is SwitchMember && node == firstDeadNode) {
_errorReporter.atToken(
node.keyword,
WarningCode.DEAD_CODE,
);
_firstDeadNode = null;
return;
}
var parent = firstDeadNode.parent;
if (parent is Assertion && identical(firstDeadNode, parent.message)) {
// Don't report "dead code" for the message part of an assert statement,
// because this causes nuisance warnings for redundant `!= null`
// asserts.
} else if (parent is ConstructorDeclaration &&
firstDeadNode is EmptyFunctionBody) {
// Don't report "dead code" for an unreachable, but syntacically required,
// semicolon that follows one or more constructor initializers.
} else if (parent is ConstructorDeclaration &&
firstDeadNode is BlockFunctionBody &&
firstDeadNode.block.statements.isEmpty) {
// Don't report "dead code" for an unreachable, but empty block body that
// follows one or more constructor initializers.
} else {
var offset = firstDeadNode.offset;
// We know that [node] is the first dead node, or contains it.
// So, technically the code interval ends at the end of [node].
// But we trim it to the last statement for presentation purposes.
if (node != firstDeadNode) {
if (node is FunctionDeclaration) {
node = node.functionExpression.body;
}
if (node is FunctionExpression) {
node = node.body;
}
if (node is MethodDeclaration) {
node = node.body;
}
if (node is BlockFunctionBody) {
node = node.block;
}
if (node is Block && node.statements.isNotEmpty) {
node = node.statements.last;
}
if (node is SwitchMember && node.statements.isNotEmpty) {
node = node.statements.last;
}
} else if (parent is BinaryExpression && node == parent.rightOperand) {
offset = parent.operator.offset;
}
if (parent is ConstructorInitializer) {
_errorReporter.atOffset(
offset: parent.offset,
length: parent.end - parent.offset,
errorCode: WarningCode.DEAD_CODE,
);
offset = node.end;
} else if (parent is DoStatement) {
var doOffset = parent.doKeyword.offset;
var doEnd = parent.doKeyword.end;
var whileOffset = parent.whileKeyword.offset;
var whileEnd = parent.semicolon.end;
var body = parent.body;
if (body is Block) {
doEnd = body.leftBracket.end;
whileOffset = body.rightBracket.offset;
}
_errorReporter.atOffset(
offset: doOffset,
length: doEnd - doOffset,
errorCode: WarningCode.DEAD_CODE,
);
_errorReporter.atOffset(
offset: whileOffset,
length: whileEnd - whileOffset,
errorCode: WarningCode.DEAD_CODE,
);
offset = parent.semicolon.next!.offset;
if (parent.hasBreakStatement) {
offset = node.end;
}
} else if (parent is ForParts) {
node = parent.updaters.last;
} else if (parent is ForStatement) {
_reportForUpdaters(parent);
} else if (parent is Block) {
var grandParent = parent.parent;
if (grandParent is ForStatement) {
_reportForUpdaters(grandParent);
}
} else if (parent is LogicalOrPattern && node == parent.rightOperand) {
offset = parent.operator.offset;
}
var length = node.end - offset;
if (length > 0) {
_errorReporter.atOffset(
offset: offset,
length: length,
errorCode: WarningCode.DEAD_CODE,
);
}
}
_firstDeadNode = null;
}
void tryStatementEnter(TryStatement node) {
var verifier = _CatchClausesVerifier(
_typeSystem,
(first, last, errorCode, arguments) {
var offset = first.offset;
var length = last.end - offset;
_errorReporter.atOffset(
offset: offset,
length: length,
errorCode: errorCode,
arguments: arguments,
);
_deadCatchClauseRanges.add(SourceRange(offset, length));
},
node.catchClauses,
);
_catchClausesVerifiers.add(verifier);
}
void tryStatementExit(TryStatement node) {
_catchClausesVerifiers.removeLast();
}
void verifyCascadeExpression(CascadeExpression node) {
var first = node.cascadeSections.firstOrNull;
if (first is PropertyAccess) {
_verifyUnassignedSimpleIdentifier(node, node.target, first.operator);
} else if (first is MethodInvocation) {
_verifyUnassignedSimpleIdentifier(node, node.target, first.operator);
} else if (first is IndexExpression) {
_verifyUnassignedSimpleIdentifier(node, node.target, first.period);
}
}
void verifyCatchClause(CatchClause node) {
var verifier = _catchClausesVerifiers.last;
if (verifier._done) return;
verifier.nextCatchClause(node);
}
void verifyIndexExpression(IndexExpression node) {
_verifyUnassignedSimpleIdentifier(node, node.target, node.question);
}
void verifyMethodInvocation(MethodInvocation node) {
_verifyUnassignedSimpleIdentifier(node, node.target, node.operator);
}
void verifyPropertyAccess(PropertyAccess node) {
_verifyUnassignedSimpleIdentifier(node, node.target, node.operator);
}
void visitNode(AstNode node) {
// Comments are visited after bodies of functions.
// So, they look unreachable, but this does not make sense.
if (node is Comment) return;
var flowAnalysis = _flowAnalysis;
if (flowAnalysis == null) return;
flowAnalysis.checkUnreachableNode(node);
// If the first dead node is not `null`, even if this new node is
// unreachable, we can ignore it as it is part of the same dead code
// range anyway.
if (_firstDeadNode != null) return;
var flow = flowAnalysis.flow;
if (flow == null) return;
if (flow.isReachable) return;
// If in a dead `CatchClause`, no need to report dead code.
for (var range in _deadCatchClauseRanges) {
if (range.contains(node.offset)) {
return;
}
}
_firstDeadNode = node;
}
bool _containsFirstDeadNode(AstNode parent) {
for (var node = _firstDeadNode; node != null; node = node.parent) {
if (node == parent) return true;
}
return false;
}
void _reportForUpdaters(ForStatement node) {
var forParts = node.forLoopParts;
if (forParts is ForParts) {
var updaters = forParts.updaters;
var beginToken = updaters.beginToken;
var endToken = updaters.endToken;
if (beginToken != null && endToken != null) {
_errorReporter.atOffset(
offset: beginToken.offset,
length: endToken.end - beginToken.offset,
errorCode: WarningCode.DEAD_CODE,
);
}
}
}
void _verifyUnassignedSimpleIdentifier(
AstNode node, Expression? target, Token? operator) {
var flowAnalysis = _flowAnalysis;
if (flowAnalysis == null) return;
var operatorType = operator?.type;
if (operatorType != TokenType.QUESTION &&
operatorType != TokenType.QUESTION_PERIOD &&
operatorType != TokenType.QUESTION_PERIOD_PERIOD) {
return;
}
if (target?.staticType?.nullabilitySuffix != NullabilitySuffix.question) {
return;
}
target = target?.unParenthesized;
if (target is SimpleIdentifier) {
var element = target.staticElement;
if (element is PromotableElement &&
flowAnalysis.isDefinitelyUnassigned(target, element)) {
var parent = node.parent;
while (parent is MethodInvocation ||
parent is PropertyAccess ||
parent is IndexExpression) {
node = parent!;
parent = node.parent;
}
_errorReporter.atNode(
node,
WarningCode.DEAD_CODE,
);
}
}
}
}
/// A visitor that finds a [BreakStatement] for a specified [DoStatement].
class _BreakDoStatementVisitor extends RecursiveAstVisitor<void> {
bool hasBreakStatement = false;
final DoStatement doStatement;
_BreakDoStatementVisitor(this.doStatement);
@override
void visitBreakStatement(BreakStatement node) {
if (node.target == doStatement) {
hasBreakStatement = true;
}
}
}
class _CatchClausesVerifier {
final TypeSystemImpl _typeSystem;
final _CatchClausesVerifierReporter _errorReporter;
final List<CatchClause> catchClauses;
bool _done = false;
final List<DartType> _visitedTypes = <DartType>[];
_CatchClausesVerifier(
this._typeSystem,
this._errorReporter,
this.catchClauses,
);
void nextCatchClause(CatchClause catchClause) {
var currentType = catchClause.exceptionType?.type;
// Found catch clause that doesn't have an exception type.
// Generate an error on any following catch clauses.
if (currentType == null || currentType.isDartCoreObject) {
if (catchClause != catchClauses.last) {
var index = catchClauses.indexOf(catchClause);
_errorReporter(
catchClauses[index + 1],
catchClauses.last,
WarningCode.DEAD_CODE_CATCH_FOLLOWING_CATCH,
const [],
);
_done = true;
}
return;
}
// An on-catch clause was found; verify that the exception type is not a
// subtype of a previous on-catch exception type.
for (var type in _visitedTypes) {
if (_typeSystem.isSubtypeOf(currentType, type)) {
_errorReporter(
catchClause,
catchClauses.last,
WarningCode.DEAD_CODE_ON_CATCH_SUBTYPE,
[currentType, type],
);
_done = true;
return;
}
}
_visitedTypes.add(currentType);
}
}
/// An object used to track the usage of labels within a single label scope.
class _LabelTracker {
/// The tracker for the outer label scope.
final _LabelTracker? outerTracker;
/// The labels whose usage is being tracked.
final List<Label> labels;
/// A list of flags corresponding to the list of [labels] indicating whether
/// the corresponding label has been used.
late final List<bool> used;
/// A map from the names of labels to the index of the label in [labels].
final Map<String, int> labelMap = <String, int>{};
/// Initialize a newly created label tracker.
_LabelTracker(this.outerTracker, this.labels) {
used = List.filled(labels.length, false);
for (int i = 0; i < labels.length; i++) {
labelMap[labels[i].label.name] = i;
}
}
/// Record that the label with the given [labelName] has been used.
void recordUsage(String? labelName) {
if (labelName != null) {
var index = labelMap[labelName];
if (index != null) {
used[index] = true;
} else {
outerTracker?.recordUsage(labelName);
}
}
}
/// Return the unused labels.
Iterable<Label> unusedLabels() sync* {
for (int i = 0; i < labels.length; i++) {
if (!used[i]) {
yield labels[i];
}
}
}
}
extension on FunctionElement {
bool get isLocal =>
enclosingElement is FunctionElement || enclosingElement is MethodElement;
}
extension DoStatementExtension on DoStatement {
bool get hasBreakStatement {
var visitor = _BreakDoStatementVisitor(this);
body.visitChildren(visitor);
return visitor.hasBreakStatement;
}
}