| #!tools/sdks/dart-sdk/bin/dart |
| // Copyright (c) 2023, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| import 'dart:convert' as convert show json; |
| import 'dart:io'; |
| |
| import 'package:pool/pool.dart'; |
| |
| final pool = Pool(Platform.numberOfProcessors); |
| |
| Future<void> buildOffsetsExtractor(List<String> args) async { |
| // Build all configurations |
| await forAllConfigurationsMode(( |
| String buildDir, |
| String mode, |
| String arch, |
| ) async { |
| print('Building $buildDir'); |
| await run([ |
| 'tools/build.py', |
| ...args, |
| '-a$arch', |
| '-m$mode', |
| '--no-rbe', |
| 'offsets_extractor', |
| 'offsets_extractor_aotruntime', |
| ]); |
| print('Building $buildDir - done'); |
| }); |
| } |
| |
| Future<String> runOffsetsExtractor() async { |
| final (jit, aot) = await ( |
| forAllConfigurationsMode((String buildDir, _, __) async { |
| return await run(['$buildDir/offsets_extractor']); |
| }).then<String>((lines) => lines.join(',\n')), |
| forAllConfigurationsMode((String buildDir, _, __) async { |
| return await run(['$buildDir/offsets_extractor_aotruntime']); |
| }).then<String>((lines) => lines.join(',\n')), |
| ).wait; |
| |
| final buf = StringBuffer(); |
| buf.writeln('['); |
| buf.writeln(jit); |
| buf.writeln(','); |
| buf.writeln(aot); |
| buf.writeln(']'); |
| return buf.toString(); |
| } |
| |
| String toCValue(Object? value) { |
| final intValue = int.parse(value as String); |
| if (intValue == -1) return '-1'; |
| return '0x${intValue.toRadixString(16)}'; |
| } |
| |
| Future<void> writeCHeaderFile(List json) async { |
| final extractedOffsetsFile = |
| 'runtime/vm/compiler/runtime_offsets_extracted.h'; |
| |
| final old = File(extractedOffsetsFile).readAsStringSync(); |
| final header = old.substring(0, old.indexOf('\n#if ')); |
| final footer = old.substring(old.lastIndexOf('\n#endif ')); |
| |
| final output = StringBuffer(); |
| output.writeln(header); |
| for (final config in json) { |
| final product = (config['product'] as bool) ? '' : '!'; |
| final productDef = '${product}defined(PRODUCT)'; |
| final arch = (config['arch'] as String).toUpperCase(); |
| final archDef = 'defined(TARGET_ARCH_$arch)'; |
| final compressed = (config['compressed'] as bool) ? '' : '!'; |
| final compressedDef = '${compressed}defined(DART_COMPRESSED_POINTERS)'; |
| final aot = (config['aot'] as bool) ? 'AOT_' : ''; |
| final prefix = 'static constexpr dart::compiler::target::word $aot'; |
| final offsets = config['offsets'] as List; |
| output.writeln('#if $productDef && $archDef && $compressedDef'); |
| for (final offset in offsets) { |
| final kind = offset['kind'] as String; |
| switch (kind) { |
| case 'value': |
| final cls = offset['class'] as String; |
| final name = offset['name'] as String; |
| final value = toCValue(offset['value']); |
| output.writeln('$prefix${cls}_$name = $value;'); |
| break; |
| case 'array': |
| final cls = offset['class'] as String; |
| final startOffset = toCValue(offset['startOffset']); |
| final elemSize = toCValue(offset['elemSize']); |
| output.writeln('$prefix${cls}_elements_start_offset = $startOffset;'); |
| output.writeln('$prefix${cls}_element_size = $elemSize;'); |
| break; |
| case 'range': |
| final cls = offset['class'] as String; |
| final name = offset['name'] as String; |
| final values = (offset['values'] as List).map(toCValue).toList(); |
| output.writeln('$prefix${cls}_$name[] = {${values.join(', ')}};'); |
| break; |
| } |
| } |
| output.writeln('#endif // $productDef &&'); |
| output.writeln(' // $archDef &&'); |
| output.writeln(' // $compressedDef'); |
| output.writeln(); |
| } |
| output.writeln(footer); |
| File(extractedOffsetsFile).writeAsStringSync(output.toString()); |
| print('Written $extractedOffsetsFile'); |
| print('Running `git cl format $extractedOffsetsFile'); |
| await run(['git', 'cl', 'format', extractedOffsetsFile]); |
| } |
| |
| Future<List<T>> forAllConfigurationsMode<T>( |
| Future<T> Function(String buildDir, String mode, String arch) fun, |
| ) async { |
| final archs = [ |
| 'simarm', |
| 'x64', |
| 'ia32', |
| 'simarm64', |
| 'x64c', |
| 'simarm64c', |
| 'simriscv32', |
| 'simriscv64', |
| ]; |
| final futures = <Future<T>>[]; |
| for (final mode in ['release', 'product']) { |
| for (final arch in archs) { |
| final buildDir = 'out/${mode.capitalized}${arch.upper}/'; |
| futures.add(pool.withResource(() => fun(buildDir, mode, arch))); |
| } |
| } |
| return await Future.wait(futures); |
| } |
| |
| Future<String> run(List<String> args) async { |
| final result = await Process.run( |
| args.first, |
| args.skip(1).toList(), |
| runInShell: true, |
| ); |
| if (result.exitCode != 0) { |
| exitCode = result.exitCode; |
| print('Running ${args.join(' ')} has failed with exit code $exitCode:'); |
| print('${result.stdout}'); |
| print('${result.stderr}'); |
| } |
| return result.stdout; |
| } |
| |
| extension on String { |
| String get capitalized => substring(0, 1).toUpperCase() + substring(1); |
| String get upper => toUpperCase(); |
| } |
| |
| void main(List<String> args) async { |
| final sdkRoot = Platform.script.resolve('../').toFilePath(); |
| Directory.current = Directory(sdkRoot); |
| |
| await buildOffsetsExtractor(args); |
| if (exitCode != 0) { |
| return; |
| } |
| |
| final text = await runOffsetsExtractor(); |
| if (exitCode != 0) { |
| return; |
| } |
| |
| final json = convert.json.decode(text) as List; |
| |
| await writeCHeaderFile(json); |
| } |