Flutter Engine
 
Loading...
Searching...
No Matches
FlutterDartProject.mm
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#define FML_USED_ON_EMBEDDER
6
8
9#import <Metal/Metal.h>
10#import <UIKit/UIKit.h>
11
12#include <sstream>
13
17#import "flutter/shell/platform/darwin/common/InternalFlutterSwiftCommon/InternalFlutterSwiftCommon.h"
19
21
22extern "C" {
23#if FLUTTER_RUNTIME_MODE == FLUTTER_RUNTIME_MODE_DEBUG
24// Used for debugging dart:* sources.
25extern const uint8_t kPlatformStrongDill[];
26extern const intptr_t kPlatformStrongDillSize;
27#endif
28}
29
30static const char* kApplicationKernelSnapshotFileName = "kernel_blob.bin";
31
33 static BOOL result = NO;
34 static dispatch_once_t once_token = 0;
35 dispatch_once(&once_token, ^{
36 id<MTLDevice> device = MTLCreateSystemDefaultDevice();
37 result = [device supportsFamily:MTLGPUFamilyApple2];
38 });
39 return result;
40}
41
42flutter::Settings FLTDefaultSettingsForBundle(NSBundle* bundle, NSProcessInfo* processInfoOrNil) {
43 auto command_line = flutter::CommandLineFromNSProcessInfo(processInfoOrNil);
44
45 // Precedence:
46 // 1. Settings from the specified NSBundle.
47 // 2. Settings passed explicitly via command-line arguments.
48 // 3. Settings from the NSBundle with the default bundle ID.
49 // 4. Settings from the main NSBundle and default values.
50
51 NSBundle* mainBundle = FLTGetApplicationBundle();
52 NSBundle* engineBundle = [NSBundle bundleForClass:[FlutterDartProject class]];
53
54 bool hasExplicitBundle = bundle != nil;
55 if (bundle == nil) {
56 bundle = FLTFrameworkBundleWithIdentifier([FlutterDartProject defaultBundleIdentifier]);
57 }
58
59 auto settings = flutter::SettingsFromCommandLine(command_line, true);
60
61 settings.task_observer_add = [](intptr_t key, const fml::closure& callback) {
64 return queue_id;
65 };
66
67 settings.task_observer_remove = [](fml::TaskQueueId queue_id, intptr_t key) {
69 };
70
71 settings.log_message_callback = [](const std::string& tag, const std::string& message) {
72 std::stringstream stream;
73 if (!tag.empty()) {
74 stream << tag << ": ";
75 }
76 stream << message;
77 std::string log = stream.str();
78 [FlutterLogger logDirect:[NSString stringWithUTF8String:log.c_str()]];
79 };
80
81 settings.enable_platform_isolates = true;
82
83 // The command line arguments may not always be complete. If they aren't, attempt to fill in
84 // defaults.
85
86 // Flutter ships the ICU data file in the bundle of the engine. Look for it there.
87 if (settings.icu_data_path.empty()) {
88 NSString* icuDataPath = [engineBundle pathForResource:@"icudtl" ofType:@"dat"];
89 if (icuDataPath.length > 0) {
90 settings.icu_data_path = icuDataPath.UTF8String;
91 }
92 }
93
95 if (hasExplicitBundle) {
96 NSString* executablePath = bundle.executablePath;
97 if ([[NSFileManager defaultManager] fileExistsAtPath:executablePath]) {
98 settings.application_library_paths.push_back(executablePath.UTF8String);
99 }
100 }
101
102 // No application bundle specified. Try a known location from the main bundle's Info.plist.
103 if (settings.application_library_paths.empty()) {
104 NSString* libraryName = [mainBundle objectForInfoDictionaryKey:@"FLTLibraryPath"];
105 NSString* libraryPath = [mainBundle pathForResource:libraryName ofType:@""];
106 if (libraryPath.length > 0) {
107 NSString* executablePath = [NSBundle bundleWithPath:libraryPath].executablePath;
108 if (executablePath.length > 0) {
109 settings.application_library_paths.push_back(executablePath.UTF8String);
110 }
111 }
112 }
113
114 // In case the application bundle is still not specified, look for the App.framework in the
115 // Frameworks directory.
116 if (settings.application_library_paths.empty()) {
117 NSString* applicationFrameworkPath = [mainBundle pathForResource:@"Frameworks/App.framework"
118 ofType:@""];
119 if (applicationFrameworkPath.length > 0) {
120 NSString* executablePath =
121 [NSBundle bundleWithPath:applicationFrameworkPath].executablePath;
122 if (executablePath.length > 0) {
123 settings.application_library_paths.push_back(executablePath.UTF8String);
124 }
125 }
126 }
127 }
128
129 // Checks to see if the flutter assets directory is already present.
130 if (settings.assets_path.empty()) {
131 NSString* assetsPath = FLTAssetsPathFromBundle(bundle);
132
133 if (assetsPath.length == 0) {
134 NSLog(@"Failed to find assets path for \"%@\"", bundle);
135 } else {
136 settings.assets_path = assetsPath.UTF8String;
137
138 // Check if there is an application kernel snapshot in the assets directory we could
139 // potentially use. Looking for the snapshot makes sense only if we have a VM that can use
140 // it.
142 NSURL* applicationKernelSnapshotURL =
143 [NSURL URLWithString:@(kApplicationKernelSnapshotFileName)
144 relativeToURL:[NSURL fileURLWithPath:assetsPath]];
145 NSError* error;
146 if ([applicationKernelSnapshotURL checkResourceIsReachableAndReturnError:&error]) {
147 settings.application_kernel_asset = applicationKernelSnapshotURL.path.UTF8String;
148 } else {
149 NSLog(@"Failed to find snapshot at %@: %@", applicationKernelSnapshotURL.path, error);
150 }
151 }
152 }
153 }
154
155 // Domain network configuration
156 // Disabled in https://github.com/flutter/flutter/issues/72723.
157 // Re-enable in https://github.com/flutter/flutter/issues/54448.
158 settings.may_insecurely_connect_to_all_domains = true;
159 settings.domain_network_policy = "";
160
161 // Whether to enable wide gamut colors.
162#if TARGET_OS_SIMULATOR
163 // As of Xcode 14.1, the wide gamut surface pixel formats are not supported by
164 // the simulator.
165 settings.enable_wide_gamut = false;
166 // Removes unused function warning.
168#else
169 NSNumber* nsEnableWideGamut = [mainBundle objectForInfoDictionaryKey:@"FLTEnableWideGamut"];
170 BOOL enableWideGamut =
171 (nsEnableWideGamut ? nsEnableWideGamut.boolValue : YES) && DoesHardwareSupportWideGamut();
172 settings.enable_wide_gamut = enableWideGamut;
173#endif
174
175 NSNumber* nsAntialiasLines = [mainBundle objectForInfoDictionaryKey:@"FLTAntialiasLines"];
176 settings.impeller_antialiased_lines = (nsAntialiasLines ? nsAntialiasLines.boolValue : NO);
177
178 settings.warn_on_impeller_opt_out = true;
179
180 NSNumber* enableTraceSystrace = [mainBundle objectForInfoDictionaryKey:@"FLTTraceSystrace"];
181 // Change the default only if the option is present.
182 if (enableTraceSystrace != nil) {
183 settings.trace_systrace = enableTraceSystrace.boolValue;
184 }
185
186 NSNumber* profileMicrotasks = [mainBundle objectForInfoDictionaryKey:@"FLTProfileMicrotasks"];
187 // Change the default only if the option is present.
188 if (profileMicrotasks != nil) {
189 settings.profile_microtasks = profileMicrotasks.boolValue;
190 }
191
192 NSNumber* enableDartAsserts = [mainBundle objectForInfoDictionaryKey:@"FLTEnableDartAsserts"];
193 if (enableDartAsserts != nil) {
194 settings.dart_flags.push_back("--enable-asserts");
195 }
196
197 NSNumber* enableDartProfiling = [mainBundle objectForInfoDictionaryKey:@"FLTEnableDartProfiling"];
198 // Change the default only if the option is present.
199 if (enableDartProfiling != nil) {
200 settings.enable_dart_profiling = enableDartProfiling.boolValue;
201 }
202
203 NSNumber* profileStartup = [mainBundle objectForInfoDictionaryKey:@"FLTProfileStartup"];
204 // Change the default only if the option is present.
205 if (profileStartup != nil) {
206 settings.profile_startup = profileStartup.boolValue;
207 }
208
209 // Leak Dart VM settings, set whether leave or clean up the VM after the last shell shuts down.
210 NSNumber* leakDartVM = [mainBundle objectForInfoDictionaryKey:@"FLTLeakDartVM"];
211 // It will change the default leak_vm value in settings only if the key exists.
212 if (leakDartVM != nil) {
213 settings.leak_vm = leakDartVM.boolValue;
214 }
215
216 NSNumber* enableMergedPlatformUIThread =
217 [mainBundle objectForInfoDictionaryKey:@"FLTEnableMergedPlatformUIThread"];
218 if (enableMergedPlatformUIThread != nil) {
219 FML_CHECK(enableMergedPlatformUIThread.boolValue)
220 << "FLTEnableMergedPlatformUIThread=false is no longer allowed.";
221 }
222
223 NSNumber* enableFlutterGPU = [mainBundle objectForInfoDictionaryKey:@"FLTEnableFlutterGPU"];
224 if (enableFlutterGPU != nil) {
225 settings.enable_flutter_gpu = enableFlutterGPU.boolValue;
226 }
227
228#if FLUTTER_RUNTIME_MODE == FLUTTER_RUNTIME_MODE_DEBUG
229 // There are no ownership concerns here as all mappings are owned by the
230 // embedder and not the engine.
231 auto make_mapping_callback = [](const uint8_t* mapping, size_t size) {
232 return [mapping, size]() { return std::make_unique<fml::NonOwnedMapping>(mapping, size); };
233 };
234
235 settings.dart_library_sources_kernel =
236 make_mapping_callback(kPlatformStrongDill, kPlatformStrongDillSize);
237#endif // FLUTTER_RUNTIME_MODE == FLUTTER_RUNTIME_MODE_DEBUG
238
239 // If we even support setting this e.g. from the command line or the plist,
240 // we should let the user override it.
241 // Otherwise, we want to set this to a value that will avoid having the OS
242 // kill us. On most iOS devices, that happens somewhere near half
243 // the available memory.
244 // The VM expects this value to be in megabytes.
245 if (settings.old_gen_heap_size <= 0) {
246 settings.old_gen_heap_size = std::round([NSProcessInfo processInfo].physicalMemory * .48 /
248 }
249
250 // This is the formula Android uses.
251 // https://android.googlesource.com/platform/frameworks/base/+/39ae5bac216757bc201490f4c7b8c0f63006c6cd/libs/hwui/renderthread/CacheManager.cpp#45
252 CGFloat scale = [UIScreen mainScreen].scale;
253 CGFloat screenWidth = [UIScreen mainScreen].bounds.size.width * scale;
254 CGFloat screenHeight = [UIScreen mainScreen].bounds.size.height * scale;
255 settings.resource_cache_max_bytes_threshold = screenWidth * screenHeight * 12 * 4;
256
257 // Whether to enable ios embedder api.
258 NSNumber* enable_embedder_api =
259 [mainBundle objectForInfoDictionaryKey:@"FLTEnableIOSEmbedderAPI"];
260 // Change the default only if the option is present.
261 if (enable_embedder_api) {
262 settings.enable_embedder_api = enable_embedder_api.boolValue;
263 }
264
265 return settings;
266}
267
268@implementation FlutterDartProject {
269 flutter::Settings _settings;
270}
271
272// This property is marked unavailable on iOS in the common header.
273// That doesn't seem to be enough to prevent this property from being synthesized.
274// Mark dynamic to avoid warnings.
275@dynamic dartEntrypointArguments;
276
277#pragma mark - Override base class designated initializers
278
279- (instancetype)init {
280 return [self initWithPrecompiledDartBundle:nil];
281}
282
283#pragma mark - Designated initializers
284
285- (instancetype)initWithPrecompiledDartBundle:(nullable NSBundle*)bundle {
286 self = [super init];
287
288 if (self) {
289 _settings = FLTDefaultSettingsForBundle(bundle);
290 }
291
292 return self;
293}
294
295- (instancetype)initWithSettings:(const flutter::Settings&)settings {
296 self = [self initWithPrecompiledDartBundle:nil];
297
298 if (self) {
299 _settings = settings;
300 }
301
302 return self;
303}
304
305#pragma mark - PlatformData accessors
306
307- (const flutter::PlatformData)defaultPlatformData {
308 flutter::PlatformData PlatformData;
309 PlatformData.lifecycle_state = std::string("AppLifecycleState.detached");
310 return PlatformData;
311}
312
313#pragma mark - Settings accessors
314
315- (const flutter::Settings&)settings {
316 return _settings;
317}
318
319- (flutter::RunConfiguration)runConfiguration {
320 return [self runConfigurationForEntrypoint:nil];
321}
322
323- (flutter::RunConfiguration)runConfigurationForEntrypoint:(nullable NSString*)entrypointOrNil {
324 return [self runConfigurationForEntrypoint:entrypointOrNil libraryOrNil:nil];
325}
326
327- (flutter::RunConfiguration)runConfigurationForEntrypoint:(nullable NSString*)entrypointOrNil
328 libraryOrNil:(nullable NSString*)dartLibraryOrNil {
329 return [self runConfigurationForEntrypoint:entrypointOrNil
330 libraryOrNil:dartLibraryOrNil
331 entrypointArgs:nil];
332}
333
334- (flutter::RunConfiguration)runConfigurationForEntrypoint:(nullable NSString*)entrypointOrNil
335 libraryOrNil:(nullable NSString*)dartLibraryOrNil
336 entrypointArgs:
337 (nullable NSArray<NSString*>*)entrypointArgs {
338 auto config = flutter::RunConfiguration::InferFromSettings(_settings);
339 if (dartLibraryOrNil && entrypointOrNil) {
340 config.SetEntrypointAndLibrary(std::string([entrypointOrNil UTF8String]),
341 std::string([dartLibraryOrNil UTF8String]));
342
343 } else if (entrypointOrNil) {
344 config.SetEntrypoint(std::string([entrypointOrNil UTF8String]));
345 }
346
347 if (entrypointArgs.count) {
348 std::vector<std::string> cppEntrypointArgs;
349 for (NSString* arg in entrypointArgs) {
350 cppEntrypointArgs.push_back(std::string([arg UTF8String]));
351 }
352 config.SetEntrypointArgs(std::move(cppEntrypointArgs));
353 }
354
355 return config;
356}
357
358#pragma mark - Assets-related utilities
359
360+ (NSString*)flutterAssetsName:(NSBundle*)bundle {
361 if (bundle == nil) {
362 bundle = FLTFrameworkBundleWithIdentifier([FlutterDartProject defaultBundleIdentifier]);
363 }
364 return FLTAssetPath(bundle);
365}
366
367+ (NSString*)domainNetworkPolicy:(NSDictionary*)appTransportSecurity {
368 // https://developer.apple.com/documentation/bundleresources/information_property_list/nsapptransportsecurity/nsexceptiondomains
369 NSDictionary* exceptionDomains = appTransportSecurity[@"NSExceptionDomains"];
370 if (exceptionDomains == nil) {
371 return @"";
372 }
373 NSMutableArray* networkConfigArray = [[NSMutableArray alloc] init];
374 for (NSString* domain in exceptionDomains) {
375 NSDictionary* domainConfiguration = exceptionDomains[domain];
376 // Default value is false.
377 bool includesSubDomains = [domainConfiguration[@"NSIncludesSubdomains"] boolValue];
378 bool allowsCleartextCommunication =
379 [domainConfiguration[@"NSExceptionAllowsInsecureHTTPLoads"] boolValue];
380 [networkConfigArray addObject:@[
381 domain, includesSubDomains ? @YES : @NO, allowsCleartextCommunication ? @YES : @NO
382 ]];
383 }
384 NSData* jsonData = [NSJSONSerialization dataWithJSONObject:networkConfigArray
385 options:0
386 error:NULL];
387 return [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding];
388}
389
390+ (bool)allowsArbitraryLoads:(NSDictionary*)appTransportSecurity {
391 return [appTransportSecurity[@"NSAllowsArbitraryLoads"] boolValue];
392}
393
394+ (NSString*)lookupKeyForAsset:(NSString*)asset {
395 return [self lookupKeyForAsset:asset fromBundle:nil];
396}
397
398+ (NSString*)lookupKeyForAsset:(NSString*)asset fromBundle:(nullable NSBundle*)bundle {
399 NSString* flutterAssetsName = [FlutterDartProject flutterAssetsName:bundle];
400 return [NSString stringWithFormat:@"%@/%@", flutterAssetsName, asset];
401}
402
403+ (NSString*)lookupKeyForAsset:(NSString*)asset fromPackage:(NSString*)package {
404 return [self lookupKeyForAsset:asset fromPackage:package fromBundle:nil];
405}
406
407+ (NSString*)lookupKeyForAsset:(NSString*)asset
408 fromPackage:(NSString*)package
409 fromBundle:(nullable NSBundle*)bundle {
410 return [self lookupKeyForAsset:[NSString stringWithFormat:@"packages/%@/%@", package, asset]
411 fromBundle:bundle];
412}
413
414+ (NSString*)defaultBundleIdentifier {
415 return @"io.flutter.flutter.app";
416}
417
418- (BOOL)isWideGamutEnabled {
419 return _settings.enable_wide_gamut;
420}
421
422@end
NSBundle * FLTFrameworkBundleWithIdentifier(NSString *flutterFrameworkBundleID)
NSString * FLTAssetsPathFromBundle(NSBundle *bundle)
NSBundle * FLTGetApplicationBundle()
NSString * FLTAssetPath(NSBundle *bundle)
static bool IsRunningPrecompiledCode()
Checks if VM instances in the process can run precompiled code. This call can be made at any time and...
Definition dart_vm.cc:176
static RunConfiguration InferFromSettings(const Settings &settings, const fml::RefPtr< fml::TaskRunner > &io_worker=nullptr, IsolateLaunchType launch_type=IsolateLaunchType::kNewGroup)
Attempts to infer a run configuration from the settings object. This tries to create a run configurat...
static TaskQueueId GetCurrentTaskQueueId()
static MessageLoopTaskQueues * GetInstance()
void AddTaskObserver(TaskQueueId queue_id, intptr_t key, const fml::closure &callback)
void RemoveTaskObserver(TaskQueueId queue_id, intptr_t key)
VkDevice device
Definition main.cc:69
G_BEGIN_DECLS GBytes * message
const uint8_t uint32_t uint32_t GError ** error
FlutterDesktopBinaryReply callback
#define FML_CHECK(condition)
Definition logging.h:104
NSString * flutterAssetsName:(NSBundle *bundle)
static const char * kApplicationKernelSnapshotFileName
const intptr_t kPlatformStrongDillSize
FLUTTER_ASSERT_ARC const uint8_t kPlatformStrongDill[]
flutter::Settings FLTDefaultSettingsForBundle(NSBundle *bundle, NSProcessInfo *processInfoOrNil)
static BOOL DoesHardwareSupportWideGamut()
Settings SettingsFromCommandLine(const fml::CommandLine &command_line, bool require_merged_platform_ui_thread)
Definition switches.cc:230
fml::CommandLine CommandLineFromNSProcessInfo(NSProcessInfo *processInfoOrNil=nil)
constexpr double kMegaByteSizeInBytes
Definition constants.h:9
std::function< void()> closure
Definition closure.h:14
std::string lifecycle_state
int BOOL