mirror of
https://github.com/encounter/engine.git
synced 2026-03-30 11:09:55 -07:00
137 lines
4.3 KiB
C++
137 lines
4.3 KiB
C++
// Copyright 2015 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.
|
|
|
|
#include "flutter/common/threads.h"
|
|
#include "flutter/fml/message_loop.h"
|
|
#include "flutter/shell/common/platform_view.h"
|
|
#include "flutter/shell/common/shell.h"
|
|
#include "flutter/shell/common/switches.h"
|
|
#include "flutter/shell/testing/test_runner.h"
|
|
#include "flutter/shell/testing/testing.h"
|
|
#include "flutter/sky/engine/public/web/Sky.h"
|
|
#include "lib/fxl/command_line.h"
|
|
#include "lib/tonic/dart_microtask_queue.h"
|
|
#include "third_party/dart/runtime/bin/embedded_dart_io.h"
|
|
|
|
namespace {
|
|
|
|
// Exit codes used by the Dart command line tool.
|
|
const int kApiErrorExitCode = 253;
|
|
const int kCompilationErrorExitCode = 254;
|
|
const int kErrorExitCode = 255;
|
|
|
|
// Checks whether the engine's main Dart isolate has no pending work. If so,
|
|
// then exit the given message loop.
|
|
class ScriptCompletionTaskObserver : public fml::TaskObserver {
|
|
public:
|
|
ScriptCompletionTaskObserver(fxl::RefPtr<fxl::TaskRunner> task_runner)
|
|
: main_task_runner_(std::move(task_runner)),
|
|
prev_live_(false),
|
|
last_error_(tonic::kNoError) {}
|
|
|
|
void DidProcessTask() override {
|
|
shell::TestRunner& test_runner = shell::TestRunner::Shared();
|
|
shell::Engine& engine = test_runner.platform_view().engine();
|
|
|
|
if (engine.GetLoadScriptError() != tonic::kNoError) {
|
|
last_error_ = engine.GetLoadScriptError();
|
|
main_task_runner_->PostTask(
|
|
[]() { fml::MessageLoop::GetCurrent().Terminate(); });
|
|
return;
|
|
}
|
|
|
|
bool live = engine.UIIsolateHasLivePorts();
|
|
if (prev_live_ && !live) {
|
|
last_error_ = engine.GetUIIsolateLastError();
|
|
main_task_runner_->PostTask(
|
|
[]() { fml::MessageLoop::GetCurrent().Terminate(); });
|
|
}
|
|
prev_live_ = live;
|
|
}
|
|
|
|
tonic::DartErrorHandleType last_error() { return last_error_; }
|
|
|
|
private:
|
|
fxl::RefPtr<fxl::TaskRunner> main_task_runner_;
|
|
bool prev_live_;
|
|
tonic::DartErrorHandleType last_error_;
|
|
};
|
|
|
|
int ConvertErrorTypeToExitCode(tonic::DartErrorHandleType error) {
|
|
switch (error) {
|
|
case tonic::kCompilationErrorType:
|
|
return kCompilationErrorExitCode;
|
|
case tonic::kApiErrorType:
|
|
return kApiErrorExitCode;
|
|
case tonic::kUnknownErrorType:
|
|
return kErrorExitCode;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void RunNonInteractive(fxl::CommandLine initial_command_line,
|
|
bool run_forever) {
|
|
// This is a platform thread (i.e not one created by fml::Thread), so perform
|
|
// one time initialization.
|
|
fml::MessageLoop::EnsureInitializedForCurrentThread();
|
|
|
|
shell::Shell::InitStandalone(initial_command_line);
|
|
|
|
// Note that this task observer must be added after the observer that drains
|
|
// the microtask queue.
|
|
ScriptCompletionTaskObserver task_observer(
|
|
fml::MessageLoop::GetCurrent().GetTaskRunner());
|
|
if (!run_forever) {
|
|
blink::Threads::UI()->PostTask([&task_observer] {
|
|
fml::MessageLoop::GetCurrent().AddTaskObserver(&task_observer);
|
|
});
|
|
}
|
|
|
|
if (!shell::InitForTesting(initial_command_line)) {
|
|
shell::PrintUsage("flutter_tester");
|
|
exit(EXIT_FAILURE);
|
|
return;
|
|
}
|
|
|
|
fml::MessageLoop::GetCurrent().Run();
|
|
|
|
shell::TestRunner& test_runner = shell::TestRunner::Shared();
|
|
tonic::DartErrorHandleType error =
|
|
test_runner.platform_view().engine().GetLoadScriptError();
|
|
if (error == tonic::kNoError)
|
|
error = task_observer.last_error();
|
|
if (error == tonic::kNoError) {
|
|
fxl::AutoResetWaitableEvent latch;
|
|
blink::Threads::UI()->PostTask([&error, &latch] {
|
|
error = tonic::DartMicrotaskQueue::GetForCurrentThread()->GetLastError();
|
|
latch.Signal();
|
|
});
|
|
latch.Wait();
|
|
}
|
|
|
|
// The script has completed and the engine may not be in a clean state,
|
|
// so just stop the process.
|
|
exit(ConvertErrorTypeToExitCode(error));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
int main(int argc, char* argv[]) {
|
|
dart::bin::SetExecutableName(argv[0]);
|
|
dart::bin::SetExecutableArguments(argc - 1, argv);
|
|
|
|
auto command_line = fxl::CommandLineFromArgcArgv(argc, argv);
|
|
|
|
if (command_line.HasOption(shell::FlagForSwitch(shell::Switch::Help))) {
|
|
shell::PrintUsage("flutter_tester");
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
bool run_forever =
|
|
command_line.HasOption(shell::FlagForSwitch(shell::Switch::RunForever));
|
|
RunNonInteractive(std::move(command_line), run_forever);
|
|
return EXIT_SUCCESS;
|
|
}
|