Presently the hfi_parser has been called only once during driver probe. To prepare the parser function to be called multiple times from recovery we need to initialize few variables which are used during parsing time. Signed-off-by: Stanimir Varbanov <stanimir.varbanov@linaro.org> Reviewed-by: Fritz Koenig <frkoenig@chromium.org> Signed-off-by: Mauro Carvalho Chehab <mchehab+huawei@kernel.org>
283 lines
6.8 KiB
C
283 lines
6.8 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (C) 2018 Linaro Ltd.
|
|
*
|
|
* Author: Stanimir Varbanov <stanimir.varbanov@linaro.org>
|
|
*/
|
|
#include <linux/bitops.h>
|
|
#include <linux/kernel.h>
|
|
|
|
#include "core.h"
|
|
#include "hfi_helper.h"
|
|
#include "hfi_parser.h"
|
|
|
|
typedef void (*func)(struct venus_caps *cap, const void *data,
|
|
unsigned int size);
|
|
|
|
static void init_codecs(struct venus_core *core)
|
|
{
|
|
struct venus_caps *caps = core->caps, *cap;
|
|
unsigned long bit;
|
|
|
|
for_each_set_bit(bit, &core->dec_codecs, MAX_CODEC_NUM) {
|
|
cap = &caps[core->codecs_count++];
|
|
cap->codec = BIT(bit);
|
|
cap->domain = VIDC_SESSION_TYPE_DEC;
|
|
cap->valid = false;
|
|
}
|
|
|
|
for_each_set_bit(bit, &core->enc_codecs, MAX_CODEC_NUM) {
|
|
cap = &caps[core->codecs_count++];
|
|
cap->codec = BIT(bit);
|
|
cap->domain = VIDC_SESSION_TYPE_ENC;
|
|
cap->valid = false;
|
|
}
|
|
}
|
|
|
|
static void for_each_codec(struct venus_caps *caps, unsigned int caps_num,
|
|
u32 codecs, u32 domain, func cb, void *data,
|
|
unsigned int size)
|
|
{
|
|
struct venus_caps *cap;
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < caps_num; i++) {
|
|
cap = &caps[i];
|
|
if (cap->valid && cap->domain == domain)
|
|
continue;
|
|
if (cap->codec & codecs && cap->domain == domain)
|
|
cb(cap, data, size);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fill_buf_mode(struct venus_caps *cap, const void *data, unsigned int num)
|
|
{
|
|
const u32 *type = data;
|
|
|
|
if (*type == HFI_BUFFER_MODE_DYNAMIC)
|
|
cap->cap_bufs_mode_dynamic = true;
|
|
}
|
|
|
|
static void
|
|
parse_alloc_mode(struct venus_core *core, u32 codecs, u32 domain, void *data)
|
|
{
|
|
struct hfi_buffer_alloc_mode_supported *mode = data;
|
|
u32 num_entries = mode->num_entries;
|
|
u32 *type;
|
|
|
|
if (num_entries > MAX_ALLOC_MODE_ENTRIES)
|
|
return;
|
|
|
|
type = mode->data;
|
|
|
|
while (num_entries--) {
|
|
if (mode->buffer_type == HFI_BUFFER_OUTPUT ||
|
|
mode->buffer_type == HFI_BUFFER_OUTPUT2)
|
|
for_each_codec(core->caps, ARRAY_SIZE(core->caps),
|
|
codecs, domain, fill_buf_mode, type, 1);
|
|
|
|
type++;
|
|
}
|
|
}
|
|
|
|
static void fill_profile_level(struct venus_caps *cap, const void *data,
|
|
unsigned int num)
|
|
{
|
|
const struct hfi_profile_level *pl = data;
|
|
|
|
memcpy(&cap->pl[cap->num_pl], pl, num * sizeof(*pl));
|
|
cap->num_pl += num;
|
|
}
|
|
|
|
static void
|
|
parse_profile_level(struct venus_core *core, u32 codecs, u32 domain, void *data)
|
|
{
|
|
struct hfi_profile_level_supported *pl = data;
|
|
struct hfi_profile_level *proflevel = pl->profile_level;
|
|
struct hfi_profile_level pl_arr[HFI_MAX_PROFILE_COUNT] = {};
|
|
|
|
if (pl->profile_count > HFI_MAX_PROFILE_COUNT)
|
|
return;
|
|
|
|
memcpy(pl_arr, proflevel, pl->profile_count * sizeof(*proflevel));
|
|
|
|
for_each_codec(core->caps, ARRAY_SIZE(core->caps), codecs, domain,
|
|
fill_profile_level, pl_arr, pl->profile_count);
|
|
}
|
|
|
|
static void
|
|
fill_caps(struct venus_caps *cap, const void *data, unsigned int num)
|
|
{
|
|
const struct hfi_capability *caps = data;
|
|
|
|
memcpy(&cap->caps[cap->num_caps], caps, num * sizeof(*caps));
|
|
cap->num_caps += num;
|
|
}
|
|
|
|
static void
|
|
parse_caps(struct venus_core *core, u32 codecs, u32 domain, void *data)
|
|
{
|
|
struct hfi_capabilities *caps = data;
|
|
struct hfi_capability *cap = caps->data;
|
|
u32 num_caps = caps->num_capabilities;
|
|
struct hfi_capability caps_arr[MAX_CAP_ENTRIES] = {};
|
|
|
|
if (num_caps > MAX_CAP_ENTRIES)
|
|
return;
|
|
|
|
memcpy(caps_arr, cap, num_caps * sizeof(*cap));
|
|
|
|
for_each_codec(core->caps, ARRAY_SIZE(core->caps), codecs, domain,
|
|
fill_caps, caps_arr, num_caps);
|
|
}
|
|
|
|
static void fill_raw_fmts(struct venus_caps *cap, const void *fmts,
|
|
unsigned int num_fmts)
|
|
{
|
|
const struct raw_formats *formats = fmts;
|
|
|
|
memcpy(&cap->fmts[cap->num_fmts], formats, num_fmts * sizeof(*formats));
|
|
cap->num_fmts += num_fmts;
|
|
}
|
|
|
|
static void
|
|
parse_raw_formats(struct venus_core *core, u32 codecs, u32 domain, void *data)
|
|
{
|
|
struct hfi_uncompressed_format_supported *fmt = data;
|
|
struct hfi_uncompressed_plane_info *pinfo = fmt->plane_info;
|
|
struct hfi_uncompressed_plane_constraints *constr;
|
|
struct raw_formats rawfmts[MAX_FMT_ENTRIES] = {};
|
|
u32 entries = fmt->format_entries;
|
|
unsigned int i = 0;
|
|
u32 num_planes;
|
|
|
|
while (entries) {
|
|
num_planes = pinfo->num_planes;
|
|
|
|
rawfmts[i].fmt = pinfo->format;
|
|
rawfmts[i].buftype = fmt->buffer_type;
|
|
i++;
|
|
|
|
if (pinfo->num_planes > MAX_PLANES)
|
|
break;
|
|
|
|
pinfo = (void *)pinfo + sizeof(*constr) * num_planes +
|
|
2 * sizeof(u32);
|
|
entries--;
|
|
}
|
|
|
|
for_each_codec(core->caps, ARRAY_SIZE(core->caps), codecs, domain,
|
|
fill_raw_fmts, rawfmts, i);
|
|
}
|
|
|
|
static void parse_codecs(struct venus_core *core, void *data)
|
|
{
|
|
struct hfi_codec_supported *codecs = data;
|
|
|
|
core->dec_codecs = codecs->dec_codecs;
|
|
core->enc_codecs = codecs->enc_codecs;
|
|
|
|
if (IS_V1(core)) {
|
|
core->dec_codecs &= ~HFI_VIDEO_CODEC_HEVC;
|
|
core->dec_codecs &= ~HFI_VIDEO_CODEC_SPARK;
|
|
core->enc_codecs &= ~HFI_VIDEO_CODEC_HEVC;
|
|
}
|
|
}
|
|
|
|
static void parse_max_sessions(struct venus_core *core, const void *data)
|
|
{
|
|
const struct hfi_max_sessions_supported *sessions = data;
|
|
|
|
core->max_sessions_supported = sessions->max_sessions;
|
|
}
|
|
|
|
static void parse_codecs_mask(u32 *codecs, u32 *domain, void *data)
|
|
{
|
|
struct hfi_codec_mask_supported *mask = data;
|
|
|
|
*codecs = mask->codecs;
|
|
*domain = mask->video_domains;
|
|
}
|
|
|
|
static void parser_init(struct venus_inst *inst, u32 *codecs, u32 *domain)
|
|
{
|
|
if (!inst || !IS_V1(inst->core))
|
|
return;
|
|
|
|
*codecs = inst->hfi_codec;
|
|
*domain = inst->session_type;
|
|
}
|
|
|
|
static void parser_fini(struct venus_inst *inst, u32 codecs, u32 domain)
|
|
{
|
|
struct venus_caps *caps, *cap;
|
|
unsigned int i;
|
|
u32 dom;
|
|
|
|
if (!inst || !IS_V1(inst->core))
|
|
return;
|
|
|
|
caps = inst->core->caps;
|
|
dom = inst->session_type;
|
|
|
|
for (i = 0; i < MAX_CODEC_NUM; i++) {
|
|
cap = &caps[i];
|
|
if (cap->codec & codecs && cap->domain == dom)
|
|
cap->valid = true;
|
|
}
|
|
}
|
|
|
|
u32 hfi_parser(struct venus_core *core, struct venus_inst *inst, void *buf,
|
|
u32 size)
|
|
{
|
|
unsigned int words_count = size >> 2;
|
|
u32 *word = buf, *data, codecs = 0, domain = 0;
|
|
|
|
if (size % 4)
|
|
return HFI_ERR_SYS_INSUFFICIENT_RESOURCES;
|
|
|
|
parser_init(inst, &codecs, &domain);
|
|
|
|
core->codecs_count = 0;
|
|
memset(core->caps, 0, sizeof(core->caps));
|
|
|
|
while (words_count) {
|
|
data = word + 1;
|
|
|
|
switch (*word) {
|
|
case HFI_PROPERTY_PARAM_CODEC_SUPPORTED:
|
|
parse_codecs(core, data);
|
|
init_codecs(core);
|
|
break;
|
|
case HFI_PROPERTY_PARAM_MAX_SESSIONS_SUPPORTED:
|
|
parse_max_sessions(core, data);
|
|
break;
|
|
case HFI_PROPERTY_PARAM_CODEC_MASK_SUPPORTED:
|
|
parse_codecs_mask(&codecs, &domain, data);
|
|
break;
|
|
case HFI_PROPERTY_PARAM_UNCOMPRESSED_FORMAT_SUPPORTED:
|
|
parse_raw_formats(core, codecs, domain, data);
|
|
break;
|
|
case HFI_PROPERTY_PARAM_CAPABILITY_SUPPORTED:
|
|
parse_caps(core, codecs, domain, data);
|
|
break;
|
|
case HFI_PROPERTY_PARAM_PROFILE_LEVEL_SUPPORTED:
|
|
parse_profile_level(core, codecs, domain, data);
|
|
break;
|
|
case HFI_PROPERTY_PARAM_BUFFER_ALLOC_MODE_SUPPORTED:
|
|
parse_alloc_mode(core, codecs, domain, data);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
word++;
|
|
words_count--;
|
|
}
|
|
|
|
parser_fini(inst, codecs, domain);
|
|
|
|
return HFI_ERR_NONE;
|
|
}
|