Index: remoting/protocol/webrtc_video_renderer_adapter.cc |
diff --git a/remoting/protocol/webrtc_video_renderer_adapter.cc b/remoting/protocol/webrtc_video_renderer_adapter.cc |
index 1320b9c6b1d86ef3d64cdf146f9c0dda93b57d8a..7d1f5adc4f969d61879faf3d1d023f9192759367 100644 |
--- a/remoting/protocol/webrtc_video_renderer_adapter.cc |
+++ b/remoting/protocol/webrtc_video_renderer_adapter.cc |
@@ -1,4 +1,4 @@ |
- // Copyright 2015 The Chromium Authors. All rights reserved. |
+// Copyright 2015 The Chromium Authors. All rights reserved. |
// Use of this source code is governed by a BSD-style license that can be |
// found in the LICENSE file. |
@@ -15,12 +15,12 @@ |
#include "base/task_runner_util.h" |
#include "base/threading/thread_task_runner_handle.h" |
#include "base/threading/worker_pool.h" |
+#include "remoting/protocol/client_video_stats_dispatcher.h" |
#include "remoting/protocol/frame_consumer.h" |
#include "remoting/protocol/frame_stats.h" |
#include "remoting/protocol/video_renderer.h" |
-#include "third_party/libyuv/include/libyuv/convert_argb.h" |
+#include "remoting/protocol/webrtc_transport.h" |
#include "third_party/libyuv/include/libyuv/convert_from.h" |
-#include "third_party/libyuv/include/libyuv/video_common.h" |
#include "third_party/webrtc/media/base/videoframe.h" |
#include "third_party/webrtc/modules/desktop_capture/desktop_frame.h" |
@@ -29,6 +29,9 @@ namespace protocol { |
namespace { |
+// Maximum number of ClientFrameStats instances to keep. |
+const int kMaxQueuedStats = 200; |
+ |
std::unique_ptr<webrtc::DesktopFrame> ConvertYuvToRgb( |
scoped_refptr<webrtc::VideoFrameBuffer> yuv_frame, |
std::unique_ptr<webrtc::DesktopFrame> rgb_frame, |
@@ -52,12 +55,27 @@ std::unique_ptr<webrtc::DesktopFrame> ConvertYuvToRgb( |
} // namespace |
WebrtcVideoRendererAdapter::WebrtcVideoRendererAdapter( |
- scoped_refptr<webrtc::MediaStreamInterface> media_stream, |
+ const std::string& label, |
VideoRenderer* video_renderer) |
- : media_stream_(std::move(media_stream)), |
+ : label_(label), |
video_renderer_(video_renderer), |
task_runner_(base::ThreadTaskRunnerHandle::Get()), |
- weak_factory_(this) { |
+ weak_factory_(this) {} |
+ |
+WebrtcVideoRendererAdapter::~WebrtcVideoRendererAdapter() { |
+ DCHECK(task_runner_->BelongsToCurrentThread()); |
+ |
+ webrtc::VideoTrackVector video_tracks = media_stream_->GetVideoTracks(); |
+ DCHECK(!video_tracks.empty()); |
+ video_tracks[0]->RemoveSink(this); |
+} |
+ |
+void WebrtcVideoRendererAdapter::SetMediaStream( |
+ scoped_refptr<webrtc::MediaStreamInterface> media_stream) { |
+ DCHECK_EQ(media_stream->label(), label()); |
+ |
+ media_stream_ = std::move(media_stream); |
+ |
webrtc::VideoTrackVector video_tracks = media_stream_->GetVideoTracks(); |
if (video_tracks.empty()) { |
LOG(ERROR) << "Received media stream with no video tracks."; |
@@ -71,12 +89,11 @@ WebrtcVideoRendererAdapter::WebrtcVideoRendererAdapter( |
video_tracks[0]->AddOrUpdateSink(this, rtc::VideoSinkWants()); |
} |
-WebrtcVideoRendererAdapter::~WebrtcVideoRendererAdapter() { |
- DCHECK(task_runner_->BelongsToCurrentThread()); |
- |
- webrtc::VideoTrackVector video_tracks = media_stream_->GetVideoTracks(); |
- DCHECK(!video_tracks.empty()); |
- video_tracks[0]->RemoveSink(this); |
+void WebrtcVideoRendererAdapter::SetVideoStatsChannel( |
+ std::unique_ptr<MessagePipe> message_pipe) { |
+ // Expect that the host also creates video_stats data channel. |
+ video_stats_dispatcher_.reset(new ClientVideoStatsDispatcher(label_, this)); |
+ video_stats_dispatcher_->Init(std::move(message_pipe), this); |
} |
void WebrtcVideoRendererAdapter::OnFrame(const cricket::VideoFrame& frame) { |
@@ -86,24 +103,72 @@ void WebrtcVideoRendererAdapter::OnFrame(const cricket::VideoFrame& frame) { |
LOG(WARNING) << "Received frame with playout delay greater than 0."; |
} |
- std::unique_ptr<ClientFrameStats> stats(new ClientFrameStats()); |
- // TODO(sergeyu): |time_received| is not reported correctly here because the |
- // frame is already decoded at this point. |
- stats->time_received = base::TimeTicks::Now(); |
- |
task_runner_->PostTask( |
FROM_HERE, |
base::Bind(&WebrtcVideoRendererAdapter::HandleFrameOnMainThread, |
- weak_factory_.GetWeakPtr(), base::Passed(&stats), |
+ weak_factory_.GetWeakPtr(), frame.transport_frame_id(), |
+ base::TimeTicks::Now(), |
scoped_refptr<webrtc::VideoFrameBuffer>( |
frame.video_frame_buffer().get()))); |
} |
+void WebrtcVideoRendererAdapter::OnVideoFrameStats( |
+ uint32_t frame_id, |
+ const HostFrameStats& host_stats) { |
+ DCHECK(task_runner_->BelongsToCurrentThread()); |
+ |
+ // Drop all ClientFrameStats for frames before |frame_id|. Stats messages are |
+ // expected to be received in the same order as the corresponding video |
+ // frames, so we are not going to receive HostFrameStats for the frames before |
+ // |frame_id|. This may happen only if for some reason the host doesn't |
+ // generate stats message for all video frames. |
+ while (!client_stats_queue_.empty() && |
+ client_stats_queue_.front().first != frame_id) { |
+ client_stats_queue_.pop_front(); |
+ } |
+ |
+ // If there are no ClientFrameStats in the queue then queue HostFrameStats |
+ // to be processed in FrameRendered(). |
+ if (client_stats_queue_.empty()) { |
+ if (host_stats_queue_.size() > kMaxQueuedStats) { |
+ LOG(ERROR) << "video_stats channel is out of sync with the video stream. " |
+ "Performance stats will not be reported."; |
+ video_stats_dispatcher_.reset(); |
+ return; |
+ } |
+ host_stats_queue_.push_back(std::make_pair(frame_id, host_stats)); |
+ return; |
+ } |
+ |
+ // The correspond frame has been received and now we have both HostFrameStats |
+ // and ClientFrameStats. Report the stats to FrameStatsConsumer. |
+ DCHECK_EQ(client_stats_queue_.front().first, frame_id); |
+ FrameStats frame_stats; |
+ frame_stats.client_stats = client_stats_queue_.front().second; |
+ client_stats_queue_.pop_front(); |
+ frame_stats.host_stats = host_stats; |
+ video_renderer_->GetFrameStatsConsumer()->OnVideoFrameStats(frame_stats); |
+} |
+ |
+void WebrtcVideoRendererAdapter::OnChannelInitialized( |
+ ChannelDispatcherBase* channel_dispatcher) {} |
+ |
+void WebrtcVideoRendererAdapter::OnChannelClosed( |
+ ChannelDispatcherBase* channel_dispatcher) { |
+ LOG(WARNING) << "video_stats channel was closed by the host."; |
+} |
+ |
void WebrtcVideoRendererAdapter::HandleFrameOnMainThread( |
- std::unique_ptr<ClientFrameStats> stats, |
+ uint32_t frame_id, |
+ base::TimeTicks time_received, |
scoped_refptr<webrtc::VideoFrameBuffer> frame) { |
DCHECK(task_runner_->BelongsToCurrentThread()); |
+ std::unique_ptr<ClientFrameStats> stats(new ClientFrameStats()); |
+ // TODO(sergeyu): |time_received| is not reported correctly here because the |
+ // frame is already decoded at this point. |
+ stats->time_received = time_received; |
+ |
std::unique_ptr<webrtc::DesktopFrame> rgb_frame = |
video_renderer_->GetFrameConsumer()->AllocateFrame( |
webrtc::DesktopSize(frame->width(), frame->height())); |
@@ -114,10 +179,11 @@ void WebrtcVideoRendererAdapter::HandleFrameOnMainThread( |
base::Passed(&rgb_frame), |
video_renderer_->GetFrameConsumer()->GetPixelFormat()), |
base::Bind(&WebrtcVideoRendererAdapter::DrawFrame, |
- weak_factory_.GetWeakPtr(), base::Passed(&stats))); |
+ weak_factory_.GetWeakPtr(), frame_id, base::Passed(&stats))); |
} |
void WebrtcVideoRendererAdapter::DrawFrame( |
+ uint32_t frame_id, |
std::unique_ptr<ClientFrameStats> stats, |
std::unique_ptr<webrtc::DesktopFrame> frame) { |
DCHECK(task_runner_->BelongsToCurrentThread()); |
@@ -125,12 +191,52 @@ void WebrtcVideoRendererAdapter::DrawFrame( |
video_renderer_->GetFrameConsumer()->DrawFrame( |
std::move(frame), |
base::Bind(&WebrtcVideoRendererAdapter::FrameRendered, |
- weak_factory_.GetWeakPtr(), base::Passed(&stats))); |
+ weak_factory_.GetWeakPtr(), frame_id, base::Passed(&stats))); |
} |
void WebrtcVideoRendererAdapter::FrameRendered( |
- std::unique_ptr<ClientFrameStats> stats) { |
- // TODO(sergeyu): Report stats here |
+ uint32_t frame_id, |
+ std::unique_ptr<ClientFrameStats> client_stats) { |
+ DCHECK(task_runner_->BelongsToCurrentThread()); |
+ |
+ if (!video_stats_dispatcher_ || !video_stats_dispatcher_->is_connected()) |
+ return; |
+ |
+ client_stats->time_rendered = base::TimeTicks::Now(); |
+ |
+ // Drop all HostFrameStats for frames before |frame_id|. Stats messages are |
+ // expected to be received in the same order as the corresponding video |
+ // frames. This may happen only if the host generates HostFrameStats without |
+ // the corresponding frame. |
+ while (!host_stats_queue_.empty() && |
+ host_stats_queue_.front().first != frame_id) { |
+ LOG(WARNING) << "Host sent VideoStats message for a frame that was never " |
+ "received."; |
+ host_stats_queue_.pop_front(); |
+ } |
+ |
+ // If HostFrameStats hasn't been received for |frame_id| then queue |
+ // ClientFrameStats to be processed in OnVideoFrameStats(). |
+ if (host_stats_queue_.empty()) { |
+ if (host_stats_queue_.size() > kMaxQueuedStats) { |
Irfan
2016/08/10 20:58:49
This should be client_stats_queue ?
Sergey Ulanov
2016/08/10 22:29:56
Done.
|
+ LOG(ERROR) << "video_stats channel is out of sync with the video " |
+ "stream. Performance stats will not be reported."; |
+ video_stats_dispatcher_.reset(); |
+ return; |
+ } |
+ client_stats_queue_.push_back(std::make_pair(frame_id, *client_stats)); |
+ return; |
+ } |
+ |
+ // The correspond HostFrameStats has been received already and now we have |
+ // both HostFrameStats and ClientFrameStats. Report the stats to |
+ // FrameStatsConsumer. |
+ DCHECK_EQ(host_stats_queue_.front().first, frame_id); |
+ FrameStats frame_stats; |
+ frame_stats.host_stats = host_stats_queue_.front().second; |
+ frame_stats.client_stats = *client_stats; |
+ host_stats_queue_.pop_front(); |
+ video_renderer_->GetFrameStatsConsumer()->OnVideoFrameStats(frame_stats); |
} |
} // namespace protocol |