 aa23198ad8
			
		
	
	
	
	
	aa23198ad8* chore: in shell/renderer/renderer_client_base.h, remove include media/base/key_systems_support_registration.h
last use removed in c670e38b (##41610)
* chore: iwyu electron/fuses.h
* chore: iwyu media/base/video_frame.h
* chore: iwyu base/functional/callback.h
* chore: iwyu base/task/cancelable_task_tracker.h
* chore: iwyu shell/browser/draggable_region_provider.h
* chore: iwyu shell/browser/ui/inspectable_web_contents_view.h
* chore: iwyu ui/aura/window.h
* chore: iwyu ui/base/win/shell.h
* chore: iwyu ui/display/win/screen_win.h
* chore: iwyu ui/gfx/geometry/insets.h
* chore: iwyu ui/display/display.h
* chore: iwyu ui/gfx/geometry/skia_conversions.h
* chore: iwyu ui/gfx/geometry/rect_conversions.h
* chore: iwyu ui/gfx/geometry/point.h
* chore: iwyu ui/gfx/scoped_canvas.h
* chore: iwyu ui/gfx/image/image.h
* chore: iwyu ui/accessibility/ax_node_data.h
* chore: iwyu ui/views/animation/ink_drop_highlight.h
* chore: iwyu ui/gfx/font_list.h
* chore: iwyu ui/linux/nav_button_provider.h
* chore: iwyu shell/browser/ui/views/frameless_view.h
* chore: iwyu services/metrics/public/cpp/ukm_source_id.h
* chore: iwyu net/http/http_util.h
* chore: iwyu net/base/mime_util.h
* chore: iwyu content/public/common/content_client.h
* chore: iwyu <list>
* chore: iwyu <optional>
* chore: iwyu <memory>
* chore: iwyu base/files/file_path.h
* chore: iwyu ui/base/cursor/cursor.h
* chore: iwyu build/build_config.h
* chore: iwyu content/public/browser/web_contents.h
* chore: iwyu shell/browser/hid/hid_chooser_context.h
* chore: iwyu shell/common/platform_util.h
* chore: iwyu base/task/single_thread_task_runner.h
* chore: iwyu content/browser/renderer_host/render_widget_host_impl.h
* chore: iwyu content/public/browser/render_widget_host.h
* chore: iwyu shell/browser/electron_browser_context.h
* chore: iwyu content/public/browser/web_contents_observer.h
* chore: iwyu content/public/browser/render_frame_host.h
* chore: iwyu content/public/browser/media_stream_request.h
* chore: iwyu chrome/common/chrome_paths.h
* chore: iwyu chrome/browser/icon_manager.h
* chore: iwyu printing/print_settings.h
* chore: iwyu renderer/pepper_helper.h
* chore: iwyu shell/browser/api/process_metric.h
* chore: iwyu shell/browser/electron_browser_client.h
* chore: iwyu shell/browser/electron_browser_context.h
* chore: iwyu shell/browser/api/electron_api_session.h
* chore: iwyu shell/browser/api/electron_api_app.h
* chore: iwyu shell/browser/ui/views/client_frame_view_linux.h
* chore: iwyu shell/browser/native_window_views.h
* chore: iwyu base/win/windows_version.h
* chore: iwyu shell/common/electron_paths.h
* chore: iwyu content/public/common/content_switches.h
* chore: iwyu third_party/skia/include/core/SkRRect.h
* chore: iwyu third_party/skia/include/core/SkBitmap.h
* chore: iwyu third_party/skia
* chore: iwyu shell/browser/osr/osr_host_display_client.h
* chore: iwyu shell/browser/login_handler.h
* chore: iwyu shell/browser/javascript_environment.h
* chore: iwyu shell/browser/event_emitter_mixin.h
* fix: mac
* fix: mac
* chore: iwyu base/nix/xdg_util.h
* fix: win
* fix: win
* fix: win
* fix: win
		
	
			
		
			
				
	
	
		
			193 lines
		
	
	
	
		
			6.4 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			193 lines
		
	
	
	
		
			6.4 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (c) 2016 GitHub, Inc.
 | |
| // Use of this source code is governed by the MIT license that can be
 | |
| // found in the LICENSE file.
 | |
| 
 | |
| #include "shell/browser/relauncher.h"
 | |
| 
 | |
| #include <utility>
 | |
| 
 | |
| #if BUILDFLAG(IS_WIN)
 | |
| #include <windows.h>
 | |
| #endif
 | |
| 
 | |
| #include "base/files/file_util.h"
 | |
| #include "base/files/scoped_file.h"
 | |
| #include "base/logging.h"
 | |
| #include "base/path_service.h"
 | |
| #include "base/process/launch.h"
 | |
| #include "content/public/common/content_paths.h"
 | |
| #include "content/public/common/main_function_params.h"
 | |
| #include "shell/common/electron_command_line.h"
 | |
| 
 | |
| #if BUILDFLAG(IS_POSIX)
 | |
| #include "base/posix/eintr_wrapper.h"
 | |
| #endif
 | |
| 
 | |
| namespace relauncher {
 | |
| 
 | |
| namespace internal {
 | |
| 
 | |
| #if BUILDFLAG(IS_POSIX)
 | |
| const int kRelauncherSyncFD = STDERR_FILENO + 1;
 | |
| #endif
 | |
| 
 | |
| const CharType* kRelauncherTypeArg = FILE_PATH_LITERAL("--type=relauncher");
 | |
| const CharType* kRelauncherArgSeparator = FILE_PATH_LITERAL("---");
 | |
| 
 | |
| }  // namespace internal
 | |
| 
 | |
| bool RelaunchApp(const StringVector& argv) {
 | |
|   // Use the currently-running application's helper process. The automatic
 | |
|   // update feature is careful to leave the currently-running version alone,
 | |
|   // so this is safe even if the relaunch is the result of an update having
 | |
|   // been applied. In fact, it's safer than using the updated version of the
 | |
|   // helper process, because there's no guarantee that the updated version's
 | |
|   // relauncher implementation will be compatible with the running version's.
 | |
|   base::FilePath child_path;
 | |
|   if (!base::PathService::Get(content::CHILD_PROCESS_EXE, &child_path)) {
 | |
|     LOG(ERROR) << "No CHILD_PROCESS_EXE";
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   StringVector relauncher_args;
 | |
|   return RelaunchAppWithHelper(child_path, relauncher_args, argv);
 | |
| }
 | |
| 
 | |
| bool RelaunchAppWithHelper(const base::FilePath& helper,
 | |
|                            const StringVector& relauncher_args,
 | |
|                            const StringVector& argv) {
 | |
|   StringVector relaunch_argv;
 | |
|   relaunch_argv.push_back(helper.value());
 | |
|   relaunch_argv.push_back(internal::kRelauncherTypeArg);
 | |
|   // Relauncher process has its own --type=relauncher which
 | |
|   // is not recognized by the service_manager, explicitly set
 | |
|   // the sandbox type to avoid CHECK failure in
 | |
|   // service_manager::SandboxTypeFromCommandLine
 | |
|   relaunch_argv.push_back(FILE_PATH_LITERAL("--no-sandbox"));
 | |
| 
 | |
|   relaunch_argv.insert(relaunch_argv.end(), relauncher_args.begin(),
 | |
|                        relauncher_args.end());
 | |
| 
 | |
|   relaunch_argv.push_back(internal::kRelauncherArgSeparator);
 | |
| 
 | |
|   relaunch_argv.insert(relaunch_argv.end(), argv.begin(), argv.end());
 | |
| 
 | |
| #if BUILDFLAG(IS_POSIX)
 | |
|   int pipe_fds[2];
 | |
|   if (HANDLE_EINTR(pipe(pipe_fds)) != 0) {
 | |
|     PLOG(ERROR) << "pipe";
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // The parent process will only use pipe_read_fd as the read side of the
 | |
|   // pipe. It can close the write side as soon as the relauncher process has
 | |
|   // forked off. The relauncher process will only use pipe_write_fd as the
 | |
|   // write side of the pipe. In that process, the read side will be closed by
 | |
|   // base::LaunchApp because it won't be present in fd_map, and the write side
 | |
|   // will be remapped to kRelauncherSyncFD by fd_map.
 | |
|   base::ScopedFD pipe_read_fd(pipe_fds[0]);
 | |
|   base::ScopedFD pipe_write_fd(pipe_fds[1]);
 | |
| 
 | |
|   // Make sure kRelauncherSyncFD is a safe value. base::LaunchProcess will
 | |
|   // preserve these three FDs in forked processes, so kRelauncherSyncFD should
 | |
|   // not conflict with them.
 | |
|   static_assert(internal::kRelauncherSyncFD != STDIN_FILENO &&
 | |
|                     internal::kRelauncherSyncFD != STDOUT_FILENO &&
 | |
|                     internal::kRelauncherSyncFD != STDERR_FILENO,
 | |
|                 "kRelauncherSyncFD must not conflict with stdio fds");
 | |
| #endif
 | |
| 
 | |
|   base::LaunchOptions options;
 | |
| #if BUILDFLAG(IS_POSIX)
 | |
|   options.fds_to_remap.emplace_back(pipe_write_fd.get(),
 | |
|                                     internal::kRelauncherSyncFD);
 | |
|   base::Process process = base::LaunchProcess(relaunch_argv, options);
 | |
| #elif BUILDFLAG(IS_WIN)
 | |
|   base::Process process = base::LaunchProcess(
 | |
|       internal::ArgvToCommandLineString(relaunch_argv), options);
 | |
| #endif
 | |
|   if (!process.IsValid()) {
 | |
|     LOG(ERROR) << "base::LaunchProcess failed";
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // The relauncher process is now starting up, or has started up. The
 | |
|   // original parent process continues.
 | |
| 
 | |
| #if BUILDFLAG(IS_WIN)
 | |
|   // Synchronize with the relauncher process.
 | |
|   StringType name = internal::GetWaitEventName(process.Pid());
 | |
|   HANDLE wait_event = ::CreateEventW(nullptr, TRUE, FALSE, name.c_str());
 | |
|   if (wait_event != nullptr) {
 | |
|     WaitForSingleObject(wait_event, 1000);
 | |
|     CloseHandle(wait_event);
 | |
|   }
 | |
| #elif BUILDFLAG(IS_POSIX)
 | |
|   pipe_write_fd.reset();  // close(pipe_fds[1]);
 | |
| 
 | |
|   // Synchronize with the relauncher process.
 | |
|   char read_char;
 | |
|   int read_result = HANDLE_EINTR(read(pipe_read_fd.get(), &read_char, 1));
 | |
|   if (read_result != 1) {
 | |
|     if (read_result < 0) {
 | |
|       PLOG(ERROR) << "read";
 | |
|     } else {
 | |
|       LOG(ERROR) << "read: unexpected result " << read_result;
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Since a byte has been successfully read from the relauncher process, it's
 | |
|   // guaranteed to have set up its kqueue monitoring this process for exit.
 | |
|   // It's safe to exit now.
 | |
| #endif
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| int RelauncherMain(const content::MainFunctionParams& main_parameters) {
 | |
|   const StringVector& argv = electron::ElectronCommandLine::argv();
 | |
| 
 | |
|   if (argv.size() < 4 || argv[1] != internal::kRelauncherTypeArg) {
 | |
|     LOG(ERROR) << "relauncher process invoked with unexpected arguments";
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   internal::RelauncherSynchronizeWithParent();
 | |
| 
 | |
|   // Figure out what to execute, what arguments to pass it, and whether to
 | |
|   // start it in the background.
 | |
|   bool in_relauncher_args = false;
 | |
|   StringType relaunch_executable;
 | |
|   StringVector relauncher_args;
 | |
|   StringVector launch_argv;
 | |
|   for (size_t argv_index = 2; argv_index < argv.size(); ++argv_index) {
 | |
|     const StringType& arg(argv[argv_index]);
 | |
|     if (!in_relauncher_args) {
 | |
|       if (arg == internal::kRelauncherArgSeparator) {
 | |
|         in_relauncher_args = true;
 | |
|       } else {
 | |
|         relauncher_args.push_back(arg);
 | |
|       }
 | |
|     } else {
 | |
|       launch_argv.push_back(arg);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (launch_argv.empty()) {
 | |
|     LOG(ERROR) << "nothing to relaunch";
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   if (internal::LaunchProgram(relauncher_args, launch_argv) != 0) {
 | |
|     LOG(ERROR) << "failed to launch program";
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   // The application should have relaunched (or is in the process of
 | |
|   // relaunching). From this point on, only clean-up tasks should occur, and
 | |
|   // failures are tolerable.
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| }  // namespace relauncher
 |