Commit 1d59fbeb authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Mauro Carvalho Chehab

media: imx: imx7-media-csi: Use V4L2 subdev active state

Use the V4L2 subdev active state API to store the active format. This
simplifies the driver not only by dropping the state stored in the
imx7_csi structure, but also by replacing the manual lock with the state
lock.

The is_streaming field is now protected by the active state lock, either
explicitly in .s_stream(), where the active state is locked manually, or
implicitly in .set_fmt(), which is called with the state locked.
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Tested-by: Adam Ford <aford173@gmail.com> #imx8mm-beacon-kit
Acked-by: default avatarRui Miguel Silva <rmfrfs@gmail.com>
Tested-by: default avatarMartin Kepplinger <martin.kepplinger@puri.sm>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@kernel.org>
parent d01a1c30
...@@ -211,7 +211,6 @@ struct imx7_csi { ...@@ -211,7 +211,6 @@ struct imx7_csi {
int irq; int irq;
struct clk *mclk; struct clk *mclk;
struct mutex lock; /* Protects is_streaming, format_mbus, cc */
spinlock_t irqlock; /* Protects last_eof */ spinlock_t irqlock; /* Protects last_eof */
/* Media and V4L2 device */ /* Media and V4L2 device */
...@@ -227,8 +226,6 @@ struct imx7_csi { ...@@ -227,8 +226,6 @@ struct imx7_csi {
struct v4l2_subdev sd; struct v4l2_subdev sd;
struct media_pad pad[IMX7_CSI_PADS_NUM]; struct media_pad pad[IMX7_CSI_PADS_NUM];
struct v4l2_mbus_framefmt format_mbus[IMX7_CSI_PADS_NUM];
/* Video device */ /* Video device */
struct video_device *vdev; /* Video device */ struct video_device *vdev; /* Video device */
struct media_pad vdev_pad; /* Video device pad */ struct media_pad vdev_pad; /* Video device pad */
...@@ -509,7 +506,8 @@ static void imx7_csi_dma_stop(struct imx7_csi *csi) ...@@ -509,7 +506,8 @@ static void imx7_csi_dma_stop(struct imx7_csi *csi)
imx7_csi_hw_disable_irq(csi); imx7_csi_hw_disable_irq(csi);
} }
static void imx7_csi_configure(struct imx7_csi *csi) static void imx7_csi_configure(struct imx7_csi *csi,
struct v4l2_subdev_state *sd_state)
{ {
struct v4l2_pix_format *out_pix = &csi->vdev_fmt; struct v4l2_pix_format *out_pix = &csi->vdev_fmt;
int width = out_pix->width; int width = out_pix->width;
...@@ -540,12 +538,17 @@ static void imx7_csi_configure(struct imx7_csi *csi) ...@@ -540,12 +538,17 @@ static void imx7_csi_configure(struct imx7_csi *csi)
out_pix->pixelformat == V4L2_PIX_FMT_YUYV) out_pix->pixelformat == V4L2_PIX_FMT_YUYV)
width *= 2; width *= 2;
} else { } else {
const struct v4l2_mbus_framefmt *sink_fmt;
sink_fmt = v4l2_subdev_get_pad_format(&csi->sd, sd_state,
IMX7_CSI_PAD_SINK);
cr1 = BIT_SOF_POL | BIT_REDGE | BIT_HSYNC_POL | BIT_FCC cr1 = BIT_SOF_POL | BIT_REDGE | BIT_HSYNC_POL | BIT_FCC
| BIT_MCLKDIV(1) | BIT_MCLKEN; | BIT_MCLKDIV(1) | BIT_MCLKEN;
cr18 |= BIT_DATA_FROM_MIPI; cr18 |= BIT_DATA_FROM_MIPI;
switch (csi->format_mbus[IMX7_CSI_PAD_SINK].code) { switch (sink_fmt->code) {
case MEDIA_BUS_FMT_Y8_1X8: case MEDIA_BUS_FMT_Y8_1X8:
case MEDIA_BUS_FMT_SBGGR8_1X8: case MEDIA_BUS_FMT_SBGGR8_1X8:
case MEDIA_BUS_FMT_SGBRG8_1X8: case MEDIA_BUS_FMT_SGBRG8_1X8:
...@@ -626,7 +629,8 @@ static void imx7_csi_configure(struct imx7_csi *csi) ...@@ -626,7 +629,8 @@ static void imx7_csi_configure(struct imx7_csi *csi)
imx7_csi_reg_write(csi, stride, CSI_CSIFBUF_PARA); imx7_csi_reg_write(csi, stride, CSI_CSIFBUF_PARA);
} }
static int imx7_csi_init(struct imx7_csi *csi) static int imx7_csi_init(struct imx7_csi *csi,
struct v4l2_subdev_state *sd_state)
{ {
int ret; int ret;
...@@ -634,7 +638,7 @@ static int imx7_csi_init(struct imx7_csi *csi) ...@@ -634,7 +638,7 @@ static int imx7_csi_init(struct imx7_csi *csi)
if (ret < 0) if (ret < 0)
return ret; return ret;
imx7_csi_configure(csi); imx7_csi_configure(csi, sd_state);
ret = imx7_csi_dma_setup(csi); ret = imx7_csi_dma_setup(csi);
if (ret < 0) { if (ret < 0) {
...@@ -1420,7 +1424,7 @@ static int imx7_csi_video_validate_fmt(struct imx7_csi *csi) ...@@ -1420,7 +1424,7 @@ static int imx7_csi_video_validate_fmt(struct imx7_csi *csi)
int ret; int ret;
/* Retrieve the media bus format on the source subdev. */ /* Retrieve the media bus format on the source subdev. */
ret = v4l2_subdev_call(&csi->sd, pad, get_fmt, NULL, &fmt_src); ret = v4l2_subdev_call_state_active(&csi->sd, pad, get_fmt, &fmt_src);
if (ret) if (ret)
return ret; return ret;
...@@ -1728,12 +1732,13 @@ static int imx7_csi_video_init(struct imx7_csi *csi) ...@@ -1728,12 +1732,13 @@ static int imx7_csi_video_init(struct imx7_csi *csi)
static int imx7_csi_s_stream(struct v4l2_subdev *sd, int enable) static int imx7_csi_s_stream(struct v4l2_subdev *sd, int enable)
{ {
struct imx7_csi *csi = v4l2_get_subdevdata(sd); struct imx7_csi *csi = v4l2_get_subdevdata(sd);
struct v4l2_subdev_state *sd_state;
int ret = 0; int ret = 0;
mutex_lock(&csi->lock); sd_state = v4l2_subdev_lock_and_get_active_state(sd);
if (enable) { if (enable) {
ret = imx7_csi_init(csi); ret = imx7_csi_init(csi, sd_state);
if (ret < 0) if (ret < 0)
goto out_unlock; goto out_unlock;
...@@ -1755,29 +1760,14 @@ static int imx7_csi_s_stream(struct v4l2_subdev *sd, int enable) ...@@ -1755,29 +1760,14 @@ static int imx7_csi_s_stream(struct v4l2_subdev *sd, int enable)
csi->is_streaming = !!enable; csi->is_streaming = !!enable;
out_unlock: out_unlock:
mutex_unlock(&csi->lock); v4l2_subdev_unlock_state(sd_state);
return ret; return ret;
} }
static struct v4l2_mbus_framefmt *
imx7_csi_get_format(struct imx7_csi *csi,
struct v4l2_subdev_state *sd_state,
unsigned int pad,
enum v4l2_subdev_format_whence which)
{
if (which == V4L2_SUBDEV_FORMAT_TRY)
return v4l2_subdev_get_try_format(&csi->sd, sd_state, pad);
return &csi->format_mbus[pad];
}
static int imx7_csi_init_cfg(struct v4l2_subdev *sd, static int imx7_csi_init_cfg(struct v4l2_subdev *sd,
struct v4l2_subdev_state *sd_state) struct v4l2_subdev_state *sd_state)
{ {
const enum v4l2_subdev_format_whence which =
sd_state ? V4L2_SUBDEV_FORMAT_TRY : V4L2_SUBDEV_FORMAT_ACTIVE;
struct imx7_csi *csi = v4l2_get_subdevdata(sd);
const struct imx7_csi_pixfmt *cc; const struct imx7_csi_pixfmt *cc;
int i; int i;
...@@ -1785,7 +1775,7 @@ static int imx7_csi_init_cfg(struct v4l2_subdev *sd, ...@@ -1785,7 +1775,7 @@ static int imx7_csi_init_cfg(struct v4l2_subdev *sd,
for (i = 0; i < IMX7_CSI_PADS_NUM; i++) { for (i = 0; i < IMX7_CSI_PADS_NUM; i++) {
struct v4l2_mbus_framefmt *mf = struct v4l2_mbus_framefmt *mf =
imx7_csi_get_format(csi, sd_state, i, which); v4l2_subdev_get_pad_format(sd, sd_state, i);
mf->code = IMX7_CSI_DEF_MBUS_CODE; mf->code = IMX7_CSI_DEF_MBUS_CODE;
mf->width = IMX7_CSI_DEF_PIX_WIDTH; mf->width = IMX7_CSI_DEF_PIX_WIDTH;
...@@ -1806,59 +1796,30 @@ static int imx7_csi_enum_mbus_code(struct v4l2_subdev *sd, ...@@ -1806,59 +1796,30 @@ static int imx7_csi_enum_mbus_code(struct v4l2_subdev *sd,
struct v4l2_subdev_state *sd_state, struct v4l2_subdev_state *sd_state,
struct v4l2_subdev_mbus_code_enum *code) struct v4l2_subdev_mbus_code_enum *code)
{ {
struct imx7_csi *csi = v4l2_get_subdevdata(sd);
struct v4l2_mbus_framefmt *in_fmt; struct v4l2_mbus_framefmt *in_fmt;
int ret = 0; int ret = 0;
mutex_lock(&csi->lock); in_fmt = v4l2_subdev_get_pad_format(sd, sd_state, IMX7_CSI_PAD_SINK);
in_fmt = imx7_csi_get_format(csi, sd_state, IMX7_CSI_PAD_SINK,
code->which);
switch (code->pad) { switch (code->pad) {
case IMX7_CSI_PAD_SINK: case IMX7_CSI_PAD_SINK:
ret = imx7_csi_enum_mbus_formats(&code->code, code->index); ret = imx7_csi_enum_mbus_formats(&code->code, code->index);
break; break;
case IMX7_CSI_PAD_SRC: case IMX7_CSI_PAD_SRC:
if (code->index != 0) { if (code->index != 0) {
ret = -EINVAL; ret = -EINVAL;
goto out_unlock; break;
} }
code->code = in_fmt->code; code->code = in_fmt->code;
break; break;
default: default:
ret = -EINVAL; ret = -EINVAL;
break;
} }
out_unlock:
mutex_unlock(&csi->lock);
return ret;
}
static int imx7_csi_get_fmt(struct v4l2_subdev *sd,
struct v4l2_subdev_state *sd_state,
struct v4l2_subdev_format *sdformat)
{
struct imx7_csi *csi = v4l2_get_subdevdata(sd);
struct v4l2_mbus_framefmt *fmt;
int ret = 0;
mutex_lock(&csi->lock);
fmt = imx7_csi_get_format(csi, sd_state, sdformat->pad,
sdformat->which);
if (!fmt) {
ret = -EINVAL;
goto out_unlock;
}
sdformat->format = *fmt;
out_unlock:
mutex_unlock(&csi->lock);
return ret; return ret;
} }
...@@ -1908,7 +1869,7 @@ static void imx7_csi_try_colorimetry(struct v4l2_mbus_framefmt *tryfmt) ...@@ -1908,7 +1869,7 @@ static void imx7_csi_try_colorimetry(struct v4l2_mbus_framefmt *tryfmt)
tryfmt->ycbcr_enc); tryfmt->ycbcr_enc);
} }
static int imx7_csi_try_fmt(struct imx7_csi *csi, static void imx7_csi_try_fmt(struct v4l2_subdev *sd,
struct v4l2_subdev_state *sd_state, struct v4l2_subdev_state *sd_state,
struct v4l2_subdev_format *sdformat, struct v4l2_subdev_format *sdformat,
const struct imx7_csi_pixfmt **cc) const struct imx7_csi_pixfmt **cc)
...@@ -1917,10 +1878,7 @@ static int imx7_csi_try_fmt(struct imx7_csi *csi, ...@@ -1917,10 +1878,7 @@ static int imx7_csi_try_fmt(struct imx7_csi *csi,
struct v4l2_mbus_framefmt *in_fmt; struct v4l2_mbus_framefmt *in_fmt;
u32 code; u32 code;
in_fmt = imx7_csi_get_format(csi, sd_state, IMX7_CSI_PAD_SINK, in_fmt = v4l2_subdev_get_pad_format(sd, sd_state, IMX7_CSI_PAD_SINK);
sdformat->which);
if (!in_fmt)
return -EINVAL;
switch (sdformat->pad) { switch (sdformat->pad) {
case IMX7_CSI_PAD_SRC: case IMX7_CSI_PAD_SRC:
...@@ -1952,8 +1910,6 @@ static int imx7_csi_try_fmt(struct imx7_csi *csi, ...@@ -1952,8 +1910,6 @@ static int imx7_csi_try_fmt(struct imx7_csi *csi,
} }
imx7_csi_try_colorimetry(&sdformat->format); imx7_csi_try_colorimetry(&sdformat->format);
return 0;
} }
static int imx7_csi_set_fmt(struct v4l2_subdev *sd, static int imx7_csi_set_fmt(struct v4l2_subdev *sd,
...@@ -1966,25 +1922,13 @@ static int imx7_csi_set_fmt(struct v4l2_subdev *sd, ...@@ -1966,25 +1922,13 @@ static int imx7_csi_set_fmt(struct v4l2_subdev *sd,
const struct imx7_csi_pixfmt *cc; const struct imx7_csi_pixfmt *cc;
struct v4l2_mbus_framefmt *fmt; struct v4l2_mbus_framefmt *fmt;
struct v4l2_subdev_format format; struct v4l2_subdev_format format;
int ret = 0;
mutex_lock(&csi->lock);
if (csi->is_streaming) { if (csi->is_streaming)
ret = -EBUSY; return -EBUSY;
goto out_unlock;
}
ret = imx7_csi_try_fmt(csi, sd_state, sdformat, &cc); imx7_csi_try_fmt(sd, sd_state, sdformat, &cc);
if (ret < 0)
goto out_unlock;
fmt = imx7_csi_get_format(csi, sd_state, sdformat->pad, fmt = v4l2_subdev_get_pad_format(sd, sd_state, sdformat->pad);
sdformat->which);
if (!fmt) {
ret = -EINVAL;
goto out_unlock;
}
*fmt = sdformat->format; *fmt = sdformat->format;
...@@ -1993,19 +1937,14 @@ static int imx7_csi_set_fmt(struct v4l2_subdev *sd, ...@@ -1993,19 +1937,14 @@ static int imx7_csi_set_fmt(struct v4l2_subdev *sd,
format.pad = IMX7_CSI_PAD_SRC; format.pad = IMX7_CSI_PAD_SRC;
format.which = sdformat->which; format.which = sdformat->which;
format.format = sdformat->format; format.format = sdformat->format;
if (imx7_csi_try_fmt(csi, sd_state, &format, &outcc)) { imx7_csi_try_fmt(sd, sd_state, &format, &outcc);
ret = -EINVAL;
goto out_unlock; outfmt = v4l2_subdev_get_pad_format(sd, sd_state,
} IMX7_CSI_PAD_SRC);
outfmt = imx7_csi_get_format(csi, sd_state, IMX7_CSI_PAD_SRC,
sdformat->which);
*outfmt = format.format; *outfmt = format.format;
} }
out_unlock: return 0;
mutex_unlock(&csi->lock);
return ret;
} }
static int imx7_csi_pad_link_validate(struct v4l2_subdev *sd, static int imx7_csi_pad_link_validate(struct v4l2_subdev *sd,
...@@ -2105,7 +2044,7 @@ static const struct v4l2_subdev_video_ops imx7_csi_video_ops = { ...@@ -2105,7 +2044,7 @@ static const struct v4l2_subdev_video_ops imx7_csi_video_ops = {
static const struct v4l2_subdev_pad_ops imx7_csi_pad_ops = { static const struct v4l2_subdev_pad_ops imx7_csi_pad_ops = {
.init_cfg = imx7_csi_init_cfg, .init_cfg = imx7_csi_init_cfg,
.enum_mbus_code = imx7_csi_enum_mbus_code, .enum_mbus_code = imx7_csi_enum_mbus_code,
.get_fmt = imx7_csi_get_fmt, .get_fmt = v4l2_subdev_get_fmt,
.set_fmt = imx7_csi_set_fmt, .set_fmt = imx7_csi_set_fmt,
.link_validate = imx7_csi_pad_link_validate, .link_validate = imx7_csi_pad_link_validate,
}; };
...@@ -2199,6 +2138,7 @@ static void imx7_csi_media_cleanup(struct imx7_csi *csi) ...@@ -2199,6 +2138,7 @@ static void imx7_csi_media_cleanup(struct imx7_csi *csi)
{ {
v4l2_device_unregister(&csi->v4l2_dev); v4l2_device_unregister(&csi->v4l2_dev);
media_device_unregister(&csi->mdev); media_device_unregister(&csi->mdev);
v4l2_subdev_cleanup(&csi->sd);
media_device_cleanup(&csi->mdev); media_device_cleanup(&csi->mdev);
} }
...@@ -2266,6 +2206,10 @@ static int imx7_csi_media_init(struct imx7_csi *csi) ...@@ -2266,6 +2206,10 @@ static int imx7_csi_media_init(struct imx7_csi *csi)
if (ret) if (ret)
goto error; goto error;
ret = v4l2_subdev_init_finalize(&csi->sd);
if (ret)
goto error;
ret = v4l2_device_register_subdev(&csi->v4l2_dev, &csi->sd); ret = v4l2_device_register_subdev(&csi->v4l2_dev, &csi->sd);
if (ret) if (ret)
goto error; goto error;
...@@ -2291,27 +2235,22 @@ static int imx7_csi_probe(struct platform_device *pdev) ...@@ -2291,27 +2235,22 @@ static int imx7_csi_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, csi); platform_set_drvdata(pdev, csi);
spin_lock_init(&csi->irqlock); spin_lock_init(&csi->irqlock);
mutex_init(&csi->lock);
/* Acquire resources and install interrupt handler. */ /* Acquire resources and install interrupt handler. */
csi->mclk = devm_clk_get(&pdev->dev, "mclk"); csi->mclk = devm_clk_get(&pdev->dev, "mclk");
if (IS_ERR(csi->mclk)) { if (IS_ERR(csi->mclk)) {
ret = PTR_ERR(csi->mclk); ret = PTR_ERR(csi->mclk);
dev_err(dev, "Failed to get mclk: %d", ret); dev_err(dev, "Failed to get mclk: %d", ret);
goto destroy_mutex; return ret;
} }
csi->irq = platform_get_irq(pdev, 0); csi->irq = platform_get_irq(pdev, 0);
if (csi->irq < 0) { if (csi->irq < 0)
ret = csi->irq; return csi->irq;
goto destroy_mutex;
}
csi->regbase = devm_platform_ioremap_resource(pdev, 0); csi->regbase = devm_platform_ioremap_resource(pdev, 0);
if (IS_ERR(csi->regbase)) { if (IS_ERR(csi->regbase))
ret = PTR_ERR(csi->regbase); return PTR_ERR(csi->regbase);
goto destroy_mutex;
}
csi->model = (enum imx_csi_model)(uintptr_t)of_device_get_match_data(&pdev->dev); csi->model = (enum imx_csi_model)(uintptr_t)of_device_get_match_data(&pdev->dev);
...@@ -2319,31 +2258,23 @@ static int imx7_csi_probe(struct platform_device *pdev) ...@@ -2319,31 +2258,23 @@ static int imx7_csi_probe(struct platform_device *pdev)
(void *)csi); (void *)csi);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "Request CSI IRQ failed.\n"); dev_err(dev, "Request CSI IRQ failed.\n");
goto destroy_mutex; return ret;
} }
/* Initialize all the media device infrastructure. */ /* Initialize all the media device infrastructure. */
ret = imx7_csi_media_init(csi); ret = imx7_csi_media_init(csi);
if (ret) if (ret)
goto destroy_mutex; return ret;
/* Set the default mbus formats. */
ret = imx7_csi_init_cfg(&csi->sd, NULL);
if (ret)
goto media_cleanup;
ret = imx7_csi_async_register(csi); ret = imx7_csi_async_register(csi);
if (ret) if (ret)
goto media_cleanup; goto err_media_cleanup;
return 0; return 0;
media_cleanup: err_media_cleanup:
imx7_csi_media_cleanup(csi); imx7_csi_media_cleanup(csi);
destroy_mutex:
mutex_destroy(&csi->lock);
return ret; return ret;
} }
...@@ -2357,8 +2288,6 @@ static int imx7_csi_remove(struct platform_device *pdev) ...@@ -2357,8 +2288,6 @@ static int imx7_csi_remove(struct platform_device *pdev)
v4l2_async_nf_cleanup(&csi->notifier); v4l2_async_nf_cleanup(&csi->notifier);
v4l2_async_unregister_subdev(&csi->sd); v4l2_async_unregister_subdev(&csi->sd);
mutex_destroy(&csi->lock);
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment