| OLD | NEW |
| 1 // Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2017, 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 import 'dart:async'; | 5 import 'dart:async'; |
| 6 | 6 |
| 7 import 'package:front_end/file_system.dart'; | 7 import 'package:front_end/file_system.dart'; |
| 8 import 'package:front_end/incremental_kernel_generator.dart'; | |
| 9 import 'package:front_end/src/base/api_signature.dart'; | 8 import 'package:front_end/src/base/api_signature.dart'; |
| 10 import 'package:front_end/src/base/performace_logger.dart'; | 9 import 'package:front_end/src/base/performace_logger.dart'; |
| 11 import 'package:front_end/src/base/processed_options.dart'; | |
| 12 import 'package:front_end/src/fasta/dill/dill_library_builder.dart'; | 10 import 'package:front_end/src/fasta/dill/dill_library_builder.dart'; |
| 13 import 'package:front_end/src/fasta/dill/dill_target.dart'; | 11 import 'package:front_end/src/fasta/dill/dill_target.dart'; |
| 14 import 'package:front_end/src/fasta/kernel/kernel_target.dart'; | 12 import 'package:front_end/src/fasta/kernel/kernel_target.dart'; |
| 15 import 'package:front_end/src/fasta/kernel/utils.dart'; | 13 import 'package:front_end/src/fasta/kernel/utils.dart'; |
| 16 import 'package:front_end/src/fasta/ticker.dart'; | 14 import 'package:front_end/src/fasta/ticker.dart'; |
| 17 import 'package:front_end/src/fasta/translate_uri.dart'; | 15 import 'package:front_end/src/fasta/translate_uri.dart'; |
| 18 import 'package:front_end/src/incremental/byte_store.dart'; | 16 import 'package:front_end/src/incremental/byte_store.dart'; |
| 19 import 'package:front_end/src/incremental/file_state.dart'; | 17 import 'package:front_end/src/incremental/file_state.dart'; |
| 20 import 'package:kernel/binary/ast_from_binary.dart'; | 18 import 'package:kernel/binary/ast_from_binary.dart'; |
| 21 import 'package:kernel/kernel.dart' hide Source; | 19 import 'package:kernel/kernel.dart' hide Source; |
| 22 import 'package:kernel/target/targets.dart' show TargetFlags; | 20 import 'package:kernel/target/targets.dart' show TargetFlags; |
| 23 import 'package:kernel/target/vm_fasta.dart' show VmFastaTarget; | 21 import 'package:kernel/target/vm_fasta.dart' show VmFastaTarget; |
| 24 import 'package:meta/meta.dart'; | 22 import 'package:meta/meta.dart'; |
| 25 | 23 |
| 26 /// Implementation of [IncrementalKernelGenerator]. | 24 /// This function is invoked for each newly discovered file, and the returned |
| 25 /// [Future] is awaited before reading the file content. |
| 26 typedef Future<Null> KernelDriverFileAddedFn(Uri uri); |
| 27 |
| 28 /// This class computes [KernelResult]s for Dart files. |
| 27 /// | 29 /// |
| 28 /// TODO(scheglov) Update the documentation. | 30 /// Let the "current file state" represent a map from file URI to the file |
| 31 /// contents most recently read from that file. When the driver needs to |
| 32 /// access a file that is not in the current file state yet, it will call |
| 33 /// the optional "file added" function, read the file and put it into the |
| 34 /// current file state. |
| 29 /// | 35 /// |
| 30 /// Theory of operation: an instance of [IncrementalResolvedAstGenerator] is | 36 /// The client invokes [getKernel] to schedule computing the [KernelResult] |
| 31 /// used to obtain resolved ASTs, and these are fed into kernel code generation | 37 /// for a Dart file. The driver will eventually use the current file state |
| 32 /// logic. | 38 /// of the specified file and all files that it transitively depends on to |
| 33 class IncrementalKernelGeneratorImpl implements IncrementalKernelGenerator { | 39 /// compute corresponding kernel files (or read them from the [ByteStore]). |
| 40 /// |
| 41 /// A call to [invalidate] removes the specified file from the current file |
| 42 /// state, so that it will be reread before any following [getKernel] will |
| 43 /// return a result. |
| 44 class KernelDriver { |
| 34 /// The version of data format, should be incremented on every format change. | 45 /// The version of data format, should be incremented on every format change. |
| 35 static const int DATA_VERSION = 1; | 46 static const int DATA_VERSION = 1; |
| 36 | 47 |
| 37 /// The compiler options, such as the [FileSystem], the SDK dill location, | 48 /// The logger to report compilation progress. |
| 38 /// etc. | 49 final PerformanceLog _logger; |
| 39 final ProcessedOptions _options; | 50 |
| 51 /// The [FileSystem] which should be used by the front end to access files. |
| 52 final FileSystem _fileSystem; |
| 53 |
| 54 /// The byte storage to get and put serialized data. |
| 55 final ByteStore _byteStore; |
| 40 | 56 |
| 41 /// The object that knows how to resolve "package:" and "dart:" URIs. | 57 /// The object that knows how to resolve "package:" and "dart:" URIs. |
| 42 final TranslateUri _uriTranslator; | 58 final TranslateUri _uriTranslator; |
| 43 | 59 |
| 44 /// The logger to report compilation progress. | 60 /// Is `true` if strong mode analysis should be used. |
| 45 final PerformanceLog _logger; | 61 final bool _strongMode; |
| 46 | 62 |
| 47 /// The byte storage to get and put serialized data. | 63 /// The function that is invoked when a new file is about to be added to |
| 48 final ByteStore _byteStore; | 64 /// the current file state. The [Future] that it returns is awaited before |
| 49 | 65 /// reading the file contents. |
| 50 /// The URI of the program entry point. | 66 final KernelDriverFileAddedFn _fileAddedFn; |
| 51 final Uri _entryPoint; | |
| 52 | |
| 53 /// The function to notify when files become used or unused, or `null`. | |
| 54 final WatchUsedFilesFn _watchFn; | |
| 55 | 67 |
| 56 /// The salt to mix into all hashes used as keys for serialized data. | 68 /// The salt to mix into all hashes used as keys for serialized data. |
| 57 List<int> _salt; | 69 List<int> _salt; |
| 58 | 70 |
| 59 /// The current file system state. | 71 /// The current file system state. |
| 60 FileSystemState _fsState; | 72 FileSystemState _fsState; |
| 61 | 73 |
| 62 /// Latest compilation signatures produced by [computeDelta] for libraries. | |
| 63 final Map<Uri, String> _latestSignature = {}; | |
| 64 | |
| 65 /// The set of absolute file URIs that were reported through [invalidate] | 74 /// The set of absolute file URIs that were reported through [invalidate] |
| 66 /// and not checked for actual changes yet. | 75 /// and not checked for actual changes yet. |
| 67 final Set<Uri> _invalidatedFiles = new Set<Uri>(); | 76 final Set<Uri> _invalidatedFiles = new Set<Uri>(); |
| 68 | 77 |
| 69 /// The object that provides additional information for tests. | 78 /// The object that provides additional information for tests. |
| 70 final _TestView _testView = new _TestView(); | 79 final _TestView _testView = new _TestView(); |
| 71 | 80 |
| 72 IncrementalKernelGeneratorImpl( | 81 KernelDriver(this._logger, this._fileSystem, this._byteStore, |
| 73 this._options, this._uriTranslator, this._entryPoint, | 82 this._uriTranslator, this._strongMode, |
| 74 {WatchUsedFilesFn watch}) | 83 {KernelDriverFileAddedFn fileAddedFn}) |
| 75 : _logger = _options.logger, | 84 : _fileAddedFn = fileAddedFn { |
| 76 _byteStore = _options.byteStore, | |
| 77 _watchFn = watch { | |
| 78 _computeSalt(); | 85 _computeSalt(); |
| 79 | 86 |
| 80 Future<Null> onFileAdded(Uri uri) { | 87 Future<Null> onFileAdded(Uri uri) { |
| 81 if (_watchFn != null) { | 88 if (_fileAddedFn != null) { |
| 82 return _watchFn(uri, true); | 89 return _fileAddedFn(uri); |
| 83 } | 90 } |
| 84 return new Future.value(); | 91 return new Future.value(); |
| 85 } | 92 } |
| 86 | 93 |
| 87 _fsState = new FileSystemState(_options.byteStore, _options.fileSystem, | 94 _fsState = new FileSystemState( |
| 88 _uriTranslator, _salt, onFileAdded); | 95 _byteStore, _fileSystem, _uriTranslator, _salt, onFileAdded); |
| 89 } | 96 } |
| 90 | 97 |
| 98 /// Return the [FileSystemState] that contains the current file state. |
| 99 FileSystemState get fsState => _fsState; |
| 100 |
| 91 /// Return the object that provides additional information for tests. | 101 /// Return the object that provides additional information for tests. |
| 92 @visibleForTesting | 102 @visibleForTesting |
| 93 _TestView get test => _testView; | 103 _TestView get test => _testView; |
| 94 | 104 |
| 95 @override | 105 /// Return the [KernelResult] for the Dart file with the given [uri]. |
| 96 Future<DeltaProgram> computeDelta() async { | 106 /// |
| 107 /// The [uri] must be absolute and normalized. |
| 108 /// |
| 109 /// The driver will update the current file state for any file previously |
| 110 /// reported using [invalidate]. |
| 111 /// |
| 112 /// If the driver has the cached result for the file with the current file |
| 113 /// state, it is returned. |
| 114 /// |
| 115 /// Otherwise the driver will compute new kernel files and return them. |
| 116 Future<KernelResult> getKernel(Uri uri) async { |
| 97 return await _logger.runAsync('Compute delta', () async { | 117 return await _logger.runAsync('Compute delta', () async { |
| 98 await _refreshInvalidatedFiles(); | 118 await _refreshInvalidatedFiles(); |
| 99 | 119 |
| 100 // Ensure that the graph starting at the entry point is ready. | 120 // Ensure that the graph starting at the entry point is ready. |
| 101 FileState entryLibrary = | 121 FileState entryLibrary = |
| 102 await _logger.runAsync('Build graph of files', () async { | 122 await _logger.runAsync('Build graph of files', () async { |
| 103 return await _fsState.getFile(_entryPoint); | 123 return await _fsState.getFile(uri); |
| 104 }); | 124 }); |
| 105 | 125 |
| 106 List<LibraryCycle> cycles = _logger.run('Compute library cycles', () { | 126 List<LibraryCycle> cycles = _logger.run('Compute library cycles', () { |
| 107 List<LibraryCycle> cycles = entryLibrary.topologicalOrder; | 127 List<LibraryCycle> cycles = entryLibrary.topologicalOrder; |
| 108 _logger.writeln('Computed ${cycles.length} cycles.'); | 128 _logger.writeln('Computed ${cycles.length} cycles.'); |
| 109 return cycles; | 129 return cycles; |
| 110 }); | 130 }); |
| 111 | 131 |
| 112 CanonicalName nameRoot = new CanonicalName.root(); | 132 CanonicalName nameRoot = new CanonicalName.root(); |
| 113 DillTarget dillTarget = new DillTarget( | 133 DillTarget dillTarget = new DillTarget( |
| 114 new Ticker(isVerbose: false), | 134 new Ticker(isVerbose: false), |
| 115 _uriTranslator, | 135 _uriTranslator, |
| 116 new VmFastaTarget(new TargetFlags(strongMode: _options.strongMode))); | 136 new VmFastaTarget(new TargetFlags(strongMode: _strongMode))); |
| 117 | 137 |
| 118 List<_LibraryCycleResult> results = []; | 138 List<LibraryCycleResult> results = []; |
| 119 _testView.compiledCycles.clear(); | 139 _testView.compiledCycles.clear(); |
| 120 await _logger.runAsync('Compute results for cycles', () async { | 140 await _logger.runAsync('Compute results for cycles', () async { |
| 121 for (LibraryCycle cycle in cycles) { | 141 for (LibraryCycle cycle in cycles) { |
| 122 _LibraryCycleResult result = | 142 LibraryCycleResult result = |
| 123 await _compileCycle(nameRoot, dillTarget, cycle); | 143 await _compileCycle(nameRoot, dillTarget, cycle); |
| 124 results.add(result); | 144 results.add(result); |
| 125 } | 145 } |
| 126 }); | 146 }); |
| 127 | 147 |
| 128 Program program = new Program(nameRoot: nameRoot); | 148 return new KernelResult(nameRoot, results); |
| 129 | |
| 130 // The set of affected library cycles (have different signatures). | |
| 131 final affectedLibraryCycles = new Set<LibraryCycle>(); | |
| 132 for (_LibraryCycleResult result in results) { | |
| 133 for (Library library in result.kernelLibraries) { | |
| 134 Uri uri = library.importUri; | |
| 135 if (_latestSignature[uri] != result.signature) { | |
| 136 _latestSignature[uri] = result.signature; | |
| 137 affectedLibraryCycles.add(result.cycle); | |
| 138 } | |
| 139 } | |
| 140 } | |
| 141 | |
| 142 // The set of affected library cycles (have different signatures), | |
| 143 // or libraries that import or export affected libraries (so VM might | |
| 144 // have inlined some code from affected libraries into them). | |
| 145 final vmRequiredLibraryCycles = new Set<LibraryCycle>(); | |
| 146 | |
| 147 void gatherVmRequiredLibraryCycles(LibraryCycle cycle) { | |
| 148 if (vmRequiredLibraryCycles.add(cycle)) { | |
| 149 cycle.directUsers.forEach(gatherVmRequiredLibraryCycles); | |
| 150 } | |
| 151 } | |
| 152 | |
| 153 affectedLibraryCycles.forEach(gatherVmRequiredLibraryCycles); | |
| 154 | |
| 155 // Add required libraries. | |
| 156 for (_LibraryCycleResult result in results) { | |
| 157 if (vmRequiredLibraryCycles.contains(result.cycle)) { | |
| 158 for (Library library in result.kernelLibraries) { | |
| 159 program.libraries.add(library); | |
| 160 library.parent = program; | |
| 161 } | |
| 162 } | |
| 163 } | |
| 164 | |
| 165 // Set the main method. | |
| 166 if (program.libraries.isNotEmpty) { | |
| 167 for (Library library in results.last.kernelLibraries) { | |
| 168 if (library.importUri == _entryPoint) { | |
| 169 program.mainMethod = library.procedures.firstWhere( | |
| 170 (procedure) => procedure.name.name == 'main', | |
| 171 orElse: () => null); | |
| 172 break; | |
| 173 } | |
| 174 } | |
| 175 } | |
| 176 | |
| 177 return new DeltaProgram(program); | |
| 178 }); | 149 }); |
| 179 } | 150 } |
| 180 | 151 |
| 181 @override | 152 /// The file with the given [uri] might have changed - updated, added, or |
| 153 /// removed. Or not, we don't know. Or it might have, but then changed back. |
| 154 /// |
| 155 /// The [uri] must be absolute and normalized file URI. |
| 156 /// |
| 157 /// Schedules the file contents for the [uri] to be read into the current |
| 158 /// file state prior the next invocation of [getKernel] returns the result. |
| 159 /// |
| 160 /// Invocation of this method will not prevent a [Future] returned from |
| 161 /// [getKernel] from completing with a result, but the result is not |
| 162 /// guaranteed to be consistent with the new current file state after this |
| 163 /// [invalidate] invocation. |
| 182 void invalidate(Uri uri) { | 164 void invalidate(Uri uri) { |
| 183 _invalidatedFiles.add(uri); | 165 _invalidatedFiles.add(uri); |
| 184 } | 166 } |
| 185 | 167 |
| 186 @override | 168 /// Flush the current file state completely. |
| 169 /// |
| 170 /// TODO(scheglov) Do we really need this functionality? |
| 187 void invalidateAll() { | 171 void invalidateAll() { |
| 188 _invalidatedFiles.addAll(_fsState.fileUris); | 172 _invalidatedFiles.addAll(_fsState.fileUris); |
| 189 } | 173 } |
| 190 | 174 |
| 191 /// Ensure that [dillTarget] includes the [cycle] libraries. It already | 175 /// Ensure that [dillTarget] includes the [cycle] libraries. It already |
| 192 /// contains all the libraries that sorted before the given [cycle] in | 176 /// contains all the libraries that sorted before the given [cycle] in |
| 193 /// topological order. Return the result with the cycle libraries. | 177 /// topological order. Return the result with the cycle libraries. |
| 194 Future<_LibraryCycleResult> _compileCycle( | 178 Future<LibraryCycleResult> _compileCycle( |
| 195 CanonicalName nameRoot, DillTarget dillTarget, LibraryCycle cycle) async { | 179 CanonicalName nameRoot, DillTarget dillTarget, LibraryCycle cycle) async { |
| 196 return _logger.runAsync('Compile cycle $cycle', () async { | 180 return _logger.runAsync('Compile cycle $cycle', () async { |
| 197 String signature = _getCycleSignature(cycle); | 181 String signature = _getCycleSignature(cycle); |
| 198 | 182 |
| 199 _logger.writeln('Signature: $signature.'); | 183 _logger.writeln('Signature: $signature.'); |
| 200 var kernelKey = '$signature.kernel'; | 184 var kernelKey = '$signature.kernel'; |
| 201 | 185 |
| 202 // We need kernel libraries for these URIs. | 186 // We need kernel libraries for these URIs. |
| 203 var libraryUris = new Set<Uri>(); | 187 var libraryUris = new Set<Uri>(); |
| 204 var libraryUriToFile = <Uri, FileState>{}; | 188 var libraryUriToFile = <Uri, FileState>{}; |
| (...skipping 17 matching lines...) Expand all Loading... |
| 222 // Check if there is already a bundle with these libraries. | 206 // Check if there is already a bundle with these libraries. |
| 223 List<int> bytes = _byteStore.get(kernelKey); | 207 List<int> bytes = _byteStore.get(kernelKey); |
| 224 if (bytes != null) { | 208 if (bytes != null) { |
| 225 return _logger.runAsync('Read serialized libraries', () async { | 209 return _logger.runAsync('Read serialized libraries', () async { |
| 226 var program = new Program(nameRoot: nameRoot); | 210 var program = new Program(nameRoot: nameRoot); |
| 227 var reader = new BinaryBuilder(bytes); | 211 var reader = new BinaryBuilder(bytes); |
| 228 reader.readProgram(program); | 212 reader.readProgram(program); |
| 229 | 213 |
| 230 await appendNewDillLibraries(program); | 214 await appendNewDillLibraries(program); |
| 231 | 215 |
| 232 return new _LibraryCycleResult(cycle, signature, program.libraries); | 216 return new LibraryCycleResult(cycle, signature, program.libraries); |
| 233 }); | 217 }); |
| 234 } | 218 } |
| 235 | 219 |
| 236 // Create KernelTarget and configure it for compiling the cycle URIs. | 220 // Create KernelTarget and configure it for compiling the cycle URIs. |
| 237 KernelTarget kernelTarget = | 221 KernelTarget kernelTarget = |
| 238 new KernelTarget(_fsState.fileSystemView, dillTarget, _uriTranslator); | 222 new KernelTarget(_fsState.fileSystemView, dillTarget, _uriTranslator); |
| 239 for (FileState library in cycle.libraries) { | 223 for (FileState library in cycle.libraries) { |
| 240 kernelTarget.read(library.uri); | 224 kernelTarget.read(library.uri); |
| 241 } | 225 } |
| 242 | 226 |
| (...skipping 13 matching lines...) Expand all Loading... |
| 256 .toList(); | 240 .toList(); |
| 257 | 241 |
| 258 _logger.run('Serialize ${kernelLibraries.length} libraries', () { | 242 _logger.run('Serialize ${kernelLibraries.length} libraries', () { |
| 259 program.uriToSource.clear(); | 243 program.uriToSource.clear(); |
| 260 List<int> bytes = | 244 List<int> bytes = |
| 261 serializeProgram(program, filter: kernelLibraries.contains); | 245 serializeProgram(program, filter: kernelLibraries.contains); |
| 262 _byteStore.put(kernelKey, bytes); | 246 _byteStore.put(kernelKey, bytes); |
| 263 _logger.writeln('Stored ${bytes.length} bytes.'); | 247 _logger.writeln('Stored ${bytes.length} bytes.'); |
| 264 }); | 248 }); |
| 265 | 249 |
| 266 return new _LibraryCycleResult(cycle, signature, kernelLibraries); | 250 return new LibraryCycleResult(cycle, signature, kernelLibraries); |
| 267 }); | 251 }); |
| 268 } | 252 } |
| 269 | 253 |
| 270 /// Compute exports scopes for a new strongly connected cycle of [libraries]. | 254 /// Compute exports scopes for a new strongly connected cycle of [libraries]. |
| 271 /// The [dillTarget] can be used to access libraries from previous cycles. | 255 /// The [dillTarget] can be used to access libraries from previous cycles. |
| 272 /// TODO(scheglov) Remove/replace this when Kernel has export scopes. | 256 /// TODO(scheglov) Remove/replace this when Kernel has export scopes. |
| 273 void _computeExportScopes(DillTarget dillTarget, | 257 void _computeExportScopes(DillTarget dillTarget, |
| 274 Map<Uri, FileState> uriToFile, List<DillLibraryBuilder> libraries) { | 258 Map<Uri, FileState> uriToFile, List<DillLibraryBuilder> libraries) { |
| 275 bool wasChanged = false; | 259 bool wasChanged = false; |
| 276 do { | 260 do { |
| (...skipping 15 matching lines...) Expand all Loading... |
| 292 } | 276 } |
| 293 } | 277 } |
| 294 } | 278 } |
| 295 } while (wasChanged); | 279 } while (wasChanged); |
| 296 } | 280 } |
| 297 | 281 |
| 298 /// Compute salt and put into [_salt]. | 282 /// Compute salt and put into [_salt]. |
| 299 void _computeSalt() { | 283 void _computeSalt() { |
| 300 var saltBuilder = new ApiSignature(); | 284 var saltBuilder = new ApiSignature(); |
| 301 saltBuilder.addInt(DATA_VERSION); | 285 saltBuilder.addInt(DATA_VERSION); |
| 302 saltBuilder.addBool(_options.strongMode); | 286 saltBuilder.addBool(_strongMode); |
| 303 saltBuilder.addString(_entryPoint.toString()); | |
| 304 _salt = saltBuilder.toByteList(); | 287 _salt = saltBuilder.toByteList(); |
| 305 } | 288 } |
| 306 | 289 |
| 307 String _getCycleSignature(LibraryCycle cycle) { | 290 String _getCycleSignature(LibraryCycle cycle) { |
| 308 bool hasMixinApplication = | 291 bool hasMixinApplication = |
| 309 cycle.libraries.any((library) => library.hasMixinApplicationLibrary); | 292 cycle.libraries.any((library) => library.hasMixinApplicationLibrary); |
| 310 var signatureBuilder = new ApiSignature(); | 293 var signatureBuilder = new ApiSignature(); |
| 311 signatureBuilder.addBytes(_salt); | 294 signatureBuilder.addBytes(_salt); |
| 312 Set<FileState> transitiveFiles = cycle.libraries | 295 Set<FileState> transitiveFiles = cycle.libraries |
| 313 .map((library) => library.transitiveFiles) | 296 .map((library) => library.transitiveFiles) |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 347 _invalidatedFiles.clear(); | 330 _invalidatedFiles.clear(); |
| 348 | 331 |
| 349 // Refresh the files. | 332 // Refresh the files. |
| 350 for (var fileUri in invalidatedFiles) { | 333 for (var fileUri in invalidatedFiles) { |
| 351 var file = _fsState.getFileByFileUri(fileUri); | 334 var file = _fsState.getFileByFileUri(fileUri); |
| 352 if (file != null) { | 335 if (file != null) { |
| 353 _logger.writeln('Refresh $fileUri'); | 336 _logger.writeln('Refresh $fileUri'); |
| 354 await file.refresh(); | 337 await file.refresh(); |
| 355 } | 338 } |
| 356 } | 339 } |
| 357 | |
| 358 // The file graph might have changed, perform GC. | |
| 359 var removedFiles = _fsState.gc(_entryPoint); | |
| 360 if (removedFiles.isNotEmpty && _watchFn != null) { | |
| 361 for (var removedFile in removedFiles) { | |
| 362 await _watchFn(removedFile.fileUri, false); | |
| 363 } | |
| 364 } | |
| 365 }); | 340 }); |
| 366 } | 341 } |
| 367 } | 342 } |
| 368 | 343 |
| 344 /// The result of compiling of a single file. |
| 345 class KernelResult { |
| 346 final CanonicalName nameRoot; |
| 347 final List<LibraryCycleResult> results; |
| 348 |
| 349 KernelResult(this.nameRoot, this.results); |
| 350 } |
| 351 |
| 369 /// Compilation result for a library cycle. | 352 /// Compilation result for a library cycle. |
| 370 class _LibraryCycleResult { | 353 class LibraryCycleResult { |
| 371 final LibraryCycle cycle; | 354 final LibraryCycle cycle; |
| 372 | 355 |
| 373 /// The signature of the result. | 356 /// The signature of the result. |
| 374 /// | 357 /// |
| 375 /// It is based on the full content of the libraries in the [cycle], and | 358 /// It is based on the full content of the libraries in the [cycle], and |
| 376 /// either API signatures of the transitive dependencies (usually), or | 359 /// either API signatures of the transitive dependencies (usually), or |
| 377 /// the full content of them (in the [cycle] has a library with a mixin | 360 /// the full content of them (in the [cycle] has a library with a mixin |
| 378 /// application). | 361 /// application). |
| 379 final String signature; | 362 final String signature; |
| 380 | 363 |
| 381 /// Kernel libraries for libraries in the [cycle]. Bodies of dependencies | 364 /// Kernel libraries for libraries in the [cycle]. Bodies of dependencies |
| 382 /// are not included, but but references to those dependencies are included. | 365 /// are not included, but but references to those dependencies are included. |
| 383 final List<Library> kernelLibraries; | 366 final List<Library> kernelLibraries; |
| 384 | 367 |
| 385 _LibraryCycleResult(this.cycle, this.signature, this.kernelLibraries); | 368 LibraryCycleResult(this.cycle, this.signature, this.kernelLibraries); |
| 386 } | 369 } |
| 387 | 370 |
| 388 @visibleForTesting | 371 @visibleForTesting |
| 389 class _TestView { | 372 class _TestView { |
| 390 /// The list of [LibraryCycle]s compiled for the last delta. | 373 /// The list of [LibraryCycle]s compiled for the last delta. |
| 391 /// It does not include libraries which were read from the cache. | 374 /// It does not include libraries which were read from the cache. |
| 392 final List<LibraryCycle> compiledCycles = []; | 375 final List<LibraryCycle> compiledCycles = []; |
| 393 } | 376 } |
| OLD | NEW |