// Copyright 2018 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. #include "flutter/shell/common/isolate_configuration.h" #include "flutter/runtime/dart_vm.h" #ifdef ERROR #undef ERROR #endif namespace shell { IsolateConfiguration::IsolateConfiguration() = default; IsolateConfiguration::~IsolateConfiguration() = default; bool IsolateConfiguration::PrepareIsolate(blink::DartIsolate& isolate) { if (isolate.GetPhase() != blink::DartIsolate::Phase::LibrariesSetup) { FML_DLOG(ERROR) << "Isolate was in incorrect phase to be prepared for running."; return false; } return DoPrepareIsolate(isolate); } class AppSnapshotIsolateConfiguration final : public IsolateConfiguration { public: AppSnapshotIsolateConfiguration() = default; // |shell::IsolateConfiguration| bool DoPrepareIsolate(blink::DartIsolate& isolate) override { return isolate.PrepareForRunningFromPrecompiledCode(); } private: FML_DISALLOW_COPY_AND_ASSIGN(AppSnapshotIsolateConfiguration); }; class SnapshotIsolateConfiguration : public IsolateConfiguration { public: SnapshotIsolateConfiguration(std::unique_ptr snapshot) : snapshot_(std::move(snapshot)) {} // |shell::IsolateConfiguration| bool DoPrepareIsolate(blink::DartIsolate& isolate) override { if (blink::DartVM::IsRunningPrecompiledCode()) { return false; } return isolate.PrepareForRunningFromSnapshot(std::move(snapshot_)); } private: std::unique_ptr snapshot_; FML_DISALLOW_COPY_AND_ASSIGN(SnapshotIsolateConfiguration); }; class SourceIsolateConfiguration final : public IsolateConfiguration { public: SourceIsolateConfiguration(std::string main_path, std::string packages_path) : main_path_(std::move(main_path)), packages_path_(std::move(packages_path)) {} // |shell::IsolateConfiguration| bool DoPrepareIsolate(blink::DartIsolate& isolate) override { if (blink::DartVM::IsRunningPrecompiledCode()) { return false; } return isolate.PrepareForRunningFromSource(std::move(main_path_), std::move(packages_path_)); } private: std::string main_path_; std::string packages_path_; FML_DISALLOW_COPY_AND_ASSIGN(SourceIsolateConfiguration); }; class KernelListIsolateConfiguration final : public IsolateConfiguration { public: KernelListIsolateConfiguration( std::vector> kernel_pieces) : kernel_pieces_(std::move(kernel_pieces)) {} // |shell::IsolateConfiguration| bool DoPrepareIsolate(blink::DartIsolate& isolate) override { if (blink::DartVM::IsRunningPrecompiledCode()) { return false; } for (size_t i = 0; i < kernel_pieces_.size(); i++) { bool last_piece = i + 1 == kernel_pieces_.size(); if (!isolate.PrepareForRunningFromSnapshot(std::move(kernel_pieces_[i]), last_piece)) { return false; } } return true; } private: std::vector> kernel_pieces_; FML_DISALLOW_COPY_AND_ASSIGN(KernelListIsolateConfiguration); }; std::unique_ptr IsolateConfiguration::InferFromSettings( const blink::Settings& settings, fml::RefPtr asset_manager) { // Running in AOT mode. if (blink::DartVM::IsRunningPrecompiledCode()) { return CreateForAppSnapshot(); } // Run from sources. { const auto& main = settings.main_dart_file_path; const auto& packages = settings.packages_file_path; if (main.size() != 0 && packages.size() != 0) { return CreateForSource(std::move(main), std::move(packages)); } } // Running from kernel snapshot. if (asset_manager) { std::unique_ptr kernel = asset_manager->GetAsMapping(settings.application_kernel_asset); if (kernel) { return CreateForSnapshot(std::move(kernel)); } } // Running from script snapshot. if (asset_manager) { std::unique_ptr script_snapshot = asset_manager->GetAsMapping(settings.script_snapshot_path); if (script_snapshot) { return CreateForSnapshot(std::move(script_snapshot)); } } // Running from kernel divided into several pieces (for sharing). // TODO(fuchsia): Use async blobfs API once it becomes available. if (asset_manager) { std::unique_ptr kernel_list = asset_manager->GetAsMapping(settings.application_kernel_list_asset); if (kernel_list) { const char* kernel_list_str = reinterpret_cast(kernel_list->GetMapping()); size_t kernel_list_size = kernel_list->GetSize(); std::vector> kernel_pieces; size_t piece_path_start = 0; while (piece_path_start < kernel_list_size) { size_t piece_path_end = piece_path_start; while ((piece_path_end < kernel_list_size) && (kernel_list_str[piece_path_end] != '\n')) { piece_path_end++; } std::string piece_path(&kernel_list_str[piece_path_start], piece_path_end - piece_path_start); std::unique_ptr piece = asset_manager->GetAsMapping(piece_path); if (piece == nullptr) { FML_LOG(ERROR) << "Failed to load: " << piece_path; return nullptr; } kernel_pieces.emplace_back(std::move(piece)); piece_path_start = piece_path_end + 1; } return CreateForKernelList(std::move(kernel_pieces)); } } return nullptr; } std::unique_ptr IsolateConfiguration::CreateForAppSnapshot() { return std::make_unique(); } std::unique_ptr IsolateConfiguration::CreateForSnapshot( std::unique_ptr snapshot) { return std::make_unique(std::move(snapshot)); } std::unique_ptr IsolateConfiguration::CreateForSource( std::string main_path, std::string packages_path) { return std::make_unique(std::move(main_path), std::move(packages_path)); } std::unique_ptr IsolateConfiguration::CreateForKernelList( std::vector> kernel_pieces) { return std::make_unique( std::move(kernel_pieces)); } } // namespace shell