| // Copyright 2013 The Chromium Authors | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "chrome/browser/win/chrome_process_finder.h" | 
 |  | 
 | #include <windows.h> | 
 |  | 
 | #include <shellapi.h> | 
 |  | 
 | #include <string> | 
 | #include <string_view> | 
 |  | 
 | #include "base/check.h" | 
 | #include "base/command_line.h" | 
 | #include "base/files/file_path.h" | 
 | #include "base/files/file_util.h" | 
 | #include "base/numerics/safe_conversions.h" | 
 | #include "base/process/process.h" | 
 | #include "base/strings/strcat.h" | 
 | #include "base/strings/string_number_conversions.h" | 
 | #include "base/strings/utf_string_conversions.h" | 
 | #include "base/trace_event/trace_event.h" | 
 | #include "base/win/message_window.h" | 
 | #include "base/win/scoped_handle.h" | 
 | #include "base/win/win_util.h" | 
 | #include "base/win/windows_version.h" | 
 | #include "chrome/common/chrome_constants.h" | 
 | #include "chrome/common/chrome_switches.h" | 
 |  | 
 | namespace { | 
 |  | 
 | uint32_t g_timeout_in_milliseconds = 20 * 1000; | 
 |  | 
 | }  // namespace | 
 |  | 
 | HWND FindRunningChromeWindow(const base::FilePath& user_data_dir) { | 
 |   TRACE_EVENT0("startup", "FindRunningChromeWindow"); | 
 |   return base::win::MessageWindow::FindWindow(user_data_dir.value()); | 
 | } | 
 |  | 
 | NotifyChromeResult AttemptToNotifyRunningChrome(HWND remote_window) { | 
 |   TRACE_EVENT0("startup", "AttemptToNotifyRunningChrome"); | 
 |  | 
 |   DCHECK(remote_window); | 
 |   DWORD process_id = 0; | 
 |   DWORD thread_id = GetWindowThreadProcessId(remote_window, &process_id); | 
 |   if (!thread_id || !process_id) { | 
 |     TRACE_EVENT0( | 
 |         "startup", | 
 |         "AttemptToNotifyRunningChrome:GetWindowThreadProcessId failed"); | 
 |     return NotifyChromeResult::kFailed; | 
 |   } | 
 |  | 
 |   base::FilePath cur_dir; | 
 |   if (!base::GetCurrentDirectory(&cur_dir)) { | 
 |     TRACE_EVENT_INSTANT( | 
 |         "startup", "AttemptToNotifyRunningChrome:GetCurrentDirectory failed"); | 
 |     return NotifyChromeResult::kFailed; | 
 |   } | 
 |   base::CommandLine new_command_line(*base::CommandLine::ForCurrentProcess()); | 
 |   // If this process was launched from a shortcut, add the shortcut path to | 
 |   // the command line, so the process we rendezvous with can record the | 
 |   // launch mode correctly. | 
 |   STARTUPINFOW si = {sizeof(si)}; | 
 |   ::GetStartupInfoW(&si); | 
 |   if (si.dwFlags & STARTF_TITLEISLINKNAME) { | 
 |     new_command_line.AppendSwitchNative(switches::kSourceShortcut, si.lpTitle); | 
 |   } else if (si.dwFlags & STARTF_TITLEISAPPID) { | 
 |     new_command_line.AppendSwitch(switches::kSourceAppId); | 
 |   } | 
 |   // Send the command line to the remote chrome window. | 
 |   // Format is "START\0<<<current directory>>>\0<<<commandline>>>". | 
 |   std::wstring to_send = base::StrCat( | 
 |       {std::wstring_view{L"START\0", 6}, cur_dir.value(), | 
 |        std::wstring_view{L"\0", 1}, new_command_line.GetCommandLineString(), | 
 |        std::wstring_view{L"\0", 1}}); | 
 |  | 
 |   // Allow the current running browser window to make itself the foreground | 
 |   // window (otherwise it will just flash in the taskbar). | 
 |   ::AllowSetForegroundWindow(process_id); | 
 |  | 
 |   COPYDATASTRUCT cds; | 
 |   cds.dwData = 0; | 
 |   cds.cbData = static_cast<DWORD>((to_send.length() + 1) * sizeof(wchar_t)); | 
 |   cds.lpData = const_cast<wchar_t*>(to_send.c_str()); | 
 |   DWORD_PTR result = 0; | 
 |   { | 
 |     TRACE_EVENT0("startup", "AttemptToNotifyRunningChrome:SendMessage"); | 
 |     if (::SendMessageTimeout(remote_window, WM_COPYDATA, NULL, | 
 |                              reinterpret_cast<LPARAM>(&cds), SMTO_ABORTIFHUNG, | 
 |                              g_timeout_in_milliseconds, &result)) { | 
 |       return result ? NotifyChromeResult::kSuccess | 
 |                     : NotifyChromeResult::kFailed; | 
 |     } | 
 |   } | 
 |  | 
 |   // If SendMessageTimeout failed to send message consider this as kFailed. | 
 |   // Timeout can be represented as either ERROR_TIMEOUT or 0... | 
 |   // https://docs.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessagetimeoutw | 
 |   // Anecdotally which error code comes out seems to depend on whether this | 
 |   // process had non-empty data to deliver via |to_send| or not. | 
 |   const auto error = ::GetLastError(); | 
 |   const bool timed_out = (error == ERROR_TIMEOUT || error == 0); | 
 |   if (!timed_out) { | 
 |     TRACE_EVENT_INSTANT("startup", | 
 |                         "AttemptToNotifyRunningChrome:Error SendFailed"); | 
 |     return NotifyChromeResult::kFailed; | 
 |   } | 
 |  | 
 |   // It is possible that the process owning this window may have died by now. | 
 |   if (!::IsWindow(remote_window)) { | 
 |     TRACE_EVENT_INSTANT("startup", | 
 |                         "AttemptToNotifyRunningChrome:Error RemoteDied"); | 
 |     return NotifyChromeResult::kFailed; | 
 |   } | 
 |  | 
 |   // If the window couldn't be notified but still exists, assume it is hung. | 
 |   TRACE_EVENT_INSTANT("startup", | 
 |                       "AttemptToNotifyRunningChrome:Error RemoteHung"); | 
 |   return NotifyChromeResult::kWindowHung; | 
 | } | 
 |  | 
 | base::TimeDelta SetNotificationTimeoutForTesting(base::TimeDelta new_timeout) { | 
 |   base::TimeDelta old_timeout = base::Milliseconds(g_timeout_in_milliseconds); | 
 |   g_timeout_in_milliseconds = | 
 |       base::checked_cast<uint32_t>(new_timeout.InMilliseconds()); | 
 |   return old_timeout; | 
 | } |