mirror of
				https://bitbucket.org/chromiumembedded/cef
				synced 2025-06-05 21:39:12 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			624 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			624 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // 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/main_runner.h"
 | |
| 
 | |
| #include "base/at_exit.h"
 | |
| #include "base/base_switches.h"
 | |
| #include "base/command_line.h"
 | |
| #include "base/debug/debugger.h"
 | |
| #include "base/memory/raw_ptr.h"
 | |
| #include "base/run_loop.h"
 | |
| #include "base/sequence_checker.h"
 | |
| #include "base/synchronization/lock.h"
 | |
| #include "base/synchronization/waitable_event.h"
 | |
| #include "base/threading/thread.h"
 | |
| #include "cef/libcef/browser/browser_message_loop.h"
 | |
| #include "cef/libcef/browser/thread_util.h"
 | |
| #include "cef/libcef/common/cef_switches.h"
 | |
| #include "cef/libcef/common/chrome/chrome_main_runner_delegate.h"
 | |
| #include "cef/libcef/features/runtime.h"
 | |
| #include "chrome/common/chrome_result_codes.h"
 | |
| #include "components/crash/core/app/crash_switches.h"
 | |
| #include "content/app/content_main_runner_impl.h"
 | |
| #include "content/browser/scheduler/browser_task_executor.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/common/content_switches.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)
 | |
| #include <Objbase.h>
 | |
| 
 | |
| #include <windows.h>
 | |
| 
 | |
| #include <memory>
 | |
| 
 | |
| #include "content/public/app/sandbox_helper_win.h"
 | |
| #include "sandbox/win/src/sandbox_types.h"
 | |
| #endif
 | |
| 
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
| #include "cef/libcef/common/alloy/alloy_main_runner_delegate.h"
 | |
| #endif
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| enum class RuntimeType {
 | |
|   UNINITIALIZED,
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
|   ALLOY,
 | |
| #endif
 | |
|   CHROME,
 | |
| };
 | |
| RuntimeType g_runtime_type = RuntimeType::UNINITIALIZED;
 | |
| 
 | |
| std::unique_ptr<CefMainRunnerDelegate> MakeDelegate(
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
|     RuntimeType type,
 | |
| #endif
 | |
|     CefMainRunnerHandler* runner,
 | |
|     CefSettings* settings,
 | |
|     CefRefPtr<CefApp> application) {
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
|   if (type == RuntimeType::ALLOY) {
 | |
|     g_runtime_type = RuntimeType::ALLOY;
 | |
|     return std::make_unique<AlloyMainRunnerDelegate>(runner, settings,
 | |
|                                                      application);
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   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
 | |
| // Remove the "--type=crashpad-handler" command-line flag that will otherwise
 | |
| // confuse the crashpad handler.
 | |
| // Chrome uses an embedded crashpad handler on Windows only and imports this
 | |
| // function via the existing "run_as_crashpad_handler" target defined in
 | |
| // components/crash/core/app/BUILD.gn. CEF uses an embedded handler on all
 | |
| // platforms so we define the function here instead of using the existing
 | |
| // target (because we can't use that target on macOS).
 | |
| int RunAsCrashpadHandler(const base::CommandLine& command_line) {
 | |
|   base::CommandLine::StringVector argv = command_line.argv();
 | |
|   const base::CommandLine::StringType process_type =
 | |
|       FILE_PATH_LITERAL("--type=");
 | |
|   argv.erase(
 | |
|       std::remove_if(argv.begin(), argv.end(),
 | |
|                      [&process_type](const base::CommandLine::StringType& str) {
 | |
|                        return base::StartsWith(str, process_type,
 | |
|                                                base::CompareCase::SENSITIVE) ||
 | |
|                               (!str.empty() && str[0] == L'/');
 | |
|                      }),
 | |
|       argv.end());
 | |
| 
 | |
| #if BUILDFLAG(IS_POSIX)
 | |
|   // HandlerMain on POSIX uses the system version of getopt_long which expects
 | |
|   // the first argument to be the program name.
 | |
|   argv.insert(argv.begin(), command_line.GetProgram().value());
 | |
| #endif
 | |
| 
 | |
|   std::unique_ptr<char*[]> argv_as_utf8(new char*[argv.size() + 1]);
 | |
|   std::vector<std::string> storage;
 | |
|   storage.reserve(argv.size());
 | |
|   for (size_t i = 0; i < argv.size(); ++i) {
 | |
| #if BUILDFLAG(IS_WIN)
 | |
|     storage.push_back(base::WideToUTF8(argv[i]));
 | |
| #else
 | |
|     storage.push_back(argv[i]);
 | |
| #endif
 | |
|     argv_as_utf8[i] = &storage[i][0];
 | |
|   }
 | |
|   argv_as_utf8[argv.size()] = nullptr;
 | |
|   argv.clear();
 | |
|   return crashpad::HandlerMain(static_cast<int>(storage.size()),
 | |
|                                argv_as_utf8.get(), nullptr);
 | |
| }
 | |
| 
 | |
| }  // 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,
 | |
|                              bool external_message_pump)
 | |
