mirror of
https://github.com/torvalds/linux.git
synced 2026-05-13 00:28:54 +02:00
media: iris: Initialize and deinitialize encoder instance structure
Introduce initialization and deinitialization for internal encoder instance structure with necessary hooks. Tested-by: Vikash Garodia <quic_vgarodia@quicinc.com> # X1E80100 Reviewed-by: Vikash Garodia <quic_vgarodia@quicinc.com> Reviewed-by: Bryan O'Donoghue <bryan.odonoghue@linaro.org> Tested-by: Neil Armstrong <neil.armstrong@linaro.org> # on SM8550-HDK Tested-by: Neil Armstrong <neil.armstrong@linaro.org> # on SM8650-HDK Signed-off-by: Dikshita Agarwal <quic_dikshita@quicinc.com> Tested-by: Bryan O'Donoghue <bryan.odonoghue@linaro.org> # x1e80100-crd Signed-off-by: Bryan O'Donoghue <bod@kernel.org> Signed-off-by: Hans Verkuil <hverkuil+cisco@kernel.org>
This commit is contained in:
parent
787c535a9f
commit
5ad964ad56
|
|
@ -19,6 +19,7 @@ qcom-iris-objs += \
|
|||
iris_vidc.o \
|
||||
iris_vb2.o \
|
||||
iris_vdec.o \
|
||||
iris_venc.o \
|
||||
iris_vpu2.o \
|
||||
iris_vpu3x.o \
|
||||
iris_vpu_buffer.o \
|
||||
|
|
|
|||
|
|
@ -63,7 +63,12 @@
|
|||
static u32 iris_yuv_buffer_size_nv12(struct iris_inst *inst)
|
||||
{
|
||||
u32 y_plane, uv_plane, y_stride, uv_stride, y_scanlines, uv_scanlines;
|
||||
struct v4l2_format *f = inst->fmt_dst;
|
||||
struct v4l2_format *f;
|
||||
|
||||
if (inst->domain == DECODER)
|
||||
f = inst->fmt_dst;
|
||||
else
|
||||
f = inst->fmt_src;
|
||||
|
||||
y_stride = ALIGN(f->fmt.pix_mp.width, Y_STRIDE_ALIGN);
|
||||
uv_stride = ALIGN(f->fmt.pix_mp.width, UV_STRIDE_ALIGN);
|
||||
|
|
@ -194,7 +199,7 @@ static u32 iris_yuv_buffer_size_qc08c(struct iris_inst *inst)
|
|||
return ALIGN(y_meta_plane + y_plane + uv_meta_plane + uv_plane, PIXELS_4K);
|
||||
}
|
||||
|
||||
static u32 iris_bitstream_buffer_size(struct iris_inst *inst)
|
||||
static u32 iris_dec_bitstream_buffer_size(struct iris_inst *inst)
|
||||
{
|
||||
struct platform_inst_caps *caps = inst->core->iris_platform_data->inst_caps;
|
||||
u32 base_res_mbs = NUM_MBS_4K;
|
||||
|
|
@ -219,18 +224,50 @@ static u32 iris_bitstream_buffer_size(struct iris_inst *inst)
|
|||
return ALIGN(frame_size, PIXELS_4K);
|
||||
}
|
||||
|
||||
static u32 iris_enc_bitstream_buffer_size(struct iris_inst *inst)
|
||||
{
|
||||
u32 aligned_width, aligned_height, bitstream_size, yuv_size;
|
||||
struct v4l2_format *f;
|
||||
|
||||
f = inst->fmt_dst;
|
||||
|
||||
aligned_width = ALIGN(f->fmt.pix_mp.width, 32);
|
||||
aligned_height = ALIGN(f->fmt.pix_mp.height, 32);
|
||||
bitstream_size = aligned_width * aligned_height * 3;
|
||||
yuv_size = (aligned_width * aligned_height * 3) >> 1;
|
||||
if (aligned_width * aligned_height > (4096 * 2176))
|
||||
/* bitstream_size = 0.25 * yuv_size; */
|
||||
bitstream_size = (bitstream_size >> 3);
|
||||
else if (aligned_width * aligned_height > (1280 * 720))
|
||||
/* bitstream_size = 0.5 * yuv_size; */
|
||||
bitstream_size = (bitstream_size >> 2);
|
||||
|
||||
return ALIGN(bitstream_size, 4096);
|
||||
}
|
||||
|
||||
int iris_get_buffer_size(struct iris_inst *inst,
|
||||
enum iris_buffer_type buffer_type)
|
||||
{
|
||||
switch (buffer_type) {
|
||||
case BUF_INPUT:
|
||||
return iris_bitstream_buffer_size(inst);
|
||||
case BUF_OUTPUT:
|
||||
return iris_yuv_buffer_size_nv12(inst);
|
||||
case BUF_DPB:
|
||||
return iris_yuv_buffer_size_qc08c(inst);
|
||||
default:
|
||||
return 0;
|
||||
if (inst->domain == DECODER) {
|
||||
switch (buffer_type) {
|
||||
case BUF_INPUT:
|
||||
return iris_dec_bitstream_buffer_size(inst);
|
||||
case BUF_OUTPUT:
|
||||
return iris_yuv_buffer_size_nv12(inst);
|
||||
case BUF_DPB:
|
||||
return iris_yuv_buffer_size_qc08c(inst);
|
||||
default:
|
||||
return 0;
|
||||
}
|
||||
} else {
|
||||
switch (buffer_type) {
|
||||
case BUF_INPUT:
|
||||
return iris_yuv_buffer_size_nv12(inst);
|
||||
case BUF_OUTPUT:
|
||||
return iris_enc_bitstream_buffer_size(inst);
|
||||
default:
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -109,7 +109,12 @@ static int iris_hfi_gen1_session_open(struct iris_inst *inst)
|
|||
packet.shdr.hdr.size = sizeof(struct hfi_session_open_pkt);
|
||||
packet.shdr.hdr.pkt_type = HFI_CMD_SYS_SESSION_INIT;
|
||||
packet.shdr.session_id = inst->session_id;
|
||||
packet.session_domain = HFI_SESSION_TYPE_DEC;
|
||||
|
||||
if (inst->domain == DECODER)
|
||||
packet.session_domain = HFI_SESSION_TYPE_DEC;
|
||||
else
|
||||
packet.session_domain = HFI_SESSION_TYPE_ENC;
|
||||
|
||||
packet.session_codec = codec;
|
||||
|
||||
reinit_completion(&inst->completion);
|
||||
|
|
|
|||
|
|
@ -10,6 +10,7 @@
|
|||
|
||||
#define HFI_VIDEO_ARCH_OX 0x1
|
||||
|
||||
#define HFI_SESSION_TYPE_ENC 1
|
||||
#define HFI_SESSION_TYPE_DEC 2
|
||||
|
||||
#define HFI_VIDEO_CODEC_H264 0x00000002
|
||||
|
|
|
|||
|
|
@ -12,6 +12,9 @@
|
|||
#include "iris_core.h"
|
||||
#include "iris_utils.h"
|
||||
|
||||
#define DEFAULT_WIDTH 320
|
||||
#define DEFAULT_HEIGHT 240
|
||||
|
||||
/**
|
||||
* struct iris_inst - holds per video instance parameters
|
||||
*
|
||||
|
|
@ -24,7 +27,9 @@
|
|||
* @fmt_src: structure of v4l2_format for source
|
||||
* @fmt_dst: structure of v4l2_format for destination
|
||||
* @ctrl_handler: reference of v4l2 ctrl handler
|
||||
* @domain: domain type: encoder or decoder
|
||||
* @crop: structure of crop info
|
||||
* @compose: structure of compose info
|
||||
* @completion: structure of signal completions
|
||||
* @flush_completion: structure of signal completions for flush cmd
|
||||
* @flush_responses_pending: counter to track number of pending flush responses
|
||||
|
|
@ -57,7 +62,9 @@ struct iris_inst {
|
|||
struct v4l2_format *fmt_src;
|
||||
struct v4l2_format *fmt_dst;
|
||||
struct v4l2_ctrl_handler ctrl_handler;
|
||||
enum domain_type domain;
|
||||
struct iris_hfi_rect_desc crop;
|
||||
struct iris_hfi_rect_desc compose;
|
||||
struct completion completion;
|
||||
struct completion flush_completion;
|
||||
u32 flush_responses_pending;
|
||||
|
|
|
|||
|
|
@ -13,8 +13,6 @@
|
|||
#include "iris_vdec.h"
|
||||
#include "iris_vpu_buffer.h"
|
||||
|
||||
#define DEFAULT_WIDTH 320
|
||||
#define DEFAULT_HEIGHT 240
|
||||
#define DEFAULT_CODEC_ALIGNMENT 16
|
||||
|
||||
int iris_vdec_inst_init(struct iris_inst *inst)
|
||||
|
|
|
|||
65
drivers/media/platform/qcom/iris/iris_venc.c
Normal file
65
drivers/media/platform/qcom/iris/iris_venc.c
Normal file
|
|
@ -0,0 +1,65 @@
|
|||
// SPDX-License-Identifier: GPL-2.0-only
|
||||
/*
|
||||
* Copyright (c) 2022-2025 Qualcomm Innovation Center, Inc. All rights reserved.
|
||||
*/
|
||||
|
||||
#include "iris_buffer.h"
|
||||
#include "iris_instance.h"
|
||||
#include "iris_venc.h"
|
||||
#include "iris_vpu_buffer.h"
|
||||
|
||||
int iris_venc_inst_init(struct iris_inst *inst)
|
||||
{
|
||||
struct v4l2_format *f;
|
||||
|
||||
inst->fmt_src = kzalloc(sizeof(*inst->fmt_src), GFP_KERNEL);
|
||||
inst->fmt_dst = kzalloc(sizeof(*inst->fmt_dst), GFP_KERNEL);
|
||||
if (!inst->fmt_src || !inst->fmt_dst)
|
||||
return -ENOMEM;
|
||||
|
||||
f = inst->fmt_dst;
|
||||
f->type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
|
||||
f->fmt.pix_mp.width = DEFAULT_WIDTH;
|
||||
f->fmt.pix_mp.height = DEFAULT_HEIGHT;
|
||||
f->fmt.pix_mp.pixelformat = V4L2_PIX_FMT_H264;
|
||||
inst->codec = f->fmt.pix_mp.pixelformat;
|
||||
f->fmt.pix_mp.num_planes = 1;
|
||||
f->fmt.pix_mp.plane_fmt[0].bytesperline = 0;
|
||||
f->fmt.pix_mp.plane_fmt[0].sizeimage = iris_get_buffer_size(inst, BUF_OUTPUT);
|
||||
f->fmt.pix_mp.field = V4L2_FIELD_NONE;
|
||||
f->fmt.pix_mp.colorspace = V4L2_COLORSPACE_DEFAULT;
|
||||
f->fmt.pix_mp.xfer_func = V4L2_XFER_FUNC_DEFAULT;
|
||||
f->fmt.pix_mp.ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT;
|
||||
f->fmt.pix_mp.quantization = V4L2_QUANTIZATION_DEFAULT;
|
||||
inst->buffers[BUF_OUTPUT].min_count = iris_vpu_buf_count(inst, BUF_OUTPUT);
|
||||
inst->buffers[BUF_OUTPUT].size = f->fmt.pix_mp.plane_fmt[0].sizeimage;
|
||||
|
||||
f = inst->fmt_src;
|
||||
f->type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
|
||||
f->fmt.pix_mp.pixelformat = V4L2_PIX_FMT_NV12;
|
||||
f->fmt.pix_mp.width = ALIGN(DEFAULT_WIDTH, 128);
|
||||
f->fmt.pix_mp.height = ALIGN(DEFAULT_HEIGHT, 32);
|
||||
f->fmt.pix_mp.num_planes = 1;
|
||||
f->fmt.pix_mp.plane_fmt[0].bytesperline = ALIGN(DEFAULT_WIDTH, 128);
|
||||
f->fmt.pix_mp.plane_fmt[0].sizeimage = iris_get_buffer_size(inst, BUF_INPUT);
|
||||
f->fmt.pix_mp.field = V4L2_FIELD_NONE;
|
||||
f->fmt.pix_mp.colorspace = V4L2_COLORSPACE_DEFAULT;
|
||||
f->fmt.pix_mp.xfer_func = V4L2_XFER_FUNC_DEFAULT;
|
||||
f->fmt.pix_mp.ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT;
|
||||
f->fmt.pix_mp.quantization = V4L2_QUANTIZATION_DEFAULT;
|
||||
inst->buffers[BUF_INPUT].min_count = iris_vpu_buf_count(inst, BUF_INPUT);
|
||||
inst->buffers[BUF_INPUT].size = f->fmt.pix_mp.plane_fmt[0].sizeimage;
|
||||
|
||||
inst->crop.left = 0;
|
||||
inst->crop.top = 0;
|
||||
inst->crop.width = f->fmt.pix_mp.width;
|
||||
inst->crop.height = f->fmt.pix_mp.height;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void iris_venc_inst_deinit(struct iris_inst *inst)
|
||||
{
|
||||
kfree(inst->fmt_dst);
|
||||
kfree(inst->fmt_src);
|
||||
}
|
||||
14
drivers/media/platform/qcom/iris/iris_venc.h
Normal file
14
drivers/media/platform/qcom/iris/iris_venc.h
Normal file
|
|
@ -0,0 +1,14 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/*
|
||||
* Copyright (c) 2022-2025 Qualcomm Innovation Center, Inc. All rights reserved.
|
||||
*/
|
||||
|
||||
#ifndef _IRIS_VENC_H_
|
||||
#define _IRIS_VENC_H_
|
||||
|
||||
struct iris_inst;
|
||||
|
||||
int iris_venc_inst_init(struct iris_inst *inst);
|
||||
void iris_venc_inst_deinit(struct iris_inst *inst);
|
||||
|
||||
#endif
|
||||
|
|
@ -12,6 +12,7 @@
|
|||
#include "iris_vidc.h"
|
||||
#include "iris_instance.h"
|
||||
#include "iris_vdec.h"
|
||||
#include "iris_venc.h"
|
||||
#include "iris_vb2.h"
|
||||
#include "iris_vpu_buffer.h"
|
||||
#include "iris_platform_common.h"
|
||||
|
|
@ -23,7 +24,10 @@
|
|||
|
||||
static void iris_v4l2_fh_init(struct iris_inst *inst, struct file *filp)
|
||||
{
|
||||
v4l2_fh_init(&inst->fh, inst->core->vdev_dec);
|
||||
if (inst->domain == ENCODER)
|
||||
v4l2_fh_init(&inst->fh, inst->core->vdev_enc);
|
||||
else if (inst->domain == DECODER)
|
||||
v4l2_fh_init(&inst->fh, inst->core->vdev_dec);
|
||||
inst->fh.ctrl_handler = &inst->ctrl_handler;
|
||||
v4l2_fh_add(&inst->fh, filp);
|
||||
}
|
||||
|
|
@ -126,9 +130,19 @@ iris_m2m_queue_init(void *priv, struct vb2_queue *src_vq, struct vb2_queue *dst_
|
|||
int iris_open(struct file *filp)
|
||||
{
|
||||
struct iris_core *core = video_drvdata(filp);
|
||||
struct video_device *vdev;
|
||||
struct iris_inst *inst;
|
||||
u32 session_type;
|
||||
int ret;
|
||||
|
||||
vdev = video_devdata(filp);
|
||||
if (strcmp(vdev->name, "qcom-iris-decoder") == 0)
|
||||
session_type = DECODER;
|
||||
else if (strcmp(vdev->name, "qcom-iris-encoder") == 0)
|
||||
session_type = ENCODER;
|
||||
else
|
||||
return -EINVAL;
|
||||
|
||||
ret = pm_runtime_resume_and_get(core->dev);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
|
@ -147,6 +161,7 @@ int iris_open(struct file *filp)
|
|||
return -ENOMEM;
|
||||
|
||||
inst->core = core;
|
||||
inst->domain = session_type;
|
||||
inst->session_id = hash32_ptr(inst);
|
||||
inst->state = IRIS_INST_DEINIT;
|
||||
|
||||
|
|
@ -178,7 +193,10 @@ int iris_open(struct file *filp)
|
|||
goto fail_m2m_release;
|
||||
}
|
||||
|
||||
ret = iris_vdec_inst_init(inst);
|
||||
if (inst->domain == DECODER)
|
||||
ret = iris_vdec_inst_init(inst);
|
||||
else if (inst->domain == ENCODER)
|
||||
ret = iris_venc_inst_init(inst);
|
||||
if (ret)
|
||||
goto fail_m2m_ctx_release;
|
||||
|
||||
|
|
@ -264,7 +282,10 @@ int iris_close(struct file *filp)
|
|||
v4l2_m2m_ctx_release(inst->m2m_ctx);
|
||||
v4l2_m2m_release(inst->m2m_dev);
|
||||
mutex_lock(&inst->lock);
|
||||
iris_vdec_inst_deinit(inst);
|
||||
if (inst->domain == DECODER)
|
||||
iris_vdec_inst_deinit(inst);
|
||||
else if (inst->domain == ENCODER)
|
||||
iris_venc_inst_deinit(inst);
|
||||
iris_session_close(inst);
|
||||
iris_inst_change_state(inst, IRIS_INST_DEINIT);
|
||||
iris_v4l2_fh_deinit(inst, filp);
|
||||
|
|
|
|||
|
|
@ -628,7 +628,10 @@ int iris_vpu_buf_count(struct iris_inst *inst, enum iris_buffer_type buffer_type
|
|||
case BUF_INPUT:
|
||||
return MIN_BUFFERS;
|
||||
case BUF_OUTPUT:
|
||||
return output_min_count(inst);
|
||||
if (inst->domain == ENCODER)
|
||||
return MIN_BUFFERS;
|
||||
else
|
||||
return output_min_count(inst);
|
||||
case BUF_BIN:
|
||||
case BUF_COMV:
|
||||
case BUF_NON_COMV:
|
||||
|
|
|
|||
Loading…
Reference in New Issue
Block a user