mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-21 06:50:25 +00:00
media: iris: implement s_fmt, g_fmt and try_fmt ioctls
Implement the s_fmt, g_fmt and try_fmt ioctl ops with the necessary hooks. Signed-off-by: Vedang Nagar <quic_vnagar@quicinc.com> Reviewed-by: Hans Verkuil <hverkuil@xs4all.nl> Tested-by: Stefan Schmidt <stefan.schmidt@linaro.org> # x1e80100 (Dell XPS 13 9345) Reviewed-by: Stefan Schmidt <stefan.schmidt@linaro.org> Tested-by: Neil Armstrong <neil.armstrong@linaro.org> # on SM8550-QRD Tested-by: Neil Armstrong <neil.armstrong@linaro.org> # on SM8550-HDK Signed-off-by: Dikshita Agarwal <quic_dikshita@quicinc.com> Signed-off-by: Hans Verkuil <hverkuil@xs4all.nl>
This commit is contained in:
parent
38fc8beaba
commit
b530b95de2
3 changed files with 172 additions and 0 deletions
|
@ -3,6 +3,8 @@
|
||||||
* Copyright (c) 2022-2024 Qualcomm Innovation Center, Inc. All rights reserved.
|
* Copyright (c) 2022-2024 Qualcomm Innovation Center, Inc. All rights reserved.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <media/v4l2-mem2mem.h>
|
||||||
|
|
||||||
#include "iris_buffer.h"
|
#include "iris_buffer.h"
|
||||||
#include "iris_instance.h"
|
#include "iris_instance.h"
|
||||||
#include "iris_vdec.h"
|
#include "iris_vdec.h"
|
||||||
|
@ -10,6 +12,7 @@
|
||||||
|
|
||||||
#define DEFAULT_WIDTH 320
|
#define DEFAULT_WIDTH 320
|
||||||
#define DEFAULT_HEIGHT 240
|
#define DEFAULT_HEIGHT 240
|
||||||
|
#define DEFAULT_CODEC_ALIGNMENT 16
|
||||||
|
|
||||||
void iris_vdec_inst_init(struct iris_inst *inst)
|
void iris_vdec_inst_init(struct iris_inst *inst)
|
||||||
{
|
{
|
||||||
|
@ -54,3 +57,122 @@ void iris_vdec_inst_deinit(struct iris_inst *inst)
|
||||||
kfree(inst->fmt_dst);
|
kfree(inst->fmt_dst);
|
||||||
kfree(inst->fmt_src);
|
kfree(inst->fmt_src);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int iris_vdec_try_fmt(struct iris_inst *inst, struct v4l2_format *f)
|
||||||
|
{
|
||||||
|
struct v4l2_pix_format_mplane *pixmp = &f->fmt.pix_mp;
|
||||||
|
struct v4l2_m2m_ctx *m2m_ctx = inst->m2m_ctx;
|
||||||
|
struct v4l2_format *f_inst;
|
||||||
|
struct vb2_queue *src_q;
|
||||||
|
|
||||||
|
memset(pixmp->reserved, 0, sizeof(pixmp->reserved));
|
||||||
|
switch (f->type) {
|
||||||
|
case V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE:
|
||||||
|
if (f->fmt.pix_mp.pixelformat != V4L2_PIX_FMT_H264) {
|
||||||
|
f_inst = inst->fmt_src;
|
||||||
|
f->fmt.pix_mp.width = f_inst->fmt.pix_mp.width;
|
||||||
|
f->fmt.pix_mp.height = f_inst->fmt.pix_mp.height;
|
||||||
|
f->fmt.pix_mp.pixelformat = f_inst->fmt.pix_mp.pixelformat;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE:
|
||||||
|
if (f->fmt.pix_mp.pixelformat != V4L2_PIX_FMT_NV12) {
|
||||||
|
f_inst = inst->fmt_dst;
|
||||||
|
f->fmt.pix_mp.pixelformat = f_inst->fmt.pix_mp.pixelformat;
|
||||||
|
f->fmt.pix_mp.width = f_inst->fmt.pix_mp.width;
|
||||||
|
f->fmt.pix_mp.height = f_inst->fmt.pix_mp.height;
|
||||||
|
}
|
||||||
|
|
||||||
|
src_q = v4l2_m2m_get_src_vq(m2m_ctx);
|
||||||
|
if (vb2_is_streaming(src_q)) {
|
||||||
|
f_inst = inst->fmt_src;
|
||||||
|
f->fmt.pix_mp.height = f_inst->fmt.pix_mp.height;
|
||||||
|
f->fmt.pix_mp.width = f_inst->fmt.pix_mp.width;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pixmp->field == V4L2_FIELD_ANY)
|
||||||
|
pixmp->field = V4L2_FIELD_NONE;
|
||||||
|
|
||||||
|
pixmp->num_planes = 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int iris_vdec_s_fmt(struct iris_inst *inst, struct v4l2_format *f)
|
||||||
|
{
|
||||||
|
struct v4l2_format *fmt, *output_fmt;
|
||||||
|
struct vb2_queue *q;
|
||||||
|
u32 codec_align;
|
||||||
|
|
||||||
|
q = v4l2_m2m_get_vq(inst->m2m_ctx, f->type);
|
||||||
|
if (!q)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (vb2_is_busy(q))
|
||||||
|
return -EBUSY;
|
||||||
|
|
||||||
|
iris_vdec_try_fmt(inst, f);
|
||||||
|
|
||||||
|
switch (f->type) {
|
||||||
|
case V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE:
|
||||||
|
if (f->fmt.pix_mp.pixelformat != V4L2_PIX_FMT_H264)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
fmt = inst->fmt_src;
|
||||||
|
fmt->type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
|
||||||
|
|
||||||
|
codec_align = DEFAULT_CODEC_ALIGNMENT;
|
||||||
|
fmt->fmt.pix_mp.width = ALIGN(f->fmt.pix_mp.width, codec_align);
|
||||||
|
fmt->fmt.pix_mp.height = ALIGN(f->fmt.pix_mp.height, codec_align);
|
||||||
|
fmt->fmt.pix_mp.num_planes = 1;
|
||||||
|
fmt->fmt.pix_mp.plane_fmt[0].bytesperline = 0;
|
||||||
|
fmt->fmt.pix_mp.plane_fmt[0].sizeimage = iris_get_buffer_size(inst, BUF_INPUT);
|
||||||
|
inst->buffers[BUF_INPUT].min_count = iris_vpu_buf_count(inst, BUF_INPUT);
|
||||||
|
inst->buffers[BUF_INPUT].size = fmt->fmt.pix_mp.plane_fmt[0].sizeimage;
|
||||||
|
|
||||||
|
fmt->fmt.pix_mp.colorspace = f->fmt.pix_mp.colorspace;
|
||||||
|
fmt->fmt.pix_mp.xfer_func = f->fmt.pix_mp.xfer_func;
|
||||||
|
fmt->fmt.pix_mp.ycbcr_enc = f->fmt.pix_mp.ycbcr_enc;
|
||||||
|
fmt->fmt.pix_mp.quantization = f->fmt.pix_mp.quantization;
|
||||||
|
|
||||||
|
output_fmt = inst->fmt_dst;
|
||||||
|
output_fmt->fmt.pix_mp.colorspace = f->fmt.pix_mp.colorspace;
|
||||||
|
output_fmt->fmt.pix_mp.xfer_func = f->fmt.pix_mp.xfer_func;
|
||||||
|
output_fmt->fmt.pix_mp.ycbcr_enc = f->fmt.pix_mp.ycbcr_enc;
|
||||||
|
output_fmt->fmt.pix_mp.quantization = f->fmt.pix_mp.quantization;
|
||||||
|
|
||||||
|
inst->crop.left = 0;
|
||||||
|
inst->crop.top = 0;
|
||||||
|
inst->crop.width = f->fmt.pix_mp.width;
|
||||||
|
inst->crop.height = f->fmt.pix_mp.height;
|
||||||
|
break;
|
||||||
|
case V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE:
|
||||||
|
fmt = inst->fmt_dst;
|
||||||
|
fmt->type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
|
||||||
|
if (fmt->fmt.pix_mp.pixelformat != V4L2_PIX_FMT_NV12)
|
||||||
|
return -EINVAL;
|
||||||
|
fmt->fmt.pix_mp.pixelformat = f->fmt.pix_mp.pixelformat;
|
||||||
|
fmt->fmt.pix_mp.width = ALIGN(f->fmt.pix_mp.width, 128);
|
||||||
|
fmt->fmt.pix_mp.height = ALIGN(f->fmt.pix_mp.height, 32);
|
||||||
|
fmt->fmt.pix_mp.num_planes = 1;
|
||||||
|
fmt->fmt.pix_mp.plane_fmt[0].bytesperline = ALIGN(f->fmt.pix_mp.width, 128);
|
||||||
|
fmt->fmt.pix_mp.plane_fmt[0].sizeimage = iris_get_buffer_size(inst, BUF_OUTPUT);
|
||||||
|
inst->buffers[BUF_OUTPUT].min_count = iris_vpu_buf_count(inst, BUF_OUTPUT);
|
||||||
|
inst->buffers[BUF_OUTPUT].size = fmt->fmt.pix_mp.plane_fmt[0].sizeimage;
|
||||||
|
|
||||||
|
inst->crop.top = 0;
|
||||||
|
inst->crop.left = 0;
|
||||||
|
inst->crop.width = f->fmt.pix_mp.width;
|
||||||
|
inst->crop.height = f->fmt.pix_mp.height;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
memcpy(f, fmt, sizeof(*fmt));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
|
@ -10,5 +10,7 @@ struct iris_inst;
|
||||||
|
|
||||||
void iris_vdec_inst_init(struct iris_inst *inst);
|
void iris_vdec_inst_init(struct iris_inst *inst);
|
||||||
void iris_vdec_inst_deinit(struct iris_inst *inst);
|
void iris_vdec_inst_deinit(struct iris_inst *inst);
|
||||||
|
int iris_vdec_try_fmt(struct iris_inst *inst, struct v4l2_format *f);
|
||||||
|
int iris_vdec_s_fmt(struct iris_inst *inst, struct v4l2_format *f);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -217,6 +217,48 @@ int iris_close(struct file *filp)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int iris_try_fmt_vid_mplane(struct file *filp, void *fh, struct v4l2_format *f)
|
||||||
|
{
|
||||||
|
struct iris_inst *inst = iris_get_inst(filp, NULL);
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
mutex_lock(&inst->lock);
|
||||||
|
ret = iris_vdec_try_fmt(inst, f);
|
||||||
|
mutex_unlock(&inst->lock);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int iris_s_fmt_vid_mplane(struct file *filp, void *fh, struct v4l2_format *f)
|
||||||
|
{
|
||||||
|
struct iris_inst *inst = iris_get_inst(filp, NULL);
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
mutex_lock(&inst->lock);
|
||||||
|
ret = iris_vdec_s_fmt(inst, f);
|
||||||
|
mutex_unlock(&inst->lock);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int iris_g_fmt_vid_mplane(struct file *filp, void *fh, struct v4l2_format *f)
|
||||||
|
{
|
||||||
|
struct iris_inst *inst = iris_get_inst(filp, NULL);
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
mutex_lock(&inst->lock);
|
||||||
|
if (V4L2_TYPE_IS_OUTPUT(f->type))
|
||||||
|
*f = *inst->fmt_src;
|
||||||
|
else if (V4L2_TYPE_IS_CAPTURE(f->type))
|
||||||
|
*f = *inst->fmt_dst;
|
||||||
|
else
|
||||||
|
ret = -EINVAL;
|
||||||
|
|
||||||
|
mutex_unlock(&inst->lock);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static struct v4l2_file_operations iris_v4l2_file_ops = {
|
static struct v4l2_file_operations iris_v4l2_file_ops = {
|
||||||
.owner = THIS_MODULE,
|
.owner = THIS_MODULE,
|
||||||
.open = iris_open,
|
.open = iris_open,
|
||||||
|
@ -231,6 +273,12 @@ static const struct vb2_ops iris_vb2_ops = {
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct v4l2_ioctl_ops iris_v4l2_ioctl_ops = {
|
static const struct v4l2_ioctl_ops iris_v4l2_ioctl_ops = {
|
||||||
|
.vidioc_try_fmt_vid_cap_mplane = iris_try_fmt_vid_mplane,
|
||||||
|
.vidioc_try_fmt_vid_out_mplane = iris_try_fmt_vid_mplane,
|
||||||
|
.vidioc_s_fmt_vid_cap_mplane = iris_s_fmt_vid_mplane,
|
||||||
|
.vidioc_s_fmt_vid_out_mplane = iris_s_fmt_vid_mplane,
|
||||||
|
.vidioc_g_fmt_vid_cap_mplane = iris_g_fmt_vid_mplane,
|
||||||
|
.vidioc_g_fmt_vid_out_mplane = iris_g_fmt_vid_mplane,
|
||||||
.vidioc_reqbufs = v4l2_m2m_ioctl_reqbufs,
|
.vidioc_reqbufs = v4l2_m2m_ioctl_reqbufs,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue