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

Unified Diff: remoting/client/software_video_renderer.cc

Issue 1288063004: Simplify FrameConsumer interface. Remove FrameProducer interface. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Created 5 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
« no previous file with comments | « remoting/client/software_video_renderer.h ('k') | remoting/client/software_video_renderer_unittest.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: remoting/client/software_video_renderer.cc
diff --git a/remoting/client/software_video_renderer.cc b/remoting/client/software_video_renderer.cc
index b063b2da6583768b164cc37a2e1d5244ed1862ba..89a86a991bd63302b161e54ba6bd42b528a2ce99 100644
--- a/remoting/client/software_video_renderer.cc
+++ b/remoting/client/software_video_renderer.cc
@@ -4,29 +4,30 @@
#include "remoting/client/software_video_renderer.h"
-#include <list>
-
#include "base/bind.h"
#include "base/callback.h"
#include "base/callback_helpers.h"
#include "base/location.h"
#include "base/logging.h"
#include "base/single_thread_task_runner.h"
+#include "base/task_runner_util.h"
#include "remoting/base/util.h"
#include "remoting/client/frame_consumer.h"
#include "remoting/codec/video_decoder.h"
#include "remoting/codec/video_decoder_verbatim.h"
#include "remoting/codec/video_decoder_vpx.h"
+#include "remoting/proto/video.pb.h"
#include "remoting/protocol/session_config.h"
#include "third_party/libyuv/include/libyuv/convert_argb.h"
#include "third_party/webrtc/modules/desktop_capture/desktop_frame.h"
-using base::Passed;
using remoting::protocol::ChannelConfig;
using remoting::protocol::SessionConfig;
namespace remoting {
+namespace {
+
// This class wraps a VideoDecoder and byte-swaps the pixels for compatibility
// with the android.graphics.Bitmap class.
// TODO(lambroslambrou): Refactor so that the VideoDecoder produces data
@@ -34,8 +35,7 @@ namespace remoting {
class RgbToBgrVideoDecoderFilter : public VideoDecoder {
public:
RgbToBgrVideoDecoderFilter(scoped_ptr<VideoDecoder> parent)
- : parent_(parent.Pass()) {
- }
+ : parent_(parent.Pass()) {}
bool DecodePacket(const VideoPacket& packet) override {
return parent_->DecodePacket(packet);
@@ -58,7 +58,7 @@ class RgbToBgrVideoDecoderFilter : public VideoDecoder {
i.Advance()) {
webrtc::DesktopRect rect = i.rect();
uint8* pixels = image_buffer + (rect.top() * image_stride) +
- (rect.left() * kBytesPerPixel);
+ (rect.left() * kBytesPerPixel);
libyuv::ABGRToARGB(pixels, image_stride, pixels, image_stride,
rect.width(), rect.height());
}
@@ -72,71 +72,41 @@ class RgbToBgrVideoDecoderFilter : public VideoDecoder {
scoped_ptr<VideoDecoder> parent_;
};
-class SoftwareVideoRenderer::Core {
- public:
- Core(scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
- scoped_refptr<base::SingleThreadTaskRunner> decode_task_runner,
- scoped_ptr<FrameConsumerProxy> consumer);
- ~Core();
-
- void OnSessionConfig(const protocol::SessionConfig& config);
- void DrawBuffer(webrtc::DesktopFrame* buffer);
- void InvalidateRegion(const webrtc::DesktopRegion& region);
- void RequestReturnBuffers(const base::Closure& done);
- void SetOutputSizeAndClip(
- const webrtc::DesktopSize& view_size,
- const webrtc::DesktopRect& clip_area);
-
- // Decodes the contents of |packet|. DecodePacket may keep a reference to
- // |packet| so the |packet| must remain alive and valid until |done| is
- // executed.
- void DecodePacket(scoped_ptr<VideoPacket> packet, const base::Closure& done);
-
- private:
- // Paints the invalidated region to the next available buffer and returns it
- // to the consumer.
- void SchedulePaint();
- void DoPaint();
-
- scoped_refptr<base::SingleThreadTaskRunner> main_task_runner_;
- scoped_refptr<base::SingleThreadTaskRunner> decode_task_runner_;
- scoped_ptr<FrameConsumerProxy> consumer_;
- scoped_ptr<VideoDecoder> decoder_;
+scoped_ptr<webrtc::DesktopFrame> DoDecodeFrame(
+ VideoDecoder* decoder,
+ scoped_ptr<VideoPacket> packet,
+ scoped_ptr<webrtc::DesktopFrame> frame) {
+ if (!decoder->DecodePacket(*packet))
+ return nullptr;
- // Remote screen size in pixels.
- webrtc::DesktopSize source_size_;
+ decoder->RenderFrame(
+ frame->size(), webrtc::DesktopRect::MakeSize(frame->size()),
+ frame->data(), frame->stride(), frame->mutable_updated_region());
- // Vertical and horizontal DPI of the remote screen.
- webrtc::DesktopVector source_dpi_;
+ const webrtc::DesktopRegion* shape = decoder->GetImageShape();
+ if (shape)
+ frame->set_shape(new webrtc::DesktopRegion(*shape));
- // The current dimensions of the frame consumer view.
- webrtc::DesktopSize view_size_;
- webrtc::DesktopRect clip_area_;
-
- // The drawing buffers supplied by the frame consumer.
- std::list<webrtc::DesktopFrame*> buffers_;
-
- // Flag used to coalesce runs of SchedulePaint()s into a single DoPaint().
- bool paint_scheduled_;
+ return frame.Pass();
+}
- base::WeakPtrFactory<Core> weak_factory_;
-};
+} // namespace
-SoftwareVideoRenderer::Core::Core(
- scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
+SoftwareVideoRenderer::SoftwareVideoRenderer(
scoped_refptr<base::SingleThreadTaskRunner> decode_task_runner,
- scoped_ptr<FrameConsumerProxy> consumer)
- : main_task_runner_(main_task_runner),
- decode_task_runner_(decode_task_runner),
- consumer_(consumer.Pass()),
- paint_scheduled_(false),
+ FrameConsumer* consumer)
+ : decode_task_runner_(decode_task_runner),
+ consumer_(consumer),
weak_factory_(this) {}
-SoftwareVideoRenderer::Core::~Core() {
+SoftwareVideoRenderer::~SoftwareVideoRenderer() {
+ if (decoder_)
+ decode_task_runner_->DeleteSoon(FROM_HERE, decoder_.release());
}
-void SoftwareVideoRenderer::Core::OnSessionConfig(const SessionConfig& config) {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
+void SoftwareVideoRenderer::OnSessionConfig(
+ const protocol::SessionConfig& config) {
+ DCHECK(thread_checker_.CalledOnValidThread());
// Initialize decoder based on the selected codec.
ChannelConfig::Codec codec = config.video_config().codec;
@@ -157,248 +127,92 @@ void SoftwareVideoRenderer::Core::OnSessionConfig(const SessionConfig& config) {
}
}
-void SoftwareVideoRenderer::Core::DecodePacket(scoped_ptr<VideoPacket> packet,
- const base::Closure& done) {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
-
- bool notify_size_or_dpi_change = false;
-
- // If the packet includes screen size or DPI information, store them.
- if (packet->format().has_screen_width() &&
- packet->format().has_screen_height()) {
- webrtc::DesktopSize source_size(packet->format().screen_width(),
- packet->format().screen_height());
- if (!source_size_.equals(source_size)) {
- source_size_ = source_size;
- notify_size_or_dpi_change = true;
- }
- }
- if (packet->format().has_x_dpi() && packet->format().has_y_dpi()) {
- webrtc::DesktopVector source_dpi(packet->format().x_dpi(),
- packet->format().y_dpi());
- if (!source_dpi.equals(source_dpi_)) {
- source_dpi_ = source_dpi;
- notify_size_or_dpi_change = true;
- }
- }
-
- // If we've never seen a screen size, ignore the packet.
- if (source_size_.is_empty()) {
- main_task_runner_->PostTask(FROM_HERE, base::Bind(done));
- return;
- }
-
- if (notify_size_or_dpi_change)
- consumer_->SetSourceSize(source_size_, source_dpi_);
-
- if (decoder_->DecodePacket(*packet.get())) {
- SchedulePaint();
- } else {
- LOG(ERROR) << "DecodePacket() failed.";
- }
-
- main_task_runner_->PostTask(FROM_HERE, base::Bind(done));
-}
-
-void SoftwareVideoRenderer::Core::SchedulePaint() {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
- if (paint_scheduled_)
- return;
- paint_scheduled_ = true;
- decode_task_runner_->PostTask(
- FROM_HERE, base::Bind(&SoftwareVideoRenderer::Core::DoPaint,
- weak_factory_.GetWeakPtr()));
-}
-
-void SoftwareVideoRenderer::Core::DoPaint() {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
- DCHECK(paint_scheduled_);
- paint_scheduled_ = false;
-
- // If the view size is empty or we have no output buffers ready, return.
- if (buffers_.empty() || view_size_.is_empty())
- return;
-
- // If no Decoder is initialized, or the host dimensions are empty, return.
- if (!decoder_.get() || source_size_.is_empty())
- return;
-
- // Draw the invalidated region to the buffer.
- webrtc::DesktopFrame* buffer = buffers_.front();
- webrtc::DesktopRegion output_region;
- decoder_->RenderFrame(view_size_, clip_area_,
- buffer->data(), buffer->stride(), &output_region);
-
- // Notify the consumer that painting is done.
- if (!output_region.is_empty()) {
- buffers_.pop_front();
- consumer_->ApplyBuffer(view_size_, clip_area_, buffer, output_region,
- decoder_->GetImageShape());
- }
-}
-
-void SoftwareVideoRenderer::Core::RequestReturnBuffers(
- const base::Closure& done) {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
-
- while (!buffers_.empty()) {
- consumer_->ReturnBuffer(buffers_.front());
- buffers_.pop_front();
- }
-
- if (!done.is_null())
- done.Run();
-}
-
-void SoftwareVideoRenderer::Core::DrawBuffer(webrtc::DesktopFrame* buffer) {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
- DCHECK(clip_area_.width() <= buffer->size().width() &&
- clip_area_.height() <= buffer->size().height());
-
- buffers_.push_back(buffer);
- SchedulePaint();
-}
-
-void SoftwareVideoRenderer::Core::InvalidateRegion(
- const webrtc::DesktopRegion& region) {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
-
- if (decoder_.get()) {
- decoder_->Invalidate(view_size_, region);
- SchedulePaint();
- }
-}
-
-void SoftwareVideoRenderer::Core::SetOutputSizeAndClip(
- const webrtc::DesktopSize& view_size,
- const webrtc::DesktopRect& clip_area) {
- DCHECK(decode_task_runner_->BelongsToCurrentThread());
-
- // The whole frame needs to be repainted if the scaling factor has changed.
- if (!view_size_.equals(view_size) && decoder_.get()) {
- webrtc::DesktopRegion region;
- region.AddRect(webrtc::DesktopRect::MakeSize(view_size));
- decoder_->Invalidate(view_size, region);
- }
-
- if (!view_size_.equals(view_size) ||
- !clip_area_.equals(clip_area)) {
- view_size_ = view_size;
- clip_area_ = clip_area;
-
- // Return buffers that are smaller than needed to the consumer for
- // reuse/reallocation.
- std::list<webrtc::DesktopFrame*>::iterator i = buffers_.begin();
- while (i != buffers_.end()) {
- if ((*i)->size().width() < clip_area_.width() ||
- (*i)->size().height() < clip_area_.height()) {
- consumer_->ReturnBuffer(*i);
- i = buffers_.erase(i);
- } else {
- ++i;
- }
- }
-
- SchedulePaint();
- }
-}
-
-SoftwareVideoRenderer::SoftwareVideoRenderer(
- scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
- scoped_refptr<base::SingleThreadTaskRunner> decode_task_runner,
- scoped_ptr<FrameConsumerProxy> consumer)
- : decode_task_runner_(decode_task_runner),
- core_(new Core(main_task_runner, decode_task_runner, consumer.Pass())),
- weak_factory_(this) {
- DCHECK(CalledOnValidThread());
-}
-
-SoftwareVideoRenderer::~SoftwareVideoRenderer() {
- DCHECK(CalledOnValidThread());
- bool result = decode_task_runner_->DeleteSoon(FROM_HERE, core_.release());
- DCHECK(result);
-}
-
-void SoftwareVideoRenderer::OnSessionConfig(
- const protocol::SessionConfig& config) {
- DCHECK(CalledOnValidThread());
- decode_task_runner_->PostTask(
- FROM_HERE, base::Bind(&SoftwareVideoRenderer::Core::OnSessionConfig,
- base::Unretained(core_.get()), config));
-}
-
ChromotingStats* SoftwareVideoRenderer::GetStats() {
- DCHECK(CalledOnValidThread());
+ DCHECK(thread_checker_.CalledOnValidThread());
return &stats_;
}
protocol::VideoStub* SoftwareVideoRenderer::GetVideoStub() {
+ DCHECK(thread_checker_.CalledOnValidThread());
return this;
}
void SoftwareVideoRenderer::ProcessVideoPacket(scoped_ptr<VideoPacket> packet,
const base::Closure& done) {
- DCHECK(CalledOnValidThread());
+ DCHECK(thread_checker_.CalledOnValidThread());
+
+ base::ScopedClosureRunner done_runner(done);
stats_.RecordVideoPacketStats(*packet);
// If the video packet is empty then drop it. Empty packets are used to
// maintain activity on the network.
if (!packet->has_data() || packet->data().size() == 0) {
- done.Run();
return;
}
- // Measure the latency between the last packet being received and presented.
- base::Time decode_start = base::Time::Now();
+ if (packet->format().has_screen_width() &&
+ packet->format().has_screen_height()) {
+ source_size_.set(packet->format().screen_width(),
+ packet->format().screen_height());
+ }
- base::Closure decode_done = base::Bind(&SoftwareVideoRenderer::OnPacketDone,
- weak_factory_.GetWeakPtr(),
- decode_start, done);
+ if (packet->format().has_x_dpi() && packet->format().has_y_dpi()) {
+ webrtc::DesktopVector source_dpi(packet->format().x_dpi(),
+ packet->format().y_dpi());
+ if (!source_dpi.equals(source_dpi_)) {
+ source_dpi_ = source_dpi;
+ }
+ }
- decode_task_runner_->PostTask(FROM_HERE, base::Bind(
- &SoftwareVideoRenderer::Core::DecodePacket,
- base::Unretained(core_.get()), base::Passed(&packet), decode_done));
-}
+ if (source_size_.is_empty()) {
+ LOG(ERROR) << "Received VideoPacket with unknown size.";
+ return;
+ }
-void SoftwareVideoRenderer::DrawBuffer(webrtc::DesktopFrame* buffer) {
- decode_task_runner_->PostTask(
- FROM_HERE, base::Bind(&SoftwareVideoRenderer::Core::DrawBuffer,
- base::Unretained(core_.get()), buffer));
-}
+ scoped_ptr<webrtc::DesktopFrame> frame =
+ consumer_->AllocateFrame(source_size_);
+ frame->set_dpi(source_dpi_);
-void SoftwareVideoRenderer::InvalidateRegion(
- const webrtc::DesktopRegion& region) {
- decode_task_runner_->PostTask(
- FROM_HERE, base::Bind(&SoftwareVideoRenderer::Core::InvalidateRegion,
- base::Unretained(core_.get()), region));
+ base::PostTaskAndReplyWithResult(
+ decode_task_runner_.get(), FROM_HERE,
+ base::Bind(&DoDecodeFrame, decoder_.get(), base::Passed(&packet),
+ base::Passed(&frame)),
+ base::Bind(&SoftwareVideoRenderer::RenderFrame,
+ weak_factory_.GetWeakPtr(), base::TimeTicks::Now(),
+ done_runner.Release()));
}
-void SoftwareVideoRenderer::RequestReturnBuffers(const base::Closure& done) {
- decode_task_runner_->PostTask(
- FROM_HERE,
- base::Bind(&SoftwareVideoRenderer::Core::RequestReturnBuffers,
- base::Unretained(core_.get()), done));
-}
+void SoftwareVideoRenderer::RenderFrame(
+ base::TimeTicks decode_start_time,
+ const base::Closure& done,
+ scoped_ptr<webrtc::DesktopFrame> frame) {
+ DCHECK(thread_checker_.CalledOnValidThread());
-void SoftwareVideoRenderer::SetOutputSizeAndClip(
- const webrtc::DesktopSize& view_size,
- const webrtc::DesktopRect& clip_area) {
- decode_task_runner_->PostTask(
- FROM_HERE,
- base::Bind(&SoftwareVideoRenderer::Core::SetOutputSizeAndClip,
- base::Unretained(core_.get()), view_size, clip_area));
+ stats_.RecordDecodeTime(
+ (base::TimeTicks::Now() - decode_start_time).InMilliseconds());
+
+ if (!frame) {
+ if (!done.is_null())
+ done.Run();
+ return;
+ }
+
+ consumer_->DrawFrame(
+ frame.Pass(),
+ base::Bind(&SoftwareVideoRenderer::OnFrameRendered,
+ weak_factory_.GetWeakPtr(), base::TimeTicks::Now(), done));
}
-void SoftwareVideoRenderer::OnPacketDone(base::Time decode_start,
- const base::Closure& done) {
- DCHECK(CalledOnValidThread());
+void SoftwareVideoRenderer::OnFrameRendered(base::TimeTicks paint_start_time,
+ const base::Closure& done) {
+ DCHECK(thread_checker_.CalledOnValidThread());
- // Record the latency between the packet being received and presented.
- base::TimeDelta decode_time = base::Time::Now() - decode_start;
- stats_.RecordDecodeTime(decode_time.InMilliseconds());
+ stats_.RecordPaintTime(
+ (base::TimeTicks::Now() - paint_start_time).InMilliseconds());
- done.Run();
+ if (!done.is_null())
+ done.Run();
}
} // namespace remoting
« no previous file with comments | « remoting/client/software_video_renderer.h ('k') | remoting/client/software_video_renderer_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698