| // Copyright (c) 2020, 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.md file. |
| |
| import '../ast.dart'; |
| import '../core_types.dart'; |
| import '../visitor.dart'; |
| |
| class DartTypeEquivalence implements DartTypeVisitor1<bool, DartType> { |
| final CoreTypes coreTypes; |
| final bool equateTopTypes; |
| final bool ignoreAllNullabilities; |
| final bool ignoreTopLevelNullability; |
| |
| bool _atTopLevel = true; |
| List<Map<TypeParameter, TypeParameter>> _alphaRenamingStack = []; |
| |
| DartTypeEquivalence(this.coreTypes, |
| {this.equateTopTypes = false, |
| this.ignoreAllNullabilities = false, |
| this.ignoreTopLevelNullability = false}); |
| |
| bool areEqual(DartType type1, DartType type2) { |
| _alphaRenamingStack.clear(); |
| _atTopLevel = true; |
| return type1.accept1(this, type2); |
| } |
| |
| @override |
| bool defaultDartType(DartType node, DartType other) { |
| throw new UnsupportedError("${node.runtimeType}"); |
| } |
| |
| @override |
| bool visitBottomType(BottomType node, DartType other) { |
| return other is BottomType; |
| } |
| |
| @override |
| bool visitDynamicType(DynamicType node, DartType other) { |
| return equateTopTypes ? coreTypes.isTop(other) : other is DynamicType; |
| } |
| |
| @override |
| bool visitFunctionType(FunctionType node, DartType other) { |
| if (other is FunctionType) { |
| if (!_checkAndRegisterNullabilities( |
| node.declaredNullability, other.declaredNullability)) { |
| return false; |
| } |
| |
| // If the two types are un-aliased typedef instantiations, check that |
| // those instantiations are equal as well. |
| bool nodeIsUnaliased = node.typedefType != null; |
| bool otherIsUnaliased = other.typedefType != null; |
| if (nodeIsUnaliased != otherIsUnaliased) { |
| return false; |
| } |
| if (node.typedefType != null) { |
| // The assert below checks the implication: if the typedef types are |
| // equal, their un-aliased types are equal as well. The reverse is not |
| // true due to possibly unused type parameters F<int> and F<String> may |
| // be equal when un-aliased if the type parameter of typedef F isn't |
| // used on the right-hand side of the definition of F. The checked |
| // proposition in the assert allows to skip checking the function types |
| // themselves if the typedef types are equal. |
| assert(() { |
| DartTypeEquivalence copy = this.copy(); |
| if (!copy.areEqual(node.typedefType, other.typedefType)) { |
| return true; |
| } |
| FunctionType nodeWithoutTypedefType = new FunctionType( |
| node.positionalParameters, |
| node.returnType, |
| node.declaredNullability, |
| namedParameters: node.namedParameters, |
| typeParameters: node.typeParameters, |
| requiredParameterCount: node.requiredParameterCount, |
| typedefType: null); |
| FunctionType otherWithoutTypedefType = new FunctionType( |
| other.positionalParameters, |
| other.returnType, |
| other.declaredNullability, |
| namedParameters: other.namedParameters, |
| typeParameters: other.typeParameters, |
| requiredParameterCount: other.requiredParameterCount, |
| typedefType: null); |
| return copy.areEqual(nodeWithoutTypedefType, otherWithoutTypedefType); |
| }()); |
| return node.typedefType.accept1(this, other.typedefType); |
| } |
| |
| // Perform simple number checks before the checks on parts. |
| if (node.typeParameters.length != other.typeParameters.length) { |
| return false; |
| } |
| if (node.positionalParameters.length != |
| other.positionalParameters.length) { |
| return false; |
| } |
| if (node.requiredParameterCount != other.requiredParameterCount) { |
| return false; |
| } |
| if (node.namedParameters.length != other.namedParameters.length) { |
| return false; |
| } |
| |
| // Enter new static scope. The scope must be exited before a return. To |
| // void multiple returns, the [result] variable is used below. |
| _pushTypeParameters(node.typeParameters, other.typeParameters); |
| bool result = true; |
| |
| for (int i = 0; result && i < node.typeParameters.length; ++i) { |
| if (!node.typeParameters[i].bound |
| .accept1(this, other.typeParameters[i].bound)) { |
| result = false; |
| } |
| // Don't check defaultTypes: they are a convenience mechanism. |
| } |
| for (int i = 0; result && i < node.positionalParameters.length; ++i) { |
| if (!node.positionalParameters[i] |
| .accept1(this, other.positionalParameters[i])) { |
| result = false; |
| } |
| } |
| Map<String, DartType> nodeNamedParameters = {}; |
| for (int i = 0; i < node.namedParameters.length; ++i) { |
| nodeNamedParameters[node.namedParameters[i].name] = |
| node.namedParameters[i].type; |
| } |
| for (int i = 0; result && i < other.namedParameters.length; ++i) { |
| String otherName = other.namedParameters[i].name; |
| DartType otherType = other.namedParameters[i].type; |
| if (!nodeNamedParameters.containsKey(otherName) || |
| !nodeNamedParameters[otherName].accept1(this, otherType)) { |
| result = false; |
| } |
| } |
| if (!node.returnType.accept1(this, other.returnType)) { |
| result = false; |
| } |
| |
| _dropTypeParameters(); |
| return result; |
| } |
| return false; |
| } |
| |
| @override |
| bool visitInterfaceType(InterfaceType node, DartType other) { |
| // First, check Object*, Object?. |
| if (equateTopTypes && coreTypes.isTop(node)) { |
| return coreTypes.isTop(other); |
| } |
| |
| if (other is InterfaceType) { |
| if (!_checkAndRegisterNullabilities( |
| node.declaredNullability, other.declaredNullability)) { |
| return false; |
| } |
| if (node.classNode != other.classNode) { |
| return false; |
| } |
| assert(node.typeArguments.length == other.typeArguments.length); |
| for (int i = 0; i < node.typeArguments.length; ++i) { |
| if (!node.typeArguments[i].accept1(this, other.typeArguments[i])) { |
| return false; |
| } |
| } |
| return true; |
| } |
| return false; |
| } |
| |
| @override |
| bool visitFutureOrType(FutureOrType node, DartType other) { |
| // First, check FutureOr<dynamic>, FutureOr<Object?>, etc. |
| if (equateTopTypes && coreTypes.isTop(node)) { |
| return coreTypes.isTop(other); |
| } |
| |
| if (other is FutureOrType) { |
| if (!_checkAndRegisterNullabilities( |
| node.declaredNullability, other.declaredNullability)) { |
| return false; |
| } |
| if (!node.typeArgument.accept1(this, other.typeArgument)) { |
| return false; |
| } |
| return true; |
| } |
| return false; |
| } |
| |
| @override |
| bool visitInvalidType(InvalidType node, DartType other) { |
| return other is InvalidType; |
| } |
| |
| @override |
| bool visitNeverType(NeverType node, DartType other) { |
| if (other is NeverType) { |
| return _checkAndRegisterNullabilities( |
| node.declaredNullability, other.declaredNullability); |
| } |
| return false; |
| } |
| |
| @override |
| bool visitTypeParameterType(TypeParameterType node, DartType other) { |
| if (other is TypeParameterType) { |
| bool nodeIsIntersection = node.promotedBound != null; |
| bool otherIsIntersection = other.promotedBound != null; |
| if (nodeIsIntersection != otherIsIntersection) { |
| return false; |
| } |
| if (!_checkAndRegisterNullabilities( |
| node.declaredNullability, other.declaredNullability)) { |
| return false; |
| } |
| if (!identical(_lookup(node.parameter), other.parameter)) { |
| return false; |
| } |
| return nodeIsIntersection |
| ? node.promotedBound.accept1(this, other.promotedBound) |
| : true; |
| } |
| return false; |
| } |
| |
| @override |
| bool visitTypedefType(TypedefType node, DartType other) { |
| if (other is TypedefType) { |
| if (!_checkAndRegisterNullabilities( |
| node.declaredNullability, other.declaredNullability)) { |
| return false; |
| } |
| if (node.typedefNode != other.typedefNode) { |
| return false; |
| } |
| assert(node.typeArguments.length == other.typeArguments.length); |
| for (int i = 0; i < node.typeArguments.length; ++i) { |
| if (!node.typeArguments[i].accept1(this, other.typeArguments[i])) { |
| return false; |
| } |
| } |
| return true; |
| } |
| return false; |
| } |
| |
| @override |
| bool visitVoidType(VoidType node, DartType other) { |
| return equateTopTypes ? coreTypes.isTop(other) : other is VoidType; |
| } |
| |
| bool _checkAndRegisterNullabilities( |
| Nullability nodeNullability, Nullability otherNullability) { |
| bool result; |
| if (nodeNullability == otherNullability || |
| ignoreAllNullabilities || |
| ignoreTopLevelNullability && _atTopLevel) { |
| result = true; |
| } else { |
| result = false; |
| } |
| _atTopLevel = false; |
| return result; |
| } |
| |
| void _pushTypeParameters( |
| List<TypeParameter> keys, List<TypeParameter> values) { |
| assert(keys.length == values.length); |
| Map<TypeParameter, TypeParameter> parameters = |
| new Map<TypeParameter, TypeParameter>.identity(); |
| for (int i = 0; i < keys.length; ++i) { |
| parameters[keys[i]] = values[i]; |
| } |
| _alphaRenamingStack.add(parameters); |
| } |
| |
| void _dropTypeParameters() { |
| _alphaRenamingStack.removeLast(); |
| } |
| |
| TypeParameter _lookup(TypeParameter parameter) { |
| for (int i = _alphaRenamingStack.length - 1; i >= 0; --i) { |
| if (_alphaRenamingStack[i].containsKey(parameter)) { |
| return _alphaRenamingStack[i][parameter]; |
| } |
| } |
| return parameter; |
| } |
| |
| DartTypeEquivalence copy() { |
| return new DartTypeEquivalence(coreTypes, |
| equateTopTypes: equateTopTypes, |
| ignoreAllNullabilities: ignoreAllNullabilities, |
| ignoreTopLevelNullability: ignoreTopLevelNullability); |
| } |
| } |