| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | // Copyright (c) 2016 GitHub, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/relauncher.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-13 18:13:34 -07:00
										 |  |  | #include "base/logging.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | #include "base/process/launch.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-02 09:51:33 +02:00
										 |  |  | #include "base/process/process_handle.h"
 | 
					
						
							| 
									
										
										
										
											2024-01-07 22:02:20 -08:00
										 |  |  | #include "base/strings/strcat_win.h"
 | 
					
						
							|  |  |  | #include "base/strings/string_number_conversions_win.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | #include "base/win/scoped_handle.h"
 | 
					
						
							|  |  |  | #include "sandbox/win/src/nt_internals.h"
 | 
					
						
							|  |  |  | #include "sandbox/win/src/win_utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 12:55:47 -07:00
										 |  |  | namespace relauncher::internal { | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  | struct PROCESS_BASIC_INFORMATION { | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     NTSTATUS ExitStatus; | 
					
						
							|  |  |  |     PVOID padding_for_x64_0; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   PPEB PebBaseAddress; | 
					
						
							|  |  |  |   KAFFINITY AffinityMask; | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     KPRIORITY BasePriority; | 
					
						
							|  |  |  |     PVOID padding_for_x64_1; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     DWORD UniqueProcessId; | 
					
						
							|  |  |  |     PVOID padding_for_x64_2; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     DWORD InheritedFromUniqueProcessId; | 
					
						
							|  |  |  |     PVOID padding_for_x64_3; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | HANDLE GetParentProcessHandle(base::ProcessHandle handle) { | 
					
						
							| 
									
										
										
										
											2024-07-02 09:51:33 +02:00
										 |  |  |   base::ProcessId ppid = base::GetParentProcessId(handle); | 
					
						
							|  |  |  |   if (ppid == 0u) { | 
					
						
							|  |  |  |     LOG(ERROR) << "Could not get parent process handle"; | 
					
						
							| 
									
										
										
										
											2023-10-03 21:26:35 +02:00
										 |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-02 09:51:33 +02:00
										 |  |  |   return ::OpenProcess(PROCESS_ALL_ACCESS, TRUE, ppid); | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 21:10:39 +09:00
										 |  |  | StringType AddQuoteForArg(const StringType& arg) { | 
					
						
							|  |  |  |   // We follow the quoting rules of CommandLineToArgvW.
 | 
					
						
							|  |  |  |   // http://msdn.microsoft.com/en-us/library/17w5ykft.aspx
 | 
					
						
							|  |  |  |   std::wstring quotable_chars(L" \\\""); | 
					
						
							|  |  |  |   if (arg.find_first_of(quotable_chars) == std::wstring::npos) { | 
					
						
							|  |  |  |     // No quoting necessary.
 | 
					
						
							|  |  |  |     return arg; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::wstring out; | 
					
						
							|  |  |  |   out.push_back(L'"'); | 
					
						
							|  |  |  |   for (size_t i = 0; i < arg.size(); ++i) { | 
					
						
							|  |  |  |     if (arg[i] == '\\') { | 
					
						
							|  |  |  |       // Find the extent of this run of backslashes.
 | 
					
						
							|  |  |  |       size_t start = i, end = start + 1; | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |       for (; end < arg.size() && arg[end] == '\\'; ++end) { | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-06-02 21:10:39 +09:00
										 |  |  |       size_t backslash_count = end - start; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Backslashes are escapes only if the run is followed by a double quote.
 | 
					
						
							|  |  |  |       // Since we also will end the string with a double quote, we escape for
 | 
					
						
							|  |  |  |       // either a double quote or the end of the string.
 | 
					
						
							|  |  |  |       if (end == arg.size() || arg[end] == '"') { | 
					
						
							|  |  |  |         // To quote, we need to output 2x as many backslashes.
 | 
					
						
							|  |  |  |         backslash_count *= 2; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       for (size_t j = 0; j < backslash_count; ++j) | 
					
						
							|  |  |  |         out.push_back('\\'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Advance i to one before the end to balance i++ in loop.
 | 
					
						
							|  |  |  |       i = end - 1; | 
					
						
							|  |  |  |     } else if (arg[i] == '"') { | 
					
						
							|  |  |  |       out.push_back('\\'); | 
					
						
							|  |  |  |       out.push_back('"'); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       out.push_back(arg[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   out.push_back('"'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | StringType GetWaitEventName(base::ProcessId pid) { | 
					
						
							| 
									
										
										
										
											2024-11-30 06:54:40 -06:00
										 |  |  |   return base::StrCat({L"ElectronRelauncherWaitEvent-", | 
					
						
							|  |  |  |                        base::NumberToWString(static_cast<int>(pid))}); | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 21:10:39 +09:00
										 |  |  | StringType ArgvToCommandLineString(const StringVector& argv) { | 
					
						
							|  |  |  |   StringType command_line; | 
					
						
							|  |  |  |   for (const StringType& arg : argv) { | 
					
						
							|  |  |  |     if (!command_line.empty()) | 
					
						
							|  |  |  |       command_line += L' '; | 
					
						
							|  |  |  |     command_line += AddQuoteForArg(arg); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return command_line; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  | void RelauncherSynchronizeWithParent() { | 
					
						
							|  |  |  |   base::Process process = base::Process::Current(); | 
					
						
							|  |  |  |   base::win::ScopedHandle parent_process( | 
					
						
							|  |  |  |       GetParentProcessHandle(process.Handle())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Notify the parent process that it can quit now.
 | 
					
						
							|  |  |  |   StringType name = internal::GetWaitEventName(process.Pid()); | 
					
						
							|  |  |  |   base::win::ScopedHandle wait_event( | 
					
						
							| 
									
										
										
										
											2023-10-03 21:26:35 +02:00
										 |  |  |       CreateEvent(nullptr, TRUE, FALSE, name.c_str())); | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  |   ::SetEvent(wait_event.Get()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Wait for parent process to quit.
 | 
					
						
							|  |  |  |   WaitForSingleObject(parent_process.Get(), INFINITE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int LaunchProgram(const StringVector& relauncher_args, | 
					
						
							| 
									
										
										
										
											2016-06-02 20:32:29 +09:00
										 |  |  |                   const StringVector& argv) { | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  |   base::LaunchOptions options; | 
					
						
							| 
									
										
										
										
											2016-06-02 20:32:29 +09:00
										 |  |  |   base::Process process = | 
					
						
							| 
									
										
										
										
											2016-06-02 21:10:39 +09:00
										 |  |  |       base::LaunchProcess(ArgvToCommandLineString(argv), options); | 
					
						
							| 
									
										
										
										
											2016-06-02 19:49:36 +09:00
										 |  |  |   return process.IsValid() ? 0 : 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 12:55:47 -07:00
										 |  |  | }  // namespace relauncher::internal
 |