diff options
-rw-r--r-- | base/process_util.h | 23 | ||||
-rw-r--r-- | base/process_util_linux.cc | 219 | ||||
-rw-r--r-- | base/process_util_unittest.cc | 41 |
3 files changed, 278 insertions, 5 deletions
diff --git a/base/process_util.h b/base/process_util.h index 503c2ec..50a5626 100644 --- a/base/process_util.h +++ b/base/process_util.h @@ -10,9 +10,13 @@ #include "base/basictypes.h" -#ifdef OS_WIN +#if defined(OS_WIN) #include <windows.h> #include <tlhelp32.h> +#elif defined(OS_LINUX) +#include <dirent.h> +#include <limits.h> +#include <sys/types.h> #endif #include <string> @@ -24,7 +28,12 @@ typedef PROCESSENTRY32 ProcessEntry; typedef IO_COUNTERS IoCounters; #elif defined(OS_POSIX) -typedef int ProcessEntry; +struct ProcessEntry { + int pid; + int ppid; + char szExeFile[NAME_MAX+1]; +}; + struct IoCounters { unsigned long long ReadOperationCount; unsigned long long WriteOperationCount; @@ -172,10 +181,16 @@ class NamedProcessIterator { void InitProcessEntry(ProcessEntry* entry); std::wstring executable_name_; -#ifdef OS_WIN + +#if defined(OS_WIN) HANDLE snapshot_; -#endif bool started_iteration_; +#elif defined(OS_LINUX) + DIR *procfs_dir_; +#elif defined(OS_MACOSX) + // probably kvm_t *kvmd_; +#endif + ProcessEntry entry_; const ProcessFilter* filter_; diff --git a/base/process_util_linux.cc b/base/process_util_linux.cc index f53ba3b..23db2bd 100644 --- a/base/process_util_linux.cc +++ b/base/process_util_linux.cc @@ -4,6 +4,8 @@ #include "base/process_util.h" +#include <ctype.h> +#include <dirent.h> #include <string> #include <sys/types.h> #include <sys/wait.h> @@ -12,6 +14,7 @@ #include "base/logging.h" #include "base/string_tokenizer.h" #include "base/string_util.h" +#include "base/time.h" namespace { @@ -35,7 +38,7 @@ bool LaunchApp(const std::vector<std::string>& argv, } argv_copy[argv.size()] = NULL; - int pid = vfork(); + int pid = fork(); if (pid == 0) { execv(argv_copy[0], argv_copy); } else if (pid < 0) { @@ -59,12 +62,226 @@ bool LaunchApp(const CommandLine& cl, return LaunchApp(cl.argv(), wait, process_handle); } +// Attempts to kill the process identified by the given process +// entry structure. Ignores specified exit_code; linux can't force that. +// Returns true if this is successful, false otherwise. +bool KillProcess(int process_id, int exit_code, bool wait) { + bool result = false; + + int status = kill(process_id, SIGTERM); + if (!status && wait) { + int tries = 60; + // The process may not end immediately due to pending I/O + while (tries-- > 0) { + int pid = waitpid(process_id, &status, WNOHANG); + if (pid == process_id) { + result = true; + break; + } + sleep(1); + } + } + if (!result) + DLOG(ERROR) << "Unable to terminate process."; + return result; +} + +bool DidProcessCrash(ProcessHandle handle) { + int status; + if (waitpid(handle, &status, WNOHANG)) { + // I feel like dancing! + return false; + } + + if (WIFSIGNALED(status)) { + int signum = WTERMSIG(status); + return (signum == SIGSEGV || signum == SIGILL || signum == SIGABRT || signum == SIGFPE); + } + + if (WIFEXITED(status)) { + int exitcode = WEXITSTATUS(status); + return (exitcode != 0); + } + + return false; +} + +NamedProcessIterator::NamedProcessIterator(const std::wstring& executable_name, + const ProcessFilter* filter) + : + executable_name_(executable_name), + filter_(filter) { + procfs_dir_ = opendir("/proc"); + } + +NamedProcessIterator::~NamedProcessIterator() { + if (procfs_dir_) { + closedir(procfs_dir_); + procfs_dir_ = 0; + } +} + +const ProcessEntry* NamedProcessIterator::NextProcessEntry() { + bool result = false; + do { + result = CheckForNextProcess(); + } while (result && !IncludeEntry()); + + if (result) + return &entry_; + + return NULL; +} + +bool NamedProcessIterator::CheckForNextProcess() { + // TODO(port): skip processes owned by different UID + + dirent* slot = 0; + const char* openparen; + const char* closeparen; + + // Arbitrarily guess that there will never be more than 200 non-process files in /proc. + // (Hardy has 53.) + int skipped = 0; + const int kSkipLimit = 200; + while (skipped < kSkipLimit) { + slot = readdir(procfs_dir_); + // all done looking through /proc? + if (!slot) + return false; + + // If not a process, keep looking for one. + bool notprocess = false; + int i; + for (i=0; i < NAME_MAX && slot->d_name[i]; ++i) { + if (!isdigit(slot->d_name[i])) { + notprocess = true; + break; + } + } + if (i == NAME_MAX || notprocess) { + skipped++; + continue; + } + + // Read the process's status. + char buf[NAME_MAX + 12]; + sprintf(buf, "/proc/%s/stat", slot->d_name); + FILE *fp = fopen(buf, "r"); + if (!fp) + return false; + const char* result = fgets(buf, sizeof(buf), fp); + fclose(fp); + if (!result) + return false; + + // Parse the status. It is formatted like this: + // %d (%s) %c %d ... + // pid (name) runstate ppid + // To avoid being fooled by names containing a closing paren, scan backwards. + openparen = strchr(buf, '('); + closeparen = strrchr(buf, ')'); + if (!openparen || !closeparen) + return false; + char runstate = closeparen[2]; + + // Is the process in 'Zombie' state, i.e. dead but waiting to be reaped? + // Allowed values: D R S T Z + if (runstate != 'Z') + break; + + // Nope, it's a zombie; somebody isn't cleaning up after their children. + // (e.g. WaitForProcessesToExit doesn't clean up after dead children yet.) + // There could be a lot of zombies, can't really decrement i here. + } + if (skipped >= kSkipLimit) { + NOTREACHED(); + return false; + } + + entry_.pid = atoi(slot->d_name); + entry_.ppid = atoi(closeparen+3); + + // TODO(port): read pid's commandline's $0, like killall does. + // Using the short name between openparen and closeparen won't work for long names! + int len = closeparen - openparen - 1; + if (len > NAME_MAX) + len = NAME_MAX; + memcpy(entry_.szExeFile, openparen + 1, len); + entry_.szExeFile[len] = 0; + + return true; +} + +bool NamedProcessIterator::IncludeEntry() { + // TODO(port): make this also work for non-ASCII filenames + bool result = strcmp(WideToASCII(executable_name_).c_str(), entry_.szExeFile) == 0 && + (!filter_ || filter_->Includes(entry_.pid, entry_.ppid)); + return result; +} + +int GetProcessCount(const std::wstring& executable_name, + const ProcessFilter* filter) { + int count = 0; + + NamedProcessIterator iter(executable_name, filter); + while (iter.NextProcessEntry()) + ++count; + return count; +} + +bool KillProcesses(const std::wstring& executable_name, int exit_code, + const ProcessFilter* filter) { + bool result = true; + const ProcessEntry* entry; + + NamedProcessIterator iter(executable_name, filter); + while ((entry = iter.NextProcessEntry()) != NULL) + result = KillProcess((*entry).pid, exit_code, true) && result; + + return result; +} + +bool WaitForProcessesToExit(const std::wstring& executable_name, + int wait_milliseconds, + const ProcessFilter* filter) { + bool result = false; + + // TODO(port): This is inefficient, but works if there are multiple procs. + // TODO(port): use waitpid to avoid leaving zombies around + + base::Time end_time = base::Time::Now() + base::TimeDelta::FromMilliseconds(wait_milliseconds); + do { + NamedProcessIterator iter(executable_name, filter); + if (!iter.NextProcessEntry()) { + result = true; + break; + } + // TODO(port): Improve resolution + sleep(1); + } while ((base::Time::Now() - end_time) > base::TimeDelta()); + + return result; +} + bool WaitForSingleProcess(ProcessHandle handle, int wait_milliseconds) { int status; waitpid(handle, &status, 0); return WIFEXITED(status); } +bool CleanupProcesses(const std::wstring& executable_name, + int wait_milliseconds, + int exit_code, + const ProcessFilter* filter) { + bool exited_cleanly = + process_util::WaitForProcessesToExit(executable_name, wait_milliseconds, + filter); + if (!exited_cleanly) + process_util::KillProcesses(executable_name, exit_code, filter); + return exited_cleanly; +} + /////////////////////////////////////////////////////////////////////////////// //// ProcessMetrics diff --git a/base/process_util_unittest.cc b/base/process_util_unittest.cc index 261eb46..1ba0ee2 100644 --- a/base/process_util_unittest.cc +++ b/base/process_util_unittest.cc @@ -5,6 +5,7 @@ #define _CRT_SECURE_NO_WARNINGS #include "base/multiprocess_test.h" +#include "base/platform_thread.h" #include "base/process_util.h" #include "testing/gtest/include/gtest/gtest.h" @@ -29,6 +30,46 @@ TEST_F(ProcessUtilTest, SpawnChild) { EXPECT_TRUE(process_util::WaitForSingleProcess(handle, 1000)); } +MULTIPROCESS_TEST_MAIN(SlowChildProcess) { + // Sleep until file "SlowChildProcess.die" is created. + FILE *fp; + do { + PlatformThread::Sleep(100); + fp = fopen("SlowChildProcess.die", "r"); + } while (!fp); + fclose(fp); + remove("SlowChildProcess.die"); + exit(0); + return 0; +} + +#if defined(OS_WIN) +#define EXE_SUFFIX ".exe" +#else +#define EXE_SUFFIX "" +#endif + +// TODO(port): finish port on Mac +#if defined(OS_MACOSX) +TEST_F(ProcessUtilTest, DISABLED_KillSlowChild) +#else +TEST_F(ProcessUtilTest, KillSlowChild) +#endif +{ + remove("SlowChildProcess.die"); + int oldcount = process_util::GetProcessCount(L"base_unittests" EXE_SUFFIX, 0); + + ProcessHandle handle = this->SpawnChild(L"SlowChildProcess"); + + ASSERT_NE(static_cast<ProcessHandle>(NULL), handle); + EXPECT_EQ(oldcount+1, process_util::GetProcessCount(L"base_unittests" EXE_SUFFIX, 0)); + FILE *fp = fopen("SlowChildProcess.die", "w"); + fclose(fp); + // TODO(port): do something less racy here + PlatformThread::Sleep(1000); + EXPECT_EQ(oldcount, process_util::GetProcessCount(L"base_unittests" EXE_SUFFIX, 0)); +} + // TODO(estade): if possible, port these 2 tests. #if defined(OS_WIN) TEST_F(ProcessUtilTest, EnableLFH) { |