blob: 574cd52f9049434dc3891a8976082888c66ff43b [file] [log] [blame]
// Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
/// This library contains support for runtime type information.
library rti;
import 'dart:_foreign_helper' show JS;
import 'dart:_interceptors' show JSArray, JSUnmodifiableArray;
/// An Rti object represents both a type (e.g `Map<int, String>`) and a type
/// environment (`Map<int, String>` binds `Map.K=int` and `Map.V=String`).
///
/// There is a single [Rti] class to help reduce polymorphism in the JavaScript
/// runtime. The class has a default constructor and no final fields so it can
/// be created before much of the runtime exists.
///
/// The fields are declared in an order that gets shorter minified names for the
/// more commonly used fields. (TODO: we should exploit the fact that an Rti
/// instance never appears in a dynamic context, so does not need field names to
/// be distinct from dynamic selectors).
///
class Rti {
/// JavaScript method for 'as' check. The method is called from generated code,
/// e.g. `o as T` generates something like `rtiForT._as(o)`.
@pragma('dart2js:noElision')
dynamic _as;
/// JavaScript method for type check. The method is called from generated
/// code, e.g. parameter check for `T param` generates something like
/// `rtiForT._check(param)`.
@pragma('dart2js:noElision')
dynamic _check;
/// JavaScript method for 'is' test. The method is called from generated
/// code, e.g. `o is T` generates something like `rtiForT._is(o)`.
@pragma('dart2js:noElision')
dynamic _is;
static void _setAsCheckFunction(Rti rti, fn) {
rti._as = fn;
}
static void _setTypeCheckFunction(Rti rti, fn) {
rti._check = fn;
}
static void _setIsTestFunction(Rti rti, fn) {
rti._is = fn;
}
/// Method called from generated code to evaluate a type environment recipe in
/// `this` type environment.
Rti _eval(String recipe) => _rtiEval(this, recipe);
/// Method called from generated code to extend `this` type environment with a
/// function type parameter.
Rti _bind1(Rti type) => _rtiBind1(this, type);
/// Method called from generated code to extend `this` type environment with a
/// tuple of function type parameters.
Rti _bind(Rti typeTuple) => _rtiBind(this, typeTuple);
// Precomputed derived types. These fields are used to hold derived types that
// are computed eagerly.
// TODO(sra): Implement precomputed type optimizations.
dynamic _precomputed1;
dynamic _precomputed2;
dynamic _precomputed3;
dynamic _precomputed4;
// The Type object corresponding to this Rti.
Type _typeCache;
/// The kind of Rti `this` is, one of the kindXXX constants below.
///
/// We don't use an enum since we need to create Rti objects very early.
///
/// The zero initializer ensures dart2js type analysis considers [_kind] is
/// non-nullable.
int _kind = 0;
static int _getKind(Rti rti) => rti._kind;
static void _setKind(Rti rti, int kind) {
rti._kind = kind;
}
// Terminal terms.
static const kindNever = 1;
static const kindDynamic = 2;
static const kindVoid = 3; // TODO(sra): Use `dynamic` instead?
static const kindAny = 4; // Dart1-style 'dynamic' for JS-interop.
// Unary terms.
static const kindStar = 5;
static const kindQuestion = 6;
static const kindFutureOr = 7;
// More complex terms.
static const kindInterface = 8;
// A vector of type parameters from enclosing functions and closures.
static const kindBinding = 9;
static const kindFunction = 10;
static const kindGenericFunction = 11;
/// Primary data associated with type.
///
/// - Minified name of interface for interface types.
/// - Underlying type for unary terms.
/// - Class part of a type environment inside a generic class, or `null` for
/// type tuple.
/// - Return type of function types.
dynamic _primary;
static Object _getPrimary(Rti rti) => rti._primary;
static void _setPrimary(Rti rti, value) {
rti._primary = value;
}
/// Additional data associated with type.
///
/// - The type arguments of an interface type.
/// - The type arguments from enclosing functions and closures for a
/// kindBinding.
/// - TBD for kindFunction and kindGenericFunction.
dynamic _rest;
static Object _getRest(Rti rti) => rti._rest;
static void _setRest(Rti rti, value) {
rti._rest = value;
}
static String _getInterfaceName(Rti rti) {
assert(_getKind(rti) == kindInterface);
return _Utils.asString(_getPrimary(rti));
}
static JSArray _getInterfaceTypeArguments(rti) {
// The array is a plain JavaScript Array, otherwise we would need the type
// `JSArray<Rti>` to exist before we could create the type `JSArray<Rti>`.
assert(_getKind(rti) == kindInterface);
return JS('JSUnmodifiableArray', '#', _getRest(rti));
}
/// On [Rti]s that are type environments, derived types are cached on the
/// environment to ensure fast canonicalization. Ground-term types (i.e. not
/// dependent on class or function type parameters) are cached in the
/// universe. This field starts as `null` and the cache is created on demand.
Object _evalCache;
static Object _getEvalCache(Rti rti) => rti._evalCache;
static void _setEvalCache(Rti rti, value) {
rti._evalCache = value;
}
static Rti allocate() {
return new Rti();
}
String _canonicalRecipe;
static String _getCanonicalRecipe(Rti rti) {
var s = rti._canonicalRecipe;
assert(_Utils.isString(s), 'Missing canonical recipe');
return _Utils.asString(s);
}
static void _setCanonicalRecipe(Rti rti, String s) {
rti._canonicalRecipe = s;
}
}
Rti _rtiEval(Rti environment, String recipe) {
throw UnimplementedError('_rtiEval');
}
Rti _rtiBind1(Rti environment, Rti type) {
throw UnimplementedError('_rtiBind1');
}
Rti _rtiBind(Rti environment, Rti typeTuple) {
throw UnimplementedError('_rtiBind');
}
Type getRuntimeType(object) {
throw UnimplementedError('getRuntimeType');
}
String _rtiToString(Rti rti, List<String> genericContext) {
int kind = Rti._getKind(rti);
if (kind == Rti.kindDynamic) return 'dynamic';
if (kind == Rti.kindInterface) {
String name = Rti._getInterfaceName(rti);
var arguments = Rti._getInterfaceTypeArguments(rti);
if (arguments.length != 0) {
name += '<';
for (int i = 0; i < arguments.length; i++) {
if (i > 0) name += ', ';
name += _rtiToString(_castToRti(arguments[i]), genericContext);
}
name += '>';
}
return name;
}
return '?';
}
/// Class of static methods for the universe of Rti objects.
///
/// The universe is the manager object for the Rti instances.
///
/// The universe itself is allocated at startup before any types or Dart objects
/// can be created, so it does not have a Dart type.
class _Universe {
_Universe._() {
throw UnimplementedError('_Universe is static methods only');
}
@pragma('dart2js:noInline')
static Object create() {
// TODO(sra): For consistency, this expression should be a JS_BUILTIN that
// uses the same template as emitted by the emitter.
return JS(
'',
'{'
'evalCache: new Map(),'
'unprocessedRules:[],'
'a0:[],' // shared empty array.
'}');
}
// Field accessors.
static evalCache(universe) => JS('', '#.evalCache', universe);
static void addRules(universe, String rules) {
JS('', '#.unprocessedRules.push(#)', universe, rules);
}
static Object sharedEmptyArray(universe) => JS('JSArray', '#.a0', universe);
/// Evaluates [recipe] in the global environment.
static Rti eval(Object universe, String recipe) {
var cache = evalCache(universe);
var probe = _cacheGet(cache, recipe);
if (probe != null) return _castToRti(probe);
var rti = _parseRecipe(universe, null, recipe);
_cacheSet(cache, recipe, rti);
return rti;
}
static Rti evalInEnvironment(
Object universe, Rti environment, String recipe) {
var cache = Rti._getEvalCache(environment);
if (cache == null) {
cache = JS('', 'new Map()');
Rti._setEvalCache(environment, cache);
}
var probe = _cacheGet(cache, recipe);
if (probe != null) return _castToRti(probe);
var rti = _parseRecipe(universe, environment, recipe);
_cacheSet(cache, recipe, rti);
return rti;
}
static Rti evalTypeVariable(Object universe, Rti environment, String name) {
throw UnimplementedError('_Universe.evalTypeVariable("$name")');
}
static _cacheGet(cache, key) => JS('', '#.get(#)', cache, key);
static void _cacheSet(cache, key, value) {
JS('', '#.set(#, #)', cache, key, value);
}
static Rti _parseRecipe(Object universe, Object environment, String recipe) {
var parser = _Parser.create(universe, environment, recipe);
Rti rti = _Parser.parse(parser);
if (rti != null) return rti;
throw UnimplementedError('_Universe._parseRecipe("$recipe")');
}
static Rti _finishRti(Object universe, Rti rti) {
// Enter fresh Rti in global table under it's canonical recipe.
String key = Rti._getCanonicalRecipe(rti);
_cacheSet(evalCache(universe), key, rti);
// Set up methods to type tests.
// TODO(sra): These are for `dynamic`. Install general functions and
// specializations.
var alwaysPasses = JS('', 'function(o) { return o; }');
Rti._setAsCheckFunction(rti, alwaysPasses);
Rti._setTypeCheckFunction(rti, alwaysPasses);
Rti._setIsTestFunction(rti, JS('', 'function(o) { return true; }'));
return rti;
}
// For each kind of Rti there are three methods:
//
// * `lookupXXX` which takes the component parts and returns an existing Rti
// object if it exists.
// * `canonicalRecipeOfXXX` that returns the compositional canonical recipe
// for the proposed type.
// * `createXXX` to create the type if it does not exist.
static String _canonicalRecipeOfDynamic() => '@';
static Rti _lookupDynamicRti(universe) {
var cache = evalCache(universe);
var probe = _cacheGet(cache, _canonicalRecipeOfDynamic());
if (probe != null) return _castToRti(probe);
return _createDynamicRti(universe);
}
static Rti _createDynamicRti(Object universe) {
var rti = Rti.allocate();
Rti._setKind(rti, Rti.kindDynamic);
Rti._setCanonicalRecipe(rti, _canonicalRecipeOfDynamic());
return _finishRti(universe, rti);
}
static String _canonicalRecipeOfInterface(String name, Object arguments) {
assert(_Utils.isString(name));
String s = _Utils.asString(name);
int length = _Utils.arrayLength(arguments);
if (length != 0) {
s += '<';
for (int i = 0; i < length; i++) {
if (i > 0) s += ',';
Rti argument = _castToRti(_Utils.arrayAt(arguments, i));
String subrecipe = Rti._getCanonicalRecipe(argument);
s += subrecipe;
}
s += '>';
}
return s;
}
static Rti _lookupInterfaceRti(
Object universe, String name, Object arguments) {
String key = _canonicalRecipeOfInterface(name, arguments);
var cache = evalCache(universe);
var probe = _cacheGet(cache, key);
if (probe != null) return _castToRti(probe);
return _createInterfaceRti(universe, name, arguments, key);
}
static Rti _createInterfaceRti(
Object universe, String name, Object typeArguments, String key) {
var rti = Rti.allocate();
Rti._setKind(rti, Rti.kindInterface);
Rti._setPrimary(rti, name);
Rti._setRest(rti, typeArguments);
Rti._setCanonicalRecipe(rti, key);
return _finishRti(universe, rti);
}
}
/// Class of static methods implementing recipe parser.
///
/// The recipe is a sequence of operations on a stack machine. The operations
/// are described below using the format
///
/// operation: stack elements before --- stack elements after
///
/// integer: --- integer-value
///
/// identifier: --- string-value
///
/// identifier-with-one-period: --- type-variable-value
///
/// Period may be in any position, including first and last e.g. `.x`.
///
/// ',': ignored
///
/// Used to separate elements.
///
/// '@': --- dynamicType
///
/// '?': type --- type?
///
/// '<': --- position
///
/// Saves (pushes) position register, sets position register to end of stack.
///
/// '>': name saved-position type ... type --- name<type, ..., type>
///
/// Creates interface type from name types pushed since the position register
/// was last set. Restores position register to previous saved value.
///
class _Parser {
_Parser._() {
throw UnimplementedError('_Parser is static methods only');
}
/// Creates a parser object for parsing a recipe against an environment in a
/// universe.
///
/// Marked as no-inline so the object literal is not cloned by inlining.
@pragma('dart2js:noInline')
static Object create(Object universe, Object environment, String recipe) {
return JS(
'',
'{'
'u:#,' // universe
'e:#,' // environment
'r:#,' // recipe
's:[],' // stack
'p:0,' // position of sequence start.
'}',
universe,
environment,
recipe);
}
// Field accessors for the parser.
static Object universe(Object parser) => JS('String', '#.u', parser);
static Rti environment(Object parser) => JS('Rti', '#.e', parser);
static String recipe(Object parser) => JS('String', '#.r', parser);
static Object stack(Object parser) => JS('', '#.s', parser);
static Object position(Object parser) => JS('int', '#.p', parser);
static void setPosition(Object parser, int p) {
JS('', '#.p = #', parser, p);
}
static int charCodeAt(String s, int i) => JS('int', '#.charCodeAt(#)', s, i);
static void push(Object stack, Object value) {
JS('', '#.push(#)', stack, value);
}
static Object pop(Object stack) => JS('', '#.pop()', stack);
static Rti parse(Object parser) {
String source = _Parser.recipe(parser);
var stack = _Parser.stack(parser);
int i = 0;
while (i < source.length) {
int ch = charCodeAt(source, i);
if (isDigit(ch)) {
i = handleDigit(i + 1, ch, source, stack);
} else if (isIdentifierStart(ch)) {
i = handleIdentifer(parser, i, source, stack, false);
} else if (ch == $PERIOD) {
i = handleIdentifer(parser, i, source, stack, true);
} else {
i++;
switch (ch) {
case $COMMA:
// ignored
break;
case $AT:
push(stack, _Universe._lookupDynamicRti(universe(parser)));
break;
case $LT:
push(stack, position(parser));
setPosition(parser, _Utils.arrayLength(stack));
break;
case $GT:
handleGenericInterfaceType(parser, stack);
break;
default:
JS('', 'throw "Bad character " + #', ch);
}
}
}
Object item = pop(stack);
return toType(universe(parser), environment(parser), item);
}
static int handleDigit(int i, int digit, String source, Object stack) {
int value = digit - $0;
for (; i < source.length; i++) {
int ch = charCodeAt(source, i);
if (!isDigit(ch)) break;
value = value * 10 + ch - $0;
}
push(stack, value);
return i;
}
static int handleIdentifer(
Object parser, int start, String source, Object stack, bool hasPeriod) {
int i = start + 1;
for (; i < source.length; i++) {
int ch = charCodeAt(source, i);
if (ch == $PERIOD) {
if (hasPeriod) break;
hasPeriod = true;
} else if (isIdentifierStart(ch) || isDigit(ch)) {
// Accept.
} else {
break;
}
}
String string = _Utils.substring(source, start, i);
if (hasPeriod) {
push(
stack,
_Universe.evalTypeVariable(
universe(parser), environment(parser), string));
} else {
push(stack, string);
}
return i;
}
static void handleGenericInterfaceType(Object parser, Object stack) {
var universe = _Parser.universe(parser);
var arguments = _Utils.arraySplice(stack, position(parser));
toTypes(universe, environment(parser), arguments);
setPosition(parser, _Utils.asInt(pop(stack)));
String name = _Utils.asString(pop(stack));
push(stack, _Universe._lookupInterfaceRti(universe, name, arguments));
}
/// Coerce a stack item into an Rti object. Strings are converted to interface
/// types, integers are looked up in the type environment.
static Rti toType(Object universe, Rti environment, Object item) {
if (_Utils.isString(item)) {
String name = _Utils.asString(item);
// TODO(sra): Compile this out for minified code.
if ('dynamic' == name) {
return _Universe._lookupDynamicRti(universe);
}
return _Universe._lookupInterfaceRti(
universe, name, _Universe.sharedEmptyArray(universe));
} else if (_Utils.isNum(item)) {
return _Parser._indexToType(universe, environment, _Utils.asInt(item));
} else {
return _castToRti(item);
}
}
static void toTypes(Object universe, Rti environment, Object items) {
int length = _Utils.arrayLength(items);
for (int i = 0; i < length; i++) {
var item = _Utils.arrayAt(items, i);
var type = toType(universe, environment, item);
_Utils.arraySetAt(items, i, type);
}
}
static Rti _indexToType(Object universe, Rti environment, int index) {
while (true) {
int kind = Rti._getKind(environment);
if (kind == Rti.kindInterface) {
var typeArguments = Rti._getInterfaceTypeArguments(environment);
int len = _Utils.arrayLength(typeArguments);
if (index < len) {
return _castToRti(_Utils.arrayAt(typeArguments, index));
}
throw AssertionError('Bad index $index for $environment');
}
// TODO(sra): Binding environment.
throw AssertionError('Recipe cannot index Rti kind $kind');
}
}
static bool isDigit(int ch) => ch >= $0 && ch <= $9;
static bool isIdentifierStart(int ch) =>
(ch >= $A && ch <= $Z) ||
(ch >= $a && ch <= $z) ||
(ch == $_) ||
(ch == $$);
static const int $$ = 0x24;
static const int $COMMA = 0x2C;
static const int $PERIOD = 0x2E;
static const int $0 = 0x30;
static const int $9 = 0x39;
static const int $LT = 0x3C;
static const int $GT = 0x3E;
static const int $A = 0x41;
static const int $AT = 0x40;
static const int $Z = $A + 26 - 1;
static const int $a = $A + 32;
static const int $z = $Z + 32;
static const int $_ = 0x5F;
}
// -------- Subtype tests ------------------------------------------------------
// Future entry point from compiled code.
bool isSubtype(Rti s, Rti t) {
return _isSubtype(s, null, t, null);
}
bool _isSubtype(Rti s, var sEnv, Rti t, var tEnv) {
if (_Utils.isIdentical(s, t)) return true;
int tKind = Rti._getKind(t);
if (tKind == Rti.kindDynamic) return true;
if (tKind == Rti.kindNever) return false;
return false;
}
/// Unchecked cast to Rti.
Rti _castToRti(s) => JS('Rti', '#', s);
///
class _Utils {
static int asInt(Object o) => JS('int', '#', o);
static String asString(Object o) => JS('String', '#', o);
static bool isString(Object o) => JS('bool', 'typeof # == "string"', o);
static bool isNum(Object o) => JS('bool', 'typeof # == "number"', o);
static bool isIdentical(s, t) => JS('bool', '# === #', s, t);
static int arrayLength(Object array) => JS('int', '#.length', array);
static Object arrayAt(Object array, int i) => JS('', '#[#]', array, i);
static Object arraySetAt(Object array, int i, Object value) {
JS('', '#[#] = #', array, i, value);
}
static JSArray arraySplice(Object array, int position) =>
JS('JSArray', '#.splice(#)', array, position);
static String substring(String s, int start, int end) =>
JS('String', '#.substring(#, #)', s, start, end);
static mapGet(cache, key) => JS('', '#.get(#)', cache, key);
static void mapSet(cache, key, value) {
JS('', '#.set(#, #)', cache, key, value);
}
}
// -------- Entry points for testing -------------------------------------------
String testingRtiToString(rti) {
return _rtiToString(_castToRti(rti), null);
}
String testingRtiToDebugString(rti) {
// TODO(sra): Create entty point for structural formatting of Rti tree.
return 'Rti';
}
Object testingCreateUniverse() {
return _Universe.create();
}
Object testingAddRules(universe, String rules) {
_Universe.addRules(universe, rules);
}
bool testingIsSubtype(rti1, rti2) {
return isSubtype(_castToRti(rti1), _castToRti(rti2));
}
Object testingUniverseEval(universe, String recipe) {
return _Universe.eval(universe, recipe);
}
Object testingEnvironmentEval(universe, environment, String recipe) {
return _Universe.evalInEnvironment(universe, _castToRti(environment), recipe);
}