|  | // Copyright 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 <errno.h> | 
|  | #include <signal.h> | 
|  | #include <stdint.h> | 
|  | #include <sys/resource.h> | 
|  | #include <sys/wait.h> | 
|  |  | 
|  | #include "base/files/scoped_file.h" | 
|  | #include "base/logging.h" | 
|  | #include "base/posix/eintr_wrapper.h" | 
|  | #include "base/process/kill.h" | 
|  | #include "base/threading/thread_restrictions.h" | 
|  | #include "build_config.h" | 
|  |  | 
|  | #if defined(OS_MACOSX) | 
|  | #include <sys/event.h> | 
|  | #endif | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | #if !defined(OS_NACL_NONSFI) | 
|  |  | 
|  | bool WaitpidWithTimeout(base::ProcessHandle handle, | 
|  | int* status, | 
|  | base::TimeDelta wait) { | 
|  | // This POSIX version of this function only guarantees that we wait no less | 
|  | // than |wait| for the process to exit.  The child process may | 
|  | // exit sometime before the timeout has ended but we may still block for up | 
|  | // to 256 milliseconds after the fact. | 
|  | // | 
|  | // waitpid() has no direct support on POSIX for specifying a timeout, you can | 
|  | // either ask it to block indefinitely or return immediately (WNOHANG). | 
|  | // When a child process terminates a SIGCHLD signal is sent to the parent. | 
|  | // Catching this signal would involve installing a signal handler which may | 
|  | // affect other parts of the application and would be difficult to debug. | 
|  | // | 
|  | // Our strategy is to call waitpid() once up front to check if the process | 
|  | // has already exited, otherwise to loop for |wait|, sleeping for | 
|  | // at most 256 milliseconds each time using usleep() and then calling | 
|  | // waitpid().  The amount of time we sleep starts out at 1 milliseconds, and | 
|  | // we double it every 4 sleep cycles. | 
|  | // | 
|  | // usleep() is speced to exit if a signal is received for which a handler | 
|  | // has been installed.  This means that when a SIGCHLD is sent, it will exit | 
|  | // depending on behavior external to this function. | 
|  | // | 
|  | // This function is used primarily for unit tests, if we want to use it in | 
|  | // the application itself it would probably be best to examine other routes. | 
|  |  | 
|  | if (wait == base::TimeDelta::Max()) { | 
|  | return HANDLE_EINTR(waitpid(handle, status, 0)) > 0; | 
|  | } | 
|  |  | 
|  | pid_t ret_pid = HANDLE_EINTR(waitpid(handle, status, WNOHANG)); | 
|  | static const int64_t kMaxSleepInMicroseconds = 1 << 18;  // ~256 milliseconds. | 
|  | int64_t max_sleep_time_usecs = 1 << 10;                  // ~1 milliseconds. | 
|  | int64_t double_sleep_time = 0; | 
|  |  | 
|  | // If the process hasn't exited yet, then sleep and try again. | 
|  | base::TimeTicks wakeup_time = base::TimeTicks::Now() + wait; | 
|  | while (ret_pid == 0) { | 
|  | base::TimeTicks now = base::TimeTicks::Now(); | 
|  | if (now > wakeup_time) | 
|  | break; | 
|  | // Guaranteed to be non-negative! | 
|  | int64_t sleep_time_usecs = (wakeup_time - now).InMicroseconds(); | 
|  | // Sleep for a bit while we wait for the process to finish. | 
|  | if (sleep_time_usecs > max_sleep_time_usecs) | 
|  | sleep_time_usecs = max_sleep_time_usecs; | 
|  |  | 
|  | // usleep() will return 0 and set errno to EINTR on receipt of a signal | 
|  | // such as SIGCHLD. | 
|  | usleep(sleep_time_usecs); | 
|  | ret_pid = HANDLE_EINTR(waitpid(handle, status, WNOHANG)); | 
|  |  | 
|  | if ((max_sleep_time_usecs < kMaxSleepInMicroseconds) && | 
|  | (double_sleep_time++ % 4 == 0)) { | 
|  | max_sleep_time_usecs *= 2; | 
|  | } | 
|  | } | 
|  |  | 
|  | return ret_pid > 0; | 
|  | } | 
|  |  | 
|  | #if defined(OS_MACOSX) | 
|  | // Using kqueue on Mac so that we can wait on non-child processes. | 
|  | // We can't use kqueues on child processes because we need to reap | 
|  | // our own children using wait. | 
|  | bool WaitForSingleNonChildProcess(base::ProcessHandle handle, | 
|  | base::TimeDelta wait) { | 
|  | DCHECK_GT(handle, 0); | 
|  |  | 
|  | base::ScopedFD kq(kqueue()); | 
|  | if (!kq.is_valid()) { | 
|  | DPLOG(ERROR) << "kqueue"; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | struct kevent change = {0}; | 
|  | EV_SET(&change, handle, EVFILT_PROC, EV_ADD, NOTE_EXIT, 0, NULL); | 
|  | int result = HANDLE_EINTR(kevent(kq.get(), &change, 1, NULL, 0, NULL)); | 
|  | if (result == -1) { | 
|  | if (errno == ESRCH) { | 
|  | // If the process wasn't found, it must be dead. | 
|  | return true; | 
|  | } | 
|  |  | 
|  | DPLOG(ERROR) << "kevent (setup " << handle << ")"; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Keep track of the elapsed time to be able to restart kevent if it's | 
|  | // interrupted. | 
|  | bool wait_forever = (wait == base::TimeDelta::Max()); | 
|  | base::TimeDelta remaining_delta; | 
|  | base::TimeTicks deadline; | 
|  | if (!wait_forever) { | 
|  | remaining_delta = wait; | 
|  | deadline = base::TimeTicks::Now() + remaining_delta; | 
|  | } | 
|  |  | 
|  | result = -1; | 
|  | struct kevent event = {0}; | 
|  |  | 
|  | do { | 
|  | struct timespec remaining_timespec; | 
|  | struct timespec* remaining_timespec_ptr; | 
|  | if (wait_forever) { | 
|  | remaining_timespec_ptr = NULL; | 
|  | } else { | 
|  | remaining_timespec = remaining_delta.ToTimeSpec(); | 
|  | remaining_timespec_ptr = &remaining_timespec; | 
|  | } | 
|  |  | 
|  | result = kevent(kq.get(), NULL, 0, &event, 1, remaining_timespec_ptr); | 
|  |  | 
|  | if (result == -1 && errno == EINTR) { | 
|  | if (!wait_forever) { | 
|  | remaining_delta = deadline - base::TimeTicks::Now(); | 
|  | } | 
|  | result = 0; | 
|  | } else { | 
|  | break; | 
|  | } | 
|  | } while (wait_forever || remaining_delta > base::TimeDelta()); | 
|  |  | 
|  | if (result < 0) { | 
|  | DPLOG(ERROR) << "kevent (wait " << handle << ")"; | 
|  | return false; | 
|  | } else if (result > 1) { | 
|  | DLOG(ERROR) << "kevent (wait " << handle << "): unexpected result " | 
|  | << result; | 
|  | return false; | 
|  | } else if (result == 0) { | 
|  | // Timed out. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | DCHECK_EQ(result, 1); | 
|  |  | 
|  | if (event.filter != EVFILT_PROC || | 
|  | (event.fflags & NOTE_EXIT) == 0 || | 
|  | event.ident != static_cast<uintptr_t>(handle)) { | 
|  | DLOG(ERROR) << "kevent (wait " << handle | 
|  | << "): unexpected event: filter=" << event.filter | 
|  | << ", fflags=" << event.fflags | 
|  | << ", ident=" << event.ident; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  | #endif  // OS_MACOSX | 
|  |  | 
|  | bool WaitForExitWithTimeoutImpl(base::ProcessHandle handle, | 
|  | int* exit_code, | 
|  | base::TimeDelta timeout) { | 
|  | const base::ProcessHandle our_pid = base::GetCurrentProcessHandle(); | 
|  | if (handle == our_pid) { | 
|  | // We won't be able to wait for ourselves to exit. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | const base::ProcessHandle parent_pid = base::GetParentProcessId(handle); | 
|  | const bool exited = (parent_pid < 0); | 
|  |  | 
|  | if (!exited && parent_pid != our_pid) { | 
|  | #if defined(OS_MACOSX) | 
|  | // On Mac we can wait on non child processes. | 
|  | return WaitForSingleNonChildProcess(handle, timeout); | 
|  | #else | 
|  | // Currently on Linux we can't handle non child processes. | 
|  | NOTIMPLEMENTED(); | 
|  | #endif  // OS_MACOSX | 
|  | } | 
|  |  | 
|  | int status; | 
|  | if (!WaitpidWithTimeout(handle, &status, timeout)) { | 
|  | // If multiple threads wait on the same |handle| then one wait will succeed | 
|  | // and the other will fail with errno set to ECHILD. | 
|  | return exited || (errno == ECHILD); | 
|  | } | 
|  | if (WIFSIGNALED(status)) { | 
|  | if (exit_code) | 
|  | *exit_code = -1; | 
|  | return true; | 
|  | } | 
|  | if (WIFEXITED(status)) { | 
|  | if (exit_code) | 
|  | *exit_code = WEXITSTATUS(status); | 
|  | return true; | 
|  | } | 
|  | return exited; | 
|  | } | 
|  | #endif  // !defined(OS_NACL_NONSFI) | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | Process::Process(ProcessHandle handle) : process_(handle) { | 
|  | } | 
|  |  | 
|  | Process::~Process() = default; | 
|  |  | 
|  | Process::Process(Process&& other) : process_(other.process_) { | 
|  | other.Close(); | 
|  | } | 
|  |  | 
|  | Process& Process::operator=(Process&& other) { | 
|  | process_ = other.process_; | 
|  | other.Close(); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::Current() { | 
|  | return Process(GetCurrentProcessHandle()); | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::Open(ProcessId pid) { | 
|  | if (pid == GetCurrentProcId()) | 
|  | return Current(); | 
|  |  | 
|  | // On POSIX process handles are the same as PIDs. | 
|  | return Process(pid); | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::OpenWithExtraPrivileges(ProcessId pid) { | 
|  | // On POSIX there are no privileges to set. | 
|  | return Open(pid); | 
|  | } | 
|  |  | 
|  | // static | 
|  | Process Process::DeprecatedGetProcessFromHandle(ProcessHandle handle) { | 
|  | DCHECK_NE(handle, GetCurrentProcessHandle()); | 
|  | return Process(handle); | 
|  | } | 
|  |  | 
|  | #if !defined(OS_LINUX) && !defined(OS_MACOSX) && !defined(OS_AIX) | 
|  | // static | 
|  | bool Process::CanBackgroundProcesses() { | 
|  | return false; | 
|  | } | 
|  | #endif  // !defined(OS_LINUX) && !defined(OS_MACOSX) && !defined(OS_AIX) | 
|  |  | 
|  | // static | 
|  | void Process::TerminateCurrentProcessImmediately(int exit_code) { | 
|  | _exit(exit_code); | 
|  | } | 
|  |  | 
|  | bool Process::IsValid() const { | 
|  | return process_ != kNullProcessHandle; | 
|  | } | 
|  |  | 
|  | ProcessHandle Process::Handle() const { | 
|  | return process_; | 
|  | } | 
|  |  | 
|  | Process Process::Duplicate() const { | 
|  | if (is_current()) | 
|  | return Current(); | 
|  |  | 
|  | return Process(process_); | 
|  | } | 
|  |  | 
|  | ProcessId Process::Pid() const { | 
|  | DCHECK(IsValid()); | 
|  | return GetProcId(process_); | 
|  | } | 
|  |  | 
|  | bool Process::is_current() const { | 
|  | return process_ == GetCurrentProcessHandle(); | 
|  | } | 
|  |  | 
|  | void Process::Close() { | 
|  | process_ = kNullProcessHandle; | 
|  | // if the process wasn't terminated (so we waited) or the state | 
|  | // wasn't already collected w/ a wait from process_utils, we're gonna | 
|  | // end up w/ a zombie when it does finally exit. | 
|  | } | 
|  |  | 
|  | #if !defined(OS_NACL_NONSFI) | 
|  | bool Process::Terminate(int exit_code, bool wait) const { | 
|  | // exit_code isn't supportable. | 
|  | DCHECK(IsValid()); | 
|  | CHECK_GT(process_, 0); | 
|  |  | 
|  | bool did_terminate = kill(process_, SIGTERM) == 0; | 
|  |  | 
|  | if (wait && did_terminate) { | 
|  | if (WaitForExitWithTimeout(TimeDelta::FromSeconds(60), nullptr)) | 
|  | return true; | 
|  | did_terminate = kill(process_, SIGKILL) == 0; | 
|  | if (did_terminate) | 
|  | return WaitForExit(nullptr); | 
|  | } | 
|  |  | 
|  | if (!did_terminate) | 
|  | DPLOG(ERROR) << "Unable to terminate process " << process_; | 
|  |  | 
|  | return did_terminate; | 
|  | } | 
|  | #endif  // !defined(OS_NACL_NONSFI) | 
|  |  | 
|  | bool Process::WaitForExit(int* exit_code) const { | 
|  | return WaitForExitWithTimeout(TimeDelta::Max(), exit_code); | 
|  | } | 
|  |  | 
|  | bool Process::WaitForExitWithTimeout(TimeDelta timeout, int* exit_code) const { | 
|  | if (!timeout.is_zero()) | 
|  | internal::AssertBaseSyncPrimitivesAllowed(); | 
|  |  | 
|  | int local_exit_code; | 
|  | bool exited = WaitForExitWithTimeoutImpl(Handle(), &local_exit_code, timeout); | 
|  | if (exited) { | 
|  | Exited(local_exit_code); | 
|  | if (exit_code) | 
|  | *exit_code = local_exit_code; | 
|  | } | 
|  | return exited; | 
|  | } | 
|  |  | 
|  | void Process::Exited(int exit_code) const {} | 
|  |  | 
|  | #if !defined(OS_LINUX) && !defined(OS_MACOSX) && !defined(OS_AIX) | 
|  | bool Process::IsProcessBackgrounded() const { | 
|  | // See SetProcessBackgrounded(). | 
|  | DCHECK(IsValid()); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool Process::SetProcessBackgrounded(bool value) { | 
|  | // Not implemented for POSIX systems other than Linux and Mac. With POSIX, if | 
|  | // we were to lower the process priority we wouldn't be able to raise it back | 
|  | // to its initial priority. | 
|  | NOTIMPLEMENTED(); | 
|  | return false; | 
|  | } | 
|  | #endif  // !defined(OS_LINUX) && !defined(OS_MACOSX) && !defined(OS_AIX) | 
|  |  | 
|  | int Process::GetPriority() const { | 
|  | DCHECK(IsValid()); | 
|  | return getpriority(PRIO_PROCESS, process_); | 
|  | } | 
|  |  | 
|  | }  // namespace base |