| OLD | NEW |
| (Empty) |
| 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 | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 class Universe { | |
| 6 Map<Element, String> generatedCode; | |
| 7 Map<Element, String> generatedBailoutCode; | |
| 8 final Set<ClassElement> instantiatedClasses; | |
| 9 final Set<SourceString> instantiatedClassInstanceFields; | |
| 10 final Set<FunctionElement> staticFunctionsNeedingGetter; | |
| 11 final Map<SourceString, Set<Selector>> invokedNames; | |
| 12 final Set<SourceString> invokedGetters; | |
| 13 final Set<SourceString> invokedSetters; | |
| 14 final Map<String, LibraryElement> libraries; | |
| 15 // TODO(ngeoffray): This should be a Set<Type>. | |
| 16 final Set<Element> isChecks; | |
| 17 | |
| 18 Universe() : generatedCode = new Map<Element, String>(), | |
| 19 generatedBailoutCode = new Map<Element, String>(), | |
| 20 libraries = new Map<String, LibraryElement>(), | |
| 21 instantiatedClasses = new Set<ClassElement>(), | |
| 22 instantiatedClassInstanceFields = new Set<SourceString>(), | |
| 23 staticFunctionsNeedingGetter = new Set<FunctionElement>(), | |
| 24 invokedNames = new Map<SourceString, Set<Selector>>(), | |
| 25 invokedGetters = new Set<SourceString>(), | |
| 26 invokedSetters = new Set<SourceString>(), | |
| 27 isChecks = new Set<Element>(); | |
| 28 | |
| 29 void addGeneratedCode(WorkItem work, String code) { | |
| 30 generatedCode[work.element] = code; | |
| 31 } | |
| 32 | |
| 33 void addBailoutCode(WorkItem work, String code) { | |
| 34 generatedBailoutCode[work.element] = code; | |
| 35 } | |
| 36 } | |
| 37 | |
| 38 class SelectorKind { | |
| 39 final String name; | |
| 40 const SelectorKind(this.name); | |
| 41 | |
| 42 static final SelectorKind GETTER = const SelectorKind('getter'); | |
| 43 static final SelectorKind SETTER = const SelectorKind('setter'); | |
| 44 static final SelectorKind INVOCATION = const SelectorKind('invocation'); | |
| 45 static final SelectorKind OPERATOR = const SelectorKind('operator'); | |
| 46 static final SelectorKind INDEX = const SelectorKind('index'); | |
| 47 | |
| 48 toString() => name; | |
| 49 } | |
| 50 | |
| 51 class Selector implements Hashable { | |
| 52 // The numbers of arguments of the selector. Includes named | |
| 53 // arguments. | |
| 54 final int argumentCount; | |
| 55 final SelectorKind kind; | |
| 56 const Selector(this.kind, this.argumentCount); | |
| 57 | |
| 58 int hashCode() => argumentCount + 1000 * namedArguments.length; | |
| 59 List<SourceString> get namedArguments() => const <SourceString>[]; | |
| 60 int get namedArgumentCount() => 0; | |
| 61 int get positionalArgumentCount() => argumentCount; | |
| 62 | |
| 63 static final Selector GETTER = const Selector(SelectorKind.GETTER, 0); | |
| 64 static final Selector SETTER = const Selector(SelectorKind.SETTER, 1); | |
| 65 static final Selector UNARY_OPERATOR = | |
| 66 const Selector(SelectorKind.OPERATOR, 0); | |
| 67 static final Selector BINARY_OPERATOR = | |
| 68 const Selector(SelectorKind.OPERATOR, 1); | |
| 69 static final Selector INDEX = const Selector(SelectorKind.INDEX, 1); | |
| 70 static final Selector INDEX_SET = const Selector(SelectorKind.INDEX, 2); | |
| 71 static final Selector INDEX_AND_INDEX_SET = | |
| 72 const Selector(SelectorKind.INDEX, 2); | |
| 73 static final Selector GETTER_AND_SETTER = | |
| 74 const Selector(SelectorKind.SETTER, 1); | |
| 75 static final Selector INVOCATION_0 = | |
| 76 const Selector(SelectorKind.INVOCATION, 0); | |
| 77 static final Selector INVOCATION_1 = | |
| 78 const Selector(SelectorKind.INVOCATION, 1); | |
| 79 static final Selector INVOCATION_2 = | |
| 80 const Selector(SelectorKind.INVOCATION, 2); | |
| 81 | |
| 82 bool applies(FunctionParameters parameters) { | |
| 83 if (argumentCount > parameters.parameterCount) return false; | |
| 84 int requiredParameterCount = parameters.requiredParameterCount; | |
| 85 int optionalParameterCount = parameters.optionalParameterCount; | |
| 86 | |
| 87 bool hasOptionalParameters = !parameters.optionalParameters.isEmpty(); | |
| 88 if (namedArguments.isEmpty()) { | |
| 89 if (!hasOptionalParameters) { | |
| 90 return requiredParameterCount == argumentCount; | |
| 91 } else { | |
| 92 return argumentCount >= requiredParameterCount && | |
| 93 argumentCount <= requiredParameterCount + optionalParameterCount; | |
| 94 } | |
| 95 } else { | |
| 96 if (!hasOptionalParameters) return false; | |
| 97 Link<Element> remainingNamedParameters = parameters.optionalParameters; | |
| 98 for (int i = requiredParameterCount; i < positionalArgumentCount; i++) { | |
| 99 remainingNamedParameters = remainingNamedParameters.tail; | |
| 100 } | |
| 101 Set<SourceString> nameSet = new Set<SourceString>(); | |
| 102 for (; | |
| 103 !remainingNamedParameters.isEmpty(); | |
| 104 remainingNamedParameters = remainingNamedParameters.tail) { | |
| 105 nameSet.add(remainingNamedParameters.head.name); | |
| 106 } | |
| 107 | |
| 108 for (SourceString name in namedArguments) { | |
| 109 if (!nameSet.contains(name)) { | |
| 110 return false; | |
| 111 } | |
| 112 nameSet.remove(name); | |
| 113 } | |
| 114 return true; | |
| 115 } | |
| 116 } | |
| 117 | |
| 118 /** | |
| 119 * Returns [:true:] if the selector and the [element] match; [:false:] | |
| 120 * otherwise. | |
| 121 */ | |
| 122 bool addSendArgumentsToList(Send send, | |
| 123 List list, | |
| 124 FunctionParameters parameters, | |
| 125 compileArgument(Node argument), | |
| 126 compileConstant(Element element)) { | |
| 127 void addMatchingSendArgumentsToList(Link<Node> link) { | |
| 128 for (; !link.isEmpty(); link = link.tail) { | |
| 129 list.add(compileArgument(link.head)); | |
| 130 } | |
| 131 } | |
| 132 | |
| 133 if (!this.applies(parameters)) return false; | |
| 134 if (this.positionalArgumentCount == parameters.parameterCount) { | |
| 135 addMatchingSendArgumentsToList(send.arguments); | |
| 136 return true; | |
| 137 } | |
| 138 | |
| 139 // If there are named arguments, provide them in the order | |
| 140 // expected by the called function, which is the source order. | |
| 141 | |
| 142 // Visit positional arguments and add them to the list. | |
| 143 Link<Node> arguments = send.arguments; | |
| 144 int positionalArgumentCount = this.positionalArgumentCount; | |
| 145 for (int i = 0; | |
| 146 i < positionalArgumentCount; | |
| 147 arguments = arguments.tail, i++) { | |
| 148 list.add(compileArgument(arguments.head)); | |
| 149 } | |
| 150 | |
| 151 // Visit named arguments and add them into a temporary list. | |
| 152 List namedArguments = []; | |
| 153 for (; !arguments.isEmpty(); arguments = arguments.tail) { | |
| 154 NamedArgument namedArgument = arguments.head; | |
| 155 namedArguments.add(compileArgument(namedArgument.expression)); | |
| 156 } | |
| 157 | |
| 158 Link<Element> remainingNamedParameters = parameters.optionalParameters; | |
| 159 // Skip the optional parameters that have been given in the | |
| 160 // positional arguments. | |
| 161 for (int i = parameters.requiredParameterCount; | |
| 162 i < positionalArgumentCount; | |
| 163 i++) { | |
| 164 remainingNamedParameters = remainingNamedParameters.tail; | |
| 165 } | |
| 166 | |
| 167 // Loop over the remaining named parameters, and try to find | |
| 168 // their values: either in the temporary list or using the | |
| 169 // default value. | |
| 170 for (; | |
| 171 !remainingNamedParameters.isEmpty(); | |
| 172 remainingNamedParameters = remainingNamedParameters.tail) { | |
| 173 Element parameter = remainingNamedParameters.head; | |
| 174 int foundIndex = -1; | |
| 175 for (int i = 0; i < this.namedArguments.length; i++) { | |
| 176 SourceString name = this.namedArguments[i]; | |
| 177 if (name == parameter.name) { | |
| 178 foundIndex = i; | |
| 179 break; | |
| 180 } | |
| 181 } | |
| 182 if (foundIndex != -1) { | |
| 183 list.add(namedArguments[foundIndex]); | |
| 184 } else { | |
| 185 list.add(compileConstant(parameter)); | |
| 186 } | |
| 187 } | |
| 188 return true; | |
| 189 } | |
| 190 | |
| 191 static bool sameNames(List<SourceString> first, List<SourceString> second) { | |
| 192 for (int i = 0; i < first.length; i++) { | |
| 193 if (first[i] != second[i]) return false; | |
| 194 } | |
| 195 return true; | |
| 196 } | |
| 197 | |
| 198 bool operator ==(other) { | |
| 199 if (other is !Selector) return false; | |
| 200 return argumentCount == other.argumentCount | |
| 201 && namedArguments.length == other.namedArguments.length | |
| 202 && sameNames(namedArguments, other.namedArguments); | |
| 203 } | |
| 204 | |
| 205 List<SourceString> getOrderedNamedArguments() => namedArguments; | |
| 206 | |
| 207 toString() => '$kind $argumentCount'; | |
| 208 } | |
| 209 | |
| 210 class Invocation extends Selector { | |
| 211 final List<SourceString> namedArguments; | |
| 212 List<SourceString> orderedNamedArguments; | |
| 213 int get namedArgumentCount() => namedArguments.length; | |
| 214 int get positionalArgumentCount() => argumentCount - namedArgumentCount; | |
| 215 | |
| 216 Invocation(int argumentCount, | |
| 217 [List<SourceString> names = const <SourceString>[]]) | |
| 218 : super(SelectorKind.INVOCATION, argumentCount), | |
| 219 namedArguments = names, | |
| 220 orderedNamedArguments = const <SourceString>[]; | |
| 221 | |
| 222 List<SourceString> getOrderedNamedArguments() { | |
| 223 if (namedArguments.isEmpty()) return namedArguments; | |
| 224 // We use the empty const List as a sentinel. | |
| 225 if (!orderedNamedArguments.isEmpty()) return orderedNamedArguments; | |
| 226 | |
| 227 List<SourceString> list = new List<SourceString>.from(namedArguments); | |
| 228 list.sort((SourceString first, SourceString second) { | |
| 229 return first.slowToString().compareTo(second.slowToString()); | |
| 230 }); | |
| 231 orderedNamedArguments = list; | |
| 232 return orderedNamedArguments; | |
| 233 } | |
| 234 } | |
| OLD | NEW |