Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1115)

Unified Diff: sdk/lib/_internal/compiler/implementation/ssa/builder.dart

Issue 11448009: Represent runtime type information as nested lists. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Address comments. Created 8 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: sdk/lib/_internal/compiler/implementation/ssa/builder.dart
diff --git a/sdk/lib/_internal/compiler/implementation/ssa/builder.dart b/sdk/lib/_internal/compiler/implementation/ssa/builder.dart
index 297c1068ab794322802f9b365489914ef512d4d2..daf3ee3a2acc86dfeefa3c826c3003f36edc1047 100644
--- a/sdk/lib/_internal/compiler/implementation/ssa/builder.dart
+++ b/sdk/lib/_internal/compiler/implementation/ssa/builder.dart
@@ -2633,6 +2633,59 @@ class SsaBuilder extends ResolvedVisitor implements Visitor {
push(result);
}
+ createForeign(String code, String type, List<HInstruction> inputs) {
ngeoffray 2012/12/06 23:21:45 HForeign createForeign
karlklose 2012/12/07 07:46:35 Done.
+ return new HForeign(new LiteralDartString(code),
+ new LiteralDartString(type),
+ inputs);
+ }
+
+ HInstruction getRuntimeTypeInfo(HInstruction target) {
+ pushInvokeHelper1(interceptors.getGetRuntimeTypeInfo(), target);
+ return pop();
+ }
+
+ // TODO(karlklose): change construction of the representations to be GVN'able.
ngeoffray 2012/12/06 23:21:45 You can also reference the bug number.
karlklose 2012/12/07 07:46:35 Done.
+ List<HInstruction> buildTypeArgumentRepresentations(DartType type) {
+ HInstruction createForeignArray(String code, inputs) {
+ return createForeign(code, 'JSArray', inputs);
ngeoffray 2012/12/06 23:21:45 Should be '=List' but i plan on fixing that.
karlklose 2012/12/07 07:46:35 I changed it to '=List'.
+ }
+ HInstruction typeInfo;
+
+ /// Helper to create an instruction that contains the runtime value of
+ /// the type variable [variable].
+ HInstruction getTypeArgument(TypeVariableType variable) {
+ if (typeInfo == null) {
+ typeInfo = getRuntimeTypeInfo(localsHandler.readThis());
+ }
+ int intIndex = RuntimeTypeInformation.getTypeVariableIndex(variable);
+ HInstruction index = graph.addConstantInt(intIndex, constantSystem);
+ return createForeignArray('#[#]', <HInstruction>[typeInfo, index]);
+ }
+
+ // Compute the representation of the type arguments, including access
+ // to the runtime type information for type variables as instructions.
+ HInstruction representations;
+ if (type.element.isTypeVariable()) {
+ return <HInstruction>[getTypeArgument(type)];
+ } else {
+ assert(type.element.isClass());
+ List<HInstruction> arguments = <HInstruction>[];
+ InterfaceType interface = type;
+ for (DartType argument in interface.typeArguments) {
+ List<HInstruction> inputs = <HInstruction>[];
+ String template = rti.getTypeRepresentation(argument, (variable) {
+ HInstruction runtimeType = getTypeArgument(variable);
+ add(runtimeType);
+ inputs.add(runtimeType);
+ });
+ HInstruction representation = createForeignArray(template, inputs);
+ add(representation);
+ arguments.add(representation);
+ }
+ return arguments;
+ }
+ }
+
visitOperatorSend(node) {
assert(node.selector is Operator);
if (!methodInterceptionEnabled) {
@@ -2668,7 +2721,6 @@ class SsaBuilder extends ResolvedVisitor implements Visitor {
typeAnnotation = argument.asSend().receiver;
isNot = true;
}
-
DartType type = elements.getType(typeAnnotation);
if (type.isMalformed) {
String reasons = fetchReasonsFromMalformedType(type);
@@ -2679,29 +2731,53 @@ class SsaBuilder extends ResolvedVisitor implements Visitor {
}
return;
}
- HInstruction typeInfo = null;
- if (RuntimeTypeInformation.hasTypeArguments(type)) {
- pushInvokeHelper1(interceptors.getGetRuntimeTypeInfo(), expression);
- typeInfo = pop();
- }
if (type.element.isTypeVariable()) {
- // TODO(karlklose): We currently answer true to any is check
- // involving a type variable -- both is T and is !T -- until
- // we have a proper implementation of reified generics.
+ // TODO(karlklose): remove this check when the backend can deal with
+ // checks of the form [:o is T:] where [:T:] is a type variable.
stack.add(graph.addConstantBool(true, constantSystem));
+ return;
+ }
+
+ HInstruction instruction;
+ if (RuntimeTypeInformation.hasTypeArguments(type) ||
+ type.element.isTypeVariable()) {
ngeoffray 2012/12/06 23:21:45 I don't think this can happen (see line 2734 and 2
karlklose 2012/12/07 07:46:35 You are right, this is a left-over from trying to
+ HInstruction typeInfo = getRuntimeTypeInfo(expression);
+ // TODO(karlklose): make isSubtype a HInstruction to enable
+ // optimizations?
ngeoffray 2012/12/06 23:21:45 No, I think we need to put compiler flags on the m
karlklose 2012/12/07 07:46:35 I will remove the comment. Let's chat about dartbu
+ Element helper = compiler.findHelper(const SourceString('isSubtype'));
+ HInstruction isSubtype = new HStatic(helper);
+ add(isSubtype);
+ // Build a list of representations for the type arguments.
+ List<HInstruction> representations =
+ buildTypeArgumentRepresentations(type);
+ // For each type argument, build a call to isSubtype, with the type
+ // argument as first and the representation of the tested type as
+ // second argument.
+ List<HInstruction> checks = <HInstruction>[];
+ int index = 0;
+ representations.forEach((HInstruction representation) {
+ HInstruction position = graph.addConstantInt(index, constantSystem);
+ // Get the index'th type argument from the runtime type information.
+ HInstruction typeArgument =
+ createForeign('#[#]', 'Object', [typeInfo, position]);
+ add(typeArgument);
+ // Create the call to isSubtype.
+ List<HInstruction> inputs =
+ <HInstruction>[isSubtype, typeArgument, representation];
+ HInstruction call = new HInvokeStatic(inputs);
+ add(call);
+ checks.add(call);
+ index++;
+ });
+ instruction = new HIs.withArgumentChecks(type, expression, checks);
} else {
- HInstruction instruction;
- if (typeInfo != null) {
- instruction = new HIs.withTypeInfoCall(type, expression, typeInfo);
- } else {
- instruction = new HIs(type, expression);
- }
- if (isNot) {
- add(instruction);
- instruction = new HNot(instruction);
- }
- push(instruction);
+ instruction = new HIs(type, expression);
+ }
+ if (isNot) {
+ add(instruction);
+ instruction = new HNot(instruction);
}
+ push(instruction);
} else if (const SourceString("as") == op.source) {
visit(node.receiver);
HInstruction expression = pop();
@@ -3104,9 +3180,8 @@ class SsaBuilder extends ResolvedVisitor implements Visitor {
return graph.addConstantNull(constantSystem);
}
- // These variables are shared between invocations of the helper methods.
+ // These variables are shared between invocations of the helper.
HInstruction typeInfo;
- StringBuffer template = new StringBuffer();
List<HInstruction> inputs = <HInstruction>[];
/**
@@ -3131,10 +3206,8 @@ class SsaBuilder extends ResolvedVisitor implements Visitor {
typeInfo = pop();
}
int index = RuntimeTypeInformation.getTypeVariableIndex(type);
- HInstruction foreign = new HForeign(
- new LiteralDartString('#[$index]'),
- new LiteralDartString('String'),
- <HInstruction>[typeInfo]);
+ HInstruction foreign = createForeign('#[$index]', 'String',
+ <HInstruction>[typeInfo]);
add(foreign);
inputs.add(foreign);
} else {
@@ -3145,53 +3218,9 @@ class SsaBuilder extends ResolvedVisitor implements Visitor {
}
}
- /**
- * Helper to build an instruction that builds the string representation for
- * this type, where type variables are substituted by their runtime value.
- *
- * Examples:
- * Type Template Inputs
- * int 'int' []
- * C<int, int> 'C<int, int>' []
- * Var # [getRuntimeType(this).Var]
- * C<int, D<Var>> 'C<int, D<' + # + '>>' [getRuntimeType(this).Var]
- */
- void buildTypeString(DartType type, {isInQuotes: false}) {
- if (type is TypeVariableType) {
- addTypeVariableReference(type);
- template.add(isInQuotes ? "' + # +'" : "#");
- } else if (type is InterfaceType) {
- bool isFirstVariable = true;
- InterfaceType interfaceType = type;
- bool hasTypeArguments = !interfaceType.isRaw;
- if (!isInQuotes) template.add("'");
- template.add(backend.namer.getName(type.element));
- if (hasTypeArguments) {
- template.add("<");
- for (DartType argument in interfaceType.typeArguments) {
- if (!isFirstVariable) {
- template.add(", ");
- } else {
- isFirstVariable = false;
- }
- buildTypeString(argument, isInQuotes: true);
- }
- template.add(">");
- }
- if (!isInQuotes) template.add("'");
- } else {
- assert(type is TypedefType);
- if (!isInQuotes) template.add("'");
- template.add(backend.namer.getName(argument.element));
- if (!isInQuotes) template.add("'");
- }
- }
-
- buildTypeString(argument, isInQuotes: false);
- HInstruction result =
- new HForeign(new LiteralDartString("$template"),
- new LiteralDartString('String'),
- inputs);
+ String template = rti.getTypeRepresentation(argument,
+ addTypeVariableReference);
+ HInstruction result = createForeign(template, 'String', inputs);
add(result);
return result;
}

Powered by Google App Engine
This is Rietveld 408576698