mirror of
https://github.com/torvalds/linux.git
synced 2024-12-20 18:11:47 +00:00
2719ef7d1b
If the first buffer contains only headers, the sequence initialization
command fails. On CodaHx4 the buffer must be padded to at least 512
bytes, on CODA960 it seems to be enough to just repeat the sequence and
extension headers (MPEG-2) or the VOS and VO headers (MPEG-4) once for
for sequence initialization to succeed without further bitstream data.
On CodaHx4 the headers can be repeated multiple times until the 512 byte
mark is reached.
A similar issue was solved for h.264 by padding with a filler NAL in
commit 0eef89403e
("[media] coda: pad first h.264 buffer to 512
bytes").
Signed-off-by: Philipp Zabel <p.zabel@pengutronix.de>
Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
88 lines
2.2 KiB
C
88 lines
2.2 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Coda multi-standard codec IP - MPEG-2 helper functions
|
|
*
|
|
* Copyright (C) 2019 Pengutronix, Philipp Zabel
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/videodev2.h>
|
|
#include "coda.h"
|
|
|
|
int coda_mpeg2_profile(int profile_idc)
|
|
{
|
|
switch (profile_idc) {
|
|
case 5:
|
|
return V4L2_MPEG_VIDEO_MPEG2_PROFILE_SIMPLE;
|
|
case 4:
|
|
return V4L2_MPEG_VIDEO_MPEG2_PROFILE_MAIN;
|
|
case 3:
|
|
return V4L2_MPEG_VIDEO_MPEG2_PROFILE_SNR_SCALABLE;
|
|
case 2:
|
|
return V4L2_MPEG_VIDEO_MPEG2_PROFILE_SPATIALLY_SCALABLE;
|
|
case 1:
|
|
return V4L2_MPEG_VIDEO_MPEG2_PROFILE_HIGH;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
int coda_mpeg2_level(int level_idc)
|
|
{
|
|
switch (level_idc) {
|
|
case 10:
|
|
return V4L2_MPEG_VIDEO_MPEG2_LEVEL_LOW;
|
|
case 8:
|
|
return V4L2_MPEG_VIDEO_MPEG2_LEVEL_MAIN;
|
|
case 6:
|
|
return V4L2_MPEG_VIDEO_MPEG2_LEVEL_HIGH_1440;
|
|
case 4:
|
|
return V4L2_MPEG_VIDEO_MPEG2_LEVEL_HIGH;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check if the buffer starts with the MPEG-2 sequence header (with or without
|
|
* quantization matrix) and extension header, for example:
|
|
*
|
|
* 00 00 01 b3 2d 01 e0 34 08 8b a3 81
|
|
* 10 11 11 12 12 12 13 13 13 13 14 14 14 14 14 15
|
|
* 15 15 15 15 15 16 16 16 16 16 16 16 17 17 17 17
|
|
* 17 17 17 17 18 18 18 19 18 18 18 19 1a 1a 1a 1a
|
|
* 19 1b 1b 1b 1b 1b 1c 1c 1c 1c 1e 1e 1e 1f 1f 21
|
|
* 00 00 01 b5 14 8a 00 01 00 00
|
|
*
|
|
* or:
|
|
*
|
|
* 00 00 01 b3 08 00 40 15 ff ff e0 28
|
|
* 00 00 01 b5 14 8a 00 01 00 00
|
|
*
|
|
* Returns the detected header size in bytes or 0.
|
|
*/
|
|
u32 coda_mpeg2_parse_headers(struct coda_ctx *ctx, u8 *buf, u32 size)
|
|
{
|
|
static const u8 sequence_header_start[4] = { 0x00, 0x00, 0x01, 0xb3 };
|
|
static const union {
|
|
u8 extension_start[4];
|
|
u8 start_code_prefix[3];
|
|
} u = { { 0x00, 0x00, 0x01, 0xb5 } };
|
|
|
|
if (size < 22 ||
|
|
memcmp(buf, sequence_header_start, 4) != 0)
|
|
return 0;
|
|
|
|
if ((size == 22 ||
|
|
(size >= 25 && memcmp(buf + 22, u.start_code_prefix, 3) == 0)) &&
|
|
memcmp(buf + 12, u.extension_start, 4) == 0)
|
|
return 22;
|
|
|
|
if ((size == 86 ||
|
|
(size > 89 && memcmp(buf + 86, u.start_code_prefix, 3) == 0)) &&
|
|
memcmp(buf + 76, u.extension_start, 4) == 0)
|
|
return 86;
|
|
|
|
return 0;
|
|
}
|