| Index: sdk/lib/_internal/compiler/js_lib/js_helper.dart
|
| diff --git a/sdk/lib/_internal/compiler/js_lib/js_helper.dart b/sdk/lib/_internal/compiler/js_lib/js_helper.dart
|
| deleted file mode 100644
|
| index 291e1784bd851e8183963890355c0cf10ec4bf73..0000000000000000000000000000000000000000
|
| --- a/sdk/lib/_internal/compiler/js_lib/js_helper.dart
|
| +++ /dev/null
|
| @@ -1,4137 +0,0 @@
|
| -// Copyright (c) 2013, 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.
|
| -
|
| -library _js_helper;
|
| -
|
| -import 'dart:_async_await_error_codes' as async_error_codes;
|
| -
|
| -import 'dart:_js_embedded_names' show
|
| - DEFERRED_LIBRARY_URIS,
|
| - DEFERRED_LIBRARY_HASHES,
|
| - GET_TYPE_FROM_NAME,
|
| - GET_ISOLATE_TAG,
|
| - INITIALIZE_LOADED_HUNK,
|
| - INTERCEPTED_NAMES,
|
| - INTERCEPTORS_BY_TAG,
|
| - IS_HUNK_LOADED,
|
| - IS_HUNK_INITIALIZED,
|
| - JsBuiltin,
|
| - JsGetName,
|
| - LEAF_TAGS,
|
| - NATIVE_SUPERCLASS_TAG_NAME;
|
| -
|
| -import 'dart:collection';
|
| -
|
| -import 'dart:_isolate_helper' show
|
| - IsolateNatives,
|
| - enterJsAsync,
|
| - isWorker,
|
| - leaveJsAsync;
|
| -
|
| -import 'dart:async' show
|
| - Completer,
|
| - DeferredLoadException,
|
| - Future,
|
| - StreamController,
|
| - Stream,
|
| - StreamSubscription,
|
| - scheduleMicrotask;
|
| -
|
| -import 'dart:_foreign_helper' show
|
| - DART_CLOSURE_TO_JS,
|
| - JS,
|
| - JS_BUILTIN,
|
| - JS_CALL_IN_ISOLATE,
|
| - JS_CONST,
|
| - JS_CURRENT_ISOLATE,
|
| - JS_CURRENT_ISOLATE_CONTEXT,
|
| - JS_EFFECT,
|
| - JS_EMBEDDED_GLOBAL,
|
| - JS_GET_FLAG,
|
| - JS_GET_NAME,
|
| - JS_HAS_EQUALS,
|
| - JS_STRING_CONCAT,
|
| - RAW_DART_FUNCTION_REF;
|
| -
|
| -import 'dart:_interceptors';
|
| -import 'dart:_internal' as _symbol_dev;
|
| -import 'dart:_internal' show EfficientLength, MappedIterable;
|
| -
|
| -import 'dart:_native_typed_data';
|
| -
|
| -import 'dart:_js_names' show
|
| - extractKeys,
|
| - mangledNames,
|
| - unmangleGlobalNameIfPreservedAnyways,
|
| - unmangleAllIdentifiersIfPreservedAnyways;
|
| -
|
| -part 'annotations.dart';
|
| -part 'constant_map.dart';
|
| -part 'native_helper.dart';
|
| -part 'regexp_helper.dart';
|
| -part 'string_helper.dart';
|
| -part 'js_rti.dart';
|
| -part 'linked_hash_map.dart';
|
| -
|
| -/// Marks the internal map in dart2js, so that internal libraries can is-check
|
| -/// them.
|
| -abstract class InternalMap {
|
| -}
|
| -
|
| -/// Extracts the JavaScript-constructor name from the given isCheckProperty.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -String isCheckPropertyToJsConstructorName(String isCheckProperty) {
|
| - return JS_BUILTIN('returns:String;depends:none;effects:none',
|
| - JsBuiltin.isCheckPropertyToJsConstructorName,
|
| - isCheckProperty);
|
| -}
|
| -
|
| -/// Returns true if the given [type] is a function type object.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -bool isDartFunctionType(Object type) {
|
| - return JS_BUILTIN('returns:bool;effects:none;depends:none',
|
| - JsBuiltin.isFunctionType, type);
|
| -}
|
| -
|
| -
|
| -/// Creates a function type object.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -createDartFunctionTypeRti() {
|
| - return JS_BUILTIN('returns:=Object;effects:none;depends:none',
|
| - JsBuiltin.createFunctionTypeRti);
|
| -}
|
| -
|
| -/// Retrieves the class name from type information stored on the constructor of
|
| -/// [type].
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -String rawRtiToJsConstructorName(Object rti) {
|
| - return JS_BUILTIN('String', JsBuiltin.rawRtiToJsConstructorName, rti);
|
| -}
|
| -
|
| -/// Returns the rti from the given [constructorName].
|
| -// TODO(floitsch): make this a builtin.
|
| -jsConstructorNameToRti(String constructorName) {
|
| - var getTypeFromName = JS_EMBEDDED_GLOBAL('', GET_TYPE_FROM_NAME);
|
| - return JS('', '#(#)', getTypeFromName, constructorName);
|
| -}
|
| -
|
| -/// Returns the raw runtime type of the given object [o].
|
| -///
|
| -/// The argument [o] must be the interceptor for primitive types. If
|
| -/// necessary run it through [getInterceptor] first.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -// TODO(floitsch): we should call getInterceptor ourselves, but currently
|
| -// getInterceptor is not GVNed.
|
| -@ForceInline()
|
| -Object getRawRuntimeType(Object o) {
|
| - return JS_BUILTIN('', JsBuiltin.rawRuntimeType, o);
|
| -}
|
| -
|
| -/// Returns whether the given [type] is a subtype of [other].
|
| -///
|
| -/// The argument [other] is the name of the other type, as computed by
|
| -/// [runtimeTypeToString].
|
| -@ForceInline()
|
| -bool builtinIsSubtype(type, String other) {
|
| - return JS_BUILTIN('returns:bool;effects:none;depends:none',
|
| - JsBuiltin.isSubtype, other, type);
|
| -}
|
| -
|
| -/// Returns true if the given [type] is _the_ `Function` type.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -bool isDartFunctionTypeRti(Object type) {
|
| - return JS_BUILTIN('returns:bool;effects:none;depends:none',
|
| - JsBuiltin.isGivenTypeRti,
|
| - type,
|
| - JS_GET_NAME(JsGetName.FUNCTION_CLASS_TYPE_NAME));
|
| -}
|
| -
|
| -/// Returns whether the given type is _the_ Dart Object type.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -bool isDartObjectTypeRti(type) {
|
| - return JS_BUILTIN('returns:bool;effects:none;depends:none',
|
| - JsBuiltin.isGivenTypeRti,
|
| - type,
|
| - JS_GET_NAME(JsGetName.OBJECT_CLASS_TYPE_NAME));
|
| -}
|
| -
|
| -/// Returns whether the given type is _the_ null type.
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -bool isNullTypeRti(type) {
|
| - return JS_BUILTIN('returns:bool;effects:none;depends:none',
|
| - JsBuiltin.isGivenTypeRti,
|
| - type,
|
| - JS_GET_NAME(JsGetName.NULL_CLASS_TYPE_NAME));
|
| -}
|
| -
|
| -/// Returns the metadata of the given [index].
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -getMetadata(int index) {
|
| - return JS_BUILTIN('returns:var;effects:none;depends:none',
|
| - JsBuiltin.getMetadata, index);
|
| -}
|
| -
|
| -/// Returns the type of the given [index].
|
| -// TODO(floitsch): move this to foreign_helper.dart or similar.
|
| -@ForceInline()
|
| -getType(int index) {
|
| - return JS_BUILTIN('returns:var;effects:none;depends:none',
|
| - JsBuiltin.getType, index);
|
| -}
|
| -
|
| -/// No-op method that is called to inform the compiler that preambles might
|
| -/// be needed when executing the resulting JS file in a command-line
|
| -/// JS engine.
|
| -requiresPreamble() {}
|
| -
|
| -bool isJsIndexable(var object, var record) {
|
| - if (record != null) {
|
| - var result = dispatchRecordIndexability(record);
|
| - if (result != null) return result;
|
| - }
|
| - return object is JavaScriptIndexingBehavior;
|
| -}
|
| -
|
| -String S(value) {
|
| - if (value is String) return value;
|
| - if (value is num) {
|
| - if (value != 0) {
|
| - // ""+x is faster than String(x) for integers on most browsers.
|
| - return JS('String', r'"" + (#)', value);
|
| - }
|
| - } else if (true == value) {
|
| - return 'true';
|
| - } else if (false == value) {
|
| - return 'false';
|
| - } else if (value == null) {
|
| - return 'null';
|
| - }
|
| - var res = value.toString();
|
| - if (res is !String) throw argumentErrorValue(value);
|
| - return res;
|
| -}
|
| -
|
| -createInvocationMirror(String name, internalName, kind, arguments,
|
| - argumentNames) {
|
| - return new JSInvocationMirror(name,
|
| - internalName,
|
| - kind,
|
| - arguments,
|
| - argumentNames);
|
| -}
|
| -
|
| -createUnmangledInvocationMirror(Symbol symbol, internalName, kind, arguments,
|
| - argumentNames) {
|
| - return new JSInvocationMirror(symbol,
|
| - internalName,
|
| - kind,
|
| - arguments,
|
| - argumentNames);
|
| -}
|
| -
|
| -void throwInvalidReflectionError(String memberName) {
|
| - throw new UnsupportedError("Can't use '$memberName' in reflection "
|
| - "because it is not included in a @MirrorsUsed annotation.");
|
| -}
|
| -
|
| -/// Helper to print the given method information to the console the first
|
| -/// time it is called with it.
|
| -@NoInline()
|
| -void traceHelper(String method) {
|
| - if (JS('bool', '!this.cache')) {
|
| - JS('', 'this.cache = Object.create(null)');
|
| - }
|
| - if (JS('bool', '!this.cache[#]', method)) {
|
| - JS('', 'console.log(#)', method);
|
| - JS('', 'this.cache[#] = true', method);
|
| - }
|
| -}
|
| -
|
| -class JSInvocationMirror implements Invocation {
|
| - static const METHOD = 0;
|
| - static const GETTER = 1;
|
| - static const SETTER = 2;
|
| -
|
| - /// When [_memberName] is a String, it holds the mangled name of this
|
| - /// invocation. When it is a Symbol, it holds the unmangled name.
|
| - var /* String or Symbol */ _memberName;
|
| - final String _internalName;
|
| - final int _kind;
|
| - final List _arguments;
|
| - final List _namedArgumentNames;
|
| - /** Map from argument name to index in _arguments. */
|
| - Map<String, dynamic> _namedIndices = null;
|
| -
|
| - JSInvocationMirror(this._memberName,
|
| - this._internalName,
|
| - this._kind,
|
| - this._arguments,
|
| - this._namedArgumentNames);
|
| -
|
| - Symbol get memberName {
|
| - if (_memberName is Symbol) return _memberName;
|
| - String name = _memberName;
|
| - String unmangledName = mangledNames[name];
|
| - if (unmangledName != null) {
|
| - name = unmangledName.split(':')[0];
|
| - } else {
|
| - if (mangledNames[_internalName] == null) {
|
| - print("Warning: '$name' is used reflectively but not in MirrorsUsed. "
|
| - "This will break minified code.");
|
| - }
|
| - }
|
| - _memberName = new _symbol_dev.Symbol.unvalidated(name);
|
| - return _memberName;
|
| - }
|
| -
|
| - bool get isMethod => _kind == METHOD;
|
| - bool get isGetter => _kind == GETTER;
|
| - bool get isSetter => _kind == SETTER;
|
| - bool get isAccessor => _kind != METHOD;
|
| -
|
| - List get positionalArguments {
|
| - if (isGetter) return const [];
|
| - var argumentCount = _arguments.length - _namedArgumentNames.length;
|
| - if (argumentCount == 0) return const [];
|
| - var list = [];
|
| - for (var index = 0 ; index < argumentCount ; index++) {
|
| - list.add(_arguments[index]);
|
| - }
|
| - return JSArray.markUnmodifiableList(list);
|
| - }
|
| -
|
| - Map<Symbol, dynamic> get namedArguments {
|
| - if (isAccessor) return const <Symbol, dynamic>{};
|
| - int namedArgumentCount = _namedArgumentNames.length;
|
| - int namedArgumentsStartIndex = _arguments.length - namedArgumentCount;
|
| - if (namedArgumentCount == 0) return const <Symbol, dynamic>{};
|
| - var map = new Map<Symbol, dynamic>();
|
| - for (int i = 0; i < namedArgumentCount; i++) {
|
| - map[new _symbol_dev.Symbol.unvalidated(_namedArgumentNames[i])] =
|
| - _arguments[namedArgumentsStartIndex + i];
|
| - }
|
| - return new ConstantMapView<Symbol, dynamic>(map);
|
| - }
|
| -
|
| - _getCachedInvocation(Object object) {
|
| - var interceptor = getInterceptor(object);
|
| - var receiver = object;
|
| - var name = _internalName;
|
| - var arguments = _arguments;
|
| - var interceptedNames = JS_EMBEDDED_GLOBAL('', INTERCEPTED_NAMES);
|
| - bool isIntercepted =
|
| - JS("bool", 'Object.prototype.hasOwnProperty.call(#, #)',
|
| - interceptedNames, name);
|
| - if (isIntercepted) {
|
| - receiver = interceptor;
|
| - if (JS('bool', '# === #', object, interceptor)) {
|
| - interceptor = null;
|
| - }
|
| - } else {
|
| - interceptor = null;
|
| - }
|
| - bool isCatchAll = false;
|
| - var method = JS('var', '#[#]', receiver, name);
|
| - if (JS('bool', 'typeof # != "function"', method) ) {
|
| - String baseName = _symbol_dev.Symbol.getName(memberName);
|
| - method = JS('', '#[# + "*"]', receiver, baseName);
|
| - if (method == null) {
|
| - interceptor = getInterceptor(object);
|
| - method = JS('', '#[# + "*"]', interceptor, baseName);
|
| - if (method != null) {
|
| - isIntercepted = true;
|
| - receiver = interceptor;
|
| - } else {
|
| - interceptor = null;
|
| - }
|
| - }
|
| - isCatchAll = true;
|
| - }
|
| - if (JS('bool', 'typeof # == "function"', method)) {
|
| - if (isCatchAll) {
|
| - return new CachedCatchAllInvocation(
|
| - name, method, isIntercepted, interceptor);
|
| - } else {
|
| - return new CachedInvocation(name, method, isIntercepted, interceptor);
|
| - }
|
| - } else {
|
| - // In this case, receiver doesn't implement name. So we should
|
| - // invoke noSuchMethod instead (which will often throw a
|
| - // NoSuchMethodError).
|
| - return new CachedNoSuchMethodInvocation(interceptor);
|
| - }
|
| - }
|
| -
|
| - /// This method is called by [InstanceMirror.delegate].
|
| - static invokeFromMirror(JSInvocationMirror invocation, Object victim) {
|
| - var cached = invocation._getCachedInvocation(victim);
|
| - if (cached.isNoSuchMethod) {
|
| - return cached.invokeOn(victim, invocation);
|
| - } else {
|
| - return cached.invokeOn(victim, invocation._arguments);
|
| - }
|
| - }
|
| -
|
| - static getCachedInvocation(JSInvocationMirror invocation, Object victim) {
|
| - return invocation._getCachedInvocation(victim);
|
| - }
|
| -}
|
| -
|
| -class CachedInvocation {
|
| - // The mangled name of this invocation.
|
| - String mangledName;
|
| -
|
| - /// The JS function to call.
|
| - var jsFunction;
|
| -
|
| - /// True if this is an intercepted call.
|
| - bool isIntercepted;
|
| -
|
| - /// Non-null interceptor if this is an intercepted call through an
|
| - /// [Interceptor].
|
| - Interceptor cachedInterceptor;
|
| -
|
| - CachedInvocation(this.mangledName,
|
| - this.jsFunction,
|
| - this.isIntercepted,
|
| - this.cachedInterceptor);
|
| -
|
| - bool get isNoSuchMethod => false;
|
| - bool get isGetterStub => JS("bool", "!!#.\$getterStub", jsFunction);
|
| -
|
| - /// Applies [jsFunction] to [victim] with [arguments].
|
| - /// Users of this class must take care to check the arguments first.
|
| - invokeOn(Object victim, List arguments) {
|
| - var receiver = victim;
|
| - if (!isIntercepted) {
|
| - if (arguments is! JSArray) arguments = new List.from(arguments);
|
| - } else {
|
| - arguments = [victim]..addAll(arguments);
|
| - if (cachedInterceptor != null) receiver = cachedInterceptor;
|
| - }
|
| - return JS("var", "#.apply(#, #)", jsFunction, receiver, arguments);
|
| - }
|
| -}
|
| -
|
| -class CachedCatchAllInvocation extends CachedInvocation {
|
| - final ReflectionInfo info;
|
| -
|
| - CachedCatchAllInvocation(String name,
|
| - jsFunction,
|
| - bool isIntercepted,
|
| - Interceptor cachedInterceptor)
|
| - : info = new ReflectionInfo(jsFunction),
|
| - super(name, jsFunction, isIntercepted, cachedInterceptor);
|
| -
|
| - bool get isGetterStub => false;
|
| -
|
| - invokeOn(Object victim, List arguments) {
|
| - var receiver = victim;
|
| - int providedArgumentCount;
|
| - int fullParameterCount =
|
| - info.requiredParameterCount + info.optionalParameterCount;
|
| - if (!isIntercepted) {
|
| - if (arguments is JSArray) {
|
| - providedArgumentCount = arguments.length;
|
| - // If we need to add extra arguments before calling, we have
|
| - // to copy the arguments array.
|
| - if (providedArgumentCount < fullParameterCount) {
|
| - arguments = new List.from(arguments);
|
| - }
|
| - } else {
|
| - arguments = new List.from(arguments);
|
| - providedArgumentCount = arguments.length;
|
| - }
|
| - } else {
|
| - arguments = [victim]..addAll(arguments);
|
| - if (cachedInterceptor != null) receiver = cachedInterceptor;
|
| - providedArgumentCount = arguments.length - 1;
|
| - }
|
| - if (info.areOptionalParametersNamed &&
|
| - (providedArgumentCount > info.requiredParameterCount)) {
|
| - throw new UnimplementedNoSuchMethodError(
|
| - "Invocation of unstubbed method '${info.reflectionName}'"
|
| - " with ${arguments.length} arguments.");
|
| - } else if (providedArgumentCount < info.requiredParameterCount) {
|
| - throw new UnimplementedNoSuchMethodError(
|
| - "Invocation of unstubbed method '${info.reflectionName}'"
|
| - " with $providedArgumentCount arguments (too few).");
|
| - } else if (providedArgumentCount > fullParameterCount) {
|
| - throw new UnimplementedNoSuchMethodError(
|
| - "Invocation of unstubbed method '${info.reflectionName}'"
|
| - " with $providedArgumentCount arguments (too many).");
|
| - }
|
| - for (int i = providedArgumentCount; i < fullParameterCount; i++) {
|
| - arguments.add(getMetadata(info.defaultValue(i)));
|
| - }
|
| - return JS("var", "#.apply(#, #)", jsFunction, receiver, arguments);
|
| - }
|
| -}
|
| -
|
| -class CachedNoSuchMethodInvocation {
|
| - /// Non-null interceptor if this is an intercepted call through an
|
| - /// [Interceptor].
|
| - var interceptor;
|
| -
|
| - CachedNoSuchMethodInvocation(this.interceptor);
|
| -
|
| - bool get isNoSuchMethod => true;
|
| - bool get isGetterStub => false;
|
| -
|
| - invokeOn(Object victim, Invocation invocation) {
|
| - var receiver = (interceptor == null) ? victim : interceptor;
|
| - return receiver.noSuchMethod(invocation);
|
| - }
|
| -}
|
| -
|
| -class ReflectionInfo {
|
| - static const int REQUIRED_PARAMETERS_INFO = 0;
|
| - static const int OPTIONAL_PARAMETERS_INFO = 1;
|
| - static const int FUNCTION_TYPE_INDEX = 2;
|
| - static const int FIRST_DEFAULT_ARGUMENT = 3;
|
| -
|
| - /// A JavaScript function object.
|
| - final jsFunction;
|
| -
|
| - /// Raw reflection information.
|
| - final List data;
|
| -
|
| - /// Is this a getter or a setter.
|
| - final bool isAccessor;
|
| -
|
| - /// Number of required parameters.
|
| - final int requiredParameterCount;
|
| -
|
| - /// Number of optional parameters.
|
| - final int optionalParameterCount;
|
| -
|
| - /// Are optional parameters named.
|
| - final bool areOptionalParametersNamed;
|
| -
|
| - /// Either an index to the function type in the embedded `metadata` global or
|
| - /// a JavaScript function object which can compute such a type (presumably
|
| - /// due to free type variables).
|
| - final functionType;
|
| -
|
| - List cachedSortedIndices;
|
| -
|
| - ReflectionInfo.internal(this.jsFunction,
|
| - this.data,
|
| - this.isAccessor,
|
| - this.requiredParameterCount,
|
| - this.optionalParameterCount,
|
| - this.areOptionalParametersNamed,
|
| - this.functionType);
|
| -
|
| - factory ReflectionInfo(jsFunction) {
|
| - List data = JS('JSExtendableArray|Null', r'#.$reflectionInfo', jsFunction);
|
| - if (data == null) return null;
|
| - data = JSArray.markFixedList(data);
|
| -
|
| - int requiredParametersInfo =
|
| - JS('int', '#[#]', data, REQUIRED_PARAMETERS_INFO);
|
| - int requiredParameterCount = JS('int', '# >> 1', requiredParametersInfo);
|
| - bool isAccessor = (requiredParametersInfo & 1) == 1;
|
| -
|
| - int optionalParametersInfo =
|
| - JS('int', '#[#]', data, OPTIONAL_PARAMETERS_INFO);
|
| - int optionalParameterCount = JS('int', '# >> 1', optionalParametersInfo);
|
| - bool areOptionalParametersNamed = (optionalParametersInfo & 1) == 1;
|
| -
|
| - var functionType = JS('', '#[#]', data, FUNCTION_TYPE_INDEX);
|
| - return new ReflectionInfo.internal(
|
| - jsFunction, data, isAccessor, requiredParameterCount,
|
| - optionalParameterCount, areOptionalParametersNamed, functionType);
|
| - }
|
| -
|
| - String parameterName(int parameter) {
|
| - int metadataIndex;
|
| - if (JS_GET_FLAG('MUST_RETAIN_METADATA')) {
|
| - metadataIndex = JS('int', '#[2 * # + # + #]', data,
|
| - parameter, optionalParameterCount, FIRST_DEFAULT_ARGUMENT);
|
| - } else {
|
| - metadataIndex = JS('int', '#[# + # + #]', data,
|
| - parameter, optionalParameterCount, FIRST_DEFAULT_ARGUMENT);
|
| - }
|
| - var name = getMetadata(metadataIndex);
|
| - return JS('String', '#', name);
|
| - }
|
| -
|
| - List<int> parameterMetadataAnnotations(int parameter) {
|
| - if (!JS_GET_FLAG('MUST_RETAIN_METADATA')) {
|
| - throw new StateError('metadata has not been preserved');
|
| - } else {
|
| - return JS('', '#[2 * # + # + # + 1]', data, parameter,
|
| - optionalParameterCount, FIRST_DEFAULT_ARGUMENT);
|
| - }
|
| - }
|
| -
|
| - int defaultValue(int parameter) {
|
| - if (parameter < requiredParameterCount) return null;
|
| - return JS('int', '#[# + # - #]', data,
|
| - FIRST_DEFAULT_ARGUMENT, parameter, requiredParameterCount);
|
| - }
|
| -
|
| - /// Returns the default value of the [parameter]th entry of the list of
|
| - /// parameters sorted by name.
|
| - int defaultValueInOrder(int parameter) {
|
| - if (parameter < requiredParameterCount) return null;
|
| -
|
| - if (!areOptionalParametersNamed || optionalParameterCount == 1) {
|
| - return defaultValue(parameter);
|
| - }
|
| -
|
| - int index = sortedIndex(parameter - requiredParameterCount);
|
| - return defaultValue(index);
|
| - }
|
| -
|
| - /// Returns the default value of the [parameter]th entry of the list of
|
| - /// parameters sorted by name.
|
| - String parameterNameInOrder(int parameter) {
|
| - if (parameter < requiredParameterCount) return null;
|
| -
|
| - if (!areOptionalParametersNamed ||
|
| - optionalParameterCount == 1) {
|
| - return parameterName(parameter);
|
| - }
|
| -
|
| - int index = sortedIndex(parameter - requiredParameterCount);
|
| - return parameterName(index);
|
| - }
|
| -
|
| - /// Computes the index of the parameter in the list of named parameters sorted
|
| - /// by their name.
|
| - int sortedIndex(int unsortedIndex) {
|
| - if (cachedSortedIndices == null) {
|
| - // TODO(karlklose): cache this between [ReflectionInfo] instances or cache
|
| - // [ReflectionInfo] instances by [jsFunction].
|
| - cachedSortedIndices = new List(optionalParameterCount);
|
| - Map<String, int> positions = <String, int>{};
|
| - for (int i = 0; i < optionalParameterCount; i++) {
|
| - int index = requiredParameterCount + i;
|
| - positions[parameterName(index)] = index;
|
| - }
|
| - int index = 0;
|
| - (positions.keys.toList()..sort()).forEach((String name) {
|
| - cachedSortedIndices[index++] = positions[name];
|
| - });
|
| - }
|
| - return cachedSortedIndices[unsortedIndex];
|
| - }
|
| -
|
| - @NoInline()
|
| - computeFunctionRti(jsConstructor) {
|
| - if (JS('bool', 'typeof # == "number"', functionType)) {
|
| - return getType(functionType);
|
| - } else if (JS('bool', 'typeof # == "function"', functionType)) {
|
| - var fakeInstance = JS('', 'new #()', jsConstructor);
|
| - setRuntimeTypeInfo(
|
| - fakeInstance, JS('JSExtendableArray', '#["<>"]', fakeInstance));
|
| - return JS('=Object|Null', r'#.apply({$receiver:#})',
|
| - functionType, fakeInstance);
|
| - } else {
|
| - throw new RuntimeError('Unexpected function type');
|
| - }
|
| - }
|
| -
|
| - String get reflectionName => JS('String', r'#.$reflectionName', jsFunction);
|
| -}
|
| -
|
| -class Primitives {
|
| - /// Isolate-unique ID for caching [JsClosureMirror.function].
|
| - /// Note the initial value is used by the first isolate (or if there are no
|
| - /// isolates), new isolates will update this value to avoid conflicts by
|
| - /// calling [initializeStatics].
|
| - static String mirrorFunctionCacheName = '\$cachedFunction';
|
| -
|
| - /// Isolate-unique ID for caching [JsInstanceMirror._invoke].
|
| - static String mirrorInvokeCacheName = '\$cachedInvocation';
|
| -
|
| - /// Called when creating a new isolate (see _IsolateContext constructor in
|
| - /// isolate_helper.dart).
|
| - /// Please don't add complicated code to this method, as it will impact
|
| - /// start-up performance.
|
| - static void initializeStatics(int id) {
|
| - // Benchmarking shows significant performance improvements if this is a
|
| - // fixed value.
|
| - mirrorFunctionCacheName += '_$id';
|
| - mirrorInvokeCacheName += '_$id';
|
| - }
|
| -
|
| - static int objectHashCode(object) {
|
| - int hash = JS('int|Null', r'#.$identityHash', object);
|
| - if (hash == null) {
|
| - hash = JS('int', '(Math.random() * 0x3fffffff) | 0');
|
| - JS('void', r'#.$identityHash = #', object, hash);
|
| - }
|
| - return JS('int', '#', hash);
|
| - }
|
| -
|
| - @NoInline()
|
| - static int _parseIntError(String source, int handleError(String source)) {
|
| - if (handleError == null) throw new FormatException(source);
|
| - return handleError(source);
|
| - }
|
| -
|
| - static int parseInt(String source,
|
| - int radix,
|
| - int handleError(String source)) {
|
| - checkString(source);
|
| - var re = JS('', r'/^\s*[+-]?((0x[a-f0-9]+)|(\d+)|([a-z0-9]+))\s*$/i');
|
| - var match = JS('JSExtendableArray|Null', '#.exec(#)', re, source);
|
| - int digitsIndex = 1;
|
| - int hexIndex = 2;
|
| - int decimalIndex = 3;
|
| - int nonDecimalHexIndex = 4;
|
| - if (match == null) {
|
| - // TODO(sra): It might be that the match failed due to unrecognized U+0085
|
| - // spaces. We could replace them with U+0020 spaces and try matching
|
| - // again.
|
| - return _parseIntError(source, handleError);
|
| - }
|
| - String decimalMatch = match[decimalIndex];
|
| - if (radix == null) {
|
| - if (decimalMatch != null) {
|
| - // Cannot fail because we know that the digits are all decimal.
|
| - return JS('int', r'parseInt(#, 10)', source);
|
| - }
|
| - if (match[hexIndex] != null) {
|
| - // Cannot fail because we know that the digits are all hex.
|
| - return JS('int', r'parseInt(#, 16)', source);
|
| - }
|
| - return _parseIntError(source, handleError);
|
| - }
|
| -
|
| - if (radix is! int) {
|
| - throw new ArgumentError.value(radix, 'radix', 'is not an integer');
|
| - }
|
| - if (radix < 2 || radix > 36) {
|
| - throw new RangeError.range(radix, 2, 36, 'radix');
|
| - }
|
| - if (radix == 10 && decimalMatch != null) {
|
| - // Cannot fail because we know that the digits are all decimal.
|
| - return JS('int', r'parseInt(#, 10)', source);
|
| - }
|
| - // If radix >= 10 and we have only decimal digits the string is safe.
|
| - // Otherwise we need to check the digits.
|
| - if (radix < 10 || decimalMatch == null) {
|
| - // We know that the characters must be ASCII as otherwise the
|
| - // regexp wouldn't have matched. Lowercasing by doing `| 0x20` is thus
|
| - // guaranteed to be a safe operation, since it preserves digits
|
| - // and lower-cases ASCII letters.
|
| - int maxCharCode;
|
| - if (radix <= 10) {
|
| - // Allow all digits less than the radix. For example 0, 1, 2 for
|
| - // radix 3.
|
| - // "0".codeUnitAt(0) + radix - 1;
|
| - maxCharCode = (0x30 - 1) + radix;
|
| - } else {
|
| - // Letters are located after the digits in ASCII. Therefore we
|
| - // only check for the character code. The regexp above made already
|
| - // sure that the string does not contain anything but digits or
|
| - // letters.
|
| - // "a".codeUnitAt(0) + (radix - 10) - 1;
|
| - maxCharCode = (0x61 - 10 - 1) + radix;
|
| - }
|
| - assert(match[digitsIndex] is String);
|
| - String digitsPart = JS('String', '#[#]', match, digitsIndex);
|
| - for (int i = 0; i < digitsPart.length; i++) {
|
| - int characterCode = digitsPart.codeUnitAt(i) | 0x20;
|
| - if (characterCode > maxCharCode) {
|
| - return _parseIntError(source, handleError);
|
| - }
|
| - }
|
| - }
|
| - // The above matching and checks ensures the source has at least one digits
|
| - // and all digits are suitable for the radix, so parseInt cannot return NaN.
|
| - return JS('int', r'parseInt(#, #)', source, radix);
|
| - }
|
| -
|
| - @NoInline()
|
| - static double _parseDoubleError(String source,
|
| - double handleError(String source)) {
|
| - if (handleError == null) {
|
| - throw new FormatException('Invalid double', source);
|
| - }
|
| - return handleError(source);
|
| - }
|
| -
|
| - static double parseDouble(String source, double handleError(String source)) {
|
| - checkString(source);
|
| - // Notice that JS parseFloat accepts garbage at the end of the string.
|
| - // Accept only:
|
| - // - [+/-]NaN
|
| - // - [+/-]Infinity
|
| - // - a Dart double literal
|
| - // We do allow leading or trailing whitespace.
|
| - if (!JS('bool',
|
| - r'/^\s*[+-]?(?:Infinity|NaN|'
|
| - r'(?:\.\d+|\d+(?:\.\d*)?)(?:[eE][+-]?\d+)?)\s*$/.test(#)',
|
| - source)) {
|
| - return _parseDoubleError(source, handleError);
|
| - }
|
| - var result = JS('num', r'parseFloat(#)', source);
|
| - if (result.isNaN) {
|
| - var trimmed = source.trim();
|
| - if (trimmed == 'NaN' || trimmed == '+NaN' || trimmed == '-NaN') {
|
| - return result;
|
| - }
|
| - return _parseDoubleError(source, handleError);
|
| - }
|
| - return result;
|
| - }
|
| -
|
| - /** [: r"$".codeUnitAt(0) :] */
|
| - static const int DOLLAR_CHAR_VALUE = 36;
|
| -
|
| - /// Creates a string containing the complete type for the class [className]
|
| - /// with the given type arguments.
|
| - ///
|
| - /// In minified mode, uses the unminified names if available.
|
| - ///
|
| - /// The given [className] string generally contains the name of the JavaScript
|
| - /// constructor of the given class.
|
| - static String formatType(String className, List typeArguments) {
|
| - return unmangleAllIdentifiersIfPreservedAnyways
|
| - ('$className${joinArguments(typeArguments, 0)}');
|
| - }
|
| -
|
| - /// Returns the type of [object] as a string (including type arguments).
|
| - ///
|
| - /// In minified mode, uses the unminified names if available.
|
| - static String objectTypeName(Object object) {
|
| - String name = constructorNameFallback(getInterceptor(object));
|
| - if (name == 'Object') {
|
| - // Try to decompile the constructor by turning it into a string and get
|
| - // the name out of that. If the decompiled name is a string containing an
|
| - // identifier, we use that instead of the very generic 'Object'.
|
| - var decompiled =
|
| - JS('var', r'#.match(/^\s*function\s*([\w$]*)\s*\(/)[1]',
|
| - JS('var', r'String(#.constructor)', object));
|
| - if (decompiled is String)
|
| - if (JS('bool', r'/^\w+$/.test(#)', decompiled))
|
| - name = decompiled;
|
| - }
|
| - // TODO(kasperl): If the namer gave us a fresh global name, we may
|
| - // want to remove the numeric suffix that makes it unique too.
|
| - if (name.length > 1 && identical(name.codeUnitAt(0), DOLLAR_CHAR_VALUE)) {
|
| - name = name.substring(1);
|
| - }
|
| - return formatType(name, getRuntimeTypeInfo(object));
|
| - }
|
| -
|
| - /// In minified mode, uses the unminified names if available.
|
| - static String objectToHumanReadableString(Object object) {
|
| - String name = objectTypeName(object);
|
| - return "Instance of '$name'";
|
| - }
|
| -
|
| - static num dateNow() => JS('int', r'Date.now()');
|
| -
|
| - static void initTicker() {
|
| - if (timerFrequency != null) return;
|
| - // Start with low-resolution. We overwrite the fields if we find better.
|
| - timerFrequency = 1000;
|
| - timerTicks = dateNow;
|
| - if (JS('bool', 'typeof window == "undefined"')) return;
|
| - var window = JS('var', 'window');
|
| - if (window == null) return;
|
| - var performance = JS('var', '#.performance', window);
|
| - if (performance == null) return;
|
| - if (JS('bool', 'typeof #.now != "function"', performance)) return;
|
| - timerFrequency = 1000000;
|
| - timerTicks = () => (1000 * JS('num', '#.now()', performance)).floor();
|
| - }
|
| -
|
| - static int timerFrequency;
|
| - static Function timerTicks;
|
| -
|
| - static String currentUri() {
|
| - requiresPreamble();
|
| - // In a browser return self.location.href.
|
| - if (JS('bool', '!!self.location')) {
|
| - return JS('String', 'self.location.href');
|
| - }
|
| -
|
| - return null;
|
| - }
|
| -
|
| - // This is to avoid stack overflows due to very large argument arrays in
|
| - // apply(). It fixes http://dartbug.com/6919
|
| - static String _fromCharCodeApply(List<int> array) {
|
| - const kMaxApply = 500;
|
| - int end = array.length;
|
| - if (end <= kMaxApply) {
|
| - return JS('String', r'String.fromCharCode.apply(null, #)', array);
|
| - }
|
| - String result = '';
|
| - for (int i = 0; i < end; i += kMaxApply) {
|
| - int chunkEnd = (i + kMaxApply < end) ? i + kMaxApply : end;
|
| - result = JS('String',
|
| - r'# + String.fromCharCode.apply(null, #.slice(#, #))',
|
| - result, array, i, chunkEnd);
|
| - }
|
| - return result;
|
| - }
|
| -
|
| - static String stringFromCodePoints(codePoints) {
|
| - List<int> a = <int>[];
|
| - for (var i in codePoints) {
|
| - if (i is !int) throw argumentErrorValue(i);
|
| - if (i <= 0xffff) {
|
| - a.add(i);
|
| - } else if (i <= 0x10ffff) {
|
| - a.add(0xd800 + ((((i - 0x10000) >> 10) & 0x3ff)));
|
| - a.add(0xdc00 + (i & 0x3ff));
|
| - } else {
|
| - throw argumentErrorValue(i);
|
| - }
|
| - }
|
| - return _fromCharCodeApply(a);
|
| - }
|
| -
|
| - static String stringFromCharCodes(charCodes) {
|
| - for (var i in charCodes) {
|
| - if (i is !int) throw argumentErrorValue(i);
|
| - if (i < 0) throw argumentErrorValue(i);
|
| - if (i > 0xffff) return stringFromCodePoints(charCodes);
|
| - }
|
| - return _fromCharCodeApply(charCodes);
|
| - }
|
| -
|
| - // [start] and [end] are validated.
|
| - static String stringFromNativeUint8List(
|
| - NativeUint8List charCodes, int start, int end) {
|
| - const kMaxApply = 500;
|
| - if (end <= kMaxApply && start == 0 && end == charCodes.length) {
|
| - return JS('String', r'String.fromCharCode.apply(null, #)', charCodes);
|
| - }
|
| - String result = '';
|
| - for (int i = start; i < end; i += kMaxApply) {
|
| - int chunkEnd = (i + kMaxApply < end) ? i + kMaxApply : end;
|
| - result = JS('String',
|
| - r'# + String.fromCharCode.apply(null, #.subarray(#, #))',
|
| - result, charCodes, i, chunkEnd);
|
| - }
|
| - return result;
|
| - }
|
| -
|
| -
|
| - static String stringFromCharCode(charCode) {
|
| - if (0 <= charCode) {
|
| - if (charCode <= 0xffff) {
|
| - return JS('String', 'String.fromCharCode(#)', charCode);
|
| - }
|
| - if (charCode <= 0x10ffff) {
|
| - var bits = charCode - 0x10000;
|
| - var low = 0xDC00 | (bits & 0x3ff);
|
| - var high = 0xD800 | (bits >> 10);
|
| - return JS('String', 'String.fromCharCode(#, #)', high, low);
|
| - }
|
| - }
|
| - throw new RangeError.range(charCode, 0, 0x10ffff);
|
| - }
|
| -
|
| - static String stringConcatUnchecked(String string1, String string2) {
|
| - return JS_STRING_CONCAT(string1, string2);
|
| - }
|
| -
|
| - static String flattenString(String str) {
|
| - return JS('String', "#.charCodeAt(0) == 0 ? # : #", str, str, str);
|
| - }
|
| -
|
| - static String getTimeZoneName(receiver) {
|
| - // Firefox and Chrome emit the timezone in parenthesis.
|
| - // Example: "Wed May 16 2012 21:13:00 GMT+0200 (CEST)".
|
| - // We extract this name using a regexp.
|
| - var d = lazyAsJsDate(receiver);
|
| - List match = JS('JSArray|Null', r'/\((.*)\)/.exec(#.toString())', d);
|
| - if (match != null) return match[1];
|
| -
|
| - // Internet Explorer 10+ emits the zone name without parenthesis:
|
| - // Example: Thu Oct 31 14:07:44 PDT 2013
|
| - match = JS('JSArray|Null',
|
| - // Thu followed by a space.
|
| - r'/^[A-Z,a-z]{3}\s'
|
| - // Oct 31 followed by space.
|
| - r'[A-Z,a-z]{3}\s\d+\s'
|
| - // Time followed by a space.
|
| - r'\d{2}:\d{2}:\d{2}\s'
|
| - // The time zone name followed by a space.
|
| - r'([A-Z]{3,5})\s'
|
| - // The year.
|
| - r'\d{4}$/'
|
| - '.exec(#.toString())',
|
| - d);
|
| - if (match != null) return match[1];
|
| -
|
| - // IE 9 and Opera don't provide the zone name. We fall back to emitting the
|
| - // UTC/GMT offset.
|
| - // Example (IE9): Wed Nov 20 09:51:00 UTC+0100 2013
|
| - // (Opera): Wed Nov 20 2013 11:03:38 GMT+0100
|
| - match = JS('JSArray|Null', r'/(?:GMT|UTC)[+-]\d{4}/.exec(#.toString())', d);
|
| - if (match != null) return match[0];
|
| - return "";
|
| - }
|
| -
|
| - static int getTimeZoneOffsetInMinutes(receiver) {
|
| - // Note that JS and Dart disagree on the sign of the offset.
|
| - return -JS('int', r'#.getTimezoneOffset()', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static valueFromDecomposedDate(years, month, day, hours, minutes, seconds,
|
| - milliseconds, isUtc) {
|
| - final int MAX_MILLISECONDS_SINCE_EPOCH = 8640000000000000;
|
| - checkInt(years);
|
| - checkInt(month);
|
| - checkInt(day);
|
| - checkInt(hours);
|
| - checkInt(minutes);
|
| - checkInt(seconds);
|
| - checkInt(milliseconds);
|
| - checkBool(isUtc);
|
| - var jsMonth = month - 1;
|
| - var value;
|
| - if (isUtc) {
|
| - value = JS('num', r'Date.UTC(#, #, #, #, #, #, #)',
|
| - years, jsMonth, day, hours, minutes, seconds, milliseconds);
|
| - } else {
|
| - value = JS('num', r'new Date(#, #, #, #, #, #, #).valueOf()',
|
| - years, jsMonth, day, hours, minutes, seconds, milliseconds);
|
| - }
|
| - if (value.isNaN ||
|
| - value < -MAX_MILLISECONDS_SINCE_EPOCH ||
|
| - value > MAX_MILLISECONDS_SINCE_EPOCH) {
|
| - return null;
|
| - }
|
| - if (years <= 0 || years < 100) return patchUpY2K(value, years, isUtc);
|
| - return value;
|
| - }
|
| -
|
| - static patchUpY2K(value, years, isUtc) {
|
| - var date = JS('', r'new Date(#)', value);
|
| - if (isUtc) {
|
| - JS('num', r'#.setUTCFullYear(#)', date, years);
|
| - } else {
|
| - JS('num', r'#.setFullYear(#)', date, years);
|
| - }
|
| - return JS('num', r'#.valueOf()', date);
|
| - }
|
| -
|
| - // Lazily keep a JS Date stored in the JS object.
|
| - static lazyAsJsDate(receiver) {
|
| - if (JS('bool', r'#.date === (void 0)', receiver)) {
|
| - JS('void', r'#.date = new Date(#)', receiver,
|
| - receiver.millisecondsSinceEpoch);
|
| - }
|
| - return JS('var', r'#.date', receiver);
|
| - }
|
| -
|
| - // The getters for date and time parts below add a positive integer to ensure
|
| - // that the result is really an integer, because the JavaScript implementation
|
| - // may return -0.0 instead of 0.
|
| -
|
| - static getYear(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('int', r'(#.getUTCFullYear() + 0)', lazyAsJsDate(receiver))
|
| - : JS('int', r'(#.getFullYear() + 0)', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getMonth(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('JSUInt31', r'#.getUTCMonth() + 1', lazyAsJsDate(receiver))
|
| - : JS('JSUInt31', r'#.getMonth() + 1', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getDay(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('JSUInt31', r'(#.getUTCDate() + 0)', lazyAsJsDate(receiver))
|
| - : JS('JSUInt31', r'(#.getDate() + 0)', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getHours(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('JSUInt31', r'(#.getUTCHours() + 0)', lazyAsJsDate(receiver))
|
| - : JS('JSUInt31', r'(#.getHours() + 0)', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getMinutes(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('JSUInt31', r'(#.getUTCMinutes() + 0)', lazyAsJsDate(receiver))
|
| - : JS('JSUInt31', r'(#.getMinutes() + 0)', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getSeconds(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('JSUInt31', r'(#.getUTCSeconds() + 0)', lazyAsJsDate(receiver))
|
| - : JS('JSUInt31', r'(#.getSeconds() + 0)', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getMilliseconds(receiver) {
|
| - return (receiver.isUtc)
|
| - ? JS('JSUInt31', r'(#.getUTCMilliseconds() + 0)', lazyAsJsDate(receiver))
|
| - : JS('JSUInt31', r'(#.getMilliseconds() + 0)', lazyAsJsDate(receiver));
|
| - }
|
| -
|
| - static getWeekday(receiver) {
|
| - int weekday = (receiver.isUtc)
|
| - ? JS('int', r'#.getUTCDay() + 0', lazyAsJsDate(receiver))
|
| - : JS('int', r'#.getDay() + 0', lazyAsJsDate(receiver));
|
| - // Adjust by one because JS weeks start on Sunday.
|
| - return (weekday + 6) % 7 + 1;
|
| - }
|
| -
|
| - static valueFromDateString(str) {
|
| - if (str is !String) throw argumentErrorValue(str);
|
| - var value = JS('num', r'Date.parse(#)', str);
|
| - if (value.isNaN) throw argumentErrorValue(str);
|
| - return value;
|
| - }
|
| -
|
| - static getProperty(object, key) {
|
| - if (object == null || object is bool || object is num || object is String) {
|
| - throw argumentErrorValue(object);
|
| - }
|
| - return JS('var', '#[#]', object, key);
|
| - }
|
| -
|
| - static void setProperty(object, key, value) {
|
| - if (object == null || object is bool || object is num || object is String) {
|
| - throw argumentErrorValue(object);
|
| - }
|
| - JS('void', '#[#] = #', object, key, value);
|
| - }
|
| -
|
| - static functionNoSuchMethod(function,
|
| - List positionalArguments,
|
| - Map<String, dynamic> namedArguments) {
|
| - int argumentCount = 0;
|
| - List arguments = [];
|
| - List namedArgumentList = [];
|
| -
|
| - if (positionalArguments != null) {
|
| - argumentCount += positionalArguments.length;
|
| - arguments.addAll(positionalArguments);
|
| - }
|
| -
|
| - String names = '';
|
| - if (namedArguments != null && !namedArguments.isEmpty) {
|
| - namedArguments.forEach((String name, argument) {
|
| - names = '$names\$$name';
|
| - namedArgumentList.add(name);
|
| - arguments.add(argument);
|
| - argumentCount++;
|
| - });
|
| - }
|
| -
|
| - String selectorName =
|
| - '${JS_GET_NAME(JsGetName.CALL_PREFIX)}\$$argumentCount$names';
|
| -
|
| - return function.noSuchMethod(
|
| - createUnmangledInvocationMirror(
|
| - #call,
|
| - selectorName,
|
| - JSInvocationMirror.METHOD,
|
| - arguments,
|
| - namedArgumentList));
|
| - }
|
| -
|
| - static applyFunctionNewEmitter(Function function,
|
| - List positionalArguments,
|
| - Map<String, dynamic> namedArguments) {
|
| - if (namedArguments == null) {
|
| - int requiredParameterCount = JS('int', r'#[#]', function,
|
| - JS_GET_NAME(JsGetName.REQUIRED_PARAMETER_PROPERTY));
|
| - int argumentCount = positionalArguments.length;
|
| - if (argumentCount < requiredParameterCount) {
|
| - return functionNoSuchMethod(function, positionalArguments, null);
|
| - }
|
| - String selectorName =
|
| - '${JS_GET_NAME(JsGetName.CALL_PREFIX)}\$$argumentCount';
|
| - var jsStub = JS('var', r'#[#]', function, selectorName);
|
| - if (jsStub == null) {
|
| - // Do a dynamic call.
|
| - var interceptor = getInterceptor(function);
|
| - var jsFunction = JS('', '#[#]', interceptor,
|
| - JS_GET_NAME(JsGetName.CALL_CATCH_ALL));
|
| - var defaultValues = JS('var', r'#[#]', function,
|
| - JS_GET_NAME(JsGetName.DEFAULT_VALUES_PROPERTY));
|
| - if (!JS('bool', '# instanceof Array', defaultValues)) {
|
| - // The function expects named arguments!
|
| - return functionNoSuchMethod(function, positionalArguments, null);
|
| - }
|
| - int defaultsLength = JS('int', "#.length", defaultValues);
|
| - int maxArguments = requiredParameterCount + defaultsLength;
|
| - if (argumentCount > maxArguments) {
|
| - // The function expects less arguments!
|
| - return functionNoSuchMethod(function, positionalArguments, null);
|
| - }
|
| - List arguments = new List.from(positionalArguments);
|
| - List missingDefaults = JS('JSArray', '#.slice(#)', defaultValues,
|
| - argumentCount - requiredParameterCount);
|
| - arguments.addAll(missingDefaults);
|
| - return JS('var', '#.apply(#, #)', jsFunction, function, arguments);
|
| - }
|
| - return JS('var', '#.apply(#, #)', jsStub, function, positionalArguments);
|
| - } else {
|
| - var interceptor = getInterceptor(function);
|
| - var jsFunction = JS('', '#[#]', interceptor,
|
| - JS_GET_NAME(JsGetName.CALL_CATCH_ALL));
|
| - var defaultValues = JS('JSArray', r'#[#]', function,
|
| - JS_GET_NAME(JsGetName.DEFAULT_VALUES_PROPERTY));
|
| - List keys = JS('JSArray', r'Object.keys(#)', defaultValues);
|
| - List arguments = new List.from(positionalArguments);
|
| - int used = 0;
|
| - for (String key in keys) {
|
| - var value = namedArguments[key];
|
| - if (value != null) {
|
| - used++;
|
| - arguments.add(value);
|
| - } else {
|
| - arguments.add(JS('var', r'#[#]', defaultValues, key));
|
| - }
|
| - }
|
| - if (used != namedArguments.length) {
|
| - return functionNoSuchMethod(function, positionalArguments,
|
| - namedArguments);
|
| - }
|
| - return JS('var', r'#.apply(#, #)', jsFunction, function, arguments);
|
| - }
|
| - }
|
| -
|
| - static applyFunction(Function function,
|
| - List positionalArguments,
|
| - Map<String, dynamic> namedArguments) {
|
| - // Dispatch on presence of named arguments to improve tree-shaking.
|
| - //
|
| - // This dispatch is as simple as possible to help the compiler detect the
|
| - // common case of `null` namedArguments, either via inlining or
|
| - // specialization.
|
| - return namedArguments == null
|
| - ? applyFunctionWithPositionalArguments(
|
| - function, positionalArguments)
|
| - : applyFunctionWithNamedArguments(
|
| - function, positionalArguments, namedArguments);
|
| - }
|
| -
|
| - static applyFunctionWithPositionalArguments(Function function,
|
| - List positionalArguments) {
|
| - List arguments;
|
| -
|
| - if (positionalArguments != null) {
|
| - if (JS('bool', '# instanceof Array', positionalArguments)) {
|
| - arguments = JS('JSArray', '#', positionalArguments);
|
| - } else {
|
| - arguments = new List.from(positionalArguments);
|
| - }
|
| - } else {
|
| - arguments = [];
|
| - }
|
| -
|
| - if (arguments.length == 0) {
|
| - String selectorName = JS_GET_NAME(JsGetName.CALL_PREFIX0);
|
| - if (JS('bool', '!!#[#]', function, selectorName)) {
|
| - return JS('', '#[#]()', function, selectorName);
|
| - }
|
| - } else if (arguments.length == 1) {
|
| - String selectorName = JS_GET_NAME(JsGetName.CALL_PREFIX1);
|
| - if (JS('bool', '!!#[#]', function, selectorName)) {
|
| - return JS('', '#[#](#[0])', function, selectorName, arguments);
|
| - }
|
| - } else if (arguments.length == 2) {
|
| - String selectorName = JS_GET_NAME(JsGetName.CALL_PREFIX2);
|
| - if (JS('bool', '!!#[#]', function, selectorName)) {
|
| - return JS('', '#[#](#[0],#[1])', function, selectorName,
|
| - arguments, arguments);
|
| - }
|
| - } else if (arguments.length == 3) {
|
| - String selectorName = JS_GET_NAME(JsGetName.CALL_PREFIX3);
|
| - if (JS('bool', '!!#[#]', function, selectorName)) {
|
| - return JS('', '#[#](#[0],#[1],#[2])', function, selectorName,
|
| - arguments, arguments, arguments);
|
| - }
|
| - }
|
| - return _genericApplyFunctionWithPositionalArguments(function, arguments);
|
| - }
|
| -
|
| - static _genericApplyFunctionWithPositionalArguments(Function function,
|
| - List arguments) {
|
| - int argumentCount = arguments.length;
|
| - String selectorName =
|
| - '${JS_GET_NAME(JsGetName.CALL_PREFIX)}\$$argumentCount';
|
| - var jsFunction = JS('var', '#[#]', function, selectorName);
|
| - if (jsFunction == null) {
|
| - var interceptor = getInterceptor(function);
|
| - jsFunction = JS('', '#["call*"]', interceptor);
|
| -
|
| - if (jsFunction == null) {
|
| - return functionNoSuchMethod(function, arguments, null);
|
| - }
|
| - ReflectionInfo info = new ReflectionInfo(jsFunction);
|
| - int requiredArgumentCount = info.requiredParameterCount;
|
| - int maxArgumentCount = requiredArgumentCount +
|
| - info.optionalParameterCount;
|
| - if (info.areOptionalParametersNamed ||
|
| - requiredArgumentCount > argumentCount ||
|
| - maxArgumentCount < argumentCount) {
|
| - return functionNoSuchMethod(function, arguments, null);
|
| - }
|
| - arguments = new List.from(arguments);
|
| - for (int pos = argumentCount; pos < maxArgumentCount; pos++) {
|
| - arguments.add(getMetadata(info.defaultValue(pos)));
|
| - }
|
| - }
|
| - // We bound 'this' to [function] because of how we compile
|
| - // closures: escaped local variables are stored and accessed through
|
| - // [function].
|
| - return JS('var', '#.apply(#, #)', jsFunction, function, arguments);
|
| - }
|
| -
|
| - static applyFunctionWithNamedArguments(Function function,
|
| - List positionalArguments,
|
| - Map<String, dynamic> namedArguments) {
|
| - if (namedArguments.isEmpty) {
|
| - return applyFunctionWithPositionalArguments(
|
| - function, positionalArguments);
|
| - }
|
| - // TODO(ahe): The following code can be shared with
|
| - // JsInstanceMirror.invoke.
|
| - var interceptor = getInterceptor(function);
|
| - var jsFunction = JS('', '#[#]', interceptor,
|
| - JS_GET_NAME(JsGetName.CALL_CATCH_ALL));
|
| -
|
| - if (jsFunction == null) {
|
| - return functionNoSuchMethod(
|
| - function, positionalArguments, namedArguments);
|
| - }
|
| - ReflectionInfo info = new ReflectionInfo(jsFunction);
|
| - if (info == null || !info.areOptionalParametersNamed) {
|
| - return functionNoSuchMethod(
|
| - function, positionalArguments, namedArguments);
|
| - }
|
| -
|
| - if (positionalArguments != null) {
|
| - positionalArguments = new List.from(positionalArguments);
|
| - } else {
|
| - positionalArguments = [];
|
| - }
|
| - // Check the number of positional arguments is valid.
|
| - if (info.requiredParameterCount != positionalArguments.length) {
|
| - return functionNoSuchMethod(
|
| - function, positionalArguments, namedArguments);
|
| - }
|
| - var defaultArguments = new Map();
|
| - for (int i = 0; i < info.optionalParameterCount; i++) {
|
| - int index = i + info.requiredParameterCount;
|
| - var parameterName = info.parameterNameInOrder(index);
|
| - var value = info.defaultValueInOrder(index);
|
| - var defaultValue = getMetadata(value);
|
| - defaultArguments[parameterName] = defaultValue;
|
| - }
|
| - bool bad = false;
|
| - namedArguments.forEach((String parameter, value) {
|
| - if (defaultArguments.containsKey(parameter)) {
|
| - defaultArguments[parameter] = value;
|
| - } else {
|
| - // Extraneous named argument.
|
| - bad = true;
|
| - }
|
| - });
|
| - if (bad) {
|
| - return functionNoSuchMethod(
|
| - function, positionalArguments, namedArguments);
|
| - }
|
| - positionalArguments.addAll(defaultArguments.values);
|
| - return JS('', '#.apply(#, #)', jsFunction, function, positionalArguments);
|
| - }
|
| -
|
| - static bool identicalImplementation(a, b) {
|
| - return JS('bool', '# == null', a)
|
| - ? JS('bool', '# == null', b)
|
| - : JS('bool', '# === #', a, b);
|
| - }
|
| -
|
| - static StackTrace extractStackTrace(Error error) {
|
| - return getTraceFromException(JS('', r'#.$thrownJsError', error));
|
| - }
|
| -}
|
| -
|
| -/// Helper class for allocating and using JS object literals as caches.
|
| -class JsCache {
|
| - /// Returns a JavaScript object suitable for use as a cache.
|
| - static allocate() {
|
| - var result = JS('=Object', 'Object.create(null)');
|
| - // Deleting a property makes V8 assume that it shouldn't create a hidden
|
| - // class for [result] and map transitions. Although these map transitions
|
| - // pay off if there are many cache hits for the same keys, it becomes
|
| - // really slow when there aren't many repeated hits.
|
| - JS('void', '#.x=0', result);
|
| - JS('void', 'delete #.x', result);
|
| - return result;
|
| - }
|
| -
|
| - static fetch(cache, String key) {
|
| - return JS('', '#[#]', cache, key);
|
| - }
|
| -
|
| - static void update(cache, String key, value) {
|
| - JS('void', '#[#] = #', cache, key, value);
|
| - }
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code to throw an illegal-argument exception,
|
| - * for example, if a non-integer index is given to an optimized
|
| - * indexed access.
|
| - */
|
| -@NoInline()
|
| -iae(argument) {
|
| - throw argumentErrorValue(argument);
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code to throw an index-out-of-range exception, for
|
| - * example, if a bounds check fails in an optimized indexed access. This may
|
| - * also be called when the index is not an integer, in which case it throws an
|
| - * illegal-argument exception instead, like [iae], or when the receiver is null.
|
| - */
|
| -@NoInline()
|
| -ioore(receiver, index) {
|
| - if (receiver == null) receiver.length; // Force a NoSuchMethodError.
|
| - throw diagnoseIndexError(receiver, index);
|
| -}
|
| -
|
| -/**
|
| - * Diagnoses an indexing error. Returns the ArgumentError or RangeError that
|
| - * describes the problem.
|
| - */
|
| -@NoInline()
|
| -Error diagnoseIndexError(indexable, index) {
|
| - if (index is !int) return new ArgumentError.value(index, 'index');
|
| - int length = indexable.length;
|
| - // The following returns the same error that would be thrown by calling
|
| - // [RangeError.checkValidIndex] with no optional parameters provided.
|
| - if (index < 0 || index >= length) {
|
| - return new RangeError.index(index, indexable, 'index', null, length);
|
| - }
|
| - // The above should always match, but if it does not, use the following.
|
| - return new RangeError.value(index, 'index');
|
| -}
|
| -
|
| -
|
| -stringLastIndexOfUnchecked(receiver, element, start)
|
| - => JS('int', r'#.lastIndexOf(#, #)', receiver, element, start);
|
| -
|
| -
|
| -/// 'factory' for constructing ArgumentError.value to keep the call sites small.
|
| -@NoInline()
|
| -ArgumentError argumentErrorValue(object) {
|
| - return new ArgumentError.value(object);
|
| -}
|
| -
|
| -checkNull(object) {
|
| - if (object == null) throw argumentErrorValue(object);
|
| - return object;
|
| -}
|
| -
|
| -checkNum(value) {
|
| - if (value is !num) throw argumentErrorValue(value);
|
| - return value;
|
| -}
|
| -
|
| -checkInt(value) {
|
| - if (value is !int) throw argumentErrorValue(value);
|
| - return value;
|
| -}
|
| -
|
| -checkBool(value) {
|
| - if (value is !bool) throw argumentErrorValue(value);
|
| - return value;
|
| -}
|
| -
|
| -checkString(value) {
|
| - if (value is !String) throw argumentErrorValue(value);
|
| - return value;
|
| -}
|
| -
|
| -/**
|
| - * Wrap the given Dart object and record a stack trace.
|
| - *
|
| - * The code in [unwrapException] deals with getting the original Dart
|
| - * object out of the wrapper again.
|
| - */
|
| -@NoInline()
|
| -wrapException(ex) {
|
| - if (ex == null) ex = new NullThrownError();
|
| - var wrapper = JS('', 'new Error()');
|
| - // [unwrapException] looks for the property 'dartException'.
|
| - JS('void', '#.dartException = #', wrapper, ex);
|
| -
|
| - if (JS('bool', '"defineProperty" in Object')) {
|
| - // Define a JavaScript getter for 'message'. This is to work around V8 bug
|
| - // (https://code.google.com/p/v8/issues/detail?id=2519). The default
|
| - // toString on Error returns the value of 'message' if 'name' is
|
| - // empty. Setting toString directly doesn't work, see the bug.
|
| - JS('void', 'Object.defineProperty(#, "message", { get: # })',
|
| - wrapper, DART_CLOSURE_TO_JS(toStringWrapper));
|
| - JS('void', '#.name = ""', wrapper);
|
| - } else {
|
| - // In the unlikely event the browser doesn't support Object.defineProperty,
|
| - // hope that it just calls toString.
|
| - JS('void', '#.toString = #', wrapper, DART_CLOSURE_TO_JS(toStringWrapper));
|
| - }
|
| -
|
| - return wrapper;
|
| -}
|
| -
|
| -/// Do not call directly.
|
| -toStringWrapper() {
|
| - // This method gets installed as toString on a JavaScript object. Due to the
|
| - // weird scope rules of JavaScript, JS 'this' will refer to that object.
|
| - return JS('', r'this.dartException').toString();
|
| -}
|
| -
|
| -/**
|
| - * This wraps the exception and does the throw. It is possible to call this in
|
| - * a JS expression context, where the throw statement is not allowed. Helpers
|
| - * are never inlined, so we don't risk inlining the throw statement into an
|
| - * expression context.
|
| - */
|
| -throwExpression(ex) {
|
| - JS('void', 'throw #', wrapException(ex));
|
| -}
|
| -
|
| -throwRuntimeError(message) {
|
| - throw new RuntimeError(message);
|
| -}
|
| -
|
| -throwAbstractClassInstantiationError(className) {
|
| - throw new AbstractClassInstantiationError(className);
|
| -}
|
| -
|
| -// This is used in open coded for-in loops on arrays.
|
| -//
|
| -// checkConcurrentModificationError(a.length == startLength, a)
|
| -//
|
| -// is replaced in codegen by:
|
| -//
|
| -// a.length == startLength || throwConcurrentModificationError(a)
|
| -//
|
| -// TODO(sra): We would like to annotate this as @NoSideEffects() so that loops
|
| -// with no other effects can recognize that the array length does not
|
| -// change. However, in the usual case where the loop does have other effects,
|
| -// that causes the length in the loop condition to be phi(startLength,a.length),
|
| -// which causes confusion in range analysis and the insertion of a bounds check.
|
| -@NoInline()
|
| -checkConcurrentModificationError(sameLength, collection) {
|
| - if (true != sameLength) {
|
| - throwConcurrentModificationError(collection);
|
| - }
|
| -}
|
| -
|
| -@NoInline()
|
| -throwConcurrentModificationError(collection) {
|
| - throw new ConcurrentModificationError(collection);
|
| -}
|
| -
|
| -/**
|
| - * Helper class for building patterns recognizing native type errors.
|
| - */
|
| -class TypeErrorDecoder {
|
| - // Field names are private to help tree-shaking.
|
| -
|
| - /// A regular expression which matches is matched against an error message.
|
| - final String _pattern;
|
| -
|
| - /// The group index of "arguments" in [_pattern], or -1 if _pattern has no
|
| - /// match for "arguments".
|
| - final int _arguments;
|
| -
|
| - /// The group index of "argumentsExpr" in [_pattern], or -1 if _pattern has
|
| - /// no match for "argumentsExpr".
|
| - final int _argumentsExpr;
|
| -
|
| - /// The group index of "expr" in [_pattern], or -1 if _pattern has no match
|
| - /// for "expr".
|
| - final int _expr;
|
| -
|
| - /// The group index of "method" in [_pattern], or -1 if _pattern has no match
|
| - /// for "method".
|
| - final int _method;
|
| -
|
| - /// The group index of "receiver" in [_pattern], or -1 if _pattern has no
|
| - /// match for "receiver".
|
| - final int _receiver;
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError error (and
|
| - /// possibly extract the method name).
|
| - static final TypeErrorDecoder noSuchMethodPattern =
|
| - extractPattern(provokeCallErrorOn(buildJavaScriptObject()));
|
| -
|
| - /// Pattern used to recognize an "object not a closure" error (and
|
| - /// possibly extract the method name).
|
| - static final TypeErrorDecoder notClosurePattern =
|
| - extractPattern(provokeCallErrorOn(buildJavaScriptObjectWithNonClosure()));
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript null
|
| - /// call.
|
| - static final TypeErrorDecoder nullCallPattern =
|
| - extractPattern(provokeCallErrorOn(JS('', 'null')));
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript literal null
|
| - /// call.
|
| - static final TypeErrorDecoder nullLiteralCallPattern =
|
| - extractPattern(provokeCallErrorOnNull());
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript
|
| - /// undefined call.
|
| - static final TypeErrorDecoder undefinedCallPattern =
|
| - extractPattern(provokeCallErrorOn(JS('', 'void 0')));
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript literal
|
| - /// undefined call.
|
| - static final TypeErrorDecoder undefinedLiteralCallPattern =
|
| - extractPattern(provokeCallErrorOnUndefined());
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript null
|
| - /// property access.
|
| - static final TypeErrorDecoder nullPropertyPattern =
|
| - extractPattern(provokePropertyErrorOn(JS('', 'null')));
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript literal null
|
| - /// property access.
|
| - static final TypeErrorDecoder nullLiteralPropertyPattern =
|
| - extractPattern(provokePropertyErrorOnNull());
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript
|
| - /// undefined property access.
|
| - static final TypeErrorDecoder undefinedPropertyPattern =
|
| - extractPattern(provokePropertyErrorOn(JS('', 'void 0')));
|
| -
|
| - /// Pattern used to recognize a NoSuchMethodError on JavaScript literal
|
| - /// undefined property access.
|
| - static final TypeErrorDecoder undefinedLiteralPropertyPattern =
|
| - extractPattern(provokePropertyErrorOnUndefined());
|
| -
|
| - TypeErrorDecoder(this._arguments,
|
| - this._argumentsExpr,
|
| - this._expr,
|
| - this._method,
|
| - this._receiver,
|
| - this._pattern);
|
| -
|
| - /// Returns a JavaScript object literal (map) with at most the
|
| - /// following keys:
|
| - ///
|
| - /// * arguments: The arguments as formatted by the JavaScript
|
| - /// engine. No browsers are known to provide this information.
|
| - ///
|
| - /// * argumentsExpr: The syntax of the arguments (JavaScript source
|
| - /// code). No browsers are known to provide this information.
|
| - ///
|
| - /// * expr: The syntax of the receiver expression (JavaScript source
|
| - /// code). Firefox provides this information, for example: "$expr$.$method$
|
| - /// is not a function".
|
| - ///
|
| - /// * method: The name of the called method (mangled name). At least Firefox
|
| - /// and Chrome/V8 provides this information, for example, "Object [object
|
| - /// Object] has no method '$method$'".
|
| - ///
|
| - /// * receiver: The string representation of the receiver. Chrome/V8
|
| - /// used to provide this information (by calling user-defined
|
| - /// JavaScript toString on receiver), but it has degenerated into
|
| - /// "[object Object]" in recent versions.
|
| - matchTypeError(message) {
|
| - var match = JS('JSExtendableArray|Null',
|
| - 'new RegExp(#).exec(#)', _pattern, message);
|
| - if (match == null) return null;
|
| - var result = JS('', 'Object.create(null)');
|
| - if (_arguments != -1) {
|
| - JS('', '#.arguments = #[# + 1]', result, match, _arguments);
|
| - }
|
| - if (_argumentsExpr != -1) {
|
| - JS('', '#.argumentsExpr = #[# + 1]', result, match, _argumentsExpr);
|
| - }
|
| - if (_expr != -1) {
|
| - JS('', '#.expr = #[# + 1]', result, match, _expr);
|
| - }
|
| - if (_method != -1) {
|
| - JS('', '#.method = #[# + 1]', result, match, _method);
|
| - }
|
| - if (_receiver != -1) {
|
| - JS('', '#.receiver = #[# + 1]', result, match, _receiver);
|
| - }
|
| -
|
| - return result;
|
| - }
|
| -
|
| - /// Builds a JavaScript Object with a toString method saying
|
| - /// r"$receiver$".
|
| - static buildJavaScriptObject() {
|
| - return JS('', r'{ toString: function() { return "$receiver$"; } }');
|
| - }
|
| -
|
| - /// Builds a JavaScript Object with a toString method saying
|
| - /// r"$receiver$". The property "$method" is defined, but is not a function.
|
| - static buildJavaScriptObjectWithNonClosure() {
|
| - return JS('', r'{ $method$: null, '
|
| - r'toString: function() { return "$receiver$"; } }');
|
| - }
|
| -
|
| - /// Extract a pattern from a JavaScript TypeError message.
|
| - ///
|
| - /// The patterns are extracted by forcing TypeErrors on known
|
| - /// objects thus forcing known strings into the error message. The
|
| - /// known strings are then replaced with wildcards which in theory
|
| - /// makes it possible to recognize the desired information even if
|
| - /// the error messages are reworded or translated.
|
| - static extractPattern(String message) {
|
| - // Some JavaScript implementations (V8 at least) include a
|
| - // representation of the receiver in the error message, however,
|
| - // this representation is not always [: receiver.toString() :],
|
| - // sometimes it is [: Object.prototype.toString(receiver) :], and
|
| - // sometimes it is an implementation specific method (but that
|
| - // doesn't seem to happen for object literals). So sometimes we
|
| - // get the text "[object Object]". The shortest way to get that
|
| - // string is using "String({})".
|
| - // See: http://code.google.com/p/v8/issues/detail?id=2519.
|
| - message = JS('String', r"#.replace(String({}), '$receiver$')", message);
|
| -
|
| - // Since we want to create a new regular expression from an unknown string,
|
| - // we must escape all regular expression syntax.
|
| - message = JS('String', r"#.replace(new RegExp(#, 'g'), '\\$&')",
|
| - message, ESCAPE_REGEXP);
|
| -
|
| - // Look for the special pattern \$camelCase\$ (all the $ symbols
|
| - // have been escaped already), as we will soon be inserting
|
| - // regular expression syntax that we want interpreted by RegExp.
|
| - List<String> match =
|
| - JS('JSExtendableArray|Null', r"#.match(/\\\$[a-zA-Z]+\\\$/g)", message);
|
| - if (match == null) match = [];
|
| -
|
| - // Find the positions within the substring matches of the error message
|
| - // components. This will help us extract information later, such as the
|
| - // method name.
|
| - int arguments = JS('int', '#.indexOf(#)', match, r'\$arguments\$');
|
| - int argumentsExpr = JS('int', '#.indexOf(#)', match, r'\$argumentsExpr\$');
|
| - int expr = JS('int', '#.indexOf(#)', match, r'\$expr\$');
|
| - int method = JS('int', '#.indexOf(#)', match, r'\$method\$');
|
| - int receiver = JS('int', '#.indexOf(#)', match, r'\$receiver\$');
|
| -
|
| - // Replace the patterns with a regular expression wildcard.
|
| - // Note: in a perfect world, one would use "(.*)", but not in
|
| - // JavaScript, "." does not match newlines.
|
| - String pattern = JS('String',
|
| - r"#.replace('\\$arguments\\$', '((?:x|[^x])*)')"
|
| - r".replace('\\$argumentsExpr\\$', '((?:x|[^x])*)')"
|
| - r".replace('\\$expr\\$', '((?:x|[^x])*)')"
|
| - r".replace('\\$method\\$', '((?:x|[^x])*)')"
|
| - r".replace('\\$receiver\\$', '((?:x|[^x])*)')",
|
| - message);
|
| -
|
| - return new TypeErrorDecoder(arguments,
|
| - argumentsExpr,
|
| - expr,
|
| - method,
|
| - receiver,
|
| - pattern);
|
| - }
|
| -
|
| - /// Provokes a TypeError and returns its message.
|
| - ///
|
| - /// The error is provoked so all known variable content can be recognized and
|
| - /// a pattern can be inferred.
|
| - static String provokeCallErrorOn(expression) {
|
| - // This function is carefully created to maximize the possibility
|
| - // of decoding the TypeError message and turning it into a general
|
| - // pattern.
|
| - //
|
| - // The idea is to inject something known into something unknown. The
|
| - // unknown entity is the error message that the browser provides with a
|
| - // TypeError. It is a human readable message, possibly localized in a
|
| - // language no dart2js engineer understand. We assume that $name$ would
|
| - // never naturally occur in a human readable error message, yet it is easy
|
| - // to decode.
|
| - //
|
| - // For example, evaluate this in V8 version 3.13.7.6:
|
| - //
|
| - // var $expr$ = null; $expr$.$method$()
|
| - //
|
| - // The VM throws an instance of TypeError whose message property contains
|
| - // "Cannot call method '$method$' of null". We can then reasonably assume
|
| - // that if the string contains $method$, that's where the method name will
|
| - // be in general. Call this automatically reverse engineering the error
|
| - // format string in V8.
|
| - //
|
| - // So the error message from V8 is turned into this regular expression:
|
| - //
|
| - // "Cannot call method '(.*)' of null"
|
| - //
|
| - // Similarly, if we evaluate:
|
| - //
|
| - // var $expr$ = {toString: function() { return '$receiver$'; }};
|
| - // $expr$.$method$()
|
| - //
|
| - // We get this message: "Object $receiver$ has no method '$method$'"
|
| - //
|
| - // Which is turned into this regular expression:
|
| - //
|
| - // "Object (.*) has no method '(.*)'"
|
| - //
|
| - // Firefox/jsshell is slightly different, it tries to include the source
|
| - // code that caused the exception, so we get this message: "$expr$.$method$
|
| - // is not a function" which is turned into this regular expression:
|
| - //
|
| - // "(.*)\\.(.*) is not a function"
|
| -
|
| - var function = JS('', r"""function($expr$) {
|
| - var $argumentsExpr$ = '$arguments$';
|
| - try {
|
| - $expr$.$method$($argumentsExpr$);
|
| - } catch (e) {
|
| - return e.message;
|
| - }
|
| -}""");
|
| - return JS('String', '(#)(#)', function, expression);
|
| - }
|
| -
|
| - /// Similar to [provokeCallErrorOn], but provokes an error directly on
|
| - /// literal "null" expression.
|
| - static String provokeCallErrorOnNull() {
|
| - // See [provokeCallErrorOn] for a detailed explanation.
|
| - var function = JS('', r"""function() {
|
| - var $argumentsExpr$ = '$arguments$';
|
| - try {
|
| - null.$method$($argumentsExpr$);
|
| - } catch (e) {
|
| - return e.message;
|
| - }
|
| -}""");
|
| - return JS('String', '(#)()', function);
|
| - }
|
| -
|
| - /// Similar to [provokeCallErrorOnNull], but provokes an error directly on
|
| - /// (void 0), that is, "undefined".
|
| - static String provokeCallErrorOnUndefined() {
|
| - // See [provokeCallErrorOn] for a detailed explanation.
|
| - var function = JS('', r"""function() {
|
| - var $argumentsExpr$ = '$arguments$';
|
| - try {
|
| - (void 0).$method$($argumentsExpr$);
|
| - } catch (e) {
|
| - return e.message;
|
| - }
|
| -}""");
|
| - return JS('String', '(#)()', function);
|
| - }
|
| -
|
| - /// Similar to [provokeCallErrorOn], but provokes a property access
|
| - /// error.
|
| - static String provokePropertyErrorOn(expression) {
|
| - // See [provokeCallErrorOn] for a detailed explanation.
|
| - var function = JS('', r"""function($expr$) {
|
| - try {
|
| - $expr$.$method$;
|
| - } catch (e) {
|
| - return e.message;
|
| - }
|
| -}""");
|
| - return JS('String', '(#)(#)', function, expression);
|
| - }
|
| -
|
| - /// Similar to [provokePropertyErrorOn], but provokes an property access
|
| - /// error directly on literal "null" expression.
|
| - static String provokePropertyErrorOnNull() {
|
| - // See [provokeCallErrorOn] for a detailed explanation.
|
| - var function = JS('', r"""function() {
|
| - try {
|
| - null.$method$;
|
| - } catch (e) {
|
| - return e.message;
|
| - }
|
| -}""");
|
| - return JS('String', '(#)()', function);
|
| - }
|
| -
|
| - /// Similar to [provokePropertyErrorOnNull], but provokes an property access
|
| - /// error directly on (void 0), that is, "undefined".
|
| - static String provokePropertyErrorOnUndefined() {
|
| - // See [provokeCallErrorOn] for a detailed explanation.
|
| - var function = JS('', r"""function() {
|
| - try {
|
| - (void 0).$method$;
|
| - } catch (e) {
|
| - return e.message;
|
| - }
|
| -}""");
|
| - return JS('String', '(#)()', function);
|
| - }
|
| -}
|
| -
|
| -class NullError extends Error implements NoSuchMethodError {
|
| - final String _message;
|
| - final String _method;
|
| -
|
| - NullError(this._message, match)
|
| - : _method = match == null ? null : JS('', '#.method', match);
|
| -
|
| - String toString() {
|
| - if (_method == null) return 'NullError: $_message';
|
| - return "NullError: method not found: '$_method' on null";
|
| - }
|
| -}
|
| -
|
| -class JsNoSuchMethodError extends Error implements NoSuchMethodError {
|
| - final String _message;
|
| - final String _method;
|
| - final String _receiver;
|
| -
|
| - JsNoSuchMethodError(this._message, match)
|
| - : _method = match == null ? null : JS('String|Null', '#.method', match),
|
| - _receiver =
|
| - match == null ? null : JS('String|Null', '#.receiver', match);
|
| -
|
| - String toString() {
|
| - if (_method == null) return 'NoSuchMethodError: $_message';
|
| - if (_receiver == null) {
|
| - return "NoSuchMethodError: method not found: '$_method' ($_message)";
|
| - }
|
| - return "NoSuchMethodError: "
|
| - "method not found: '$_method' on '$_receiver' ($_message)";
|
| - }
|
| -}
|
| -
|
| -class UnknownJsTypeError extends Error {
|
| - final String _message;
|
| -
|
| - UnknownJsTypeError(this._message);
|
| -
|
| - String toString() => _message.isEmpty ? 'Error' : 'Error: $_message';
|
| -}
|
| -
|
| -/**
|
| - * Called from catch blocks in generated code to extract the Dart
|
| - * exception from the thrown value. The thrown value may have been
|
| - * created by [wrapException] or it may be a 'native' JS exception.
|
| - *
|
| - * Some native exceptions are mapped to new Dart instances, others are
|
| - * returned unmodified.
|
| - */
|
| -unwrapException(ex) {
|
| - /// If error implements Error, save [ex] in [error.$thrownJsError].
|
| - /// Otherwise, do nothing. Later, the stack trace can then be extraced from
|
| - /// [ex].
|
| - saveStackTrace(error) {
|
| - if (error is Error) {
|
| - var thrownStackTrace = JS('', r'#.$thrownJsError', error);
|
| - if (thrownStackTrace == null) {
|
| - JS('void', r'#.$thrownJsError = #', error, ex);
|
| - }
|
| - }
|
| - return error;
|
| - }
|
| -
|
| - // Note that we are checking if the object has the property. If it
|
| - // has, it could be set to null if the thrown value is null.
|
| - if (ex == null) return null;
|
| - if (ex is ExceptionAndStackTrace) {
|
| - return saveStackTrace(ex.dartException);
|
| - }
|
| - if (JS('bool', 'typeof # !== "object"', ex)) return ex;
|
| -
|
| - if (JS('bool', r'"dartException" in #', ex)) {
|
| - return saveStackTrace(JS('', r'#.dartException', ex));
|
| - } else if (!JS('bool', r'"message" in #', ex)) {
|
| - return ex;
|
| - }
|
| -
|
| - // Grab hold of the exception message. This field is available on
|
| - // all supported browsers.
|
| - var message = JS('var', r'#.message', ex);
|
| -
|
| - // Internet Explorer has an error number. This is the most reliable way to
|
| - // detect specific errors, so check for this first.
|
| - if (JS('bool', '"number" in #', ex)
|
| - && JS('bool', 'typeof #.number == "number"', ex)) {
|
| - int number = JS('int', '#.number', ex);
|
| -
|
| - // From http://msdn.microsoft.com/en-us/library/ie/hc53e755(v=vs.94).aspx
|
| - // "number" is a 32-bit word. The error code is the low 16 bits, and the
|
| - // facility code is the upper 16 bits.
|
| - var ieErrorCode = number & 0xffff;
|
| - var ieFacilityNumber = (number >> 16) & 0x1fff;
|
| -
|
| - // http://msdn.microsoft.com/en-us/library/aa264975(v=vs.60).aspx
|
| - // http://msdn.microsoft.com/en-us/library/ie/1dk3k160(v=vs.94).aspx
|
| - if (ieFacilityNumber == 10) {
|
| - switch (ieErrorCode) {
|
| - case 438:
|
| - return saveStackTrace(
|
| - new JsNoSuchMethodError('$message (Error $ieErrorCode)', null));
|
| - case 445:
|
| - case 5007:
|
| - return saveStackTrace(
|
| - new NullError('$message (Error $ieErrorCode)', null));
|
| - }
|
| - }
|
| - }
|
| -
|
| - if (JS('bool', r'# instanceof TypeError', ex)) {
|
| - var match;
|
| - // Using JS to give type hints to the compiler to help tree-shaking.
|
| - // TODO(ahe): That should be unnecessary due to type inference.
|
| - var nsme = TypeErrorDecoder.noSuchMethodPattern;
|
| - var notClosure = TypeErrorDecoder.notClosurePattern;
|
| - var nullCall = TypeErrorDecoder.nullCallPattern;
|
| - var nullLiteralCall = TypeErrorDecoder.nullLiteralCallPattern;
|
| - var undefCall = TypeErrorDecoder.undefinedCallPattern;
|
| - var undefLiteralCall = TypeErrorDecoder.undefinedLiteralCallPattern;
|
| - var nullProperty = TypeErrorDecoder.nullPropertyPattern;
|
| - var nullLiteralProperty = TypeErrorDecoder.nullLiteralPropertyPattern;
|
| - var undefProperty = TypeErrorDecoder.undefinedPropertyPattern;
|
| - var undefLiteralProperty =
|
| - TypeErrorDecoder.undefinedLiteralPropertyPattern;
|
| - if ((match = nsme.matchTypeError(message)) != null) {
|
| - return saveStackTrace(new JsNoSuchMethodError(message, match));
|
| - } else if ((match = notClosure.matchTypeError(message)) != null) {
|
| - // notClosure may match "({c:null}).c()" or "({c:1}).c()", so we
|
| - // cannot tell if this an attempt to invoke call on null or a
|
| - // non-function object.
|
| - // But we do know the method name is "call".
|
| - JS('', '#.method = "call"', match);
|
| - return saveStackTrace(new JsNoSuchMethodError(message, match));
|
| - } else if ((match = nullCall.matchTypeError(message)) != null ||
|
| - (match = nullLiteralCall.matchTypeError(message)) != null ||
|
| - (match = undefCall.matchTypeError(message)) != null ||
|
| - (match = undefLiteralCall.matchTypeError(message)) != null ||
|
| - (match = nullProperty.matchTypeError(message)) != null ||
|
| - (match = nullLiteralCall.matchTypeError(message)) != null ||
|
| - (match = undefProperty.matchTypeError(message)) != null ||
|
| - (match = undefLiteralProperty.matchTypeError(message)) != null) {
|
| - return saveStackTrace(new NullError(message, match));
|
| - }
|
| -
|
| - // If we cannot determine what kind of error this is, we fall back
|
| - // to reporting this as a generic error. It's probably better than
|
| - // nothing.
|
| - return saveStackTrace(
|
| - new UnknownJsTypeError(message is String ? message : ''));
|
| - }
|
| -
|
| - if (JS('bool', r'# instanceof RangeError', ex)) {
|
| - if (message is String && contains(message, 'call stack')) {
|
| - return new StackOverflowError();
|
| - }
|
| -
|
| - // In general, a RangeError is thrown when trying to pass a number as an
|
| - // argument to a function that does not allow a range that includes that
|
| - // number. Translate to a Dart ArgumentError with the same message.
|
| - // TODO(sra): Translate to RangeError.
|
| - String message = tryStringifyException(ex);
|
| - if (message is String) {
|
| - message = JS('String', r'#.replace(/^RangeError:\s*/, "")', message);
|
| - }
|
| - return saveStackTrace(new ArgumentError(message));
|
| - }
|
| -
|
| - // Check for the Firefox specific stack overflow signal.
|
| - if (JS('bool',
|
| - r'typeof InternalError == "function" && # instanceof InternalError',
|
| - ex)) {
|
| - if (message is String && message == 'too much recursion') {
|
| - return new StackOverflowError();
|
| - }
|
| - }
|
| -
|
| - // Just return the exception. We should not wrap it because in case
|
| - // the exception comes from the DOM, it is a JavaScript
|
| - // object backed by a native Dart class.
|
| - return ex;
|
| -}
|
| -
|
| -String tryStringifyException(ex) {
|
| - // Since this function is called from [unwrapException] which is called from
|
| - // code injected into a catch-clause, use JavaScript try-catch to avoid a
|
| - // potential loop if stringifying crashes.
|
| - return JS('String|Null', r'''
|
| - (function(ex) {
|
| - try {
|
| - return String(ex);
|
| - } catch (e) {}
|
| - return null;
|
| - })(#)
|
| - ''', ex);
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code to fetch the stack trace from an
|
| - * exception. Should never return null.
|
| - */
|
| -StackTrace getTraceFromException(exception) {
|
| - if (exception is ExceptionAndStackTrace) {
|
| - return exception.stackTrace;
|
| - }
|
| - if (exception == null) return new _StackTrace(exception);
|
| - _StackTrace trace = JS('_StackTrace|Null', r'#.$cachedTrace', exception);
|
| - if (trace != null) return trace;
|
| - trace = new _StackTrace(exception);
|
| - return JS('_StackTrace', r'#.$cachedTrace = #', exception, trace);
|
| -}
|
| -
|
| -class _StackTrace implements StackTrace {
|
| - var _exception;
|
| - String _trace;
|
| - _StackTrace(this._exception);
|
| -
|
| - String toString() {
|
| - if (_trace != null) return JS('String', '#', _trace);
|
| -
|
| - String trace;
|
| - if (JS('bool', '# !== null', _exception) &&
|
| - JS('bool', 'typeof # === "object"', _exception)) {
|
| - trace = JS("String|Null", r"#.stack", _exception);
|
| - }
|
| - return _trace = (trace == null) ? '' : trace;
|
| - }
|
| -}
|
| -
|
| -int objectHashCode(var object) {
|
| - if (object == null || JS('bool', "typeof # != 'object'", object)) {
|
| - return object.hashCode;
|
| - } else {
|
| - return Primitives.objectHashCode(object);
|
| - }
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code to build a map literal. [keyValuePairs] is
|
| - * a list of key, value, key, value, ..., etc.
|
| - */
|
| -fillLiteralMap(keyValuePairs, Map result) {
|
| - // TODO(johnniwinther): Use JSArray to optimize this code instead of calling
|
| - // [getLength] and [getIndex].
|
| - int index = 0;
|
| - int length = getLength(keyValuePairs);
|
| - while (index < length) {
|
| - var key = getIndex(keyValuePairs, index++);
|
| - var value = getIndex(keyValuePairs, index++);
|
| - result[key] = value;
|
| - }
|
| - return result;
|
| -}
|
| -
|
| -invokeClosure(Function closure,
|
| - var isolate,
|
| - int numberOfArguments,
|
| - var arg1,
|
| - var arg2,
|
| - var arg3,
|
| - var arg4) {
|
| - if (numberOfArguments == 0) {
|
| - return JS_CALL_IN_ISOLATE(isolate, () => closure());
|
| - } else if (numberOfArguments == 1) {
|
| - return JS_CALL_IN_ISOLATE(isolate, () => closure(arg1));
|
| - } else if (numberOfArguments == 2) {
|
| - return JS_CALL_IN_ISOLATE(isolate, () => closure(arg1, arg2));
|
| - } else if (numberOfArguments == 3) {
|
| - return JS_CALL_IN_ISOLATE(isolate, () => closure(arg1, arg2, arg3));
|
| - } else if (numberOfArguments == 4) {
|
| - return JS_CALL_IN_ISOLATE(isolate, () => closure(arg1, arg2, arg3, arg4));
|
| - } else {
|
| - throw new Exception(
|
| - 'Unsupported number of arguments for wrapped closure');
|
| - }
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code to convert a Dart closure to a JS
|
| - * closure when the Dart closure is passed to the DOM.
|
| - */
|
| -convertDartClosureToJS(closure, int arity) {
|
| - if (closure == null) return null;
|
| - var function = JS('var', r'#.$identity', closure);
|
| - if (JS('bool', r'!!#', function)) return function;
|
| -
|
| - // We use $0 and $1 to not clash with variable names used by the
|
| - // compiler and/or minifier.
|
| - function = JS('var',
|
| - '(function(closure, arity, context, invoke) {'
|
| - ' return function(a1, a2, a3, a4) {'
|
| - ' return invoke(closure, context, arity, a1, a2, a3, a4);'
|
| - ' };'
|
| - '})(#,#,#,#)',
|
| - closure,
|
| - arity,
|
| - // Capture the current isolate now. Remember that "#"
|
| - // in JS is simply textual substitution of compiled
|
| - // expressions.
|
| - JS_CURRENT_ISOLATE_CONTEXT(),
|
| - DART_CLOSURE_TO_JS(invokeClosure));
|
| -
|
| - JS('void', r'#.$identity = #', closure, function);
|
| - return function;
|
| -}
|
| -
|
| -/**
|
| - * Super class for Dart closures.
|
| - */
|
| -abstract class Closure implements Function {
|
| - // TODO(ahe): These constants must be in sync with
|
| - // reflection_data_parser.dart.
|
| - static const FUNCTION_INDEX = 0;
|
| - static const NAME_INDEX = 1;
|
| - static const CALL_NAME_INDEX = 2;
|
| - static const REQUIRED_PARAMETER_INDEX = 3;
|
| - static const OPTIONAL_PARAMETER_INDEX = 4;
|
| - static const DEFAULT_ARGUMENTS_INDEX = 5;
|
| -
|
| - /**
|
| - * Global counter to prevent reusing function code objects.
|
| - *
|
| - * V8 will share the underlying function code objects when the same string is
|
| - * passed to "new Function". Shared function code objects can lead to
|
| - * sub-optimal performance due to polymorhism, and can be prevented by
|
| - * ensuring the strings are different.
|
| - */
|
| - static int functionCounter = 0;
|
| -
|
| - Closure();
|
| -
|
| - /**
|
| - * Creates a new closure class for use by implicit getters associated with a
|
| - * method.
|
| - *
|
| - * In other words, creates a tear-off closure.
|
| - *
|
| - * Called from [closureFromTearOff] as well as from reflection when tearing
|
| - * of a method via [:getField:].
|
| - *
|
| - * This method assumes that [functions] was created by the JavaScript function
|
| - * `addStubs` in `reflection_data_parser.dart`. That is, a list of JavaScript
|
| - * function objects with properties `$stubName` and `$callName`.
|
| - *
|
| - * Further assumes that [reflectionInfo] is the end of the array created by
|
| - * [dart2js.js_emitter.ContainerBuilder.addMemberMethod] starting with
|
| - * required parameter count or, in case of the new emitter, the runtime
|
| - * representation of the function's type.
|
| - *
|
| - * Caution: this function may be called when building constants.
|
| - * TODO(ahe): Don't call this function when building constants.
|
| - */
|
| - static fromTearOff(receiver,
|
| - List functions,
|
| - var reflectionInfo,
|
| - bool isStatic,
|
| - jsArguments,
|
| - String propertyName) {
|
| - JS_EFFECT(() {
|
| - BoundClosure.receiverOf(JS('BoundClosure', 'void 0'));
|
| - BoundClosure.selfOf(JS('BoundClosure', 'void 0'));
|
| - });
|
| - // TODO(ahe): All the place below using \$ should be rewritten to go
|
| - // through the namer.
|
| - var function = JS('', '#[#]', functions, 0);
|
| - String name = JS('String|Null', '#.\$stubName', function);
|
| - String callName = JS('String|Null', '#[#]', function,
|
| - JS_GET_NAME(JsGetName.CALL_NAME_PROPERTY));
|
| -
|
| - // This variable holds either an index into the types-table, or a function
|
| - // that can compute a function-rti. (The latter is necessary if the type
|
| - // is dependent on generic arguments).
|
| - var functionType;
|
| - if (reflectionInfo is List) {
|
| - JS('', '#.\$reflectionInfo = #', function, reflectionInfo);
|
| - ReflectionInfo info = new ReflectionInfo(function);
|
| - functionType = info.functionType;
|
| - } else {
|
| - functionType = reflectionInfo;
|
| - }
|
| -
|
| -
|
| - // function tmp() {};
|
| - // tmp.prototype = BC.prototype;
|
| - // var proto = new tmp;
|
| - // for each computed prototype property:
|
| - // proto[property] = ...;
|
| - // proto._init = BC;
|
| - // var dynClosureConstructor =
|
| - // new Function('self', 'target', 'receiver', 'name',
|
| - // 'this._init(self, target, receiver, name)');
|
| - // proto.constructor = dynClosureConstructor;
|
| - // dynClosureConstructor.prototype = proto;
|
| - // return dynClosureConstructor;
|
| -
|
| - // We need to create a new subclass of TearOffClosure, one of StaticClosure
|
| - // or BoundClosure. For this, we need to create an object whose prototype
|
| - // is the prototype is either StaticClosure.prototype or
|
| - // BoundClosure.prototype, respectively in pseudo JavaScript code. The
|
| - // simplest way to access the JavaScript construction function of a Dart
|
| - // class is to create an instance and access its constructor property.
|
| - // Creating an instance ensures that any lazy class initialization has taken
|
| - // place. The newly created instance could in theory be used directly as the
|
| - // prototype, but it might include additional fields that we don't need. So
|
| - // we only use the new instance to access the constructor property and use
|
| - // Object.create to create the desired prototype.
|
| - //
|
| - // TODO(sra): Perhaps cache the prototype to avoid the allocation.
|
| - var prototype = isStatic
|
| - ? JS('StaticClosure', 'Object.create(#.constructor.prototype)',
|
| - new StaticClosure())
|
| - : JS('BoundClosure', 'Object.create(#.constructor.prototype)',
|
| - new BoundClosure(null, null, null, null));
|
| -
|
| - JS('', '#.\$initialize = #', prototype, JS('', '#.constructor', prototype));
|
| - var constructor = isStatic
|
| - ? JS('', 'function(){this.\$initialize()}')
|
| - : isCsp
|
| - ? JS('', 'function(a,b,c,d) {this.\$initialize(a,b,c,d)}')
|
| - : JS('',
|
| - 'new Function("a,b,c,d", "this.\$initialize(a,b,c,d);" + #)',
|
| - functionCounter++);
|
| -
|
| - // It is necessary to set the constructor property, otherwise it will be
|
| - // "Object".
|
| - JS('', '#.constructor = #', prototype, constructor);
|
| -
|
| - JS('', '#.prototype = #', constructor, prototype);
|
| -
|
| - // Create a closure and "monkey" patch it with call stubs.
|
| - var trampoline = function;
|
| - var isIntercepted = false;
|
| - if (!isStatic) {
|
| - if (JS('bool', '#.length == 1', jsArguments)) {
|
| - // Intercepted call.
|
| - isIntercepted = true;
|
| - }
|
| - trampoline = forwardCallTo(receiver, function, isIntercepted);
|
| - JS('', '#.\$reflectionInfo = #', trampoline, reflectionInfo);
|
| - } else {
|
| - JS('', '#.\$name = #', prototype, propertyName);
|
| - }
|
| -
|
| - var signatureFunction;
|
| - if (JS('bool', 'typeof # == "number"', functionType)) {
|
| - // We cannot call [getType] here, since the types-metadata might not be
|
| - // set yet. This is, because fromTearOff might be called for constants
|
| - // when the program isn't completely set up yet.
|
| - //
|
| - // Note that we cannot just textually inline the call
|
| - // `getType(functionType)` since we cannot guarantee that the (then)
|
| - // captured variable `functionType` isn't reused.
|
| - signatureFunction =
|
| - JS('',
|
| - '''(function(t) {
|
| - return function(){ return #(t); };
|
| - })(#)''',
|
| - RAW_DART_FUNCTION_REF(getType),
|
| - functionType);
|
| - } else if (!isStatic
|
| - && JS('bool', 'typeof # == "function"', functionType)) {
|
| - var getReceiver = isIntercepted
|
| - ? RAW_DART_FUNCTION_REF(BoundClosure.receiverOf)
|
| - : RAW_DART_FUNCTION_REF(BoundClosure.selfOf);
|
| - signatureFunction = JS(
|
| - '',
|
| - 'function(f,r){'
|
| - 'return function(){'
|
| - 'return f.apply({\$receiver:r(this)},arguments)'
|
| - '}'
|
| - '}(#,#)', functionType, getReceiver);
|
| - } else {
|
| - throw 'Error in reflectionInfo.';
|
| - }
|
| -
|
| - JS('', '#[#] = #', prototype, JS_GET_NAME(JsGetName.SIGNATURE_NAME),
|
| - signatureFunction);
|
| -
|
| - JS('', '#[#] = #', prototype, callName, trampoline);
|
| - for (int i = 1; i < functions.length; i++) {
|
| - var stub = functions[i];
|
| - var stubCallName = JS('String|Null', '#[#]', stub,
|
| - JS_GET_NAME(JsGetName.CALL_NAME_PROPERTY));
|
| - if (stubCallName != null) {
|
| - JS('', '#[#] = #', prototype, stubCallName,
|
| - isStatic ? stub : forwardCallTo(receiver, stub, isIntercepted));
|
| - }
|
| - }
|
| -
|
| - JS('', '#[#] = #', prototype, JS_GET_NAME(JsGetName.CALL_CATCH_ALL),
|
| - trampoline);
|
| - String reqArgProperty = JS_GET_NAME(JsGetName.REQUIRED_PARAMETER_PROPERTY);
|
| - String defValProperty = JS_GET_NAME(JsGetName.DEFAULT_VALUES_PROPERTY);
|
| - JS('', '#.# = #.#', prototype, reqArgProperty, function, reqArgProperty);
|
| - JS('', '#.# = #.#', prototype, defValProperty, function, defValProperty);
|
| -
|
| - return constructor;
|
| - }
|
| -
|
| - static cspForwardCall(int arity, bool isSuperCall, String stubName,
|
| - function) {
|
| - var getSelf = RAW_DART_FUNCTION_REF(BoundClosure.selfOf);
|
| - // Handle intercepted stub-names with the default slow case.
|
| - if (isSuperCall) arity = -1;
|
| - switch (arity) {
|
| - case 0:
|
| - return JS(
|
| - '',
|
| - 'function(n,S){'
|
| - 'return function(){'
|
| - 'return S(this)[n]()'
|
| - '}'
|
| - '}(#,#)', stubName, getSelf);
|
| - case 1:
|
| - return JS(
|
| - '',
|
| - 'function(n,S){'
|
| - 'return function(a){'
|
| - 'return S(this)[n](a)'
|
| - '}'
|
| - '}(#,#)', stubName, getSelf);
|
| - case 2:
|
| - return JS(
|
| - '',
|
| - 'function(n,S){'
|
| - 'return function(a,b){'
|
| - 'return S(this)[n](a,b)'
|
| - '}'
|
| - '}(#,#)', stubName, getSelf);
|
| - case 3:
|
| - return JS(
|
| - '',
|
| - 'function(n,S){'
|
| - 'return function(a,b,c){'
|
| - 'return S(this)[n](a,b,c)'
|
| - '}'
|
| - '}(#,#)', stubName, getSelf);
|
| - case 4:
|
| - return JS(
|
| - '',
|
| - 'function(n,S){'
|
| - 'return function(a,b,c,d){'
|
| - 'return S(this)[n](a,b,c,d)'
|
| - '}'
|
| - '}(#,#)', stubName, getSelf);
|
| - case 5:
|
| - return JS(
|
| - '',
|
| - 'function(n,S){'
|
| - 'return function(a,b,c,d,e){'
|
| - 'return S(this)[n](a,b,c,d,e)'
|
| - '}'
|
| - '}(#,#)', stubName, getSelf);
|
| - default:
|
| - return JS(
|
| - '',
|
| - 'function(f,s){'
|
| - 'return function(){'
|
| - 'return f.apply(s(this),arguments)'
|
| - '}'
|
| - '}(#,#)', function, getSelf);
|
| - }
|
| - }
|
| -
|
| - static bool get isCsp => JS_GET_FLAG("USE_CONTENT_SECURITY_POLICY");
|
| -
|
| - static forwardCallTo(receiver, function, bool isIntercepted) {
|
| - if (isIntercepted) return forwardInterceptedCallTo(receiver, function);
|
| - String stubName = JS('String|Null', '#.\$stubName', function);
|
| - int arity = JS('int', '#.length', function);
|
| - var lookedUpFunction = JS("", "#[#]", receiver, stubName);
|
| - // The receiver[stubName] may not be equal to the function if we try to
|
| - // forward to a super-method. Especially when we create a bound closure
|
| - // of a super-call we need to make sure that we don't forward back to the
|
| - // dynamically looked up function.
|
| - bool isSuperCall = !identical(function, lookedUpFunction);
|
| -
|
| - if (isCsp || isSuperCall || arity >= 27) {
|
| - return cspForwardCall(arity, isSuperCall, stubName, function);
|
| - }
|
| -
|
| - if (arity == 0) {
|
| - return JS(
|
| - '',
|
| - '(new Function(#))()',
|
| - 'return function(){'
|
| - 'return this.${BoundClosure.selfFieldName()}.$stubName();'
|
| - '${functionCounter++}'
|
| - '}');
|
| - }
|
| - assert (1 <= arity && arity < 27);
|
| - String arguments = JS(
|
| - 'String',
|
| - '"abcdefghijklmnopqrstuvwxyz".split("").splice(0,#).join(",")',
|
| - arity);
|
| - return JS(
|
| - '',
|
| - '(new Function(#))()',
|
| - 'return function($arguments){'
|
| - 'return this.${BoundClosure.selfFieldName()}.$stubName($arguments);'
|
| - '${functionCounter++}'
|
| - '}');
|
| - }
|
| -
|
| - static cspForwardInterceptedCall(int arity, bool isSuperCall,
|
| - String name, function) {
|
| - var getSelf = RAW_DART_FUNCTION_REF(BoundClosure.selfOf);
|
| - var getReceiver = RAW_DART_FUNCTION_REF(BoundClosure.receiverOf);
|
| - // Handle intercepted stub-names with the default slow case.
|
| - if (isSuperCall) arity = -1;
|
| - switch (arity) {
|
| - case 0:
|
| - // Intercepted functions always takes at least one argument (the
|
| - // receiver).
|
| - throw new RuntimeError('Intercepted function with no arguments.');
|
| - case 1:
|
| - return JS(
|
| - '',
|
| - 'function(n,s,r){'
|
| - 'return function(){'
|
| - 'return s(this)[n](r(this))'
|
| - '}'
|
| - '}(#,#,#)', name, getSelf, getReceiver);
|
| - case 2:
|
| - return JS(
|
| - '',
|
| - 'function(n,s,r){'
|
| - 'return function(a){'
|
| - 'return s(this)[n](r(this),a)'
|
| - '}'
|
| - '}(#,#,#)', name, getSelf, getReceiver);
|
| - case 3:
|
| - return JS(
|
| - '',
|
| - 'function(n,s,r){'
|
| - 'return function(a,b){'
|
| - 'return s(this)[n](r(this),a,b)'
|
| - '}'
|
| - '}(#,#,#)', name, getSelf, getReceiver);
|
| - case 4:
|
| - return JS(
|
| - '',
|
| - 'function(n,s,r){'
|
| - 'return function(a,b,c){'
|
| - 'return s(this)[n](r(this),a,b,c)'
|
| - '}'
|
| - '}(#,#,#)', name, getSelf, getReceiver);
|
| - case 5:
|
| - return JS(
|
| - '',
|
| - 'function(n,s,r){'
|
| - 'return function(a,b,c,d){'
|
| - 'return s(this)[n](r(this),a,b,c,d)'
|
| - '}'
|
| - '}(#,#,#)', name, getSelf, getReceiver);
|
| - case 6:
|
| - return JS(
|
| - '',
|
| - 'function(n,s,r){'
|
| - 'return function(a,b,c,d,e){'
|
| - 'return s(this)[n](r(this),a,b,c,d,e)'
|
| - '}'
|
| - '}(#,#,#)', name, getSelf, getReceiver);
|
| - default:
|
| - return JS(
|
| - '',
|
| - 'function(f,s,r,a){'
|
| - 'return function(){'
|
| - 'a=[r(this)];'
|
| - 'Array.prototype.push.apply(a,arguments);'
|
| - 'return f.apply(s(this),a)'
|
| - '}'
|
| - '}(#,#,#)', function, getSelf, getReceiver);
|
| - }
|
| - }
|
| -
|
| - static forwardInterceptedCallTo(receiver, function) {
|
| - String selfField = BoundClosure.selfFieldName();
|
| - String receiverField = BoundClosure.receiverFieldName();
|
| - String stubName = JS('String|Null', '#.\$stubName', function);
|
| - int arity = JS('int', '#.length', function);
|
| - bool isCsp = JS_GET_FLAG("USE_CONTENT_SECURITY_POLICY");
|
| - var lookedUpFunction = JS("", "#[#]", receiver, stubName);
|
| - // The receiver[stubName] may not be equal to the function if we try to
|
| - // forward to a super-method. Especially when we create a bound closure
|
| - // of a super-call we need to make sure that we don't forward back to the
|
| - // dynamically looked up function.
|
| - bool isSuperCall = !identical(function, lookedUpFunction);
|
| -
|
| - if (isCsp || isSuperCall || arity >= 28) {
|
| - return cspForwardInterceptedCall(arity, isSuperCall, stubName,
|
| - function);
|
| - }
|
| - if (arity == 1) {
|
| - return JS(
|
| - '',
|
| - '(new Function(#))()',
|
| - 'return function(){'
|
| - 'return this.$selfField.$stubName(this.$receiverField);'
|
| - '${functionCounter++}'
|
| - '}');
|
| - }
|
| - assert(1 < arity && arity < 28);
|
| - String arguments = JS(
|
| - 'String',
|
| - '"abcdefghijklmnopqrstuvwxyz".split("").splice(0,#).join(",")',
|
| - arity - 1);
|
| - return JS(
|
| - '',
|
| - '(new Function(#))()',
|
| - 'return function($arguments){'
|
| - 'return this.$selfField.$stubName(this.$receiverField, $arguments);'
|
| - '${functionCounter++}'
|
| - '}');
|
| - }
|
| -
|
| - // The backend adds a special getter of the form
|
| - //
|
| - // Closure get call => this;
|
| - //
|
| - // to allow tearing off a closure from itself. We do this magically in the
|
| - // backend rather than simply adding it here, as we do not want this getter
|
| - // to be visible to resolution and the generation of extra stubs.
|
| -
|
| - String toString() {
|
| - String name = Primitives.objectTypeName(this);
|
| - return "Closure '$name'";
|
| - }
|
| -}
|
| -
|
| -/// Called from implicit method getter (aka tear-off).
|
| -closureFromTearOff(receiver,
|
| - functions,
|
| - reflectionInfo,
|
| - isStatic,
|
| - jsArguments,
|
| - name) {
|
| - return Closure.fromTearOff(
|
| - receiver,
|
| - JSArray.markFixedList(functions),
|
| - reflectionInfo is List ? JSArray.markFixedList(reflectionInfo)
|
| - : reflectionInfo,
|
| - JS('bool', '!!#', isStatic),
|
| - jsArguments,
|
| - JS('String', '#', name));
|
| -}
|
| -
|
| -/// Represents an implicit closure of a function.
|
| -abstract class TearOffClosure extends Closure {
|
| -}
|
| -
|
| -class StaticClosure extends TearOffClosure {
|
| - String toString() {
|
| - String name = JS('String|Null', '#.\$name', this);
|
| - if (name == null) return "Closure of unknown static method";
|
| - return "Closure '$name'";
|
| - }
|
| -}
|
| -
|
| -/// Represents a 'tear-off' or property extraction closure of an instance
|
| -/// method, that is an instance method bound to a specific receiver (instance).
|
| -class BoundClosure extends TearOffClosure {
|
| - /// The receiver or interceptor.
|
| - // TODO(ahe): This could just be the interceptor, we always know if
|
| - // we need the interceptor when generating the call method.
|
| - final _self;
|
| -
|
| - /// The method.
|
| - final _target;
|
| -
|
| - /// The receiver. Null if [_self] is not an interceptor.
|
| - final _receiver;
|
| -
|
| - /// The name of the function. Only used by the mirror system.
|
| - final String _name;
|
| -
|
| - BoundClosure(this._self, this._target, this._receiver, this._name);
|
| -
|
| - bool operator==(other) {
|
| - if (identical(this, other)) return true;
|
| - if (other is! BoundClosure) return false;
|
| - return JS('bool', '# === # && # === # && # === #',
|
| - _self, other._self,
|
| - _target, other._target,
|
| - _receiver, other._receiver);
|
| - }
|
| -
|
| - int get hashCode {
|
| - int receiverHashCode;
|
| - if (_receiver == null) {
|
| - // A bound closure on a regular Dart object, just use the
|
| - // identity hash code.
|
| - receiverHashCode = Primitives.objectHashCode(_self);
|
| - } else if (JS('String', 'typeof #', _receiver) != 'object') {
|
| - // A bound closure on a primitive JavaScript type. We
|
| - // use the hashCode method we define for those primitive types.
|
| - receiverHashCode = _receiver.hashCode;
|
| - } else {
|
| - // A bound closure on an intercepted native class, just use the
|
| - // identity hash code.
|
| - receiverHashCode = Primitives.objectHashCode(_receiver);
|
| - }
|
| - return receiverHashCode ^ Primitives.objectHashCode(_target);
|
| - }
|
| -
|
| - toString() {
|
| - var receiver = _receiver == null ? _self : _receiver;
|
| - return "Closure '$_name' of ${Primitives.objectToHumanReadableString(receiver)}";
|
| - }
|
| -
|
| - @NoInline()
|
| - static selfOf(BoundClosure closure) => closure._self;
|
| -
|
| - static targetOf(BoundClosure closure) => closure._target;
|
| -
|
| - @NoInline()
|
| - static receiverOf(BoundClosure closure) => closure._receiver;
|
| -
|
| - static nameOf(BoundClosure closure) => closure._name;
|
| -
|
| - static String selfFieldNameCache;
|
| -
|
| - static String selfFieldName() {
|
| - if (selfFieldNameCache == null) {
|
| - selfFieldNameCache = computeFieldNamed('self');
|
| - }
|
| - return selfFieldNameCache;
|
| - }
|
| -
|
| - static String receiverFieldNameCache;
|
| -
|
| - static String receiverFieldName() {
|
| - if (receiverFieldNameCache == null) {
|
| - receiverFieldNameCache = computeFieldNamed('receiver');
|
| - }
|
| - return receiverFieldNameCache;
|
| - }
|
| -
|
| - @NoInline() @NoSideEffects()
|
| - static String computeFieldNamed(String fieldName) {
|
| - var template = new BoundClosure('self', 'target', 'receiver', 'name');
|
| - var names = JSArray.markFixedList(
|
| - JS('', 'Object.getOwnPropertyNames(#)', template));
|
| - for (int i = 0; i < names.length; i++) {
|
| - var name = names[i];
|
| - if (JS('bool', '#[#] === #', template, name, fieldName)) {
|
| - return JS('String', '#', name);
|
| - }
|
| - }
|
| - }
|
| -}
|
| -
|
| -bool jsHasOwnProperty(var jsObject, String property) {
|
| - return JS('bool', r'#.hasOwnProperty(#)', jsObject, property);
|
| -}
|
| -
|
| -jsPropertyAccess(var jsObject, String property) {
|
| - return JS('var', r'#[#]', jsObject, property);
|
| -}
|
| -
|
| -/**
|
| - * Called at the end of unaborted switch cases to get the singleton
|
| - * FallThroughError exception that will be thrown.
|
| - */
|
| -getFallThroughError() => new FallThroughErrorImplementation();
|
| -
|
| -/**
|
| - * A metadata annotation describing the types instantiated by a native element.
|
| - *
|
| - * The annotation is valid on a native method and a field of a native class.
|
| - *
|
| - * By default, a field of a native class is seen as an instantiation point for
|
| - * all native classes that are a subtype of the field's type, and a native
|
| - * method is seen as an instantiation point fo all native classes that are a
|
| - * subtype of the method's return type, or the argument types of the declared
|
| - * type of the method's callback parameter.
|
| - *
|
| - * An @[Creates] annotation overrides the default set of instantiated types. If
|
| - * one or more @[Creates] annotations are present, the type of the native
|
| - * element is ignored, and the union of @[Creates] annotations is used instead.
|
| - * The names in the strings are resolved and the program will fail to compile
|
| - * with dart2js if they do not name types.
|
| - *
|
| - * The argument to [Creates] is a string. The string is parsed as the names of
|
| - * one or more types, separated by vertical bars `|`. There are some special
|
| - * names:
|
| - *
|
| - * * `=Object`. This means 'exactly Object', which is a plain JavaScript object
|
| - * with properties and none of the subtypes of Object.
|
| - *
|
| - * Example: we may know that a method always returns a specific implementation:
|
| - *
|
| - * @Creates('_NodeList')
|
| - * List<Node> getElementsByTagName(String tag) native;
|
| - *
|
| - * Useful trick: A method can be marked as not instantiating any native classes
|
| - * with the annotation `@Creates('Null')`. This is useful for fields on native
|
| - * classes that are used only in Dart code.
|
| - *
|
| - * @Creates('Null')
|
| - * var _cachedFoo;
|
| - */
|
| -class Creates {
|
| - final String types;
|
| - const Creates(this.types);
|
| -}
|
| -
|
| -/**
|
| - * A metadata annotation describing the types returned or yielded by a native
|
| - * element.
|
| - *
|
| - * The annotation is valid on a native method and a field of a native class.
|
| - *
|
| - * By default, a native method or field is seen as returning or yielding all
|
| - * subtypes if the method return type or field type. This annotation allows a
|
| - * more precise set of types to be specified.
|
| - *
|
| - * See [Creates] for the syntax of the argument.
|
| - *
|
| - * Example: IndexedDB keys are numbers, strings and JavaScript Arrays of keys.
|
| - *
|
| - * @Returns('String|num|JSExtendableArray')
|
| - * dynamic key;
|
| - *
|
| - * // Equivalent:
|
| - * @Returns('String') @Returns('num') @Returns('JSExtendableArray')
|
| - * dynamic key;
|
| - */
|
| -class Returns {
|
| - final String types;
|
| - const Returns(this.types);
|
| -}
|
| -
|
| -/**
|
| - * A metadata annotation placed on native methods and fields of native classes
|
| - * to specify the JavaScript name.
|
| - *
|
| - * This example declares a Dart field + getter + setter called `$dom_title` that
|
| - * corresponds to the JavaScript property `title`.
|
| - *
|
| - * class Docmument native "*Foo" {
|
| - * @JSName('title')
|
| - * String $dom_title;
|
| - * }
|
| - */
|
| -class JSName {
|
| - final String name;
|
| - const JSName(this.name);
|
| -}
|
| -
|
| -/**
|
| - * The following methods are called by the runtime to implement
|
| - * checked mode and casts. We specialize each primitive type (eg int, bool), and
|
| - * use the compiler's convention to do is-checks on regular objects.
|
| - */
|
| -boolConversionCheck(value) {
|
| - if (value is bool) return value;
|
| - // One of the following checks will always fail.
|
| - boolTypeCheck(value);
|
| - assert(value != null);
|
| - return false;
|
| -}
|
| -
|
| -stringTypeCheck(value) {
|
| - if (value == null) return value;
|
| - if (value is String) return value;
|
| - throw new TypeErrorImplementation(value, 'String');
|
| -}
|
| -
|
| -stringTypeCast(value) {
|
| - if (value is String || value == null) return value;
|
| - // TODO(lrn): When reified types are available, pass value.class and String.
|
| - throw new CastErrorImplementation(
|
| - Primitives.objectTypeName(value), 'String');
|
| -}
|
| -
|
| -doubleTypeCheck(value) {
|
| - if (value == null) return value;
|
| - if (value is double) return value;
|
| - throw new TypeErrorImplementation(value, 'double');
|
| -}
|
| -
|
| -doubleTypeCast(value) {
|
| - if (value is double || value == null) return value;
|
| - throw new CastErrorImplementation(
|
| - Primitives.objectTypeName(value), 'double');
|
| -}
|
| -
|
| -numTypeCheck(value) {
|
| - if (value == null) return value;
|
| - if (value is num) return value;
|
| - throw new TypeErrorImplementation(value, 'num');
|
| -}
|
| -
|
| -numTypeCast(value) {
|
| - if (value is num || value == null) return value;
|
| - throw new CastErrorImplementation(
|
| - Primitives.objectTypeName(value), 'num');
|
| -}
|
| -
|
| -boolTypeCheck(value) {
|
| - if (value == null) return value;
|
| - if (value is bool) return value;
|
| - throw new TypeErrorImplementation(value, 'bool');
|
| -}
|
| -
|
| -boolTypeCast(value) {
|
| - if (value is bool || value == null) return value;
|
| - throw new CastErrorImplementation(
|
| - Primitives.objectTypeName(value), 'bool');
|
| -}
|
| -
|
| -intTypeCheck(value) {
|
| - if (value == null) return value;
|
| - if (value is int) return value;
|
| - throw new TypeErrorImplementation(value, 'int');
|
| -}
|
| -
|
| -intTypeCast(value) {
|
| - if (value is int || value == null) return value;
|
| - throw new CastErrorImplementation(
|
| - Primitives.objectTypeName(value), 'int');
|
| -}
|
| -
|
| -void propertyTypeError(value, property) {
|
| - String name = isCheckPropertyToJsConstructorName(property);
|
| - throw new TypeErrorImplementation(value, name);
|
| -}
|
| -
|
| -void propertyTypeCastError(value, property) {
|
| - // Cuts the property name to the class name.
|
| - String actualType = Primitives.objectTypeName(value);
|
| - String expectedType = property.substring(3, property.length);
|
| - throw new CastErrorImplementation(actualType, expectedType);
|
| -}
|
| -
|
| -/**
|
| - * For types that are not supertypes of native (eg DOM) types,
|
| - * we emit a simple property check to check that an object implements
|
| - * that type.
|
| - */
|
| -propertyTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (JS('bool', '!!#[#]', value, property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -/**
|
| - * For types that are not supertypes of native (eg DOM) types,
|
| - * we emit a simple property check to check that an object implements
|
| - * that type.
|
| - */
|
| -propertyTypeCast(value, property) {
|
| - if (value == null || JS('bool', '!!#[#]', value, property)) return value;
|
| - propertyTypeCastError(value, property);
|
| -}
|
| -
|
| -/**
|
| - * For types that are supertypes of native (eg DOM) types, we use the
|
| - * interceptor for the class because we cannot add a JS property to the
|
| - * prototype at load time.
|
| - */
|
| -interceptedTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if ((identical(JS('String', 'typeof #', value), 'object'))
|
| - && JS('bool', '#[#]', getInterceptor(value), property)) {
|
| - return value;
|
| - }
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -/**
|
| - * For types that are supertypes of native (eg DOM) types, we use the
|
| - * interceptor for the class because we cannot add a JS property to the
|
| - * prototype at load time.
|
| - */
|
| -interceptedTypeCast(value, property) {
|
| - if (value == null
|
| - || ((JS('bool', 'typeof # === "object"', value))
|
| - && JS('bool', '#[#]', getInterceptor(value), property))) {
|
| - return value;
|
| - }
|
| - propertyTypeCastError(value, property);
|
| -}
|
| -
|
| -/**
|
| - * Specialization of the type check for num and String and their
|
| - * supertype since [value] can be a JS primitive.
|
| - */
|
| -numberOrStringSuperTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (value is String) return value;
|
| - if (value is num) return value;
|
| - if (JS('bool', '!!#[#]', value, property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -numberOrStringSuperTypeCast(value, property) {
|
| - if (value is String) return value;
|
| - if (value is num) return value;
|
| - return propertyTypeCast(value, property);
|
| -}
|
| -
|
| -numberOrStringSuperNativeTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (value is String) return value;
|
| - if (value is num) return value;
|
| - if (JS('bool', '#[#]', getInterceptor(value), property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -numberOrStringSuperNativeTypeCast(value, property) {
|
| - if (value == null) return value;
|
| - if (value is String) return value;
|
| - if (value is num) return value;
|
| - if (JS('bool', '#[#]', getInterceptor(value), property)) return value;
|
| - propertyTypeCastError(value, property);
|
| -}
|
| -
|
| -/**
|
| - * Specialization of the type check for String and its supertype
|
| - * since [value] can be a JS primitive.
|
| - */
|
| -stringSuperTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (value is String) return value;
|
| - if (JS('bool', '!!#[#]', value, property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -stringSuperTypeCast(value, property) {
|
| - if (value is String) return value;
|
| - return propertyTypeCast(value, property);
|
| -}
|
| -
|
| -stringSuperNativeTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (value is String) return value;
|
| - if (JS('bool', '#[#]', getInterceptor(value), property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -stringSuperNativeTypeCast(value, property) {
|
| - if (value is String || value == null) return value;
|
| - if (JS('bool', '#[#]', getInterceptor(value), property)) return value;
|
| - propertyTypeCastError(value, property);
|
| -}
|
| -
|
| -/**
|
| - * Specialization of the type check for List and its supertypes,
|
| - * since [value] can be a JS array.
|
| - */
|
| -listTypeCheck(value) {
|
| - if (value == null) return value;
|
| - if (value is List) return value;
|
| - throw new TypeErrorImplementation(value, 'List');
|
| -}
|
| -
|
| -listTypeCast(value) {
|
| - if (value is List || value == null) return value;
|
| - throw new CastErrorImplementation(
|
| - Primitives.objectTypeName(value), 'List');
|
| -}
|
| -
|
| -listSuperTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (value is List) return value;
|
| - if (JS('bool', '!!#[#]', value, property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -listSuperTypeCast(value, property) {
|
| - if (value is List) return value;
|
| - return propertyTypeCast(value, property);
|
| -}
|
| -
|
| -listSuperNativeTypeCheck(value, property) {
|
| - if (value == null) return value;
|
| - if (value is List) return value;
|
| - if (JS('bool', '#[#]', getInterceptor(value), property)) return value;
|
| - propertyTypeError(value, property);
|
| -}
|
| -
|
| -listSuperNativeTypeCast(value, property) {
|
| - if (value is List || value == null) return value;
|
| - if (JS('bool', '#[#]', getInterceptor(value), property)) return value;
|
| - propertyTypeCastError(value, property);
|
| -}
|
| -
|
| -voidTypeCheck(value) {
|
| - if (value == null) return value;
|
| - throw new TypeErrorImplementation(value, 'void');
|
| -}
|
| -
|
| -checkMalformedType(value, message) {
|
| - if (value == null) return value;
|
| - throw new TypeErrorImplementation.fromMessage(message);
|
| -}
|
| -
|
| -@NoInline()
|
| -void checkDeferredIsLoaded(String loadId, String uri) {
|
| - if (!_loadedLibraries.contains(loadId)) {
|
| - throw new DeferredNotLoadedError(uri);
|
| - }
|
| -}
|
| -
|
| -/**
|
| - * Special interface recognized by the compiler and implemented by DOM
|
| - * objects that support integer indexing. This interface is not
|
| - * visible to anyone, and is only injected into special libraries.
|
| - */
|
| -abstract class JavaScriptIndexingBehavior extends JSMutableIndexable {
|
| -}
|
| -
|
| -// TODO(lrn): These exceptions should be implemented in core.
|
| -// When they are, remove the 'Implementation' here.
|
| -
|
| -/** Thrown by type assertions that fail. */
|
| -class TypeErrorImplementation extends Error implements TypeError {
|
| - final String message;
|
| -
|
| - /**
|
| - * Normal type error caused by a failed subtype test.
|
| - */
|
| - TypeErrorImplementation(Object value, String type)
|
| - : message = "type '${Primitives.objectTypeName(value)}' is not a subtype "
|
| - "of type '$type'";
|
| -
|
| - TypeErrorImplementation.fromMessage(String this.message);
|
| -
|
| - String toString() => message;
|
| -}
|
| -
|
| -/** Thrown by the 'as' operator if the cast isn't valid. */
|
| -class CastErrorImplementation extends Error implements CastError {
|
| - // TODO(lrn): Rename to CastError (and move implementation into core).
|
| - final String message;
|
| -
|
| - /**
|
| - * Normal cast error caused by a failed type cast.
|
| - */
|
| - CastErrorImplementation(Object actualType, Object expectedType)
|
| - : message = "CastError: Casting value of type $actualType to"
|
| - " incompatible type $expectedType";
|
| -
|
| - String toString() => message;
|
| -}
|
| -
|
| -class FallThroughErrorImplementation extends FallThroughError {
|
| - FallThroughErrorImplementation();
|
| - String toString() => "Switch case fall-through.";
|
| -}
|
| -
|
| -/**
|
| - * Helper function for implementing asserts. The compiler treats this specially.
|
| - */
|
| -void assertHelper(condition) {
|
| - // Do a bool check first because it is common and faster than 'is Function'.
|
| - if (condition is !bool) {
|
| - if (condition is Function) condition = condition();
|
| - if (condition is !bool) {
|
| - throw new TypeErrorImplementation(condition, 'bool');
|
| - }
|
| - }
|
| - // Compare to true to avoid boolean conversion check in checked
|
| - // mode.
|
| - if (true != condition) throw new AssertionError();
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code when a method that must be statically
|
| - * resolved cannot be found.
|
| - */
|
| -void throwNoSuchMethod(obj, name, arguments, expectedArgumentNames) {
|
| - Symbol memberName = new _symbol_dev.Symbol.unvalidated(name);
|
| - throw new NoSuchMethodError(obj, memberName, arguments,
|
| - new Map<Symbol, dynamic>(),
|
| - expectedArgumentNames);
|
| -}
|
| -
|
| -/**
|
| - * Called by generated code when a static field's initializer references the
|
| - * field that is currently being initialized.
|
| - */
|
| -void throwCyclicInit(String staticName) {
|
| - throw new CyclicInitializationError(
|
| - "Cyclic initialization for static $staticName");
|
| -}
|
| -
|
| -/**
|
| - * Error thrown when a runtime error occurs.
|
| - */
|
| -class RuntimeError extends Error {
|
| - final message;
|
| - RuntimeError(this.message);
|
| - String toString() => "RuntimeError: $message";
|
| -}
|
| -
|
| -class DeferredNotLoadedError extends Error implements NoSuchMethodError {
|
| - String libraryName;
|
| -
|
| - DeferredNotLoadedError(this.libraryName);
|
| -
|
| - String toString() {
|
| - return "Deferred library $libraryName was not loaded.";
|
| - }
|
| -}
|
| -
|
| -abstract class RuntimeType {
|
| - const RuntimeType();
|
| -
|
| - toRti();
|
| -}
|
| -
|
| -class RuntimeFunctionType extends RuntimeType {
|
| - final RuntimeType returnType;
|
| - final List<RuntimeType> parameterTypes;
|
| - final List<RuntimeType> optionalParameterTypes;
|
| - final namedParameters;
|
| -
|
| - static var /* bool */ inAssert = false;
|
| -
|
| - RuntimeFunctionType(this.returnType,
|
| - this.parameterTypes,
|
| - this.optionalParameterTypes,
|
| - this.namedParameters);
|
| -
|
| - bool get isVoid => returnType is VoidRuntimeType;
|
| -
|
| - /// Called from generated code. [expression] is a Dart object and this method
|
| - /// returns true if [this] is a supertype of [expression].
|
| - @NoInline() @NoSideEffects()
|
| - bool _isTest(expression) {
|
| - var functionTypeObject = _extractFunctionTypeObjectFrom(expression);
|
| - return functionTypeObject == null
|
| - ? false
|
| - : isFunctionSubtype(functionTypeObject, toRti());
|
| - }
|
| -
|
| - @NoInline() @NoSideEffects()
|
| - _asCheck(expression) {
|
| - // Type inferrer doesn't think this is called with dynamic arguments.
|
| - return _check(JS('', '#', expression), true);
|
| - }
|
| -
|
| - @NoInline() @NoSideEffects()
|
| - _assertCheck(expression) {
|
| - if (inAssert) return null;
|
| - inAssert = true; // Don't try to check this library itself.
|
| - try {
|
| - // Type inferrer don't think this is called with dynamic arguments.
|
| - return _check(JS('', '#', expression), false);
|
| - } finally {
|
| - inAssert = false;
|
| - }
|
| - }
|
| -
|
| - _check(expression, bool isCast) {
|
| - if (expression == null) return null;
|
| - if (_isTest(expression)) return expression;
|
| -
|
| - var self = new FunctionTypeInfoDecoderRing(toRti()).toString();
|
| - if (isCast) {
|
| - var functionTypeObject = _extractFunctionTypeObjectFrom(expression);
|
| - var pretty;
|
| - if (functionTypeObject != null) {
|
| - pretty = new FunctionTypeInfoDecoderRing(functionTypeObject).toString();
|
| - } else {
|
| - pretty = Primitives.objectTypeName(expression);
|
| - }
|
| - throw new CastErrorImplementation(pretty, self);
|
| - } else {
|
| - // TODO(ahe): Pass "pretty" function-type to TypeErrorImplementation?
|
| - throw new TypeErrorImplementation(expression, self);
|
| - }
|
| - }
|
| -
|
| - _extractFunctionTypeObjectFrom(o) {
|
| - var interceptor = getInterceptor(o);
|
| - var signatureName = JS_GET_NAME(JsGetName.SIGNATURE_NAME);
|
| - return JS('bool', '# in #', signatureName, interceptor)
|
| - ? JS('', '#[#]()', interceptor, JS_GET_NAME(JsGetName.SIGNATURE_NAME))
|
| - : null;
|
| - }
|
| -
|
| - toRti() {
|
| - var result = createDartFunctionTypeRti();
|
| - if (isVoid) {
|
| - JS('', '#[#] = true', result,
|
| - JS_GET_NAME(JsGetName.FUNCTION_TYPE_VOID_RETURN_TAG));
|
| - } else {
|
| - if (returnType is! DynamicRuntimeType) {
|
| - JS('', '#[#] = #', result,
|
| - JS_GET_NAME(JsGetName.FUNCTION_TYPE_RETURN_TYPE_TAG),
|
| - returnType.toRti());
|
| - }
|
| - }
|
| - if (parameterTypes != null && !parameterTypes.isEmpty) {
|
| - JS('', '#[#] = #', result,
|
| - JS_GET_NAME(JsGetName.FUNCTION_TYPE_REQUIRED_PARAMETERS_TAG),
|
| - listToRti(parameterTypes));
|
| - }
|
| -
|
| - if (optionalParameterTypes != null && !optionalParameterTypes.isEmpty) {
|
| - JS('', '#[#] = #', result,
|
| - JS_GET_NAME(JsGetName.FUNCTION_TYPE_OPTIONAL_PARAMETERS_TAG),
|
| - listToRti(optionalParameterTypes));
|
| - }
|
| -
|
| - if (namedParameters != null) {
|
| - var namedRti = JS('=Object', 'Object.create(null)');
|
| - var keys = extractKeys(namedParameters);
|
| - for (var i = 0; i < keys.length; i++) {
|
| - var name = keys[i];
|
| - var rti = JS('', '#[#]', namedParameters, name).toRti();
|
| - JS('', '#[#] = #', namedRti, name, rti);
|
| - }
|
| - JS('', '#[#] = #', result,
|
| - JS_GET_NAME(JsGetName.FUNCTION_TYPE_NAMED_PARAMETERS_TAG),
|
| - namedRti);
|
| - }
|
| -
|
| - return result;
|
| - }
|
| -
|
| - static listToRti(list) {
|
| - list = JS('JSFixedArray', '#', list);
|
| - var result = JS('JSExtendableArray', '[]');
|
| - for (var i = 0; i < list.length; i++) {
|
| - JS('', '#.push(#)', result, list[i].toRti());
|
| - }
|
| - return result;
|
| - }
|
| -
|
| - String toString() {
|
| - String result = '(';
|
| - bool needsComma = false;
|
| - if (parameterTypes != null) {
|
| - for (var i = 0; i < parameterTypes.length; i++) {
|
| - RuntimeType type = parameterTypes[i];
|
| - if (needsComma) result += ', ';
|
| - result += '$type';
|
| - needsComma = true;
|
| - }
|
| - }
|
| - if (optionalParameterTypes != null && !optionalParameterTypes.isEmpty) {
|
| - if (needsComma) result += ', ';
|
| - needsComma = false;
|
| - result += '[';
|
| - for (var i = 0; i < optionalParameterTypes.length; i++) {
|
| - RuntimeType type = optionalParameterTypes[i];
|
| - if (needsComma) result += ', ';
|
| - result += '$type';
|
| - needsComma = true;
|
| - }
|
| - result += ']';
|
| - } else if (namedParameters != null) {
|
| - if (needsComma) result += ', ';
|
| - needsComma = false;
|
| - result += '{';
|
| - var keys = extractKeys(namedParameters);
|
| - for (var i = 0; i < keys.length; i++) {
|
| - var name = keys[i];
|
| - if (needsComma) result += ', ';
|
| - var rti = JS('', '#[#]', namedParameters, name).toRti();
|
| - result += '$rti ${JS("String", "#", name)}';
|
| - needsComma = true;
|
| - }
|
| - result += '}';
|
| - }
|
| -
|
| - result += ') -> $returnType';
|
| - return result;
|
| - }
|
| -}
|
| -
|
| -RuntimeFunctionType buildFunctionType(returnType,
|
| - parameterTypes,
|
| - optionalParameterTypes) {
|
| - return new RuntimeFunctionType(
|
| - returnType,
|
| - parameterTypes,
|
| - optionalParameterTypes,
|
| - null);
|
| -}
|
| -
|
| -RuntimeFunctionType buildNamedFunctionType(returnType,
|
| - parameterTypes,
|
| - namedParameters) {
|
| - return new RuntimeFunctionType(
|
| - returnType,
|
| - parameterTypes,
|
| - null,
|
| - namedParameters);
|
| -}
|
| -
|
| -RuntimeType buildInterfaceType(rti, typeArguments) {
|
| - String jsConstructorName = rawRtiToJsConstructorName(rti);
|
| - if (typeArguments == null || typeArguments.isEmpty) {
|
| - return new RuntimeTypePlain(jsConstructorName);
|
| - }
|
| - return new RuntimeTypeGeneric(jsConstructorName, typeArguments, null);
|
| -}
|
| -
|
| -class DynamicRuntimeType extends RuntimeType {
|
| - const DynamicRuntimeType();
|
| -
|
| - String toString() => 'dynamic';
|
| -
|
| - toRti() => null;
|
| -}
|
| -
|
| -RuntimeType getDynamicRuntimeType() => const DynamicRuntimeType();
|
| -
|
| -class VoidRuntimeType extends RuntimeType {
|
| - const VoidRuntimeType();
|
| -
|
| - String toString() => 'void';
|
| -
|
| - toRti() => throw 'internal error';
|
| -}
|
| -
|
| -RuntimeType getVoidRuntimeType() => const VoidRuntimeType();
|
| -
|
| -/**
|
| - * Meta helper for function type tests.
|
| - *
|
| - * A "meta helper" is a helper function that is never called but simulates how
|
| - * generated code behaves as far as resolution and type inference is concerned.
|
| - */
|
| -functionTypeTestMetaHelper() {
|
| - var dyn = JS('', 'x');
|
| - var dyn2 = JS('', 'x');
|
| - List fixedListOrNull = JS('JSFixedArray|Null', 'x');
|
| - List fixedListOrNull2 = JS('JSFixedArray|Null', 'x');
|
| - List fixedList = JS('JSFixedArray', 'x');
|
| - // TODO(ahe): Can we use [UnknownJavaScriptObject] below?
|
| - var /* UnknownJavaScriptObject */ jsObject = JS('=Object', 'x');
|
| -
|
| - buildFunctionType(dyn, fixedListOrNull, fixedListOrNull2);
|
| - buildNamedFunctionType(dyn, fixedList, jsObject);
|
| - buildInterfaceType(dyn, fixedListOrNull);
|
| - getDynamicRuntimeType();
|
| - getVoidRuntimeType();
|
| - convertRtiToRuntimeType(dyn);
|
| - dyn._isTest(dyn2);
|
| - dyn._asCheck(dyn2);
|
| - dyn._assertCheck(dyn2);
|
| -}
|
| -
|
| -RuntimeType convertRtiToRuntimeType(rti) {
|
| - if (rti == null) {
|
| - return getDynamicRuntimeType();
|
| - } else if (JS('bool', 'typeof # == "function"', rti)) {
|
| - return new RuntimeTypePlain(JS('String', r'#.name', rti));
|
| - } else if (JS('bool', '#.constructor == Array', rti)) {
|
| - List list = JS('JSFixedArray', '#', rti);
|
| - String name = JS('String', r'#.name', list[0]);
|
| - List arguments = [];
|
| - for (int i = 1; i < list.length; i++) {
|
| - arguments.add(convertRtiToRuntimeType(list[i]));
|
| - }
|
| - return new RuntimeTypeGeneric(name, arguments, rti);
|
| - } else if (JS('bool', '"func" in #', rti)) {
|
| - return new FunctionTypeInfoDecoderRing(rti).toRuntimeType();
|
| - } else {
|
| - throw new RuntimeError(
|
| - "Cannot convert "
|
| - "'${JS('String', 'JSON.stringify(#)', rti)}' to RuntimeType.");
|
| - }
|
| -}
|
| -
|
| -class RuntimeTypePlain extends RuntimeType {
|
| - /// The constructor name of this raw type.
|
| - final String _jsConstructorName;
|
| -
|
| - RuntimeTypePlain(this._jsConstructorName);
|
| -
|
| - toRti() {
|
| - var rti = jsConstructorNameToRti(_jsConstructorName);
|
| - if (rti == null) throw "no type for '$_jsConstructorName'";
|
| - return rti;
|
| - }
|
| -
|
| - String toString() => _jsConstructorName;
|
| -}
|
| -
|
| -class RuntimeTypeGeneric extends RuntimeType {
|
| - /// The constructor name of the raw type for this generic type.
|
| - final String _jsConstructorName;
|
| - final List<RuntimeType> arguments;
|
| - var rti;
|
| -
|
| - RuntimeTypeGeneric(this._jsConstructorName, this.arguments, this.rti);
|
| -
|
| - toRti() {
|
| - if (rti != null) return rti;
|
| - var result = [jsConstructorNameToRti(_jsConstructorName)];
|
| - if (result[0] == null) {
|
| - throw "no type for '$_jsConstructorName<...>'";
|
| - }
|
| - for (RuntimeType argument in arguments) {
|
| - result.add(argument.toRti());
|
| - }
|
| - return rti = result;
|
| - }
|
| -
|
| - String toString() => '$_jsConstructorName<${arguments.join(", ")}>';
|
| -}
|
| -
|
| -class FunctionTypeInfoDecoderRing {
|
| - final _typeData;
|
| - String _cachedToString;
|
| -
|
| - FunctionTypeInfoDecoderRing(this._typeData);
|
| -
|
| - bool get _hasReturnType => JS('bool', '"ret" in #', _typeData);
|
| - get _returnType => JS('', '#.ret', _typeData);
|
| -
|
| - bool get _isVoid => JS('bool', '!!#.void', _typeData);
|
| -
|
| - bool get _hasArguments => JS('bool', '"args" in #', _typeData);
|
| - List get _arguments => JS('JSExtendableArray', '#.args', _typeData);
|
| -
|
| - bool get _hasOptionalArguments => JS('bool', '"opt" in #', _typeData);
|
| - List get _optionalArguments => JS('JSExtendableArray', '#.opt', _typeData);
|
| -
|
| - bool get _hasNamedArguments => JS('bool', '"named" in #', _typeData);
|
| - get _namedArguments => JS('=Object', '#.named', _typeData);
|
| -
|
| - RuntimeType toRuntimeType() {
|
| - // TODO(ahe): Implement this (and update return type).
|
| - return const DynamicRuntimeType();
|
| - }
|
| -
|
| - String _convert(type) {
|
| - String result = runtimeTypeToString(type);
|
| - if (result != null) return result;
|
| - // Currently the [runtimeTypeToString] method doesn't handle function rtis.
|
| - if (JS('bool', '"func" in #', type)) {
|
| - return new FunctionTypeInfoDecoderRing(type).toString();
|
| - } else {
|
| - throw 'bad type';
|
| - }
|
| - }
|
| -
|
| - String toString() {
|
| - if (_cachedToString != null) return _cachedToString;
|
| - var s = "(";
|
| - var sep = '';
|
| - if (_hasArguments) {
|
| - for (var argument in _arguments) {
|
| - s += sep;
|
| - s += _convert(argument);
|
| - sep = ', ';
|
| - }
|
| - }
|
| - if (_hasOptionalArguments) {
|
| - s += '$sep[';
|
| - sep = '';
|
| - for (var argument in _optionalArguments) {
|
| - s += sep;
|
| - s += _convert(argument);
|
| - sep = ', ';
|
| - }
|
| - s += ']';
|
| - }
|
| - if (_hasNamedArguments) {
|
| - s += '$sep{';
|
| - sep = '';
|
| - for (var name in extractKeys(_namedArguments)) {
|
| - s += sep;
|
| - s += '$name: ';
|
| - s += _convert(JS('', '#[#]', _namedArguments, name));
|
| - sep = ', ';
|
| - }
|
| - s += '}';
|
| - }
|
| - s += ') -> ';
|
| - if (_isVoid) {
|
| - s += 'void';
|
| - } else if (_hasReturnType) {
|
| - s += _convert(_returnType);
|
| - } else {
|
| - s += 'dynamic';
|
| - }
|
| - return _cachedToString = "$s";
|
| - }
|
| -}
|
| -
|
| -// TODO(ahe): Remove this class and call noSuchMethod instead.
|
| -class UnimplementedNoSuchMethodError extends Error
|
| - implements NoSuchMethodError {
|
| - final String _message;
|
| -
|
| - UnimplementedNoSuchMethodError(this._message);
|
| -
|
| - String toString() => "Unsupported operation: $_message";
|
| -}
|
| -
|
| -/**
|
| - * Creates a random number with 64 bits of randomness.
|
| - *
|
| - * This will be truncated to the 53 bits available in a double.
|
| - */
|
| -int random64() {
|
| - // TODO(lrn): Use a secure random source.
|
| - int int32a = JS("int", "(Math.random() * 0x100000000) >>> 0");
|
| - int int32b = JS("int", "(Math.random() * 0x100000000) >>> 0");
|
| - return int32a + int32b * 0x100000000;
|
| -}
|
| -
|
| -String jsonEncodeNative(String string) {
|
| - return JS("String", "JSON.stringify(#)", string);
|
| -}
|
| -
|
| -/**
|
| - * Returns a property name for placing data on JavaScript objects shared between
|
| - * DOM isolates. This happens when multiple programs are loaded in the same
|
| - * JavaScript context (i.e. page). The name is based on [name] but with an
|
| - * additional part that is unique for each isolate.
|
| - *
|
| - * The form of the name is '___dart_$name_$id'.
|
| - */
|
| -String getIsolateAffinityTag(String name) {
|
| - var isolateTagGetter =
|
| - JS_EMBEDDED_GLOBAL('', GET_ISOLATE_TAG);
|
| - return JS('String', '#(#)', isolateTagGetter, name);
|
| -}
|
| -
|
| -typedef Future<Null> LoadLibraryFunctionType();
|
| -
|
| -LoadLibraryFunctionType _loadLibraryWrapper(String loadId) {
|
| - return () => loadDeferredLibrary(loadId);
|
| -}
|
| -
|
| -final Map<String, Future<Null>> _loadingLibraries = <String, Future<Null>>{};
|
| -final Set<String> _loadedLibraries = new Set<String>();
|
| -
|
| -typedef void DeferredLoadCallback();
|
| -
|
| -// Function that will be called every time a new deferred import is loaded.
|
| -DeferredLoadCallback deferredLoadHook;
|
| -
|
| -Future<Null> loadDeferredLibrary(String loadId) {
|
| - // For each loadId there is a list of hunk-uris to load, and a corresponding
|
| - // list of hashes. These are stored in the app-global scope.
|
| - var urisMap = JS_EMBEDDED_GLOBAL('', DEFERRED_LIBRARY_URIS);
|
| - List<String> uris = JS('JSExtendableArray|Null', '#[#]', urisMap, loadId);
|
| - var hashesMap = JS_EMBEDDED_GLOBAL('', DEFERRED_LIBRARY_HASHES);
|
| - List<String> hashes = JS('JSExtendableArray|Null', '#[#]', hashesMap, loadId);
|
| - if (uris == null) return new Future.value(null);
|
| - // The indices into `uris` and `hashes` that we want to load.
|
| - List<int> indices = new List.generate(uris.length, (i) => i);
|
| - var isHunkLoaded = JS_EMBEDDED_GLOBAL('', IS_HUNK_LOADED);
|
| - var isHunkInitialized = JS_EMBEDDED_GLOBAL('', IS_HUNK_INITIALIZED);
|
| - // Filter away indices for hunks that have already been loaded.
|
| - List<int> indicesToLoad = indices
|
| - .where((int i) => !JS('bool','#(#)', isHunkLoaded, hashes[i]))
|
| - .toList();
|
| - return Future.wait(indicesToLoad
|
| - .map((int i) => _loadHunk(uris[i]))).then((_) {
|
| - // Now all hunks have been loaded, we run the needed initializers.
|
| - List<int> indicesToInitialize = indices
|
| - .where((int i) => !JS('bool','#(#)', isHunkInitialized, hashes[i]))
|
| - .toList(); // Load the needed hunks.
|
| - for (int i in indicesToInitialize) {
|
| - var initializer = JS_EMBEDDED_GLOBAL('', INITIALIZE_LOADED_HUNK);
|
| - JS('void', '#(#)', initializer, hashes[i]);
|
| - }
|
| - bool updated = _loadedLibraries.add(loadId);
|
| - if (updated && deferredLoadHook != null) {
|
| - deferredLoadHook();
|
| - }
|
| - });
|
| -}
|
| -
|
| -Future<Null> _loadHunk(String hunkName) {
|
| - Future<Null> future = _loadingLibraries[hunkName];
|
| - if (future != null) {
|
| - return future.then((_) => null);
|
| - }
|
| -
|
| - String uri = IsolateNatives.thisScript;
|
| -
|
| - int index = uri.lastIndexOf('/');
|
| - uri = '${uri.substring(0, index + 1)}$hunkName';
|
| -
|
| - var deferredLibraryLoader = JS('', 'self.dartDeferredLibraryLoader');
|
| - Completer<Null> completer = new Completer<Null>();
|
| -
|
| - void success() {
|
| - completer.complete(null);
|
| - }
|
| -
|
| - void failure([error, StackTrace stackTrace]) {
|
| - _loadingLibraries[hunkName] = null;
|
| - completer.completeError(
|
| - new DeferredLoadException("Loading $uri failed: $error"),
|
| - stackTrace);
|
| - }
|
| -
|
| - var jsSuccess = convertDartClosureToJS(success, 0);
|
| - var jsFailure = convertDartClosureToJS((error) {
|
| - failure(unwrapException(error), getTraceFromException(error));
|
| - }, 1);
|
| -
|
| - if (JS('bool', 'typeof # === "function"', deferredLibraryLoader)) {
|
| - try {
|
| - JS('void', '#(#, #, #)', deferredLibraryLoader, uri,
|
| - jsSuccess, jsFailure);
|
| - } catch (error, stackTrace) {
|
| - failure(error, stackTrace);
|
| - }
|
| - } else if (isWorker()) {
|
| - // We are in a web worker. Load the code with an XMLHttpRequest.
|
| - enterJsAsync();
|
| - Future<Null> leavingFuture = completer.future.whenComplete(() {
|
| - leaveJsAsync();
|
| - });
|
| -
|
| - int index = uri.lastIndexOf('/');
|
| - uri = '${uri.substring(0, index + 1)}$hunkName';
|
| - var xhr = JS('var', 'new XMLHttpRequest()');
|
| - JS('void', '#.open("GET", #)', xhr, uri);
|
| - JS('void', '#.addEventListener("load", #, false)',
|
| - xhr, convertDartClosureToJS((event) {
|
| - if (JS('int', '#.status', xhr) != 200) {
|
| - failure("");
|
| - }
|
| - String code = JS('String', '#.responseText', xhr);
|
| - try {
|
| - // Create a new function to avoid getting access to current function
|
| - // context.
|
| - JS('void', '(new Function(#))()', code);
|
| - success();
|
| - } catch (error, stackTrace) {
|
| - failure(error, stackTrace);
|
| - }
|
| - }, 1));
|
| -
|
| - JS('void', '#.addEventListener("error", #, false)', xhr, failure);
|
| - JS('void', '#.addEventListener("abort", #, false)', xhr, failure);
|
| - JS('void', '#.send()', xhr);
|
| - } else {
|
| - // We are in a dom-context.
|
| - // Inject a script tag.
|
| - var script = JS('', 'document.createElement("script")');
|
| - JS('', '#.type = "text/javascript"', script);
|
| - JS('', '#.src = #', script, uri);
|
| - JS('', '#.addEventListener("load", #, false)', script, jsSuccess);
|
| - JS('', '#.addEventListener("error", #, false)', script, jsFailure);
|
| - JS('', 'document.body.appendChild(#)', script);
|
| - }
|
| - _loadingLibraries[hunkName] = completer.future;
|
| - return completer.future;
|
| -}
|
| -
|
| -class MainError extends Error implements NoSuchMethodError {
|
| - final String _message;
|
| -
|
| - MainError(this._message);
|
| -
|
| - String toString() => 'NoSuchMethodError: $_message';
|
| -}
|
| -
|
| -void missingMain() {
|
| - throw new MainError("No top-level function named 'main'.");
|
| -}
|
| -
|
| -void badMain() {
|
| - throw new MainError("'main' is not a function.");
|
| -}
|
| -
|
| -void mainHasTooManyParameters() {
|
| - throw new MainError("'main' expects too many parameters.");
|
| -}
|
| -
|
| -/// A wrapper around an exception, much like the one created by [wrapException]
|
| -/// but with a pre-given stack-trace.
|
| -class ExceptionAndStackTrace {
|
| - dynamic dartException;
|
| - StackTrace stackTrace;
|
| -
|
| - ExceptionAndStackTrace(this.dartException, this.stackTrace);
|
| -}
|
| -
|
| -/// Runtime support for async-await transformation.
|
| -///
|
| -/// This function is called by a transformed function on each await and return
|
| -/// in the untransformed function, and before starting.
|
| -///
|
| -/// If [object] is not a future it will be wrapped in a `new Future.value`.
|
| -///
|
| -/// If [asyncBody] is [async_error_codes.SUCCESS]/[async_error_codes.ERROR] it
|
| -/// indicates a return or throw from the async function, and
|
| -/// complete/completeError is called on [completer] with [object].
|
| -///
|
| -/// Otherwise [asyncBody] is set up to be called when the future is completed
|
| -/// with a code [async_error_codes.SUCCESS]/[async_error_codes.ERROR] depending
|
| -/// on the success of the future.
|
| -///
|
| -/// Returns the future of the completer for convenience of the first call.
|
| -dynamic asyncHelper(dynamic object,
|
| - dynamic /* js function */ bodyFunctionOrErrorCode,
|
| - Completer completer) {
|
| - if (identical(bodyFunctionOrErrorCode, async_error_codes.SUCCESS)) {
|
| - completer.complete(object);
|
| - return;
|
| - } else if (identical(bodyFunctionOrErrorCode, async_error_codes.ERROR)) {
|
| - // The error is a js-error.
|
| - completer.completeError(unwrapException(object),
|
| - getTraceFromException(object));
|
| - return;
|
| - }
|
| - Future future = object is Future ? object : new Future.value(object);
|
| - future.then(_wrapJsFunctionForAsync(bodyFunctionOrErrorCode,
|
| - async_error_codes.SUCCESS),
|
| - onError: (dynamic error, StackTrace stackTrace) {
|
| - ExceptionAndStackTrace wrappedException =
|
| - new ExceptionAndStackTrace(error, stackTrace);
|
| - Function wrapped =_wrapJsFunctionForAsync(bodyFunctionOrErrorCode,
|
| - async_error_codes.ERROR);
|
| - wrapped(wrappedException);
|
| - });
|
| - return completer.future;
|
| -}
|
| -
|
| -Function _wrapJsFunctionForAsync(dynamic /* js function */ function,
|
| - int errorCode) {
|
| - var protected = JS('', """
|
| - // Invokes [function] with [errorCode] and [result].
|
| - //
|
| - // If (and as long as) the invocation throws, calls [function] again,
|
| - // with an error-code.
|
| - function(errorCode, result) {
|
| - while (true) {
|
| - try {
|
| - #(errorCode, result);
|
| - break;
|
| - } catch (error) {
|
| - result = error;
|
| - errorCode = #;
|
| - }
|
| - }
|
| - }""", function, async_error_codes.ERROR);
|
| - return (result) {
|
| - JS('', '#(#, #)', protected, errorCode, result);
|
| - };
|
| -}
|
| -
|
| -/// Implements the runtime support for async* functions.
|
| -///
|
| -/// Called by the transformed function for each original return, await, yield,
|
| -/// yield* and before starting the function.
|
| -///
|
| -/// When the async* function wants to return it calls this function with
|
| -/// [asyncBody] == [async_error_codes.SUCCESS], the asyncStarHelper takes this
|
| -/// as signal to close the stream.
|
| -///
|
| -/// When the async* function wants to signal that an uncaught error was thrown,
|
| -/// it calls this function with [asyncBody] == [async_error_codes.ERROR],
|
| -/// the streamHelper takes this as signal to addError [object] to the
|
| -/// [controller] and close it.
|
| -///
|
| -/// If the async* function wants to do a yield or yield*, it calls this function
|
| -/// with [object] being an [IterationMarker].
|
| -///
|
| -/// In the case of a yield or yield*, if the stream subscription has been
|
| -/// canceled, schedules [asyncBody] to be called with
|
| -/// [async_error_codes.STREAM_WAS_CANCELED].
|
| -///
|
| -/// If [object] is a single-yield [IterationMarker], adds the value of the
|
| -/// [IterationMarker] to the stream. If the stream subscription has been
|
| -/// paused, return early. Otherwise schedule the helper function to be
|
| -/// executed again.
|
| -///
|
| -/// If [object] is a yield-star [IterationMarker], starts listening to the
|
| -/// yielded stream, and adds all events and errors to our own controller (taking
|
| -/// care if the subscription has been paused or canceled) - when the sub-stream
|
| -/// is done, schedules [asyncBody] again.
|
| -///
|
| -/// If the async* function wants to do an await it calls this function with
|
| -/// [object] not and [IterationMarker].
|
| -///
|
| -/// If [object] is not a [Future], it is wrapped in a `Future.value`.
|
| -/// The [asyncBody] is called on completion of the future (see [asyncHelper].
|
| -void asyncStarHelper(dynamic object,
|
| - dynamic /* int | js function */ bodyFunctionOrErrorCode,
|
| - AsyncStarStreamController controller) {
|
| - if (identical(bodyFunctionOrErrorCode, async_error_codes.SUCCESS)) {
|
| - // This happens on return from the async* function.
|
| - if (controller.isCanceled) {
|
| - controller.cancelationCompleter.complete();
|
| - } else {
|
| - controller.close();
|
| - }
|
| - return;
|
| - } else if (identical(bodyFunctionOrErrorCode, async_error_codes.ERROR)) {
|
| - // The error is a js-error.
|
| - if (controller.isCanceled) {
|
| - controller.cancelationCompleter.completeError(
|
| - unwrapException(object),
|
| - getTraceFromException(object));
|
| - } else {
|
| - controller.addError(unwrapException(object),
|
| - getTraceFromException(object));
|
| - controller.close();
|
| - }
|
| - return;
|
| - }
|
| -
|
| - if (object is IterationMarker) {
|
| - if (controller.isCanceled) {
|
| - Function wrapped = _wrapJsFunctionForAsync(bodyFunctionOrErrorCode,
|
| - async_error_codes.STREAM_WAS_CANCELED);
|
| - wrapped(null);
|
| - return;
|
| - }
|
| - if (object.state == IterationMarker.YIELD_SINGLE) {
|
| - controller.add(object.value);
|
| -
|
| - scheduleMicrotask(() {
|
| - if (controller.isPaused) {
|
| - // We only suspend the thread inside the microtask in order to allow
|
| - // listeners on the output stream to pause in response to the just
|
| - // output value, and have the stream immediately stop producing.
|
| - controller.isSuspended = true;
|
| - return;
|
| - }
|
| - Function wrapped = _wrapJsFunctionForAsync(bodyFunctionOrErrorCode,
|
| - async_error_codes.SUCCESS);
|
| - wrapped(null);
|
| - });
|
| - return;
|
| - } else if (object.state == IterationMarker.YIELD_STAR) {
|
| - Stream stream = object.value;
|
| - // Errors of [stream] are passed though to the main stream. (see
|
| - // [AsyncStreamController.addStream]).
|
| - // TODO(sigurdm): The spec is not very clear here. Clarify with Gilad.
|
| - controller.addStream(stream).then((_) {
|
| - // No check for isPaused here because the spec 17.16.2 only
|
| - // demands checks *before* each element in [stream] not after the last
|
| - // one. On the other hand we check for isCanceled, as that check happens
|
| - // after insertion of each element.
|
| - int errorCode = controller.isCanceled
|
| - ? async_error_codes.STREAM_WAS_CANCELED
|
| - : async_error_codes.SUCCESS;
|
| - Function wrapped = _wrapJsFunctionForAsync(bodyFunctionOrErrorCode,
|
| - errorCode);
|
| - wrapped(null);
|
| - });
|
| - return;
|
| - }
|
| - }
|
| -
|
| - Future future = object is Future ? object : new Future.value(object);
|
| - future.then(_wrapJsFunctionForAsync(bodyFunctionOrErrorCode,
|
| - async_error_codes.SUCCESS),
|
| - onError: (error, StackTrace stackTrace) {
|
| - ExceptionAndStackTrace wrappedException =
|
| - new ExceptionAndStackTrace(error, stackTrace);
|
| - Function wrapped = _wrapJsFunctionForAsync(
|
| - bodyFunctionOrErrorCode, async_error_codes.ERROR);
|
| - return wrapped(wrappedException);
|
| - });
|
| -}
|
| -
|
| -Stream streamOfController(AsyncStarStreamController controller) {
|
| - return controller.stream;
|
| -}
|
| -
|
| -/// A wrapper around a [StreamController] that keeps track of the state of
|
| -/// the execution of an async* function.
|
| -/// It can be in 1 of 3 states:
|
| -///
|
| -/// - running/scheduled
|
| -/// - suspended
|
| -/// - canceled
|
| -///
|
| -/// If yielding while the subscription is paused it will become suspended. And
|
| -/// only resume after the subscription is resumed or canceled.
|
| -class AsyncStarStreamController {
|
| - StreamController controller;
|
| - Stream get stream => controller.stream;
|
| -
|
| - /// True when the async* function has yielded while being paused.
|
| - /// When true execution will only resume after a `onResume` or `onCancel`
|
| - /// event.
|
| - bool isSuspended = false;
|
| -
|
| - bool get isPaused => controller.isPaused;
|
| -
|
| - Completer cancelationCompleter = null;
|
| -
|
| - /// True after the StreamSubscription has been cancelled.
|
| - /// When this is true, errors thrown from the async* body should go to the
|
| - /// [cancelationCompleter] instead of adding them to [controller], and
|
| - /// returning from the async function should complete [cancelationCompleter].
|
| - bool get isCanceled => cancelationCompleter != null;
|
| -
|
| - add(event) => controller.add(event);
|
| -
|
| - addStream(Stream stream) {
|
| - return controller.addStream(stream, cancelOnError: false);
|
| - }
|
| -
|
| - addError(error, stackTrace) => controller.addError(error, stackTrace);
|
| -
|
| - close() => controller.close();
|
| -
|
| - AsyncStarStreamController(body) {
|
| -
|
| - _resumeBody() {
|
| - scheduleMicrotask(() {
|
| - Function wrapped =
|
| - _wrapJsFunctionForAsync(body, async_error_codes.SUCCESS);
|
| - wrapped(null);
|
| - });
|
| - }
|
| -
|
| - controller = new StreamController(
|
| - onListen: () {
|
| - _resumeBody();
|
| - }, onResume: () {
|
| - // Only schedule again if the async* function actually is suspended.
|
| - // Resume directly instead of scheduling, so that the sequence
|
| - // `pause-resume-pause` will result in one extra event produced.
|
| - if (isSuspended) {
|
| - isSuspended = false;
|
| - _resumeBody();
|
| - }
|
| - }, onCancel: () {
|
| - // If the async* is finished we ignore cancel events.
|
| - if (!controller.isClosed) {
|
| - cancelationCompleter = new Completer();
|
| - if (isSuspended) {
|
| - // Resume the suspended async* function to run finalizers.
|
| - isSuspended = false;
|
| - scheduleMicrotask(() {
|
| - Function wrapped =_wrapJsFunctionForAsync(body,
|
| - async_error_codes.STREAM_WAS_CANCELED);
|
| - wrapped(null);
|
| - });
|
| - }
|
| - return cancelationCompleter.future;
|
| - }
|
| - });
|
| - }
|
| -}
|
| -
|
| -makeAsyncStarController(body) {
|
| - return new AsyncStarStreamController(body);
|
| -}
|
| -
|
| -class IterationMarker {
|
| - static const YIELD_SINGLE = 0;
|
| - static const YIELD_STAR = 1;
|
| - static const ITERATION_ENDED = 2;
|
| - static const UNCAUGHT_ERROR = 3;
|
| -
|
| - final value;
|
| - final int state;
|
| -
|
| - IterationMarker._(this.state, this.value);
|
| -
|
| - static yieldStar(dynamic /* Iterable or Stream */ values) {
|
| - return new IterationMarker._(YIELD_STAR, values);
|
| - }
|
| -
|
| - static endOfIteration() {
|
| - return new IterationMarker._(ITERATION_ENDED, null);
|
| - }
|
| -
|
| - static yieldSingle(dynamic value) {
|
| - return new IterationMarker._(YIELD_SINGLE, value);
|
| - }
|
| -
|
| - static uncaughtError(dynamic error) {
|
| - return new IterationMarker._(UNCAUGHT_ERROR, error);
|
| - }
|
| -
|
| - toString() => "IterationMarker($state, $value)";
|
| -}
|
| -
|
| -class SyncStarIterator implements Iterator {
|
| - final dynamic _body;
|
| -
|
| - // If [runningNested] this is the nested iterator, otherwise it is the
|
| - // current value.
|
| - dynamic _current = null;
|
| - bool _runningNested = false;
|
| -
|
| - get current => _runningNested ? _current.current : _current;
|
| -
|
| - SyncStarIterator(this._body);
|
| -
|
| - _runBody() {
|
| - return JS('', '''
|
| - // Invokes [body] with [errorCode] and [result].
|
| - //
|
| - // If (and as long as) the invocation throws, calls [function] again,
|
| - // with an error-code.
|
| - (function(body) {
|
| - var errorValue, errorCode = #;
|
| - while (true) {
|
| - try {
|
| - return body(errorCode, errorValue);
|
| - } catch (error) {
|
| - errorValue = error;
|
| - errorCode = #
|
| - }
|
| - }
|
| - })(#)''', async_error_codes.SUCCESS, async_error_codes.ERROR, _body);
|
| - }
|
| -
|
| -
|
| - bool moveNext() {
|
| - if (_runningNested) {
|
| - if (_current.moveNext()) {
|
| - return true;
|
| - } else {
|
| - _runningNested = false;
|
| - }
|
| - }
|
| - _current = _runBody();
|
| - if (_current is IterationMarker) {
|
| - if (_current.state == IterationMarker.ITERATION_ENDED) {
|
| - _current = null;
|
| - // Rely on [_body] to repeatedly return `ITERATION_ENDED`.
|
| - return false;
|
| - } else if (_current.state == IterationMarker.UNCAUGHT_ERROR) {
|
| - // Rely on [_body] to repeatedly return `UNCAUGHT_ERROR`.
|
| - // This is a wrapped exception, so we use JavaScript throw to throw it.
|
| - JS('', 'throw #', _current.value);
|
| - } else {
|
| - assert(_current.state == IterationMarker.YIELD_STAR);
|
| - _current = _current.value.iterator;
|
| - _runningNested = true;
|
| - return moveNext();
|
| - }
|
| - }
|
| - return true;
|
| - }
|
| -}
|
| -
|
| -/// An Iterable corresponding to a sync* method.
|
| -///
|
| -/// Each invocation of a sync* method will return a new instance of this class.
|
| -class SyncStarIterable extends IterableBase {
|
| - // This is a function that will return a helper function that does the
|
| - // iteration of the sync*.
|
| - //
|
| - // Each invocation should give a body with fresh state.
|
| - final dynamic /* js function */ _outerHelper;
|
| -
|
| - SyncStarIterable(this._outerHelper);
|
| -
|
| - Iterator get iterator => new SyncStarIterator(JS('', '#()', _outerHelper));
|
| -}
|
|
|