
This PR introduces a `NativeAssetsManifest.json` next to the `AssetManifest.bin` and `FontManifest.json`. This removes the need for embedding the native assets mapping inside the kernel file and enables decoupling native assets building and bundling from the kernel compilation in flutter tools. This means `flutter run` no longer does a dry run of `hook/build.dart` hooks. (It also means all isolate groups will have the same native assets. However, since Flutter does not support `Isolate.spawnUri` from kernel files, this is not a regression.) Implementation details: * g3 is still using kernel embedding. https://github.com/flutter/flutter/pull/142016 introduced an argument to embed a `native_assets.yaml` inside `flutter attach` and `flutter run` (the outer flutter process), but it is not used in `flutter assemble` (the inner process when doing `flutter run`). So, those arguments need to still be respected. However, all other logic related to embedding a yaml encoding in the kernel file has been removed. * All dry-run logic has been removed. 🎉 * The `KernelSnapshot` target no longer depends on the `InstallCodeAssets` target. Instead, the various OS-specific "BundleAsset" targets now depend on the `InstallCodeAssets` target. The `InstallCodeAssets` invokes the build hooks and produces the `NativeAssetsManifest.json`. The various "BundleAsset" commands synchronize the `NativeAssetsManifest.json` to the app bundle. * `InstallCodeAssets` produces a `native_assets.json`, which is renamed to `NativeAssetsManifest.json` in the various "Bundle" targets. This means that all unit tests of the "Bundle" targets now need to create this file. (Similar to how `app.dill` is expected to exist because `KernelSnapshot` is a dependency of the "Bundle" targets.) * Because dynamic libraries need to be code signed (at least on iOS and MacOS), the bundling of the dylibs is _not_ migrated to reuse `_updateDevFS` (which is used for ordinary assets). Only the 2nd and 3rd invocation of `flutter assemble` from `xcodebuild` has access to the code signing identity. Relevant tests: * test/integration.shard/isolated/native_assets_test.dart - runs `flutter run` with native assets including hot restart and hot reload. TODO: * Undo engine-roll in this PR after engine has rolled in. Issue: * https://github.com/flutter/flutter/issues/154425 Related PRs: * https://dart-review.googlesource.com/c/sdk/+/388161 * https://github.com/flutter/engine/pull/56727 ## Pre-launch Checklist - [x] I read the [Contributor Guide] and followed the process outlined there for submitting PRs. - [x] I read the [Tree Hygiene] wiki page, which explains my responsibilities. - [x] I read and followed the [Flutter Style Guide], including [Features we expect every widget to implement]. - [x] I signed the [CLA]. - [x] I listed at least one issue that this PR fixes in the description above. - [x] I updated/added relevant documentation (doc comments with `///`). - [x] I added new tests to check the change I am making, or this PR is [test-exempt]. - [x] I followed the [breaking change policy] and added [Data Driven Fixes] where supported. - [x] All existing and new tests are passing. <!-- Links --> [Contributor Guide]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#overview [Tree Hygiene]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md [test-exempt]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#tests [Flutter Style Guide]: https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md [Features we expect every widget to implement]: https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md#features-we-expect-every-widget-to-implement [CLA]: https://cla.developers.google.com/ [flutter/tests]: https://github.com/flutter/tests [breaking change policy]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#handling-breaking-changes [Discord]: https://github.com/flutter/flutter/blob/main/docs/contributing/Chat.md [Data Driven Fixes]: https://github.com/flutter/flutter/blob/main/docs/contributing/Data-driven-Fixes.md
258 lines
7.5 KiB
Dart
258 lines
7.5 KiB
Dart
// Copyright 2014 The Flutter 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 '../../artifacts.dart';
|
|
import '../../base/file_system.dart';
|
|
import '../../build_info.dart';
|
|
import '../../convert.dart';
|
|
import '../../devfs.dart';
|
|
import '../../project.dart';
|
|
import '../build_system.dart';
|
|
import '../depfile.dart';
|
|
import '../exceptions.dart';
|
|
import 'assets.dart';
|
|
import 'common.dart';
|
|
import 'desktop.dart';
|
|
import 'icon_tree_shaker.dart';
|
|
import 'native_assets.dart';
|
|
|
|
/// The only files/subdirectories we care out.
|
|
const List<String> _kLinuxArtifacts = <String>[
|
|
'libflutter_linux_gtk.so',
|
|
];
|
|
|
|
const String _kLinuxDepfile = 'linux_engine_sources.d';
|
|
|
|
/// Copies the Linux desktop embedding files to the copy directory.
|
|
class UnpackLinux extends Target {
|
|
const UnpackLinux(this.targetPlatform);
|
|
|
|
final TargetPlatform targetPlatform;
|
|
|
|
@override
|
|
String get name => 'unpack_linux';
|
|
|
|
@override
|
|
List<Source> get inputs => const <Source>[
|
|
Source.pattern('{FLUTTER_ROOT}/packages/flutter_tools/lib/src/build_system/targets/linux.dart'),
|
|
];
|
|
|
|
@override
|
|
List<Source> get outputs => const <Source>[];
|
|
|
|
@override
|
|
List<String> get depfiles => const <String>[_kLinuxDepfile];
|
|
|
|
@override
|
|
List<Target> get dependencies => <Target>[];
|
|
|
|
@override
|
|
Future<void> build(Environment environment) async {
|
|
final String? buildModeEnvironment = environment.defines[kBuildMode];
|
|
if (buildModeEnvironment == null) {
|
|
throw MissingDefineException(kBuildMode, name);
|
|
}
|
|
final BuildMode buildMode = BuildMode.fromCliName(buildModeEnvironment);
|
|
final String engineSourcePath = environment.artifacts
|
|
.getArtifactPath(
|
|
Artifact.linuxDesktopPath,
|
|
mode: buildMode,
|
|
platform: targetPlatform,
|
|
);
|
|
final String headersPath = environment.artifacts
|
|
.getArtifactPath(
|
|
Artifact.linuxHeaders,
|
|
mode: buildMode,
|
|
platform: targetPlatform,
|
|
);
|
|
final Directory outputDirectory = environment.fileSystem.directory(
|
|
environment.fileSystem.path.join(
|
|
environment.projectDir.path,
|
|
'linux',
|
|
'flutter',
|
|
'ephemeral',
|
|
));
|
|
final Depfile depfile = unpackDesktopArtifacts(
|
|
fileSystem: environment.fileSystem,
|
|
engineSourcePath: engineSourcePath,
|
|
outputDirectory: outputDirectory,
|
|
artifacts: _kLinuxArtifacts,
|
|
clientSourcePaths: <String>[headersPath],
|
|
icuDataPath: environment.artifacts.getArtifactPath(
|
|
Artifact.icuData,
|
|
platform: targetPlatform,
|
|
)
|
|
);
|
|
environment.depFileService.writeToFile(
|
|
depfile,
|
|
environment.buildDir.childFile(_kLinuxDepfile),
|
|
);
|
|
}
|
|
}
|
|
|
|
/// Creates a bundle for the Linux desktop target.
|
|
abstract class BundleLinuxAssets extends Target {
|
|
const BundleLinuxAssets(this.targetPlatform);
|
|
|
|
final TargetPlatform targetPlatform;
|
|
|
|
@override
|
|
List<Target> get dependencies => <Target>[
|
|
const KernelSnapshot(),
|
|
const InstallCodeAssets(),
|
|
UnpackLinux(targetPlatform),
|
|
];
|
|
|
|
@override
|
|
List<Source> get inputs => const <Source>[
|
|
Source.pattern('{FLUTTER_ROOT}/packages/flutter_tools/lib/src/build_system/targets/linux.dart'),
|
|
Source.pattern('{PROJECT_DIR}/pubspec.yaml'),
|
|
...IconTreeShaker.inputs,
|
|
];
|
|
|
|
@override
|
|
List<String> get depfiles => const <String>[
|
|
'flutter_assets.d',
|
|
];
|
|
|
|
@override
|
|
Future<void> build(Environment environment) async {
|
|
final String? buildModeEnvironment = environment.defines[kBuildMode];
|
|
if (buildModeEnvironment == null) {
|
|
throw MissingDefineException(kBuildMode, 'bundle_linux_assets');
|
|
}
|
|
final BuildMode buildMode = BuildMode.fromCliName(buildModeEnvironment);
|
|
final Directory outputDirectory = environment.outputDir
|
|
.childDirectory('flutter_assets');
|
|
if (!outputDirectory.existsSync()) {
|
|
outputDirectory.createSync();
|
|
}
|
|
|
|
// Only copy the kernel blob in debug mode.
|
|
if (buildMode == BuildMode.debug) {
|
|
environment.buildDir.childFile('app.dill')
|
|
.copySync(outputDirectory.childFile('kernel_blob.bin').path);
|
|
}
|
|
final String versionInfo = getVersionInfo(environment.defines);
|
|
final Depfile depfile = await copyAssets(
|
|
environment,
|
|
outputDirectory,
|
|
targetPlatform: targetPlatform,
|
|
buildMode: buildMode,
|
|
additionalContent: <String, DevFSContent>{
|
|
'version.json': DevFSStringContent(versionInfo),
|
|
'NativeAssetsManifest.json':
|
|
DevFSFileContent(environment.buildDir.childFile('native_assets.json')),
|
|
},
|
|
);
|
|
environment.depFileService.writeToFile(
|
|
depfile,
|
|
environment.buildDir.childFile('flutter_assets.d'),
|
|
);
|
|
}
|
|
|
|
/// Return json encoded string that contains data about version for package_info
|
|
String getVersionInfo(Map<String, String> defines) {
|
|
final Map<String, dynamic> versionInfo =
|
|
jsonDecode(FlutterProject.current().getVersionInfo())
|
|
as Map<String, dynamic>;
|
|
|
|
if (defines.containsKey(kBuildNumber)) {
|
|
versionInfo['build_number'] = defines[kBuildNumber];
|
|
}
|
|
|
|
if (defines.containsKey(kBuildName)) {
|
|
versionInfo['version'] = defines[kBuildName];
|
|
}
|
|
|
|
return jsonEncode(versionInfo);
|
|
}
|
|
}
|
|
|
|
/// A wrapper for AOT compilation that copies app.so into the output directory.
|
|
class LinuxAotBundle extends Target {
|
|
/// Create a [LinuxAotBundle] wrapper for [aotTarget].
|
|
const LinuxAotBundle(this.aotTarget);
|
|
|
|
/// The [AotElfBase] subclass that produces the app.so.
|
|
final AotElfBase aotTarget;
|
|
|
|
@override
|
|
String get name => 'linux_aot_bundle';
|
|
|
|
@override
|
|
List<Source> get inputs => const <Source>[
|
|
Source.pattern('{BUILD_DIR}/app.so'),
|
|
];
|
|
|
|
@override
|
|
List<Source> get outputs => const <Source>[
|
|
Source.pattern('{OUTPUT_DIR}/lib/libapp.so'),
|
|
];
|
|
|
|
@override
|
|
List<Target> get dependencies => <Target>[
|
|
aotTarget,
|
|
];
|
|
|
|
@override
|
|
Future<void> build(Environment environment) async {
|
|
final File outputFile = environment.buildDir.childFile('app.so');
|
|
final Directory outputDirectory = environment.outputDir.childDirectory('lib');
|
|
if (!outputDirectory.existsSync()) {
|
|
outputDirectory.createSync(recursive: true);
|
|
}
|
|
outputFile.copySync(outputDirectory.childFile('libapp.so').path);
|
|
}
|
|
}
|
|
|
|
class DebugBundleLinuxAssets extends BundleLinuxAssets {
|
|
const DebugBundleLinuxAssets(super.targetPlatform);
|
|
|
|
@override
|
|
String get name => 'debug_bundle_${getNameForTargetPlatform(targetPlatform)}_assets';
|
|
|
|
@override
|
|
List<Source> get inputs => <Source>[
|
|
const Source.pattern('{BUILD_DIR}/app.dill'),
|
|
];
|
|
|
|
@override
|
|
List<Source> get outputs => <Source>[
|
|
const Source.pattern('{OUTPUT_DIR}/flutter_assets/kernel_blob.bin'),
|
|
];
|
|
}
|
|
|
|
class ProfileBundleLinuxAssets extends BundleLinuxAssets {
|
|
const ProfileBundleLinuxAssets(super.targetPlatform);
|
|
|
|
@override
|
|
String get name => 'profile_bundle_${getNameForTargetPlatform(targetPlatform)}_assets';
|
|
|
|
@override
|
|
List<Source> get outputs => const <Source>[];
|
|
|
|
@override
|
|
List<Target> get dependencies => <Target>[
|
|
...super.dependencies,
|
|
LinuxAotBundle(AotElfProfile(targetPlatform)),
|
|
];
|
|
}
|
|
|
|
class ReleaseBundleLinuxAssets extends BundleLinuxAssets {
|
|
const ReleaseBundleLinuxAssets(super.targetPlatform);
|
|
|
|
@override
|
|
String get name => 'release_bundle_${getNameForTargetPlatform(targetPlatform)}_assets';
|
|
|
|
@override
|
|
List<Source> get outputs => const <Source>[];
|
|
|
|
@override
|
|
List<Target> get dependencies => <Target>[
|
|
...super.dependencies,
|
|
LinuxAotBundle(AotElfRelease(targetPlatform)),
|
|
];
|
|
}
|