|  | // Copyright (c) 2011 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 "base/process/process.h" | 
|  |  | 
|  | #include "base/debug/activity_tracker.h" | 
|  | #include "base/logging.h" | 
|  | #include "base/numerics/safe_conversions.h" | 
|  | #include "base/process/kill.h" | 
|  | #include "base/threading/thread_restrictions.h" | 
|  |  | 
|  | #include <windows.h> | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | DWORD kBasicProcessAccess = | 
|  | PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION | SYNCHRONIZE; | 
|  |  | 
|  | } // namespace | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | Process::Process(ProcessHandle handle) | 
|  | : process_(handle), is_current_process_(false) { | 
|  | CHECK_NE(handle, ::GetCurrentProcess()); | 
|  | } | 
|  |  | 
|  | Process::Process(Process&& other) | 
|  | : process_(other.process_.Take()), | 
|  | is_current_process_(other.is_current_process_) { | 
|  | other.Close(); | 
|  | } | 
|  |  | 
|  | Process::~Process() { | 
|  | } | 
|  |  | 
|  | Process& Process::operator=(Process&& other) { | 
|  | DCHECK_NE(this, &other); | 
|  | process_.Set(other.process_.Take()); | 
|  | is_current_process_ = other.is_current_process_; | 
|  | other.Close(); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::Current() { | 
|  | Process process; | 
|  | process.is_current_process_ = true; | 
|  | return process; | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::Open(ProcessId pid) { | 
|  | return Process(::OpenProcess(kBasicProcessAccess, FALSE, pid)); | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::OpenWithExtraPrivileges(ProcessId pid) { | 
|  | DWORD access = kBasicProcessAccess | PROCESS_DUP_HANDLE | PROCESS_VM_READ; | 
|  | return Process(::OpenProcess(access, FALSE, pid)); | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::OpenWithAccess(ProcessId pid, DWORD desired_access) { | 
|  | return Process(::OpenProcess(desired_access, FALSE, pid)); | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::DeprecatedGetProcessFromHandle(ProcessHandle handle) { | 
|  | DCHECK_NE(handle, ::GetCurrentProcess()); | 
|  | ProcessHandle out_handle; | 
|  | if (!::DuplicateHandle(GetCurrentProcess(), handle, | 
|  | GetCurrentProcess(), &out_handle, | 
|  | 0, FALSE, DUPLICATE_SAME_ACCESS)) { | 
|  | return Process(); | 
|  | } | 
|  | return Process(out_handle); | 
|  | } | 
|  |  | 
|  | // static | 
|  | bool Process::CanBackgroundProcesses() { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // static | 
|  | void Process::TerminateCurrentProcessImmediately(int exit_code) { | 
|  | ::TerminateProcess(GetCurrentProcess(), exit_code); | 
|  | // There is some ambiguity over whether the call above can return. Rather than | 
|  | // hitting confusing crashes later on we should crash right here. | 
|  | IMMEDIATE_CRASH(); | 
|  | } | 
|  |  | 
|  | bool Process::IsValid() const { | 
|  | return process_.IsValid() || is_current(); | 
|  | } | 
|  |  | 
|  | ProcessHandle Process::Handle() const { | 
|  | return is_current_process_ ? GetCurrentProcess() : process_.Get(); | 
|  | } | 
|  |  | 
|  | Process Process::Duplicate() const { | 
|  | if (is_current()) | 
|  | return Current(); | 
|  |  | 
|  | ProcessHandle out_handle; | 
|  | if (!IsValid() || !::DuplicateHandle(GetCurrentProcess(), | 
|  | Handle(), | 
|  | GetCurrentProcess(), | 
|  | &out_handle, | 
|  | 0, | 
|  | FALSE, | 
|  | DUPLICATE_SAME_ACCESS)) { | 
|  | return Process(); | 
|  | } | 
|  | return Process(out_handle); | 
|  | } | 
|  |  | 
|  | ProcessId Process::Pid() const { | 
|  | DCHECK(IsValid()); | 
|  | return GetProcId(Handle()); | 
|  | } | 
|  |  | 
|  | bool Process::is_current() const { | 
|  | return is_current_process_; | 
|  | } | 
|  |  | 
|  | void Process::Close() { | 
|  | is_current_process_ = false; | 
|  | if (!process_.IsValid()) | 
|  | return; | 
|  |  | 
|  | process_.Close(); | 
|  | } | 
|  |  | 
|  | bool Process::Terminate(int exit_code, bool wait) const { | 
|  | constexpr DWORD kWaitMs = 60 * 1000; | 
|  |  | 
|  | // exit_code cannot be implemented. | 
|  | DCHECK(IsValid()); | 
|  | bool result = (::TerminateProcess(Handle(), exit_code) != FALSE); | 
|  | if (result) { | 
|  | // The process may not end immediately due to pending I/O | 
|  | if (wait && ::WaitForSingleObject(Handle(), kWaitMs) != WAIT_OBJECT_0) | 
|  | DPLOG(ERROR) << "Error waiting for process exit"; | 
|  | Exited(exit_code); | 
|  | } else { | 
|  | // The process can't be terminated, perhaps because it has already | 
|  | // exited or is in the process of exiting. A non-zero timeout is necessary | 
|  | // here for the same reasons as above. | 
|  | DPLOG(ERROR) << "Unable to terminate process"; | 
|  | if (::WaitForSingleObject(Handle(), kWaitMs) == WAIT_OBJECT_0) { | 
|  | DWORD actual_exit; | 
|  | Exited(::GetExitCodeProcess(Handle(), &actual_exit) ? actual_exit | 
|  | : exit_code); | 
|  | result = true; | 
|  | } | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool Process::WaitForExit(int* exit_code) const { | 
|  | return WaitForExitWithTimeout(TimeDelta::FromMilliseconds(INFINITE), | 
|  | exit_code); | 
|  | } | 
|  |  | 
|  | bool Process::WaitForExitWithTimeout(TimeDelta timeout, int* exit_code) const { | 
|  | if (!timeout.is_zero()) | 
|  | internal::AssertBaseSyncPrimitivesAllowed(); | 
|  |  | 
|  | // Record the event that this thread is blocking upon (for hang diagnosis). | 
|  | base::debug::ScopedProcessWaitActivity process_activity(this); | 
|  |  | 
|  | // Limit timeout to INFINITE. | 
|  | DWORD timeout_ms = saturated_cast<DWORD>(timeout.InMilliseconds()); | 
|  | if (::WaitForSingleObject(Handle(), timeout_ms) != WAIT_OBJECT_0) | 
|  | return false; | 
|  |  | 
|  | DWORD temp_code;  // Don't clobber out-parameters in case of failure. | 
|  | if (!::GetExitCodeProcess(Handle(), &temp_code)) | 
|  | return false; | 
|  |  | 
|  | if (exit_code) | 
|  | *exit_code = temp_code; | 
|  |  | 
|  | Exited(temp_code); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void Process::Exited(int exit_code) const { | 
|  | base::debug::GlobalActivityTracker::RecordProcessExitIfEnabled(Pid(), | 
|  | exit_code); | 
|  | } | 
|  |  | 
|  | bool Process::IsProcessBackgrounded() const { | 
|  | DCHECK(IsValid()); | 
|  | DWORD priority = GetPriority(); | 
|  | if (priority == 0) | 
|  | return false;  // Failure case. | 
|  | return ((priority == BELOW_NORMAL_PRIORITY_CLASS) || | 
|  | (priority == IDLE_PRIORITY_CLASS)); | 
|  | } | 
|  |  | 
|  | bool Process::SetProcessBackgrounded(bool value) { | 
|  | DCHECK(IsValid()); | 
|  | // Vista and above introduce a real background mode, which not only | 
|  | // sets the priority class on the threads but also on the IO generated | 
|  | // by it. Unfortunately it can only be set for the calling process. | 
|  | DWORD priority; | 
|  | if (is_current()) { | 
|  | priority = value ? PROCESS_MODE_BACKGROUND_BEGIN : | 
|  | PROCESS_MODE_BACKGROUND_END; | 
|  | } else { | 
|  | priority = value ? IDLE_PRIORITY_CLASS : NORMAL_PRIORITY_CLASS; | 
|  | } | 
|  |  | 
|  | return (::SetPriorityClass(Handle(), priority) != 0); | 
|  | } | 
|  |  | 
|  | int Process::GetPriority() const { | 
|  | DCHECK(IsValid()); | 
|  | return ::GetPriorityClass(Handle()); | 
|  | } | 
|  |  | 
|  | }  // namespace base |