123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329 |
- // Copyright (c) 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 "chrome/browser/process_singleton.h"
- #include <shellapi.h>
- #include "base/base_paths.h"
- #include "base/bind.h"
- #include "base/command_line.h"
- #include "base/files/file_path.h"
- #include "base/files/file_util.h"
- #include "base/process/process.h"
- #include "base/process/process_info.h"
- #include "base/strings/string_number_conversions.h"
- #include "base/strings/stringprintf.h"
- #include "base/strings/utf_string_conversions.h"
- #include "base/time/time.h"
- #include "base/win/registry.h"
- #include "base/win/scoped_handle.h"
- #include "base/win/windows_version.h"
- #include "chrome/browser/chrome_process_finder_win.h"
- #include "content/public/common/result_codes.h"
- #include "net/base/escape.h"
- #include "ui/base/l10n/l10n_util.h"
- #include "ui/gfx/win/hwnd_util.h"
- namespace {
- const char kLockfile[] = "lockfile";
- // A helper class that acquires the given |mutex| while the AutoLockMutex is in
- // scope.
- class AutoLockMutex {
- public:
- explicit AutoLockMutex(HANDLE mutex) : mutex_(mutex) {
- DWORD result = ::WaitForSingleObject(mutex_, INFINITE);
- DPCHECK(result == WAIT_OBJECT_0) << "Result = " << result;
- }
- ~AutoLockMutex() {
- BOOL released = ::ReleaseMutex(mutex_);
- DPCHECK(released);
- }
- private:
- HANDLE mutex_;
- DISALLOW_COPY_AND_ASSIGN(AutoLockMutex);
- };
- // A helper class that releases the given |mutex| while the AutoUnlockMutex is
- // in scope and immediately re-acquires it when going out of scope.
- class AutoUnlockMutex {
- public:
- explicit AutoUnlockMutex(HANDLE mutex) : mutex_(mutex) {
- BOOL released = ::ReleaseMutex(mutex_);
- DPCHECK(released);
- }
- ~AutoUnlockMutex() {
- DWORD result = ::WaitForSingleObject(mutex_, INFINITE);
- DPCHECK(result == WAIT_OBJECT_0) << "Result = " << result;
- }
- private:
- HANDLE mutex_;
- DISALLOW_COPY_AND_ASSIGN(AutoUnlockMutex);
- };
- // Checks the visibility of the enumerated window and signals once a visible
- // window has been found.
- BOOL CALLBACK BrowserWindowEnumeration(HWND window, LPARAM param) {
- bool* result = reinterpret_cast<bool*>(param);
- *result = ::IsWindowVisible(window) != 0;
- // Stops enumeration if a visible window has been found.
- return !*result;
- }
- // Convert Command line string to argv.
- base::CommandLine::StringVector CommandLineStringToArgv(
- const std::wstring& command_line_string) {
- int num_args = 0;
- wchar_t** args = NULL;
- args = ::CommandLineToArgvW(command_line_string.c_str(), &num_args);
- base::CommandLine::StringVector argv;
- for (int i = 0; i < num_args; ++i)
- argv.push_back(std::wstring(args[i]));
- LocalFree(args);
- return argv;
- }
- bool ParseCommandLine(const COPYDATASTRUCT* cds,
- base::CommandLine::StringVector* parsed_command_line,
- base::FilePath* current_directory) {
- // We should have enough room for the shortest command (min_message_size)
- // and also be a multiple of wchar_t bytes. The shortest command
- // possible is L"START\0\0" (empty current directory and command line).
- static const int min_message_size = 7;
- if (cds->cbData < min_message_size * sizeof(wchar_t) ||
- cds->cbData % sizeof(wchar_t) != 0) {
- LOG(WARNING) << "Invalid WM_COPYDATA, length = " << cds->cbData;
- return false;
- }
- // We split the string into 4 parts on NULLs.
- DCHECK(cds->lpData);
- const std::wstring msg(static_cast<wchar_t*>(cds->lpData),
- cds->cbData / sizeof(wchar_t));
- const std::wstring::size_type first_null = msg.find_first_of(L'\0');
- if (first_null == 0 || first_null == std::wstring::npos) {
- // no NULL byte, don't know what to do
- LOG(WARNING) << "Invalid WM_COPYDATA, length = " << msg.length()
- << ", first null = " << first_null;
- return false;
- }
- // Decode the command, which is everything until the first NULL.
- if (msg.substr(0, first_null) == L"START") {
- // Another instance is starting parse the command line & do what it would
- // have done.
- VLOG(1) << "Handling STARTUP request from another process";
- const std::wstring::size_type second_null =
- msg.find_first_of(L'\0', first_null + 1);
- if (second_null == std::wstring::npos || first_null == msg.length() - 1 ||
- second_null == msg.length()) {
- LOG(WARNING) << "Invalid format for start command, we need a string in 4 "
- "parts separated by NULLs";
- return false;
- }
- // Get current directory.
- *current_directory =
- base::FilePath(msg.substr(first_null + 1, second_null - first_null));
- const std::wstring::size_type third_null =
- msg.find_first_of(L'\0', second_null + 1);
- if (third_null == std::wstring::npos || third_null == msg.length()) {
- LOG(WARNING) << "Invalid format for start command, we need a string in 4 "
- "parts separated by NULLs";
- }
- // Get command line.
- const std::wstring cmd_line =
- msg.substr(second_null + 1, third_null - second_null);
- *parsed_command_line = CommandLineStringToArgv(cmd_line);
- return true;
- }
- return false;
- }
- bool ProcessLaunchNotification(
- const ProcessSingleton::NotificationCallback& notification_callback,
- UINT message,
- WPARAM wparam,
- LPARAM lparam,
- LRESULT* result) {
- if (message != WM_COPYDATA)
- return false;
- // Handle the WM_COPYDATA message from another process.
- const COPYDATASTRUCT* cds = reinterpret_cast<COPYDATASTRUCT*>(lparam);
- base::CommandLine::StringVector parsed_command_line;
- base::FilePath current_directory;
- if (!ParseCommandLine(cds, &parsed_command_line, ¤t_directory)) {
- *result = TRUE;
- return true;
- }
- *result = notification_callback.Run(parsed_command_line, current_directory)
- ? TRUE
- : FALSE;
- return true;
- }
- bool TerminateAppWithError() {
- // TODO: This is called when the secondary process can't ping the primary
- // process. Need to find out what to do here.
- return false;
- }
- } // namespace
- ProcessSingleton::ProcessSingleton(
- const base::FilePath& user_data_dir,
- const NotificationCallback& notification_callback)
- : notification_callback_(notification_callback),
- is_virtualized_(false),
- lock_file_(INVALID_HANDLE_VALUE),
- user_data_dir_(user_data_dir),
- should_kill_remote_process_callback_(base::Bind(&TerminateAppWithError)) {
- // The user_data_dir may have not been created yet.
- base::CreateDirectoryAndGetError(user_data_dir, nullptr);
- }
- ProcessSingleton::~ProcessSingleton() {
- DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
- if (lock_file_ != INVALID_HANDLE_VALUE)
- ::CloseHandle(lock_file_);
- }
- // Code roughly based on Mozilla.
- ProcessSingleton::NotifyResult ProcessSingleton::NotifyOtherProcess() {
- if (is_virtualized_)
- return PROCESS_NOTIFIED; // We already spawned the process in this case.
- if (lock_file_ == INVALID_HANDLE_VALUE && !remote_window_) {
- return LOCK_ERROR;
- } else if (!remote_window_) {
- return PROCESS_NONE;
- }
- switch (chrome::AttemptToNotifyRunningChrome(remote_window_, false)) {
- case chrome::NOTIFY_SUCCESS:
- return PROCESS_NOTIFIED;
- case chrome::NOTIFY_FAILED:
- remote_window_ = NULL;
- return PROCESS_NONE;
- case chrome::NOTIFY_WINDOW_HUNG:
- // Fall through and potentially terminate the hung browser.
- break;
- }
- DWORD process_id = 0;
- DWORD thread_id = ::GetWindowThreadProcessId(remote_window_, &process_id);
- if (!thread_id || !process_id) {
- remote_window_ = NULL;
- return PROCESS_NONE;
- }
- base::Process process = base::Process::Open(process_id);
- // The window is hung. Scan for every window to find a visible one.
- bool visible_window = false;
- ::EnumThreadWindows(thread_id, &BrowserWindowEnumeration,
- reinterpret_cast<LPARAM>(&visible_window));
- // If there is a visible browser window, ask the user before killing it.
- if (visible_window && !should_kill_remote_process_callback_.Run()) {
- // The user denied. Quit silently.
- return PROCESS_NOTIFIED;
- }
- // Time to take action. Kill the browser process.
- process.Terminate(content::RESULT_CODE_HUNG, true);
- remote_window_ = NULL;
- return PROCESS_NONE;
- }
- ProcessSingleton::NotifyResult ProcessSingleton::NotifyOtherProcessOrCreate() {
- ProcessSingleton::NotifyResult result = PROCESS_NONE;
- if (!Create()) {
- result = NotifyOtherProcess();
- if (result == PROCESS_NONE)
- result = PROFILE_IN_USE;
- }
- return result;
- }
- void ProcessSingleton::StartListeningOnSocket() {}
- void ProcessSingleton::OnBrowserReady() {}
- // Look for a Chrome instance that uses the same profile directory. If there
- // isn't one, create a message window with its title set to the profile
- // directory path.
- bool ProcessSingleton::Create() {
- static const wchar_t kMutexName[] = L"Local\\AtomProcessSingletonStartup!";
- remote_window_ = chrome::FindRunningChromeWindow(user_data_dir_);
- if (!remote_window_) {
- // Make sure we will be the one and only process creating the window.
- // We use a named Mutex since we are protecting against multi-process
- // access. As documented, it's clearer to NOT request ownership on creation
- // since it isn't guaranteed we will get it. It is better to create it
- // without ownership and explicitly get the ownership afterward.
- base::win::ScopedHandle only_me(::CreateMutex(NULL, FALSE, kMutexName));
- if (!only_me.IsValid()) {
- DPLOG(FATAL) << "CreateMutex failed";
- return false;
- }
- AutoLockMutex auto_lock_only_me(only_me.Get());
- // We now own the mutex so we are the only process that can create the
- // window at this time, but we must still check if someone created it
- // between the time where we looked for it above and the time the mutex
- // was given to us.
- remote_window_ = chrome::FindRunningChromeWindow(user_data_dir_);
- if (!remote_window_) {
- // We have to make sure there is no Chrome instance running on another
- // machine that uses the same profile.
- base::FilePath lock_file_path = user_data_dir_.AppendASCII(kLockfile);
- lock_file_ =
- ::CreateFile(lock_file_path.value().c_str(), GENERIC_WRITE,
- FILE_SHARE_READ, NULL, CREATE_ALWAYS,
- FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, NULL);
- DWORD error = ::GetLastError();
- LOG_IF(WARNING, lock_file_ != INVALID_HANDLE_VALUE &&
- error == ERROR_ALREADY_EXISTS)
- << "Lock file exists but is writable.";
- LOG_IF(ERROR, lock_file_ == INVALID_HANDLE_VALUE)
- << "Lock file can not be created! Error code: " << error;
- if (lock_file_ != INVALID_HANDLE_VALUE) {
- // Set the window's title to the path of our user data directory so
- // other Chrome instances can decide if they should forward to us.
- bool result = window_.CreateNamed(
- base::Bind(&ProcessLaunchNotification, notification_callback_),
- user_data_dir_.value());
- // NB: Ensure that if the primary app gets started as elevated
- // admin inadvertently, secondary windows running not as elevated
- // will still be able to send messages
- ::ChangeWindowMessageFilterEx(window_.hwnd(), WM_COPYDATA, MSGFLT_ALLOW,
- NULL);
- CHECK(result && window_.hwnd());
- }
- }
- }
- return window_.hwnd() != NULL;
- }
- void ProcessSingleton::Cleanup() {}
- void ProcessSingleton::OverrideShouldKillRemoteProcessCallbackForTesting(
- const ShouldKillRemoteProcessCallback& display_dialog_callback) {
- should_kill_remote_process_callback_ = display_dialog_callback;
- }
|