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
3 // found in the LICENSE file.
5 #include "media/cast/receiver/video_decoder.h"
8 #include "base/bind_helpers.h"
9 #include "base/json/json_reader.h"
10 #include "base/location.h"
11 #include "base/logging.h"
12 #include "base/values.h"
13 #include "media/base/video_util.h"
14 #include "media/cast/cast_defines.h"
15 #include "media/cast/cast_environment.h"
16 // VPX_CODEC_DISABLE_COMPAT excludes parts of the libvpx API that provide
17 // backwards compatibility for legacy applications using the library.
18 #define VPX_CODEC_DISABLE_COMPAT 1
19 #include "third_party/libvpx/source/libvpx/vpx/vp8dx.h"
20 #include "third_party/libvpx/source/libvpx/vpx/vpx_decoder.h"
21 #include "ui/gfx/geometry/size.h"
26 // Base class that handles the common problem of detecting dropped frames, and
27 // then invoking the Decode() method implemented by the subclasses to convert
28 // the encoded payload data into a usable video frame.
29 class VideoDecoder::ImplBase
30 : public base::RefCountedThreadSafe
<VideoDecoder::ImplBase
> {
32 ImplBase(const scoped_refptr
<CastEnvironment
>& cast_environment
,
34 : cast_environment_(cast_environment
),
36 operational_status_(STATUS_UNINITIALIZED
),
37 seen_first_frame_(false) {}
39 OperationalStatus
InitializationResult() const {
40 return operational_status_
;
43 void DecodeFrame(scoped_ptr
<EncodedFrame
> encoded_frame
,
44 const DecodeFrameCallback
& callback
) {
45 DCHECK_EQ(operational_status_
, STATUS_INITIALIZED
);
47 static_assert(sizeof(encoded_frame
->frame_id
) == sizeof(last_frame_id_
),
48 "size of frame_id types do not match");
49 bool is_continuous
= true;
50 if (seen_first_frame_
) {
51 const uint32 frames_ahead
= encoded_frame
->frame_id
- last_frame_id_
;
52 if (frames_ahead
> 1) {
53 RecoverBecauseFramesWereDropped();
54 is_continuous
= false;
57 seen_first_frame_
= true;
59 last_frame_id_
= encoded_frame
->frame_id
;
61 const scoped_refptr
<VideoFrame
> decoded_frame
= Decode(
62 encoded_frame
->mutable_bytes(),
63 static_cast<int>(encoded_frame
->data
.size()));
64 cast_environment_
->PostTask(
65 CastEnvironment::MAIN
,
67 base::Bind(callback
, decoded_frame
, is_continuous
));
71 friend class base::RefCountedThreadSafe
<ImplBase
>;
72 virtual ~ImplBase() {}
74 virtual void RecoverBecauseFramesWereDropped() {}
76 // Note: Implementation of Decode() is allowed to mutate |data|.
77 virtual scoped_refptr
<VideoFrame
> Decode(uint8
* data
, int len
) = 0;
79 const scoped_refptr
<CastEnvironment
> cast_environment_
;
82 // Subclass' ctor is expected to set this to STATUS_INITIALIZED.
83 OperationalStatus operational_status_
;
86 bool seen_first_frame_
;
87 uint32 last_frame_id_
;
89 DISALLOW_COPY_AND_ASSIGN(ImplBase
);
92 class VideoDecoder::Vp8Impl
: public VideoDecoder::ImplBase
{
94 explicit Vp8Impl(const scoped_refptr
<CastEnvironment
>& cast_environment
)
95 : ImplBase(cast_environment
, CODEC_VIDEO_VP8
) {
96 if (ImplBase::operational_status_
!= STATUS_UNINITIALIZED
)
99 vpx_codec_dec_cfg_t cfg
= {0};
100 // TODO(miu): Revisit this for typical multi-core desktop use case. This
101 // feels like it should be 4 or 8.
104 DCHECK(vpx_codec_get_caps(vpx_codec_vp8_dx()) & VPX_CODEC_CAP_POSTPROC
);
105 if (vpx_codec_dec_init(&context_
,
108 VPX_CODEC_USE_POSTPROC
) != VPX_CODEC_OK
) {
109 ImplBase::operational_status_
= STATUS_INVALID_CONFIGURATION
;
112 ImplBase::operational_status_
= STATUS_INITIALIZED
;
116 ~Vp8Impl() override
{
117 if (ImplBase::operational_status_
== STATUS_INITIALIZED
)
118 CHECK_EQ(VPX_CODEC_OK
, vpx_codec_destroy(&context_
));
121 scoped_refptr
<VideoFrame
> Decode(uint8
* data
, int len
) override
{
122 if (len
<= 0 || vpx_codec_decode(&context_
,
124 static_cast<unsigned int>(len
),
126 0) != VPX_CODEC_OK
) {
130 vpx_codec_iter_t iter
= NULL
;
131 vpx_image_t
* const image
= vpx_codec_get_frame(&context_
, &iter
);
134 if (image
->fmt
!= VPX_IMG_FMT_I420
&& image
->fmt
!= VPX_IMG_FMT_YV12
) {
138 DCHECK(vpx_codec_get_frame(&context_
, &iter
) == NULL
)
139 << "Should have only decoded exactly one frame.";
141 const gfx::Size
frame_size(image
->d_w
, image
->d_h
);
142 // Note: Timestamp for the VideoFrame will be set in VideoReceiver.
143 const scoped_refptr
<VideoFrame
> decoded_frame
=
144 VideoFrame::CreateFrame(VideoFrame::YV12
,
146 gfx::Rect(frame_size
),
149 CopyYPlane(image
->planes
[VPX_PLANE_Y
],
150 image
->stride
[VPX_PLANE_Y
],
152 decoded_frame
.get());
153 CopyUPlane(image
->planes
[VPX_PLANE_U
],
154 image
->stride
[VPX_PLANE_U
],
155 (image
->d_h
+ 1) / 2,
156 decoded_frame
.get());
157 CopyVPlane(image
->planes
[VPX_PLANE_V
],
158 image
->stride
[VPX_PLANE_V
],
159 (image
->d_h
+ 1) / 2,
160 decoded_frame
.get());
161 return decoded_frame
;
164 // VPX decoder context (i.e., an instantiation).
165 vpx_codec_ctx_t context_
;
167 DISALLOW_COPY_AND_ASSIGN(Vp8Impl
);
170 #ifndef OFFICIAL_BUILD
171 // A fake video decoder that always output 2x2 black frames.
172 class VideoDecoder::FakeImpl
: public VideoDecoder::ImplBase
{
174 explicit FakeImpl(const scoped_refptr
<CastEnvironment
>& cast_environment
)
175 : ImplBase(cast_environment
, CODEC_VIDEO_FAKE
),
176 last_decoded_id_(-1) {
177 if (ImplBase::operational_status_
!= STATUS_UNINITIALIZED
)
179 ImplBase::operational_status_
= STATUS_INITIALIZED
;
183 ~FakeImpl() override
{}
185 scoped_refptr
<VideoFrame
> Decode(uint8
* data
, int len
) override
{
186 // Make sure this is a JSON string.
187 if (!len
|| data
[0] != '{')
189 base::JSONReader reader
;
190 scoped_ptr
<base::Value
> values(
191 reader
.Read(base::StringPiece(reinterpret_cast<char*>(data
), len
)));
194 base::DictionaryValue
* dict
= NULL
;
195 values
->GetAsDictionary(&dict
);
200 dict
->GetBoolean("key", &key
);
201 dict
->GetInteger("id", &id
);
202 dict
->GetInteger("ref", &ref
);
203 DCHECK(id
== last_decoded_id_
+ 1);
204 last_decoded_id_
= id
;
205 return media::VideoFrame::CreateBlackFrame(gfx::Size(2, 2));
208 int last_decoded_id_
;
210 DISALLOW_COPY_AND_ASSIGN(FakeImpl
);
214 VideoDecoder::VideoDecoder(
215 const scoped_refptr
<CastEnvironment
>& cast_environment
,
217 : cast_environment_(cast_environment
) {
219 #ifndef OFFICIAL_BUILD
220 case CODEC_VIDEO_FAKE
:
221 impl_
= new FakeImpl(cast_environment
);
224 case CODEC_VIDEO_VP8
:
225 impl_
= new Vp8Impl(cast_environment
);
227 case CODEC_VIDEO_H264
:
228 // TODO(miu): Need implementation.
232 NOTREACHED() << "Unknown or unspecified codec.";
237 VideoDecoder::~VideoDecoder() {}
239 OperationalStatus
VideoDecoder::InitializationResult() const {
241 return impl_
->InitializationResult();
242 return STATUS_UNSUPPORTED_CODEC
;
245 void VideoDecoder::DecodeFrame(
246 scoped_ptr
<EncodedFrame
> encoded_frame
,
247 const DecodeFrameCallback
& callback
) {
248 DCHECK(encoded_frame
.get());
249 DCHECK(!callback
.is_null());
250 if (!impl_
.get() || impl_
->InitializationResult() != STATUS_INITIALIZED
) {
251 callback
.Run(make_scoped_refptr
<VideoFrame
>(NULL
), false);
254 cast_environment_
->PostTask(CastEnvironment::VIDEO
,
256 base::Bind(&VideoDecoder::ImplBase::DecodeFrame
,
258 base::Passed(&encoded_frame
),