|     : multi_threaded_message_loop_(multi_threaded_message_loop),
 | |
|       external_message_pump_(external_message_pump) {}
 | |
| 
 | |
| CefMainRunner::~CefMainRunner() = default;
 | |
| 
 | |
| bool CefMainRunner::Initialize(CefSettings* settings,
 | |
|                                CefRefPtr<CefApp> application,
 | |
|                                const CefMainArgs& args,
 | |
|                                void* windows_sandbox_info,
 | |
|                                bool* initialized,
 | |
|                                base::OnceClosure context_initialized) {
 | |
|   DCHECK(!main_delegate_);
 | |
|   main_delegate_ = MakeDelegate(
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
|       settings->chrome_runtime ? RuntimeType::CHROME : RuntimeType::ALLOY,
 | |
| #endif
 | |
|       this, settings, application);
 | |
| 
 | |
|   exit_code_ =
 | |
|       ContentMainInitialize(args, windows_sandbox_info, &settings->no_sandbox,
 | |
| #if BUILDFLAG(IS_POSIX) && !BUILDFLAG(IS_ANDROID)
 | |
|                             settings->disable_signal_handlers
 | |
| #else
 | |
|                             false
 | |
| #endif
 | |
|       );
 | |
|   if (exit_code_ >= 0) {
 | |
|     LOG(ERROR) << "ContentMainInitialize failed with exit code " << exit_code_;
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   exit_code_ = ContentMainRun(initialized, std::move(context_initialized));
 | |
|   if (exit_code_ != content::RESULT_CODE_NORMAL_EXIT) {
 | |
|     // Some exit codes are used to exit early, but are otherwise a normal
 | |
|     // result. Don't log for those codes.
 | |
|     if (!chrome::IsNormalResultCode(
 | |
|             static_cast<chrome::ResultCode>(exit_code_))) {
 | |
|       LOG(ERROR) << "ContentMainRun failed with exit code " << exit_code_;
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void CefMainRunner::Shutdown(base::OnceClosure shutdown_on_ui_thread,
 | |
|                              base::OnceClosure finalize_shutdown) {
 | |
|   if (multi_threaded_message_loop_) {
 | |
|     // Start shutdown on the UI thread. This is guaranteed to run before the
 | |
|     // thread RunLoop has stopped.
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&CefMainRunner::StartShutdownOnUIThread,
 | |
|                                  base::Unretained(this),
 | |
|                                  std::move(shutdown_on_ui_thread)));
 | |
| 
 | |
|     // Finish shutdown on the UI thread after the thread RunLoop has stopped and
 | |
|     // before running exit callbacks.
 | |
|     ui_thread_->set_shutdown_callback(base::BindOnce(
 | |
|         &CefMainRunner::FinishShutdownOnUIThread, base::Unretained(this)));
 | |
| 
 | |
|     // Blocks until the thread has stopped.
 | |
|     ui_thread_->Stop();
 | |
|     ui_thread_.reset();
 | |
|   }
 | |
| 
 | |
|   main_delegate_->BeforeMainThreadShutdown();
 | |
| 
 | |
|   if (!multi_threaded_message_loop_) {
 | |
|     // Main thread and UI thread are the same.
 | |
|     StartShutdownOnUIThread(std::move(shutdown_on_ui_thread));
 | |
| 
 | |
|     browser_runner_->Shutdown();
 | |
|     browser_runner_.reset();
 | |
| 
 | |
|     FinishShutdownOnUIThread();
 | |
|   }
 | |
| 
 | |
|   // Shut down the content runner.
 | |
|   content::ContentMainShutdown(main_runner_.get());
 | |
| 
 | |
|   main_runner_.reset();
 | |
| 
 | |
|   std::move(finalize_shutdown).Run();
 | |
|   main_delegate_->AfterMainThreadShutdown();
 | |
| 
 | |
|   main_delegate_.reset();
 | |
|   g_runtime_type = RuntimeType::UNINITIALIZED;
 | |
| }
 | |
| 
 | |
| void CefMainRunner::RunMessageLoop() {
 | |
|   base::RunLoop run_loop;
 | |
| 
 | |
|   DCHECK(quit_callback_.is_null());
 | |
|   quit_callback_ = run_loop.QuitClosure();
 | |
| 
 | |
|   main_delegate_->BeforeMainMessageLoopRun(&run_loop);
 | |
| 
 | |
|   // Blocks until QuitMessageLoop() is called.
 | |
|   run_loop.Run();
 | |
| }
 | |
| 
 | |
| void CefMainRunner::QuitMessageLoop() {
 | |
|   if (!quit_callback_.is_null()) {
 | |
|     if (main_delegate_->HandleMainMessageLoopQuit()) {
 | |
|       return;
 | |
|     }
 | |
|     std::move(quit_callback_).Run();
 | |
|   }
 | |
| }
 | |
| 
 | |
| // static
 | |
| int CefMainRunner::RunAsHelperProcess(const CefMainArgs& args,
 | |
|                                       CefRefPtr<CefApp> application,
 | |
|                                       void* windows_sandbox_info) {
 | |
|   base::CommandLine command_line(base::CommandLine::NO_PROGRAM);
 | |
| #if BUILDFLAG(IS_WIN)
 | |
|   command_line.ParseFromString(::GetCommandLineW());
 | |
| #else
 | |
|   command_line.InitFromArgv(args.argc, args.argv);
 | |
| #endif
 | |
| 
 | |
|   // Wait for the debugger as early in process initialization as possible.
 | |
|   if (command_line.HasSwitch(switches::kWaitForDebugger)) {
 | |
|     base::debug::WaitForDebugger(60, true);
 | |
|   }
 | |
| 
 | |
|   // If no process type is specified then it represents the browser process and
 | |
|   // we do nothing.
 | |
|   const std::string& process_type =
 | |
|       command_line.GetSwitchValueASCII(switches::kProcessType);
 | |
|   if (process_type.empty()) {
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   auto main_delegate = MakeDelegate(
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
|       command_line.HasSwitch(switches::kEnableChromeRuntime)
 | |
|           ? RuntimeType::CHROME
 | |
|           : RuntimeType::ALLOY,
 | |
| #endif
 | |
|       /*runner=*/nullptr, /*settings=*/nullptr, application);
 | |
|   main_delegate->BeforeExecuteProcess(args);
 | |
| 
 | |
|   int result;
 | |
| 
 | |
|   if (process_type == crash_reporter::switches::kCrashpadHandler) {
 | |
|     result = RunAsCrashpadHandler(command_line);
 | |
|     main_delegate->AfterExecuteProcess();
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   // Execute the secondary process.
 | |
|   content::ContentMainParams main_params(
 | |
|       main_delegate->GetContentMainDelegate());
 | |
| #if BUILDFLAG(IS_WIN)
 | |
|   sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
 | |
|   if (windows_sandbox_info == nullptr) {
 | |
|     content::InitializeSandboxInfo(&sandbox_info);
 | |
|     windows_sandbox_info = &sandbox_info;
 | |
|   }
 | |
| 
 | |
|   main_params.instance = args.instance;
 | |
|   main_params.sandbox_info =
 | |
|       static_cast<sandbox::SandboxInterfaceInfo*>(windows_sandbox_info);
 | |
| #else
 | |
|   main_params.argc = args.argc;
 | |
|   main_params.argv = const_cast<const char**>(args.argv);
 | |
| #endif
 | |
|   result = content::ContentMain(std::move(main_params));
 | |
| 
 | |
|   main_delegate->AfterExecuteProcess();
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| int CefMainRunner::ContentMainInitialize(const CefMainArgs& args,
 | |
|                                          void* windows_sandbox_info,
 | |
|                                          int* no_sandbox,
 | |
|                                          bool disable_signal_handlers) {
 | |
|   main_delegate_->BeforeMainThreadInitialize(args);
 | |
| 
 | |
|   // Initialize the content runner.
 | |
|   main_runner_ = content::ContentMainRunner::Create();
 | |
|   content::ContentMainParams main_params(
 | |
|       main_delegate_->GetContentMainDelegate());
 | |
| 
 | |
| #if BUILDFLAG(IS_WIN)
 | |
|   sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
 | |
|   if (windows_sandbox_info == nullptr) {
 | |
|     windows_sandbox_info = &sandbox_info;
 | |
|     *no_sandbox = true;
 | |
|   }
 | |
| 
 | |
|   main_params.instance = args.instance;
 | |
|   main_params.sandbox_info =
 | |
|       static_cast<sandbox::SandboxInterfaceInfo*>(windows_sandbox_info);
 | |
| #else
 | |
|   main_params.argc = args.argc;
 | |
|   main_params.argv = const_cast<const char**>(args.argv);
 | |
| #endif
 | |
| 
 | |
| #if BUILDFLAG(IS_POSIX) && !BUILDFLAG(IS_ANDROID)
 | |
|   main_params.disable_signal_handlers = disable_signal_handlers;
 | |
| #endif
 | |
| 
 | |
|   return content::ContentMainInitialize(std::move(main_params),
 | |
|                                         main_runner_.get());
 | |
| }
 | |
| 
 | |
| int CefMainRunner::ContentMainRun(bool* initialized,
 | |
|                                   base::OnceClosure context_initialized) {
 | |
|   main_delegate_->BeforeMainThreadRun(multi_threaded_message_loop_);
 | |
| 
 | |
|   int exit_code = -1;
 | |
| 
 | |
|   if (multi_threaded_message_loop_) {
 | |
|     // Detach the CommandLine from the main thread so that it can be
 | |
|     // attached and modified from the UI thread going forward.
 | |
|     base::CommandLine::ForCurrentProcess()->DetachFromCurrentSequence();
 | |
| 
 | |
|     base::WaitableEvent uithread_startup_event(
 | |
|         base::WaitableEvent::ResetPolicy::AUTOMATIC,
 | |
|         base::WaitableEvent::InitialState::NOT_SIGNALED);
 | |
| 
 | |
|     if (!CreateUIThread(base::BindOnce(
 | |
|             [](CefMainRunner* runner, base::WaitableEvent* event,
 | |
|                int* exit_code) {
 | |
|               runner->main_delegate_->BeforeUIThreadInitialize();
 | |
|               *exit_code = content::ContentMainRun(runner->main_runner_.get());
 | |
| 
 | |
|               if (*exit_code != content::RESULT_CODE_NORMAL_EXIT) {
 | |
|                 runner->FinishShutdownOnUIThread();
 | |
|               }
 | |
| 
 | |
|               event->Signal();
 | |
|             },
 | |
|             base::Unretained(this), base::Unretained(&uithread_startup_event),
 | |
|             base::Unretained(&exit_code)))) {
 | |
|       return exit_code;
 | |
|     }
 | |
| 
 | |
|     *initialized = true;
 | |
| 
 | |
|     // We need to wait until content::ContentMainRun has finished.
 | |
|     uithread_startup_event.Wait();
 | |
| 
 | |
|     if (exit_code != content::RESULT_CODE_NORMAL_EXIT) {
 | |
|       // content::ContentMainRun was not successful (for example, due to process
 | |
|       // singleton relaunch). Stop the UI thread and block until done.
 | |
|       ui_thread_->Stop();
 | |
|       ui_thread_.reset();
 | |
|     }
 | |
|   } else {
 | |
|     *initialized = true;
 | |
|     main_delegate_->BeforeUIThreadInitialize();
 | |
|     exit_code = content::ContentMainRun(main_runner_.get());
 | |
|   }
 | |
| 
 | |
|   if (exit_code == content::RESULT_CODE_NORMAL_EXIT) {
 | |
|     // content::ContentMainRun was successful and we're not exiting early.
 | |
|     if (CEF_CURRENTLY_ON_UIT()) {
 | |
|       OnContextInitialized(std::move(context_initialized));
 | |
|     } else {
 | |
|       // Continue initialization on the UI thread.
 | |
|       CEF_POST_TASK(CEF_UIT,
 | |
|                     base::BindOnce(&CefMainRunner::OnContextInitialized,
 | |
|                                    base::Unretained(this),
 | |
|                                    std::move(context_initialized)));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return exit_code;
 | |
| }
 | |
| 
 | |
| void CefMainRunner::PreBrowserMain() {
 | |
|   if (external_message_pump_) {
 | |
|     InitExternalMessagePumpFactoryForUI();
 | |
|   }
 | |
| }
 | |
| 
 | |
| int CefMainRunner::RunMainProcess(
 | |
|     content::MainFunctionParams main_function_params) {
 | |
|   if (!multi_threaded_message_loop_) {
 | |
|     // Use our own browser process runner.
 | |
|     browser_runner_ = content::BrowserMainRunner::Create();
 | |
| 
 | |
|     // Initialize browser process state. Results in a call to
 | |
|     // AlloyBrowserMain::PreBrowserMain() which creates the UI message
 | |
|     // loop.
 | |
|     int exit_code =
 | |
|         browser_runner_->Initialize(std::move(main_function_params));
 | |
|     if (exit_code >= 0) {
 | |
|       return exit_code;
 | |
|     }
 | |
|   } else {
 | |
|     // Running on the separate UI thread.
 | |
|     DCHECK(ui_thread_);
 | |
|     ui_thread_->InitializeBrowserRunner(std::move(main_function_params));
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| bool CefMainRunner::CreateUIThread(base::OnceClosure setup_callback) {
 | |
|   DCHECK(!ui_thread_);
 | |
| 
 | |
|   ui_thread_ = std::make_unique<CefUIThread>(this, std::move(setup_callback));
 | |
|   ui_thread_->Start();
 | |
|   ui_thread_->WaitUntilThreadStarted();
 | |
| 
 | |
|   if (external_message_pump_) {
 | |
|     InitExternalMessagePumpFactoryForUI();
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void CefMainRunner::OnContextInitialized(
 | |
|     base::OnceClosure context_initialized) {
 | |
|   CEF_REQUIRE_UIT();
 | |
| 
 | |
|   main_delegate_->AfterUIThreadInitialize();
 | |
|   std::move(context_initialized).Run();
 | |
| }
 | |
| 
 | |
| void CefMainRunner::StartShutdownOnUIThread(
 | |
|     base::OnceClosure shutdown_on_ui_thread) {
 | |
|   CEF_REQUIRE_UIT();
 | |
| 
 | |
|   // Execute all pending tasks now before proceeding with shutdown. Otherwise,
 | |
|   // objects bound to tasks and released at the end of shutdown via
 | |
|   // BrowserTaskExecutor::Shutdown may attempt to access other objects that have
 | |
|   // already been destroyed (for example, if teardown results in a call to
 | |
|   // RenderProcessHostImpl::Cleanup).
 | |
|   content::BrowserTaskExecutor::RunAllPendingTasksOnThreadForTesting(
 | |
|       content::BrowserThread::UI);
 | |
|   content::BrowserTaskExecutor::RunAllPendingTasksOnThreadForTesting(
 | |
|       content::BrowserThread::IO);
 | |
| 
 | |
|   static_cast<content::ContentMainRunnerImpl*>(main_runner_.get())
 | |
|       ->ShutdownOnUIThread();
 | |
| 
 | |
|   std::move(shutdown_on_ui_thread).Run();
 | |
|   main_delegate_->BeforeUIThreadShutdown();
 | |
| }
 | |
| 
 | |
| void CefMainRunner::FinishShutdownOnUIThread() {
 | |
|   main_delegate_->AfterUIThreadShutdown();
 | |
| }
 | |
| 
 | |
| // From libcef/features/runtime.h:
 | |
| namespace cef {
 | |
| 
 | |
| #if BUILDFLAG(ENABLE_ALLOY_BOOTSTRAP)
 | |
| bool IsAlloyRuntimeEnabled() {
 | |
|   return g_runtime_type == RuntimeType::ALLOY;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| bool IsChromeRuntimeEnabled() {
 | |
|   return g_runtime_type == RuntimeType::CHROME;
 | |
| }
 | |
| 
 | |
| }  // namespace cef
 |