| OLD | NEW |
| 1 // Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/common/process_watcher.h" | 5 #include "chrome/common/process_watcher.h" |
| 6 | 6 |
| 7 #include "base/basictypes.h" |
| 7 #include "base/message_loop.h" | 8 #include "base/message_loop.h" |
| 8 #include "base/object_watcher.h" | 9 #include "base/process.h" |
| 10 #include "base/process_util.h" |
| 9 #include "base/sys_info.h" | 11 #include "base/sys_info.h" |
| 12 #include "base/timer.h" |
| 13 #include "base/worker_pool.h" |
| 10 #include "chrome/app/result_codes.h" | 14 #include "chrome/app/result_codes.h" |
| 11 #include "chrome/common/env_vars.h" | 15 #include "chrome/common/env_vars.h" |
| 12 | 16 |
| 13 // Maximum amount of time (in milliseconds) to wait for the process to exit. | 17 // Maximum amount of time (in milliseconds) to wait for the process to exit. |
| 14 static const int kWaitInterval = 2000; | 18 static const int kWaitInterval = 2000; |
| 15 | 19 |
| 16 namespace { | 20 namespace { |
| 17 | 21 |
| 18 class TimerExpiredTask : public Task, public base::ObjectWatcher::Delegate { | 22 class TerminatorTask : public Task { |
| 19 public: | 23 public: |
| 20 explicit TimerExpiredTask(base::ProcessHandle process) : process_(process) { | 24 explicit TerminatorTask(base::ProcessHandle process) : process_(process) { |
| 21 watcher_.StartWatching(process_, this); | 25 timer_.Start(base::TimeDelta::FromMilliseconds(kWaitInterval), |
| 26 this, &TerminatorTask::KillProcess); |
| 22 } | 27 } |
| 23 | 28 |
| 24 virtual ~TimerExpiredTask() { | 29 virtual ~TerminatorTask() { |
| 25 if (process_) { | 30 if (process_) { |
| 26 KillProcess(); | 31 KillProcess(); |
| 27 DCHECK(!process_) << "Make sure to close the handle."; | 32 DCHECK(!process_); |
| 28 } | 33 } |
| 29 } | 34 } |
| 30 | 35 |
| 31 // Task --------------------------------------------------------------------- | |
| 32 | |
| 33 virtual void Run() { | 36 virtual void Run() { |
| 37 base::WaitForSingleProcess(process_, kWaitInterval); |
| 38 timer_.Stop(); |
| 34 if (process_) | 39 if (process_) |
| 35 KillProcess(); | 40 KillProcess(); |
| 36 } | 41 } |
| 37 | 42 |
| 38 // MessageLoop::Watcher ----------------------------------------------------- | |
| 39 | |
| 40 virtual void OnObjectSignaled(HANDLE object) { | |
| 41 // When we're called from KillProcess, the ObjectWatcher may still be | |
| 42 // watching. the process handle, so make sure it has stopped. | |
| 43 watcher_.StopWatching(); | |
| 44 | |
| 45 CloseHandle(process_); | |
| 46 process_ = NULL; | |
| 47 } | |
| 48 | |
| 49 private: | 43 private: |
| 50 void KillProcess() { | 44 void KillProcess() { |
| 51 if (base::SysInfo::HasEnvVar(env_vars::kHeadless)) { | 45 if (base::SysInfo::HasEnvVar(env_vars::kHeadless)) { |
| 52 // If running the distributed tests, give the renderer a little time | 46 // If running the distributed tests, give the renderer a little time |
| 53 // to figure out that the channel is shutdown and unwind. | 47 // to figure out that the channel is shutdown and unwind. |
| 54 if (WaitForSingleObject(process_, kWaitInterval) == WAIT_OBJECT_0) { | 48 if (base::WaitForSingleProcess(process_, kWaitInterval)) { |
| 55 OnObjectSignaled(process_); | 49 Cleanup(); |
| 56 return; | 50 return; |
| 57 } | 51 } |
| 58 } | 52 } |
| 59 | 53 |
| 60 // OK, time to get frisky. We don't actually care when the process | 54 // OK, time to get frisky. We don't actually care when the process |
| 61 // terminates. We just care that it eventually terminates, and that's what | 55 // terminates. We just care that it eventually terminates. |
| 62 // TerminateProcess should do for us. Don't check for the result code since | 56 base::KillProcess(base::Process(process_).pid(), |
| 63 // it fails quite often. This should be investigated eventually. | 57 ResultCodes::HUNG, |
| 64 TerminateProcess(process_, ResultCodes::HUNG); | 58 false /* don't wait */); |
| 65 | 59 |
| 66 // Now, just cleanup as if the process exited normally. | 60 Cleanup(); |
| 67 OnObjectSignaled(process_); | 61 } |
| 62 |
| 63 void Cleanup() { |
| 64 timer_.Stop(); |
| 65 base::CloseProcessHandle(process_); |
| 66 process_ = NULL; |
| 68 } | 67 } |
| 69 | 68 |
| 70 // The process that we are watching. | 69 // The process that we are watching. |
| 71 base::ProcessHandle process_; | 70 base::ProcessHandle process_; |
| 72 | 71 |
| 73 base::ObjectWatcher watcher_; | 72 base::OneShotTimer<TerminatorTask> timer_; |
| 74 | 73 |
| 75 DISALLOW_EVIL_CONSTRUCTORS(TimerExpiredTask); | 74 DISALLOW_COPY_AND_ASSIGN(TerminatorTask); |
| 76 }; | 75 }; |
| 77 | 76 |
| 78 } // namespace | 77 } // namespace |
| 79 | 78 |
| 80 // static | 79 // static |
| 81 void ProcessWatcher::EnsureProcessTerminated(base::ProcessHandle process) { | 80 void ProcessWatcher::EnsureProcessTerminated(base::ProcessHandle process) { |
| 82 DCHECK(process != GetCurrentProcess()); | 81 DCHECK(base::GetProcId(process) != base::GetCurrentProcId()); |
| 83 | 82 |
| 84 // If already signaled, then we are done! | 83 // Check if the process has already exited. |
| 85 if (WaitForSingleObject(process, 0) == WAIT_OBJECT_0) { | 84 if (base::WaitForSingleProcess(process, 0)) { |
| 86 CloseHandle(process); | 85 base::CloseProcessHandle(process); |
| 87 return; | 86 return; |
| 88 } | 87 } |
| 89 | 88 |
| 90 MessageLoop::current()->PostDelayedTask(FROM_HERE, | 89 WorkerPool::PostTask(FROM_HERE, new TerminatorTask(process), true); |
| 91 new TimerExpiredTask(process), | |
| 92 kWaitInterval); | |
| 93 } | 90 } |
| 94 | |
| OLD | NEW |