| Index: webkit/glue/media/video_renderer_impl.cc
|
| diff --git a/webkit/glue/media/video_renderer_impl.cc b/webkit/glue/media/video_renderer_impl.cc
|
| deleted file mode 100644
|
| index b3bf06323da5d3dbbf279e5c9bea6ce5917b09d6..0000000000000000000000000000000000000000
|
| --- a/webkit/glue/media/video_renderer_impl.cc
|
| +++ /dev/null
|
| @@ -1,278 +0,0 @@
|
| -// Copyright (c) 2011 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.
|
| -
|
| -#include "webkit/glue/media/video_renderer_impl.h"
|
| -
|
| -#include "base/logging.h"
|
| -#include "media/base/video_frame.h"
|
| -#include "media/base/yuv_convert.h"
|
| -#include "third_party/skia/include/core/SkCanvas.h"
|
| -#include "third_party/skia/include/core/SkDevice.h"
|
| -#include "webkit/glue/webmediaplayer_proxy.h"
|
| -
|
| -namespace webkit_glue {
|
| -
|
| -VideoRendererImpl::VideoRendererImpl(bool pts_logging)
|
| - : last_converted_frame_(NULL),
|
| - pts_logging_(pts_logging) {
|
| -}
|
| -
|
| -VideoRendererImpl::~VideoRendererImpl() {}
|
| -
|
| -bool VideoRendererImpl::OnInitialize(media::VideoDecoder* decoder) {
|
| - natural_size_ = decoder->natural_size();
|
| - bitmap_.setConfig(SkBitmap::kARGB_8888_Config,
|
| - natural_size_.width(), natural_size_.height());
|
| - bitmap_.allocPixels();
|
| - bitmap_.eraseRGB(0x00, 0x00, 0x00);
|
| - bitmap_.setIsVolatile(true);
|
| - return true;
|
| -}
|
| -
|
| -void VideoRendererImpl::OnStop(const base::Closure& callback) {
|
| - if (!callback.is_null())
|
| - callback.Run();
|
| -}
|
| -
|
| -void VideoRendererImpl::OnFrameAvailable() {
|
| - proxy_->Repaint();
|
| -}
|
| -
|
| -void VideoRendererImpl::SetWebMediaPlayerProxy(WebMediaPlayerProxy* proxy) {
|
| - proxy_ = proxy;
|
| -}
|
| -
|
| -void VideoRendererImpl::SetRect(const gfx::Rect& rect) {}
|
| -
|
| -// This method is always called on the renderer's thread.
|
| -void VideoRendererImpl::Paint(SkCanvas* canvas, const gfx::Rect& dest_rect) {
|
| - scoped_refptr<media::VideoFrame> video_frame;
|
| - GetCurrentFrame(&video_frame);
|
| - if (!video_frame) {
|
| - SkPaint paint;
|
| - paint.setColor(SK_ColorBLACK);
|
| - canvas->drawRectCoords(
|
| - static_cast<float>(dest_rect.x()),
|
| - static_cast<float>(dest_rect.y()),
|
| - static_cast<float>(dest_rect.right()),
|
| - static_cast<float>(dest_rect.bottom()),
|
| - paint);
|
| - } else {
|
| - if (CanFastPaint(canvas, dest_rect)) {
|
| - FastPaint(video_frame, canvas, dest_rect);
|
| - } else {
|
| - SlowPaint(video_frame, canvas, dest_rect);
|
| - }
|
| -
|
| - // Presentation timestamp logging is primarily used to measure performance
|
| - // on low-end devices. When profiled on an Intel Atom N280 @ 1.66GHz this
|
| - // code had a ~63 microsecond perf hit when logging to a file (not stdout),
|
| - // which is neglible enough for measuring playback performance.
|
| - if (pts_logging_)
|
| - VLOG(1) << "pts=" << video_frame->GetTimestamp().InMicroseconds();
|
| - }
|
| -
|
| - PutCurrentFrame(video_frame);
|
| -}
|
| -
|
| -// CanFastPaint is a helper method to determine the conditions for fast
|
| -// painting. The conditions are:
|
| -// 1. No skew in canvas matrix.
|
| -// 2. No flipping nor mirroring.
|
| -// 3. Canvas has pixel format ARGB8888.
|
| -// 4. Canvas is opaque.
|
| -// TODO(hclam): The fast paint method should support flipping and mirroring.
|
| -// Disable the flipping and mirroring checks once we have it.
|
| -bool VideoRendererImpl::CanFastPaint(SkCanvas* canvas,
|
| - const gfx::Rect& dest_rect) {
|
| - // Fast paint does not handle opacity value other than 1.0. Hence use slow
|
| - // paint if opacity is not 1.0. Since alpha = opacity * 0xFF, we check that
|
| - // alpha != 0xFF.
|
| - //
|
| - // Additonal notes: If opacity = 0.0, the chrome display engine does not try
|
| - // to render the video. So, this method is never called. However, if the
|
| - // opacity = 0.0001, alpha is again 0, but the display engine tries to render
|
| - // the video. If we use Fast paint, the video shows up with opacity = 1.0.
|
| - // Hence we use slow paint also in the case where alpha = 0. It would be ideal
|
| - // if rendering was never called even for cases where alpha is 0. Created
|
| - // bug 48090 for this.
|
| - SkCanvas::LayerIter layer_iter(canvas, false);
|
| - SkColor sk_color = layer_iter.paint().getColor();
|
| - SkAlpha sk_alpha = SkColorGetA(sk_color);
|
| - if (sk_alpha != 0xFF) {
|
| - return false;
|
| - }
|
| -
|
| - const SkMatrix& total_matrix = canvas->getTotalMatrix();
|
| - // Perform the following checks here:
|
| - // 1. Check for skewing factors of the transformation matrix. They should be
|
| - // zero.
|
| - // 2. Check for mirroring and flipping. Make sure they are greater than zero.
|
| - if (SkScalarNearlyZero(total_matrix.getSkewX()) &&
|
| - SkScalarNearlyZero(total_matrix.getSkewY()) &&
|
| - total_matrix.getScaleX() > 0 &&
|
| - total_matrix.getScaleY() > 0) {
|
| - SkDevice* device = canvas->getDevice();
|
| - const SkBitmap::Config config = device->config();
|
| -
|
| - if (config == SkBitmap::kARGB_8888_Config && device->isOpaque()) {
|
| - return true;
|
| - }
|
| - }
|
| -
|
| - return false;
|
| -}
|
| -
|
| -void VideoRendererImpl::SlowPaint(media::VideoFrame* video_frame,
|
| - SkCanvas* canvas,
|
| - const gfx::Rect& dest_rect) {
|
| - // 1. Convert YUV frame to RGB.
|
| - base::TimeDelta timestamp = video_frame->GetTimestamp();
|
| - if (video_frame != last_converted_frame_ ||
|
| - timestamp != last_converted_timestamp_) {
|
| - last_converted_frame_ = video_frame;
|
| - last_converted_timestamp_ = timestamp;
|
| - DCHECK(video_frame->format() == media::VideoFrame::YV12 ||
|
| - video_frame->format() == media::VideoFrame::YV16);
|
| - DCHECK(video_frame->stride(media::VideoFrame::kUPlane) ==
|
| - video_frame->stride(media::VideoFrame::kVPlane));
|
| - DCHECK(video_frame->planes() == media::VideoFrame::kNumYUVPlanes);
|
| - bitmap_.lockPixels();
|
| - media::YUVType yuv_type =
|
| - (video_frame->format() == media::VideoFrame::YV12) ?
|
| - media::YV12 : media::YV16;
|
| - media::ConvertYUVToRGB32(video_frame->data(media::VideoFrame::kYPlane),
|
| - video_frame->data(media::VideoFrame::kUPlane),
|
| - video_frame->data(media::VideoFrame::kVPlane),
|
| - static_cast<uint8*>(bitmap_.getPixels()),
|
| - video_frame->width(),
|
| - video_frame->height(),
|
| - video_frame->stride(media::VideoFrame::kYPlane),
|
| - video_frame->stride(media::VideoFrame::kUPlane),
|
| - bitmap_.rowBytes(),
|
| - yuv_type);
|
| - bitmap_.notifyPixelsChanged();
|
| - bitmap_.unlockPixels();
|
| - }
|
| -
|
| - // 2. Paint the bitmap to canvas.
|
| - SkMatrix matrix;
|
| - matrix.setTranslate(static_cast<SkScalar>(dest_rect.x()),
|
| - static_cast<SkScalar>(dest_rect.y()));
|
| - if (dest_rect.width() != natural_size_.width() ||
|
| - dest_rect.height() != natural_size_.height()) {
|
| - matrix.preScale(SkIntToScalar(dest_rect.width()) /
|
| - SkIntToScalar(natural_size_.width()),
|
| - SkIntToScalar(dest_rect.height()) /
|
| - SkIntToScalar(natural_size_.height()));
|
| - }
|
| - SkPaint paint;
|
| - paint.setFlags(SkPaint::kFilterBitmap_Flag);
|
| - canvas->drawBitmapMatrix(bitmap_, matrix, &paint);
|
| -}
|
| -
|
| -void VideoRendererImpl::FastPaint(media::VideoFrame* video_frame,
|
| - SkCanvas* canvas,
|
| - const gfx::Rect& dest_rect) {
|
| - DCHECK(video_frame->format() == media::VideoFrame::YV12 ||
|
| - video_frame->format() == media::VideoFrame::YV16);
|
| - DCHECK(video_frame->stride(media::VideoFrame::kUPlane) ==
|
| - video_frame->stride(media::VideoFrame::kVPlane));
|
| - DCHECK(video_frame->planes() == media::VideoFrame::kNumYUVPlanes);
|
| - const SkBitmap& bitmap = canvas->getDevice()->accessBitmap(true);
|
| - media::YUVType yuv_type = (video_frame->format() == media::VideoFrame::YV12) ?
|
| - media::YV12 : media::YV16;
|
| - int y_shift = yuv_type; // 1 for YV12, 0 for YV16.
|
| -
|
| - // Create a rectangle backed by SkScalar.
|
| - SkRect scalar_dest_rect;
|
| - scalar_dest_rect.iset(dest_rect.x(), dest_rect.y(),
|
| - dest_rect.right(), dest_rect.bottom());
|
| -
|
| - // Transform the destination rectangle to local coordinates.
|
| - const SkMatrix& local_matrix = canvas->getTotalMatrix();
|
| - SkRect local_dest_rect;
|
| - local_matrix.mapRect(&local_dest_rect, scalar_dest_rect);
|
| -
|
| - // After projecting the destination rectangle to local coordinates, round
|
| - // the projected rectangle to integer values, this will give us pixel values
|
| - // of the rectangle.
|
| - SkIRect local_dest_irect, local_dest_irect_saved;
|
| - local_dest_rect.round(&local_dest_irect);
|
| - local_dest_rect.round(&local_dest_irect_saved);
|
| -
|
| - // Only does the paint if the destination rect intersects with the clip
|
| - // rect.
|
| - if (local_dest_irect.intersect(canvas->getTotalClip().getBounds())) {
|
| - // At this point |local_dest_irect| contains the rect that we should draw
|
| - // to within the clipping rect.
|
| -
|
| - // Calculate the address for the top left corner of destination rect in
|
| - // the canvas that we will draw to. The address is obtained by the base
|
| - // address of the canvas shifted by "left" and "top" of the rect.
|
| - uint8* dest_rect_pointer = static_cast<uint8*>(bitmap.getPixels()) +
|
| - local_dest_irect.fTop * bitmap.rowBytes() +
|
| - local_dest_irect.fLeft * 4;
|
| -
|
| - // Project the clip rect to the original video frame, obtains the
|
| - // dimensions of the projected clip rect, "left" and "top" of the rect.
|
| - // The math here are all integer math so we won't have rounding error and
|
| - // write outside of the canvas.
|
| - // We have the assumptions of dest_rect.width() and dest_rect.height()
|
| - // being non-zero, these are valid assumptions since finding intersection
|
| - // above rejects empty rectangle so we just do a DCHECK here.
|
| - DCHECK_NE(0, dest_rect.width());
|
| - DCHECK_NE(0, dest_rect.height());
|
| - size_t frame_clip_width = local_dest_irect.width() *
|
| - video_frame->width() / local_dest_irect_saved.width();
|
| - size_t frame_clip_height = local_dest_irect.height() *
|
| - video_frame->height() / local_dest_irect_saved.height();
|
| -
|
| - // Project the "left" and "top" of the final destination rect to local
|
| - // coordinates of the video frame, use these values to find the offsets
|
| - // in the video frame to start reading.
|
| - size_t frame_clip_left =
|
| - (local_dest_irect.fLeft - local_dest_irect_saved.fLeft) *
|
| - video_frame->width() / local_dest_irect_saved.width();
|
| - size_t frame_clip_top =
|
| - (local_dest_irect.fTop - local_dest_irect_saved.fTop) *
|
| - video_frame->height() / local_dest_irect_saved.height();
|
| -
|
| - // Use the "left" and "top" of the destination rect to locate the offset
|
| - // in Y, U and V planes.
|
| - size_t y_offset = video_frame->stride(media::VideoFrame::kYPlane) *
|
| - frame_clip_top + frame_clip_left;
|
| - // For format YV12, there is one U, V value per 2x2 block.
|
| - // For format YV16, there is one u, V value per 2x1 block.
|
| - size_t uv_offset = (video_frame->stride(media::VideoFrame::kUPlane) *
|
| - (frame_clip_top >> y_shift)) + (frame_clip_left >> 1);
|
| - uint8* frame_clip_y =
|
| - video_frame->data(media::VideoFrame::kYPlane) + y_offset;
|
| - uint8* frame_clip_u =
|
| - video_frame->data(media::VideoFrame::kUPlane) + uv_offset;
|
| - uint8* frame_clip_v =
|
| - video_frame->data(media::VideoFrame::kVPlane) + uv_offset;
|
| - bitmap.lockPixels();
|
| -
|
| - // TODO(hclam): do rotation and mirroring here.
|
| - // TODO(fbarchard): switch filtering based on performance.
|
| - media::ScaleYUVToRGB32(frame_clip_y,
|
| - frame_clip_u,
|
| - frame_clip_v,
|
| - dest_rect_pointer,
|
| - frame_clip_width,
|
| - frame_clip_height,
|
| - local_dest_irect.width(),
|
| - local_dest_irect.height(),
|
| - video_frame->stride(media::VideoFrame::kYPlane),
|
| - video_frame->stride(media::VideoFrame::kUPlane),
|
| - bitmap.rowBytes(),
|
| - yuv_type,
|
| - media::ROTATE_0,
|
| - media::FILTER_BILINEAR);
|
| - bitmap.unlockPixels();
|
| - }
|
| -}
|
| -
|
| -} // namespace webkit_glue
|
|
|