| 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 library native; | 5 library native; |
| 6 | 6 |
| 7 import 'dart:uri'; | 7 import 'dart:uri'; |
| 8 import 'dart2jslib.dart' hide SourceString; | 8 import 'dart2jslib.dart' hide SourceString; |
| 9 import 'elements/elements.dart'; | 9 import 'elements/elements.dart'; |
| 10 import 'js_backend/js_backend.dart'; | 10 import 'js_backend/js_backend.dart'; |
| (...skipping 15 matching lines...) Expand all Loading... |
| 26 /// The specific implementation of List that is JavaScript Array: | 26 /// The specific implementation of List that is JavaScript Array: |
| 27 static const JsArray = const SpecialType._('=List'); | 27 static const JsArray = const SpecialType._('=List'); |
| 28 } | 28 } |
| 29 | 29 |
| 30 | 30 |
| 31 /** | 31 /** |
| 32 * This could be an abstract class but we use it as a stub for the dart_backend. | 32 * This could be an abstract class but we use it as a stub for the dart_backend. |
| 33 */ | 33 */ |
| 34 class NativeEnqueuer { | 34 class NativeEnqueuer { |
| 35 /// Initial entry point to native enqueuer. | 35 /// Initial entry point to native enqueuer. |
| 36 void processNativeClasses(Collection<LibraryElement> libraries) {} | 36 void processNativeClasses(Iterable<LibraryElement> libraries) {} |
| 37 | 37 |
| 38 /// Notification of a main Enqueuer worklist element. For methods, adds | 38 /// Notification of a main Enqueuer worklist element. For methods, adds |
| 39 /// information from metadata attributes, and computes types instantiated due | 39 /// information from metadata attributes, and computes types instantiated due |
| 40 /// to calling the method. | 40 /// to calling the method. |
| 41 void registerElement(Element element) {} | 41 void registerElement(Element element) {} |
| 42 | 42 |
| 43 /// Notification of native field. Adds information from metadata attributes. | 43 /// Notification of native field. Adds information from metadata attributes. |
| 44 void handleFieldAnnotations(Element field) {} | 44 void handleFieldAnnotations(Element field) {} |
| 45 | 45 |
| 46 /// Computes types instantiated due to getting a native field. | 46 /// Computes types instantiated due to getting a native field. |
| (...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 94 final Compiler compiler; | 94 final Compiler compiler; |
| 95 final bool enableLiveTypeAnalysis; | 95 final bool enableLiveTypeAnalysis; |
| 96 | 96 |
| 97 ClassElement _annotationCreatesClass; | 97 ClassElement _annotationCreatesClass; |
| 98 ClassElement _annotationReturnsClass; | 98 ClassElement _annotationReturnsClass; |
| 99 ClassElement _annotationJsNameClass; | 99 ClassElement _annotationJsNameClass; |
| 100 | 100 |
| 101 /// Subclasses of [NativeEnqueuerBase] are constructed by the backend. | 101 /// Subclasses of [NativeEnqueuerBase] are constructed by the backend. |
| 102 NativeEnqueuerBase(this.world, this.compiler, this.enableLiveTypeAnalysis); | 102 NativeEnqueuerBase(this.world, this.compiler, this.enableLiveTypeAnalysis); |
| 103 | 103 |
| 104 void processNativeClasses(Collection<LibraryElement> libraries) { | 104 void processNativeClasses(Iterable<LibraryElement> libraries) { |
| 105 libraries.forEach(processNativeClassesInLibrary); | 105 libraries.forEach(processNativeClassesInLibrary); |
| 106 processNativeClassesInLibrary(compiler.isolateHelperLibrary); | 106 processNativeClassesInLibrary(compiler.isolateHelperLibrary); |
| 107 if (!enableLiveTypeAnalysis) { | 107 if (!enableLiveTypeAnalysis) { |
| 108 nativeClasses.forEach((c) => enqueueClass(c, 'forced')); | 108 nativeClasses.forEach((c) => enqueueClass(c, 'forced')); |
| 109 flushQueue(); | 109 flushQueue(); |
| 110 } | 110 } |
| 111 } | 111 } |
| 112 | 112 |
| 113 void processNativeClassesInLibrary(LibraryElement library) { | 113 void processNativeClassesInLibrary(LibraryElement library) { |
| 114 // Use implementation to ensure the inclusion of injected members. | 114 // Use implementation to ensure the inclusion of injected members. |
| (...skipping 228 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 343 // Give an info so that library developers can compile with -v to find why | 343 // Give an info so that library developers can compile with -v to find why |
| 344 // all the native classes are included. | 344 // all the native classes are included. |
| 345 if (unusedClasses.isEmpty && !allUsedBefore) { | 345 if (unusedClasses.isEmpty && !allUsedBefore) { |
| 346 compiler.log('All native types marked as used due to $cause.'); | 346 compiler.log('All native types marked as used due to $cause.'); |
| 347 } | 347 } |
| 348 } | 348 } |
| 349 | 349 |
| 350 enqueueUnusedClassesMatching(bool predicate(classElement), | 350 enqueueUnusedClassesMatching(bool predicate(classElement), |
| 351 cause, | 351 cause, |
| 352 [String reason]) { | 352 [String reason]) { |
| 353 Collection matches = unusedClasses.filter(predicate); | 353 Iterable matches = unusedClasses.where(predicate); |
| 354 matches.forEach((c) => enqueueClass(c, cause)); | 354 matches.forEach((c) => enqueueClass(c, cause)); |
| 355 } | 355 } |
| 356 | 356 |
| 357 onFirstNativeClass() { | 357 onFirstNativeClass() { |
| 358 staticUse(name) => world.registerStaticUse(compiler.findHelper(name)); | 358 staticUse(name) => world.registerStaticUse(compiler.findHelper(name)); |
| 359 | 359 |
| 360 staticUse(const SourceString('dynamicFunction')); | 360 staticUse(const SourceString('dynamicFunction')); |
| 361 staticUse(const SourceString('dynamicSetMetadata')); | 361 staticUse(const SourceString('dynamicSetMetadata')); |
| 362 staticUse(const SourceString('defineProperty')); | 362 staticUse(const SourceString('defineProperty')); |
| 363 staticUse(const SourceString('toStringForNativeObject')); | 363 staticUse(const SourceString('toStringForNativeObject')); |
| (...skipping 29 matching lines...) Expand all Loading... |
| 393 | 393 |
| 394 class NativeCodegenEnqueuer extends NativeEnqueuerBase { | 394 class NativeCodegenEnqueuer extends NativeEnqueuerBase { |
| 395 | 395 |
| 396 final CodeEmitterTask emitter; | 396 final CodeEmitterTask emitter; |
| 397 | 397 |
| 398 final Set<ClassElement> doneAddSubtypes = new Set<ClassElement>(); | 398 final Set<ClassElement> doneAddSubtypes = new Set<ClassElement>(); |
| 399 | 399 |
| 400 NativeCodegenEnqueuer(Enqueuer world, Compiler compiler, this.emitter) | 400 NativeCodegenEnqueuer(Enqueuer world, Compiler compiler, this.emitter) |
| 401 : super(world, compiler, compiler.enableNativeLiveTypeAnalysis); | 401 : super(world, compiler, compiler.enableNativeLiveTypeAnalysis); |
| 402 | 402 |
| 403 void processNativeClasses(Collection<LibraryElement> libraries) { | 403 void processNativeClasses(Iterable<LibraryElement> libraries) { |
| 404 super.processNativeClasses(libraries); | 404 super.processNativeClasses(libraries); |
| 405 | 405 |
| 406 // HACK HACK - add all the resolved classes. | 406 // HACK HACK - add all the resolved classes. |
| 407 NativeEnqueuerBase enqueuer = compiler.enqueuer.resolution.nativeEnqueuer; | 407 NativeEnqueuerBase enqueuer = compiler.enqueuer.resolution.nativeEnqueuer; |
| 408 for (final classElement in enqueuer.registeredClasses) { | 408 for (final classElement in enqueuer.registeredClasses) { |
| 409 if (unusedClasses.contains(classElement)) { | 409 if (unusedClasses.contains(classElement)) { |
| 410 enqueueClass(classElement, 'was resolved'); | 410 enqueueClass(classElement, 'was resolved'); |
| 411 } | 411 } |
| 412 } | 412 } |
| 413 flushQueue(); | 413 flushQueue(); |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 447 '${unusedClasses.length} native classes omitted.'); | 447 '${unusedClasses.length} native classes omitted.'); |
| 448 } | 448 } |
| 449 } | 449 } |
| 450 | 450 |
| 451 void maybeEnableNative(Compiler compiler, | 451 void maybeEnableNative(Compiler compiler, |
| 452 LibraryElement library, | 452 LibraryElement library, |
| 453 Uri uri) { | 453 Uri uri) { |
| 454 String libraryName = uri.toString(); | 454 String libraryName = uri.toString(); |
| 455 if (library.entryCompilationUnit.script.name.contains( | 455 if (library.entryCompilationUnit.script.name.contains( |
| 456 'dart/tests/compiler/dart2js_native') | 456 'dart/tests/compiler/dart2js_native') |
| 457 || libraryName == 'dart:async' |
| 457 || libraryName == 'dart:html' | 458 || libraryName == 'dart:html' |
| 458 || libraryName == 'dart:html_common' | 459 || libraryName == 'dart:html_common' |
| 459 || libraryName == 'dart:indexed_db' | 460 || libraryName == 'dart:indexed_db' |
| 460 || libraryName == 'dart:svg' | 461 || libraryName == 'dart:svg' |
| 461 || libraryName == 'dart:web_audio') { | 462 || libraryName == 'dart:web_audio') { |
| 462 library.canUseNative = true; | 463 library.canUseNative = true; |
| 463 } | 464 } |
| 464 } | 465 } |
| 465 | 466 |
| 466 /** | 467 /** |
| (...skipping 407 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 874 } else { | 875 } else { |
| 875 if (parameters.parameterCount != 0) { | 876 if (parameters.parameterCount != 0) { |
| 876 compiler.cancel( | 877 compiler.cancel( |
| 877 'native "..." syntax is restricted to functions with zero parameters', | 878 'native "..." syntax is restricted to functions with zero parameters', |
| 878 node: nativeBody); | 879 node: nativeBody); |
| 879 } | 880 } |
| 880 LiteralString jsCode = nativeBody.asLiteralString(); | 881 LiteralString jsCode = nativeBody.asLiteralString(); |
| 881 builder.push(new HForeign.statement(jsCode.dartString, <HInstruction>[])); | 882 builder.push(new HForeign.statement(jsCode.dartString, <HInstruction>[])); |
| 882 } | 883 } |
| 883 } | 884 } |
| OLD | NEW |