OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 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/blink/video_frame_compositor.h" | 5 #include "media/blink/video_frame_compositor.h" |
6 | 6 |
| 7 #include "base/bind.h" |
| 8 #include "base/message_loop/message_loop.h" |
7 #include "media/base/video_frame.h" | 9 #include "media/base/video_frame.h" |
8 | 10 |
9 namespace media { | 11 namespace media { |
10 | 12 |
11 static bool IsOpaque(const scoped_refptr<VideoFrame>& frame) { | 13 static bool IsOpaque(const scoped_refptr<VideoFrame>& frame) { |
12 switch (frame->format()) { | 14 switch (frame->format()) { |
13 case VideoFrame::UNKNOWN: | 15 case VideoFrame::UNKNOWN: |
14 case VideoFrame::YV12: | 16 case VideoFrame::YV12: |
15 case VideoFrame::YV12J: | 17 case VideoFrame::YV12J: |
16 case VideoFrame::YV12HD: | 18 case VideoFrame::YV12HD: |
17 case VideoFrame::YV16: | 19 case VideoFrame::YV16: |
18 case VideoFrame::I420: | 20 case VideoFrame::I420: |
19 case VideoFrame::YV24: | 21 case VideoFrame::YV24: |
20 case VideoFrame::NV12: | 22 case VideoFrame::NV12: |
21 return true; | 23 return true; |
22 | 24 |
23 case VideoFrame::YV12A: | 25 case VideoFrame::YV12A: |
24 #if defined(VIDEO_HOLE) | 26 #if defined(VIDEO_HOLE) |
25 case VideoFrame::HOLE: | 27 case VideoFrame::HOLE: |
26 #endif // defined(VIDEO_HOLE) | 28 #endif // defined(VIDEO_HOLE) |
27 case VideoFrame::NATIVE_TEXTURE: | 29 case VideoFrame::NATIVE_TEXTURE: |
28 case VideoFrame::ARGB: | 30 case VideoFrame::ARGB: |
29 break; | 31 break; |
30 } | 32 } |
31 return false; | 33 return false; |
32 } | 34 } |
33 | 35 |
34 VideoFrameCompositor::VideoFrameCompositor( | 36 VideoFrameCompositor::VideoFrameCompositor( |
| 37 const scoped_refptr<base::SingleThreadTaskRunner>& compositor_task_runner, |
35 const base::Callback<void(gfx::Size)>& natural_size_changed_cb, | 38 const base::Callback<void(gfx::Size)>& natural_size_changed_cb, |
36 const base::Callback<void(bool)>& opacity_changed_cb) | 39 const base::Callback<void(bool)>& opacity_changed_cb) |
37 : natural_size_changed_cb_(natural_size_changed_cb), | 40 : compositor_task_runner_(compositor_task_runner), |
| 41 natural_size_changed_cb_(natural_size_changed_cb), |
38 opacity_changed_cb_(opacity_changed_cb), | 42 opacity_changed_cb_(opacity_changed_cb), |
39 client_(NULL) { | 43 client_(nullptr), |
| 44 callback_(nullptr), |
| 45 rendering_(false) { |
40 } | 46 } |
41 | 47 |
42 VideoFrameCompositor::~VideoFrameCompositor() { | 48 VideoFrameCompositor::~VideoFrameCompositor() { |
| 49 DCHECK(compositor_task_runner_->BelongsToCurrentThread()); |
| 50 DCHECK(!callback_); |
| 51 DCHECK(!rendering_); |
43 if (client_) | 52 if (client_) |
44 client_->StopUsingProvider(); | 53 client_->StopUsingProvider(); |
45 } | 54 } |
46 | 55 |
| 56 void VideoFrameCompositor::OnRendererStateUpdate() { |
| 57 DCHECK(compositor_task_runner_->BelongsToCurrentThread()); |
| 58 if (!client_) |
| 59 return; |
| 60 |
| 61 base::AutoLock lock(lock_); |
| 62 if (callback_) { |
| 63 if (rendering_) |
| 64 client_->StartRendering(); |
| 65 |
| 66 // TODO(dalecurtis): This will need to request the first frame so we have |
| 67 // something to show, even if playback hasn't started yet. |
| 68 } else if (rendering_) { |
| 69 client_->StopRendering(); |
| 70 } |
| 71 } |
| 72 |
47 void VideoFrameCompositor::SetVideoFrameProviderClient( | 73 void VideoFrameCompositor::SetVideoFrameProviderClient( |
48 cc::VideoFrameProvider::Client* client) { | 74 cc::VideoFrameProvider::Client* client) { |
| 75 DCHECK(compositor_task_runner_->BelongsToCurrentThread()); |
49 if (client_) | 76 if (client_) |
50 client_->StopUsingProvider(); | 77 client_->StopUsingProvider(); |
51 client_ = client; | 78 client_ = client; |
| 79 OnRendererStateUpdate(); |
52 } | 80 } |
53 | 81 |
54 scoped_refptr<VideoFrame> VideoFrameCompositor::GetCurrentFrame() { | 82 scoped_refptr<VideoFrame> VideoFrameCompositor::GetCurrentFrame() { |
| 83 base::AutoLock lock(lock_); |
55 return current_frame_; | 84 return current_frame_; |
56 } | 85 } |
57 | 86 |
58 void VideoFrameCompositor::PutCurrentFrame( | 87 void VideoFrameCompositor::PutCurrentFrame() { |
59 const scoped_refptr<VideoFrame>& frame) { | 88 DCHECK(compositor_task_runner_->BelongsToCurrentThread()); |
| 89 // TODO(dalecurtis): Wire up a flag for RenderCallback::OnFrameDropped(). |
60 } | 90 } |
61 | 91 |
62 void VideoFrameCompositor::UpdateCurrentFrame( | 92 void VideoFrameCompositor::SetVisible(bool visible) { |
| 93 DCHECK(compositor_task_runner_->BelongsToCurrentThread()); |
| 94 // TODO(dalecurtis): Wire up to RenderCallback::OnRenderSuspended() and |
| 95 // OnRenderResumed(). |
| 96 } |
| 97 |
| 98 bool VideoFrameCompositor::UpdateCurrentFrame(base::TimeTicks deadline_min, |
| 99 base::TimeTicks deadline_max) { |
| 100 // TODO(dalecurtis): Wire this up to RenderCallback::Render(). |
| 101 base::AutoLock lock(lock_); |
| 102 return false; |
| 103 } |
| 104 |
| 105 void VideoFrameCompositor::Start(RenderCallback* callback) { |
| 106 NOTREACHED(); |
| 107 |
| 108 // Called from the media thread. |
| 109 base::AutoLock lock(lock_); |
| 110 callback_ = callback; |
| 111 rendering_ = true; |
| 112 compositor_task_runner_->PostTask( |
| 113 FROM_HERE, base::Bind(&VideoFrameCompositor::OnRendererStateUpdate, |
| 114 base::Unretained(this))); |
| 115 } |
| 116 |
| 117 void VideoFrameCompositor::Stop() { |
| 118 NOTREACHED(); |
| 119 |
| 120 base::AutoLock lock(lock_); |
| 121 callback_ = nullptr; |
| 122 rendering_ = false; |
| 123 compositor_task_runner_->PostTask( |
| 124 FROM_HERE, base::Bind(&VideoFrameCompositor::OnRendererStateUpdate, |
| 125 base::Unretained(this))); |
| 126 } |
| 127 |
| 128 void VideoFrameCompositor::PaintFrameUsingOldRenderingPath( |
63 const scoped_refptr<VideoFrame>& frame) { | 129 const scoped_refptr<VideoFrame>& frame) { |
| 130 if (!compositor_task_runner_->BelongsToCurrentThread()) { |
| 131 compositor_task_runner_->PostTask( |
| 132 FROM_HERE, |
| 133 base::Bind(&VideoFrameCompositor::PaintFrameUsingOldRenderingPath, |
| 134 base::Unretained(this), frame)); |
| 135 return; |
| 136 } |
| 137 |
64 if (current_frame_.get() && | 138 if (current_frame_.get() && |
65 current_frame_->natural_size() != frame->natural_size()) { | 139 current_frame_->natural_size() != frame->natural_size()) { |
66 natural_size_changed_cb_.Run(frame->natural_size()); | 140 natural_size_changed_cb_.Run(frame->natural_size()); |
67 } | 141 } |
68 | 142 |
69 if (!current_frame_.get() || IsOpaque(current_frame_) != IsOpaque(frame)) { | 143 if (!current_frame_.get() || IsOpaque(current_frame_) != IsOpaque(frame)) { |
70 opacity_changed_cb_.Run(IsOpaque(frame)); | 144 opacity_changed_cb_.Run(IsOpaque(frame)); |
71 } | 145 } |
72 | 146 |
73 current_frame_ = frame; | 147 current_frame_ = frame; |
74 | 148 |
75 if (client_) | 149 if (client_) |
76 client_->DidReceiveFrame(); | 150 client_->DidReceiveFrame(); |
77 } | 151 } |
78 | 152 |
79 } // namespace media | 153 } // namespace media |
OLD | NEW |