blob: c3489dd1fbf727d823dd12d9c772587b8c4907b4 [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/visitor.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/dart/element/nullability_suffix.dart';
import 'package:analyzer/dart/element/type.dart';
import 'package:analyzer/error/error.dart';
import 'package:analyzer/error/listener.dart';
import 'package:analyzer/src/dart/ast/ast.dart';
import 'package:analyzer/src/dart/element/element.dart';
import 'package:analyzer/src/dart/element/type.dart';
import 'package:analyzer/src/dart/resolver/ast_rewrite.dart';
import 'package:analyzer/src/dart/resolver/scope.dart';
import 'package:analyzer/src/error/codes.dart';
import 'package:analyzer/src/generated/declaration_resolver.dart';
import 'package:analyzer/src/generated/resolver.dart';
import 'package:analyzer/src/generated/utilities_dart.dart';
import 'package:meta/meta.dart';
class ElementHolder {
final ElementImpl _element;
final List<TypeParameterElementImpl> _typeParameters = [];
final List<ParameterElementImpl> _parameters = [];
ElementHolder(this._element) : assert(_element != null);
List<ParameterElementImpl> get parameters => _parameters;
List<TypeParameterElementImpl> get typeParameters => _typeParameters;
void addParameter(ParameterElementImpl element) {
_parameters.add(element);
}
void addTypeParameter(TypeParameterElementImpl element) {
_typeParameters.add(element);
}
void enclose(ElementImpl element) {
element.enclosingElement = _element;
}
}
/// Recursively visit AST and perform following resolution tasks:
///
/// 1. Set existing top-level elements from [_elementWalker] to corresponding
/// nodes in AST.
/// 2. Create and set new elements for local declarations.
/// 3. Resolve all [TypeName]s - set elements and types.
/// 4. Resolve all [GenericFunctionType]s - set their types.
class ResolutionVisitor extends RecursiveAstVisitor<void> {
final TypeProvider _typeProvider;
final CompilationUnitElementImpl _unitElement;
final bool _nonNullableEnabled;
final ErrorReporter _errorReporter;
final AstRewriter _astRewriter;
final TypeNameResolver _typeNameResolver;
/// The provider of pre-built children elements from the element being
/// visited. For example when we visit a method, its element is resynthesized
/// from the summary, and we get resynthesized elements for type parameters
/// and formal parameters to apply to corresponding AST nodes.
ElementWalker _elementWalker;
/// The scope used to resolve identifiers.
Scope _nameScope;
/// The scope used to resolve labels for `break` and `continue` statements,
/// or `null` if no labels have been defined in the current context.
LabelScope _labelScope;
/// The container to add newly created elements that should be put into the
/// enclosing element.
ElementHolder _elementHolder;
/// The flag specifying if currently visited class references 'super'
/// expression.
/// TODO(scheglov) put into summary
bool _hasReferenceToSuper = false;
factory ResolutionVisitor({
@required CompilationUnitElementImpl unitElement,
@required AnalysisErrorListener errorListener,
@required FeatureSet featureSet,
@required Scope nameScope,
ElementWalker elementWalker,
}) {
var libraryElement = unitElement.library;
var typeProvider = libraryElement.context.typeProvider;
var unitSource = unitElement.source;
var nonNullableEnabled = featureSet.isEnabled(Feature.non_nullable);
var errorReporter = ErrorReporter(errorListener, unitSource);
var typeNameResolver = TypeNameResolver(
unitElement.context.typeSystem,
typeProvider,
nonNullableEnabled,
libraryElement,
unitSource,
errorListener,
);
return ResolutionVisitor._(
typeProvider,
unitElement,
nonNullableEnabled,
errorReporter,
AstRewriter(libraryElement, errorReporter),
typeNameResolver,
nameScope,
elementWalker,
ElementHolder(unitElement),
);
}
ResolutionVisitor._(
this._typeProvider,
this._unitElement,
this._nonNullableEnabled,
this._errorReporter,
this._astRewriter,
this._typeNameResolver,
this._nameScope,
this._elementWalker,
this._elementHolder,
);
DartType get _dynamicType => _typeProvider.dynamicType;
@override
void visitAnnotation(Annotation node) {
_withElementWalker(null, () {
super.visitAnnotation(node);
});
}
@override
void visitBlock(Block node) {
var outerScope = _nameScope;
try {
_nameScope = EnclosedScope(_nameScope);
var statements = node.statements;
_buildLocalElements(statements);
statements.accept(this);
} finally {
_nameScope = outerScope;
}
}
@override
void visitCatchClause(CatchClause node) {
var exceptionTypeNode = node.exceptionType;
exceptionTypeNode?.accept(this);
_withNameScope(() {
var exceptionNode = node.exceptionParameter;
if (exceptionNode != null) {
var element = LocalVariableElementImpl.forNode(exceptionNode);
_elementHolder.enclose(element);
_nameScope.define(element);
exceptionNode.staticElement = element;
if (exceptionTypeNode == null) {
element.hasImplicitType = true;
element.type = _dynamicType;
exceptionNode.staticType = _dynamicType;
} else {
element.type = exceptionTypeNode.type;
exceptionNode.staticType = exceptionTypeNode.type;
}
_setCodeRange(element, exceptionNode);
}
var stackTraceNode = node.stackTraceParameter;
if (stackTraceNode != null) {
var element = LocalVariableElementImpl.forNode(stackTraceNode);
_elementHolder.enclose(element);
_nameScope.define(element);
stackTraceNode.staticElement = element;
element.type = _typeProvider.stackTraceType;
stackTraceNode.staticType = _typeProvider.stackTraceType;
_setCodeRange(element, stackTraceNode);
}
node.body.accept(this);
});
}
@override
void visitClassDeclaration(ClassDeclaration node) {
ClassElementImpl element = _elementWalker.getClass();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forClass(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
var extendsClause = node.extendsClause;
var withClause = node.withClause;
if (extendsClause != null) {
ErrorCode errorCode = withClause == null
? CompileTimeErrorCode.EXTENDS_NON_CLASS
: CompileTimeErrorCode.MIXIN_WITH_NON_CLASS_SUPERCLASS;
_resolveType(extendsClause.superclass, errorCode, asClass: true);
}
_resolveWithClause(withClause);
_resolveImplementsClause(node.implementsClause);
_hasReferenceToSuper = false;
_defineElements(element.accessors);
_defineElements(element.methods);
node.members.accept(this);
element.hasReferenceToSuper = _hasReferenceToSuper;
});
});
}
@override
void visitClassTypeAlias(ClassTypeAlias node) {
ClassElementImpl element = _elementWalker.getClass();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forClass(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
_resolveType(
node.superclass,
CompileTimeErrorCode.MIXIN_WITH_NON_CLASS_SUPERCLASS,
asClass: true,
);
_resolveWithClause(node.withClause);
_resolveImplementsClause(node.implementsClause);
});
});
}
@override
void visitConstructorDeclaration(ConstructorDeclaration node) {
ConstructorElementImpl element = _elementWalker.getConstructor();
(node as ConstructorDeclarationImpl).declaredElement = element;
node.name?.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementHolder(ElementHolder(element), () {
_withElementWalker(null, () {
_withNameScope(() {
node.returnType.accept(this);
_withElementWalker(
ElementWalker.forExecutable(element, _unitElement),
() {
node.parameters.accept(this);
},
);
_defineParameters(element.parameters);
node.redirectedConstructor?.accept(this);
node.initializers.accept(this);
node.body?.accept(this);
});
});
});
}
@override
void visitDeclaredIdentifier(DeclaredIdentifier node) {
var nameNode = node.identifier;
var element = LocalVariableElementImpl.forNode(nameNode);
_elementHolder.enclose(element);
nameNode.staticElement = element;
node.metadata.accept(this);
element.metadata = _createElementAnnotations(node.metadata);
element.isConst = node.isConst;
element.isFinal = node.isFinal;
if (node.type == null) {
element.hasImplicitType = true;
element.type = _dynamicType;
} else {
node.type.accept(this);
element.type = node.type.type;
}
_setCodeRange(element, node);
}
@override
void visitDefaultFormalParameter(DefaultFormalParameter node) {
NormalFormalParameter normalParameter = node.parameter;
SimpleIdentifier nameNode = normalParameter.identifier;
ParameterElementImpl element;
if (_elementWalker != null) {
element = _elementWalker.getParameter();
} else {
element = DefaultParameterElementImpl.forNode(nameNode);
_elementHolder.addParameter(element);
_setCodeRange(element, node);
element.isConst = node.isConst;
element.isExplicitlyCovariant = node.parameter.covariantKeyword != null;
element.isFinal = node.isFinal;
// ignore: deprecated_member_use_from_same_package
element.parameterKind = node.kind;
if (normalParameter is SimpleFormalParameter &&
normalParameter.type == null) {
element.hasImplicitType = true;
}
}
if (normalParameter is SimpleFormalParameterImpl) {
normalParameter.declaredElement = element;
}
nameNode?.staticElement = element;
normalParameter.accept(this);
var defaultValue = node.defaultValue;
if (defaultValue != null) {
_withElementWalker(null, () {
var offset = defaultValue.offset;
var initializer = FunctionElementImpl.forOffset(offset);
element.initializer = initializer;
initializer.hasImplicitReturnType = true;
initializer.isSynthetic = true;
_withElementHolder(ElementHolder(initializer), () {
defaultValue.accept(this);
});
});
element.defaultValueCode = defaultValue.toSource();
}
}
@override
void visitEnumConstantDeclaration(EnumConstantDeclaration node) {
var element = _elementWalker.getVariable();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
}
@override
void visitEnumDeclaration(EnumDeclaration node) {
EnumElementImpl element = _elementWalker.getEnum();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forClass(element), () {
_withNameScope(() {
_defineElements(element.accessors);
node.constants.accept(this);
});
});
}
@override
void visitExportDirective(ExportDirective node) {
super.visitExportDirective(node);
if (node.element != null) {
_setElementAnnotations(node.metadata, node.element.metadata);
}
}
@override
void visitExtensionDeclaration(ExtensionDeclaration node) {
var element = _elementWalker.getExtension();
(node as ExtensionDeclarationImpl).declaredElement = element;
node.name?.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forExtension(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
node.extendedType.accept(this);
_defineElements(element.accessors);
_defineElements(element.methods);
node.members.accept(this);
});
});
}
@override
void visitFieldDeclaration(FieldDeclaration node) {
super.visitFieldDeclaration(node);
FieldElement firstFieldElement = node.fields.variables[0].declaredElement;
_setElementAnnotations(node.metadata, firstFieldElement.metadata);
}
@override
void visitFieldFormalParameter(FieldFormalParameter node) {
FieldFormalParameterElementImpl element;
if (node.parent is DefaultFormalParameter) {
element = node.declaredElement;
} else if (_elementWalker != null) {
element = _elementWalker.getParameter();
} else {
// Only for recovery, this should not happen in valid code.
element = FieldFormalParameterElementImpl.forNode(node.identifier);
_elementHolder.enclose(element);
element.isConst = node.isConst;
element.isExplicitlyCovariant = node.covariantKeyword != null;
element.isFinal = node.isFinal;
// ignore: deprecated_member_use_from_same_package
element.parameterKind = node.kind;
_setCodeRange(element, node);
}
node.identifier.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementHolder(ElementHolder(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
node.type?.accept(this);
if (_elementWalker != null) {
_withElementWalker(ElementWalker.forParameter(element), () {
node.parameters?.accept(this);
});
} else {
// Only for recovery, this should not happen in valid code.
element.type = node.type?.type ?? _dynamicType;
_withElementWalker(null, () {
node.parameters?.accept(this);
});
}
});
});
}
@override
void visitFunctionDeclaration(FunctionDeclaration node) {
ExecutableElementImpl element;
if (_elementWalker != null) {
element = node.isGetter || node.isSetter
? _elementWalker.getAccessor()
: _elementWalker.getFunction();
node.name.staticElement = element;
} else {
element = node.declaredElement;
_setCodeRange(element, node);
setElementDocumentationComment(element, node);
element.metadata = _createElementAnnotations(node.metadata);
FunctionBody body = node.functionExpression.body;
if (node.externalKeyword != null || body is NativeFunctionBody) {
element.external = true;
}
element.asynchronous = body.isAsynchronous;
element.generator = body.isGenerator;
if (node.returnType == null) {
element.hasImplicitReturnType = true;
}
}
FunctionExpressionImpl expression = node.functionExpression;
expression.declaredElement = element;
node.metadata?.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
var holder = ElementHolder(element);
_withElementHolder(holder, () {
_withElementWalker(
_elementWalker != null
? ElementWalker.forExecutable(element, _unitElement)
: null,
() {
_withNameScope(() {
_buildTypeParameterElements(expression.typeParameters);
expression.typeParameters?.accept(this);
if (_elementWalker == null) {
element.typeParameters = holder.typeParameters;
}
expression.parameters?.accept(this);
if (_elementWalker == null) {
element.parameters = holder.parameters;
}
node.returnType?.accept(this);
if (_elementWalker == null) {
element.returnType = node.returnType?.type ?? _dynamicType;
}
_defineParameters(element.parameters);
_withElementWalker(null, () {
expression.body.accept(this);
});
});
},
);
});
}
@override
void visitFunctionDeclarationStatement(FunctionDeclarationStatement node) {
if (!_hasLocalElementsBuilt(node)) {
_buildLocalFunctionElement(node);
}
node.functionDeclaration.accept(this);
}
@override
void visitFunctionExpression(FunctionExpression node) {
var element = FunctionElementImpl.forOffset(node.offset);
_elementHolder.enclose(element);
(node as FunctionExpressionImpl).declaredElement = element;
element.hasImplicitReturnType = true;
FunctionBody body = node.body;
element.asynchronous = body.isAsynchronous;
element.generator = body.isGenerator;
var holder = ElementHolder(element);
_withElementHolder(holder, () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
element.typeParameters = holder.typeParameters;
node.parameters.accept(this);
element.parameters = holder.parameters;
_defineParameters(element.parameters);
node.body.accept(this);
});
});
_setCodeRange(element, node);
}
@override
void visitFunctionTypeAlias(FunctionTypeAlias node) {
var element = _elementWalker.getTypedef();
node.name.staticElement = element;
node.metadata?.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forTypedef(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
node.returnType?.accept(this);
node.parameters?.accept(this);
});
});
}
@override
void visitFunctionTypedFormalParameter(FunctionTypedFormalParameter node) {
ParameterElementImpl element;
if (node.parent is DefaultFormalParameter) {
element = node.declaredElement;
} else {
SimpleIdentifier nameNode = node.identifier;
if (_elementWalker != null) {
element = _elementWalker.getParameter();
} else {
element = new ParameterElementImpl.forNode(nameNode);
_elementHolder.addParameter(element);
element.isConst = node.isConst;
element.isExplicitlyCovariant = node.covariantKeyword != null;
element.isFinal = node.isFinal;
// ignore: deprecated_member_use_from_same_package
element.parameterKind = node.kind;
_setCodeRange(element, node);
}
nameNode.staticElement = element;
}
node.metadata?.accept(this);
element.metadata = _createElementAnnotations(node.metadata);
var holder = ElementHolder(element);
_withElementHolder(holder, () {
_withElementWalker(
_elementWalker != null ? ElementWalker.forParameter(element) : null,
() {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
if (_elementWalker == null) {
element.typeParameters = holder.typeParameters;
}
node.parameters.accept(this);
if (_elementWalker == null) {
element.parameters = holder.parameters;
}
node.returnType?.accept(this);
if (_elementWalker == null) {
element.type = FunctionTypeImpl.synthetic(
node.returnType?.type ?? _dynamicType,
element.typeParameters,
element.parameters,
nullabilitySuffix: _getNullability(node.question != null),
);
}
});
},
);
});
}
@override
void visitGenericFunctionType(GenericFunctionType node) {
var element = GenericFunctionTypeElementImpl.forOffset(node.offset);
_unitElement.encloseElement(element);
(node as GenericFunctionTypeImpl).declaredElement = element;
_setCodeRange(element, node);
var holder = ElementHolder(element);
_withElementHolder(holder, () {
_withElementWalker(null, () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
element.typeParameters = holder.typeParameters;
node.parameters.accept(this);
element.parameters = holder.parameters;
node.returnType?.accept(this);
element.returnType = node.returnType?.type ?? _dynamicType;
});
});
});
var type = FunctionTypeImpl.synthetic(
element.returnType,
element.typeParameters,
element.parameters,
nullabilitySuffix: _getNullability(node.question != null),
);
element.type = type;
(node as GenericFunctionTypeImpl).type = type;
}
@override
void visitGenericTypeAlias(GenericTypeAlias node) {
var element = _elementWalker.getTypedef();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forGenericTypeAlias(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
node.functionType?.accept(this);
});
});
}
@override
void visitImportDirective(ImportDirective node) {
super.visitImportDirective(node);
if (node.element != null) {
_setElementAnnotations(node.metadata, node.element.metadata);
}
}
@override
void visitLabeledStatement(LabeledStatement node) {
bool onSwitchStatement = node.statement is SwitchStatement;
_buildLabelElements(node.labels, onSwitchStatement, false);
var outerScope = _labelScope;
try {
var unlabeled = node.unlabeled;
for (Label label in node.labels) {
SimpleIdentifier labelNameNode = label.label;
_labelScope = LabelScope(
_labelScope,
labelNameNode.name,
unlabeled,
labelNameNode.staticElement as LabelElement,
);
}
unlabeled.accept(this);
} finally {
_labelScope = outerScope;
}
}
@override
void visitLibraryDirective(LibraryDirective node) {
super.visitLibraryDirective(node);
if (node.element != null) {
_setElementAnnotations(node.metadata, node.element.metadata);
}
}
@override
void visitMethodDeclaration(MethodDeclaration node) {
ExecutableElementImpl element = node.isGetter || node.isSetter
? _elementWalker.getAccessor()
: _elementWalker.getFunction();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forExecutable(element, _unitElement), () {
node.metadata.accept(this);
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
node.parameters?.accept(this);
node.returnType?.accept(this);
_withElementWalker(null, () {
_withElementHolder(ElementHolder(element), () {
_defineParameters(element.parameters);
node.body?.accept(this);
});
});
});
});
}
@override
void visitMethodInvocation(MethodInvocation node) {
var newNode = _astRewriter.methodInvocation(_nameScope, node);
if (newNode != node) {
return newNode.accept(this);
}
super.visitMethodInvocation(node);
}
@override
void visitMixinDeclaration(MixinDeclaration node) {
var element = _elementWalker.getMixin();
node.name.staticElement = element;
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
_withElementWalker(ElementWalker.forClass(element), () {
_withNameScope(() {
_buildTypeParameterElements(node.typeParameters);
node.typeParameters?.accept(this);
_resolveOnClause(node.onClause);
_resolveImplementsClause(node.implementsClause);
_defineElements(element.accessors);
_defineElements(element.methods);
node.members.accept(this);
});
});
}
@override
void visitPartDirective(PartDirective node) {
super.visitPartDirective(node);
if (node.element != null) {
_setElementAnnotations(node.metadata, node.element.metadata);
}
}
@override
void visitSimpleFormalParameter(SimpleFormalParameter node) {
ParameterElementImpl element;
if (node.parent is DefaultFormalParameter) {
element = node.declaredElement;
} else {
SimpleIdentifier nameNode = node.identifier;
if (_elementWalker != null) {
element = _elementWalker.getParameter();
} else {
element = ParameterElementImpl.forNode(nameNode);
_elementHolder.addParameter(element);
_setCodeRange(element, node);
element.isConst = node.isConst;
element.isExplicitlyCovariant = node.covariantKeyword != null;
element.isFinal = node.isFinal;
// ignore: deprecated_member_use_from_same_package
element.parameterKind = node.kind;
if (node.type == null) {
element.hasImplicitType = true;
}
(node as SimpleFormalParameterImpl).declaredElement = element;
}
nameNode?.staticElement = element;
(node as SimpleFormalParameterImpl).declaredElement = element;
}
node.type?.accept(this);
if (_elementWalker == null) {
element.metadata = _createElementAnnotations(node.metadata);
element.type = node.type?.type ?? _dynamicType;
}
node.metadata.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
}
@override
void visitSuperExpression(SuperExpression node) {
_hasReferenceToSuper = true;
super.visitSuperExpression(node);
}
@override
void visitSwitchCase(SwitchCase node) {
_buildLabelElements(node.labels, false, true);
node.expression.accept(this);
_withNameScope(() {
var statements = node.statements;
_buildLocalElements(statements);
statements.accept(this);
});
}
@override
void visitSwitchDefault(SwitchDefault node) {
_buildLabelElements(node.labels, false, true);
_withNameScope(() {
var statements = node.statements;
_buildLocalElements(statements);
statements.accept(this);
});
}
@override
void visitTypeName(TypeName node) {
node.typeArguments?.accept(this);
_typeNameResolver.nameScope = _nameScope;
_typeNameResolver.resolveTypeName(node);
if (_typeNameResolver.rewriteResult != null) {
_typeNameResolver.rewriteResult.accept(this);
}
}
@override
void visitTypeParameter(TypeParameter node) {
TypeParameterElementImpl element = node.declaredElement;
node.metadata?.accept(this);
_setElementAnnotations(node.metadata, element.metadata);
var boundNode = node.bound;
if (boundNode != null) {
boundNode.accept(this);
if (_elementWalker == null) {
element.bound = boundNode.type;
element.metadata = _createElementAnnotations(node.metadata);
_setCodeRange(element, node);
}
}
}
@override
void visitVariableDeclaration(VariableDeclaration node) {
Expression initializerNode = node.initializer;
VariableElementImpl element;
if (_elementWalker != null) {
element = _elementWalker.getVariable();
node.name.staticElement = element;
} else {
LocalVariableElementImpl localElement = node.declaredElement;
element = localElement;
VariableDeclarationList varList = node.parent;
localElement.hasImplicitType = varList.type == null;
localElement.type = varList.type?.type ?? _dynamicType;
}
if (initializerNode != null) {
_withElementWalker(null, () {
var offset = initializerNode.offset;
var initializer = FunctionElementImpl.forOffset(offset);
element.initializer = initializer;
initializer.hasImplicitReturnType = true;
initializer.isSynthetic = true;
_withElementHolder(ElementHolder(initializer), () {
initializerNode.accept(this);
});
});
}
}
@override
void visitVariableDeclarationList(VariableDeclarationList node) {
var parent = node.parent;
if (parent is ForPartsWithDeclarations ||
parent is VariableDeclarationStatement &&
!_hasLocalElementsBuilt(parent)) {
_buildLocalVariableElements(node);
}
node.visitChildren(this);
List<ElementAnnotation> elementAnnotations;
if (parent is FieldDeclaration) {
elementAnnotations = _createElementAnnotations(parent.metadata);
} else if (parent is TopLevelVariableDeclaration) {
elementAnnotations = _createElementAnnotations(parent.metadata);
} else {
// Local variable declaration
elementAnnotations = _createElementAnnotations(node.metadata);
}
var variables = node.variables;
for (var i = 0; i < variables.length; i++) {
var variable = variables[i];
var element = variable.declaredElement as ElementImpl;
element.metadata = elementAnnotations;
var offset = (i == 0 ? node.parent : variable).offset;
var length = variable.end - offset;
element.setCodeRange(offset, length);
}
}
/// Builds the label elements associated with [labels] and stores them in the
/// element holder.
void _buildLabelElements(
List<Label> labels, bool onSwitchStatement, bool onSwitchMember) {
for (Label label in labels) {
var labelName = label.label;
var element = LabelElementImpl.forNode(
labelName, onSwitchStatement, onSwitchMember);
labelName.staticElement = element;
_elementHolder.enclose(element);
}
}
void _buildLocalElements(List<Statement> statements) {
for (var statement in statements) {
if (statement is FunctionDeclarationStatement) {
_buildLocalFunctionElement(statement);
} else if (statement is VariableDeclarationStatement) {
_buildLocalVariableElements(statement.variables);
}
}
}
void _buildLocalFunctionElement(FunctionDeclarationStatement statement) {
var node = statement.functionDeclaration;
var element = FunctionElementImpl.forNode(node.name);
node.name.staticElement = element;
_nameScope.define(element);
_elementHolder.enclose(element);
}
void _buildLocalVariableElements(VariableDeclarationList variableList) {
var isConst = variableList.isConst;
var isFinal = variableList.isFinal;
var isLate = variableList.isLate;
for (var variable in variableList.variables) {
var variableName = variable.name;
LocalVariableElementImpl element;
if (isConst && variable.initializer != null) {
element = ConstLocalVariableElementImpl.forNode(variableName);
} else {
element = LocalVariableElementImpl.forNode(variableName);
}
variableName.staticElement = element;
_elementHolder.enclose(element);
_nameScope.define(element);
element.isConst = isConst;
element.isFinal = isFinal;
element.isLate = isLate;
}
}
/// Ensure that each type parameters from the [typeParameterList] has its
/// element set, either from the [_elementWalker] or new, and define these
/// elements in the [_nameScope].
void _buildTypeParameterElements(TypeParameterList typeParameterList) {
if (typeParameterList == null) return;
for (var typeParameter in typeParameterList.typeParameters) {
var name = typeParameter.name;
TypeParameterElementImpl element;
if (_elementWalker != null) {
element = _elementWalker.getTypeParameter();
} else {
element = TypeParameterElementImpl.forNode(name);
_elementHolder.addTypeParameter(element);
}
name.staticElement = element;
_nameScope.define(element);
}
}
/**
* For each [Annotation] found in [annotations], create a new
* [ElementAnnotation] object and set the [Annotation] to point to it.
*/
List<ElementAnnotation> _createElementAnnotations(
List<Annotation> annotations) {
if (annotations.isEmpty) {
return const <ElementAnnotation>[];
}
return annotations.map((Annotation a) {
var elementAnnotation = ElementAnnotationImpl(_unitElement);
a.elementAnnotation = elementAnnotation;
return elementAnnotation;
}).toList();
}
/// Define given [elements] in the [_nameScope].
void _defineElements(List<Element> elements) {
int length = elements.length;
for (int i = 0; i < length; i++) {
var element = elements[i];
_nameScope.define(element);
}
}
/// Define given [parameters] in the [_nameScope].
void _defineParameters(List<ParameterElement> parameters) {
int length = parameters.length;
for (int i = 0; i < length; i++) {
ParameterElement parameter = parameters[i];
if (!parameter.isInitializingFormal) {
_nameScope.define(parameter);
}
}
}
NullabilitySuffix _getNullability(bool hasQuestion) {
NullabilitySuffix nullability;
if (_nonNullableEnabled) {
if (hasQuestion) {
nullability = NullabilitySuffix.question;
} else {
nullability = NullabilitySuffix.none;
}
} else {
nullability = NullabilitySuffix.star;
}
return nullability;
}
void _resolveImplementsClause(ImplementsClause clause) {
if (clause == null) return;
_resolveTypes(
clause.interfaces,
CompileTimeErrorCode.IMPLEMENTS_NON_CLASS,
);
}
void _resolveOnClause(OnClause clause) {
if (clause == null) return;
_resolveTypes(
clause.superclassConstraints,
CompileTimeErrorCode.MIXIN_SUPER_CLASS_CONSTRAINT_NON_INTERFACE,
);
}
/// Return the [InterfaceType] of the given [typeName].
///
/// If the resulting type is not a valid interface type, return `null`.
///
/// The flag [asClass] specifies if the type will be used as a class, so mixin
/// declarations are not valid (they declare interfaces and mixins, but not
/// classes).
void _resolveType(TypeName typeName, ErrorCode errorCode,
{bool asClass: false}) {
visitTypeName(typeName);
DartType type = typeName.type;
if (type is InterfaceType) {
ClassElement element = type.element;
if (element.isEnum || element.isMixin && asClass) {
_errorReporter.reportErrorForNode(errorCode, typeName);
return;
}
return;
}
// If the type is not an InterfaceType, then visitTypeName() sets the type
// to be a DynamicTypeImpl
Identifier name = typeName.name;
if (!_nameScope.shouldIgnoreUndefined(name)) {
_errorReporter.reportErrorForNode(errorCode, name, [name.name]);
}
}
/// Resolve the types in the given list of type names.
///
/// @param typeNames the type names to be resolved
/// @param nonTypeError the error to produce if the type name is defined to be
/// something other than a type
/// @param enumTypeError the error to produce if the type name is defined to
/// be an enum
/// @param dynamicTypeError the error to produce if the type name is "dynamic"
/// @return an array containing all of the types that were resolved.
void _resolveTypes(NodeList<TypeName> typeNames, ErrorCode errorCode) {
for (TypeName typeName in typeNames) {
_resolveType(typeName, errorCode);
}
}
void _resolveWithClause(WithClause clause) {
if (clause == null) return;
_resolveTypes(
clause.mixinTypes,
CompileTimeErrorCode.MIXIN_OF_NON_CLASS,
);
}
void _setCodeRange(ElementImpl element, AstNode node) {
element.setCodeRange(node.offset, node.length);
}
/// Make the given [holder] be the current one while running [f].
void _withElementHolder(ElementHolder holder, void Function() f) {
var previousHolder = _elementHolder;
_elementHolder = holder;
try {
f();
} finally {
_elementHolder = previousHolder;
}
}
/// Make the given [walker] be the current one while running [f].
void _withElementWalker(ElementWalker walker, void Function() f) {
var current = _elementWalker;
try {
_elementWalker = walker;
f();
} finally {
_elementWalker = current;
}
}
/// Run [f] with the new name scope.
void _withNameScope(void Function() f) {
var current = _nameScope;
try {
_nameScope = EnclosedScope(current);
f();
} finally {
_nameScope = current;
}
}
/// We always build local elements for [VariableDeclarationStatement]s and
/// [FunctionDeclarationStatement]s in blocks, because invalid code might try
/// to use forward references.
static bool _hasLocalElementsBuilt(Statement node) {
var parent = node.parent;
return parent is Block || parent is SwitchMember;
}
/// Associate each of the annotation [nodes] with the corresponding
/// [ElementAnnotation] in [annotations].
static void _setElementAnnotations(
List<Annotation> nodes, List<ElementAnnotation> annotations) {
int nodeCount = nodes.length;
if (nodeCount != annotations.length) {
throw StateError(
'Found $nodeCount annotation nodes and '
'${annotations.length} element annotations',
);
}
for (int i = 0; i < nodeCount; i++) {
nodes[i].elementAnnotation = annotations[i];
}
}
}