mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-01-30 11:04:52 +01:00
Remove main runner abstractions (see #3685)
Remove code abstractions that are no longer required after deletion of the Alloy bootstrap. This is a functional no-op.
This commit is contained in:
parent
231701d98b
commit
c0a3c897cf
6
BUILD.gn
6
BUILD.gn
@ -687,6 +687,8 @@ source_set("libcef_static") {
|
|||||||
"libcef/browser/trace_subscriber.cc",
|
"libcef/browser/trace_subscriber.cc",
|
||||||
"libcef/browser/trace_subscriber.h",
|
"libcef/browser/trace_subscriber.h",
|
||||||
"libcef/browser/thread_util.h",
|
"libcef/browser/thread_util.h",
|
||||||
|
"libcef/browser/ui_thread.cc",
|
||||||
|
"libcef/browser/ui_thread.h",
|
||||||
"libcef/browser/views/basic_label_button_impl.cc",
|
"libcef/browser/views/basic_label_button_impl.cc",
|
||||||
"libcef/browser/views/basic_label_button_impl.h",
|
"libcef/browser/views/basic_label_button_impl.h",
|
||||||
"libcef/browser/views/basic_label_button_view.cc",
|
"libcef/browser/views/basic_label_button_view.cc",
|
||||||
@ -768,8 +770,6 @@ source_set("libcef_static") {
|
|||||||
"libcef/common/chrome/chrome_content_client_cef.h",
|
"libcef/common/chrome/chrome_content_client_cef.h",
|
||||||
"libcef/common/chrome/chrome_main_delegate_cef.cc",
|
"libcef/common/chrome/chrome_main_delegate_cef.cc",
|
||||||
"libcef/common/chrome/chrome_main_delegate_cef.h",
|
"libcef/common/chrome/chrome_main_delegate_cef.h",
|
||||||
"libcef/common/chrome/chrome_main_runner_delegate.cc",
|
|
||||||
"libcef/common/chrome/chrome_main_runner_delegate.h",
|
|
||||||
"libcef/common/command_line_impl.cc",
|
"libcef/common/command_line_impl.cc",
|
||||||
"libcef/common/command_line_impl.h",
|
"libcef/common/command_line_impl.h",
|
||||||
"libcef/common/crash_reporter_client.cc",
|
"libcef/common/crash_reporter_client.cc",
|
||||||
@ -783,8 +783,6 @@ source_set("libcef_static") {
|
|||||||
"libcef/common/frame_util.h",
|
"libcef/common/frame_util.h",
|
||||||
"libcef/common/i18n_util_impl.cc",
|
"libcef/common/i18n_util_impl.cc",
|
||||||
"libcef/common/json_impl.cc",
|
"libcef/common/json_impl.cc",
|
||||||
"libcef/common/main_runner_delegate.h",
|
|
||||||
"libcef/common/main_runner_handler.h",
|
|
||||||
"libcef/common/net/http_header_utils.cc",
|
"libcef/common/net/http_header_utils.cc",
|
||||||
"libcef/common/net/http_header_utils.h",
|
"libcef/common/net/http_header_utils.h",
|
||||||
"libcef/common/net/scheme_registration.cc",
|
"libcef/common/net/scheme_registration.cc",
|
||||||
|
@ -5,40 +5,31 @@
|
|||||||
|
|
||||||
#include "cef/libcef/browser/main_runner.h"
|
#include "cef/libcef/browser/main_runner.h"
|
||||||
|
|
||||||
#include "base/at_exit.h"
|
|
||||||
#include "base/base_switches.h"
|
#include "base/base_switches.h"
|
||||||
#include "base/command_line.h"
|
#include "base/command_line.h"
|
||||||
#include "base/debug/debugger.h"
|
#include "base/debug/debugger.h"
|
||||||
#include "base/memory/raw_ptr.h"
|
#include "base/memory/raw_ptr.h"
|
||||||
#include "base/run_loop.h"
|
#include "base/run_loop.h"
|
||||||
#include "base/sequence_checker.h"
|
|
||||||
#include "base/synchronization/lock.h"
|
|
||||||
#include "base/synchronization/waitable_event.h"
|
#include "base/synchronization/waitable_event.h"
|
||||||
#include "base/threading/thread.h"
|
|
||||||
#include "cef/libcef/browser/browser_message_loop.h"
|
#include "cef/libcef/browser/browser_message_loop.h"
|
||||||
|
#include "cef/libcef/browser/chrome/chrome_content_browser_client_cef.h"
|
||||||
#include "cef/libcef/browser/thread_util.h"
|
#include "cef/libcef/browser/thread_util.h"
|
||||||
|
#include "cef/libcef/common/app_manager.h"
|
||||||
#include "cef/libcef/common/cef_switches.h"
|
#include "cef/libcef/common/cef_switches.h"
|
||||||
#include "cef/libcef/common/chrome/chrome_main_runner_delegate.h"
|
#include "chrome/browser/browser_process_impl.h"
|
||||||
|
#include "chrome/browser/chrome_process_singleton.h"
|
||||||
#include "chrome/common/chrome_result_codes.h"
|
#include "chrome/common/chrome_result_codes.h"
|
||||||
#include "components/crash/core/app/crash_switches.h"
|
#include "components/crash/core/app/crash_switches.h"
|
||||||
|
#include "components/keep_alive_registry/keep_alive_types.h"
|
||||||
|
#include "components/metrics/persistent_system_profile.h"
|
||||||
#include "content/app/content_main_runner_impl.h"
|
#include "content/app/content_main_runner_impl.h"
|
||||||
#include "content/browser/scheduler/browser_task_executor.h"
|
#include "content/browser/scheduler/browser_task_executor.h"
|
||||||
#include "content/public/app/content_main.h"
|
#include "content/public/app/content_main.h"
|
||||||
#include "content/public/app/content_main_runner.h"
|
|
||||||
#include "content/public/browser/render_process_host.h"
|
#include "content/public/browser/render_process_host.h"
|
||||||
#include "content/public/common/content_switches.h"
|
#include "content/public/common/content_switches.h"
|
||||||
#include "third_party/crashpad/crashpad/handler/handler_main.h"
|
#include "third_party/crashpad/crashpad/handler/handler_main.h"
|
||||||
|
|
||||||
#if BUILDFLAG(IS_LINUX)
|
|
||||||
#include "ui/base/ozone_buildflags.h"
|
|
||||||
#if BUILDFLAG(IS_OZONE_X11)
|
|
||||||
#include "ui/ozone/platform/x11/ozone_platform_x11.h"
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if BUILDFLAG(IS_WIN)
|
#if BUILDFLAG(IS_WIN)
|
||||||
#include <Objbase.h>
|
|
||||||
|
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
@ -49,20 +40,7 @@
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
enum class RuntimeType {
|
bool g_initialized = false;
|
||||||
UNINITIALIZED,
|
|
||||||
CHROME,
|
|
||||||
};
|
|
||||||
RuntimeType g_runtime_type = RuntimeType::UNINITIALIZED;
|
|
||||||
|
|
||||||
std::unique_ptr<CefMainRunnerDelegate> MakeDelegate(
|
|
||||||
CefMainRunnerHandler* runner,
|
|
||||||
CefSettings* settings,
|
|
||||||
CefRefPtr<CefApp> application) {
|
|
||||||
g_runtime_type = RuntimeType::CHROME;
|
|
||||||
return std::make_unique<ChromeMainRunnerDelegate>(runner, settings,
|
|
||||||
application);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Based on components/crash/core/app/run_as_crashpad_handler_win.cc
|
// Based on components/crash/core/app/run_as_crashpad_handler_win.cc
|
||||||
// Remove the "--type=crashpad-handler" command-line flag that will otherwise
|
// Remove the "--type=crashpad-handler" command-line flag that will otherwise
|
||||||
@ -110,148 +88,20 @@ int RunAsCrashpadHandler(const base::CommandLine& command_line) {
|
|||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
// Used to run the UI on a separate thread.
|
|
||||||
class CefUIThread : public base::PlatformThread::Delegate {
|
|
||||||
public:
|
|
||||||
CefUIThread(CefMainRunner* runner, base::OnceClosure setup_callback)
|
|
||||||
: runner_(runner), setup_callback_(std::move(setup_callback)) {}
|
|
||||||
~CefUIThread() override { Stop(); }
|
|
||||||
|
|
||||||
void Start() {
|
|
||||||
base::AutoLock lock(thread_lock_);
|
|
||||||
bool success = base::PlatformThread::CreateWithType(
|
|
||||||
0, this, &thread_, base::ThreadType::kDefault);
|
|
||||||
if (!success) {
|
|
||||||
LOG(FATAL) << "failed to UI create thread";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Stop() {
|
|
||||||
base::AutoLock lock(thread_lock_);
|
|
||||||
|
|
||||||
if (!stopping_) {
|
|
||||||
stopping_ = true;
|
|
||||||
CEF_POST_TASK(CEF_UIT, base::BindOnce(&CefMainRunner::QuitMessageLoop,
|
|
||||||
base::Unretained(runner_)));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Can't join if the |thread_| is either already gone or is non-joinable.
|
|
||||||
if (thread_.is_null()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
base::PlatformThread::Join(thread_);
|
|
||||||
thread_ = base::PlatformThreadHandle();
|
|
||||||
|
|
||||||
stopping_ = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WaitUntilThreadStarted() const {
|
|
||||||
DCHECK(owning_sequence_checker_.CalledOnValidSequence());
|
|
||||||
start_event_.Wait();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void InitializeBrowserRunner(
|
|
||||||
content::MainFunctionParams main_function_params) {
|
|
||||||
#if BUILDFLAG(IS_LINUX)
|
|
||||||
#if BUILDFLAG(IS_OZONE_X11)
|
|
||||||
// Disable creation of GtkUi (interface to GTK desktop features) and cause
|
|
||||||
// ui::GetDefaultLinuxUi() (and related functions) to return nullptr. We
|
|
||||||
// can't use GtkUi in combination with multi-threaded-message-loop because
|
|
||||||
// Chromium's GTK implementation doesn't use GDK threads. Light/dark theme
|
|
||||||
// changes will still be detected via DarkModeManagerLinux.
|
|
||||||
ui::SetMultiThreadedMessageLoopX11();
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Use our own browser process runner.
|
|
||||||
browser_runner_ = content::BrowserMainRunner::Create();
|
|
||||||
|
|
||||||
// Initialize browser process state. Uses the current thread's message loop.
|
|
||||||
int exit_code =
|
|
||||||
browser_runner_->Initialize(std::move(main_function_params));
|
|
||||||
CHECK_EQ(exit_code, -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_shutdown_callback(base::OnceClosure shutdown_callback) {
|
|
||||||
shutdown_callback_ = std::move(shutdown_callback);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void ThreadMain() override {
|
|
||||||
base::PlatformThread::SetName("CefUIThread");
|
|
||||||
|
|
||||||
#if BUILDFLAG(IS_WIN)
|
|
||||||
// Initializes the COM library on the current thread.
|
|
||||||
CoInitialize(nullptr);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
start_event_.Signal();
|
|
||||||
|
|
||||||
std::move(setup_callback_).Run();
|
|
||||||
|
|
||||||
runner_->RunMessageLoop();
|
|
||||||
|
|
||||||
// Stop may be called before InitializeBrowserRunner if
|
|
||||||
// content::ContentMainRun was not successful (for example, due to process
|
|
||||||
// singleton relaunch).
|
|
||||||
if (browser_runner_) {
|
|
||||||
browser_runner_->Shutdown();
|
|
||||||
browser_runner_.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
// This will be a no-op if there is no BrowserTaskExecutor.
|
|
||||||
content::BrowserTaskExecutor::Shutdown();
|
|
||||||
|
|
||||||
if (!shutdown_callback_.is_null()) {
|
|
||||||
std::move(shutdown_callback_).Run();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run exit callbacks on the UI thread to avoid sequence check failures.
|
|
||||||
base::AtExitManager::ProcessCallbacksNow();
|
|
||||||
|
|
||||||
#if BUILDFLAG(IS_WIN)
|
|
||||||
// Closes the COM library on the current thread. CoInitialize must
|
|
||||||
// be balanced by a corresponding call to CoUninitialize.
|
|
||||||
CoUninitialize();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
const raw_ptr<CefMainRunner> runner_;
|
|
||||||
base::OnceClosure setup_callback_;
|
|
||||||
base::OnceClosure shutdown_callback_;
|
|
||||||
|
|
||||||
std::unique_ptr<content::BrowserMainRunner> browser_runner_;
|
|
||||||
|
|
||||||
bool stopping_ = false;
|
|
||||||
|
|
||||||
// The thread's handle.
|
|
||||||
base::PlatformThreadHandle thread_;
|
|
||||||
mutable base::Lock thread_lock_; // Protects |thread_|.
|
|
||||||
|
|
||||||
mutable base::WaitableEvent start_event_;
|
|
||||||
|
|
||||||
// This class is not thread-safe, use this to verify access from the owning
|
|
||||||
// sequence of the Thread.
|
|
||||||
base::SequenceChecker owning_sequence_checker_;
|
|
||||||
};
|
|
||||||
|
|
||||||
CefMainRunner::CefMainRunner(bool multi_threaded_message_loop,
|
CefMainRunner::CefMainRunner(bool multi_threaded_message_loop,
|
||||||
bool external_message_pump)
|
bool external_message_pump)
|
||||||
: multi_threaded_message_loop_(multi_threaded_message_loop),
|
: multi_threaded_message_loop_(multi_threaded_message_loop),
|
||||||
external_message_pump_(external_message_pump) {}
|
external_message_pump_(external_message_pump) {}
|
||||||
|
|
||||||
CefMainRunner::~CefMainRunner() = default;
|
|
||||||
|
|
||||||
bool CefMainRunner::Initialize(CefSettings* settings,
|
bool CefMainRunner::Initialize(CefSettings* settings,
|
||||||
CefRefPtr<CefApp> application,
|
CefRefPtr<CefApp> application,
|
||||||
const CefMainArgs& args,
|
const CefMainArgs& args,
|
||||||
void* windows_sandbox_info,
|
void* windows_sandbox_info,
|
||||||
bool* initialized,
|
bool* initialized,
|
||||||
base::OnceClosure context_initialized) {
|
base::OnceClosure context_initialized) {
|
||||||
DCHECK(!main_delegate_);
|
g_initialized = true;
|
||||||
main_delegate_ = MakeDelegate(this, settings, application);
|
settings_ = settings;
|
||||||
|
application_ = application;
|
||||||
|
|
||||||
exit_code_ =
|
exit_code_ =
|
||||||
ContentMainInitialize(args, windows_sandbox_info, &settings->no_sandbox);
|
ContentMainInitialize(args, windows_sandbox_info, &settings->no_sandbox);
|
||||||
@ -294,8 +144,6 @@ void CefMainRunner::Shutdown(base::OnceClosure shutdown_on_ui_thread,
|
|||||||
ui_thread_.reset();
|
ui_thread_.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
main_delegate_->BeforeMainThreadShutdown();
|
|
||||||
|
|
||||||
if (!multi_threaded_message_loop_) {
|
if (!multi_threaded_message_loop_) {
|
||||||
// Main thread and UI thread are the same.
|
// Main thread and UI thread are the same.
|
||||||
StartShutdownOnUIThread(std::move(shutdown_on_ui_thread));
|
StartShutdownOnUIThread(std::move(shutdown_on_ui_thread));
|
||||||
@ -312,10 +160,14 @@ void CefMainRunner::Shutdown(base::OnceClosure shutdown_on_ui_thread,
|
|||||||
main_runner_.reset();
|
main_runner_.reset();
|
||||||
|
|
||||||
std::move(finalize_shutdown).Run();
|
std::move(finalize_shutdown).Run();
|
||||||
main_delegate_->AfterMainThreadShutdown();
|
|
||||||
|
|
||||||
main_delegate_.reset();
|
main_delegate_.reset();
|
||||||
g_runtime_type = RuntimeType::UNINITIALIZED;
|
sampling_profiler_.reset();
|
||||||
|
keep_alive_.reset();
|
||||||
|
settings_ = nullptr;
|
||||||
|
application_ = nullptr;
|
||||||
|
|
||||||
|
g_initialized = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CefMainRunner::RunMessageLoop() {
|
void CefMainRunner::RunMessageLoop() {
|
||||||
@ -324,7 +176,21 @@ void CefMainRunner::RunMessageLoop() {
|
|||||||
DCHECK(quit_callback_.is_null());
|
DCHECK(quit_callback_.is_null());
|
||||||
quit_callback_ = run_loop.QuitClosure();
|
quit_callback_ = run_loop.QuitClosure();
|
||||||
|
|
||||||
main_delegate_->BeforeMainMessageLoopRun(&run_loop);
|
// May be nullptr if content::ContentMainRun exits early.
|
||||||
|
if (g_browser_process) {
|
||||||
|
// The ScopedKeepAlive instance triggers shutdown logic when released on the
|
||||||
|
// UI thread before terminating the message loop (e.g. from
|
||||||
|
// CefQuitMessageLoop or FinishShutdownOnUIThread when running with
|
||||||
|
// multi-threaded message loop).
|
||||||
|
keep_alive_ = std::make_unique<ScopedKeepAlive>(
|
||||||
|
KeepAliveOrigin::APP_CONTROLLER, KeepAliveRestartOption::DISABLED);
|
||||||
|
|
||||||
|
// The QuitClosure will be executed from BrowserProcessImpl::Unpin() via
|
||||||
|
// KeepAliveRegistry when the last ScopedKeepAlive is released.
|
||||||
|
// ScopedKeepAlives are also held by Browser objects.
|
||||||
|
static_cast<BrowserProcessImpl*>(g_browser_process)
|
||||||
|
->SetQuitClosure(run_loop.QuitClosure());
|
||||||
|
}
|
||||||
|
|
||||||
// Blocks until QuitMessageLoop() is called.
|
// Blocks until QuitMessageLoop() is called.
|
||||||
run_loop.Run();
|
run_loop.Run();
|
||||||
@ -332,7 +198,7 @@ void CefMainRunner::RunMessageLoop() {
|
|||||||
|
|
||||||
void CefMainRunner::QuitMessageLoop() {
|
void CefMainRunner::QuitMessageLoop() {
|
||||||
if (!quit_callback_.is_null()) {
|
if (!quit_callback_.is_null()) {
|
||||||
if (main_delegate_->HandleMainMessageLoopQuit()) {
|
if (HandleMainMessageLoopQuit()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
std::move(quit_callback_).Run();
|
std::move(quit_callback_).Run();
|
||||||
@ -363,21 +229,18 @@ int CefMainRunner::RunAsHelperProcess(const CefMainArgs& args,
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto main_delegate = MakeDelegate(
|
g_initialized = true;
|
||||||
/*runner=*/nullptr, /*settings=*/nullptr, application);
|
|
||||||
main_delegate->BeforeExecuteProcess(args);
|
|
||||||
|
|
||||||
int result;
|
auto main_delegate = std::make_unique<ChromeMainDelegateCef>(
|
||||||
|
/*runner=*/nullptr, /*settings=*/nullptr, application);
|
||||||
|
BeforeMainInitialize(args);
|
||||||
|
|
||||||
if (process_type == crash_reporter::switches::kCrashpadHandler) {
|
if (process_type == crash_reporter::switches::kCrashpadHandler) {
|
||||||
result = RunAsCrashpadHandler(command_line);
|
return RunAsCrashpadHandler(command_line);
|
||||||
main_delegate->AfterExecuteProcess();
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Execute the secondary process.
|
// Execute the secondary process.
|
||||||
content::ContentMainParams main_params(
|
content::ContentMainParams main_params(main_delegate.get());
|
||||||
main_delegate->GetContentMainDelegate());
|
|
||||||
#if BUILDFLAG(IS_WIN)
|
#if BUILDFLAG(IS_WIN)
|
||||||
sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
|
sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
|
||||||
if (windows_sandbox_info == nullptr) {
|
if (windows_sandbox_info == nullptr) {
|
||||||
@ -392,22 +255,20 @@ int CefMainRunner::RunAsHelperProcess(const CefMainArgs& args,
|
|||||||
main_params.argc = args.argc;
|
main_params.argc = args.argc;
|
||||||
main_params.argv = const_cast<const char**>(args.argv);
|
main_params.argv = const_cast<const char**>(args.argv);
|
||||||
#endif
|
#endif
|
||||||
result = content::ContentMain(std::move(main_params));
|
return content::ContentMain(std::move(main_params));
|
||||||
|
|
||||||
main_delegate->AfterExecuteProcess();
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int CefMainRunner::ContentMainInitialize(const CefMainArgs& args,
|
int CefMainRunner::ContentMainInitialize(const CefMainArgs& args,
|
||||||
void* windows_sandbox_info,
|
void* windows_sandbox_info,
|
||||||
int* no_sandbox) {
|
int* no_sandbox) {
|
||||||
main_delegate_->BeforeMainThreadInitialize(args);
|
BeforeMainInitialize(args);
|
||||||
|
|
||||||
|
main_delegate_ =
|
||||||
|
std::make_unique<ChromeMainDelegateCef>(this, settings_, application_);
|
||||||
|
|
||||||
// Initialize the content runner.
|
// Initialize the content runner.
|
||||||
main_runner_ = content::ContentMainRunner::Create();
|
main_runner_ = content::ContentMainRunner::Create();
|
||||||
content::ContentMainParams main_params(
|
content::ContentMainParams main_params(main_delegate_.get());
|
||||||
main_delegate_->GetContentMainDelegate());
|
|
||||||
|
|
||||||
#if BUILDFLAG(IS_WIN)
|
#if BUILDFLAG(IS_WIN)
|
||||||
sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
|
sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
|
||||||
@ -430,7 +291,12 @@ int CefMainRunner::ContentMainInitialize(const CefMainArgs& args,
|
|||||||
|
|
||||||
int CefMainRunner::ContentMainRun(bool* initialized,
|
int CefMainRunner::ContentMainRun(bool* initialized,
|
||||||
base::OnceClosure context_initialized) {
|
base::OnceClosure context_initialized) {
|
||||||
main_delegate_->BeforeMainThreadRun(multi_threaded_message_loop_);
|
if (multi_threaded_message_loop_) {
|
||||||
|
// Detach from the main thread so that these objects can be attached and
|
||||||
|
// modified from the UI thread going forward.
|
||||||
|
metrics::GlobalPersistentSystemProfile::GetInstance()
|
||||||
|
->DetachFromCurrentThread();
|
||||||
|
}
|
||||||
|
|
||||||
int exit_code = -1;
|
int exit_code = -1;
|
||||||
|
|
||||||
@ -446,7 +312,7 @@ int CefMainRunner::ContentMainRun(bool* initialized,
|
|||||||
if (!CreateUIThread(base::BindOnce(
|
if (!CreateUIThread(base::BindOnce(
|
||||||
[](CefMainRunner* runner, base::WaitableEvent* event,
|
[](CefMainRunner* runner, base::WaitableEvent* event,
|
||||||
int* exit_code) {
|
int* exit_code) {
|
||||||
runner->main_delegate_->BeforeUIThreadInitialize();
|
runner->BeforeUIThreadInitialize();
|
||||||
*exit_code = content::ContentMainRun(runner->main_runner_.get());
|
*exit_code = content::ContentMainRun(runner->main_runner_.get());
|
||||||
|
|
||||||
if (*exit_code != content::RESULT_CODE_NORMAL_EXIT) {
|
if (*exit_code != content::RESULT_CODE_NORMAL_EXIT) {
|
||||||
@ -473,7 +339,7 @@ int CefMainRunner::ContentMainRun(bool* initialized,
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
*initialized = true;
|
*initialized = true;
|
||||||
main_delegate_->BeforeUIThreadInitialize();
|
BeforeUIThreadInitialize();
|
||||||
exit_code = content::ContentMainRun(main_runner_.get());
|
exit_code = content::ContentMainRun(main_runner_.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -493,6 +359,30 @@ int CefMainRunner::ContentMainRun(bool* initialized,
|
|||||||
return exit_code;
|
return exit_code;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
void CefMainRunner::BeforeMainInitialize(const CefMainArgs& args) {
|
||||||
|
#if BUILDFLAG(IS_WIN)
|
||||||
|
base::CommandLine::Init(0, nullptr);
|
||||||
|
#else
|
||||||
|
base::CommandLine::Init(args.argc, args.argv);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CefMainRunner::HandleMainMessageLoopQuit() {
|
||||||
|
// May be nullptr if content::ContentMainRun exits early.
|
||||||
|
if (!g_browser_process) {
|
||||||
|
// Proceed with direct execution of the QuitClosure().
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// May be called multiple times. See comments in RunMainMessageLoopBefore.
|
||||||
|
keep_alive_.reset();
|
||||||
|
|
||||||
|
// Cancel direct execution of the QuitClosure() in QuitMessageLoop. We
|
||||||
|
// instead wait for all Chrome browser windows to exit.
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
void CefMainRunner::PreBrowserMain() {
|
void CefMainRunner::PreBrowserMain() {
|
||||||
if (external_message_pump_) {
|
if (external_message_pump_) {
|
||||||
InitExternalMessagePumpFactoryForUI();
|
InitExternalMessagePumpFactoryForUI();
|
||||||
@ -506,8 +396,7 @@ int CefMainRunner::RunMainProcess(
|
|||||||
browser_runner_ = content::BrowserMainRunner::Create();
|
browser_runner_ = content::BrowserMainRunner::Create();
|
||||||
|
|
||||||
// Initialize browser process state. Results in a call to
|
// Initialize browser process state. Results in a call to
|
||||||
// AlloyBrowserMain::PreBrowserMain() which creates the UI message
|
// PreBrowserMain() which creates the UI message loop.
|
||||||
// loop.
|
|
||||||
int exit_code =
|
int exit_code =
|
||||||
browser_runner_->Initialize(std::move(main_function_params));
|
browser_runner_->Initialize(std::move(main_function_params));
|
||||||
if (exit_code >= 0) {
|
if (exit_code >= 0) {
|
||||||
@ -539,7 +428,6 @@ void CefMainRunner::OnContextInitialized(
|
|||||||
base::OnceClosure context_initialized) {
|
base::OnceClosure context_initialized) {
|
||||||
CEF_REQUIRE_UIT();
|
CEF_REQUIRE_UIT();
|
||||||
|
|
||||||
main_delegate_->AfterUIThreadInitialize();
|
|
||||||
std::move(context_initialized).Run();
|
std::move(context_initialized).Run();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -561,18 +449,35 @@ void CefMainRunner::StartShutdownOnUIThread(
|
|||||||
->ShutdownOnUIThread();
|
->ShutdownOnUIThread();
|
||||||
|
|
||||||
std::move(shutdown_on_ui_thread).Run();
|
std::move(shutdown_on_ui_thread).Run();
|
||||||
main_delegate_->BeforeUIThreadShutdown();
|
BeforeUIThreadShutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
void CefMainRunner::FinishShutdownOnUIThread() {
|
void CefMainRunner::FinishShutdownOnUIThread() {
|
||||||
main_delegate_->AfterUIThreadShutdown();
|
if (multi_threaded_message_loop_) {
|
||||||
|
// Don't wait for this to be called in ChromeMainDelegate::ProcessExiting.
|
||||||
|
// It is safe to call multiple times.
|
||||||
|
ChromeProcessSingleton::DeleteInstance();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CefMainRunner::BeforeUIThreadInitialize() {
|
||||||
|
sampling_profiler_ = std::make_unique<MainThreadStackSamplingProfiler>();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CefMainRunner::BeforeUIThreadShutdown() {
|
||||||
|
static_cast<ChromeContentBrowserClientCef*>(
|
||||||
|
CefAppManager::Get()->GetContentClient()->browser())
|
||||||
|
->CleanupOnUIThread();
|
||||||
|
main_delegate_->CleanupOnUIThread();
|
||||||
|
|
||||||
|
sampling_profiler_.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
// From libcef/features/runtime.h:
|
// From libcef/features/runtime.h:
|
||||||
namespace cef {
|
namespace cef {
|
||||||
|
|
||||||
bool IsChromeRuntimeEnabled() {
|
bool IsChromeRuntimeEnabled() {
|
||||||
return g_runtime_type == RuntimeType::CHROME;
|
return g_initialized;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace cef
|
} // namespace cef
|
||||||
|
@ -7,32 +7,30 @@
|
|||||||
#define CEF_LIBCEF_BROWSER_MAIN_RUNNER_H_
|
#define CEF_LIBCEF_BROWSER_MAIN_RUNNER_H_
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#include "base/functional/callback.h"
|
#include "base/functional/callback.h"
|
||||||
|
#include "base/memory/raw_ptr.h"
|
||||||
#include "cef/include/cef_app.h"
|
#include "cef/include/cef_app.h"
|
||||||
#include "cef/libcef/common/main_runner_delegate.h"
|
#include "cef/libcef/browser/ui_thread.h"
|
||||||
#include "cef/libcef/common/main_runner_handler.h"
|
#include "cef/libcef/common/chrome/chrome_main_delegate_cef.h"
|
||||||
|
#include "chrome/common/profiler/main_thread_stack_sampling_profiler.h"
|
||||||
|
#include "components/keep_alive_registry/scoped_keep_alive.h"
|
||||||
|
#include "content/public/app/content_main_runner.h"
|
||||||
#include "content/public/browser/browser_main_runner.h"
|
#include "content/public/browser/browser_main_runner.h"
|
||||||
|
|
||||||
namespace base {
|
namespace base {
|
||||||
class WaitableEvent;
|
class WaitableEvent;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace content {
|
|
||||||
class ContentMainRunner;
|
|
||||||
} // namespace content
|
|
||||||
|
|
||||||
class CefUIThread;
|
|
||||||
|
|
||||||
// Manages the main process lifespan and related objects.
|
// Manages the main process lifespan and related objects.
|
||||||
class CefMainRunner : public CefMainRunnerHandler {
|
class CefMainRunner final {
|
||||||
public:
|
public:
|
||||||
CefMainRunner(bool multi_threaded_message_loop, bool external_message_pump);
|
CefMainRunner(bool multi_threaded_message_loop, bool external_message_pump);
|
||||||
|
|
||||||
CefMainRunner(const CefMainRunner&) = delete;
|
CefMainRunner(const CefMainRunner&) = delete;
|
||||||
CefMainRunner& operator=(const CefMainRunner&) = delete;
|
CefMainRunner& operator=(const CefMainRunner&) = delete;
|
||||||
|
|
||||||
~CefMainRunner() override;
|
|
||||||
|
|
||||||
// Called from CefContext::Initialize.
|
// Called from CefContext::Initialize.
|
||||||
bool Initialize(CefSettings* settings,
|
bool Initialize(CefSettings* settings,
|
||||||
CefRefPtr<CefApp> application,
|
CefRefPtr<CefApp> application,
|
||||||
@ -56,6 +54,10 @@ class CefMainRunner : public CefMainRunnerHandler {
|
|||||||
CefRefPtr<CefApp> application,
|
CefRefPtr<CefApp> application,
|
||||||
void* windows_sandbox_info);
|
void* windows_sandbox_info);
|
||||||
|
|
||||||
|
// Called from ChromeMainDelegateCef.
|
||||||
|
void PreBrowserMain();
|
||||||
|
int RunMainProcess(content::MainFunctionParams main_function_params);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Called from Initialize().
|
// Called from Initialize().
|
||||||
int ContentMainInitialize(const CefMainArgs& args,
|
int ContentMainInitialize(const CefMainArgs& args,
|
||||||
@ -63,9 +65,8 @@ class CefMainRunner : public CefMainRunnerHandler {
|
|||||||
int* no_sandbox);
|
int* no_sandbox);
|
||||||
int ContentMainRun(bool* initialized, base::OnceClosure context_initialized);
|
int ContentMainRun(bool* initialized, base::OnceClosure context_initialized);
|
||||||
|
|
||||||
// CefMainRunnerHandler methods:
|
static void BeforeMainInitialize(const CefMainArgs& args);
|
||||||
void PreBrowserMain() override;
|
bool HandleMainMessageLoopQuit();
|
||||||
int RunMainProcess(content::MainFunctionParams main_function_params) override;
|
|
||||||
|
|
||||||
// Create the UI thread when running with multi-threaded message loop mode.
|
// Create the UI thread when running with multi-threaded message loop mode.
|
||||||
bool CreateUIThread(base::OnceClosure setup_callback);
|
bool CreateUIThread(base::OnceClosure setup_callback);
|
||||||
@ -81,10 +82,12 @@ class CefMainRunner : public CefMainRunnerHandler {
|
|||||||
// thread RunLoop has stopped and before running exit callbacks.
|
// thread RunLoop has stopped and before running exit callbacks.
|
||||||
void FinishShutdownOnUIThread();
|
void FinishShutdownOnUIThread();
|
||||||
|
|
||||||
|
void BeforeUIThreadInitialize();
|
||||||
|
void BeforeUIThreadShutdown();
|
||||||
|
|
||||||
const bool multi_threaded_message_loop_;
|
const bool multi_threaded_message_loop_;
|
||||||
const bool external_message_pump_;
|
const bool external_message_pump_;
|
||||||
|
|
||||||
std::unique_ptr<CefMainRunnerDelegate> main_delegate_;
|
|
||||||
std::unique_ptr<content::ContentMainRunner> main_runner_;
|
std::unique_ptr<content::ContentMainRunner> main_runner_;
|
||||||
|
|
||||||
std::unique_ptr<content::BrowserMainRunner> browser_runner_;
|
std::unique_ptr<content::BrowserMainRunner> browser_runner_;
|
||||||
@ -94,6 +97,14 @@ class CefMainRunner : public CefMainRunnerHandler {
|
|||||||
base::OnceClosure quit_callback_;
|
base::OnceClosure quit_callback_;
|
||||||
|
|
||||||
int exit_code_ = -1;
|
int exit_code_ = -1;
|
||||||
|
|
||||||
|
std::unique_ptr<ChromeMainDelegateCef> main_delegate_;
|
||||||
|
|
||||||
|
std::unique_ptr<MainThreadStackSamplingProfiler> sampling_profiler_;
|
||||||
|
std::unique_ptr<ScopedKeepAlive> keep_alive_;
|
||||||
|
|
||||||
|
raw_ptr<CefSettings> settings_ = nullptr;
|
||||||
|
CefRefPtr<CefApp> application_;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // CEF_LIBCEF_BROWSER_MAIN_RUNNER_H_
|
#endif // CEF_LIBCEF_BROWSER_MAIN_RUNNER_H_
|
||||||
|
126
libcef/browser/ui_thread.cc
Normal file
126
libcef/browser/ui_thread.cc
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
// Copyright 2020 The Chromium Embedded Framework Authors.
|
||||||
|
// Portions copyright 2014 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 "cef/libcef/browser/ui_thread.h"
|
||||||
|
|
||||||
|
#include "base/at_exit.h"
|
||||||
|
#include "base/logging.h"
|
||||||
|
#include "cef/libcef/browser/main_runner.h"
|
||||||
|
#include "cef/libcef/browser/thread_util.h"
|
||||||
|
#include "content/browser/scheduler/browser_task_executor.h"
|
||||||
|
|
||||||
|
#if BUILDFLAG(IS_LINUX)
|
||||||
|
#include "ui/base/ozone_buildflags.h"
|
||||||
|
#if BUILDFLAG(IS_OZONE_X11)
|
||||||
|
#include "ui/ozone/platform/x11/ozone_platform_x11.h"
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if BUILDFLAG(IS_WIN)
|
||||||
|
#include <Objbase.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
CefUIThread::CefUIThread(CefMainRunner* runner,
|
||||||
|
base::OnceClosure setup_callback)
|
||||||
|
: runner_(runner), setup_callback_(std::move(setup_callback)) {}
|
||||||
|
|
||||||
|
CefUIThread::~CefUIThread() {
|
||||||
|
Stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CefUIThread::Start() {
|
||||||
|
base::AutoLock lock(thread_lock_);
|
||||||
|
bool success = base::PlatformThread::CreateWithType(
|
||||||
|
0, this, &thread_, base::ThreadType::kDefault);
|
||||||
|
if (!success) {
|
||||||
|
LOG(FATAL) << "failed to UI create thread";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CefUIThread::Stop() {
|
||||||
|
base::AutoLock lock(thread_lock_);
|
||||||
|
|
||||||
|
if (!stopping_) {
|
||||||
|
stopping_ = true;
|
||||||
|
CEF_POST_TASK(CEF_UIT, base::BindOnce(&CefMainRunner::QuitMessageLoop,
|
||||||
|
base::Unretained(runner_)));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Can't join if the |thread_| is either already gone or is non-joinable.
|
||||||
|
if (thread_.is_null()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
base::PlatformThread::Join(thread_);
|
||||||
|
thread_ = base::PlatformThreadHandle();
|
||||||
|
|
||||||
|
stopping_ = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CefUIThread::WaitUntilThreadStarted() const {
|
||||||
|
DCHECK(owning_sequence_checker_.CalledOnValidSequence());
|
||||||
|
start_event_.Wait();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CefUIThread::InitializeBrowserRunner(
|
||||||
|
content::MainFunctionParams main_function_params) {
|
||||||
|
#if BUILDFLAG(IS_LINUX)
|
||||||
|
#if BUILDFLAG(IS_OZONE_X11)
|
||||||
|
// Disable creation of GtkUi (interface to GTK desktop features) and cause
|
||||||
|
// ui::GetDefaultLinuxUi() (and related functions) to return nullptr. We
|
||||||
|
// can't use GtkUi in combination with multi-threaded-message-loop because
|
||||||
|
// Chromium's GTK implementation doesn't use GDK threads. Light/dark theme
|
||||||
|
// changes will still be detected via DarkModeManagerLinux.
|
||||||
|
ui::SetMultiThreadedMessageLoopX11();
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Use our own browser process runner.
|
||||||
|
browser_runner_ = content::BrowserMainRunner::Create();
|
||||||
|
|
||||||
|
// Initialize browser process state. Uses the current thread's message loop.
|
||||||
|
int exit_code = browser_runner_->Initialize(std::move(main_function_params));
|
||||||
|
CHECK_EQ(exit_code, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CefUIThread::ThreadMain() {
|
||||||
|
base::PlatformThread::SetName("CefUIThread");
|
||||||
|
|
||||||
|
#if BUILDFLAG(IS_WIN)
|
||||||
|
// Initializes the COM library on the current thread.
|
||||||
|
CoInitialize(nullptr);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
start_event_.Signal();
|
||||||
|
|
||||||
|
std::move(setup_callback_).Run();
|
||||||
|
|
||||||
|
runner_->RunMessageLoop();
|
||||||
|
|
||||||
|
// Stop may be called before InitializeBrowserRunner if
|
||||||
|
// content::ContentMainRun was not successful (for example, due to process
|
||||||
|
// singleton relaunch).
|
||||||
|
if (browser_runner_) {
|
||||||
|
browser_runner_->Shutdown();
|
||||||
|
browser_runner_.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
// This will be a no-op if there is no BrowserTaskExecutor.
|
||||||
|
content::BrowserTaskExecutor::Shutdown();
|
||||||
|
|
||||||
|
if (!shutdown_callback_.is_null()) {
|
||||||
|
std::move(shutdown_callback_).Run();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run exit callbacks on the UI thread to avoid sequence check failures.
|
||||||
|
base::AtExitManager::ProcessCallbacksNow();
|
||||||
|
|
||||||
|
#if BUILDFLAG(IS_WIN)
|
||||||
|
// Closes the COM library on the current thread. CoInitialize must
|
||||||
|
// be balanced by a corresponding call to CoUninitialize.
|
||||||
|
CoUninitialize();
|
||||||
|
#endif
|
||||||
|
}
|
63
libcef/browser/ui_thread.h
Normal file
63
libcef/browser/ui_thread.h
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
// Copyright 2020 The Chromium Embedded Framework Authors.
|
||||||
|
// Portions copyright 2014 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.
|
||||||
|
|
||||||
|
#ifndef CEF_LIBCEF_BROWSER_UI_THREAD_H_
|
||||||
|
#define CEF_LIBCEF_BROWSER_UI_THREAD_H_
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "base/functional/callback_forward.h"
|
||||||
|
#include "base/memory/raw_ptr.h"
|
||||||
|
#include "base/sequence_checker.h"
|
||||||
|
#include "base/synchronization/lock.h"
|
||||||
|
#include "base/synchronization/waitable_event.h"
|
||||||
|
#include "base/threading/platform_thread.h"
|
||||||
|
#include "content/public/browser/browser_main_runner.h"
|
||||||
|
#include "content/public/common/main_function_params.h"
|
||||||
|
|
||||||
|
class CefMainRunner;
|
||||||
|
|
||||||
|
// Used to run the UI on a separate thread.
|
||||||
|
class CefUIThread final : public base::PlatformThread::Delegate {
|
||||||
|
public:
|
||||||
|
CefUIThread(CefMainRunner* runner, base::OnceClosure setup_callback);
|
||||||
|
~CefUIThread() override;
|
||||||
|
|
||||||
|
void Start();
|
||||||
|
void Stop();
|
||||||
|
|
||||||
|
bool WaitUntilThreadStarted() const;
|
||||||
|
|
||||||
|
void InitializeBrowserRunner(
|
||||||
|
content::MainFunctionParams main_function_params);
|
||||||
|
|
||||||
|
void set_shutdown_callback(base::OnceClosure shutdown_callback) {
|
||||||
|
shutdown_callback_ = std::move(shutdown_callback);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void ThreadMain() override;
|
||||||
|
|
||||||
|
const raw_ptr<CefMainRunner> runner_;
|
||||||
|
base::OnceClosure setup_callback_;
|
||||||
|
base::OnceClosure shutdown_callback_;
|
||||||
|
|
||||||
|
std::unique_ptr<content::BrowserMainRunner> browser_runner_;
|
||||||
|
|
||||||
|
bool stopping_ = false;
|
||||||
|
|
||||||
|
// The thread's handle.
|
||||||
|
base::PlatformThreadHandle thread_;
|
||||||
|
mutable base::Lock thread_lock_; // Protects |thread_|.
|
||||||
|
|
||||||
|
mutable base::WaitableEvent start_event_;
|
||||||
|
|
||||||
|
// This class is not thread-safe, use this to verify access from the owning
|
||||||
|
// sequence of the Thread.
|
||||||
|
base::SequenceChecker owning_sequence_checker_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // CEF_LIBCEF_BROWSER_UI_THREAD_H_
|
@ -14,6 +14,7 @@
|
|||||||
#include "base/threading/threading_features.h"
|
#include "base/threading/threading_features.h"
|
||||||
#include "cef/libcef/browser/chrome/chrome_browser_context.h"
|
#include "cef/libcef/browser/chrome/chrome_browser_context.h"
|
||||||
#include "cef/libcef/browser/chrome/chrome_content_browser_client_cef.h"
|
#include "cef/libcef/browser/chrome/chrome_content_browser_client_cef.h"
|
||||||
|
#include "cef/libcef/browser/main_runner.h"
|
||||||
#include "cef/libcef/common/cef_switches.h"
|
#include "cef/libcef/common/cef_switches.h"
|
||||||
#include "cef/libcef/common/command_line_impl.h"
|
#include "cef/libcef/common/command_line_impl.h"
|
||||||
#include "cef/libcef/common/crash_reporting.h"
|
#include "cef/libcef/common/crash_reporting.h"
|
||||||
@ -114,7 +115,7 @@ void InitLogging(const base::CommandLine* command_line) {
|
|||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
ChromeMainDelegateCef::ChromeMainDelegateCef(CefMainRunnerHandler* runner,
|
ChromeMainDelegateCef::ChromeMainDelegateCef(CefMainRunner* runner,
|
||||||
CefSettings* settings,
|
CefSettings* settings,
|
||||||
CefRefPtr<CefApp> application)
|
CefRefPtr<CefApp> application)
|
||||||
: ChromeMainDelegate(base::TimeTicks::Now()),
|
: ChromeMainDelegate(base::TimeTicks::Now()),
|
||||||
|
@ -12,11 +12,11 @@
|
|||||||
#include "cef/include/cef_app.h"
|
#include "cef/include/cef_app.h"
|
||||||
#include "cef/libcef/common/app_manager.h"
|
#include "cef/libcef/common/app_manager.h"
|
||||||
#include "cef/libcef/common/chrome/chrome_content_client_cef.h"
|
#include "cef/libcef/common/chrome/chrome_content_client_cef.h"
|
||||||
#include "cef/libcef/common/main_runner_handler.h"
|
|
||||||
#include "cef/libcef/common/resource_bundle_delegate.h"
|
#include "cef/libcef/common/resource_bundle_delegate.h"
|
||||||
#include "cef/libcef/common/task_runner_manager.h"
|
#include "cef/libcef/common/task_runner_manager.h"
|
||||||
#include "chrome/app/chrome_main_delegate.h"
|
#include "chrome/app/chrome_main_delegate.h"
|
||||||
|
|
||||||
|
class CefMainRunner;
|
||||||
class ChromeContentBrowserClientCef;
|
class ChromeContentBrowserClientCef;
|
||||||
class ChromeContentRendererClientCef;
|
class ChromeContentRendererClientCef;
|
||||||
|
|
||||||
@ -27,7 +27,7 @@ class ChromeMainDelegateCef : public ChromeMainDelegate,
|
|||||||
public:
|
public:
|
||||||
// |runner| will be non-nullptr for the main process only, and will outlive
|
// |runner| will be non-nullptr for the main process only, and will outlive
|
||||||
// this object.
|
// this object.
|
||||||
ChromeMainDelegateCef(CefMainRunnerHandler* runner,
|
ChromeMainDelegateCef(CefMainRunner* runner,
|
||||||
CefSettings* settings,
|
CefSettings* settings,
|
||||||
CefRefPtr<CefApp> application);
|
CefRefPtr<CefApp> application);
|
||||||
|
|
||||||
@ -36,6 +36,7 @@ class ChromeMainDelegateCef : public ChromeMainDelegate,
|
|||||||
|
|
||||||
~ChromeMainDelegateCef() override;
|
~ChromeMainDelegateCef() override;
|
||||||
|
|
||||||
|
protected:
|
||||||
// ChromeMainDelegate overrides.
|
// ChromeMainDelegate overrides.
|
||||||
std::optional<int> BasicStartupComplete() override;
|
std::optional<int> BasicStartupComplete() override;
|
||||||
void PreSandboxStartup() override;
|
void PreSandboxStartup() override;
|
||||||
@ -51,8 +52,10 @@ class ChromeMainDelegateCef : public ChromeMainDelegate,
|
|||||||
content::ContentClient* CreateContentClient() override;
|
content::ContentClient* CreateContentClient() override;
|
||||||
content::ContentBrowserClient* CreateContentBrowserClient() override;
|
content::ContentBrowserClient* CreateContentBrowserClient() override;
|
||||||
content::ContentRendererClient* CreateContentRendererClient() override;
|
content::ContentRendererClient* CreateContentRendererClient() override;
|
||||||
|
ui::ResourceBundle::Delegate* GetResourceBundleDelegate() override {
|
||||||
|
return &resource_bundle_delegate_;
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
|
||||||
// CefAppManager overrides.
|
// CefAppManager overrides.
|
||||||
CefRefPtr<CefApp> GetApplication() override { return application_; }
|
CefRefPtr<CefApp> GetApplication() override { return application_; }
|
||||||
content::ContentClient* GetContentClient() override {
|
content::ContentClient* GetContentClient() override {
|
||||||
@ -73,16 +76,11 @@ class ChromeMainDelegateCef : public ChromeMainDelegate,
|
|||||||
scoped_refptr<base::SingleThreadTaskRunner> GetRenderTaskRunner() override;
|
scoped_refptr<base::SingleThreadTaskRunner> GetRenderTaskRunner() override;
|
||||||
scoped_refptr<base::SingleThreadTaskRunner> GetWebWorkerTaskRunner() override;
|
scoped_refptr<base::SingleThreadTaskRunner> GetWebWorkerTaskRunner() override;
|
||||||
|
|
||||||
// ChromeMainDelegate overrides.
|
|
||||||
ui::ResourceBundle::Delegate* GetResourceBundleDelegate() override {
|
|
||||||
return &resource_bundle_delegate_;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ChromeContentBrowserClientCef* content_browser_client() const;
|
ChromeContentBrowserClientCef* content_browser_client() const;
|
||||||
ChromeContentRendererClientCef* content_renderer_client() const;
|
ChromeContentRendererClientCef* content_renderer_client() const;
|
||||||
|
|
||||||
const raw_ptr<CefMainRunnerHandler> runner_;
|
const raw_ptr<CefMainRunner> runner_;
|
||||||
const raw_ptr<CefSettings> settings_;
|
const raw_ptr<CefSettings> settings_;
|
||||||
CefRefPtr<CefApp> application_;
|
CefRefPtr<CefApp> application_;
|
||||||
|
|
||||||
|
@ -1,121 +0,0 @@
|
|||||||
// Copyright 2020 The Chromium Embedded Framework Authors.
|
|
||||||
// Portions copyright 2012 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 "cef/libcef/common/chrome/chrome_main_runner_delegate.h"
|
|
||||||
|
|
||||||
#include "base/command_line.h"
|
|
||||||
#include "base/run_loop.h"
|
|
||||||
#include "cef/libcef/browser/chrome/chrome_content_browser_client_cef.h"
|
|
||||||
#include "cef/libcef/common/app_manager.h"
|
|
||||||
#include "cef/libcef/common/chrome/chrome_main_delegate_cef.h"
|
|
||||||
#include "chrome/browser/browser_process_impl.h"
|
|
||||||
#include "chrome/browser/chrome_process_singleton.h"
|
|
||||||
#include "chrome/common/profiler/main_thread_stack_sampling_profiler.h"
|
|
||||||
#include "components/keep_alive_registry/keep_alive_types.h"
|
|
||||||
#include "components/keep_alive_registry/scoped_keep_alive.h"
|
|
||||||
#include "components/metrics/persistent_system_profile.h"
|
|
||||||
|
|
||||||
ChromeMainRunnerDelegate::ChromeMainRunnerDelegate(
|
|
||||||
CefMainRunnerHandler* runner,
|
|
||||||
CefSettings* settings,
|
|
||||||
CefRefPtr<CefApp> application)
|
|
||||||
: runner_(runner), settings_(settings), application_(application) {}
|
|
||||||
|
|
||||||
ChromeMainRunnerDelegate::~ChromeMainRunnerDelegate() = default;
|
|
||||||
|
|
||||||
content::ContentMainDelegate*
|
|
||||||
ChromeMainRunnerDelegate::GetContentMainDelegate() {
|
|
||||||
if (!main_delegate_) {
|
|
||||||
main_delegate_ = std::make_unique<ChromeMainDelegateCef>(runner_, settings_,
|
|
||||||
application_);
|
|
||||||
}
|
|
||||||
return main_delegate_.get();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::BeforeMainThreadInitialize(
|
|
||||||
const CefMainArgs& args) {
|
|
||||||
#if BUILDFLAG(IS_WIN)
|
|
||||||
base::CommandLine::Init(0, nullptr);
|
|
||||||
#else
|
|
||||||
base::CommandLine::Init(args.argc, args.argv);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::BeforeMainThreadRun(
|
|
||||||
bool multi_threaded_message_loop) {
|
|
||||||
if (multi_threaded_message_loop) {
|
|
||||||
multi_threaded_message_loop_ = true;
|
|
||||||
|
|
||||||
// Detach from the main thread so that these objects can be attached and
|
|
||||||
// modified from the UI thread going forward.
|
|
||||||
metrics::GlobalPersistentSystemProfile::GetInstance()
|
|
||||||
->DetachFromCurrentThread();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::BeforeMainMessageLoopRun(
|
|
||||||
base::RunLoop* run_loop) {
|
|
||||||
// May be nullptr if content::ContentMainRun exits early.
|
|
||||||
if (!g_browser_process) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The ScopedKeepAlive instance triggers shutdown logic when released on the
|
|
||||||
// UI thread before terminating the message loop (e.g. from CefQuitMessageLoop
|
|
||||||
// or FinishShutdownOnUIThread when running with multi-threaded message loop).
|
|
||||||
keep_alive_ = std::make_unique<ScopedKeepAlive>(
|
|
||||||
KeepAliveOrigin::APP_CONTROLLER, KeepAliveRestartOption::DISABLED);
|
|
||||||
|
|
||||||
// The QuitClosure will be executed from BrowserProcessImpl::Unpin() via
|
|
||||||
// KeepAliveRegistry when the last ScopedKeepAlive is released.
|
|
||||||
// ScopedKeepAlives are also held by Browser objects.
|
|
||||||
static_cast<BrowserProcessImpl*>(g_browser_process)
|
|
||||||
->SetQuitClosure(run_loop->QuitClosure());
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ChromeMainRunnerDelegate::HandleMainMessageLoopQuit() {
|
|
||||||
// May be nullptr if content::ContentMainRun exits early.
|
|
||||||
if (!g_browser_process) {
|
|
||||||
// Proceed with direct execution of the QuitClosure().
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// May be called multiple times. See comments in RunMainMessageLoopBefore.
|
|
||||||
keep_alive_.reset();
|
|
||||||
|
|
||||||
// Cancel direct execution of the QuitClosure() in
|
|
||||||
// CefMainRunner::QuitMessageLoop. We instead wait for all Chrome browser
|
|
||||||
// windows to exit.
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::BeforeUIThreadInitialize() {
|
|
||||||
sampling_profiler_ = std::make_unique<MainThreadStackSamplingProfiler>();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::BeforeUIThreadShutdown() {
|
|
||||||
static_cast<ChromeContentBrowserClientCef*>(
|
|
||||||
CefAppManager::Get()->GetContentClient()->browser())
|
|
||||||
->CleanupOnUIThread();
|
|
||||||
main_delegate_->CleanupOnUIThread();
|
|
||||||
|
|
||||||
sampling_profiler_.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::AfterUIThreadShutdown() {
|
|
||||||
if (multi_threaded_message_loop_) {
|
|
||||||
// Don't wait for this to be called in ChromeMainDelegate::ProcessExiting.
|
|
||||||
// It is safe to call multiple times.
|
|
||||||
ChromeProcessSingleton::DeleteInstance();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::BeforeExecuteProcess(const CefMainArgs& args) {
|
|
||||||
BeforeMainThreadInitialize(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ChromeMainRunnerDelegate::AfterExecuteProcess() {
|
|
||||||
AfterMainThreadShutdown();
|
|
||||||
}
|
|
@ -1,59 +0,0 @@
|
|||||||
// Copyright 2020 The Chromium Embedded Framework Authors.
|
|
||||||
// Portions copyright 2012 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.
|
|
||||||
|
|
||||||
#ifndef CEF_LIBCEF_COMMON_CHROME_CHROME_MAIN_RUNNER_DELEGATE_CEF_
|
|
||||||
#define CEF_LIBCEF_COMMON_CHROME_CHROME_MAIN_RUNNER_DELEGATE_CEF_
|
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
|
|
||||||
#include "base/memory/raw_ptr.h"
|
|
||||||
#include "cef/include/cef_app.h"
|
|
||||||
#include "cef/libcef/common/main_runner_delegate.h"
|
|
||||||
#include "cef/libcef/common/main_runner_handler.h"
|
|
||||||
|
|
||||||
class ChromeMainDelegateCef;
|
|
||||||
class MainThreadStackSamplingProfiler;
|
|
||||||
class ScopedKeepAlive;
|
|
||||||
|
|
||||||
class ChromeMainRunnerDelegate : public CefMainRunnerDelegate {
|
|
||||||
public:
|
|
||||||
// |runner| will be non-nullptr for the main process only, and will outlive
|
|
||||||
// this object.
|
|
||||||
ChromeMainRunnerDelegate(CefMainRunnerHandler* runner,
|
|
||||||
CefSettings* settings,
|
|
||||||
CefRefPtr<CefApp> application);
|
|
||||||
|
|
||||||
ChromeMainRunnerDelegate(const ChromeMainRunnerDelegate&) = delete;
|
|
||||||
ChromeMainRunnerDelegate& operator=(const ChromeMainRunnerDelegate&) = delete;
|
|
||||||
|
|
||||||
~ChromeMainRunnerDelegate() override;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
// CefMainRunnerDelegate overrides.
|
|
||||||
content::ContentMainDelegate* GetContentMainDelegate() override;
|
|
||||||
void BeforeMainThreadInitialize(const CefMainArgs& args) override;
|
|
||||||
void BeforeMainThreadRun(bool multi_threaded_message_loop) override;
|
|
||||||
void BeforeMainMessageLoopRun(base::RunLoop* run_loop) override;
|
|
||||||
bool HandleMainMessageLoopQuit() override;
|
|
||||||
void BeforeUIThreadInitialize() override;
|
|
||||||
void BeforeUIThreadShutdown() override;
|
|
||||||
void AfterUIThreadShutdown() override;
|
|
||||||
void BeforeExecuteProcess(const CefMainArgs& args) override;
|
|
||||||
void AfterExecuteProcess() override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::unique_ptr<ChromeMainDelegateCef> main_delegate_;
|
|
||||||
|
|
||||||
std::unique_ptr<MainThreadStackSamplingProfiler> sampling_profiler_;
|
|
||||||
std::unique_ptr<ScopedKeepAlive> keep_alive_;
|
|
||||||
|
|
||||||
const raw_ptr<CefMainRunnerHandler> runner_;
|
|
||||||
const raw_ptr<CefSettings> settings_;
|
|
||||||
CefRefPtr<CefApp> application_;
|
|
||||||
|
|
||||||
bool multi_threaded_message_loop_ = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // CEF_LIBCEF_COMMON_CHROME_CHROME_MAIN_RUNNER_DELEGATE_CEF_
|
|
@ -1,39 +0,0 @@
|
|||||||
// Copyright 2020 The Chromium Embedded Framework Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style license that can be
|
|
||||||
// found in the LICENSE file.
|
|
||||||
|
|
||||||
#ifndef CEF_LIBCEF_COMMON_MAIN_RUNNER_DELEGATE_H_
|
|
||||||
#define CEF_LIBCEF_COMMON_MAIN_RUNNER_DELEGATE_H_
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "cef/include/cef_app.h"
|
|
||||||
|
|
||||||
namespace base {
|
|
||||||
class RunLoop;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace content {
|
|
||||||
class ContentMainDelegate;
|
|
||||||
}
|
|
||||||
|
|
||||||
class CefMainRunnerDelegate {
|
|
||||||
public:
|
|
||||||
virtual content::ContentMainDelegate* GetContentMainDelegate() = 0;
|
|
||||||
|
|
||||||
virtual void BeforeMainThreadInitialize(const CefMainArgs& args) {}
|
|
||||||
virtual void BeforeMainThreadRun(bool multi_threaded_message_loop) {}
|
|
||||||
virtual void BeforeMainMessageLoopRun(base::RunLoop* run_loop) {}
|
|
||||||
virtual bool HandleMainMessageLoopQuit() { return false; }
|
|
||||||
virtual void BeforeUIThreadInitialize() {}
|
|
||||||
virtual void AfterUIThreadInitialize() {}
|
|
||||||
virtual void BeforeUIThreadShutdown() {}
|
|
||||||
virtual void AfterUIThreadShutdown() {}
|
|
||||||
virtual void BeforeMainThreadShutdown() {}
|
|
||||||
virtual void AfterMainThreadShutdown() {}
|
|
||||||
virtual void BeforeExecuteProcess(const CefMainArgs& args) {}
|
|
||||||
virtual void AfterExecuteProcess() {}
|
|
||||||
|
|
||||||
virtual ~CefMainRunnerDelegate() = default;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // CEF_LIBCEF_COMMON_MAIN_RUNNER_DELEGATE_H_
|
|
@ -1,24 +0,0 @@
|
|||||||
// Copyright 2020 The Chromium Embedded Framework Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style license that can be
|
|
||||||
// found in the LICENSE file.
|
|
||||||
|
|
||||||
#ifndef CEF_LIBCEF_COMMON_MAIN_RUNNER_HANDLER_H_
|
|
||||||
#define CEF_LIBCEF_COMMON_MAIN_RUNNER_HANDLER_H_
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
namespace content {
|
|
||||||
struct MainFunctionParams;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handles running of the main process.
|
|
||||||
class CefMainRunnerHandler {
|
|
||||||
public:
|
|
||||||
virtual void PreBrowserMain() = 0;
|
|
||||||
virtual int RunMainProcess(
|
|
||||||
content::MainFunctionParams main_function_params) = 0;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
virtual ~CefMainRunnerHandler() = default;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // CEF_LIBCEF_COMMON_MAIN_RUNNER_HANDLER_H_
|
|
Loading…
x
Reference in New Issue
Block a user