| 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
 | 
| 
 |