Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(149)

Unified Diff: remoting/host/win/wts_session_process_launcher.cc

Issue 10828181: [Chromoting] Moving common logic responsible for launching child processes to WorkerProcessLauncher… (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: CR feedback. Created 8 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: remoting/host/win/wts_session_process_launcher.cc
diff --git a/remoting/host/win/wts_session_process_launcher.cc b/remoting/host/win/wts_session_process_launcher.cc
index 6548fd4fbe2149a6275606ee306e3abb0bcc65d3..e57d37c93e03cd573d8db8ddb0e28543c9353256 100644
--- a/remoting/host/win/wts_session_process_launcher.cc
+++ b/remoting/host/win/wts_session_process_launcher.cc
@@ -23,6 +23,7 @@
#include "base/process_util.h"
#include "base/rand_util.h"
#include "base/stringprintf.h"
+#include "base/utf_string_conversions.h"
#include "base/win/scoped_handle.h"
#include "ipc/ipc_channel_proxy.h"
#include "ipc/ipc_message.h"
@@ -46,9 +47,6 @@ const int kMinLaunchDelaySeconds = 1;
const FilePath::CharType kMe2meHostBinaryName[] =
FILE_PATH_LITERAL("remoting_me2me_host.exe");
-// Match the pipe name prefix used by Chrome IPC channels.
-const wchar_t kChromePipeNamePrefix[] = L"\\\\.\\pipe\\chrome.";
-
// The IPC channel name is passed to the host in the command line.
const char kChromotingIpcSwitchName[] = "chromoting-ipc";
@@ -59,70 +57,7 @@ const char* kCopiedSwitchNames[] = {
// The security descriptor of the Chromoting IPC channel. It gives full access
// to LocalSystem and denies access by anyone else.
-const wchar_t kChromotingChannelSecurityDescriptor[] =
- L"O:SYG:SYD:(A;;GA;;;SY)";
-
-// Generates random channel ID.
-// N.B. Stolen from src/content/common/child_process_host_impl.cc
-std::wstring GenerateRandomChannelId(void* instance) {
- return base::StringPrintf(L"%d.%p.%d",
- base::GetCurrentProcId(), instance,
- base::RandInt(0, std::numeric_limits<int>::max()));
-}
-
-// Creates the server end of the Chromoting IPC channel.
-// N.B. This code is based on IPC::Channel's implementation.
-bool CreatePipeForIpcChannel(void* instance,
- std::wstring* channel_name_out,
- ScopedHandle* pipe_out) {
- // Create security descriptor for the channel.
- SECURITY_ATTRIBUTES security_attributes;
- security_attributes.nLength = sizeof(security_attributes);
- security_attributes.bInheritHandle = FALSE;
-
- ULONG security_descriptor_length = 0;
- if (!ConvertStringSecurityDescriptorToSecurityDescriptorW(
- kChromotingChannelSecurityDescriptor,
- SDDL_REVISION_1,
- reinterpret_cast<PSECURITY_DESCRIPTOR*>(
- &security_attributes.lpSecurityDescriptor),
- &security_descriptor_length)) {
- LOG_GETLASTERROR(ERROR) <<
- "Failed to create a security descriptor for the Chromoting IPC channel";
- return false;
- }
-
- // Generate a random channel name.
- std::wstring channel_name(GenerateRandomChannelId(instance));
-
- // Convert it to the pipe name.
- std::wstring pipe_name(kChromePipeNamePrefix);
- pipe_name.append(channel_name);
-
- // Create the server end of the pipe. This code should match the code in
- // IPC::Channel with exception of passing a non-default security descriptor.
- HANDLE pipe = CreateNamedPipeW(pipe_name.c_str(),
- PIPE_ACCESS_DUPLEX | FILE_FLAG_OVERLAPPED |
- FILE_FLAG_FIRST_PIPE_INSTANCE,
- PIPE_TYPE_BYTE | PIPE_READMODE_BYTE,
- 1,
- IPC::Channel::kReadBufferSize,
- IPC::Channel::kReadBufferSize,
- 5000,
- &security_attributes);
- if (pipe == INVALID_HANDLE_VALUE) {
- LOG_GETLASTERROR(ERROR) <<
- "Failed to create the server end of the Chromoting IPC channel";
- LocalFree(security_attributes.lpSecurityDescriptor);
- return false;
- }
-
- LocalFree(security_attributes.lpSecurityDescriptor);
-
- *channel_name_out = channel_name;
- pipe_out->Set(pipe);
- return true;
-}
+const char kChromotingChannelSecurityDescriptor[] = "O:SYG:SYD:(A;;GA;;;SY)";
} // namespace
@@ -134,151 +69,157 @@ WtsSessionProcessLauncher::WtsSessionProcessLauncher(
scoped_refptr<base::SingleThreadTaskRunner> main_message_loop,
scoped_refptr<base::SingleThreadTaskRunner> ipc_message_loop)
: Stoppable(main_message_loop, stopped_callback),
+ attached_(false),
main_message_loop_(main_message_loop),
ipc_message_loop_(ipc_message_loop),
- monitor_(monitor),
- state_(StateDetached) {
+ monitor_(monitor) {
monitor_->AddWtsConsoleObserver(this);
}
WtsSessionProcessLauncher::~WtsSessionProcessLauncher() {
monitor_->RemoveWtsConsoleObserver(this);
- if (state_ != StateDetached) {
- OnSessionDetached();
- }
- DCHECK(state_ == StateDetached);
+ DCHECK(!attached_);
DCHECK(!timer_.IsRunning());
- DCHECK(process_.handle() == NULL);
- DCHECK(process_watcher_.GetWatchedObject() == NULL);
- DCHECK(chromoting_channel_.get() == NULL);
}
void WtsSessionProcessLauncher::LaunchProcess() {
DCHECK(main_message_loop_->BelongsToCurrentThread());
- DCHECK(state_ == StateStarting);
+ DCHECK(attached_);
+ DCHECK(launcher_.get() == NULL);
DCHECK(!timer_.IsRunning());
- DCHECK(process_.handle() == NULL);
- DCHECK(process_watcher_.GetWatchedObject() == NULL);
- DCHECK(chromoting_channel_.get() == NULL);
+ DCHECK(!worker_process_.IsValid());
launch_time_ = base::Time::Now();
+ launcher_.reset(new WorkerProcessLauncher(
+ this,
+ base::Bind(&WtsSessionProcessLauncher::OnLauncherStopped,
+ base::Unretained(this)),
+ main_message_loop_,
+ ipc_message_loop_));
+ launcher_->Start(kChromotingChannelSecurityDescriptor);
+}
+
+void WtsSessionProcessLauncher::OnLauncherStopped() {
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
+
+ DWORD exit_code;
+ if (!::GetExitCodeProcess(worker_process_, &exit_code)) {
+ LOG_GETLASTERROR(INFO)
+ << "Failed to query the exit code of the worker process";
+ exit_code = CONTROL_C_EXIT;
+ }
+
+ launcher_.reset(NULL);
+ worker_process_.Close();
+
+ // Do not relaunch the worker process if the caller has asked us to stop.
+ if (stoppable_state() != Stoppable::kRunning) {
+ CompleteStopping();
+ return;
+ }
+
+ // Stop trying to restart the worker process if its process exited due to
+ // misconfiguration.
+ if (kMinPermanentErrorExitCode <= exit_code &&
+ exit_code <= kMaxPermanentErrorExitCode) {
+ Stop();
+ return;
+ }
+
+ // Try to restart the worker process if we are still attached to a session.
+ if (attached_) {
+ // Expand the backoff interval if the process has died quickly or reset it
+ // if it was up longer than the maximum backoff delay.
+ base::TimeDelta delta = base::Time::Now() - launch_time_;
+ if (delta < base::TimeDelta() ||
+ delta >= base::TimeDelta::FromSeconds(kMaxLaunchDelaySeconds)) {
+ launch_backoff_ = base::TimeDelta();
+ } else {
+ launch_backoff_ = std::max(
+ launch_backoff_ * 2, TimeDelta::FromSeconds(kMinLaunchDelaySeconds));
+ launch_backoff_ = std::min(
+ launch_backoff_, TimeDelta::FromSeconds(kMaxLaunchDelaySeconds));
+ }
+
+ // Try to launch the worker process.
+ timer_.Start(FROM_HERE, launch_backoff_,
+ this, &WtsSessionProcessLauncher::LaunchProcess);
+ }
+}
+
+bool WtsSessionProcessLauncher::DoLaunchProcess(
+ const std::string& channel_name,
+ ScopedHandle* process_exit_event_out) {
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
+ DCHECK(!worker_process_.IsValid());
// Construct the host binary name.
FilePath dir_path;
if (!PathService::Get(base::DIR_EXE, &dir_path)) {
LOG(ERROR) << "Failed to get the executable file name.";
- Stop();
- return;
+ return false;
}
FilePath host_binary = dir_path.Append(kMe2meHostBinaryName);
- std::wstring channel_name;
- ScopedHandle pipe;
- if (CreatePipeForIpcChannel(this, &channel_name, &pipe)) {
- // Wrap the pipe into an IPC channel.
- chromoting_channel_.reset(new IPC::ChannelProxy(
- IPC::ChannelHandle(pipe.Get()),
- IPC::Channel::MODE_SERVER,
- this,
- ipc_message_loop_));
-
- // Create the host process command line passing the name of the IPC channel
- // to use and copying known switches from the service's command line.
- CommandLine command_line(host_binary);
- command_line.AppendSwitchNative(kChromotingIpcSwitchName, channel_name);
- command_line.CopySwitchesFrom(*CommandLine::ForCurrentProcess(),
- kCopiedSwitchNames,
- _countof(kCopiedSwitchNames));
-
- // Try to launch the process and attach an object watcher to the returned
- // handle so that we get notified when the process terminates.
- if (LaunchProcessWithToken(host_binary,
- command_line.GetCommandLineString(),
- session_token_,
- &process_)) {
- if (process_watcher_.StartWatching(process_.handle(), this)) {
- state_ = StateAttached;
- return;
- } else {
- LOG(ERROR) << "Failed to arm the process watcher.";
- process_.Terminate(0);
- process_.Close();
- }
- }
+ // Create the host process command line passing the name of the IPC channel
+ // to use and copying known switches from the service's command line.
+ CommandLine command_line(host_binary);
+ command_line.AppendSwitchNative(kChromotingIpcSwitchName,
+ UTF8ToWide(channel_name));
+ command_line.CopySwitchesFrom(*CommandLine::ForCurrentProcess(),
+ kCopiedSwitchNames,
+ _countof(kCopiedSwitchNames));
+
+ // Try to launch the process and attach an object watcher to the returned
+ // handle so that we get notified when the process terminates.
+ if (!LaunchProcessWithToken(host_binary,
+ command_line.GetCommandLineString(),
+ session_token_,
+ &worker_process_)) {
+ return false;
+ }
- chromoting_channel_.reset();
+ ScopedHandle process_exit_event;
+ if (!DuplicateHandle(GetCurrentProcess(),
+ worker_process_,
+ GetCurrentProcess(),
+ process_exit_event.Receive(),
+ SYNCHRONIZE,
+ FALSE,
+ 0)) {
+ LOG_GETLASTERROR(ERROR) << "Failed to duplicate a handle";
+ DoKillProcess(CONTROL_C_EXIT);
+ return false;
}
- // Something went wrong. Try to launch the host again later. The attempts rate
- // is limited by exponential backoff.
- launch_backoff_ = std::max(launch_backoff_ * 2,
- TimeDelta::FromSeconds(kMinLaunchDelaySeconds));
- launch_backoff_ = std::min(launch_backoff_,
- TimeDelta::FromSeconds(kMaxLaunchDelaySeconds));
- timer_.Start(FROM_HERE, launch_backoff_,
- this, &WtsSessionProcessLauncher::LaunchProcess);
+ *process_exit_event_out = process_exit_event.Pass();
+ return true;
}
-void WtsSessionProcessLauncher::OnObjectSignaled(HANDLE object) {
- if (!main_message_loop_->BelongsToCurrentThread()) {
- main_message_loop_->PostTask(
- FROM_HERE, base::Bind(&WtsSessionProcessLauncher::OnObjectSignaled,
- base::Unretained(this), object));
- return;
- }
+void WtsSessionProcessLauncher::DoKillProcess(DWORD exit_code) {
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
- // It is possible that OnObjectSignaled() task will be queued by another
- // thread right before |process_watcher_| was stopped. It such a case it is
- // safe to ignore this notification.
- if (state_ != StateAttached) {
- return;
+ if (worker_process_.IsValid()) {
+ TerminateProcess(worker_process_, exit_code);
}
+}
- DCHECK(!timer_.IsRunning());
- DCHECK(process_.handle() != NULL);
- DCHECK(process_watcher_.GetWatchedObject() == NULL);
- DCHECK(chromoting_channel_.get() != NULL);
+void WtsSessionProcessLauncher::OnChannelConnected(DWORD peer_pid) {
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
- // Stop trying to restart the host if its process exited due to
- // misconfiguration.
- int exit_code;
- bool stop_trying =
- base::WaitForExitCodeWithTimeout(
- process_.handle(), &exit_code, base::TimeDelta()) &&
- kMinPermanentErrorExitCode <= exit_code &&
- exit_code <= kMaxPermanentErrorExitCode;
-
- // The host process has been terminated for some reason. The handle can now be
- // closed.
- process_.Close();
- chromoting_channel_.reset();
- state_ = StateStarting;
-
- if (stop_trying) {
+ DWORD expected_pid = GetProcessId(worker_process_);
+ if (peer_pid != expected_pid) {
+ LOG(ERROR)
+ << "Unexpected client connected: expected=" << expected_pid
+ << ", actual=" << peer_pid;
Stop();
- return;
- }
-
- // Expand the backoff interval if the process has died quickly or reset it if
- // it was up longer than the maximum backoff delay.
- base::TimeDelta delta = base::Time::Now() - launch_time_;
- if (delta < base::TimeDelta() ||
- delta >= base::TimeDelta::FromSeconds(kMaxLaunchDelaySeconds)) {
- launch_backoff_ = base::TimeDelta();
- } else {
- launch_backoff_ = std::max(launch_backoff_ * 2,
- TimeDelta::FromSeconds(kMinLaunchDelaySeconds));
- launch_backoff_ = std::min(launch_backoff_,
- TimeDelta::FromSeconds(kMaxLaunchDelaySeconds));
}
-
- // Try to restart the host.
- timer_.Start(FROM_HERE, launch_backoff_,
- this, &WtsSessionProcessLauncher::LaunchProcess);
}
bool WtsSessionProcessLauncher::OnMessageReceived(const IPC::Message& message) {
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
+
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(WtsSessionProcessLauncher, message)
IPC_MESSAGE_HANDLER(ChromotingHostMsg_SendSasToConsole,
@@ -289,14 +230,9 @@ bool WtsSessionProcessLauncher::OnMessageReceived(const IPC::Message& message) {
}
void WtsSessionProcessLauncher::OnSendSasToConsole() {
- if (!main_message_loop_->BelongsToCurrentThread()) {
- main_message_loop_->PostTask(
- FROM_HERE, base::Bind(&WtsSessionProcessLauncher::OnSendSasToConsole,
- base::Unretained(this)));
- return;
- }
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
- if (state_ == StateAttached) {
+ if (attached_) {
if (sas_injector_.get() == NULL) {
sas_injector_ = SasInjector::Create();
}
@@ -314,68 +250,43 @@ void WtsSessionProcessLauncher::OnSessionAttached(uint32 session_id) {
return;
}
- DCHECK(state_ == StateDetached);
+ DCHECK(!attached_);
DCHECK(!timer_.IsRunning());
- DCHECK(process_.handle() == NULL);
- DCHECK(process_watcher_.GetWatchedObject() == NULL);
- DCHECK(chromoting_channel_.get() == NULL);
+
+ attached_ = true;
// Create a session token for the launched process.
if (!CreateSessionToken(session_id, &session_token_))
return;
// Now try to launch the host.
- state_ = StateStarting;
LaunchProcess();
}
void WtsSessionProcessLauncher::OnSessionDetached() {
DCHECK(main_message_loop_->BelongsToCurrentThread());
- DCHECK(state_ == StateDetached ||
- state_ == StateStarting ||
- state_ == StateAttached);
-
- switch (state_) {
- case StateDetached:
- DCHECK(!timer_.IsRunning());
- DCHECK(process_.handle() == NULL);
- DCHECK(process_watcher_.GetWatchedObject() == NULL);
- DCHECK(chromoting_channel_.get() == NULL);
- break;
-
- case StateStarting:
- DCHECK(process_.handle() == NULL);
- DCHECK(process_watcher_.GetWatchedObject() == NULL);
- DCHECK(chromoting_channel_.get() == NULL);
-
- timer_.Stop();
- launch_backoff_ = base::TimeDelta();
- state_ = StateDetached;
- break;
-
- case StateAttached:
- DCHECK(!timer_.IsRunning());
- DCHECK(process_.handle() != NULL);
- DCHECK(process_watcher_.GetWatchedObject() != NULL);
- DCHECK(chromoting_channel_.get() != NULL);
-
- process_watcher_.StopWatching();
- process_.Terminate(0);
- process_.Close();
- chromoting_channel_.reset();
- state_ = StateDetached;
- break;
- }
+ DCHECK(attached_);
+ attached_ = false;
+ launch_backoff_ = base::TimeDelta();
session_token_.Close();
+ timer_.Stop();
+
+ if (launcher_.get() != NULL) {
+ launcher_->Stop();
+ }
}
void WtsSessionProcessLauncher::DoStop() {
- if (state_ != StateDetached) {
+ DCHECK(main_message_loop_->BelongsToCurrentThread());
+
+ if (attached_) {
OnSessionDetached();
}
- CompleteStopping();
+ if (launcher_.get() == NULL) {
+ CompleteStopping();
+ }
}
} // namespace remoting

Powered by Google App Engine
This is Rietveld 408576698