1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "content/renderer/media/webmediaplayer_ms.h"
10 #include "base/callback.h"
11 #include "base/message_loop/message_loop.h"
12 #include "base/metrics/histogram.h"
13 #include "cc/blink/context_provider_web_context.h"
14 #include "cc/blink/web_layer_impl.h"
15 #include "cc/layers/video_layer.h"
16 #include "content/public/renderer/media_stream_audio_renderer.h"
17 #include "content/public/renderer/media_stream_renderer_factory.h"
18 #include "content/public/renderer/render_view.h"
19 #include "content/public/renderer/video_frame_provider.h"
20 #include "content/renderer/render_frame_impl.h"
21 #include "content/renderer/render_thread_impl.h"
22 #include "gpu/blink/webgraphicscontext3d_impl.h"
23 #include "media/base/media_log.h"
24 #include "media/base/video_frame.h"
25 #include "media/base/video_rotation.h"
26 #include "media/base/video_util.h"
27 #include "media/blink/webmediaplayer_delegate.h"
28 #include "third_party/WebKit/public/platform/WebMediaPlayerClient.h"
29 #include "third_party/WebKit/public/platform/WebRect.h"
30 #include "third_party/WebKit/public/platform/WebSize.h"
31 #include "third_party/WebKit/public/platform/WebURL.h"
32 #include "third_party/WebKit/public/web/WebFrame.h"
33 #include "third_party/WebKit/public/web/WebView.h"
34 #include "third_party/skia/include/core/SkBitmap.h"
36 using blink::WebCanvas
;
37 using blink::WebMediaPlayer
;
45 // This function copies |frame| to a new YV12 media::VideoFrame.
46 scoped_refptr
<media::VideoFrame
> CopyFrameToYV12(
47 const scoped_refptr
<media::VideoFrame
>& frame
,
48 media::SkCanvasVideoRenderer
* video_renderer
) {
49 const scoped_refptr
<media::VideoFrame
> new_frame
=
50 media::VideoFrame::CreateFrame(media::PIXEL_FORMAT_YV12
,
51 frame
->coded_size(), frame
->visible_rect(),
52 frame
->natural_size(), frame
->timestamp());
54 if (frame
->HasTextures()) {
55 DCHECK(frame
->format() == media::PIXEL_FORMAT_ARGB
||
56 frame
->format() == media::PIXEL_FORMAT_XRGB
);
58 bitmap
.allocN32Pixels(frame
->visible_rect().width(),
59 frame
->visible_rect().height());
60 SkCanvas
canvas(bitmap
);
62 cc::ContextProvider
* const provider
=
63 RenderThreadImpl::current()->SharedMainThreadContextProvider().get();
65 const media::Context3D context_3d
=
66 media::Context3D(provider
->ContextGL(), provider
->GrContext());
67 DCHECK(context_3d
.gl
);
68 video_renderer
->Copy(frame
.get(), &canvas
, context_3d
);
70 // GPU Process crashed.
71 bitmap
.eraseColor(SK_ColorTRANSPARENT
);
73 media::CopyRGBToVideoFrame(reinterpret_cast<uint8
*>(bitmap
.getPixels()),
75 frame
->visible_rect(),
78 DCHECK(frame
->IsMappable());
79 DCHECK(frame
->format() == media::PIXEL_FORMAT_YV12
||
80 frame
->format() == media::PIXEL_FORMAT_I420
);
81 for (size_t i
= 0; i
< media::VideoFrame::NumPlanes(frame
->format()); ++i
) {
82 media::CopyPlane(i
, frame
->data(i
), frame
->stride(i
),
83 frame
->rows(i
), new_frame
.get());
89 } // anonymous namespace
91 WebMediaPlayerMS::WebMediaPlayerMS(
92 blink::WebFrame
* frame
,
93 blink::WebMediaPlayerClient
* client
,
94 base::WeakPtr
<media::WebMediaPlayerDelegate
> delegate
,
95 media::MediaLog
* media_log
,
96 scoped_ptr
<MediaStreamRendererFactory
> factory
)
98 network_state_(WebMediaPlayer::NetworkStateEmpty
),
99 ready_state_(WebMediaPlayer::ReadyStateHaveNothing
),
100 buffered_(static_cast<size_t>(0)),
105 current_frame_used_(false),
106 video_frame_provider_client_(NULL
),
107 received_first_frame_(false),
108 total_frame_count_(0),
109 dropped_frame_count_(0),
110 media_log_(media_log
),
111 renderer_factory_(factory
.Pass()) {
112 DVLOG(1) << "WebMediaPlayerMS::ctor";
113 media_log_
->AddEvent(
114 media_log_
->CreateEvent(media::MediaLogEvent::WEBMEDIAPLAYER_CREATED
));
117 WebMediaPlayerMS::~WebMediaPlayerMS() {
118 DVLOG(1) << "WebMediaPlayerMS::dtor";
119 DCHECK(thread_checker_
.CalledOnValidThread());
121 SetVideoFrameProviderClient(NULL
);
122 GetClient()->setWebLayer(NULL
);
124 if (video_frame_provider_
.get())
125 video_frame_provider_
->Stop();
127 if (audio_renderer_
.get())
128 audio_renderer_
->Stop();
130 media_log_
->AddEvent(
131 media_log_
->CreateEvent(media::MediaLogEvent::WEBMEDIAPLAYER_DESTROYED
));
134 delegate_
->PlayerGone(this);
137 void WebMediaPlayerMS::load(LoadType load_type
,
138 const blink::WebURL
& url
,
139 CORSMode cors_mode
) {
140 DVLOG(1) << "WebMediaPlayerMS::load";
141 DCHECK(thread_checker_
.CalledOnValidThread());
143 // TODO(acolwell): Change this to DCHECK_EQ(load_type,
144 // LoadTypeMediaStream) once Blink-side changes land.
145 DCHECK_NE(load_type
, LoadTypeMediaSource
);
149 SetNetworkState(WebMediaPlayer::NetworkStateLoading
);
150 SetReadyState(WebMediaPlayer::ReadyStateHaveNothing
);
151 media_log_
->AddEvent(media_log_
->CreateLoadEvent(url
.spec()));
153 video_frame_provider_
= renderer_factory_
->GetVideoFrameProvider(
155 base::Bind(&WebMediaPlayerMS::OnSourceError
, AsWeakPtr()),
156 base::Bind(&WebMediaPlayerMS::OnFrameAvailable
, AsWeakPtr()));
158 RenderFrame
* const frame
= RenderFrame::FromWebFrame(frame_
);
159 audio_renderer_
= renderer_factory_
->GetAudioRenderer(
161 frame
->GetRoutingID());
163 if (!video_frame_provider_
&& !audio_renderer_
) {
164 SetNetworkState(WebMediaPlayer::NetworkStateNetworkError
);
168 if (audio_renderer_
) {
169 audio_renderer_
->SetVolume(volume_
);
170 audio_renderer_
->Start();
172 if (video_frame_provider_
)
173 video_frame_provider_
->Start();
174 if (audio_renderer_
&& !video_frame_provider_
) {
175 // This is audio-only mode.
176 SetReadyState(WebMediaPlayer::ReadyStateHaveMetadata
);
177 SetReadyState(WebMediaPlayer::ReadyStateHaveEnoughData
);
181 void WebMediaPlayerMS::play() {
182 DVLOG(1) << "WebMediaPlayerMS::play";
183 DCHECK(thread_checker_
.CalledOnValidThread());
186 if (video_frame_provider_
.get())
187 video_frame_provider_
->Play();
189 if (audio_renderer_
.get())
190 audio_renderer_
->Play();
193 delegate_
->DidPlay(this);
198 media_log_
->AddEvent(media_log_
->CreateEvent(media::MediaLogEvent::PLAY
));
201 void WebMediaPlayerMS::pause() {
202 DVLOG(1) << "WebMediaPlayerMS::pause";
203 DCHECK(thread_checker_
.CalledOnValidThread());
205 if (video_frame_provider_
.get())
206 video_frame_provider_
->Pause();
209 if (audio_renderer_
.get())
210 audio_renderer_
->Pause();
213 delegate_
->DidPause(this);
218 media_log_
->AddEvent(media_log_
->CreateEvent(media::MediaLogEvent::PAUSE
));
220 if (!current_frame_
.get())
223 // Copy the frame so that rendering can show the last received frame.
224 // The original frame must not be referenced when the player is paused since
225 // there might be a finite number of available buffers. E.g, video that
226 // originates from a video camera.
227 scoped_refptr
<media::VideoFrame
> new_frame
=
228 CopyFrameToYV12(current_frame_
, &video_renderer_
);
230 base::AutoLock
auto_lock(current_frame_lock_
);
231 current_frame_
= new_frame
;
234 bool WebMediaPlayerMS::supportsSave() const {
235 DCHECK(thread_checker_
.CalledOnValidThread());
239 void WebMediaPlayerMS::seek(double seconds
) {
240 DCHECK(thread_checker_
.CalledOnValidThread());
243 void WebMediaPlayerMS::setRate(double rate
) {
244 DCHECK(thread_checker_
.CalledOnValidThread());
247 void WebMediaPlayerMS::setVolume(double volume
) {
248 DCHECK(thread_checker_
.CalledOnValidThread());
249 DVLOG(1) << "WebMediaPlayerMS::setVolume(volume=" << volume
<< ")";
251 if (audio_renderer_
.get())
252 audio_renderer_
->SetVolume(volume_
);
255 void WebMediaPlayerMS::setSinkId(const blink::WebString
& device_id
,
256 media::WebSetSinkIdCB
* web_callback
) {
257 DCHECK(thread_checker_
.CalledOnValidThread());
258 DVLOG(1) << __FUNCTION__
;
259 media::SwitchOutputDeviceCB callback
=
260 media::ConvertToSwitchOutputDeviceCB(web_callback
);
261 if (audio_renderer_
.get()) {
262 media::OutputDevice
* output_device
= audio_renderer_
->GetOutputDevice();
264 const std::string
device_id_str(device_id
.utf8());
265 const GURL
security_origin(frame_
->securityOrigin().toString().utf8());
266 output_device
->SwitchOutputDevice(device_id_str
, security_origin
,
271 callback
.Run(media::SWITCH_OUTPUT_DEVICE_RESULT_ERROR_NOT_SUPPORTED
);
274 void WebMediaPlayerMS::setPreload(WebMediaPlayer::Preload preload
) {
275 DCHECK(thread_checker_
.CalledOnValidThread());
278 bool WebMediaPlayerMS::hasVideo() const {
279 DCHECK(thread_checker_
.CalledOnValidThread());
280 return (video_frame_provider_
.get() != NULL
);
283 bool WebMediaPlayerMS::hasAudio() const {
284 DCHECK(thread_checker_
.CalledOnValidThread());
285 return (audio_renderer_
.get() != NULL
);
288 blink::WebSize
WebMediaPlayerMS::naturalSize() const {
289 DCHECK(thread_checker_
.CalledOnValidThread());
291 if (!current_frame_
.get())
292 return blink::WebSize();
294 const gfx::Size
& size
= current_frame_
->natural_size();
295 DVLOG(3) << "WebMediaPlayerMS::naturalSize, " << size
.ToString();
296 return blink::WebSize(size
);
299 bool WebMediaPlayerMS::paused() const {
300 DCHECK(thread_checker_
.CalledOnValidThread());
304 bool WebMediaPlayerMS::seeking() const {
305 DCHECK(thread_checker_
.CalledOnValidThread());
309 double WebMediaPlayerMS::duration() const {
310 DCHECK(thread_checker_
.CalledOnValidThread());
311 return std::numeric_limits
<double>::infinity();
314 double WebMediaPlayerMS::currentTime() const {
315 DCHECK(thread_checker_
.CalledOnValidThread());
316 if (current_time_
.ToInternalValue() != 0) {
317 return current_time_
.InSecondsF();
318 } else if (audio_renderer_
.get()) {
319 return audio_renderer_
->GetCurrentRenderTime().InSecondsF();
324 WebMediaPlayer::NetworkState
WebMediaPlayerMS::networkState() const {
325 DCHECK(thread_checker_
.CalledOnValidThread());
326 DVLOG(1) << "WebMediaPlayerMS::networkState, state:" << network_state_
;
327 return network_state_
;
330 WebMediaPlayer::ReadyState
WebMediaPlayerMS::readyState() const {
331 DCHECK(thread_checker_
.CalledOnValidThread());
332 DVLOG(1) << "WebMediaPlayerMS::readyState, state:" << ready_state_
;
336 blink::WebTimeRanges
WebMediaPlayerMS::buffered() const {
337 DCHECK(thread_checker_
.CalledOnValidThread());
341 blink::WebTimeRanges
WebMediaPlayerMS::seekable() const {
342 DCHECK(thread_checker_
.CalledOnValidThread());
343 return blink::WebTimeRanges();
346 bool WebMediaPlayerMS::didLoadingProgress() {
347 DCHECK(thread_checker_
.CalledOnValidThread());
351 void WebMediaPlayerMS::paint(blink::WebCanvas
* canvas
,
352 const blink::WebRect
& rect
,
354 SkXfermode::Mode mode
) {
355 DVLOG(3) << "WebMediaPlayerMS::paint";
356 DCHECK(thread_checker_
.CalledOnValidThread());
358 media::Context3D context_3d
;
359 if (current_frame_
.get() && current_frame_
->HasTextures()) {
360 cc::ContextProvider
* provider
=
361 RenderThreadImpl::current()->SharedMainThreadContextProvider().get();
362 // GPU Process crashed.
365 context_3d
= media::Context3D(provider
->ContextGL(), provider
->GrContext());
366 DCHECK(context_3d
.gl
);
368 gfx::RectF
dest_rect(rect
.x
, rect
.y
, rect
.width
, rect
.height
);
369 video_renderer_
.Paint(current_frame_
, canvas
, dest_rect
, alpha
, mode
,
370 media::VIDEO_ROTATION_0
, context_3d
);
373 base::AutoLock
auto_lock(current_frame_lock_
);
374 if (current_frame_
.get())
375 current_frame_used_
= true;
379 bool WebMediaPlayerMS::hasSingleSecurityOrigin() const {
380 DCHECK(thread_checker_
.CalledOnValidThread());
384 bool WebMediaPlayerMS::didPassCORSAccessCheck() const {
385 DCHECK(thread_checker_
.CalledOnValidThread());
389 double WebMediaPlayerMS::mediaTimeForTimeValue(double timeValue
) const {
390 return base::TimeDelta::FromSecondsD(timeValue
).InSecondsF();
393 unsigned WebMediaPlayerMS::decodedFrameCount() const {
394 DCHECK(thread_checker_
.CalledOnValidThread());
395 DVLOG(1) << "WebMediaPlayerMS::decodedFrameCount, " << total_frame_count_
;
396 return total_frame_count_
;
399 unsigned WebMediaPlayerMS::droppedFrameCount() const {
400 DCHECK(thread_checker_
.CalledOnValidThread());
401 DVLOG(1) << "WebMediaPlayerMS::droppedFrameCount, " << dropped_frame_count_
;
402 return dropped_frame_count_
;
405 unsigned WebMediaPlayerMS::audioDecodedByteCount() const {
406 DCHECK(thread_checker_
.CalledOnValidThread());
411 unsigned WebMediaPlayerMS::videoDecodedByteCount() const {
412 DCHECK(thread_checker_
.CalledOnValidThread());
417 bool WebMediaPlayerMS::copyVideoTextureToPlatformTexture(
418 blink::WebGraphicsContext3D
* web_graphics_context
,
419 unsigned int texture
,
420 unsigned int internal_format
,
422 bool premultiply_alpha
,
424 TRACE_EVENT0("media", "WebMediaPlayerMS:copyVideoTextureToPlatformTexture");
425 DCHECK(thread_checker_
.CalledOnValidThread());
427 scoped_refptr
<media::VideoFrame
> video_frame
;
429 base::AutoLock
auto_lock(current_frame_lock_
);
430 video_frame
= current_frame_
;
433 if (!video_frame
.get() || video_frame
->HasTextures() ||
434 media::VideoFrame::NumPlanes(video_frame
->format()) != 1) {
438 // TODO(dshwang): need more elegant way to convert WebGraphicsContext3D to
440 gpu::gles2::GLES2Interface
* const gl
=
441 static_cast<gpu_blink::WebGraphicsContext3DImpl
*>(web_graphics_context
)
443 media::SkCanvasVideoRenderer::CopyVideoFrameSingleTextureToGLTexture(
444 gl
, video_frame
.get(), texture
, internal_format
, type
, premultiply_alpha
,
449 void WebMediaPlayerMS::SetVideoFrameProviderClient(
450 cc::VideoFrameProvider::Client
* client
) {
451 // This is called from both the main renderer thread and the compositor
452 // thread (when the main thread is blocked).
453 if (video_frame_provider_client_
)
454 video_frame_provider_client_
->StopUsingProvider();
455 video_frame_provider_client_
= client
;
458 bool WebMediaPlayerMS::UpdateCurrentFrame(base::TimeTicks deadline_min
,
459 base::TimeTicks deadline_max
) {
460 // TODO(dalecurtis): This should make use of the deadline interval to ensure
461 // the painted frame is correct for the given interval.
466 bool WebMediaPlayerMS::HasCurrentFrame() {
467 base::AutoLock
auto_lock(current_frame_lock_
);
468 return current_frame_
;
471 scoped_refptr
<media::VideoFrame
> WebMediaPlayerMS::GetCurrentFrame() {
472 DVLOG(3) << "WebMediaPlayerMS::GetCurrentFrame";
473 base::AutoLock
auto_lock(current_frame_lock_
);
474 if (!current_frame_
.get())
476 current_frame_used_
= true;
477 return current_frame_
;
480 void WebMediaPlayerMS::PutCurrentFrame() {
481 DVLOG(3) << "WebMediaPlayerMS::PutCurrentFrame";
484 void WebMediaPlayerMS::OnFrameAvailable(
485 const scoped_refptr
<media::VideoFrame
>& frame
) {
486 DVLOG(3) << "WebMediaPlayerMS::OnFrameAvailable";
487 DCHECK(thread_checker_
.CalledOnValidThread());
488 ++total_frame_count_
;
489 if (!received_first_frame_
) {
490 received_first_frame_
= true;
492 base::AutoLock
auto_lock(current_frame_lock_
);
493 DCHECK(!current_frame_used_
);
494 current_frame_
= frame
;
496 SetReadyState(WebMediaPlayer::ReadyStateHaveMetadata
);
497 SetReadyState(WebMediaPlayer::ReadyStateHaveEnoughData
);
498 GetClient()->sizeChanged();
500 if (video_frame_provider_
.get()) {
501 video_weblayer_
.reset(new cc_blink::WebLayerImpl(
502 cc::VideoLayer::Create(cc_blink::WebLayerImpl::LayerSettings(), this,
503 media::VIDEO_ROTATION_0
)));
504 video_weblayer_
->setOpaque(true);
505 GetClient()->setWebLayer(video_weblayer_
.get());
509 // Do not update |current_frame_| when paused.
513 const bool size_changed
=
514 !current_frame_
.get() ||
515 current_frame_
->natural_size() != frame
->natural_size();
518 base::AutoLock
auto_lock(current_frame_lock_
);
519 if (!current_frame_used_
&& current_frame_
.get())
520 ++dropped_frame_count_
;
521 current_frame_
= frame
;
522 current_time_
= frame
->timestamp();
523 current_frame_used_
= false;
527 GetClient()->sizeChanged();
529 GetClient()->repaint();
532 void WebMediaPlayerMS::RepaintInternal() {
533 DVLOG(1) << "WebMediaPlayerMS::RepaintInternal";
534 DCHECK(thread_checker_
.CalledOnValidThread());
535 GetClient()->repaint();
538 void WebMediaPlayerMS::OnSourceError() {
539 DVLOG(1) << "WebMediaPlayerMS::OnSourceError";
540 DCHECK(thread_checker_
.CalledOnValidThread());
541 SetNetworkState(WebMediaPlayer::NetworkStateFormatError
);
545 void WebMediaPlayerMS::SetNetworkState(WebMediaPlayer::NetworkState state
) {
546 DCHECK(thread_checker_
.CalledOnValidThread());
547 network_state_
= state
;
548 // Always notify to ensure client has the latest value.
549 GetClient()->networkStateChanged();
552 void WebMediaPlayerMS::SetReadyState(WebMediaPlayer::ReadyState state
) {
553 DCHECK(thread_checker_
.CalledOnValidThread());
554 ready_state_
= state
;
555 // Always notify to ensure client has the latest value.
556 GetClient()->readyStateChanged();
559 blink::WebMediaPlayerClient
* WebMediaPlayerMS::GetClient() {
560 DCHECK(thread_checker_
.CalledOnValidThread());
565 } // namespace content