162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Coda multi-standard codec IP - MPEG-2 helper functions
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2019 Pengutronix, Philipp Zabel
662306a36Sopenharmony_ci */
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#include <linux/kernel.h>
962306a36Sopenharmony_ci#include <linux/videodev2.h>
1062306a36Sopenharmony_ci#include "coda.h"
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ciint coda_mpeg2_profile(int profile_idc)
1362306a36Sopenharmony_ci{
1462306a36Sopenharmony_ci	switch (profile_idc) {
1562306a36Sopenharmony_ci	case 5:
1662306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_PROFILE_SIMPLE;
1762306a36Sopenharmony_ci	case 4:
1862306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_PROFILE_MAIN;
1962306a36Sopenharmony_ci	case 3:
2062306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_PROFILE_SNR_SCALABLE;
2162306a36Sopenharmony_ci	case 2:
2262306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_PROFILE_SPATIALLY_SCALABLE;
2362306a36Sopenharmony_ci	case 1:
2462306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_PROFILE_HIGH;
2562306a36Sopenharmony_ci	default:
2662306a36Sopenharmony_ci		return -EINVAL;
2762306a36Sopenharmony_ci	}
2862306a36Sopenharmony_ci}
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ciint coda_mpeg2_level(int level_idc)
3162306a36Sopenharmony_ci{
3262306a36Sopenharmony_ci	switch (level_idc) {
3362306a36Sopenharmony_ci	case 10:
3462306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_LEVEL_LOW;
3562306a36Sopenharmony_ci	case 8:
3662306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_LEVEL_MAIN;
3762306a36Sopenharmony_ci	case 6:
3862306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_LEVEL_HIGH_1440;
3962306a36Sopenharmony_ci	case 4:
4062306a36Sopenharmony_ci		return V4L2_MPEG_VIDEO_MPEG2_LEVEL_HIGH;
4162306a36Sopenharmony_ci	default:
4262306a36Sopenharmony_ci		return -EINVAL;
4362306a36Sopenharmony_ci	}
4462306a36Sopenharmony_ci}
4562306a36Sopenharmony_ci
4662306a36Sopenharmony_ci/*
4762306a36Sopenharmony_ci * Check if the buffer starts with the MPEG-2 sequence header (with or without
4862306a36Sopenharmony_ci * quantization matrix) and extension header, for example:
4962306a36Sopenharmony_ci *
5062306a36Sopenharmony_ci *   00 00 01 b3 2d 01 e0 34 08 8b a3 81
5162306a36Sopenharmony_ci *               10 11 11 12 12 12 13 13 13 13 14 14 14 14 14 15
5262306a36Sopenharmony_ci *               15 15 15 15 15 16 16 16 16 16 16 16 17 17 17 17
5362306a36Sopenharmony_ci *               17 17 17 17 18 18 18 19 18 18 18 19 1a 1a 1a 1a
5462306a36Sopenharmony_ci *               19 1b 1b 1b 1b 1b 1c 1c 1c 1c 1e 1e 1e 1f 1f 21
5562306a36Sopenharmony_ci *   00 00 01 b5 14 8a 00 01 00 00
5662306a36Sopenharmony_ci *
5762306a36Sopenharmony_ci * or:
5862306a36Sopenharmony_ci *
5962306a36Sopenharmony_ci *   00 00 01 b3 08 00 40 15 ff ff e0 28
6062306a36Sopenharmony_ci *   00 00 01 b5 14 8a 00 01 00 00
6162306a36Sopenharmony_ci *
6262306a36Sopenharmony_ci * Returns the detected header size in bytes or 0.
6362306a36Sopenharmony_ci */
6462306a36Sopenharmony_ciu32 coda_mpeg2_parse_headers(struct coda_ctx *ctx, u8 *buf, u32 size)
6562306a36Sopenharmony_ci{
6662306a36Sopenharmony_ci	static const u8 sequence_header_start[4] = { 0x00, 0x00, 0x01, 0xb3 };
6762306a36Sopenharmony_ci	static const union {
6862306a36Sopenharmony_ci		u8 extension_start[4];
6962306a36Sopenharmony_ci		u8 start_code_prefix[3];
7062306a36Sopenharmony_ci	} u = { { 0x00, 0x00, 0x01, 0xb5 } };
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	if (size < 22 ||
7362306a36Sopenharmony_ci	    memcmp(buf, sequence_header_start, 4) != 0)
7462306a36Sopenharmony_ci		return 0;
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_ci	if ((size == 22 ||
7762306a36Sopenharmony_ci	     (size >= 25 && memcmp(buf + 22, u.start_code_prefix, 3) == 0)) &&
7862306a36Sopenharmony_ci	    memcmp(buf + 12, u.extension_start, 4) == 0)
7962306a36Sopenharmony_ci		return 22;
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	if ((size == 86 ||
8262306a36Sopenharmony_ci	     (size > 89 && memcmp(buf + 86, u.start_code_prefix, 3) == 0)) &&
8362306a36Sopenharmony_ci	    memcmp(buf + 76, u.extension_start, 4) == 0)
8462306a36Sopenharmony_ci		return 86;
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_ci	return 0;
8762306a36Sopenharmony_ci}
88