| OLD | NEW |
| 1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2015, 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 library dart2js.semantics_visitor.resolver; | 5 library dart2js.semantics_visitor.resolver; |
| 6 | 6 |
| 7 import '../constants/expressions.dart'; | 7 import '../constants/expressions.dart'; |
| 8 import '../dart_types.dart'; | 8 import '../dart_types.dart'; |
| 9 import '../diagnostics/invariant.dart' show | |
| 10 invariant; | |
| 11 import '../diagnostics/messages.dart' show | 9 import '../diagnostics/messages.dart' show |
| 12 MessageKind; | 10 MessageKind; |
| 13 import '../diagnostics/spannable.dart' show | 11 import '../diagnostics/spannable.dart' show |
| 14 Spannable, | 12 Spannable, |
| 15 SpannableAssertionFailure; | 13 SpannableAssertionFailure; |
| 16 import '../elements/elements.dart'; | 14 import '../elements/elements.dart'; |
| 17 import '../tree/tree.dart'; | 15 import '../tree/tree.dart'; |
| 18 import '../universe/universe.dart'; | 16 import '../universe/universe.dart'; |
| 19 | 17 |
| 20 import 'access_semantics.dart'; | 18 import 'access_semantics.dart'; |
| 21 import 'operators.dart'; | |
| 22 import 'semantic_visitor.dart'; | 19 import 'semantic_visitor.dart'; |
| 23 import 'send_structure.dart'; | 20 import 'send_structure.dart'; |
| 24 import 'tree_elements.dart'; | 21 import 'tree_elements.dart'; |
| 25 | 22 |
| 26 enum SendStructureKind { | |
| 27 GET, | |
| 28 SET, | |
| 29 INVOKE, | |
| 30 UNARY, | |
| 31 NOT, | |
| 32 BINARY, | |
| 33 EQ, | |
| 34 NOT_EQ, | |
| 35 COMPOUND, | |
| 36 INDEX, | |
| 37 INDEX_SET, | |
| 38 COMPOUND_INDEX_SET, | |
| 39 PREFIX, | |
| 40 POSTFIX, | |
| 41 INDEX_PREFIX, | |
| 42 INDEX_POSTFIX, | |
| 43 } | |
| 44 | |
| 45 abstract class SendResolverMixin { | 23 abstract class SendResolverMixin { |
| 46 TreeElements get elements; | 24 TreeElements get elements; |
| 47 | 25 |
| 48 internalError(Spannable spannable, String message); | 26 internalError(Spannable spannable, String message); |
| 49 | 27 |
| 50 AccessSemantics handleCompoundErroneousSetterAccess( | |
| 51 Send node, | |
| 52 Element setter, | |
| 53 Element getter) { | |
| 54 assert(invariant(node, Elements.isUnresolved(setter), | |
| 55 message: "Unexpected erreneous compound setter: $setter.")); | |
| 56 if (getter.isStatic) { | |
| 57 if (getter.isGetter) { | |
| 58 return new CompoundAccessSemantics( | |
| 59 CompoundAccessKind.UNRESOLVED_STATIC_SETTER, getter, setter); | |
| 60 } else if (getter.isField) { | |
| 61 // TODO(johnniwinther): Handle const field separately. | |
| 62 assert(invariant(node, getter.isFinal || getter.isConst, | |
| 63 message: "Field expected to be final or const.")); | |
| 64 return new StaticAccess.finalStaticField(getter); | |
| 65 } else if (getter.isFunction) { | |
| 66 return new StaticAccess.staticMethod(getter); | |
| 67 } else { | |
| 68 return internalError(node, | |
| 69 "Unexpected erroneous static compound: getter=$getter"); | |
| 70 } | |
| 71 } else if (getter.isTopLevel) { | |
| 72 if (getter.isGetter) { | |
| 73 return new CompoundAccessSemantics( | |
| 74 CompoundAccessKind.UNRESOLVED_TOPLEVEL_SETTER, getter, setter); | |
| 75 } else if (getter.isField) { | |
| 76 // TODO(johnniwinther): Handle const field separately. | |
| 77 assert(invariant(node, getter.isFinal || getter.isConst, | |
| 78 message: "Field expected to be final or const.")); | |
| 79 return new StaticAccess.finalTopLevelField(getter); | |
| 80 } else if (getter.isFunction) { | |
| 81 return new StaticAccess.topLevelMethod(getter); | |
| 82 } else { | |
| 83 return internalError(node, | |
| 84 "Unexpected erroneous top level compound: getter=$getter"); | |
| 85 } | |
| 86 } else if (getter.isParameter) { | |
| 87 assert(invariant(node, getter.isFinal, | |
| 88 message: "Parameter expected to be final.")); | |
| 89 return new StaticAccess.finalParameter(getter); | |
| 90 } else if (getter.isLocal) { | |
| 91 if (getter.isVariable) { | |
| 92 // TODO(johnniwinther): Handle const variable separately. | |
| 93 assert(invariant(node, getter.isFinal || getter.isConst, | |
| 94 message: "Variable expected to be final or const.")); | |
| 95 return new StaticAccess.finalLocalVariable(getter); | |
| 96 } else if (getter.isFunction) { | |
| 97 return new StaticAccess.localFunction(getter); | |
| 98 } else { | |
| 99 return internalError(node, | |
| 100 "Unexpected erroneous local compound: getter=$getter"); | |
| 101 } | |
| 102 } else if (getter.isErroneous) { | |
| 103 return new StaticAccess.unresolved(getter); | |
| 104 } else { | |
| 105 return internalError(node, | |
| 106 "Unexpected erroneous compound: getter=$getter"); | |
| 107 } | |
| 108 } | |
| 109 | |
| 110 AccessSemantics handleStaticallyResolvedAccess( | |
| 111 Send node, | |
| 112 Element element, | |
| 113 Element getter, | |
| 114 {bool isCompound}) { | |
| 115 if (element == null) { | |
| 116 assert(invariant(node, isCompound, message: | |
| 117 "Non-compound static access without element.")); | |
| 118 assert(invariant(node, getter != null, message: | |
| 119 "Compound static access without element.")); | |
| 120 return handleCompoundErroneousSetterAccess(node, element, getter); | |
| 121 } | |
| 122 if (element.isErroneous) { | |
| 123 if (isCompound) { | |
| 124 return handleCompoundErroneousSetterAccess(node, element, getter); | |
| 125 } | |
| 126 return new StaticAccess.unresolved(element); | |
| 127 } else if (element.isParameter) { | |
| 128 if (element.isFinal) { | |
| 129 return new StaticAccess.finalParameter(element); | |
| 130 } else { | |
| 131 return new StaticAccess.parameter(element); | |
| 132 } | |
| 133 } else if (element.isLocal) { | |
| 134 if (element.isFunction) { | |
| 135 return new StaticAccess.localFunction(element); | |
| 136 } else if (element.isFinal || element.isConst) { | |
| 137 return new StaticAccess.finalLocalVariable(element); | |
| 138 } else { | |
| 139 return new StaticAccess.localVariable(element); | |
| 140 } | |
| 141 } else if (element.isStatic) { | |
| 142 if (element.isField) { | |
| 143 if (element.isFinal || element.isConst) { | |
| 144 // TODO(johnniwinther): Handle const field separately. | |
| 145 return new StaticAccess.finalStaticField(element); | |
| 146 } | |
| 147 return new StaticAccess.staticField(element); | |
| 148 } else if (element.isGetter) { | |
| 149 if (isCompound) { | |
| 150 return new CompoundAccessSemantics( | |
| 151 CompoundAccessKind.UNRESOLVED_STATIC_SETTER, element, null); | |
| 152 } | |
| 153 return new StaticAccess.staticGetter(element); | |
| 154 } else if (element.isSetter) { | |
| 155 if (getter != null) { | |
| 156 CompoundAccessKind accessKind; | |
| 157 if (getter.isErroneous) { | |
| 158 accessKind = CompoundAccessKind.UNRESOLVED_STATIC_GETTER; | |
| 159 } else if (getter.isAbstractField) { | |
| 160 AbstractFieldElement abstractField = getter; | |
| 161 if (abstractField.getter == null) { | |
| 162 accessKind = CompoundAccessKind.UNRESOLVED_STATIC_GETTER; | |
| 163 } else { | |
| 164 // TODO(johnniwinther): This might be dead code. | |
| 165 getter = abstractField.getter; | |
| 166 accessKind = CompoundAccessKind.STATIC_GETTER_SETTER; | |
| 167 } | |
| 168 } else if (getter.isGetter) { | |
| 169 accessKind = CompoundAccessKind.STATIC_GETTER_SETTER; | |
| 170 } else { | |
| 171 accessKind = CompoundAccessKind.STATIC_METHOD_SETTER; | |
| 172 } | |
| 173 return new CompoundAccessSemantics( | |
| 174 accessKind, getter, element); | |
| 175 } else { | |
| 176 return new StaticAccess.staticSetter(element); | |
| 177 } | |
| 178 } else { | |
| 179 return new StaticAccess.staticMethod(element); | |
| 180 } | |
| 181 } else if (element.isTopLevel) { | |
| 182 if (element.isField) { | |
| 183 if (element.isFinal || element.isConst) { | |
| 184 // TODO(johnniwinther): Handle const field separately. | |
| 185 return new StaticAccess.finalTopLevelField(element); | |
| 186 } | |
| 187 return new StaticAccess.topLevelField(element); | |
| 188 } else if (element.isGetter) { | |
| 189 return new StaticAccess.topLevelGetter(element); | |
| 190 } else if (element.isSetter) { | |
| 191 if (getter != null) { | |
| 192 CompoundAccessKind accessKind; | |
| 193 if (getter.isErroneous) { | |
| 194 accessKind = CompoundAccessKind.UNRESOLVED_TOPLEVEL_GETTER; | |
| 195 } else if (getter.isAbstractField) { | |
| 196 AbstractFieldElement abstractField = getter; | |
| 197 if (abstractField.getter == null) { | |
| 198 accessKind = CompoundAccessKind.UNRESOLVED_TOPLEVEL_GETTER; | |
| 199 } else { | |
| 200 // TODO(johnniwinther): This might be dead code. | |
| 201 getter = abstractField.getter; | |
| 202 accessKind = CompoundAccessKind.TOPLEVEL_GETTER_SETTER; | |
| 203 } | |
| 204 } else if (getter.isGetter) { | |
| 205 accessKind = CompoundAccessKind.TOPLEVEL_GETTER_SETTER; | |
| 206 } else { | |
| 207 accessKind = CompoundAccessKind.TOPLEVEL_METHOD_SETTER; | |
| 208 } | |
| 209 return new CompoundAccessSemantics( | |
| 210 accessKind, getter, element); | |
| 211 } else { | |
| 212 return new StaticAccess.topLevelSetter(element); | |
| 213 } | |
| 214 } else { | |
| 215 return new StaticAccess.topLevelMethod(element); | |
| 216 } | |
| 217 } else { | |
| 218 return internalError( | |
| 219 node, "Unhandled resolved property access: $element"); | |
| 220 } | |
| 221 } | |
| 222 | |
| 223 SendStructure computeSendStructure(Send node) { | |
| 224 SendStructure sendStructure = elements.getSendStructure(node); | |
| 225 if (sendStructure != null) { | |
| 226 return sendStructure; | |
| 227 } | |
| 228 | |
| 229 if (elements.isAssert(node)) { | |
| 230 return internalError(node, "Unexpected assert."); | |
| 231 } | |
| 232 | |
| 233 AssignmentOperator assignmentOperator; | |
| 234 BinaryOperator binaryOperator; | |
| 235 IncDecOperator incDecOperator; | |
| 236 | |
| 237 if (node.isOperator) { | |
| 238 String operatorText = node.selector.asOperator().source; | |
| 239 if (operatorText == 'is') { | |
| 240 return internalError(node, "Unexpected is test."); | |
| 241 } else if (operatorText == 'as') { | |
| 242 return internalError(node, "Unexpected as cast."); | |
| 243 } else if (operatorText == '&&') { | |
| 244 return internalError(node, "Unexpected logical and."); | |
| 245 } else if (operatorText == '||') { | |
| 246 return internalError(node, "Unexpected logical or."); | |
| 247 } else if (operatorText == '??') { | |
| 248 return internalError(node, "Unexpected if-null."); | |
| 249 } | |
| 250 } | |
| 251 | |
| 252 SendStructureKind kind; | |
| 253 | |
| 254 if (node.asSendSet() != null) { | |
| 255 SendSet sendSet = node.asSendSet(); | |
| 256 String operatorText = sendSet.assignmentOperator.source; | |
| 257 if (sendSet.isPrefix || sendSet.isPostfix) { | |
| 258 kind = sendSet.isPrefix | |
| 259 ? SendStructureKind.PREFIX | |
| 260 : SendStructureKind.POSTFIX; | |
| 261 incDecOperator = IncDecOperator.parse(operatorText); | |
| 262 if (incDecOperator == null) { | |
| 263 return internalError( | |
| 264 node, "No inc/dec operator for '$operatorText'."); | |
| 265 } | |
| 266 } else { | |
| 267 assignmentOperator = AssignmentOperator.parse(operatorText); | |
| 268 if (assignmentOperator != null) { | |
| 269 switch (assignmentOperator.kind) { | |
| 270 case AssignmentOperatorKind.ASSIGN: | |
| 271 kind = SendStructureKind.SET; | |
| 272 break; | |
| 273 default: | |
| 274 kind = SendStructureKind.COMPOUND; | |
| 275 } | |
| 276 } else { | |
| 277 return internalError( | |
| 278 node, "No assignment operator for '$operatorText'."); | |
| 279 } | |
| 280 } | |
| 281 } else if (!node.isPropertyAccess) { | |
| 282 kind = SendStructureKind.INVOKE; | |
| 283 } else { | |
| 284 kind = SendStructureKind.GET; | |
| 285 } | |
| 286 | |
| 287 if (node.isOperator) { | |
| 288 String operatorText = node.selector.asOperator().source; | |
| 289 if (node.arguments.isEmpty) { | |
| 290 return internalError(node, "Unexpected unary $operatorText."); | |
| 291 } else { | |
| 292 binaryOperator = BinaryOperator.parse(operatorText); | |
| 293 if (binaryOperator != null) { | |
| 294 switch (binaryOperator.kind) { | |
| 295 case BinaryOperatorKind.EQ: | |
| 296 kind = SendStructureKind.EQ; | |
| 297 return internalError(node, "Unexpected binary $kind."); | |
| 298 case BinaryOperatorKind.NOT_EQ: | |
| 299 kind = SendStructureKind.NOT_EQ; | |
| 300 return internalError(node, "Unexpected binary $kind."); | |
| 301 case BinaryOperatorKind.INDEX: | |
| 302 if (node.isPrefix) { | |
| 303 kind = SendStructureKind.INDEX_PREFIX; | |
| 304 } else if (node.isPostfix) { | |
| 305 kind = SendStructureKind.INDEX_POSTFIX; | |
| 306 } else if (node.arguments.tail.isEmpty) { | |
| 307 // a[b] | |
| 308 kind = SendStructureKind.INDEX; | |
| 309 return internalError(node, "Unexpected binary $kind."); | |
| 310 } else { | |
| 311 if (kind == SendStructureKind.COMPOUND) { | |
| 312 // a[b] += c | |
| 313 kind = SendStructureKind.COMPOUND_INDEX_SET; | |
| 314 } else { | |
| 315 // a[b] = c | |
| 316 kind = SendStructureKind.INDEX_SET; | |
| 317 } | |
| 318 } | |
| 319 break; | |
| 320 default: | |
| 321 kind = SendStructureKind.BINARY; | |
| 322 return internalError(node, "Unexpected binary $kind."); | |
| 323 } | |
| 324 } else { | |
| 325 return internalError( | |
| 326 node, "Unexpected invalid binary $operatorText."); | |
| 327 } | |
| 328 } | |
| 329 } | |
| 330 AccessSemantics semantics = computeAccessSemantics( | |
| 331 node, | |
| 332 isSet: kind == SendStructureKind.SET, | |
| 333 isInvoke: kind == SendStructureKind.INVOKE, | |
| 334 isCompound: kind == SendStructureKind.COMPOUND || | |
| 335 kind == SendStructureKind.COMPOUND_INDEX_SET || | |
| 336 kind == SendStructureKind.PREFIX || | |
| 337 kind == SendStructureKind.POSTFIX || | |
| 338 kind == SendStructureKind.INDEX_PREFIX || | |
| 339 kind == SendStructureKind.INDEX_POSTFIX); | |
| 340 if (semantics == null) { | |
| 341 return internalError(node, 'No semantics for $node'); | |
| 342 } | |
| 343 Selector selector = elements.getSelector(node); | |
| 344 switch (kind) { | |
| 345 case SendStructureKind.GET: | |
| 346 return new GetStructure(semantics, selector); | |
| 347 case SendStructureKind.SET: | |
| 348 return new SetStructure(semantics, selector); | |
| 349 case SendStructureKind.INVOKE: | |
| 350 switch (semantics.kind) { | |
| 351 case AccessKind.STATIC_METHOD: | |
| 352 case AccessKind.SUPER_METHOD: | |
| 353 case AccessKind.TOPLEVEL_METHOD: | |
| 354 // TODO(johnniwinther): Should local function also be handled here? | |
| 355 FunctionElement function = semantics.element; | |
| 356 FunctionSignature signature = function.functionSignature; | |
| 357 if (!selector.callStructure.signatureApplies(signature)) { | |
| 358 return new IncompatibleInvokeStructure(semantics, selector); | |
| 359 } | |
| 360 break; | |
| 361 default: | |
| 362 break; | |
| 363 } | |
| 364 return new InvokeStructure(semantics, selector); | |
| 365 case SendStructureKind.UNARY: | |
| 366 return internalError(node, "Unexpected unary."); | |
| 367 case SendStructureKind.NOT: | |
| 368 return internalError(node, "Unexpected not."); | |
| 369 case SendStructureKind.BINARY: | |
| 370 return internalError(node, "Unexpected binary."); | |
| 371 case SendStructureKind.INDEX: | |
| 372 return internalError(node, "Unexpected index."); | |
| 373 case SendStructureKind.EQ: | |
| 374 return internalError(node, "Unexpected equals."); | |
| 375 case SendStructureKind.NOT_EQ: | |
| 376 return internalError(node, "Unexpected not equals."); | |
| 377 case SendStructureKind.COMPOUND: | |
| 378 Selector getterSelector = | |
| 379 elements.getGetterSelectorInComplexSendSet(node); | |
| 380 return new CompoundStructure( | |
| 381 semantics, | |
| 382 assignmentOperator, | |
| 383 getterSelector, | |
| 384 selector); | |
| 385 case SendStructureKind.INDEX_SET: | |
| 386 return new IndexSetStructure(semantics, selector); | |
| 387 case SendStructureKind.COMPOUND_INDEX_SET: | |
| 388 Selector getterSelector = | |
| 389 elements.getGetterSelectorInComplexSendSet(node); | |
| 390 return new CompoundIndexSetStructure( | |
| 391 semantics, | |
| 392 assignmentOperator, | |
| 393 getterSelector, | |
| 394 selector); | |
| 395 case SendStructureKind.INDEX_PREFIX: | |
| 396 Selector getterSelector = | |
| 397 elements.getGetterSelectorInComplexSendSet(node); | |
| 398 return new IndexPrefixStructure( | |
| 399 semantics, | |
| 400 incDecOperator, | |
| 401 getterSelector, | |
| 402 selector); | |
| 403 case SendStructureKind.INDEX_POSTFIX: | |
| 404 Selector getterSelector = | |
| 405 elements.getGetterSelectorInComplexSendSet(node); | |
| 406 return new IndexPostfixStructure( | |
| 407 semantics, | |
| 408 incDecOperator, | |
| 409 getterSelector, | |
| 410 selector); | |
| 411 case SendStructureKind.PREFIX: | |
| 412 Selector getterSelector = | |
| 413 elements.getGetterSelectorInComplexSendSet(node); | |
| 414 return new PrefixStructure( | |
| 415 semantics, | |
| 416 incDecOperator, | |
| 417 getterSelector, | |
| 418 selector); | |
| 419 case SendStructureKind.POSTFIX: | |
| 420 Selector getterSelector = | |
| 421 elements.getGetterSelectorInComplexSendSet(node); | |
| 422 return new PostfixStructure( | |
| 423 semantics, | |
| 424 incDecOperator, | |
| 425 getterSelector, | |
| 426 selector); | |
| 427 } | |
| 428 } | |
| 429 | |
| 430 AccessSemantics computeAccessSemantics(Send node, | |
| 431 {bool isSet: false, | |
| 432 bool isInvoke: false, | |
| 433 bool isCompound: false}) { | |
| 434 Element element = elements[node]; | |
| 435 Element getter = isCompound ? elements[node.selector] : null; | |
| 436 if (elements.isTypeLiteral(node)) { | |
| 437 DartType dartType = elements.getTypeLiteralType(node); | |
| 438 // TODO(johnniwinther): Handle deferred constants. There are runtime | |
| 439 // but not compile-time constants and should have their own | |
| 440 // [DeferredConstantExpression] class. | |
| 441 ConstantExpression constant = elements.getConstant( | |
| 442 isInvoke || isSet || isCompound ? node.selector : node); | |
| 443 switch (dartType.kind) { | |
| 444 case TypeKind.INTERFACE: | |
| 445 return new ConstantAccess.classTypeLiteral(constant); | |
| 446 case TypeKind.TYPEDEF: | |
| 447 return new ConstantAccess.typedefTypeLiteral(constant); | |
| 448 case TypeKind.TYPE_VARIABLE: | |
| 449 return new StaticAccess.typeParameterTypeLiteral(dartType.element); | |
| 450 case TypeKind.DYNAMIC: | |
| 451 return new ConstantAccess.dynamicTypeLiteral(constant); | |
| 452 default: | |
| 453 return internalError(node, "Unexpected type literal type: $dartType"); | |
| 454 } | |
| 455 } else if (node.isSuperCall) { | |
| 456 if (Elements.isUnresolved(element)) { | |
| 457 if (isCompound) { | |
| 458 if (Elements.isUnresolved(getter)) { | |
| 459 // TODO(johnniwinther): Ensure that [getter] is not null. This | |
| 460 // happens in the case of missing super getter. | |
| 461 return new StaticAccess.unresolvedSuper(element); | |
| 462 } else if (getter.isField) { | |
| 463 assert(invariant(node, getter.isFinal, | |
| 464 message: "Super field expected to be final.")); | |
| 465 return new StaticAccess.superFinalField(getter); | |
| 466 } else if (getter.isFunction) { | |
| 467 if (node.isIndex) { | |
| 468 return new CompoundAccessSemantics( | |
| 469 CompoundAccessKind.UNRESOLVED_SUPER_SETTER, getter, element); | |
| 470 } else { | |
| 471 return new StaticAccess.superMethod(getter); | |
| 472 } | |
| 473 } else { | |
| 474 return new CompoundAccessSemantics( | |
| 475 CompoundAccessKind.UNRESOLVED_SUPER_SETTER, getter, element); | |
| 476 } | |
| 477 } else { | |
| 478 return new StaticAccess.unresolvedSuper(element); | |
| 479 } | |
| 480 } else if (isCompound && Elements.isUnresolved(getter)) { | |
| 481 // TODO(johnniwinther): Ensure that [getter] is not null. This happens | |
| 482 // in the case of missing super getter. | |
| 483 return new CompoundAccessSemantics( | |
| 484 CompoundAccessKind.UNRESOLVED_SUPER_GETTER, getter, element); | |
| 485 } else if (element.isField) { | |
| 486 if (getter != null && getter != element) { | |
| 487 CompoundAccessKind accessKind; | |
| 488 if (getter.isField) { | |
| 489 accessKind = CompoundAccessKind.SUPER_FIELD_FIELD; | |
| 490 } else if (getter.isGetter) { | |
| 491 accessKind = CompoundAccessKind.SUPER_GETTER_FIELD; | |
| 492 } else { | |
| 493 return internalError(node, | |
| 494 "Unsupported super call: $node : $element/$getter."); | |
| 495 } | |
| 496 return new CompoundAccessSemantics(accessKind, getter, element); | |
| 497 } else if (element.isFinal) { | |
| 498 return new StaticAccess.superFinalField(element); | |
| 499 } | |
| 500 return new StaticAccess.superField(element); | |
| 501 } else if (element.isGetter) { | |
| 502 return new StaticAccess.superGetter(element); | |
| 503 } else if (element.isSetter) { | |
| 504 if (getter != null) { | |
| 505 CompoundAccessKind accessKind; | |
| 506 if (getter.isField) { | |
| 507 accessKind = CompoundAccessKind.SUPER_FIELD_SETTER; | |
| 508 } else if (getter.isGetter) { | |
| 509 accessKind = CompoundAccessKind.SUPER_GETTER_SETTER; | |
| 510 } else { | |
| 511 accessKind = CompoundAccessKind.SUPER_METHOD_SETTER; | |
| 512 } | |
| 513 return new CompoundAccessSemantics(accessKind, getter, element); | |
| 514 } | |
| 515 return new StaticAccess.superSetter(element); | |
| 516 } else if (isCompound) { | |
| 517 return new CompoundAccessSemantics( | |
| 518 CompoundAccessKind.SUPER_GETTER_SETTER, getter, element); | |
| 519 } else { | |
| 520 return new StaticAccess.superMethod(element); | |
| 521 } | |
| 522 } else if (node.isConditional) { | |
| 523 // Conditional sends (e?.x) are treated as dynamic property reads because | |
| 524 // they are equivalent to do ((a) => a == null ? null : a.x)(e). If `e` is | |
| 525 // a type `A`, this is equivalent to write `(A).x`. | |
| 526 return const DynamicAccess.ifNotNullProperty(); | |
| 527 } else if (node.isOperator) { | |
| 528 return const DynamicAccess.dynamicProperty(); | |
| 529 } else if (Elements.isClosureSend(node, element)) { | |
| 530 if (element == null) { | |
| 531 if (node.selector.isThis()) { | |
| 532 return new DynamicAccess.thisAccess(); | |
| 533 } else { | |
| 534 return new DynamicAccess.expression(); | |
| 535 } | |
| 536 } else if (Elements.isErroneous(element)) { | |
| 537 return new StaticAccess.unresolved(element); | |
| 538 } else { | |
| 539 return handleStaticallyResolvedAccess( | |
| 540 node, element, getter, isCompound: isCompound); | |
| 541 } | |
| 542 } else { | |
| 543 bool isDynamicAccess(Element e) => e == null || e.isInstanceMember; | |
| 544 | |
| 545 if (isDynamicAccess(element) && | |
| 546 (!isCompound || isDynamicAccess(getter))) { | |
| 547 if (node.receiver == null || node.receiver.isThis()) { | |
| 548 return const DynamicAccess.thisProperty(); | |
| 549 } else { | |
| 550 return const DynamicAccess.dynamicProperty(); | |
| 551 } | |
| 552 } else if (element != null && element.impliesType) { | |
| 553 // TODO(johnniwinther): Provide an [ErroneousElement]. | |
| 554 // This happens for code like `C.this`. | |
| 555 return new StaticAccess.unresolved(null); | |
| 556 } else { | |
| 557 return handleStaticallyResolvedAccess( | |
| 558 node, element, getter, isCompound: isCompound); | |
| 559 } | |
| 560 } | |
| 561 } | |
| 562 | |
| 563 ConstructorAccessSemantics computeConstructorAccessSemantics( | 28 ConstructorAccessSemantics computeConstructorAccessSemantics( |
| 564 ConstructorElement constructor, | 29 ConstructorElement constructor, |
| 565 CallStructure callStructure, | 30 CallStructure callStructure, |
| 566 DartType type, | 31 DartType type, |
| 567 {bool mustBeConstant: false}) { | 32 {bool mustBeConstant: false}) { |
| 568 if (mustBeConstant && !constructor.isConst) { | 33 if (mustBeConstant && !constructor.isConst) { |
| 569 return new ConstructorAccessSemantics( | 34 return new ConstructorAccessSemantics( |
| 570 ConstructorAccessKind.NON_CONSTANT_CONSTRUCTOR, constructor, type); | 35 ConstructorAccessKind.NON_CONSTANT_CONSTRUCTOR, constructor, type); |
| 571 } | 36 } |
| 572 if (constructor.isErroneous) { | 37 if (constructor.isErroneous) { |
| (...skipping 431 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1004 return internalError(node, "Unexpected variable $element."); | 469 return internalError(node, "Unexpected variable $element."); |
| 1005 } | 470 } |
| 1006 if (element.isConst) { | 471 if (element.isConst) { |
| 1007 ConstantExpression constant = elements.getConstant(element.initializer); | 472 ConstantExpression constant = elements.getConstant(element.initializer); |
| 1008 return new ConstantVariableStructure(kind, node, element, constant); | 473 return new ConstantVariableStructure(kind, node, element, constant); |
| 1009 } else { | 474 } else { |
| 1010 return new NonConstantVariableStructure(kind, node, element); | 475 return new NonConstantVariableStructure(kind, node, element); |
| 1011 } | 476 } |
| 1012 } | 477 } |
| 1013 } | 478 } |
| OLD | NEW |