Index: remoting/host/client_session.cc |
diff --git a/remoting/host/client_session.cc b/remoting/host/client_session.cc |
index 540277ecc6d041fa2262597be002c0ee79d5833c..d415edf2b99278a2a3f85b271131030f24e44e4e 100644 |
--- a/remoting/host/client_session.cc |
+++ b/remoting/host/client_session.cc |
@@ -7,6 +7,16 @@ |
#include <algorithm> |
#include "base/message_loop_proxy.h" |
+#include "remoting/base/encoder.h" |
+#include "remoting/base/encoder_row_based.h" |
+#include "remoting/base/encoder_vp8.h" |
+#include "remoting/codec/audio_encoder.h" |
+#include "remoting/codec/audio_encoder_verbatim.h" |
+#include "remoting/host/audio_capturer.h" |
+#include "remoting/host/audio_scheduler.h" |
+#include "remoting/host/desktop_environment.h" |
+#include "remoting/host/event_executor.h" |
+#include "remoting/host/screen_recorder.h" |
#include "remoting/host/video_frame_capturer.h" |
#include "remoting/proto/control.pb.h" |
#include "remoting/proto/event.pb.h" |
@@ -17,23 +27,29 @@ namespace remoting { |
ClientSession::ClientSession( |
EventHandler* event_handler, |
+ scoped_refptr<base::SingleThreadTaskRunner> capture_task_runner, |
+ scoped_refptr<base::SingleThreadTaskRunner> encode_task_runner, |
+ scoped_refptr<base::SingleThreadTaskRunner> network_task_runner, |
scoped_ptr<protocol::ConnectionToClient> connection, |
- protocol::ClipboardStub* host_clipboard_stub, |
- protocol::InputStub* host_input_stub, |
- VideoFrameCapturer* capturer, |
+ scoped_ptr<DesktopEnvironment> desktop_environment, |
const base::TimeDelta& max_duration) |
: event_handler_(event_handler), |
connection_(connection.Pass()), |
+ desktop_environment_(desktop_environment.Pass()), |
client_jid_(connection_->session()->jid()), |
is_authenticated_(false), |
- host_clipboard_stub_(host_clipboard_stub), |
- host_input_stub_(host_input_stub), |
+ host_clipboard_stub_(desktop_environment_->event_executor()), |
+ host_input_stub_(desktop_environment_->event_executor()), |
input_tracker_(host_input_stub_), |
remote_input_filter_(&input_tracker_), |
mouse_input_filter_(&remote_input_filter_), |
client_clipboard_factory_(clipboard_echo_filter_.client_filter()), |
- capturer_(capturer), |
- max_duration_(max_duration) { |
+ capturer_(desktop_environment_->video_capturer()), |
+ max_duration_(max_duration), |
+ capture_task_runner_(capture_task_runner), |
+ encode_task_runner_(encode_task_runner), |
+ network_task_runner_(network_task_runner), |
+ active_recorders_(0) { |
connection_->SetEventHandler(this); |
// TODO(sergeyu): Currently ConnectionToClient expects stubs to be |
@@ -45,9 +61,6 @@ ClientSession::ClientSession( |
clipboard_echo_filter_.set_host_stub(host_clipboard_stub_); |
} |
-ClientSession::~ClientSession() { |
-} |
- |
void ClientSession::InjectKeyEvent(const protocol::KeyEvent& event) { |
DCHECK(CalledOnValidThread()); |
auth_input_filter_.InjectKeyEvent(event); |
@@ -107,6 +120,34 @@ void ClientSession::OnConnectionChannelsConnected( |
DCHECK(CalledOnValidThread()); |
DCHECK_EQ(connection_.get(), connection); |
SetDisableInputs(false); |
+ |
+ // Create a ScreenRecorder passing the message loops that it should run on. |
Wez
2012/09/06 00:13:23
nit: ...ScreenRecorder passing... -> ScreenRecorde
alexeypa (please no reviews)
2012/09/06 17:12:03
Done.
|
+ Encoder* encoder = CreateEncoder(connection_->session()->config()); |
+ video_recorder_ = new ScreenRecorder( |
+ capture_task_runner_, |
+ encode_task_runner_, |
+ network_task_runner_, |
+ desktop_environment_->video_capturer(), |
+ encoder); |
+ ++active_recorders_; |
+ |
+ if (connection_->session()->config().is_audio_enabled()) { |
+ scoped_ptr<AudioEncoder> audio_encoder = |
+ CreateAudioEncoder(connection_->session()->config()); |
+ audio_scheduler_ = new AudioScheduler( |
+ capture_task_runner_, |
+ network_task_runner_, |
+ desktop_environment_->audio_capturer(), |
+ audio_encoder.Pass(), |
+ connection_->audio_stub()); |
+ ++active_recorders_; |
+ } |
+ |
+ // Start the session. |
+ video_recorder_->AddConnection(connection_.get()); |
+ video_recorder_->Start(); |
+ desktop_environment_->Start(CreateClipboardProxy()); |
+ |
event_handler_->OnSessionChannelsConnected(this); |
} |
@@ -131,6 +172,10 @@ void ClientSession::OnSequenceNumberUpdated( |
protocol::ConnectionToClient* connection, int64 sequence_number) { |
DCHECK(CalledOnValidThread()); |
DCHECK_EQ(connection_.get(), connection); |
+ |
+ if (video_recorder_.get()) |
+ video_recorder_->UpdateSequenceNumber(sequence_number); |
+ |
event_handler_->OnSessionSequenceNumber(this, sequence_number); |
} |
@@ -153,6 +198,28 @@ void ClientSession::Disconnect() { |
connection_->Disconnect(); |
} |
+void ClientSession::StopAndDelete() { |
+ DCHECK(CalledOnValidThread()); |
+ |
+ if (audio_scheduler_.get()) { |
+ audio_scheduler_->OnClientDisconnected(); |
+ audio_scheduler_->Stop(base::Bind(&ClientSession::OnRecorderStopped, |
+ base::Unretained(this))); |
+ audio_scheduler_ = NULL; |
+ } |
+ |
+ if (video_recorder_.get()) { |
+ video_recorder_->RemoveConnection(connection_.get()); |
+ video_recorder_->Stop(base::Bind(&ClientSession::OnRecorderStopped, |
+ base::Unretained(this))); |
+ video_recorder_ = NULL; |
+ } |
+ |
+ if (!active_recorders_) { |
+ delete this; |
+ } |
+} |
+ |
void ClientSession::LocalMouseMoved(const SkIPoint& mouse_pos) { |
DCHECK(CalledOnValidThread()); |
remote_input_filter_.LocalMouseMoved(mouse_pos); |
@@ -172,6 +239,11 @@ void ClientSession::SetDisableInputs(bool disable_inputs) { |
} |
} |
+ClientSession::~ClientSession() { |
+ DCHECK(audio_scheduler_.get() == NULL); |
+ DCHECK(video_recorder_.get() == NULL); |
+} |
+ |
scoped_ptr<protocol::ClipboardStub> ClientSession::CreateClipboardProxy() { |
DCHECK(CalledOnValidThread()); |
@@ -181,4 +253,47 @@ scoped_ptr<protocol::ClipboardStub> ClientSession::CreateClipboardProxy() { |
base::MessageLoopProxy::current())); |
} |
+void ClientSession::OnRecorderStopped() { |
+ if (!network_task_runner_->BelongsToCurrentThread()) { |
+ network_task_runner_->PostTask( |
+ FROM_HERE, base::Bind(&ClientSession::OnRecorderStopped, |
+ base::Unretained(this))); |
+ return; |
+ } |
+ |
+ --active_recorders_; |
+ DCHECK_GE(active_recorders_, 0); |
+ |
+ StopAndDelete(); |
+} |
+ |
+// TODO(sergeyu): Move this to SessionManager? |
+// static |
+Encoder* ClientSession::CreateEncoder(const protocol::SessionConfig& config) { |
+ const protocol::ChannelConfig& video_config = config.video_config(); |
+ |
+ if (video_config.codec == protocol::ChannelConfig::CODEC_VERBATIM) { |
+ return EncoderRowBased::CreateVerbatimEncoder(); |
+ } else if (video_config.codec == protocol::ChannelConfig::CODEC_ZIP) { |
+ return EncoderRowBased::CreateZlibEncoder(); |
+ } else if (video_config.codec == protocol::ChannelConfig::CODEC_VP8) { |
+ return new remoting::EncoderVp8(); |
+ } |
+ |
+ return NULL; |
+} |
+ |
+// static |
+scoped_ptr<AudioEncoder> ClientSession::CreateAudioEncoder( |
+ const protocol::SessionConfig& config) { |
+ const protocol::ChannelConfig& audio_config = config.audio_config(); |
+ |
+ if (audio_config.codec == protocol::ChannelConfig::CODEC_VERBATIM) { |
+ return scoped_ptr<AudioEncoder>(new AudioEncoderVerbatim()); |
+ } |
+ |
+ NOTIMPLEMENTED(); |
+ return scoped_ptr<AudioEncoder>(NULL); |
+} |
+ |
} // namespace remoting |