OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "media/filters/skcanvas_video_renderer.h" | 5 #include "media/filters/skcanvas_video_renderer.h" |
6 | 6 |
7 #include "base/logging.h" | 7 #include "base/logging.h" |
8 #include "media/base/video_frame.h" | 8 #include "media/base/video_frame.h" |
9 #include "media/base/yuv_convert.h" | 9 #include "media/base/yuv_convert.h" |
10 #include "third_party/libyuv/include/libyuv.h" | 10 #include "third_party/libyuv/include/libyuv.h" |
11 #include "third_party/skia/include/core/SkCanvas.h" | 11 #include "third_party/skia/include/core/SkCanvas.h" |
12 #include "third_party/skia/include/core/SkImageGenerator.h" | |
12 #include "ui/gfx/skbitmap_operations.h" | 13 #include "ui/gfx/skbitmap_operations.h" |
13 | 14 |
14 // Skia internal format depends on a platform. On Android it is ABGR, on others | 15 // Skia internal format depends on a platform. On Android it is ABGR, on others |
15 // it is ARGB. | 16 // it is ARGB. |
16 #if SK_B32_SHIFT == 0 && SK_G32_SHIFT == 8 && SK_R32_SHIFT == 16 && \ | 17 #if SK_B32_SHIFT == 0 && SK_G32_SHIFT == 8 && SK_R32_SHIFT == 16 && \ |
17 SK_A32_SHIFT == 24 | 18 SK_A32_SHIFT == 24 |
18 #define LIBYUV_I420_TO_ARGB libyuv::I420ToARGB | 19 #define LIBYUV_I420_TO_ARGB libyuv::I420ToARGB |
19 #define LIBYUV_I422_TO_ARGB libyuv::I422ToARGB | 20 #define LIBYUV_I422_TO_ARGB libyuv::I422ToARGB |
20 #elif SK_R32_SHIFT == 0 && SK_G32_SHIFT == 8 && SK_B32_SHIFT == 16 && \ | 21 #elif SK_R32_SHIFT == 0 && SK_G32_SHIFT == 8 && SK_B32_SHIFT == 16 && \ |
21 SK_A32_SHIFT == 24 | 22 SK_A32_SHIFT == 24 |
(...skipping 11 matching lines...) Expand all Loading... | |
33 format == media::VideoFrame::I420 || | 34 format == media::VideoFrame::I420 || |
34 format == media::VideoFrame::YV12A || | 35 format == media::VideoFrame::YV12A || |
35 format == media::VideoFrame::YV12J || | 36 format == media::VideoFrame::YV12J || |
36 format == media::VideoFrame::YV24; | 37 format == media::VideoFrame::YV24; |
37 } | 38 } |
38 | 39 |
39 static bool IsYUVOrNative(media::VideoFrame::Format format) { | 40 static bool IsYUVOrNative(media::VideoFrame::Format format) { |
40 return IsYUV(format) || format == media::VideoFrame::NATIVE_TEXTURE; | 41 return IsYUV(format) || format == media::VideoFrame::NATIVE_TEXTURE; |
41 } | 42 } |
42 | 43 |
43 // Converts a VideoFrame containing YUV data to a SkBitmap containing RGB data. | 44 // Converts a |video_frame| to raw |rgb_pixels|. |
44 // | 45 static void ConvertVideoFrameToRGBPixels( |
45 // |bitmap| will be (re)allocated to match the dimensions of |video_frame|. | |
46 static void ConvertVideoFrameToBitmap( | |
47 const scoped_refptr<media::VideoFrame>& video_frame, | 46 const scoped_refptr<media::VideoFrame>& video_frame, |
48 SkBitmap* bitmap) { | 47 void* rgb_pixels, |
48 size_t row_bytes) { | |
49 DCHECK(IsYUVOrNative(video_frame->format())) | 49 DCHECK(IsYUVOrNative(video_frame->format())) |
50 << video_frame->format(); | 50 << video_frame->format(); |
51 if (IsYUV(video_frame->format())) { | 51 if (IsYUV(video_frame->format())) { |
52 DCHECK_EQ(video_frame->stride(media::VideoFrame::kUPlane), | 52 DCHECK_EQ(video_frame->stride(media::VideoFrame::kUPlane), |
53 video_frame->stride(media::VideoFrame::kVPlane)); | 53 video_frame->stride(media::VideoFrame::kVPlane)); |
54 } | 54 } |
55 | 55 |
56 // Check if |bitmap| needs to be (re)allocated. | |
57 if (bitmap->isNull() || | |
58 bitmap->width() != video_frame->visible_rect().width() || | |
59 bitmap->height() != video_frame->visible_rect().height()) { | |
60 bitmap->allocN32Pixels(video_frame->visible_rect().width(), | |
61 video_frame->visible_rect().height()); | |
62 bitmap->setIsVolatile(true); | |
63 } | |
64 | |
65 bitmap->lockPixels(); | |
66 | |
67 size_t y_offset = 0; | 56 size_t y_offset = 0; |
68 size_t uv_offset = 0; | 57 size_t uv_offset = 0; |
69 if (IsYUV(video_frame->format())) { | 58 if (IsYUV(video_frame->format())) { |
70 int y_shift = (video_frame->format() == media::VideoFrame::YV16) ? 0 : 1; | 59 int y_shift = (video_frame->format() == media::VideoFrame::YV16) ? 0 : 1; |
71 // Use the "left" and "top" of the destination rect to locate the offset | 60 // Use the "left" and "top" of the destination rect to locate the offset |
72 // in Y, U and V planes. | 61 // in Y, U and V planes. |
73 y_offset = (video_frame->stride(media::VideoFrame::kYPlane) * | 62 y_offset = (video_frame->stride(media::VideoFrame::kYPlane) * |
74 video_frame->visible_rect().y()) + | 63 video_frame->visible_rect().y()) + |
75 video_frame->visible_rect().x(); | 64 video_frame->visible_rect().x(); |
76 // For format YV12, there is one U, V value per 2x2 block. | 65 // For format YV12, there is one U, V value per 2x2 block. |
77 // For format YV16, there is one U, V value per 2x1 block. | 66 // For format YV16, there is one U, V value per 2x1 block. |
78 uv_offset = (video_frame->stride(media::VideoFrame::kUPlane) * | 67 uv_offset = (video_frame->stride(media::VideoFrame::kUPlane) * |
79 (video_frame->visible_rect().y() >> y_shift)) + | 68 (video_frame->visible_rect().y() >> y_shift)) + |
80 (video_frame->visible_rect().x() >> 1); | 69 (video_frame->visible_rect().x() >> 1); |
81 } | 70 } |
82 | 71 |
83 switch (video_frame->format()) { | 72 switch (video_frame->format()) { |
84 case media::VideoFrame::YV12: | 73 case media::VideoFrame::YV12: |
85 case media::VideoFrame::I420: | 74 case media::VideoFrame::I420: |
86 LIBYUV_I420_TO_ARGB( | 75 LIBYUV_I420_TO_ARGB( |
87 video_frame->data(media::VideoFrame::kYPlane) + y_offset, | 76 video_frame->data(media::VideoFrame::kYPlane) + y_offset, |
88 video_frame->stride(media::VideoFrame::kYPlane), | 77 video_frame->stride(media::VideoFrame::kYPlane), |
89 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, | 78 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, |
90 video_frame->stride(media::VideoFrame::kUPlane), | 79 video_frame->stride(media::VideoFrame::kUPlane), |
91 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, | 80 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, |
92 video_frame->stride(media::VideoFrame::kVPlane), | 81 video_frame->stride(media::VideoFrame::kVPlane), |
93 static_cast<uint8*>(bitmap->getPixels()), | 82 static_cast<uint8*>(rgb_pixels), |
94 bitmap->rowBytes(), | 83 row_bytes, |
95 video_frame->visible_rect().width(), | 84 video_frame->visible_rect().width(), |
96 video_frame->visible_rect().height()); | 85 video_frame->visible_rect().height()); |
97 break; | 86 break; |
98 | 87 |
99 case media::VideoFrame::YV12J: | 88 case media::VideoFrame::YV12J: |
100 media::ConvertYUVToRGB32( | 89 media::ConvertYUVToRGB32( |
101 video_frame->data(media::VideoFrame::kYPlane) + y_offset, | 90 video_frame->data(media::VideoFrame::kYPlane) + y_offset, |
102 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, | 91 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, |
103 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, | 92 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, |
104 static_cast<uint8*>(bitmap->getPixels()), | 93 static_cast<uint8*>(rgb_pixels), |
105 video_frame->visible_rect().width(), | 94 video_frame->visible_rect().width(), |
106 video_frame->visible_rect().height(), | 95 video_frame->visible_rect().height(), |
107 video_frame->stride(media::VideoFrame::kYPlane), | 96 video_frame->stride(media::VideoFrame::kYPlane), |
108 video_frame->stride(media::VideoFrame::kUPlane), | 97 video_frame->stride(media::VideoFrame::kUPlane), |
109 bitmap->rowBytes(), | 98 row_bytes, |
110 media::YV12J); | 99 media::YV12J); |
111 break; | 100 break; |
112 | 101 |
113 case media::VideoFrame::YV16: | 102 case media::VideoFrame::YV16: |
114 LIBYUV_I422_TO_ARGB( | 103 LIBYUV_I422_TO_ARGB( |
115 video_frame->data(media::VideoFrame::kYPlane) + y_offset, | 104 video_frame->data(media::VideoFrame::kYPlane) + y_offset, |
116 video_frame->stride(media::VideoFrame::kYPlane), | 105 video_frame->stride(media::VideoFrame::kYPlane), |
117 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, | 106 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, |
118 video_frame->stride(media::VideoFrame::kUPlane), | 107 video_frame->stride(media::VideoFrame::kUPlane), |
119 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, | 108 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, |
120 video_frame->stride(media::VideoFrame::kVPlane), | 109 video_frame->stride(media::VideoFrame::kVPlane), |
121 static_cast<uint8*>(bitmap->getPixels()), | 110 static_cast<uint8*>(rgb_pixels), |
122 bitmap->rowBytes(), | 111 row_bytes, |
123 video_frame->visible_rect().width(), | 112 video_frame->visible_rect().width(), |
124 video_frame->visible_rect().height()); | 113 video_frame->visible_rect().height()); |
125 break; | 114 break; |
126 | 115 |
127 case media::VideoFrame::YV12A: | 116 case media::VideoFrame::YV12A: |
128 // Since libyuv doesn't support YUVA, fallback to media, which is not ARM | 117 // Since libyuv doesn't support YUVA, fallback to media, which is not ARM |
129 // optimized. | 118 // optimized. |
130 // TODO(fbarchard, mtomasz): Use libyuv, then copy the alpha channel. | 119 // TODO(fbarchard, mtomasz): Use libyuv, then copy the alpha channel. |
131 media::ConvertYUVAToARGB( | 120 media::ConvertYUVAToARGB( |
132 video_frame->data(media::VideoFrame::kYPlane) + y_offset, | 121 video_frame->data(media::VideoFrame::kYPlane) + y_offset, |
133 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, | 122 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, |
134 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, | 123 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, |
135 video_frame->data(media::VideoFrame::kAPlane), | 124 video_frame->data(media::VideoFrame::kAPlane), |
136 static_cast<uint8*>(bitmap->getPixels()), | 125 static_cast<uint8*>(rgb_pixels), |
137 video_frame->visible_rect().width(), | 126 video_frame->visible_rect().width(), |
138 video_frame->visible_rect().height(), | 127 video_frame->visible_rect().height(), |
139 video_frame->stride(media::VideoFrame::kYPlane), | 128 video_frame->stride(media::VideoFrame::kYPlane), |
140 video_frame->stride(media::VideoFrame::kUPlane), | 129 video_frame->stride(media::VideoFrame::kUPlane), |
141 video_frame->stride(media::VideoFrame::kAPlane), | 130 video_frame->stride(media::VideoFrame::kAPlane), |
142 bitmap->rowBytes(), | 131 row_bytes, |
143 media::YV12); | 132 media::YV12); |
144 break; | 133 break; |
145 | 134 |
146 case media::VideoFrame::YV24: | 135 case media::VideoFrame::YV24: |
147 libyuv::I444ToARGB( | 136 libyuv::I444ToARGB( |
148 video_frame->data(media::VideoFrame::kYPlane) + y_offset, | 137 video_frame->data(media::VideoFrame::kYPlane) + y_offset, |
149 video_frame->stride(media::VideoFrame::kYPlane), | 138 video_frame->stride(media::VideoFrame::kYPlane), |
150 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, | 139 video_frame->data(media::VideoFrame::kUPlane) + uv_offset, |
151 video_frame->stride(media::VideoFrame::kUPlane), | 140 video_frame->stride(media::VideoFrame::kUPlane), |
152 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, | 141 video_frame->data(media::VideoFrame::kVPlane) + uv_offset, |
153 video_frame->stride(media::VideoFrame::kVPlane), | 142 video_frame->stride(media::VideoFrame::kVPlane), |
154 static_cast<uint8*>(bitmap->getPixels()), | 143 static_cast<uint8*>(rgb_pixels), |
155 bitmap->rowBytes(), | 144 row_bytes, |
156 video_frame->visible_rect().width(), | 145 video_frame->visible_rect().width(), |
157 video_frame->visible_rect().height()); | 146 video_frame->visible_rect().height()); |
158 #if SK_R32_SHIFT == 0 && SK_G32_SHIFT == 8 && SK_B32_SHIFT == 16 && \ | 147 #if SK_R32_SHIFT == 0 && SK_G32_SHIFT == 8 && SK_B32_SHIFT == 16 && \ |
159 SK_A32_SHIFT == 24 | 148 SK_A32_SHIFT == 24 |
160 libyuv::ARGBToABGR( | 149 libyuv::ARGBToABGR(static_cast<uint8*>(rgb_pixels), |
161 static_cast<uint8*>(bitmap->getPixels()), | 150 row_bytes, |
162 bitmap->rowBytes(), | 151 static_cast<uint8*>(rgb_pixels), |
163 static_cast<uint8*>(bitmap->getPixels()), | 152 row_bytes, |
164 bitmap->rowBytes(), | 153 video_frame->visible_rect().width(), |
165 video_frame->visible_rect().width(), | 154 video_frame->visible_rect().height()); |
166 video_frame->visible_rect().height()); | |
167 #endif | 155 #endif |
168 break; | 156 break; |
169 | 157 |
170 case media::VideoFrame::NATIVE_TEXTURE: | 158 case media::VideoFrame::NATIVE_TEXTURE: { |
171 DCHECK_EQ(video_frame->format(), media::VideoFrame::NATIVE_TEXTURE); | 159 DCHECK_EQ(video_frame->format(), media::VideoFrame::NATIVE_TEXTURE); |
172 video_frame->ReadPixelsFromNativeTexture(*bitmap); | 160 SkBitmap tmp; |
161 tmp.installPixels( | |
162 SkImageInfo::MakeN32Premul(video_frame->visible_rect().width(), | |
163 video_frame->visible_rect().height()), | |
164 rgb_pixels, | |
165 row_bytes); | |
166 video_frame->ReadPixelsFromNativeTexture(tmp); | |
173 break; | 167 break; |
174 | 168 } |
175 default: | 169 default: |
rileya (GONE FROM CHROMIUM)
2014/09/09 00:50:14
Off-topic: Should this be here?
scherkus (not reviewing)
2014/09/09 00:55:35
no -- being explicit would be preferable
| |
176 NOTREACHED(); | 170 NOTREACHED(); |
177 break; | 171 break; |
178 } | 172 } |
179 bitmap->notifyPixelsChanged(); | |
180 bitmap->unlockPixels(); | |
181 } | 173 } |
182 | 174 |
175 // Generates an RGB image from a VideoFrame. | |
176 class VideoImageGenerator : public SkImageGenerator { | |
177 public: | |
178 VideoImageGenerator(const scoped_refptr<VideoFrame>& frame) : frame_(frame) {} | |
179 virtual ~VideoImageGenerator() {} | |
180 | |
181 protected: | |
182 virtual bool onGetInfo(SkImageInfo* info) OVERRIDE { | |
183 info->fWidth = frame_->visible_rect().width(); | |
184 info->fHeight = frame_->visible_rect().height(); | |
185 info->fColorType = kBGRA_8888_SkColorType; | |
186 info->fAlphaType = kPremul_SkAlphaType; | |
187 return true; | |
188 } | |
189 | |
190 virtual bool onGetPixels(const SkImageInfo& info, | |
191 void* pixels, | |
192 size_t row_bytes, | |
193 SkPMColor c_table[], | |
194 int* c_table_count) OVERRIDE { | |
195 // If skia couldn't do the YUV conversion, we will. | |
196 ConvertVideoFrameToRGBPixels(frame_, pixels, row_bytes); | |
197 return true; | |
198 } | |
199 | |
200 virtual bool onGetYUV8Planes(SkISize sizes[3], | |
201 void* planes[3], | |
202 size_t row_bytes[3]) OVERRIDE { | |
203 // Currently Skia only supports JPEG color range YUV. | |
204 if (frame_->format() != VideoFrame::YV12J) | |
205 return false; | |
206 for (int plane = VideoFrame::kYPlane; plane <= VideoFrame::kVPlane; | |
207 ++plane) { | |
208 if (sizes) { | |
209 gfx::Size size; | |
210 size = VideoFrame::PlaneSize( | |
211 frame_->format(), plane, frame_->coded_size()); | |
212 sizes[plane].set(size.width(), size.height()); | |
213 } | |
214 if (row_bytes) | |
215 row_bytes[plane] = frame_->row_bytes(plane); | |
216 if (planes) { | |
217 memcpy(planes[plane], | |
218 frame_->data(plane), | |
219 VideoFrame::PlaneAllocationSize( | |
220 frame_->format(), plane, frame_->coded_size())); | |
221 } | |
222 } | |
223 return true; | |
224 } | |
225 | |
226 private: | |
227 scoped_refptr<VideoFrame> frame_; | |
228 }; | |
229 | |
183 SkCanvasVideoRenderer::SkCanvasVideoRenderer() | 230 SkCanvasVideoRenderer::SkCanvasVideoRenderer() |
184 : last_frame_timestamp_(media::kNoTimestamp()) { | 231 : last_frame_timestamp_(media::kNoTimestamp()) { |
185 } | 232 } |
186 | 233 |
187 SkCanvasVideoRenderer::~SkCanvasVideoRenderer() {} | 234 SkCanvasVideoRenderer::~SkCanvasVideoRenderer() {} |
188 | 235 |
189 void SkCanvasVideoRenderer::Paint(media::VideoFrame* video_frame, | 236 void SkCanvasVideoRenderer::Paint(const scoped_refptr<VideoFrame>& video_frame, |
190 SkCanvas* canvas, | 237 SkCanvas* canvas, |
191 const gfx::RectF& dest_rect, | 238 const gfx::RectF& dest_rect, |
192 uint8 alpha, | 239 uint8 alpha, |
193 SkXfermode::Mode mode, | 240 SkXfermode::Mode mode, |
194 VideoRotation video_rotation) { | 241 VideoRotation video_rotation) { |
195 if (alpha == 0) { | 242 if (alpha == 0) { |
196 return; | 243 return; |
197 } | 244 } |
198 | 245 |
199 SkRect dest; | 246 SkRect dest; |
200 dest.set(dest_rect.x(), dest_rect.y(), dest_rect.right(), dest_rect.bottom()); | 247 dest.set(dest_rect.x(), dest_rect.y(), dest_rect.right(), dest_rect.bottom()); |
201 | 248 |
202 SkPaint paint; | 249 SkPaint paint; |
203 paint.setAlpha(alpha); | 250 paint.setAlpha(alpha); |
204 | 251 |
205 // Paint black rectangle if there isn't a frame available or the | 252 // Paint black rectangle if there isn't a frame available or the |
206 // frame has an unexpected format. | 253 // frame has an unexpected format. |
207 if (!video_frame || !IsYUVOrNative(video_frame->format())) { | 254 if (!video_frame || !IsYUVOrNative(video_frame->format())) { |
208 canvas->drawRect(dest, paint); | 255 canvas->drawRect(dest, paint); |
209 return; | 256 return; |
210 } | 257 } |
211 | 258 |
212 // Check if we should convert and update |last_frame_|. | 259 // Check if we should convert and update |last_frame_|. |
213 if (last_frame_.isNull() || | 260 if (last_frame_.isNull() || |
214 video_frame->timestamp() != last_frame_timestamp_) { | 261 video_frame->timestamp() != last_frame_timestamp_) { |
215 ConvertVideoFrameToBitmap(video_frame, &last_frame_); | 262 if (!SkInstallDiscardablePixelRef(new VideoImageGenerator(video_frame), |
263 &last_frame_)) { | |
264 NOTREACHED(); | |
265 } | |
216 | 266 |
267 // TODO(rileya): Perform this rotation on the canvas, rather than allocating | |
268 // a new bitmap and copying. | |
217 switch (video_rotation) { | 269 switch (video_rotation) { |
218 case VIDEO_ROTATION_0: | 270 case VIDEO_ROTATION_0: |
219 break; | 271 break; |
220 case VIDEO_ROTATION_90: | 272 case VIDEO_ROTATION_90: |
221 last_frame_ = SkBitmapOperations::Rotate( | 273 last_frame_ = SkBitmapOperations::Rotate( |
222 last_frame_, SkBitmapOperations::ROTATION_90_CW); | 274 last_frame_, SkBitmapOperations::ROTATION_90_CW); |
223 break; | 275 break; |
224 case VIDEO_ROTATION_180: | 276 case VIDEO_ROTATION_180: |
225 last_frame_ = SkBitmapOperations::Rotate( | 277 last_frame_ = SkBitmapOperations::Rotate( |
226 last_frame_, SkBitmapOperations::ROTATION_180_CW); | 278 last_frame_, SkBitmapOperations::ROTATION_180_CW); |
227 break; | 279 break; |
228 case VIDEO_ROTATION_270: | 280 case VIDEO_ROTATION_270: |
229 last_frame_ = SkBitmapOperations::Rotate( | 281 last_frame_ = SkBitmapOperations::Rotate( |
230 last_frame_, SkBitmapOperations::ROTATION_270_CW); | 282 last_frame_, SkBitmapOperations::ROTATION_270_CW); |
231 break; | 283 break; |
232 } | 284 } |
233 | 285 |
234 last_frame_timestamp_ = video_frame->timestamp(); | 286 last_frame_timestamp_ = video_frame->timestamp(); |
235 } | 287 } |
236 | 288 |
237 paint.setXfermodeMode(mode); | 289 paint.setXfermodeMode(mode); |
238 | 290 |
239 // Paint using |last_frame_|. | 291 // Paint using |last_frame_|. |
240 paint.setFilterLevel(SkPaint::kLow_FilterLevel); | 292 paint.setFilterLevel(SkPaint::kLow_FilterLevel); |
241 canvas->drawBitmapRect(last_frame_, NULL, dest, &paint); | 293 canvas->drawBitmapRect(last_frame_, NULL, dest, &paint); |
242 } | 294 } |
243 | 295 |
244 void SkCanvasVideoRenderer::Copy(media::VideoFrame* video_frame, | 296 void SkCanvasVideoRenderer::Copy(const scoped_refptr<VideoFrame>& video_frame, |
245 SkCanvas* canvas) { | 297 SkCanvas* canvas) { |
246 Paint(video_frame, | 298 Paint(video_frame, |
247 canvas, | 299 canvas, |
248 video_frame->visible_rect(), | 300 video_frame->visible_rect(), |
249 0xff, | 301 0xff, |
250 SkXfermode::kSrc_Mode, | 302 SkXfermode::kSrc_Mode, |
251 media::VIDEO_ROTATION_0); | 303 media::VIDEO_ROTATION_0); |
252 } | 304 } |
253 | 305 |
254 } // namespace media | 306 } // namespace media |
OLD | NEW |