Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 | 4 |
| 5 class Constant implements Hashable { | 5 class Constant implements Hashable { |
| 6 const Constant(); | 6 const Constant(); |
| 7 | 7 |
| 8 bool isNull() => false; | 8 bool isNull() => false; |
| 9 bool isBool() => false; | 9 bool isBool() => false; |
| 10 bool isTrue() => false; | 10 bool isTrue() => false; |
| 11 bool isFalse() => false; | 11 bool isFalse() => false; |
| 12 bool isInt() => false; | 12 bool isInt() => false; |
| 13 bool isDouble() => false; | 13 bool isDouble() => false; |
| 14 bool isNum() => false; | 14 bool isNum() => false; |
| 15 bool isString() => false; | 15 bool isString() => false; |
| 16 bool isList() => false; | 16 bool isList() => false; |
| 17 bool isMap() => false; | 17 bool isMap() => false; |
| 18 bool isConstructedObject() => false; | 18 bool isConstructedObject() => false; |
| 19 bool isFunction() => false; | |
| 19 /** Returns true if the constant is null, a bool, a number or a string. */ | 20 /** Returns true if the constant is null, a bool, a number or a string. */ |
| 20 bool isPrimitive() => false; | 21 bool isPrimitive() => false; |
| 21 /** Returns true if the constant is a list, a map or a constructed object. */ | 22 /** Returns true if the constant is a list, a map or a constructed object. */ |
| 22 bool isObject() => false; | 23 bool isObject() => false; |
| 23 | 24 |
| 24 bool isNaN() => false; | 25 bool isNaN() => false; |
| 25 | 26 |
| 26 abstract void _writeJsCode(CodeBuffer buffer, ConstantHandler handler); | 27 abstract void _writeJsCode(CodeBuffer buffer, ConstantHandler handler); |
| 27 /** | 28 /** |
| 28 * Unless the constant can be emitted multiple times (as for numbers and | 29 * Unless the constant can be emitted multiple times (as for numbers and |
| 29 * strings) adds its canonical name to the buffer. | 30 * strings) adds its canonical name to the buffer. |
| 30 */ | 31 */ |
| 31 abstract void _writeCanonicalizedJsCode(CodeBuffer buffer, | 32 abstract void _writeCanonicalizedJsCode(CodeBuffer buffer, |
| 32 ConstantHandler handler); | 33 ConstantHandler handler); |
| 33 abstract List<Constant> getDependencies(); | 34 abstract List<Constant> getDependencies(); |
| 34 } | 35 } |
| 35 | 36 |
| 37 class FunctionConstant extends Constant { | |
| 38 Element element; | |
| 39 | |
| 40 FunctionConstant(this.element); | |
| 41 | |
| 42 bool isFunction() => true; | |
| 43 | |
| 44 bool operator ==(var other) { | |
| 45 if (other is !FunctionConstant) return false; | |
| 46 return other.element === element; | |
| 47 } | |
| 48 | |
| 49 String toString() => element.toString(); | |
| 50 List<Constant> getDependencies() => const <Constant>[]; | |
| 51 DartString toDartString() { | |
| 52 return new DartString.literal(element.name.slowToString()); | |
| 53 } | |
| 54 | |
| 55 void _writeJsCode(CodeBuffer buffer, ConstantHandler handler) { | |
| 56 compiler.internalError( | |
| 57 "A constant function does not need specific JS code"); | |
| 58 } | |
| 59 | |
| 60 void _writeCanonicalizedJsCode(CodeBuffer buffer, ConstantHandler handler) { | |
| 61 buffer.add(handler.compiler.namer.isolatePropertiesAccess(element)); | |
| 62 } | |
| 63 | |
| 64 int hashCode() => element.hashCode(); | |
|
ahe
2012/08/28 13:35:54
You should make sure that FunctionConstant has a d
| |
| 65 } | |
| 66 | |
| 36 class PrimitiveConstant extends Constant { | 67 class PrimitiveConstant extends Constant { |
| 37 abstract get value; | 68 abstract get value; |
| 38 const PrimitiveConstant(); | 69 const PrimitiveConstant(); |
| 39 bool isPrimitive() => true; | 70 bool isPrimitive() => true; |
| 40 | 71 |
| 41 bool operator ==(var other) { | 72 bool operator ==(var other) { |
| 42 if (other is !PrimitiveConstant) return false; | 73 if (other is !PrimitiveConstant) return false; |
| 43 PrimitiveConstant otherPrimitive = other; | 74 PrimitiveConstant otherPrimitive = other; |
| 44 // We use == instead of === so that DartStrings compare correctly. | 75 // We use == instead of === so that DartStrings compare correctly. |
| 45 return value == otherPrimitive.value; | 76 return value == otherPrimitive.value; |
| (...skipping 446 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 492 final Set<VariableElement> pendingVariables; | 523 final Set<VariableElement> pendingVariables; |
| 493 | 524 |
| 494 ConstantHandler(Compiler compiler) | 525 ConstantHandler(Compiler compiler) |
| 495 : initialVariableValues = new Map<VariableElement, Dynamic>(), | 526 : initialVariableValues = new Map<VariableElement, Dynamic>(), |
| 496 compiledConstants = new Map<Constant, String>(), | 527 compiledConstants = new Map<Constant, String>(), |
| 497 pendingVariables = new Set<VariableElement>(), | 528 pendingVariables = new Set<VariableElement>(), |
| 498 super(compiler); | 529 super(compiler); |
| 499 String get name => 'ConstantHandler'; | 530 String get name => 'ConstantHandler'; |
| 500 | 531 |
| 501 void registerCompileTimeConstant(Constant constant) { | 532 void registerCompileTimeConstant(Constant constant) { |
| 502 Function ifAbsentThunk = (() => compiler.namer.getFreshGlobalName("CTC")); | 533 Function ifAbsentThunk = (() { |
| 534 return constant.isFunction() | |
| 535 ? null : compiler.namer.getFreshGlobalName("CTC"); | |
| 536 }); | |
| 503 compiledConstants.putIfAbsent(constant, ifAbsentThunk); | 537 compiledConstants.putIfAbsent(constant, ifAbsentThunk); |
| 504 } | 538 } |
| 505 | 539 |
| 506 /** | 540 /** |
| 507 * Compiles the initial value of the given field and stores it in an internal | 541 * Compiles the initial value of the given field and stores it in an internal |
| 508 * map. | 542 * map. |
| 509 * | 543 * |
| 510 * [WorkItem] must contain a [VariableElement] refering to a global or | 544 * [WorkItem] must contain a [VariableElement] refering to a global or |
| 511 * static field. | 545 * static field. |
| 512 */ | 546 */ |
| (...skipping 324 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 837 // TODO(floitsch): provide better error-messages. | 871 // TODO(floitsch): provide better error-messages. |
| 838 Constant visitSend(Send send) { | 872 Constant visitSend(Send send) { |
| 839 Element element = elements[send]; | 873 Element element = elements[send]; |
| 840 if (Elements.isStaticOrTopLevelField(element)) { | 874 if (Elements.isStaticOrTopLevelField(element)) { |
| 841 if (element.modifiers === null || | 875 if (element.modifiers === null || |
| 842 // TODO(johnniwinther): This should eventually be [isConst]. | 876 // TODO(johnniwinther): This should eventually be [isConst]. |
| 843 !element.modifiers.isFinalOrConst()) { | 877 !element.modifiers.isFinalOrConst()) { |
| 844 error(send); | 878 error(send); |
| 845 } | 879 } |
| 846 return compiler.compileVariable(element); | 880 return compiler.compileVariable(element); |
| 881 } else if (Elements.isStaticOrTopLevelFunction(element) | |
| 882 && send.isPropertyAccess) { | |
| 883 compiler.codegenWorld.staticFunctionsNeedingGetter.add(element); | |
| 884 Constant constant = new FunctionConstant(element); | |
| 885 compiler.constantHandler.registerCompileTimeConstant(constant); | |
| 886 return constant; | |
| 847 } else if (send.isPrefix) { | 887 } else if (send.isPrefix) { |
| 848 assert(send.isOperator); | 888 assert(send.isOperator); |
| 849 Constant receiverConstant = evaluate(send.receiver); | 889 Constant receiverConstant = evaluate(send.receiver); |
| 850 Operator op = send.selector; | 890 Operator op = send.selector; |
| 851 Constant folded; | 891 Constant folded; |
| 852 switch (op.source.stringValue) { | 892 switch (op.source.stringValue) { |
| 853 case "!": | 893 case "!": |
| 854 folded = const NotOperation().fold(receiverConstant); | 894 folded = const NotOperation().fold(receiverConstant); |
| 855 break; | 895 break; |
| 856 case "-": | 896 case "-": |
| (...skipping 313 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1170 Constant fieldValue = fieldValues[field]; | 1210 Constant fieldValue = fieldValues[field]; |
| 1171 if (fieldValue === null) { | 1211 if (fieldValue === null) { |
| 1172 // Use the default value. | 1212 // Use the default value. |
| 1173 fieldValue = compiler.compileVariable(field); | 1213 fieldValue = compiler.compileVariable(field); |
| 1174 } | 1214 } |
| 1175 jsNewArguments.add(fieldValue); | 1215 jsNewArguments.add(fieldValue); |
| 1176 }); | 1216 }); |
| 1177 return jsNewArguments; | 1217 return jsNewArguments; |
| 1178 } | 1218 } |
| 1179 } | 1219 } |
| OLD | NEW |