1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "remoting/host/host_window_proxy.h"
6
7 #include "base/bind.h"
8 #include "base/location.h"
9 #include "base/logging.h"
10 #include "base/single_thread_task_runner.h"
11 #include "remoting/host/client_session_control.h"
12 #include "third_party/webrtc/modules/desktop_capture/desktop_geometry.h"
13
14 namespace remoting {
15
16 // Runs an instance of |HostWindow| on the |ui_task_runner_| thread.
17 class HostWindowProxy::Core
18 : public base::RefCountedThreadSafe<Core>,
19 public ClientSessionControl {
20 public:
21 Core(scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner,
22 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner,
23 scoped_ptr<HostWindow> host_window);
24
25 // Starts |host_window_| on the |ui_task_runner_| thread.
26 void Start(const base::WeakPtr<ClientSessionControl>& client_session_control);
27
28 // Destroys |host_window_| on the |ui_task_runner_| thread.
29 void Stop();
30
31 private:
32 friend class base::RefCountedThreadSafe<Core>;
33 virtual ~Core();
34
35 // Start() and Stop() equivalents called on the |ui_task_runner_| thread.
36 void StartOnUiThread(const std::string& client_jid);
37 void StopOnUiThread();
38
39 // ClientSessionControl interface.
40 virtual const std::string& client_jid() const OVERRIDE;
41 virtual void DisconnectSession() OVERRIDE;
42 virtual void OnLocalMouseMoved(
43 const webrtc::DesktopVector& position) OVERRIDE;
44 virtual void SetDisableInputs(bool disable_inputs) OVERRIDE;
45 virtual void ResetVideoPipeline() OVERRIDE;
46
47 // Task runner on which public methods of this class must be called.
48 scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner_;
49
50 // Task runner on which |host_window_| is running.
51 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner_;
52
53 // Stores the client's JID so it can be read on the |ui_task_runner_| thread.
54 std::string client_jid_;
55
56 // Used to notify the caller about the local user's actions on
57 // the |caller_task_runner| thread.
58 base::WeakPtr<ClientSessionControl> client_session_control_;
59
60 // The wrapped |HostWindow| instance running on the |ui_task_runner_| thread.
61 scoped_ptr<HostWindow> host_window_;
62
63 // Used to create the control pointer passed to |host_window_|.
64 base::WeakPtrFactory<ClientSessionControl> weak_factory_;
65
66 DISALLOW_COPY_AND_ASSIGN(Core);
67 };
68
HostWindowProxy(scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner,scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner,scoped_ptr<HostWindow> host_window)69 HostWindowProxy::HostWindowProxy(
70 scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner,
71 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner,
72 scoped_ptr<HostWindow> host_window) {
73 DCHECK(caller_task_runner->BelongsToCurrentThread());
74
75 // Detach |host_window| from the calling thread so that |Core| could run it on
76 // the |ui_task_runner_| thread.
77 host_window->DetachFromThread();
78 core_ = new Core(caller_task_runner, ui_task_runner, host_window.Pass());
79 }
80
~HostWindowProxy()81 HostWindowProxy::~HostWindowProxy() {
82 DCHECK(CalledOnValidThread());
83
84 core_->Stop();
85 }
86
Start(const base::WeakPtr<ClientSessionControl> & client_session_control)87 void HostWindowProxy::Start(
88 const base::WeakPtr<ClientSessionControl>& client_session_control) {
89 DCHECK(CalledOnValidThread());
90
91 core_->Start(client_session_control);
92 }
93
Core(scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner,scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner,scoped_ptr<HostWindow> host_window)94 HostWindowProxy::Core::Core(
95 scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner,
96 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner,
97 scoped_ptr<HostWindow> host_window)
98 : caller_task_runner_(caller_task_runner),
99 ui_task_runner_(ui_task_runner),
100 host_window_(host_window.Pass()),
101 weak_factory_(this) {
102 DCHECK(caller_task_runner->BelongsToCurrentThread());
103 }
104
Start(const base::WeakPtr<ClientSessionControl> & client_session_control)105 void HostWindowProxy::Core::Start(
106 const base::WeakPtr<ClientSessionControl>& client_session_control) {
107 DCHECK(caller_task_runner_->BelongsToCurrentThread());
108 DCHECK(!client_session_control_.get());
109 DCHECK(client_session_control.get());
110
111 client_session_control_ = client_session_control;
112 ui_task_runner_->PostTask(
113 FROM_HERE, base::Bind(&Core::StartOnUiThread, this,
114 client_session_control->client_jid()));
115 }
116
Stop()117 void HostWindowProxy::Core::Stop() {
118 DCHECK(caller_task_runner_->BelongsToCurrentThread());
119
120 ui_task_runner_->PostTask(FROM_HERE, base::Bind(&Core::StopOnUiThread, this));
121 }
122
~Core()123 HostWindowProxy::Core::~Core() {
124 DCHECK(!host_window_);
125 }
126
StartOnUiThread(const std::string & client_jid)127 void HostWindowProxy::Core::StartOnUiThread(const std::string& client_jid) {
128 DCHECK(ui_task_runner_->BelongsToCurrentThread());
129 DCHECK(client_jid_.empty());
130
131 client_jid_ = client_jid;
132 host_window_->Start(weak_factory_.GetWeakPtr());
133 }
134
StopOnUiThread()135 void HostWindowProxy::Core::StopOnUiThread() {
136 DCHECK(ui_task_runner_->BelongsToCurrentThread());
137
138 host_window_.reset();
139 }
140
client_jid() const141 const std::string& HostWindowProxy::Core::client_jid() const {
142 DCHECK(ui_task_runner_->BelongsToCurrentThread());
143
144 return client_jid_;
145 }
146
DisconnectSession()147 void HostWindowProxy::Core::DisconnectSession() {
148 if (!caller_task_runner_->BelongsToCurrentThread()) {
149 caller_task_runner_->PostTask(FROM_HERE,
150 base::Bind(&Core::DisconnectSession, this));
151 return;
152 }
153
154 if (client_session_control_.get())
155 client_session_control_->DisconnectSession();
156 }
157
OnLocalMouseMoved(const webrtc::DesktopVector & position)158 void HostWindowProxy::Core::OnLocalMouseMoved(
159 const webrtc::DesktopVector& position) {
160 if (!caller_task_runner_->BelongsToCurrentThread()) {
161 caller_task_runner_->PostTask(
162 FROM_HERE, base::Bind(&Core::OnLocalMouseMoved, this, position));
163 return;
164 }
165
166 if (client_session_control_.get())
167 client_session_control_->OnLocalMouseMoved(position);
168 }
169
SetDisableInputs(bool disable_inputs)170 void HostWindowProxy::Core::SetDisableInputs(bool disable_inputs) {
171 if (!caller_task_runner_->BelongsToCurrentThread()) {
172 caller_task_runner_->PostTask(
173 FROM_HERE, base::Bind(&Core::SetDisableInputs, this, disable_inputs));
174 return;
175 }
176
177 if (client_session_control_.get())
178 client_session_control_->SetDisableInputs(disable_inputs);
179 }
180
ResetVideoPipeline()181 void HostWindowProxy::Core::ResetVideoPipeline() {
182 // ResetVideoPipeline is only used by HostExtensionSession implementations.
183 NOTREACHED();
184 }
185
186 } // namespace remoting
187