build_aot.dart 6.15 KB
Newer Older
1 2 3 4 5 6
// Copyright 2016 The Chromium Authors. 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:async';

7
import '../base/build.dart';
8
import '../base/common.dart';
9
import '../base/file_system.dart';
10
import '../base/logger.dart';
11
import '../base/process.dart';
12
import '../build_info.dart';
13
import '../dart/package_map.dart';
14
import '../globals.dart';
15
import '../resident_runner.dart';
16
import '../runner/flutter_command.dart';
17
import 'build.dart';
18

19
class BuildAotCommand extends BuildSubCommand {
20
  BuildAotCommand({bool verboseHelp: false}) {
21 22 23
    usesTargetOption();
    addBuildModeFlags();
    usesPubOption();
24
    argParser
25
      ..addOption('output-dir', defaultsTo: getAotBuildDirectory())
26 27
      ..addOption('target-platform',
        defaultsTo: 'android-arm',
28
        allowed: <String>['android-arm', 'android-arm64', 'ios']
29
      )
30
      ..addFlag('quiet', defaultsTo: false)
31 32 33 34 35
      ..addFlag('preview-dart-2',
        defaultsTo: true,
        hide: !verboseHelp,
        help: 'Preview Dart 2.0 functionality.',
      )
36 37 38 39 40
      ..addFlag('build-shared-library',
        negatable: false,
        defaultsTo: false,
        help: 'Compile to a *.so file (requires NDK when building for Android).'
      )
41 42 43 44
      ..addMultiOption('ios-arch',
        splitCommas: true,
        defaultsTo: defaultIOSArchs.map(getNameForIOSArch),
        allowed: IOSArch.values.map(getNameForIOSArch),
45
        help: 'iOS architectures to build.',
46
      )
47
      ..addMultiOption(FlutterOptions.kExtraFrontEndOptions,
48 49 50
        splitCommas: true,
        hide: true,
      )
51
      ..addMultiOption(FlutterOptions.kExtraGenSnapshotOptions,
52 53
        splitCommas: true,
        hide: true,
54
      );
55 56 57 58 59 60
  }

  @override
  final String name = 'aot';

  @override
61
  final String description = "Build an ahead-of-time compiled snapshot of your app's Dart code.";
62 63

  @override
64
  Future<Null> runCommand() async {
65
    await super.runCommand();
66

67 68
    final String targetPlatform = argResults['target-platform'];
    final TargetPlatform platform = getTargetPlatformForName(targetPlatform);
69 70
    if (platform == null)
      throwToolExit('Unknown platform: $targetPlatform');
71

72 73
    final BuildMode buildMode = getBuildMode();

74 75
    Status status;
    if (!argResults['quiet']) {
76
      final String typeName = artifacts.getEngineType(platform, buildMode);
77 78 79
      status = logger.startProgress('Building AOT snapshot in ${getModeName(getBuildMode())} mode ($typeName)...',
          expectSlowOperation: true);
    }
80 81
    final String outputPath = argResults['output-dir'] ?? getAotBuildDirectory();
    try {
82 83
      final bool previewDart2 = argResults['preview-dart-2'];
      String mainPath = findMainDartFile(targetFile);
84
      final AOTSnapshotter snapshotter = new AOTSnapshotter();
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101

      // Compile to kernel, if Dart 2.
      if (previewDart2) {
        mainPath = await snapshotter.compileKernel(
          platform: platform,
          buildMode: buildMode,
          mainPath: mainPath,
          outputPath: outputPath,
          extraFrontEndOptions: argResults[FlutterOptions.kExtraFrontEndOptions],
        );
        if (mainPath == null) {
          printError('Compiler terminated unexpectedly.');
          return;
        }
      }

      // Build AOT snapshot.
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
      if (platform == TargetPlatform.ios) {
        // Determine which iOS architectures to build for.
        final Iterable<IOSArch> buildArchs = argResults['ios-arch'].map(getIOSArchForName);
        final Map<IOSArch, String> iosBuilds = <IOSArch, String>{};
        for (IOSArch arch in buildArchs)
          iosBuilds[arch] = fs.path.join(outputPath, getNameForIOSArch(arch));

        // Generate AOT snapshot and compile to arch-specific App.framework.
        final Map<IOSArch, Future<int>> exitCodes = <IOSArch, Future<int>>{};
        iosBuilds.forEach((IOSArch iosArch, String outputPath) {
          exitCodes[iosArch] = snapshotter.build(
            platform: platform,
            iosArch: iosArch,
            buildMode: buildMode,
            mainPath: mainPath,
            packagesPath: PackageMap.globalPackagesPath,
            outputPath: outputPath,
            previewDart2: previewDart2,
120
            buildSharedLibrary: false,
121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146
            extraGenSnapshotOptions: argResults[FlutterOptions.kExtraGenSnapshotOptions],
          ).then((int buildExitCode) {
            if (buildExitCode != 0)
              printError('Snapshotting ($iosArch) exited with non-zero exit code: $buildExitCode');
            return buildExitCode;
          });
        });

        // Merge arch-specific App.frameworks into a multi-arch App.framework.
        if ((await Future.wait(exitCodes.values)).every((int buildExitCode) => buildExitCode == 0)) {
          final Iterable<String> dylibs = iosBuilds.values.map((String outputDir) => fs.path.join(outputDir, 'App.framework', 'App'));
          fs.directory(fs.path.join(outputPath, 'App.framework'))..createSync();
          await runCheckedAsync(<String>['lipo']
            ..addAll(dylibs)
            ..addAll(<String>['-create', '-output', fs.path.join(outputPath, 'App.framework', 'App')]),
          );
        }
      } else {
        // Android AOT snapshot.
        final int snapshotExitCode = await snapshotter.build(
          platform: platform,
          buildMode: buildMode,
          mainPath: mainPath,
          packagesPath: PackageMap.globalPackagesPath,
          outputPath: outputPath,
          previewDart2: previewDart2,
147
          buildSharedLibrary: argResults['build-shared-library'],
148 149 150 151
          extraGenSnapshotOptions: argResults[FlutterOptions.kExtraGenSnapshotOptions],
        );
        if (snapshotExitCode != 0) {
          printError('Snapshotting exited with non-zero exit code: $snapshotExitCode');
152
          return;
153
        }
154 155 156 157
      }
    } on String catch (error) {
      // Catch the String exceptions thrown from the `runCheckedSync` methods below.
      printError(error);
158
      return;
159
    }
160
    status?.stop();
161

162
    if (outputPath == null)
163
      throwToolExit(null);
164

165 166 167 168 169 170
    final String builtMessage = 'Built to $outputPath${fs.path.separator}.';
    if (argResults['quiet']) {
      printTrace(builtMessage);
    } else {
      printStatus(builtMessage);
    }
171 172
  }
}