| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 // This file implements the Windows service controlling Me2Me host processes | 5 // This file implements the Windows service controlling Me2Me host processes |
| 6 // running within user sessions. | 6 // running within user sessions. |
| 7 | 7 |
| 8 #include "remoting/host/win/wts_session_process_delegate.h" | 8 #include "remoting/host/win/wts_session_process_delegate.h" |
| 9 | 9 |
| 10 #include <sddl.h> | |
| 11 #include <limits> | |
| 12 | |
| 13 #include "base/base_switches.h" | 10 #include "base/base_switches.h" |
| 14 #include "base/bind.h" | 11 #include "base/bind.h" |
| 15 #include "base/bind_helpers.h" | 12 #include "base/bind_helpers.h" |
| 16 #include "base/command_line.h" | 13 #include "base/command_line.h" |
| 17 #include "base/file_path.h" | 14 #include "base/file_path.h" |
| 18 #include "base/file_util.h" | 15 #include "base/file_util.h" |
| 19 #include "base/logging.h" | 16 #include "base/logging.h" |
| 20 #include "base/memory/scoped_ptr.h" | 17 #include "base/memory/scoped_ptr.h" |
| 21 #include "base/message_loop.h" | 18 #include "base/message_loop.h" |
| 22 #include "base/path_service.h" | |
| 23 #include "base/single_thread_task_runner.h" | 19 #include "base/single_thread_task_runner.h" |
| 24 #include "base/time.h" | |
| 25 #include "base/timer.h" | |
| 26 #include "base/utf_string_conversions.h" | 20 #include "base/utf_string_conversions.h" |
| 27 #include "base/win/scoped_handle.h" | 21 #include "base/win/scoped_handle.h" |
| 28 #include "base/win/windows_version.h" | 22 #include "base/win/windows_version.h" |
| 29 #include "ipc/ipc_channel.h" | 23 #include "ipc/ipc_channel.h" |
| 30 #include "ipc/ipc_channel_proxy.h" | 24 #include "ipc/ipc_channel_proxy.h" |
| 31 #include "ipc/ipc_message.h" | 25 #include "ipc/ipc_message.h" |
| 32 #include "remoting/host/host_exit_codes.h" | 26 #include "remoting/host/host_exit_codes.h" |
| 27 #include "remoting/host/ipc_consts.h" |
| 33 #include "remoting/host/win/launch_process_with_token.h" | 28 #include "remoting/host/win/launch_process_with_token.h" |
| 34 #include "remoting/host/win/worker_process_launcher.h" | 29 #include "remoting/host/win/worker_process_launcher.h" |
| 35 #include "remoting/host/win/wts_console_monitor.h" | 30 #include "remoting/host/win/wts_console_monitor.h" |
| 36 #include "remoting/host/worker_process_ipc_delegate.h" | 31 #include "remoting/host/worker_process_ipc_delegate.h" |
| 37 | 32 |
| 38 using base::TimeDelta; | |
| 39 using base::win::ScopedHandle; | 33 using base::win::ScopedHandle; |
| 40 | 34 |
| 41 const FilePath::CharType kDaemonBinaryName[] = | |
| 42 FILE_PATH_LITERAL("remoting_daemon.exe"); | |
| 43 | |
| 44 // The command line switch specifying the name of the daemon IPC endpoint. | |
| 45 const char kDaemonIpcSwitchName[] = "daemon-pipe"; | |
| 46 | |
| 47 const char kElevateSwitchName[] = "elevate"; | 35 const char kElevateSwitchName[] = "elevate"; |
| 48 | 36 |
| 49 // The command line parameters that should be copied from the service's command | 37 // The command line parameters that should be copied from the service's command |
| 50 // line to the host process. | 38 // line to the host process. |
| 51 const char* kCopiedSwitchNames[] = { | 39 const char* kCopiedSwitchNames[] = { |
| 52 "host-config", switches::kV, switches::kVModule }; | 40 "host-config", switches::kV, switches::kVModule }; |
| 53 | 41 |
| 54 namespace remoting { | 42 namespace remoting { |
| 55 | 43 |
| 56 // A private class actually implementing the functionality provided by | 44 // A private class actually implementing the functionality provided by |
| 57 // |WtsSessionProcessDelegate|. This class is ref-counted and implements | 45 // |WtsSessionProcessDelegate|. This class is ref-counted and implements |
| 58 // asynchronous fire-and-forget shutdown. | 46 // asynchronous fire-and-forget shutdown. |
| 59 class WtsSessionProcessDelegate::Core | 47 class WtsSessionProcessDelegate::Core |
| 60 : public base::RefCountedThreadSafe<WtsSessionProcessDelegate::Core>, | 48 : public base::RefCountedThreadSafe<WtsSessionProcessDelegate::Core>, |
| 61 public base::MessagePumpForIO::IOHandler, | 49 public base::MessagePumpForIO::IOHandler, |
| 62 public WorkerProcessLauncher::Delegate { | 50 public WorkerProcessLauncher::Delegate { |
| 63 public: | 51 public: |
| 64 // The caller must ensure that |delegate| remains valid at least until | 52 // The caller must ensure that |delegate| remains valid at least until |
| 65 // Stop() method has been called. | 53 // Stop() method has been called. |
| 66 Core(scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, | 54 Core(scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, |
| 67 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner, | 55 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner, |
| 68 const FilePath& binary_path, | 56 const FilePath& binary_path, |
| 69 bool launch_elevated); | 57 bool launch_elevated, |
| 58 const std::string& channel_security); |
| 70 | 59 |
| 71 // base::MessagePumpForIO::IOHandler implementation. | 60 // base::MessagePumpForIO::IOHandler implementation. |
| 72 virtual void OnIOCompleted(base::MessagePumpForIO::IOContext* context, | 61 virtual void OnIOCompleted(base::MessagePumpForIO::IOContext* context, |
| 73 DWORD bytes_transferred, | 62 DWORD bytes_transferred, |
| 74 DWORD error) OVERRIDE; | 63 DWORD error) OVERRIDE; |
| 75 | 64 |
| 65 // IPC::Sender implementation. |
| 66 virtual bool Send(IPC::Message* message) OVERRIDE; |
| 67 |
| 76 // WorkerProcessLauncher::Delegate implementation. | 68 // WorkerProcessLauncher::Delegate implementation. |
| 77 virtual DWORD GetExitCode() OVERRIDE; | 69 virtual DWORD GetExitCode() OVERRIDE; |
| 78 virtual void KillProcess(DWORD exit_code) OVERRIDE; | 70 virtual void KillProcess(DWORD exit_code) OVERRIDE; |
| 79 virtual bool LaunchProcess( | 71 virtual bool LaunchProcess( |
| 80 const std::string& channel_name, | 72 IPC::Listener* delegate, |
| 81 base::win::ScopedHandle* process_exit_event_out) OVERRIDE; | 73 base::win::ScopedHandle* process_exit_event_out) OVERRIDE; |
| 82 | 74 |
| 83 // Initializes the object returning true on success. | 75 // Initializes the object returning true on success. |
| 84 bool Initialize(uint32 session_id); | 76 bool Initialize(uint32 session_id); |
| 85 | 77 |
| 86 // Stops the object asynchronously. | 78 // Stops the object asynchronously. |
| 87 void Stop(); | 79 void Stop(); |
| 88 | 80 |
| 89 private: | 81 private: |
| 90 friend class base::RefCountedThreadSafe<Core>; | 82 friend class base::RefCountedThreadSafe<Core>; |
| (...skipping 18 matching lines...) Expand all Loading... |
| 109 | 101 |
| 110 // The task runner all public methods of this class should be called on. | 102 // The task runner all public methods of this class should be called on. |
| 111 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner_; | 103 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner_; |
| 112 | 104 |
| 113 // The task runner serving job object notifications. | 105 // The task runner serving job object notifications. |
| 114 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner_; | 106 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner_; |
| 115 | 107 |
| 116 // Path to the worker process binary. | 108 // Path to the worker process binary. |
| 117 FilePath binary_path_; | 109 FilePath binary_path_; |
| 118 | 110 |
| 111 // The server end of the IPC channel used to communicate to the worker |
| 112 // process. |
| 113 scoped_ptr<IPC::ChannelProxy> channel_; |
| 114 |
| 115 // Security descriptor (as SDDL) to be applied to |channel_|. |
| 116 std::string channel_security_; |
| 117 |
| 119 // The job object used to control the lifetime of child processes. | 118 // The job object used to control the lifetime of child processes. |
| 120 base::win::ScopedHandle job_; | 119 base::win::ScopedHandle job_; |
| 121 | 120 |
| 122 // True if the worker process should be launched elevated. | 121 // True if the worker process should be launched elevated. |
| 123 bool launch_elevated_; | 122 bool launch_elevated_; |
| 124 | 123 |
| 125 // A handle that becomes signalled once all processes associated with the job | 124 // A handle that becomes signalled once all processes associated with the job |
| 126 // have been terminated. | 125 // have been terminated. |
| 127 base::win::ScopedHandle process_exit_event_; | 126 base::win::ScopedHandle process_exit_event_; |
| 128 | 127 |
| 129 // The token to be used to launch a process in a different session. | 128 // The token to be used to launch a process in a different session. |
| 130 base::win::ScopedHandle session_token_; | 129 base::win::ScopedHandle session_token_; |
| 131 | 130 |
| 132 // True if Stop() has been called. | 131 // True if Stop() has been called. |
| 133 bool stopping_; | 132 bool stopping_; |
| 134 | 133 |
| 135 // The handle of the worker process, if launched. | 134 // The handle of the worker process, if launched. |
| 136 base::win::ScopedHandle worker_process_; | 135 base::win::ScopedHandle worker_process_; |
| 137 | 136 |
| 138 DISALLOW_COPY_AND_ASSIGN(Core); | 137 DISALLOW_COPY_AND_ASSIGN(Core); |
| 139 }; | 138 }; |
| 140 | 139 |
| 141 WtsSessionProcessDelegate::Core::Core( | 140 WtsSessionProcessDelegate::Core::Core( |
| 142 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, | 141 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, |
| 143 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner, | 142 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner, |
| 144 const FilePath& binary_path, | 143 const FilePath& binary_path, |
| 145 bool launch_elevated) | 144 bool launch_elevated, |
| 145 const std::string& channel_security) |
| 146 : main_task_runner_(main_task_runner), | 146 : main_task_runner_(main_task_runner), |
| 147 io_task_runner_(io_task_runner), | 147 io_task_runner_(io_task_runner), |
| 148 binary_path_(binary_path), | 148 binary_path_(binary_path), |
| 149 channel_security_(channel_security), |
| 149 launch_elevated_(launch_elevated), | 150 launch_elevated_(launch_elevated), |
| 150 stopping_(false) { | 151 stopping_(false) { |
| 151 DCHECK(main_task_runner_->BelongsToCurrentThread()); | 152 DCHECK(main_task_runner_->BelongsToCurrentThread()); |
| 152 } | 153 } |
| 153 | 154 |
| 154 void WtsSessionProcessDelegate::Core::OnIOCompleted( | 155 void WtsSessionProcessDelegate::Core::OnIOCompleted( |
| 155 base::MessagePumpForIO::IOContext* context, | 156 base::MessagePumpForIO::IOContext* context, |
| 156 DWORD bytes_transferred, | 157 DWORD bytes_transferred, |
| 157 DWORD error) { | 158 DWORD error) { |
| 158 DCHECK(io_task_runner_->BelongsToCurrentThread()); | 159 DCHECK(io_task_runner_->BelongsToCurrentThread()); |
| 159 | 160 |
| 160 // |bytes_transferred| is used in job object notifications to supply | 161 // |bytes_transferred| is used in job object notifications to supply |
| 161 // the message ID; |context| carries process ID. | 162 // the message ID; |context| carries process ID. |
| 162 main_task_runner_->PostTask(FROM_HERE, base::Bind( | 163 main_task_runner_->PostTask(FROM_HERE, base::Bind( |
| 163 &Core::OnJobNotification, this, bytes_transferred, | 164 &Core::OnJobNotification, this, bytes_transferred, |
| 164 reinterpret_cast<DWORD>(context))); | 165 reinterpret_cast<DWORD>(context))); |
| 165 } | 166 } |
| 166 | 167 |
| 168 bool WtsSessionProcessDelegate::Core::Send(IPC::Message* message) { |
| 169 DCHECK(main_task_runner_->BelongsToCurrentThread()); |
| 170 |
| 171 if (channel_.get()) { |
| 172 return channel_->Send(message); |
| 173 } else { |
| 174 delete message; |
| 175 return false; |
| 176 } |
| 177 } |
| 178 |
| 167 DWORD WtsSessionProcessDelegate::Core::GetExitCode() { | 179 DWORD WtsSessionProcessDelegate::Core::GetExitCode() { |
| 168 DCHECK(main_task_runner_->BelongsToCurrentThread()); | 180 DCHECK(main_task_runner_->BelongsToCurrentThread()); |
| 169 | 181 |
| 170 DWORD exit_code = CONTROL_C_EXIT; | 182 DWORD exit_code = CONTROL_C_EXIT; |
| 171 if (worker_process_.IsValid()) { | 183 if (worker_process_.IsValid()) { |
| 172 if (!::GetExitCodeProcess(worker_process_, &exit_code)) { | 184 if (!::GetExitCodeProcess(worker_process_, &exit_code)) { |
| 173 LOG_GETLASTERROR(INFO) | 185 LOG_GETLASTERROR(INFO) |
| 174 << "Failed to query the exit code of the worker process"; | 186 << "Failed to query the exit code of the worker process"; |
| 175 exit_code = CONTROL_C_EXIT; | 187 exit_code = CONTROL_C_EXIT; |
| 176 } | 188 } |
| 177 } | 189 } |
| 178 | 190 |
| 179 return exit_code; | 191 return exit_code; |
| 180 } | 192 } |
| 181 | 193 |
| 182 void WtsSessionProcessDelegate::Core::KillProcess(DWORD exit_code) { | 194 void WtsSessionProcessDelegate::Core::KillProcess(DWORD exit_code) { |
| 183 DCHECK(main_task_runner_->BelongsToCurrentThread()); | 195 DCHECK(main_task_runner_->BelongsToCurrentThread()); |
| 184 | 196 |
| 197 channel_.reset(); |
| 198 |
| 185 if (launch_elevated_) { | 199 if (launch_elevated_) { |
| 186 if (job_.IsValid()) { | 200 if (job_.IsValid()) { |
| 187 TerminateJobObject(job_, exit_code); | 201 TerminateJobObject(job_, exit_code); |
| 188 } | 202 } |
| 189 } else { | 203 } else { |
| 190 if (worker_process_.IsValid()) { | 204 if (worker_process_.IsValid()) { |
| 191 TerminateProcess(worker_process_, exit_code); | 205 TerminateProcess(worker_process_, exit_code); |
| 192 } | 206 } |
| 193 } | 207 } |
| 194 } | 208 } |
| 195 | 209 |
| 196 bool WtsSessionProcessDelegate::Core::LaunchProcess( | 210 bool WtsSessionProcessDelegate::Core::LaunchProcess( |
| 197 const std::string& channel_name, | 211 IPC::Listener* delegate, |
| 198 ScopedHandle* process_exit_event_out) { | 212 ScopedHandle* process_exit_event_out) { |
| 199 DCHECK(main_task_runner_->BelongsToCurrentThread()); | 213 DCHECK(main_task_runner_->BelongsToCurrentThread()); |
| 200 | 214 |
| 201 CommandLine command_line(CommandLine::NO_PROGRAM); | 215 CommandLine command_line(CommandLine::NO_PROGRAM); |
| 202 if (launch_elevated_) { | 216 if (launch_elevated_) { |
| 203 // The job object is not ready. Retry starting the host process later. | 217 // The job object is not ready. Retry starting the host process later. |
| 204 if (!job_.IsValid()) { | 218 if (!job_.IsValid()) { |
| 205 return false; | 219 return false; |
| 206 } | 220 } |
| 207 | 221 |
| 208 // Construct the helper binary name. | 222 // Construct the helper binary name. |
| 209 FilePath dir_path; | 223 FilePath daemon_binary; |
| 210 if (!PathService::Get(base::DIR_EXE, &dir_path)) { | 224 if (!GetInstalledBinaryPath(kDaemonBinaryName, &daemon_binary)) |
| 211 LOG(ERROR) << "Failed to get the executable file name."; | |
| 212 return false; | 225 return false; |
| 213 } | |
| 214 FilePath daemon_binary = dir_path.Append(kDaemonBinaryName); | |
| 215 | 226 |
| 216 // Create the command line passing the name of the IPC channel to use and | 227 // Create the command line passing the name of the IPC channel to use and |
| 217 // copying known switches from the caller's command line. | 228 // copying known switches from the caller's command line. |
| 218 command_line.SetProgram(daemon_binary); | 229 command_line.SetProgram(daemon_binary); |
| 219 command_line.AppendSwitchPath(kElevateSwitchName, binary_path_); | 230 command_line.AppendSwitchPath(kElevateSwitchName, binary_path_); |
| 220 | 231 |
| 221 CHECK(ResetEvent(process_exit_event_)); | 232 CHECK(ResetEvent(process_exit_event_)); |
| 222 } else { | 233 } else { |
| 223 command_line.SetProgram(binary_path_); | 234 command_line.SetProgram(binary_path_); |
| 224 } | 235 } |
| 225 | 236 |
| 237 // Create the server end of the IPC channel. |
| 238 scoped_ptr<IPC::ChannelProxy> channel; |
| 239 std::string channel_name = GenerateIpcChannelName(this); |
| 240 if (!CreateIpcChannel(channel_name, channel_security_, io_task_runner_, |
| 241 delegate, &channel)) |
| 242 return false; |
| 243 |
| 226 // Create the command line passing the name of the IPC channel to use and | 244 // Create the command line passing the name of the IPC channel to use and |
| 227 // copying known switches from the caller's command line. | 245 // copying known switches from the caller's command line. |
| 228 command_line.AppendSwitchNative(kDaemonIpcSwitchName, | 246 command_line.AppendSwitchNative(kDaemonPipeSwitchName, |
| 229 UTF8ToWide(channel_name)); | 247 UTF8ToWide(channel_name)); |
| 230 command_line.CopySwitchesFrom(*CommandLine::ForCurrentProcess(), | 248 command_line.CopySwitchesFrom(*CommandLine::ForCurrentProcess(), |
| 231 kCopiedSwitchNames, | 249 kCopiedSwitchNames, |
| 232 arraysize(kCopiedSwitchNames)); | 250 arraysize(kCopiedSwitchNames)); |
| 233 | 251 |
| 234 // Try to launch the process. | 252 // Try to launch the process. |
| 235 ScopedHandle worker_process; | 253 ScopedHandle worker_process; |
| 236 ScopedHandle worker_thread; | 254 ScopedHandle worker_thread; |
| 237 if (!LaunchProcessWithToken(command_line.GetProgram(), | 255 if (!LaunchProcessWithToken(command_line.GetProgram(), |
| 238 command_line.GetCommandLineString(), | 256 command_line.GetCommandLineString(), |
| 239 session_token_, | 257 session_token_, |
| 258 false, |
| 240 CREATE_SUSPENDED | CREATE_BREAKAWAY_FROM_JOB, | 259 CREATE_SUSPENDED | CREATE_BREAKAWAY_FROM_JOB, |
| 241 &worker_process, | 260 &worker_process, |
| 242 &worker_thread)) { | 261 &worker_thread)) { |
| 243 return false; | 262 return false; |
| 244 } | 263 } |
| 245 | 264 |
| 246 HANDLE local_process_exit_event; | 265 HANDLE local_process_exit_event; |
| 247 if (launch_elevated_) { | 266 if (launch_elevated_) { |
| 248 if (!AssignProcessToJobObject(job_, worker_process)) { | 267 if (!AssignProcessToJobObject(job_, worker_process)) { |
| 249 LOG_GETLASTERROR(ERROR) | 268 LOG_GETLASTERROR(ERROR) |
| (...skipping 22 matching lines...) Expand all Loading... |
| 272 GetCurrentProcess(), | 291 GetCurrentProcess(), |
| 273 process_exit_event.Receive(), | 292 process_exit_event.Receive(), |
| 274 SYNCHRONIZE, | 293 SYNCHRONIZE, |
| 275 FALSE, | 294 FALSE, |
| 276 0)) { | 295 0)) { |
| 277 LOG_GETLASTERROR(ERROR) << "Failed to duplicate a handle"; | 296 LOG_GETLASTERROR(ERROR) << "Failed to duplicate a handle"; |
| 278 KillProcess(CONTROL_C_EXIT); | 297 KillProcess(CONTROL_C_EXIT); |
| 279 return false; | 298 return false; |
| 280 } | 299 } |
| 281 | 300 |
| 301 channel_ = channel.Pass(); |
| 282 *process_exit_event_out = process_exit_event.Pass(); | 302 *process_exit_event_out = process_exit_event.Pass(); |
| 283 return true; | 303 return true; |
| 284 } | 304 } |
| 285 | 305 |
| 286 bool WtsSessionProcessDelegate::Core::Initialize(uint32 session_id) { | 306 bool WtsSessionProcessDelegate::Core::Initialize(uint32 session_id) { |
| 287 if (base::win::GetVersion() == base::win::VERSION_XP) | 307 if (base::win::GetVersion() == base::win::VERSION_XP) |
| 288 launch_elevated_ = false; | 308 launch_elevated_ = false; |
| 289 | 309 |
| 290 if (launch_elevated_) { | 310 if (launch_elevated_) { |
| 291 process_exit_event_.Set(CreateEvent(NULL, TRUE, FALSE, NULL)); | 311 process_exit_event_.Set(CreateEvent(NULL, TRUE, FALSE, NULL)); |
| (...skipping 120 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 412 worker_process_.Set(OpenProcess(PROCESS_QUERY_INFORMATION, FALSE, pid)); | 432 worker_process_.Set(OpenProcess(PROCESS_QUERY_INFORMATION, FALSE, pid)); |
| 413 break; | 433 break; |
| 414 } | 434 } |
| 415 } | 435 } |
| 416 | 436 |
| 417 WtsSessionProcessDelegate::WtsSessionProcessDelegate( | 437 WtsSessionProcessDelegate::WtsSessionProcessDelegate( |
| 418 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, | 438 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, |
| 419 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner, | 439 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner, |
| 420 const FilePath& binary_path, | 440 const FilePath& binary_path, |
| 421 uint32 session_id, | 441 uint32 session_id, |
| 422 bool launch_elevated) { | 442 bool launch_elevated, |
| 443 const std::string& channel_security) { |
| 423 core_ = new Core(main_task_runner, io_task_runner, binary_path, | 444 core_ = new Core(main_task_runner, io_task_runner, binary_path, |
| 424 launch_elevated); | 445 launch_elevated, channel_security); |
| 425 if (!core_->Initialize(session_id)) { | 446 if (!core_->Initialize(session_id)) { |
| 426 core_->Stop(); | 447 core_->Stop(); |
| 427 core_ = NULL; | 448 core_ = NULL; |
| 428 } | 449 } |
| 429 } | 450 } |
| 430 | 451 |
| 431 WtsSessionProcessDelegate::~WtsSessionProcessDelegate() { | 452 WtsSessionProcessDelegate::~WtsSessionProcessDelegate() { |
| 432 core_->Stop(); | 453 core_->Stop(); |
| 433 } | 454 } |
| 434 | 455 |
| 456 bool WtsSessionProcessDelegate::Send(IPC::Message* message) { |
| 457 return core_->Send(message); |
| 458 } |
| 459 |
| 435 DWORD WtsSessionProcessDelegate::GetExitCode() { | 460 DWORD WtsSessionProcessDelegate::GetExitCode() { |
| 436 if (!core_) | 461 if (!core_) |
| 437 return CONTROL_C_EXIT; | 462 return CONTROL_C_EXIT; |
| 438 | 463 |
| 439 return core_->GetExitCode(); | 464 return core_->GetExitCode(); |
| 440 } | 465 } |
| 441 | 466 |
| 442 void WtsSessionProcessDelegate::KillProcess(DWORD exit_code) { | 467 void WtsSessionProcessDelegate::KillProcess(DWORD exit_code) { |
| 443 if (core_) { | 468 if (core_) { |
| 444 core_->KillProcess(exit_code); | 469 core_->KillProcess(exit_code); |
| 445 } | 470 } |
| 446 } | 471 } |
| 447 | 472 |
| 448 bool WtsSessionProcessDelegate::LaunchProcess( | 473 bool WtsSessionProcessDelegate::LaunchProcess( |
| 449 const std::string& channel_name, | 474 IPC::Listener* delegate, |
| 450 base::win::ScopedHandle* process_exit_event_out) { | 475 base::win::ScopedHandle* process_exit_event_out) { |
| 451 if (!core_) | 476 if (!core_) |
| 452 return false; | 477 return false; |
| 453 | 478 |
| 454 return core_->LaunchProcess(channel_name, process_exit_event_out); | 479 return core_->LaunchProcess(delegate, process_exit_event_out); |
| 455 } | 480 } |
| 456 | 481 |
| 457 } // namespace remoting | 482 } // namespace remoting |
| OLD | NEW |