| // Copyright (c) 2022, 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:js_shared/variance.dart'; |
| |
| import '../common.dart'; |
| import '../constants/constant_system.dart' as constant_system; |
| import '../constants/values.dart'; |
| import '../elements/entities.dart'; |
| import '../elements/names.dart'; |
| import '../elements/types.dart'; |
| import '../inferrer/abstract_value_domain.dart'; |
| import '../ir/element_map.dart'; |
| import '../js_backend/native_data.dart' show NativeBasicData; |
| import '../js_model/locals.dart'; |
| import '../universe/selector.dart' show Selector; |
| import '../universe/world_builder.dart'; |
| |
| import 'names.dart' show Identifiers, Uris; |
| |
| /// The common elements and types in Dart. |
| abstract class CommonElements { |
| final DartTypes dartTypes; |
| final ElementEnvironment _env; |
| ClassEntity? _symbolClass; |
| ConstructorEntity? _symbolConstructorTarget; |
| bool _computedSymbolConstructorDependencies = false; |
| ConstructorEntity? _symbolConstructorImplementationTarget; |
| ClassEntity? _mapLiteralClass; |
| ClassEntity? _symbolImplementationClass; |
| FieldEntity? _symbolImplementationField; |
| |
| CommonElements(this.dartTypes, this._env); |
| |
| /// The `Object` class defined in 'dart:core'. |
| late final ClassEntity objectClass = _findClass(coreLibrary, 'Object'); |
| |
| /// The `bool` class defined in 'dart:core'. |
| late final ClassEntity boolClass = _findClass(coreLibrary, 'bool'); |
| |
| /// The `num` class defined in 'dart:core'. |
| late final ClassEntity numClass = _findClass(coreLibrary, 'num'); |
| |
| /// The `int` class defined in 'dart:core'. |
| late final ClassEntity intClass = _findClass(coreLibrary, 'int'); |
| |
| /// The `double` class defined in 'dart:core'. |
| late final ClassEntity doubleClass = _findClass(coreLibrary, 'double'); |
| |
| /// The `String` class defined in 'dart:core'. |
| late final ClassEntity stringClass = _findClass(coreLibrary, 'String'); |
| |
| /// The `Function` class defined in 'dart:core'. |
| late final ClassEntity functionClass = _findClass(coreLibrary, 'Function'); |
| |
| /// The `Record` class defined in 'dart:core'. |
| late final ClassEntity recordClass = _findClass(coreLibrary, 'Record'); |
| |
| /// The `Resource` class defined in 'dart:core'. |
| late final ClassEntity resourceClass = _findClass(coreLibrary, 'Resource'); |
| |
| /// The `Symbol` class defined in 'dart:core'. |
| late final ClassEntity symbolClass = _findClass(coreLibrary, 'Symbol'); |
| |
| /// The `Null` class defined in 'dart:core'. |
| late final ClassEntity nullClass = _findClass(coreLibrary, 'Null'); |
| |
| /// The `Type` class defined in 'dart:core'. |
| late final ClassEntity typeClass = _findClass(coreLibrary, 'Type'); |
| |
| /// The `StackTrace` class defined in 'dart:core'; |
| late final ClassEntity stackTraceClass = _findClass( |
| coreLibrary, |
| 'StackTrace', |
| ); |
| |
| /// The `List` class defined in 'dart:core'; |
| late final ClassEntity listClass = _findClass(coreLibrary, 'List'); |
| |
| /// The `Set` class defined in 'dart:core'; |
| late final ClassEntity setClass = _findClass(coreLibrary, 'Set'); |
| |
| /// The `Map` class defined in 'dart:core'; |
| late final ClassEntity mapClass = _findClass(coreLibrary, 'Map'); |
| |
| /// The `Set` class defined in 'dart:core'; |
| late final ClassEntity unmodifiableSetClass = _findClass( |
| collectionLibrary, |
| '_UnmodifiableSet', |
| ); |
| |
| /// The `Iterable` class defined in 'dart:core'; |
| late final ClassEntity iterableClass = _findClass(coreLibrary, 'Iterable'); |
| |
| /// The `Future` class defined in 'async';. |
| late final ClassEntity futureClass = _findClass(asyncLibrary, 'Future'); |
| |
| /// The `Stream` class defined in 'async'; |
| late final ClassEntity streamClass = _findClass(asyncLibrary, 'Stream'); |
| |
| /// The dart:core library. |
| late final LibraryEntity coreLibrary = _env.lookupLibrary( |
| Uris.dartCore, |
| required: true, |
| )!; |
| |
| /// The dart:async library. |
| late final LibraryEntity? asyncLibrary = _env.lookupLibrary(Uris.dartAsync); |
| |
| /// The dart:collection library. |
| late final LibraryEntity? collectionLibrary = _env.lookupLibrary( |
| Uris.dartCollection, |
| ); |
| |
| /// The dart:mirrors library. |
| /// Null if the program doesn't access dart:mirrors. |
| late final LibraryEntity? mirrorsLibrary = _env.lookupLibrary( |
| Uris.dartMirrors, |
| ); |
| |
| /// The dart:typed_data library. |
| late final LibraryEntity typedDataLibrary = _env.lookupLibrary( |
| Uris.dartNativeTypedData, |
| required: true, |
| )!; |
| |
| /// The dart:_js_shared_embedded_names library. |
| late final LibraryEntity sharedEmbeddedNamesLibrary = _env.lookupLibrary( |
| Uris.dartJSSharedEmbeddedNames, |
| required: true, |
| )!; |
| |
| /// The dart:_js_helper library. |
| late final LibraryEntity? jsHelperLibrary = _env.lookupLibrary( |
| Uris.dartJSHelper, |
| ); |
| |
| /// The dart:_late_helper library |
| late final LibraryEntity? lateHelperLibrary = _env.lookupLibrary( |
| Uris.dartLateHelper, |
| ); |
| |
| /// The dart:_interceptors library. |
| late final LibraryEntity? interceptorsLibrary = _env.lookupLibrary( |
| Uris.dartInterceptors, |
| ); |
| |
| /// The dart:_foreign_helper library. |
| late final LibraryEntity? foreignLibrary = _env.lookupLibrary( |
| Uris.dartForeignHelper, |
| ); |
| |
| /// The dart:_rti library. |
| late final LibraryEntity rtiLibrary = _env.lookupLibrary( |
| Uris.dartRti, |
| required: true, |
| )!; |
| |
| /// The dart:_internal library. |
| late final LibraryEntity internalLibrary = _env.lookupLibrary( |
| Uris.dartInternal, |
| required: true, |
| )!; |
| |
| /// The dart:js_util library. |
| late final LibraryEntity? dartJsUtilLibrary = _env.lookupLibrary( |
| Uris.dartJSUtil, |
| ); |
| |
| /// The package:js library. |
| late final LibraryEntity? packageJsLibrary = _env.lookupLibrary( |
| Uris.packageJS, |
| ); |
| |
| /// The dart:_js_annotations library. |
| late final LibraryEntity? dartJsAnnotationsLibrary = _env.lookupLibrary( |
| Uris.dartJSAnnotations, |
| ); |
| |
| /// The dart:js_interop library. |
| late final LibraryEntity? dartJsInteropLibrary = _env.lookupLibrary( |
| Uris.dartJSInterop, |
| ); |
| |
| /// The `NativeTypedData` class from dart:typed_data. |
| ClassEntity get typedDataClass => |
| _findClass(typedDataLibrary, 'NativeTypedData'); |
| |
| /// Constructor of the `Symbol` class in dart:internal. |
| /// |
| /// This getter will ensure that `Symbol` is resolved and lookup the |
| /// constructor on demand. |
| ConstructorEntity get symbolConstructorTarget { |
| // TODO(johnniwinther): Kernel does not include redirecting factories |
| // so this cannot be found in kernel. Find a consistent way to handle |
| // this and similar cases. |
| return _symbolConstructorTarget ??= _env.lookupConstructor( |
| symbolImplementationClass, |
| '', |
| )!; |
| } |
| |
| void _ensureSymbolConstructorDependencies() { |
| if (_computedSymbolConstructorDependencies) return; |
| _computedSymbolConstructorDependencies = true; |
| if (_symbolConstructorTarget == null) { |
| _symbolImplementationClass ??= _findClassOrNull( |
| internalLibrary, |
| 'Symbol', |
| ); |
| if (_symbolImplementationClass != null) { |
| _symbolConstructorTarget = _env.lookupConstructor( |
| _symbolImplementationClass!, |
| '', |
| required: false, |
| ); |
| } |
| } |
| _symbolClass ??= _findClassOrNull(coreLibrary, 'Symbol'); |
| if (_symbolClass == null) { |
| return; |
| } |
| _symbolConstructorImplementationTarget = _env.lookupConstructor( |
| symbolClass, |
| '', |
| required: false, |
| ); |
| } |
| |
| /// Whether [element] is the same as [symbolConstructor]. |
| /// |
| /// Used to check for the constructor without computing it until it is likely |
| /// to be seen. |
| bool isSymbolConstructor(ConstructorEntity element) { |
| _ensureSymbolConstructorDependencies(); |
| return element == _symbolConstructorImplementationTarget || |
| element == _symbolConstructorTarget; |
| } |
| |
| /// The function `identical` in dart:core. |
| late final FunctionEntity identicalFunction = _findLibraryMember( |
| coreLibrary, |
| 'identical', |
| )!; |
| |
| /// Whether [element] is the `Function.apply` method. |
| /// |
| /// This will not resolve the apply method if it hasn't been seen yet during |
| /// compilation. |
| bool isFunctionApplyMethod(MemberEntity element) => |
| element.name == 'apply' && element.enclosingClass == functionClass; |
| |
| /// The `dynamic` type. |
| DynamicType get dynamicType => _env.dynamicType as DynamicType; |
| |
| /// The `Object` type defined in 'dart:core'. |
| InterfaceType get objectType => _getRawType(objectClass); |
| |
| /// The `bool` type defined in 'dart:core'. |
| InterfaceType get boolType => _getRawType(boolClass); |
| |
| /// The `num` type defined in 'dart:core'. |
| InterfaceType get numType => _getRawType(numClass); |
| |
| /// The `int` type defined in 'dart:core'. |
| InterfaceType get intType => _getRawType(intClass); |
| |
| /// The `double` type defined in 'dart:core'. |
| InterfaceType get doubleType => _getRawType(doubleClass); |
| |
| /// The `String` type defined in 'dart:core'. |
| InterfaceType get stringType => _getRawType(stringClass); |
| |
| /// The `Symbol` type defined in 'dart:core'. |
| InterfaceType get symbolType => _getRawType(symbolClass); |
| |
| /// The `Function` type defined in 'dart:core'. |
| InterfaceType get functionType => _getRawType(functionClass); |
| |
| /// The `Null` type defined in 'dart:core'. |
| InterfaceType get nullType => _getRawType(nullClass); |
| |
| /// The `Record` type defined in 'dart:core'. |
| InterfaceType get recordType => _getRawType(recordClass); |
| |
| /// The `Type` type defined in 'dart:core'. |
| InterfaceType get typeType => _getRawType(typeClass); |
| |
| InterfaceType get typeLiteralType => _getRawType(typeLiteralClass); |
| |
| /// The `StackTrace` type defined in 'dart:core'; |
| InterfaceType get stackTraceType => _getRawType(stackTraceClass); |
| |
| /// Returns an instance of the `List` type defined in 'dart:core' with |
| /// [elementType] as its type argument. |
| /// |
| /// If no type argument is provided, the canonical raw type is returned. |
| InterfaceType listType([DartType? elementType]) { |
| if (elementType == null) { |
| return _getRawType(listClass); |
| } |
| return _createInterfaceType(listClass, [elementType]); |
| } |
| |
| /// Returns an instance of the `Set` type defined in 'dart:core' with |
| /// [elementType] as its type argument. |
| /// |
| /// If no type argument is provided, the canonical raw type is returned. |
| InterfaceType setType([DartType? elementType]) { |
| if (elementType == null) { |
| return _getRawType(setClass); |
| } |
| return _createInterfaceType(setClass, [elementType]); |
| } |
| |
| /// Returns an instance of the `Map` type defined in 'dart:core' with |
| /// [keyType] and [valueType] as its type arguments. |
| /// |
| /// If no type arguments are provided, the canonical raw type is returned. |
| |
| InterfaceType mapType([DartType? keyType, DartType? valueType]) { |
| if (keyType == null && valueType == null) { |
| return _getRawType(mapClass); |
| } else if (keyType == null) { |
| keyType = dynamicType; |
| } else { |
| valueType ??= dynamicType; |
| } |
| return _createInterfaceType(mapClass, [keyType, valueType!]); |
| } |
| |
| /// Returns an instance of the `Iterable` type defined in 'dart:core' with |
| /// [elementType] as its type argument. |
| /// |
| /// If no type argument is provided, the canonical raw type is returned. |
| InterfaceType iterableType([DartType? elementType]) { |
| if (elementType == null) { |
| return _getRawType(iterableClass); |
| } |
| return _createInterfaceType(iterableClass, [elementType]); |
| } |
| |
| /// Returns an instance of the `Future` type defined in 'dart:async' with |
| /// [elementType] as its type argument. |
| /// |
| /// If no type argument is provided, the canonical raw type is returned. |
| // CommonElementsForDartTypes |
| InterfaceType futureType([DartType? elementType]) { |
| if (elementType == null) { |
| return _getRawType(futureClass); |
| } |
| return _createInterfaceType(futureClass, [elementType]); |
| } |
| |
| /// Returns an instance of the `Stream` type defined in 'dart:async' with |
| /// [elementType] as its type argument. |
| /// |
| /// If no type argument is provided, the canonical raw type is returned. |
| InterfaceType streamType([DartType? elementType]) { |
| if (elementType == null) { |
| return _getRawType(streamClass); |
| } |
| return _createInterfaceType(streamClass, [elementType]); |
| } |
| |
| ClassEntity _findClass(LibraryEntity? library, String name) { |
| return _env.lookupClass(library!, name, required: true)!; |
| } |
| |
| ClassEntity? _findClassOrNull(LibraryEntity? library, String name) { |
| if (library == null) return null; |
| return _env.lookupClass(library, name, required: false); |
| } |
| |
| T? _findLibraryMember<T extends MemberEntity>( |
| LibraryEntity? library, |
| String name, { |
| bool setter = false, |
| bool required = true, |
| }) { |
| if (library == null) return null; |
| return _env.lookupLibraryMember( |
| library, |
| name, |
| setter: setter, |
| required: required, |
| ) |
| as T?; |
| } |
| |
| T? _findClassMemberOrNull<T extends MemberEntity>( |
| ClassEntity cls, |
| String name, |
| ) { |
| return _env.lookupLocalClassMember( |
| cls, |
| Name(name, cls.library.canonicalUri, isSetter: false), |
| required: false, |
| ) |
| as T?; |
| } |
| |
| T _findClassMember<T extends MemberEntity>(ClassEntity cls, String name) { |
| return _env.lookupLocalClassMember( |
| cls, |
| Name(name, cls.library.canonicalUri, isSetter: false), |
| required: true, |
| ) |
| as T; |
| } |
| |
| /// Return the raw type of [cls]. |
| InterfaceType _getRawType(ClassEntity cls) { |
| return _env.getRawType(cls); |
| } |
| |
| /// Create the instantiation of [cls] with the given [typeArguments] and |
| /// [nullability]. |
| InterfaceType _createInterfaceType( |
| ClassEntity cls, |
| List<DartType> typeArguments, |
| ) { |
| return _env.createInterfaceType(cls, typeArguments); |
| } |
| |
| InterfaceType getConstantListTypeFor(InterfaceType sourceType) { |
| // TODO(51534): Use CONST_CANONICAL_TYPE(T_i) for arguments. |
| return _env.createInterfaceType(jsArrayClass, sourceType.typeArguments); |
| } |
| |
| InterfaceType getConstantMapTypeFor( |
| InterfaceType sourceType, { |
| bool onlyStringKeys = false, |
| }) { |
| // TODO(51534): Use CONST_CANONICAL_TYPE(T_i) for arguments. |
| ClassEntity classElement = onlyStringKeys |
| ? constantStringMapClass |
| : generalConstantMapClass; |
| return _env.createInterfaceType(classElement, sourceType.typeArguments); |
| } |
| |
| InterfaceType getConstantSetTypeFor( |
| InterfaceType sourceType, { |
| bool onlyStringKeys = false, |
| }) { |
| // TODO(51534): Use CONST_CANONICAL_TYPE(T_i) for arguments. |
| ClassEntity classElement = onlyStringKeys |
| ? constantStringSetClass |
| : generalConstantSetClass; |
| return _env.createInterfaceType(classElement, sourceType.typeArguments); |
| } |
| |
| /// Returns the field that holds the internal name in the implementation class |
| /// for `Symbol`. |
| |
| FieldEntity get symbolField => _symbolImplementationField ??= |
| _env.lookupLocalClassMember( |
| symbolImplementationClass, |
| PrivateName( |
| '_name', |
| symbolImplementationClass.library.canonicalUri, |
| ), |
| required: true, |
| ) |
| as FieldEntity; |
| |
| InterfaceType get symbolImplementationType => |
| _env.getRawType(symbolImplementationClass); |
| |
| // From dart:core |
| ClassEntity get mapLiteralClass { |
| if (_mapLiteralClass == null) { |
| _mapLiteralClass = _env.lookupClass(coreLibrary, 'LinkedHashMap'); |
| _mapLiteralClass ??= _findClass(collectionLibrary, 'LinkedHashMap'); |
| } |
| return _mapLiteralClass!; |
| } |
| |
| late final ConstructorEntity mapLiteralConstructor = _env.lookupConstructor( |
| mapLiteralClass, |
| '_literal', |
| )!; |
| late final ConstructorEntity mapLiteralConstructorEmpty = _env |
| .lookupConstructor(mapLiteralClass, '_empty')!; |
| |
| late final ClassEntity setLiteralClass = _findClass( |
| collectionLibrary, |
| 'LinkedHashSet', |
| ); |
| |
| late final ConstructorEntity setLiteralConstructor = _env.lookupConstructor( |
| setLiteralClass, |
| '_literal', |
| )!; |
| late final ConstructorEntity setLiteralConstructorEmpty = _env |
| .lookupConstructor(setLiteralClass, '_empty')!; |
| |
| late final FunctionEntity? objectNoSuchMethod = |
| _env.lookupLocalClassMember( |
| objectClass, |
| const PublicName(Identifiers.noSuchMethod_), |
| ) |
| as FunctionEntity?; |
| |
| bool isDefaultNoSuchMethodImplementation(FunctionEntity element) { |
| ClassEntity? classElement = element.enclosingClass; |
| return classElement == objectClass || |
| classElement == jsInterceptorClass || |
| classElement == jsNullClass; |
| } |
| |
| // From dart:async |
| ClassEntity _findAsyncHelperClass(String name) => |
| _findClass(asyncLibrary, name); |
| |
| FunctionEntity _findAsyncHelperFunction(String name) => |
| _findLibraryMember(asyncLibrary, name)!; |
| |
| FunctionEntity get asyncHelperStartSync => |
| _findAsyncHelperFunction("_asyncStartSync"); |
| |
| FunctionEntity get asyncHelperAwait => |
| _findAsyncHelperFunction("_asyncAwait"); |
| |
| FunctionEntity get asyncHelperReturn => |
| _findAsyncHelperFunction("_asyncReturn"); |
| |
| FunctionEntity get asyncHelperRethrow => |
| _findAsyncHelperFunction("_asyncRethrow"); |
| |
| FunctionEntity get wrapBody => |
| _findAsyncHelperFunction("_wrapJsFunctionForAsync"); |
| |
| FunctionEntity get yieldStar => |
| _env.lookupLocalClassMember( |
| _findAsyncHelperClass("_IterationMarker"), |
| const PublicName("yieldStar"), |
| ) |
| as FunctionEntity; |
| |
| FunctionEntity get yieldSingle => |
| _env.lookupLocalClassMember( |
| _findAsyncHelperClass("_IterationMarker"), |
| const PublicName("yieldSingle"), |
| ) |
| as FunctionEntity; |
| |
| FunctionEntity get syncStarUncaughtError => |
| _env.lookupLocalClassMember( |
| _findAsyncHelperClass("_IterationMarker"), |
| const PublicName("uncaughtError"), |
| ) |
| as FunctionEntity; |
| |
| FunctionEntity get asyncStarHelper => |
| _findAsyncHelperFunction("_asyncStarHelper"); |
| |
| FunctionEntity get streamOfController => |
| _findAsyncHelperFunction("_streamOfController"); |
| |
| FunctionEntity get endOfIteration => |
| _env.lookupLocalClassMember( |
| _findAsyncHelperClass("_IterationMarker"), |
| const PublicName("endOfIteration"), |
| ) |
| as FunctionEntity; |
| |
| ClassEntity get syncStarIterable => |
| _findAsyncHelperClass("_SyncStarIterable"); |
| |
| late final ClassEntity _syncStarIteratorClass = _findAsyncHelperClass( |
| '_SyncStarIterator', |
| ); |
| |
| late final FieldEntity syncStarIteratorCurrentField = _findClassMember( |
| _syncStarIteratorClass, |
| '_current', |
| ); |
| |
| late final FieldEntity syncStarIteratorDatumField = _findClassMember( |
| _syncStarIteratorClass, |
| '_datum', |
| ); |
| |
| late final FunctionEntity syncStarIteratorYieldStarMethod = _findClassMember( |
| _syncStarIteratorClass, |
| '_yieldStar', |
| ); |
| |
| ClassEntity get futureImplementation => _findAsyncHelperClass('_Future'); |
| |
| ClassEntity get controllerStream => |
| _findAsyncHelperClass("_ControllerStream"); |
| |
| ClassEntity get streamIterator => _findAsyncHelperClass("StreamIterator"); |
| |
| ConstructorEntity get streamIteratorConstructor => |
| _env.lookupConstructor(streamIterator, "")!; |
| |
| late final FunctionEntity syncStarIterableFactory = _findAsyncHelperFunction( |
| '_makeSyncStarIterable', |
| ); |
| |
| late final FunctionEntity asyncAwaitCompleterFactory = |
| _findAsyncHelperFunction('_makeAsyncAwaitCompleter'); |
| |
| late final FunctionEntity asyncStarStreamControllerFactory = |
| _findAsyncHelperFunction('_makeAsyncStarStreamController'); |
| |
| // From dart:_interceptors |
| ClassEntity _findInterceptorsClass(String name) => |
| _findClass(interceptorsLibrary, name); |
| |
| FunctionEntity _findInterceptorsFunction(String name) => |
| _findLibraryMember(interceptorsLibrary, name)!; |
| |
| late final ClassEntity jsInterceptorClass = _findInterceptorsClass( |
| 'Interceptor', |
| ); |
| |
| late final ClassEntity jsStringClass = _findInterceptorsClass('JSString'); |
| |
| late final ClassEntity jsArrayClass = _findInterceptorsClass('JSArray'); |
| |
| late final ClassEntity jsNumberClass = _findInterceptorsClass('JSNumber'); |
| |
| late final ClassEntity jsIntClass = _findInterceptorsClass('JSInt'); |
| |
| late final ClassEntity jsNumNotIntClass = _findInterceptorsClass( |
| 'JSNumNotInt', |
| ); |
| |
| late final ClassEntity jsNullClass = _findInterceptorsClass('JSNull'); |
| |
| late final ClassEntity jsBoolClass = _findInterceptorsClass('JSBool'); |
| |
| late final ClassEntity jsPlainJavaScriptObjectClass = _findInterceptorsClass( |
| 'PlainJavaScriptObject', |
| ); |
| |
| late final ClassEntity jsUnknownJavaScriptObjectClass = |
| _findInterceptorsClass('UnknownJavaScriptObject'); |
| |
| late final ClassEntity jsJavaScriptBigIntClass = _findInterceptorsClass( |
| 'JavaScriptBigInt', |
| ); |
| |
| late final ClassEntity jsJavaScriptFunctionClass = _findInterceptorsClass( |
| 'JavaScriptFunction', |
| ); |
| |
| late final ClassEntity jsJavaScriptSymbolClass = _findInterceptorsClass( |
| 'JavaScriptSymbol', |
| ); |
| |
| InterfaceType get jsJavaScriptFunctionType => |
| _getRawType(jsJavaScriptFunctionClass); |
| |
| late final ClassEntity jsLegacyJavaScriptObjectClass = _findInterceptorsClass( |
| 'LegacyJavaScriptObject', |
| ); |
| |
| late final ClassEntity jsJavaScriptObjectClass = _findInterceptorsClass( |
| 'JavaScriptObject', |
| ); |
| |
| InterfaceType get jsJavaScriptObjectType => |
| _getRawType(jsJavaScriptObjectClass); |
| |
| late final ClassEntity jsIndexableClass = _findInterceptorsClass( |
| 'JSIndexable', |
| ); |
| |
| late final ClassEntity jsMutableIndexableClass = _findInterceptorsClass( |
| 'JSMutableIndexable', |
| ); |
| |
| late final ClassEntity jsMutableArrayClass = _findInterceptorsClass( |
| 'JSMutableArray', |
| ); |
| |
| late final ClassEntity jsFixedArrayClass = _findInterceptorsClass( |
| 'JSFixedArray', |
| ); |
| |
| late final ClassEntity jsExtendableArrayClass = _findInterceptorsClass( |
| 'JSExtendableArray', |
| ); |
| |
| late final ClassEntity jsUnmodifiableArrayClass = _findInterceptorsClass( |
| 'JSUnmodifiableArray', |
| ); |
| |
| late final ClassEntity jsPositiveIntClass = _findInterceptorsClass( |
| 'JSPositiveInt', |
| ); |
| |
| late final ClassEntity jsUInt32Class = _findInterceptorsClass('JSUInt32'); |
| |
| late final ClassEntity jsUInt31Class = _findInterceptorsClass('JSUInt31'); |
| |
| /// Returns `true` member is the 'findIndexForNativeSubclassType' method |
| /// declared in `dart:_interceptors`. |
| bool isFindIndexForNativeSubclassType(MemberEntity member) { |
| return member.name == 'findIndexForNativeSubclassType' && |
| member.isTopLevel && |
| member.library == interceptorsLibrary; |
| } |
| |
| late final FunctionEntity getNativeInterceptorMethod = |
| _findInterceptorsFunction('getNativeInterceptor'); |
| |
| late final ConstructorEntity jsArrayTypedConstructor = _env.lookupConstructor( |
| jsArrayClass, |
| 'typed', |
| )!; |
| |
| // From dart:_js_helper |
| // TODO(johnniwinther): Avoid the need for this (from [CheckedModeHelper]). |
| FunctionEntity findHelperFunction(String name) => _findHelperFunction(name); |
| |
| FunctionEntity _findHelperFunction(String name) => |
| _findLibraryMember(jsHelperLibrary, name)!; |
| |
| ClassEntity _findHelperClass(String name) => |
| _findClass(jsHelperLibrary, name); |
| |
| FunctionEntity _findLateHelperFunction(String name) => |
| _findLibraryMember(lateHelperLibrary, name)!; |
| |
| late final ClassEntity closureClass = _findHelperClass('Closure'); |
| |
| late final ClassEntity closureClass0Args = _findHelperClass('Closure0Args'); |
| |
| late final ClassEntity closureClass2Args = _findHelperClass('Closure2Args'); |
| |
| late final ClassEntity boundClosureClass = _findHelperClass('BoundClosure'); |
| |
| late final ClassEntity typeLiteralClass = _findRtiClass('_Type'); |
| |
| late final ClassEntity constMapLiteralClass = _findHelperClass('ConstantMap'); |
| |
| late final ClassEntity constSetLiteralClass = _findHelperClass('ConstantSet'); |
| |
| /// Base class for all records. |
| late final ClassEntity recordBaseClass = _findHelperClass('_Record'); |
| |
| /// A function that is used to model the back-end impacts of record lowering |
| /// in the front-end. |
| late final FunctionEntity recordImpactModel = _findHelperFunction( |
| '_recordImpactModel', |
| ); |
| |
| /// Base class for records with N fields. Can be a fixed-arity class or a |
| /// general class that works for any arity. |
| ClassEntity recordArityClass(int n) { |
| return _findClassOrNull(jsHelperLibrary, '_Record$n') ?? |
| (n == 0 ? emptyRecordClass : recordGeneralBaseClass); |
| } |
| |
| late final ClassEntity recordGeneralBaseClass = _findHelperClass('_RecordN'); |
| |
| late final ClassEntity emptyRecordClass = _findHelperClass('_EmptyRecord'); |
| |
| late final FunctionEntity recordTestByListHelper = _findHelperFunction( |
| '_testRecordValues', |
| ); |
| |
| late final ClassEntity jsInvocationMirrorClass = _findHelperClass( |
| 'JSInvocationMirror', |
| ); |
| |
| late final ClassEntity requiredSentinelClass = _findHelperClass('_Required'); |
| |
| InterfaceType get requiredSentinelType => _getRawType(requiredSentinelClass); |
| |
| late final MemberEntity invocationTypeArgumentGetter = _findClassMember( |
| jsInvocationMirrorClass, |
| 'typeArguments', |
| ); |
| |
| /// Interface used to determine if an object has the JavaScript |
| /// indexing behavior. The interface is only visible to specific libraries. |
| late final ClassEntity jsIndexingBehaviorInterface = _findHelperClass( |
| 'JavaScriptIndexingBehavior', |
| ); |
| |
| late final ClassEntity stackTraceHelperClass = _findHelperClass( |
| '_StackTrace', |
| ); |
| |
| late final ClassEntity constantMapClass = _findHelperClass( |
| constant_system.JavaScriptMapConstant.dartClass, |
| ); |
| |
| late final ClassEntity constantStringMapClass = _findHelperClass( |
| constant_system.JavaScriptMapConstant.dartStringClass, |
| ); |
| |
| late final ClassEntity generalConstantMapClass = _findHelperClass( |
| constant_system.JavaScriptMapConstant.dartGeneralClass, |
| ); |
| |
| late final ClassEntity constantStringSetClass = _findHelperClass( |
| constant_system.JavaScriptSetConstant.dartStringClass, |
| ); |
| |
| late final ClassEntity generalConstantSetClass = _findHelperClass( |
| constant_system.JavaScriptSetConstant.dartGeneralClass, |
| ); |
| |
| late final ClassEntity annotationCreatesClass = _findHelperClass('Creates'); |
| |
| late final ClassEntity annotationReturnsClass = _findHelperClass('Returns'); |
| |
| late final ClassEntity annotationJSNameClass = _findHelperClass('JSName'); |
| |
| /// The class for native annotations defined in dart:_js_helper. |
| late final ClassEntity nativeAnnotationClass = _findHelperClass('Native'); |
| |
| bool isAssertTest(MemberEntity member) => |
| member.name == 'assertTest' && |
| member.isFunction && |
| member.isTopLevel && |
| member.library == jsHelperLibrary; |
| |
| late final assertTest = _findHelperFunction('assertTest'); |
| |
| late final assertThrow = _findHelperFunction('assertThrow'); |
| |
| bool isAssertHelper(MemberEntity member) => |
| member.name == 'assertHelper' && |
| member.isFunction && |
| member.isTopLevel && |
| member.library == jsHelperLibrary; |
| |
| late final assertHelper = _findHelperFunction('assertHelper'); |
| |
| late final assertUnreachableMethod = _findHelperFunction('assertUnreachable'); |
| |
| /// Holds the method "getIsolateAffinityTag" when dart:_js_helper has been |
| /// loaded. |
| late final getIsolateAffinityTagMarker = _findHelperFunction( |
| 'getIsolateAffinityTag', |
| ); |
| |
| /// Holds the method "requiresPreamble" in _js_helper. |
| late final requiresPreambleMarker = _findHelperFunction('requiresPreamble'); |
| |
| /// Holds the method "_rawStartupMetrics" in _js_helper. |
| late final rawStartupMetrics = _findHelperFunction('rawStartupMetrics'); |
| |
| FunctionEntity get loadDeferredLibrary => |
| _findHelperFunction("loadDeferredLibrary"); |
| |
| FunctionEntity get traceHelper => _findHelperFunction('traceHelper'); |
| |
| FunctionEntity get closureFromTearOff => |
| _findHelperFunction('closureFromTearOff'); |
| |
| FunctionEntity get isJsIndexable => _findHelperFunction('isJsIndexable'); |
| |
| FunctionEntity get throwIllegalArgumentException => |
| _findHelperFunction('iae'); |
| |
| FunctionEntity get throwIndexOutOfRangeException => |
| _findHelperFunction('ioore'); |
| |
| FunctionEntity get exceptionUnwrapper => |
| _findHelperFunction('unwrapException'); |
| |
| FunctionEntity get throwUnsupportedError => |
| _findHelperFunction('throwUnsupportedError'); |
| |
| FunctionEntity get throwTypeError => _findRtiFunction('throwTypeError'); |
| |
| /// Recognizes the `checkConcurrentModificationError` helper without needing |
| /// it to be resolved. |
| bool isCheckConcurrentModificationError(MemberEntity member) { |
| return member.name == 'checkConcurrentModificationError' && |
| member.isFunction && |
| member.isTopLevel && |
| member.library == jsHelperLibrary; |
| } |
| |
| late final FunctionEntity checkConcurrentModificationError = |
| _findHelperFunction('checkConcurrentModificationError'); |
| |
| FunctionEntity get throwConcurrentModificationError => |
| _findHelperFunction('throwConcurrentModificationError'); |
| |
| FunctionEntity get stringInterpolationHelper => _findHelperFunction('S'); |
| |
| FunctionEntity get initializeExceptionWrapper => |
| _findHelperFunction('initializeExceptionWrapper'); |
| |
| FunctionEntity get wrapExceptionHelper => |
| _findHelperFunction('wrapException'); |
| |
| FunctionEntity get throwExpressionHelper => |
| _findHelperFunction('throwExpression'); |
| |
| FunctionEntity get closureConverter => |
| _findHelperFunction('convertDartClosureToJS'); |
| |
| FunctionEntity get traceFromException => |
| _findHelperFunction('getTraceFromException'); |
| |
| FunctionEntity get checkDeferredIsLoaded => |
| _findHelperFunction('checkDeferredIsLoaded'); |
| |
| FunctionEntity get createRuntimeType => _findRtiFunction('createRuntimeType'); |
| |
| FunctionEntity get createInvocationMirror => |
| _findHelperFunction('createInvocationMirror'); |
| |
| FunctionEntity get createUnmangledInvocationMirror => |
| _findHelperFunction('createUnmangledInvocationMirror'); |
| |
| FunctionEntity get throwUnsupportedOperation => |
| _findHelperFunction('throwUnsupportedOperation'); |
| |
| FunctionEntity get defineProperty => _findHelperFunction('defineProperty'); |
| |
| FunctionEntity get throwLateFieldNI => |
| _findLateHelperFunction('throwLateFieldNI'); |
| |
| FunctionEntity get throwLateFieldAI => |
| _findLateHelperFunction('throwLateFieldAI'); |
| |
| FunctionEntity get throwLateFieldADI => |
| _findLateHelperFunction('throwLateFieldADI'); |
| |
| FunctionEntity get throwUnnamedLateFieldNI => |
| _findLateHelperFunction('throwUnnamedLateFieldNI'); |
| |
| FunctionEntity get throwUnnamedLateFieldAI => |
| _findLateHelperFunction('throwUnnamedLateFieldAI'); |
| |
| FunctionEntity get throwUnnamedLateFieldADI => |
| _findLateHelperFunction('throwUnnamedLateFieldADI'); |
| |
| bool isExtractTypeArguments(FunctionEntity member) { |
| return member.name == 'extractTypeArguments' && |
| member.library == internalLibrary; |
| } |
| |
| // TODO(johnniwinther,sra): Support arbitrary type argument count. |
| void _checkTypeArgumentCount(int typeArgumentCount) { |
| assert(typeArgumentCount > 0); |
| if (typeArgumentCount > 20) { |
| failedAt( |
| noLocationSpannable, |
| "Unsupported instantiation argument count: " |
| "$typeArgumentCount", |
| ); |
| } |
| } |
| |
| ClassEntity getInstantiationClass(int typeArgumentCount) { |
| _checkTypeArgumentCount(typeArgumentCount); |
| return _findHelperClass('Instantiation$typeArgumentCount'); |
| } |
| |
| FunctionEntity getInstantiateFunction(int typeArgumentCount) { |
| _checkTypeArgumentCount(typeArgumentCount); |
| return _findHelperFunction('instantiate$typeArgumentCount'); |
| } |
| |
| FunctionEntity get convertMainArgumentList => |
| _findHelperFunction('convertMainArgumentList'); |
| |
| // From dart:_rti |
| |
| ClassEntity _findRtiClass(String name) => _findClass(rtiLibrary, name); |
| |
| FunctionEntity _findRtiFunction(String name) => |
| _findLibraryMember(rtiLibrary, name)!; |
| |
| late final FunctionEntity interopNullAssertion = _findRtiFunction( |
| '_interopNullAssertion', |
| ); |
| |
| late final FunctionEntity setArrayType = _findRtiFunction('_setArrayType'); |
| |
| late final FunctionEntity findType = _findRtiFunction('findType'); |
| |
| late final FunctionEntity instanceType = _findRtiFunction('instanceType'); |
| |
| late final FunctionEntity arrayInstanceType = _findRtiFunction( |
| '_arrayInstanceType', |
| ); |
| |
| late final FunctionEntity simpleInstanceType = _findRtiFunction( |
| '_instanceType', |
| ); |
| |
| late final FunctionEntity typeLiteralMaker = _findRtiFunction('typeLiteral'); |
| |
| late final FunctionEntity checkTypeBound = _findRtiFunction('checkTypeBound'); |
| |
| late final FunctionEntity pairwiseIsTest = _findRtiFunction('pairwiseIsTest'); |
| |
| ClassEntity get _rtiImplClass => _findRtiClass('Rti'); |
| |
| ClassEntity get _rtiUniverseClass => _findRtiClass('_Universe'); |
| |
| FieldEntity _findRtiClassField(String name) => |
| _findClassMember(_rtiImplClass, name); |
| |
| late final FieldEntity rtiAsField = _findRtiClassField('_as'); |
| |
| late final FieldEntity rtiIsField = _findRtiClassField('_is'); |
| |
| late final FieldEntity rtiRestField = _findRtiClassField('_rest'); |
| |
| late final FieldEntity rtiPrecomputed1Field = _findRtiClassField( |
| '_precomputed1', |
| ); |
| |
| late final FunctionEntity rtiEvalMethod = _findClassMember( |
| _rtiImplClass, |
| '_eval', |
| ); |
| |
| late final FunctionEntity rtiBindMethod = _findClassMember( |
| _rtiImplClass, |
| '_bind', |
| ); |
| |
| late final FunctionEntity rtiAddRulesMethod = _findClassMember( |
| _rtiUniverseClass, |
| 'addRules', |
| ); |
| |
| late final FunctionEntity rtiAddErasedTypesMethod = _findClassMember( |
| _rtiUniverseClass, |
| 'addErasedTypes', |
| ); |
| |
| late final FunctionEntity rtiAddTypeParameterVariancesMethod = |
| _findClassMember(_rtiUniverseClass, 'addTypeParameterVariances'); |
| |
| FunctionEntity get installSpecializedIsTest => |
| _findRtiFunction('_installSpecializedIsTest'); |
| |
| FunctionEntity get installSpecializedAsCheck => |
| _findRtiFunction('_installSpecializedAsCheck'); |
| |
| late final FunctionEntity generalIsTestImplementation = _findRtiFunction( |
| '_generalIsTestImplementation', |
| ); |
| |
| late final FunctionEntity generalNullableIsTestImplementation = |
| _findRtiFunction('_generalNullableIsTestImplementation'); |
| |
| late final FunctionEntity generalAsCheckImplementation = _findRtiFunction( |
| '_generalAsCheckImplementation', |
| ); |
| |
| late final FunctionEntity generalNullableAsCheckImplementation = |
| _findRtiFunction('_generalNullableAsCheckImplementation'); |
| |
| late final FunctionEntity specializedIsObject = _findRtiFunction('_isObject'); |
| |
| late final FunctionEntity specializedAsObject = _findRtiFunction('_asObject'); |
| |
| FunctionEntity get specializedIsTop => _findRtiFunction('_isTop'); |
| |
| FunctionEntity get specializedAsTop => _findRtiFunction('_asTop'); |
| |
| FunctionEntity get specializedIsBool => _findRtiFunction('_isBool'); |
| |
| FunctionEntity get specializedAsBool => _findRtiFunction('_asBool'); |
| |
| FunctionEntity get specializedAsBoolNullable => _findRtiFunction('_asBoolQ'); |
| |
| FunctionEntity get specializedAsDouble => _findRtiFunction('_asDouble'); |
| |
| FunctionEntity get specializedAsDoubleNullable => |
| _findRtiFunction('_asDoubleQ'); |
| |
| FunctionEntity get specializedIsInt => _findRtiFunction('_isInt'); |
| |
| FunctionEntity get specializedAsInt => _findRtiFunction('_asInt'); |
| |
| FunctionEntity get specializedAsIntNullable => _findRtiFunction('_asIntQ'); |
| |
| FunctionEntity get specializedIsNum => _findRtiFunction('_isNum'); |
| |
| FunctionEntity get specializedAsNum => _findRtiFunction('_asNum'); |
| |
| FunctionEntity get specializedAsNumNullable => _findRtiFunction('_asNumQ'); |
| |
| FunctionEntity get specializedIsString => _findRtiFunction('_isString'); |
| |
| FunctionEntity get specializedAsString => _findRtiFunction('_asString'); |
| |
| FunctionEntity get specializedAsStringNullable => |
| _findRtiFunction('_asStringQ'); |
| |
| FunctionEntity get instantiatedGenericFunctionType => |
| _findRtiFunction('instantiatedGenericFunctionType'); |
| |
| FunctionEntity get closureFunctionType => |
| _findRtiFunction('closureFunctionType'); |
| |
| // From dart:_internal |
| |
| late final ClassEntity symbolImplementationClass = _findClass( |
| internalLibrary, |
| 'Symbol', |
| ); |
| |
| /// Used to annotate items that have the keyword "native". |
| late final ClassEntity externalNameClass = _findClass( |
| internalLibrary, |
| 'ExternalName', |
| ); |
| |
| InterfaceType get externalNameType => _getRawType(externalNameClass); |
| |
| // From dart:_js_embedded_names |
| |
| late final ClassEntity jsGetNameEnum = _findClass( |
| sharedEmbeddedNamesLibrary, |
| 'JsGetName', |
| ); |
| |
| /// Returns `true` if [member] is a "foreign helper", that is, a member whose |
| /// semantics is defined synthetically and not through Dart code. |
| /// |
| /// Most foreign helpers are located in the `dart:_foreign_helper` library. |
| bool isForeignHelper(MemberEntity member) { |
| return member.library == foreignLibrary || |
| isLateReadCheck(member) || |
| isLateWriteOnceCheck(member) || |
| isLateInitializeOnceCheck(member) || |
| isCreateInvocationMirrorHelper(member); |
| } |
| |
| bool _isTopLevelFunctionNamed(String name, MemberEntity member) => |
| member.name == name && member.isFunction && member.isTopLevel; |
| |
| /// Returns `true` if [member] is the `createJsSentinel` function defined in |
| /// dart:_foreign_helper. |
| bool isCreateJsSentinel(MemberEntity member) => |
| member.library == foreignLibrary && |
| _isTopLevelFunctionNamed('createJsSentinel', member); |
| |
| /// Returns `true` if [member] is the `isJsSentinel` function defined in |
| /// dart:_foreign_helper. |
| bool isIsJsSentinel(MemberEntity member) => |
| member.library == foreignLibrary && |
| _isTopLevelFunctionNamed('isJsSentinel', member); |
| |
| /// Returns `true` if [member] is the `_lateReadCheck` function defined in |
| /// dart:_late_helper. |
| bool isLateReadCheck(MemberEntity member) => |
| member.library == lateHelperLibrary && |
| _isTopLevelFunctionNamed('_lateReadCheck', member); |
| |
| /// Returns `true` if [member] is the `_lateWriteOnceCheck` function defined |
| /// in dart:_late_helper. |
| bool isLateWriteOnceCheck(MemberEntity member) => |
| member.library == lateHelperLibrary && |
| _isTopLevelFunctionNamed('_lateWriteOnceCheck', member); |
| |
| /// Returns `true` if [member] is the `_lateInitializeOnceCheck` function |
| /// defined in dart:_late_helper. |
| bool isLateInitializeOnceCheck(MemberEntity member) => |
| member.library == lateHelperLibrary && |
| _isTopLevelFunctionNamed('_lateInitializeOnceCheck', member); |
| |
| /// Returns `true` if [member] is the `createSentinel` function defined in |
| /// dart:_internal. |
| bool isCreateSentinel(MemberEntity member) => |
| member.library == internalLibrary && |
| _isTopLevelFunctionNamed('createSentinel', member); |
| |
| /// Returns `true` if [member] is the `isSentinel` function defined in |
| /// dart:_internal. |
| bool isIsSentinel(MemberEntity member) => |
| member.library == internalLibrary && |
| _isTopLevelFunctionNamed('isSentinel', member); |
| |
| /// Returns `true` if [member] is the `allowInterop` function defined in |
| /// dart:js_util. |
| bool isAllowInterop(MemberEntity member) => |
| member.library == dartJsUtilLibrary && |
| _isTopLevelFunctionNamed('allowInterop', member); |
| |
| ClassEntity getDefaultSuperclass( |
| ClassEntity cls, |
| NativeBasicData nativeBasicData, |
| ) { |
| final defaultedClass = defaultReceiverClass(this, nativeBasicData, cls); |
| if (defaultedClass != cls) return defaultedClass; |
| // Native classes inherit from Interceptor. |
| return nativeBasicData.isNativeClass(cls) |
| ? jsInterceptorClass |
| : objectClass; |
| } |
| |
| // From dart:js_util |
| late final FunctionEntity? jsAllowInterop = _findLibraryMember( |
| dartJsUtilLibrary, |
| 'allowInterop', |
| required: false, |
| ); |
| |
| bool isCreateInvocationMirrorHelper(MemberEntity member) { |
| return member.isTopLevel && |
| member.name == '_createInvocationMirror' && |
| member.library == coreLibrary; |
| } |
| } |
| |
| class KCommonElements extends CommonElements { |
| KCommonElements(super.dartTypes, super.env); |
| |
| // From package:js |
| |
| late final ClassEntity? jsAnnotationClass1 = _findClassOrNull( |
| packageJsLibrary, |
| 'JS', |
| ); |
| |
| late final ClassEntity? jsAnonymousClass1 = _findClassOrNull( |
| packageJsLibrary, |
| '_Anonymous', |
| ); |
| |
| // From dart:_js_annotations |
| |
| late final ClassEntity? jsAnnotationClass2 = _findClassOrNull( |
| dartJsAnnotationsLibrary, |
| 'JS', |
| ); |
| |
| late final ClassEntity? jsAnonymousClass2 = _findClassOrNull( |
| dartJsAnnotationsLibrary, |
| '_Anonymous', |
| ); |
| |
| // From dart:js_interop |
| |
| late final ClassEntity? jsAnnotationClass3 = _findClassOrNull( |
| dartJsInteropLibrary, |
| 'JS', |
| ); |
| |
| /// Returns `true` if [cls] is a @JS() annotation. |
| /// |
| /// The class can come from either `package:js`, `dart:_js_annotations`, or |
| /// `dart:js_interop`. |
| bool isJsAnnotationClass(ClassEntity cls) { |
| return cls == jsAnnotationClass1 || |
| cls == jsAnnotationClass2 || |
| cls == jsAnnotationClass3; |
| } |
| |
| /// Returns `true` if [cls] is an @anonymous annotation. |
| /// |
| /// The class can come from either `package:js` or `dart:_js_annotations`. |
| bool isJsAnonymousClass(ClassEntity cls) { |
| return cls == jsAnonymousClass1 || cls == jsAnonymousClass2; |
| } |
| |
| late final ClassEntity pragmaClass = _findClass(coreLibrary, 'pragma'); |
| |
| late final FieldEntity pragmaClassNameField = _findClassMember( |
| pragmaClass, |
| 'name', |
| ); |
| |
| late final FieldEntity pragmaClassOptionsField = _findClassMember( |
| pragmaClass, |
| 'options', |
| ); |
| } |
| |
| class JCommonElements extends CommonElements { |
| JCommonElements(super.dartTypes, super.env); |
| |
| /// Returns `true` if [element] is the named constructor of `List`, |
| /// e.g. `List.of`. |
| /// |
| /// This will not resolve the constructor if it hasn't been seen yet during |
| /// compilation. |
| bool isNamedListConstructor(String name, ConstructorEntity element) => |
| element.name == name && element.enclosingClass == listClass; |
| |
| /// Returns `true` if [element] is the named constructor of `JSArray`, |
| /// e.g. `JSArray.fixed`. |
| /// |
| /// This will not resolve the constructor if it hasn't been seen yet during |
| /// compilation. |
| bool isNamedJSArrayConstructor(String name, ConstructorEntity element) => |
| element.name == name && element.enclosingClass == jsArrayClass; |
| |
| bool isDefaultEqualityImplementation(MemberEntity element) { |
| assert(element.name == '=='); |
| ClassEntity? classElement = element.enclosingClass; |
| return classElement == objectClass || |
| classElement == jsInterceptorClass || |
| classElement == jsNullClass; |
| } |
| |
| /// Returns `true` if [selector] applies to `JSIndexable.length`. |
| bool appliesToJsIndexableLength(Selector selector) { |
| return selector.name == 'length' && (selector.isGetter || selector.isCall); |
| } |
| |
| late final FunctionEntity jsArrayRemoveLast = _findClassMember( |
| jsArrayClass, |
| 'removeLast', |
| ); |
| |
| late final FunctionEntity jsArrayAdd = _findClassMember(jsArrayClass, 'add'); |
| |
| bool _isJsStringClass(ClassEntity cls) { |
| return cls.name == 'JSString' && cls.library == interceptorsLibrary; |
| } |
| |
| bool isJsStringSplit(MemberEntity member) { |
| return member.name == 'split' && |
| member.isInstanceMember && |
| _isJsStringClass(member.enclosingClass!); |
| } |
| |
| /// Returns `true` if [selector] applies to `JSString.split` on [receiver] |
| /// in the given [world]. |
| /// |
| /// Returns `false` if `JSString.split` is not available (e.g. the |
| /// method was tree-shaken in an earlier compilation phase). |
| bool appliesToJsStringSplit( |
| Selector selector, |
| AbstractValue? receiver, |
| AbstractValueDomain abstractValueDomain, |
| ) { |
| final splitMember = jsStringSplit; |
| if (splitMember == null) return false; |
| return selector.applies(splitMember) && |
| (receiver == null || |
| abstractValueDomain |
| .isTargetingMember(receiver, splitMember, selector.memberName) |
| .isPotentiallyTrue); |
| } |
| |
| late final FunctionEntity? jsStringSplit = _findClassMemberOrNull( |
| jsStringClass, |
| 'split', |
| ); |
| |
| late final FunctionEntity jsStringToString = _findClassMember( |
| jsStringClass, |
| 'toString', |
| ); |
| |
| late final FunctionEntity jsStringOperatorAdd = _findClassMember( |
| jsStringClass, |
| '+', |
| ); |
| |
| late final ClassEntity jsConstClass = _findClass(foreignLibrary, 'JS_CONST'); |
| |
| /// Return `true` if [member] is the 'checkInt' function defined in |
| /// dart:_js_helpers. |
| bool isCheckInt(MemberEntity member) { |
| return member.isFunction && |
| member.isTopLevel && |
| member.library == jsHelperLibrary && |
| member.name == 'checkInt'; |
| } |
| |
| /// Return `true` if [member] is the 'checkNum' function defined in |
| /// dart:_js_helpers. |
| bool isCheckNum(MemberEntity member) { |
| return member.isFunction && |
| member.isTopLevel && |
| member.library == jsHelperLibrary && |
| member.name == 'checkNum'; |
| } |
| |
| /// Return `true` if [member] is the 'checkString' function defined in |
| /// dart:_js_helpers. |
| bool isCheckString(MemberEntity member) { |
| return member.isFunction && |
| member.isTopLevel && |
| member.library == jsHelperLibrary && |
| member.name == 'checkString'; |
| } |
| |
| bool isInstantiationClass(ClassEntity cls) { |
| return cls.library == jsHelperLibrary && |
| cls.name != 'Instantiation' && |
| cls.name.startsWith('Instantiation'); |
| } |
| |
| // From dart:_native_typed_data |
| |
| late final ClassEntity? typedArrayOfIntClass = _findClass( |
| typedDataLibrary, |
| 'NativeTypedArrayOfInt', |
| ); |
| |
| late final ClassEntity typedArrayOfDoubleClass = _findClass( |
| typedDataLibrary, |
| 'NativeTypedArrayOfDouble', |
| ); |
| |
| late final ClassEntity jsBuiltinEnum = _findClass( |
| sharedEmbeddedNamesLibrary, |
| 'JsBuiltin', |
| ); |
| |
| bool isForeign(MemberEntity element) => element.library == foreignLibrary; |
| |
| /// Returns `true` if the implementation of the 'operator ==' [function] is |
| /// known to handle `null` as argument. |
| bool operatorEqHandlesNullArgument(FunctionEntity function) { |
| assert( |
| function.name == '==', |
| failedAt(function, "Unexpected function $function."), |
| ); |
| ClassEntity? cls = function.enclosingClass; |
| return cls == objectClass || |
| cls == jsInterceptorClass || |
| cls == jsNullClass; |
| } |
| } |
| |
| /// Interface for accessing libraries, classes and members. |
| /// |
| /// The element environment makes private and injected members directly |
| /// available and should therefore not be used to determine scopes. |
| /// |
| /// The properties exposed are Dart-centric and should therefore, long-term, not |
| /// be used during codegen, expect for mirrors. |
| // TODO(johnniwinther): Split this into an element environment and a type query |
| // interface, the first should only be used during resolution and the latter in |
| // both resolution and codegen. |
| abstract class ElementEnvironment { |
| IrToElementMap get elementMap; |
| |
| /// Returns the main library for the compilation. |
| LibraryEntity? get mainLibrary; |
| |
| /// Returns the main method for the compilation. |
| FunctionEntity? get mainFunction; |
| |
| /// Returns all known libraries. |
| Iterable<LibraryEntity> get libraries; |
| |
| /// Returns the library name of [library] or '' if the library is unnamed. |
| String getLibraryName(LibraryEntity library); |
| |
| /// Lookup the library with the canonical [uri], fail if the library is |
| /// missing and [required]; |
| LibraryEntity? lookupLibrary(Uri uri, {bool required = false}); |
| |
| /// Calls [f] for every class declared in [library]. |
| void forEachClass(LibraryEntity library, void Function(ClassEntity cls) f); |
| |
| /// Lookup the class [name] in [library], fail if the class is missing and |
| /// [required]. |
| ClassEntity? lookupClass( |
| LibraryEntity library, |
| String name, { |
| bool required = false, |
| }); |
| |
| /// Calls [f] for every top level member in [library]. |
| void forEachLibraryMember( |
| LibraryEntity library, |
| void Function(MemberEntity member) f, |
| ); |
| |
| /// Lookup the member [name] in [library], fail if the class is missing and |
| /// [required]. |
| MemberEntity? lookupLibraryMember( |
| LibraryEntity library, |
| String name, { |
| bool setter = false, |
| bool required = false, |
| }); |
| |
| /// Lookup the member [name] in [cls], fail if the class is missing and |
| /// [required]. |
| MemberEntity? lookupLocalClassMember( |
| ClassEntity cls, |
| Name name, { |
| bool required = false, |
| }); |
| |
| /// Lookup the member [name] in [cls] and its superclasses. |
| /// |
| /// Return `null` if the member is not found in the class or any superclass. |
| MemberEntity? lookupClassMember(ClassEntity cls, Name name) { |
| ClassEntity? clsLocal = cls; |
| while (clsLocal != null) { |
| final entity = lookupLocalClassMember(clsLocal, name); |
| if (entity != null) return entity; |
| |
| clsLocal = getSuperClass(clsLocal); |
| } |
| return null; |
| } |
| |
| /// Lookup the constructor [name] in [cls], fail if the class is missing and |
| /// [required]. |
| ConstructorEntity? lookupConstructor( |
| ClassEntity cls, |
| String name, { |
| bool required = false, |
| }); |
| |
| /// Calls [f] for each class member declared in [cls]. |
| void forEachLocalClassMember( |
| ClassEntity cls, |
| void Function(MemberEntity member) f, |
| ); |
| |
| /// Calls [f] for each class member declared or inherited in [cls] together |
| /// with the class that declared the member. |
| /// |
| /// TODO(johnniwinther): This should not include static members of |
| /// superclasses. |
| void forEachClassMember( |
| ClassEntity cls, |
| void Function(ClassEntity declarer, MemberEntity member) f, |
| ); |
| |
| /// Calls [f] for every constructor declared in [cls]. |
| /// |
| /// Will ensure that the class and all constructors are resolved if |
| /// [ensureResolved] is `true`. |
| void forEachConstructor( |
| ClassEntity cls, |
| void Function(ConstructorEntity constructor) f, |
| ); |
| |
| /// Returns the superclass of [cls]. |
| /// |
| /// If [skipUnnamedMixinApplications] is `true`, unnamed mixin applications |
| /// are excluded, for instance for these classes |
| /// |
| /// class S {} |
| /// class M {} |
| /// class C extends S with M {} |
| /// |
| /// the result of `getSuperClass(C)` is the unnamed mixin application |
| /// typically named `S+M` and `getSuperClass(S+M)` is `S`, whereas |
| /// the result of `getSuperClass(C, skipUnnamedMixinApplications: false)` is |
| /// `S`. |
| ClassEntity? getSuperClass( |
| ClassEntity cls, { |
| bool skipUnnamedMixinApplications = false, |
| }); |
| |
| /// Calls [f] for each supertype of [cls]. |
| void forEachSupertype( |
| ClassEntity cls, |
| void Function(InterfaceType supertype) f, |
| ); |
| |
| /// Calls [f] for each SuperClass of [cls]. |
| void forEachSuperClass( |
| ClassEntity cls, |
| void Function(ClassEntity superClass) f, |
| ) { |
| for ( |
| var superClass = getSuperClass(cls); |
| superClass != null; |
| superClass = getSuperClass(superClass) |
| ) { |
| f(superClass); |
| } |
| } |
| |
| /// Create the instantiation of [cls] with the given [typeArguments] and |
| /// [nullability]. |
| InterfaceType createInterfaceType( |
| ClassEntity cls, |
| List<DartType> typeArguments, |
| ); |
| |
| /// Returns the `dynamic` type. |
| DartType get dynamicType; |
| |
| /// Returns the 'raw type' of [cls]. That is, the instantiation of [cls] |
| /// where all types arguments are `dynamic`. |
| InterfaceType getRawType(ClassEntity cls); |
| |
| /// Returns the 'JS-interop type' of [cls]; that is, the instantiation of |
| /// [cls] where all type arguments are 'any'. |
| InterfaceType getJsInteropType(ClassEntity cls); |
| |
| /// Returns the 'this type' of [cls]. That is, the instantiation of [cls] |
| /// where the type arguments are the type variables of [cls]. |
| InterfaceType getThisType(ClassEntity cls); |
| |
| /// Returns the instantiation of [cls] to bounds. |
| InterfaceType getClassInstantiationToBounds(ClassEntity cls); |
| |
| /// Returns `true` if [cls] is generic. |
| bool isGenericClass(ClassEntity cls); |
| |
| /// Returns `true` if [cls] is a mixin application (named or unnamed). |
| bool isMixinApplication(ClassEntity cls); |
| |
| /// Returns `true` if [cls] is an unnamed mixin application. |
| bool isUnnamedMixinApplication(ClassEntity cls); |
| |
| /// The upper bound on the [typeVariable]. If not explicitly declared, this is |
| /// `Object`. |
| DartType getTypeVariableBound(TypeVariableEntity typeVariable); |
| |
| /// Returns the variances for each type parameter in [cls]. |
| List<Variance> getTypeVariableVariances(ClassEntity cls); |
| |
| /// Returns the type of [function]. |
| FunctionType getFunctionType(FunctionEntity function); |
| |
| /// Returns the function type variables defined on [function]. |
| List<TypeVariableType> getFunctionTypeVariables(FunctionEntity function); |
| |
| /// Returns the type of the [local] function. |
| FunctionType getLocalFunctionType(Local local); |
| |
| /// Returns the type of [field]. |
| DartType getFieldType(FieldEntity field); |
| |
| /// Returns `true` if [cls] is a Dart enum class. |
| bool isEnumClass(ClassEntity cls); |
| |
| /// Returns the 'effective' mixin class if [cls] is a mixin application, and |
| /// `null` otherwise. |
| /// |
| /// The 'effective' mixin class is the class from which members are mixed in. |
| /// Normally this is the mixin class itself, but not if the mixin class itself |
| /// is a mixin application. |
| /// |
| /// Consider this hierarchy: |
| /// |
| /// class A {} |
| /// class B = Object with A {} |
| /// class C = Object with B {} |
| /// |
| /// The mixin classes of `B` and `C` are `A` and `B`, respectively, but the |
| /// _effective_ mixin class of both is `A`. |
| ClassEntity? getEffectiveMixinClass(ClassEntity cls); |
| } |
| |
| abstract class KElementEnvironment extends ElementEnvironment { |
| /// Calls [f] for each class that is mixed into [cls] or one of its |
| /// superclasses. |
| void forEachMixin(ClassEntity cls, void Function(ClassEntity mixin) f); |
| |
| /// Returns the imports seen in [library] |
| Iterable<ImportEntity> getImports(LibraryEntity library); |
| |
| /// Returns the metadata constants declared on [member]. |
| Iterable<ConstantValue> getMemberMetadata( |
| MemberEntity member, { |
| bool includeParameterMetadata = false, |
| }); |
| |
| /// `true` if field is the backing field for a `late` or `late final` instance |
| /// field. |
| bool isLateBackingField(FieldEntity field); |
| |
| /// `true` if field is the backing field for a `late final` instance field. If |
| /// this is true, so is [isLateBackingField]. |
| bool isLateFinalBackingField(FieldEntity field); |
| } |
| |
| abstract class JElementEnvironment extends ElementEnvironment { |
| /// Calls [f] for each class member added to [cls] during compilation. |
| void forEachInjectedClassMember( |
| ClassEntity cls, |
| void Function(MemberEntity member) f, |
| ); |
| |
| /// Calls [f] for every constructor body in [cls]. |
| void forEachConstructorBody( |
| ClassEntity cls, |
| void Function(ConstructorBodyEntity constructorBody) f, |
| ); |
| |
| /// Calls [f] for each nested closure in [member]. |
| void forEachNestedClosure( |
| MemberEntity member, |
| void Function(FunctionEntity closure) f, |
| ); |
| |
| /// Returns `true` if [cls] is a mixin application with its own members. |
| /// |
| /// This occurs when a mixin contains methods with super calls or when |
| /// the mixin application contains concrete forwarding stubs. |
| bool isMixinApplicationWithMembers(ClassEntity cls); |
| |
| /// The default type of the [typeVariable]. |
| /// |
| /// This is the type used as the default type argument when no explicit type |
| /// argument is passed. |
| DartType getTypeVariableDefaultType(TypeVariableEntity typeVariable); |
| |
| /// Returns the 'element' type of a [function] with the async, async* or sync* |
| /// marker. [returnType] is the return type of the marked function. |
| DartType getAsyncOrSyncStarElementType( |
| FunctionEntity function, |
| DartType returnType, |
| ); |
| |
| /// Returns the 'element' type of a function with an async, async* or sync* |
| /// marker. The return type of the method is inspected to determine the type |
| /// parameter of the Future, Stream or Iterable. |
| DartType getFunctionAsyncOrSyncStarElementType(FunctionEntity function); |
| |
| /// Returns `true` if [field] has an explicit `covariant` declaration. |
| bool isFieldCovariantByDeclaration(FieldEntity field); |
| |
| /// Calls [f] with every instance field, together with its declarer, in an |
| /// instance of [cls]. All fields inherited from superclasses and mixins are |
| /// included. |
| /// |
| /// If [isElided] is `true`, the field is not read and should therefore not |
| /// be emitted. |
| void forEachInstanceField( |
| ClassEntity cls, |
| void Function(ClassEntity declarer, FieldEntity field) f, |
| ); |
| |
| /// Calls [f] with every instance field declared directly in class [cls] |
| /// (i.e. no inherited fields). Fields are presented in initialization |
| /// (i.e. textual) order. |
| /// |
| /// If [isElided] is `true`, the field is not read and should therefore not |
| /// be emitted. |
| void forEachDirectInstanceField( |
| ClassEntity cls, |
| void Function(FieldEntity field) f, |
| ); |
| |
| /// Calls [f] for each parameter of [function] providing the type and name of |
| /// the parameter and the [defaultValue] if the parameter is optional. |
| void forEachParameter( |
| covariant FunctionEntity function, |
| void Function(DartType type, String? name, ConstantValue? defaultValue) f, |
| ); |
| |
| /// Calls [f] for each parameter - given as a [Local] - of [function]. |
| void forEachParameterAsLocal( |
| GlobalLocalsMap globalLocalsMap, |
| FunctionEntity function, |
| void Function(Local parameter) f, |
| ); |
| } |