Start on parser
Change-Id: Ibcc1dcb04735bcefd43f4239dcac28329f519c95
Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/105470
Commit-Queue: Stephen Adams <sra@google.com>
Reviewed-by: Sigmund Cherem <sigmund@google.com>
Reviewed-by: Mayank Patke <fishythefish@google.com>
diff --git a/sdk/lib/_internal/js_runtime/lib/rti.dart b/sdk/lib/_internal/js_runtime/lib/rti.dart
index 5e57518..574cd52 100644
--- a/sdk/lib/_internal/js_runtime/lib/rti.dart
+++ b/sdk/lib/_internal/js_runtime/lib/rti.dart
@@ -23,15 +23,18 @@
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) {
@@ -107,7 +110,10 @@
/// - Return type of function types.
dynamic _primary;
- static _getPrimary(Rti rti) => rti._primary;
+ static Object _getPrimary(Rti rti) => rti._primary;
+ static void _setPrimary(Rti rti, value) {
+ rti._primary = value;
+ }
/// Additional data associated with type.
///
@@ -117,22 +123,49 @@
/// - 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', '#', _getPrimary(rti));
+ 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.
- dynamic _evalCache;
+ 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) {
@@ -154,16 +187,31 @@
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');
+class _Universe {
+ _Universe._() {
+ throw UnimplementedError('_Universe is static methods only');
}
@pragma('dart2js:noInline')
@@ -175,44 +223,361 @@
'{'
'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 eval(universe, String recipe) {
+ 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 probe;
- var rti = _parseRecipe(universe, 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 _parseRecipe(universe, recipe) {
- if (recipe == 'dynamic') return _createDynamicRti(universe);
- throw UnimplementedError('Universe._parseRecipe("$recipe")');
+ 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 _createDynamicRti(universe) {
- var rti = Rti.allocate();
- Rti._setKind(rti, Rti.kindDynamic);
+ 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 ------------------------------------------------------
@@ -223,7 +588,7 @@
}
bool _isSubtype(Rti s, var sEnv, Rti t, var tEnv) {
- if (_isIdentical(s, t)) return true;
+ if (_Utils.isIdentical(s, t)) return true;
int tKind = Rti._getKind(t);
if (tKind == Rti.kindDynamic) return true;
if (tKind == Rti.kindNever) return false;
@@ -233,8 +598,36 @@
/// Unchecked cast to Rti.
Rti _castToRti(s) => JS('Rti', '#', s);
-bool _isIdentical(s, t) => JS('bool', '# === #', s, t);
+///
+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) {
@@ -247,11 +640,11 @@
}
Object testingCreateUniverse() {
- return Universe.create();
+ return _Universe.create();
}
Object testingAddRules(universe, String rules) {
- Universe.addRules(universe, rules);
+ _Universe.addRules(universe, rules);
}
bool testingIsSubtype(rti1, rti2) {
@@ -259,5 +652,9 @@
}
Object testingUniverseEval(universe, String recipe) {
- return Universe.eval(universe, recipe);
+ return _Universe.eval(universe, recipe);
+}
+
+Object testingEnvironmentEval(universe, environment, String recipe) {
+ return _Universe.evalInEnvironment(universe, _castToRti(environment), recipe);
}
diff --git a/tests/compiler/dart2js_extra/rti/class_environment_test.dart b/tests/compiler/dart2js_extra/rti/class_environment_test.dart
new file mode 100644
index 0000000..0cad566
--- /dev/null
+++ b/tests/compiler/dart2js_extra/rti/class_environment_test.dart
@@ -0,0 +1,44 @@
+// Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+import 'dart:_rti' as rti;
+import "package:expect/expect.dart";
+
+void checkRtiIdentical(Object rti1, Object rti2) {
+ var format = rti.testingRtiToString;
+ Expect.isTrue(
+ identical(rti1, rti2), 'identical(${format(rti1)}, ${format(rti2)}');
+}
+
+testInterface1() {
+ var universe = rti.testingCreateUniverse();
+
+ var env = rti.testingUniverseEval(universe, 'Foo<int>');
+ var rti1 = rti.testingUniverseEval(universe, 'int');
+ var rti2 = rti.testingEnvironmentEval(universe, env, '0');
+
+ Expect.equals('int', rti.testingRtiToString(rti1));
+ Expect.equals('int', rti.testingRtiToString(rti2));
+ checkRtiIdentical(rti1, rti2);
+}
+
+testInterface2() {
+ var universe = rti.testingCreateUniverse();
+
+ var env = rti.testingUniverseEval(universe, 'Foo<int,List<int>>');
+ var rti1 = rti.testingUniverseEval(universe, 'List<int>');
+ var rti2 = rti.testingEnvironmentEval(universe, env, '1');
+ var rti3 = rti.testingEnvironmentEval(universe, env, 'List<0>');
+
+ Expect.equals('List<int>', rti.testingRtiToString(rti1));
+ Expect.equals('List<int>', rti.testingRtiToString(rti2));
+ Expect.equals('List<int>', rti.testingRtiToString(rti3));
+ checkRtiIdentical(rti1, rti2);
+ checkRtiIdentical(rti1, rti3);
+}
+
+main() {
+ testInterface1();
+ testInterface2();
+}
diff --git a/tests/compiler/dart2js_extra/rti/simple_2_test.dart b/tests/compiler/dart2js_extra/rti/simple_2_test.dart
new file mode 100644
index 0000000..ce81c7b
--- /dev/null
+++ b/tests/compiler/dart2js_extra/rti/simple_2_test.dart
@@ -0,0 +1,83 @@
+// Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+import 'dart:_rti' as rti;
+import "package:expect/expect.dart";
+
+testDynamic1() {
+ var universe = rti.testingCreateUniverse();
+
+ var dynamicRti1 = rti.testingUniverseEval(universe, 'dynamic');
+ var dynamicRti2 = rti.testingUniverseEval(universe, ',,dynamic,,');
+
+ Expect.isTrue(
+ identical(dynamicRti1, dynamicRti2), 'dynamic should be identical');
+ Expect.isFalse(dynamicRti1 is String);
+ Expect.equals('dynamic', rti.testingRtiToString(dynamicRti1));
+}
+
+testDynamic2() {
+ var universe = rti.testingCreateUniverse();
+
+ var dynamicRti1 = rti.testingUniverseEval(universe, 'dynamic');
+ var dynamicRti2 = rti.testingUniverseEval(universe, ',,@,,');
+
+ Expect.isTrue(
+ identical(dynamicRti1, dynamicRti2), 'dynamic should be identical');
+ Expect.isFalse(dynamicRti1 is String);
+ Expect.equals('dynamic', rti.testingRtiToString(dynamicRti1));
+}
+
+testInterface1() {
+ var universe = rti.testingCreateUniverse();
+
+ var rti1 = rti.testingUniverseEval(universe, 'int');
+ var rti2 = rti.testingUniverseEval(universe, ',,int,,');
+
+ Expect.isTrue(identical(rti1, rti2));
+ Expect.isFalse(rti1 is String);
+ Expect.equals('int', rti.testingRtiToString(rti1));
+}
+
+testInterface2() {
+ var universe = rti.testingCreateUniverse();
+
+ var rti1 = rti.testingUniverseEval(universe, 'Foo<int,bool>');
+ var rti2 = rti.testingUniverseEval(universe, 'Foo<int,bool>');
+
+ Expect.isTrue(identical(rti1, rti2));
+ Expect.isFalse(rti1 is String);
+ Expect.equals('Foo<int, bool>', rti.testingRtiToString(rti1));
+}
+
+testInterface3() {
+ var universe = rti.testingCreateUniverse();
+
+ var rti1 = rti.testingUniverseEval(universe, 'Foo<Bar<int>,Bar<bool>>');
+ var rti2 = rti.testingUniverseEval(universe, 'Foo<Bar<int>,Bar<bool>>');
+
+ Expect.isTrue(identical(rti1, rti2));
+ Expect.isFalse(rti1 is String);
+ Expect.equals('Foo<Bar<int>, Bar<bool>>', rti.testingRtiToString(rti1));
+}
+
+testInterface4() {
+ var universe = rti.testingCreateUniverse();
+
+ var rti1 = rti.testingUniverseEval(universe, 'Foo<Foo<Foo<Foo<int>>>>');
+ var rti2 = rti.testingUniverseEval(universe, 'Foo<Foo<Foo<Foo<int>>>>');
+
+ Expect.isTrue(identical(rti1, rti2));
+ Expect.isFalse(rti1 is String);
+ Expect.equals('Foo<Foo<Foo<Foo<int>>>>', rti.testingRtiToString(rti1));
+}
+
+main() {
+ testDynamic1();
+ testDynamic2();
+ testInterface1();
+ testInterface2();
+ testInterface3();
+ testInterface4();
+}