blob: d22f77bf56906f4e39efdbf4e86e82af98564f6c [file] [log] [blame]
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +09001// Copyright 2020 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.
4
5//#define LOG_NDEBUG 0
6#define LOG_TAG "V4L2DecodeInterface"
7
8#include <v4l2_codec2/components/V4L2DecodeInterface.h>
9
10#include <C2PlatformSupport.h>
11#include <SimpleC2Interface.h>
12#include <android/hardware/graphics/common/1.0/types.h>
13#include <log/log.h>
14#include <media/stagefright/foundation/MediaDefs.h>
15
16#include <v4l2_codec2/common/V4L2ComponentCommon.h>
David Staessens69f5a3b2021-03-18 13:14:34 +090017#include <v4l2_codec2/common/V4L2Device.h>
Chih-Yu Huangbcc6ef62020-05-27 17:52:33 +090018#include <v4l2_codec2/plugin_store/V4L2AllocatorId.h>
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +090019
20namespace android {
21namespace {
22
23constexpr size_t k1080pArea = 1920 * 1088;
24constexpr size_t k4KArea = 3840 * 2160;
25// Input bitstream buffer size for up to 1080p streams.
26constexpr size_t kInputBufferSizeFor1080p = 1024 * 1024; // 1MB
27// Input bitstream buffer size for up to 4k streams.
28constexpr size_t kInputBufferSizeFor4K = 4 * kInputBufferSizeFor1080p;
29
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +090030std::optional<VideoCodec> getCodecFromComponentName(const std::string& name) {
31 if (name == V4L2ComponentName::kH264Decoder || name == V4L2ComponentName::kH264SecureDecoder)
32 return VideoCodec::H264;
33 if (name == V4L2ComponentName::kVP8Decoder || name == V4L2ComponentName::kVP8SecureDecoder)
34 return VideoCodec::VP8;
35 if (name == V4L2ComponentName::kVP9Decoder || name == V4L2ComponentName::kVP9SecureDecoder)
36 return VideoCodec::VP9;
Jeffrey Kardatzke58a30542022-05-23 14:19:40 -070037 if (name == V4L2ComponentName::kHEVCDecoder || name == V4L2ComponentName::kHEVCSecureDecoder)
38 return VideoCodec::HEVC;
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +090039
40 ALOGE("Unknown name: %s", name.c_str());
41 return std::nullopt;
42}
43
44size_t calculateInputBufferSize(size_t area) {
45 if (area > k4KArea) {
46 ALOGW("Input buffer size for video size (%zu) larger than 4K (%zu) might be too small.",
47 area, k4KArea);
48 }
49
50 // Enlarge the input buffer for 4k video
51 if (area > k1080pArea) return kInputBufferSizeFor4K;
52 return kInputBufferSizeFor1080p;
53}
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +090054} // namespace
55
56// static
57C2R V4L2DecodeInterface::ProfileLevelSetter(bool /* mayBlock */,
58 C2P<C2StreamProfileLevelInfo::input>& info) {
59 return info.F(info.v.profile)
60 .validatePossible(info.v.profile)
61 .plus(info.F(info.v.level).validatePossible(info.v.level));
62}
63
64// static
65C2R V4L2DecodeInterface::SizeSetter(bool /* mayBlock */,
66 C2P<C2StreamPictureSizeInfo::output>& videoSize) {
67 return videoSize.F(videoSize.v.width)
68 .validatePossible(videoSize.v.width)
69 .plus(videoSize.F(videoSize.v.height).validatePossible(videoSize.v.height));
70}
71
72// static
73template <typename T>
74C2R V4L2DecodeInterface::DefaultColorAspectsSetter(bool /* mayBlock */, C2P<T>& def) {
75 if (def.v.range > C2Color::RANGE_OTHER) {
76 def.set().range = C2Color::RANGE_OTHER;
77 }
78 if (def.v.primaries > C2Color::PRIMARIES_OTHER) {
79 def.set().primaries = C2Color::PRIMARIES_OTHER;
80 }
81 if (def.v.transfer > C2Color::TRANSFER_OTHER) {
82 def.set().transfer = C2Color::TRANSFER_OTHER;
83 }
84 if (def.v.matrix > C2Color::MATRIX_OTHER) {
85 def.set().matrix = C2Color::MATRIX_OTHER;
86 }
87 return C2R::Ok();
88}
89
90// static
91C2R V4L2DecodeInterface::MergedColorAspectsSetter(
92 bool /* mayBlock */, C2P<C2StreamColorAspectsInfo::output>& merged,
93 const C2P<C2StreamColorAspectsTuning::output>& def,
94 const C2P<C2StreamColorAspectsInfo::input>& coded) {
95 // Take coded values for all specified fields, and default values for unspecified ones.
96 merged.set().range = coded.v.range == RANGE_UNSPECIFIED ? def.v.range : coded.v.range;
97 merged.set().primaries =
98 coded.v.primaries == PRIMARIES_UNSPECIFIED ? def.v.primaries : coded.v.primaries;
99 merged.set().transfer =
100 coded.v.transfer == TRANSFER_UNSPECIFIED ? def.v.transfer : coded.v.transfer;
101 merged.set().matrix = coded.v.matrix == MATRIX_UNSPECIFIED ? def.v.matrix : coded.v.matrix;
102 return C2R::Ok();
103}
104
105// static
106C2R V4L2DecodeInterface::MaxInputBufferSizeCalculator(
107 bool /* mayBlock */, C2P<C2StreamMaxBufferSizeInfo::input>& me,
108 const C2P<C2StreamPictureSizeInfo::output>& size) {
109 me.set().value = calculateInputBufferSize(size.v.width * size.v.height);
110 return C2R::Ok();
111}
112
113V4L2DecodeInterface::V4L2DecodeInterface(const std::string& name,
114 const std::shared_ptr<C2ReflectorHelper>& helper)
115 : C2InterfaceHelper(helper), mInitStatus(C2_OK) {
116 ALOGV("%s(%s)", __func__, name.c_str());
117
118 setDerivedInstance(this);
119
120 mVideoCodec = getCodecFromComponentName(name);
121 if (!mVideoCodec) {
122 ALOGE("Invalid component name: %s", name.c_str());
123 mInitStatus = C2_BAD_VALUE;
124 return;
125 }
126
Chih-Yu Huang5aaed6a2021-03-29 14:52:14 +0900127 addParameter(DefineParam(mKind, C2_PARAMKEY_COMPONENT_KIND)
128 .withConstValue(new C2ComponentKindSetting(C2Component::KIND_DECODER))
129 .build());
130
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900131 std::string inputMime;
132 switch (*mVideoCodec) {
133 case VideoCodec::H264:
134 inputMime = MEDIA_MIMETYPE_VIDEO_AVC;
135 addParameter(
136 DefineParam(mProfileLevel, C2_PARAMKEY_PROFILE_LEVEL)
137 .withDefault(new C2StreamProfileLevelInfo::input(
138 0u, C2Config::PROFILE_AVC_MAIN, C2Config::LEVEL_AVC_4))
139 .withFields(
140 {C2F(mProfileLevel, profile)
141 .oneOf({C2Config::PROFILE_AVC_BASELINE,
142 C2Config::PROFILE_AVC_CONSTRAINED_BASELINE,
143 C2Config::PROFILE_AVC_MAIN,
144 C2Config::PROFILE_AVC_HIGH,
145 C2Config::PROFILE_AVC_CONSTRAINED_HIGH}),
146 C2F(mProfileLevel, level)
147 .oneOf({C2Config::LEVEL_AVC_1, C2Config::LEVEL_AVC_1B,
148 C2Config::LEVEL_AVC_1_1, C2Config::LEVEL_AVC_1_2,
149 C2Config::LEVEL_AVC_1_3, C2Config::LEVEL_AVC_2,
150 C2Config::LEVEL_AVC_2_1, C2Config::LEVEL_AVC_2_2,
151 C2Config::LEVEL_AVC_3, C2Config::LEVEL_AVC_3_1,
152 C2Config::LEVEL_AVC_3_2, C2Config::LEVEL_AVC_4,
153 C2Config::LEVEL_AVC_4_1, C2Config::LEVEL_AVC_4_2,
154 C2Config::LEVEL_AVC_5, C2Config::LEVEL_AVC_5_1,
155 C2Config::LEVEL_AVC_5_2})})
156 .withSetter(ProfileLevelSetter)
157 .build());
158 break;
159
160 case VideoCodec::VP8:
161 inputMime = MEDIA_MIMETYPE_VIDEO_VP8;
162 addParameter(DefineParam(mProfileLevel, C2_PARAMKEY_PROFILE_LEVEL)
163 .withConstValue(new C2StreamProfileLevelInfo::input(
164 0u, C2Config::PROFILE_UNUSED, C2Config::LEVEL_UNUSED))
165 .build());
166 break;
167
168 case VideoCodec::VP9:
169 inputMime = MEDIA_MIMETYPE_VIDEO_VP9;
170 addParameter(
171 DefineParam(mProfileLevel, C2_PARAMKEY_PROFILE_LEVEL)
172 .withDefault(new C2StreamProfileLevelInfo::input(
173 0u, C2Config::PROFILE_VP9_0, C2Config::LEVEL_VP9_5))
174 .withFields({C2F(mProfileLevel, profile).oneOf({C2Config::PROFILE_VP9_0}),
175 C2F(mProfileLevel, level)
176 .oneOf({C2Config::LEVEL_VP9_1, C2Config::LEVEL_VP9_1_1,
177 C2Config::LEVEL_VP9_2, C2Config::LEVEL_VP9_2_1,
178 C2Config::LEVEL_VP9_3, C2Config::LEVEL_VP9_3_1,
179 C2Config::LEVEL_VP9_4, C2Config::LEVEL_VP9_4_1,
180 C2Config::LEVEL_VP9_5})})
181 .withSetter(ProfileLevelSetter)
182 .build());
183 break;
Jeffrey Kardatzke58a30542022-05-23 14:19:40 -0700184
185 case VideoCodec::HEVC:
186 inputMime = MEDIA_MIMETYPE_VIDEO_HEVC;
187 addParameter(
188 DefineParam(mProfileLevel, C2_PARAMKEY_PROFILE_LEVEL)
189 .withDefault(new C2StreamProfileLevelInfo::input(
190 0u, C2Config::PROFILE_HEVC_MAIN, C2Config::LEVEL_HEVC_MAIN_5_1))
191 .withFields({C2F(mProfileLevel, profile)
192 .oneOf({C2Config::PROFILE_HEVC_MAIN,
193 C2Config::PROFILE_HEVC_MAIN_STILL}),
194 C2F(mProfileLevel, level)
195 .oneOf({C2Config::LEVEL_HEVC_MAIN_1,
196 C2Config::LEVEL_HEVC_MAIN_2,
197 C2Config::LEVEL_HEVC_MAIN_2_1,
198 C2Config::LEVEL_HEVC_MAIN_3,
199 C2Config::LEVEL_HEVC_MAIN_3_1,
200 C2Config::LEVEL_HEVC_MAIN_4,
201 C2Config::LEVEL_HEVC_MAIN_4_1,
202 C2Config::LEVEL_HEVC_MAIN_5,
203 C2Config::LEVEL_HEVC_MAIN_5_1,
204 C2Config::LEVEL_HEVC_MAIN_5_2,
205 C2Config::LEVEL_HEVC_HIGH_4,
206 C2Config::LEVEL_HEVC_HIGH_4_1,
207 C2Config::LEVEL_HEVC_HIGH_5,
208 C2Config::LEVEL_HEVC_HIGH_5_1,
209 C2Config::LEVEL_HEVC_HIGH_5_2})})
210 .withSetter(ProfileLevelSetter)
211 .build());
212 break;
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900213 }
214
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900215 addParameter(
216 DefineParam(mInputFormat, C2_PARAMKEY_INPUT_STREAM_BUFFER_TYPE)
217 .withConstValue(new C2StreamBufferTypeSetting::input(0u, C2BufferData::LINEAR))
218 .build());
219 addParameter(
220 DefineParam(mInputMemoryUsage, C2_PARAMKEY_INPUT_STREAM_USAGE)
221 .withConstValue(new C2StreamUsageTuning::input(
222 0u, static_cast<uint64_t>(android::hardware::graphics::common::V1_0::
223 BufferUsage::VIDEO_DECODER)))
224 .build());
225
226 addParameter(DefineParam(mOutputFormat, C2_PARAMKEY_OUTPUT_STREAM_BUFFER_TYPE)
227 .withConstValue(
228 new C2StreamBufferTypeSetting::output(0u, C2BufferData::GRAPHIC))
229 .build());
Chih-Yu Huangb65c2122020-05-18 17:29:18 +0900230 addParameter(
231 DefineParam(mOutputDelay, C2_PARAMKEY_OUTPUT_DELAY)
232 .withConstValue(new C2PortDelayTuning::output(getOutputDelay(*mVideoCodec)))
233 .build());
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900234
235 addParameter(DefineParam(mInputMediaType, C2_PARAMKEY_INPUT_MEDIA_TYPE)
236 .withConstValue(AllocSharedString<C2PortMediaTypeSetting::input>(
237 inputMime.c_str()))
238 .build());
239
240 addParameter(DefineParam(mOutputMediaType, C2_PARAMKEY_OUTPUT_MEDIA_TYPE)
241 .withConstValue(AllocSharedString<C2PortMediaTypeSetting::output>(
242 MEDIA_MIMETYPE_VIDEO_RAW))
243 .build());
244
Chih-Yu Huang776b2ad2020-08-26 13:14:51 +0900245 // Note(b/165826281): The check is not used at Android framework currently.
246 // In order to fasten the bootup time, we use the maximum supported size instead of querying the
247 // capability from the V4L2 device.
Chih-Yu Huangd23406c2020-08-26 10:12:58 +0900248 addParameter(DefineParam(mSize, C2_PARAMKEY_PICTURE_SIZE)
249 .withDefault(new C2StreamPictureSizeInfo::output(0u, 320, 240))
250 .withFields({
251 C2F(mSize, width).inRange(16, 4096, 16),
252 C2F(mSize, height).inRange(16, 4096, 16),
253 })
254 .withSetter(SizeSetter)
255 .build());
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900256
257 addParameter(
258 DefineParam(mMaxInputSize, C2_PARAMKEY_INPUT_MAX_BUFFER_SIZE)
259 .withDefault(new C2StreamMaxBufferSizeInfo::input(0u, kInputBufferSizeFor1080p))
260 .withFields({
261 C2F(mMaxInputSize, value).any(),
262 })
263 .calculatedAs(MaxInputBufferSizeCalculator, mSize)
264 .build());
265
Chih-Yu Huangbcc6ef62020-05-27 17:52:33 +0900266 bool secureMode = name.find(".secure") != std::string::npos;
267 const C2Allocator::id_t inputAllocators[] = {secureMode ? V4L2AllocatorId::SECURE_LINEAR
Chih-Yu Huang818ead62021-03-25 16:35:14 +0900268 : C2AllocatorStore::DEFAULT_LINEAR};
Chih-Yu Huangbcc6ef62020-05-27 17:52:33 +0900269
Chih-Yu Huang50b5cdf2020-07-22 12:37:00 +0900270 const C2Allocator::id_t outputAllocators[] = {V4L2AllocatorId::V4L2_BUFFERPOOL};
Chih-Yu Huangbcc6ef62020-05-27 17:52:33 +0900271 const C2Allocator::id_t surfaceAllocator =
272 secureMode ? V4L2AllocatorId::SECURE_GRAPHIC : V4L2AllocatorId::V4L2_BUFFERQUEUE;
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900273 const C2BlockPool::local_id_t outputBlockPools[] = {C2BlockPool::BASIC_GRAPHIC};
274
275 addParameter(
276 DefineParam(mInputAllocatorIds, C2_PARAMKEY_INPUT_ALLOCATORS)
277 .withConstValue(C2PortAllocatorsTuning::input::AllocShared(inputAllocators))
278 .build());
279
280 addParameter(
281 DefineParam(mOutputAllocatorIds, C2_PARAMKEY_OUTPUT_ALLOCATORS)
282 .withConstValue(C2PortAllocatorsTuning::output::AllocShared(outputAllocators))
283 .build());
284
285 addParameter(DefineParam(mOutputSurfaceAllocatorId, C2_PARAMKEY_OUTPUT_SURFACE_ALLOCATOR)
286 .withConstValue(new C2PortSurfaceAllocatorTuning::output(surfaceAllocator))
287 .build());
288
289 addParameter(
290 DefineParam(mOutputBlockPoolIds, C2_PARAMKEY_OUTPUT_BLOCK_POOLS)
291 .withDefault(C2PortBlockPoolsTuning::output::AllocShared(outputBlockPools))
292 .withFields({C2F(mOutputBlockPoolIds, m.values[0]).any(),
293 C2F(mOutputBlockPoolIds, m.values).inRange(0, 1)})
294 .withSetter(Setter<C2PortBlockPoolsTuning::output>::NonStrictValuesWithNoDeps)
295 .build());
296
297 addParameter(
298 DefineParam(mDefaultColorAspects, C2_PARAMKEY_DEFAULT_COLOR_ASPECTS)
299 .withDefault(new C2StreamColorAspectsTuning::output(
300 0u, C2Color::RANGE_UNSPECIFIED, C2Color::PRIMARIES_UNSPECIFIED,
301 C2Color::TRANSFER_UNSPECIFIED, C2Color::MATRIX_UNSPECIFIED))
302 .withFields(
303 {C2F(mDefaultColorAspects, range)
304 .inRange(C2Color::RANGE_UNSPECIFIED, C2Color::RANGE_OTHER),
305 C2F(mDefaultColorAspects, primaries)
306 .inRange(C2Color::PRIMARIES_UNSPECIFIED,
307 C2Color::PRIMARIES_OTHER),
308 C2F(mDefaultColorAspects, transfer)
309 .inRange(C2Color::TRANSFER_UNSPECIFIED,
310 C2Color::TRANSFER_OTHER),
311 C2F(mDefaultColorAspects, matrix)
312 .inRange(C2Color::MATRIX_UNSPECIFIED, C2Color::MATRIX_OTHER)})
313 .withSetter(DefaultColorAspectsSetter)
314 .build());
315
316 addParameter(
317 DefineParam(mCodedColorAspects, C2_PARAMKEY_VUI_COLOR_ASPECTS)
318 .withDefault(new C2StreamColorAspectsInfo::input(
319 0u, C2Color::RANGE_LIMITED, C2Color::PRIMARIES_UNSPECIFIED,
320 C2Color::TRANSFER_UNSPECIFIED, C2Color::MATRIX_UNSPECIFIED))
321 .withFields(
322 {C2F(mCodedColorAspects, range)
323 .inRange(C2Color::RANGE_UNSPECIFIED, C2Color::RANGE_OTHER),
324 C2F(mCodedColorAspects, primaries)
325 .inRange(C2Color::PRIMARIES_UNSPECIFIED,
326 C2Color::PRIMARIES_OTHER),
327 C2F(mCodedColorAspects, transfer)
328 .inRange(C2Color::TRANSFER_UNSPECIFIED,
329 C2Color::TRANSFER_OTHER),
330 C2F(mCodedColorAspects, matrix)
331 .inRange(C2Color::MATRIX_UNSPECIFIED, C2Color::MATRIX_OTHER)})
332 .withSetter(DefaultColorAspectsSetter)
333 .build());
334
335 addParameter(
336 DefineParam(mColorAspects, C2_PARAMKEY_COLOR_ASPECTS)
337 .withDefault(new C2StreamColorAspectsInfo::output(
338 0u, C2Color::RANGE_UNSPECIFIED, C2Color::PRIMARIES_UNSPECIFIED,
339 C2Color::TRANSFER_UNSPECIFIED, C2Color::MATRIX_UNSPECIFIED))
340 .withFields(
341 {C2F(mColorAspects, range)
342 .inRange(C2Color::RANGE_UNSPECIFIED, C2Color::RANGE_OTHER),
343 C2F(mColorAspects, primaries)
344 .inRange(C2Color::PRIMARIES_UNSPECIFIED,
345 C2Color::PRIMARIES_OTHER),
346 C2F(mColorAspects, transfer)
347 .inRange(C2Color::TRANSFER_UNSPECIFIED,
348 C2Color::TRANSFER_OTHER),
349 C2F(mColorAspects, matrix)
350 .inRange(C2Color::MATRIX_UNSPECIFIED, C2Color::MATRIX_OTHER)})
351 .withSetter(MergedColorAspectsSetter, mDefaultColorAspects, mCodedColorAspects)
352 .build());
353}
354
355size_t V4L2DecodeInterface::getInputBufferSize() const {
Chih-Yu Huang37ed3062020-12-10 18:07:34 +0900356 return calculateInputBufferSize(mSize->width * mSize->height);
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900357}
358
359c2_status_t V4L2DecodeInterface::queryColorAspects(
360 std::shared_ptr<C2StreamColorAspectsInfo::output>* targetColorAspects) {
361 std::unique_ptr<C2StreamColorAspectsInfo::output> colorAspects =
362 std::make_unique<C2StreamColorAspectsInfo::output>(
363 0u, C2Color::RANGE_UNSPECIFIED, C2Color::PRIMARIES_UNSPECIFIED,
364 C2Color::TRANSFER_UNSPECIFIED, C2Color::MATRIX_UNSPECIFIED);
365 c2_status_t status = query({colorAspects.get()}, {}, C2_DONT_BLOCK, nullptr);
366 if (status == C2_OK) {
367 *targetColorAspects = std::move(colorAspects);
368 }
369 return status;
370}
371
David Stevens79dd1272020-11-06 19:16:39 +0900372uint32_t V4L2DecodeInterface::getOutputDelay(VideoCodec codec) {
373 switch (codec) {
374 case VideoCodec::H264:
375 // Due to frame reordering an H264 decoder might need multiple additional input frames to be
376 // queued before being able to output the associated decoded buffers. We need to tell the
377 // codec2 framework that it should not stop queuing new work items until the maximum number
378 // of frame reordering is reached, to avoid stalling the decoder.
379 return 16;
Jeffrey Kardatzke58a30542022-05-23 14:19:40 -0700380 case VideoCodec::HEVC:
381 return 16;
David Stevens79dd1272020-11-06 19:16:39 +0900382 case VideoCodec::VP8:
383 return 0;
384 case VideoCodec::VP9:
385 return 0;
386 }
387}
388
Chih-Yu Huang6a7255a2020-03-25 17:01:47 +0900389} // namespace android