From: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
To: Sakari Ailus <sakari.ailus@linux.intel.com>
Cc: Wentong Wu <wentong.wu@intel.com>,
hdegoede@redhat.com, djrscally@gmail.com,
linux-media@vger.kernel.org, bingbu.cao@linux.intel.com,
zhifeng.wang@intel.com, xiang.ye@intel.com,
tian.shu.qiu@intel.com
Subject: Re: [PATCH v11 1/2] media: pci: intel: ivsc: Add CSI submodule
Date: Fri, 4 Aug 2023 00:58:42 +0300 [thread overview]
Message-ID: <20230803215842.GJ27752@pendragon.ideasonboard.com> (raw)
In-Reply-To: <20230803115550.1601965-2-sakari.ailus@linux.intel.com>
Hi Sakari and Wentong,
Thank you for the patch.
On Thu, Aug 03, 2023 at 02:55:49PM +0300, Sakari Ailus wrote:
> From: Wentong Wu <wentong.wu@intel.com>
>
> CSI is a submodule of IVSC which can route camera sensor data
> to the outbound MIPI CSI-2 interface.
>
> The interface communicating with firmware is via MEI. There is
> a separate MEI UUID, which this driver uses to enumerate.
>
> To route camera sensor data to host, the information of link
> frequency and number of data lanes is sent to firmware by
> sending MEI command when starting stream.
>
> CSI also provides a privacy mode. When privacy mode is turned
> on, camera sensor can't be used. This means that both IVSC and
> host Image Processing Unit(IPU) can't get image data. And when
s/Unit/Unit /
I'd like to see a proof of that statement though :-) Handling camera
privacy in a closed-source firmware known to have backdoors doesn't seem
very private to me.
Is the IVSC found in lots of IPU6-based machines ?
> this mode is turned on, user is notified via v4l2 control
> callback.
>
> Signed-off-by: Wentong Wu <wentong.wu@intel.com>
> Signed-off-by: Sakari Ailus <sakari.ailus@linux.intel.com>
> ---
> drivers/media/pci/intel/Kconfig | 1 +
> drivers/media/pci/intel/Makefile | 1 +
> drivers/media/pci/intel/ivsc/Kconfig | 12 +
> drivers/media/pci/intel/ivsc/Makefile | 6 +
> drivers/media/pci/intel/ivsc/mei_csi.c | 825 +++++++++++++++++++++++++
> 5 files changed, 845 insertions(+)
> create mode 100644 drivers/media/pci/intel/ivsc/Kconfig
> create mode 100644 drivers/media/pci/intel/ivsc/Makefile
> create mode 100644 drivers/media/pci/intel/ivsc/mei_csi.c
>
> diff --git a/drivers/media/pci/intel/Kconfig b/drivers/media/pci/intel/Kconfig
> index 51b18fce6a1d..e113902fa806 100644
> --- a/drivers/media/pci/intel/Kconfig
> +++ b/drivers/media/pci/intel/Kconfig
> @@ -8,3 +8,4 @@ config IPU_BRIDGE
> dependencies, this is selected by each driver that needs it.
>
> source "drivers/media/pci/intel/ipu3/Kconfig"
> +source "drivers/media/pci/intel/ivsc/Kconfig"
> diff --git a/drivers/media/pci/intel/Makefile b/drivers/media/pci/intel/Makefile
> index 951191a7e401..f199a97e1d78 100644
> --- a/drivers/media/pci/intel/Makefile
> +++ b/drivers/media/pci/intel/Makefile
> @@ -4,3 +4,4 @@
> #
> obj-$(CONFIG_IPU_BRIDGE) += ipu-bridge.o
> obj-y += ipu3/
> +obj-y += ivsc/
> diff --git a/drivers/media/pci/intel/ivsc/Kconfig b/drivers/media/pci/intel/ivsc/Kconfig
> new file mode 100644
> index 000000000000..9535ac10f4f7
> --- /dev/null
> +++ b/drivers/media/pci/intel/ivsc/Kconfig
> @@ -0,0 +1,12 @@
> +# SPDX-License-Identifier: GPL-2.0-only
> +# Copyright (C) 2023, Intel Corporation. All rights reserved.
> +
> +config INTEL_VSC
> + tristate "Intel Visual Sensing Controller"
> + depends on INTEL_MEI
> + help
> + This adds support for Intel Visual Sensing Controller (IVSC).
> +
> + Enables the IVSC firmware services required for controlling
> + camera sensor ownership and CSI-2 link through Image Processing
> + Unit(IPU) driver of Intel.
s/Unit/Unit /
This would benefit from being reworded, it's not very clear.
> diff --git a/drivers/media/pci/intel/ivsc/Makefile b/drivers/media/pci/intel/ivsc/Makefile
> new file mode 100644
> index 000000000000..cbd194a26f03
> --- /dev/null
> +++ b/drivers/media/pci/intel/ivsc/Makefile
> @@ -0,0 +1,6 @@
> +# SPDX-License-Identifier: GPL-2.0-only
> +#
> +# Copyright (C) 2023, Intel Corporation. All rights reserved.
> +
> +obj-$(CONFIG_INTEL_VSC) += ivsc-csi.o
> +ivsc-csi-y += mei_csi.o
> diff --git a/drivers/media/pci/intel/ivsc/mei_csi.c b/drivers/media/pci/intel/ivsc/mei_csi.c
> new file mode 100644
> index 000000000000..00ba611e0f68
> --- /dev/null
> +++ b/drivers/media/pci/intel/ivsc/mei_csi.c
> @@ -0,0 +1,825 @@
> +// SPDX-License-Identifier: GPL-2.0-only
> +/*
> + * Copyright (C) 2023 Intel Corporation. All rights reserved.
> + * Intel Visual Sensing Controller CSI Linux driver
> + */
> +
> +/*
> + * To set ownership of CSI-2 link and to configure CSI-2 link, there
> + * are specific commands, which are sent via MEI protocol. The send
> + * command function uses "completion" as a synchronization mechanism.
> + * The response for command is received via a mei callback which wakes
> + * up the caller. There can be only one outstanding command at a time.
> + */
> +
> +#include <linux/completion.h>
> +#include <linux/delay.h>
> +#include <linux/kernel.h>
> +#include <linux/math64.h>
> +#include <linux/mei_cl_bus.h>
> +#include <linux/module.h>
> +#include <linux/mutex.h>
> +#include <linux/pm_runtime.h>
> +#include <linux/slab.h>
> +#include <linux/units.h>
> +#include <linux/uuid.h>
> +#include <linux/workqueue.h>
> +
> +#include <media/v4l2-async.h>
> +#include <media/v4l2-ctrls.h>
> +#include <media/v4l2-fwnode.h>
> +#include <media/v4l2-subdev.h>
> +
> +#define MEI_CSI_DRIVER_NAME "ivsc_csi"
> +#define MEI_CSI_ENTITY_NAME "Intel IVSC CSI"
> +
> +#define MEI_CSI_LINK_FREQ_400MHZ 400000000ULL
> +
> +/* the 5s used here is based on experiment */
> +#define CSI_CMD_TIMEOUT (5 * HZ)
> +/* to setup CSI-2 link an extra delay needed and determined experimentally */
> +#define CSI_FW_READY_DELAY_MS 100
> +/* link frequency unit is 100kHz */
> +#define CSI_LINK_FREQ(x) ((u32)(div_u64(x, 100 * HZ_PER_KHZ)))
> +
> +/*
> + * identify the command id supported by firmware
> + * IPC, as well as the privacy notification id
> + * used when processing privacy event.
> + */
> +enum csi_cmd_id {
> + /* used to set csi ownership */
> + CSI_SET_OWNER = 0,
> +
> + /* used to configure CSI-2 link */
> + CSI_SET_CONF = 2,
> +
> + /* privacy notification id used when privacy state changes */
> + CSI_PRIVACY_NOTIF = 6,
> +};
> +
> +/* CSI-2 link ownership definition */
> +enum csi_link_owner {
> + CSI_LINK_IVSC,
> + CSI_LINK_HOST,
> +};
> +
> +/* privacy status definition */
> +enum ivsc_privacy_status {
> + CSI_PRIVACY_OFF,
> + CSI_PRIVACY_ON,
> + CSI_PRIVACY_MAX,
> +};
> +
> +enum csi_pads {
> + CSI_PAD_SOURCE,
> + CSI_PAD_SINK,
> + CSI_NUM_PADS
> +};
> +
> +/* configuration of the CSI-2 link between host and IVSC */
> +struct csi_link_cfg {
> + /* number of data lanes used on the CSI-2 link */
> + u32 nr_of_lanes;
> +
> + /* frequency of the CSI-2 link */
> + u32 link_freq;
> +
> + /* for future use */
> + u32 rsvd[2];
> +} __packed;
> +
> +/* CSI command structure */
> +struct csi_cmd {
> + u32 cmd_id;
> + union _cmd_param {
> + u32 param;
> + struct csi_link_cfg conf;
> + } param;
> +} __packed;
> +
> +/* CSI notification structure */
> +struct csi_notif {
> + u32 cmd_id;
> + int status;
> + union _resp_cont {
> + u32 cont;
> + struct csi_link_cfg conf;
> + } cont;
> +} __packed;
> +
> +struct mei_csi {
> + struct mei_cl_device *cldev;
> +
> + /* command response */
> + struct csi_notif cmd_response;
> + /* used to wait for command response from firmware */
> + struct completion cmd_completion;
> + /* protect command download */
> + struct mutex lock;
> +
> + struct v4l2_subdev subdev;
> + struct v4l2_subdev *remote;
> + struct v4l2_async_notifier notifier;
> + struct v4l2_ctrl_handler ctrl_handler;
> + struct v4l2_ctrl *freq_ctrl;
> + struct v4l2_ctrl *privacy_ctrl;
> + unsigned int remote_pad;
> + /* start streaming or not */
> + int streaming;
> +
> + struct media_pad pads[CSI_NUM_PADS];
> + struct v4l2_mbus_framefmt format_mbus[CSI_NUM_PADS];
> +
> + /* number of data lanes used on the CSI-2 link */
> + u32 nr_of_lanes;
> + /* frequency of the CSI-2 link */
> + u64 link_freq;
> +
> + /* privacy status */
> + enum ivsc_privacy_status status;
> +};
> +
> +static const struct v4l2_mbus_framefmt mei_csi_format_mbus_default = {
> + .width = 1,
> + .height = 1,
> + .code = MEDIA_BUS_FMT_Y8_1X8,
> + .field = V4L2_FIELD_NONE,
> +};
> +
> +static s64 link_freq_menu_items[] = {
> + MEI_CSI_LINK_FREQ_400MHZ
> +};
> +
> +static inline struct mei_csi *notifier_to_csi(struct v4l2_async_notifier *n)
> +{
> + return container_of(n, struct mei_csi, notifier);
> +}
> +
> +static inline struct mei_csi *sd_to_csi(struct v4l2_subdev *sd)
> +{
> + return container_of(sd, struct mei_csi, subdev);
> +}
> +
> +static inline struct mei_csi *ctrl_to_csi(struct v4l2_ctrl *ctrl)
> +{
> + return container_of(ctrl->handler, struct mei_csi, ctrl_handler);
> +}
> +
> +/* send a command to firmware and mutex must be held by caller */
> +static int mei_csi_send(struct mei_csi *csi, u8 *buf, size_t len)
> +{
> + struct csi_cmd *cmd = (struct csi_cmd *)buf;
> + int ret;
> +
> + reinit_completion(&csi->cmd_completion);
> +
> + ret = mei_cldev_send(csi->cldev, buf, len);
> + if (ret < 0)
> + goto out;
> +
> + ret = wait_for_completion_killable_timeout(&csi->cmd_completion,
> + CSI_CMD_TIMEOUT);
> + if (ret < 0) {
> + goto out;
> + } else if (!ret) {
> + ret = -ETIMEDOUT;
> + goto out;
> + }
> +
> + /* command response status */
> + ret = csi->cmd_response.status;
> + if (ret) {
> + ret = -EINVAL;
> + goto out;
> + }
> +
> + if (csi->cmd_response.cmd_id != cmd->cmd_id)
> + ret = -EINVAL;
> +
> +out:
> + return ret;
> +}
> +
> +/* set CSI-2 link ownership */
> +static int csi_set_link_owner(struct mei_csi *csi, enum csi_link_owner owner)
> +{
> + struct csi_cmd cmd = { 0 };
> + size_t cmd_size;
> + int ret;
> +
> + cmd.cmd_id = CSI_SET_OWNER;
> + cmd.param.param = owner;
> + cmd_size = sizeof(cmd.cmd_id) + sizeof(cmd.param.param);
> +
> + mutex_lock(&csi->lock);
> +
> + ret = mei_csi_send(csi, (u8 *)&cmd, cmd_size);
> +
> + mutex_unlock(&csi->lock);
> +
> + return ret;
> +}
> +
> +/* configure CSI-2 link between host and IVSC */
> +static int csi_set_link_cfg(struct mei_csi *csi)
> +{
> + struct csi_cmd cmd = { 0 };
> + size_t cmd_size;
> + int ret;
> +
> + cmd.cmd_id = CSI_SET_CONF;
> + cmd.param.conf.nr_of_lanes = csi->nr_of_lanes;
> + cmd.param.conf.link_freq = CSI_LINK_FREQ(csi->link_freq);
> + cmd_size = sizeof(cmd.cmd_id) + sizeof(cmd.param.conf);
> +
> + mutex_lock(&csi->lock);
> +
> + ret = mei_csi_send(csi, (u8 *)&cmd, cmd_size);
> + /*
> + * wait configuration ready if download success. placing
> + * delay under mutex is to make sure current command flow
> + * completed before starting a possible new one.
> + */
> + if (!ret)
> + msleep(CSI_FW_READY_DELAY_MS);
> +
> + mutex_unlock(&csi->lock);
> +
> + return ret;
> +}
> +
> +/* callback for receive */
> +static void mei_csi_rx(struct mei_cl_device *cldev)
> +{
> + struct mei_csi *csi = mei_cldev_get_drvdata(cldev);
> + struct csi_notif notif = { 0 };
> + int ret;
> +
> + ret = mei_cldev_recv(cldev, (u8 *)¬if, sizeof(notif));
> + if (ret < 0) {
> + dev_err(&cldev->dev, "recv error: %d\n", ret);
> + return;
> + }
> +
> + switch (notif.cmd_id) {
> + case CSI_PRIVACY_NOTIF:
> + if (notif.cont.cont < CSI_PRIVACY_MAX) {
> + csi->status = notif.cont.cont;
> + v4l2_ctrl_s_ctrl(csi->privacy_ctrl, csi->status);
> + }
> + break;
> + case CSI_SET_OWNER:
> + case CSI_SET_CONF:
> + memcpy(&csi->cmd_response, ¬if, ret);
> +
> + complete(&csi->cmd_completion);
> + break;
> + default:
> + break;
> + }
> +}
> +
> +static int mei_csi_set_stream(struct v4l2_subdev *sd, int enable)
> +{
> + struct mei_csi *csi = sd_to_csi(sd);
> + s64 freq;
> + int ret;
> +
> + if (enable && csi->streaming == 0) {
> + freq = v4l2_get_link_freq(csi->remote->ctrl_handler, 0, 0);
> + if (freq < 0) {
> + dev_err(&csi->cldev->dev,
> + "error %lld, invalid link_freq\n", freq);
> + ret = freq;
> + goto err;
> + }
> + csi->link_freq = freq;
> +
> + /* switch CSI-2 link to host */
> + ret = csi_set_link_owner(csi, CSI_LINK_HOST);
> + if (ret < 0)
> + goto err;
> +
> + /* configure CSI-2 link */
> + ret = csi_set_link_cfg(csi);
> + if (ret < 0)
> + goto err_switch;
> +
> + ret = v4l2_subdev_call(csi->remote, video, s_stream, 1);
> + if (ret)
> + goto err_switch;
> + } else if (!enable && csi->streaming == 1) {
> + v4l2_subdev_call(csi->remote, video, s_stream, 0);
> +
> + /* switch CSI-2 link to IVSC */
> + ret = csi_set_link_owner(csi, CSI_LINK_IVSC);
> + if (ret < 0)
> + dev_warn(&csi->cldev->dev,
> + "failed to switch CSI2 link: %d\n", ret);
> + }
> +
> + csi->streaming = enable;
> +
> + return 0;
> +
> +err_switch:
> + csi_set_link_owner(csi, CSI_LINK_IVSC);
> +
> +err:
> + return ret;
> +}
> +
> +static struct v4l2_mbus_framefmt *
> +mei_csi_get_pad_format(struct v4l2_subdev *sd,
> + struct v4l2_subdev_state *sd_state,
> + unsigned int pad, u32 which)
> +{
> + struct mei_csi *csi = sd_to_csi(sd);
> +
> + switch (which) {
> + case V4L2_SUBDEV_FORMAT_TRY:
> + return v4l2_subdev_get_try_format(sd, sd_state, pad);
> + case V4L2_SUBDEV_FORMAT_ACTIVE:
> + return &csi->format_mbus[pad];
> + default:
> + return NULL;
> + }
> +}
> +
> +static int mei_csi_init_cfg(struct v4l2_subdev *sd,
> + struct v4l2_subdev_state *sd_state)
> +{
> + struct v4l2_mbus_framefmt *mbusformat;
> + struct mei_csi *csi = sd_to_csi(sd);
> + unsigned int i;
> +
> + mutex_lock(&csi->lock);
> +
> + for (i = 0; i < sd->entity.num_pads; i++) {
> + mbusformat = v4l2_subdev_get_try_format(sd, sd_state, i);
> + *mbusformat = mei_csi_format_mbus_default;
> + }
> +
> + mutex_unlock(&csi->lock);
> +
> + return 0;
> +}
> +
> +static int mei_csi_get_fmt(struct v4l2_subdev *sd,
> + struct v4l2_subdev_state *sd_state,
> + struct v4l2_subdev_format *format)
> +{
> + struct v4l2_mbus_framefmt *mbusformat;
> + struct mei_csi *csi = sd_to_csi(sd);
> +
> + mutex_lock(&csi->lock);
> +
> + mbusformat = mei_csi_get_pad_format(sd, sd_state, format->pad,
> + format->which);
> + if (mbusformat)
> + format->format = *mbusformat;
> +
> + mutex_unlock(&csi->lock);
> +
> + return 0;
> +}
> +
> +static int mei_csi_set_fmt(struct v4l2_subdev *sd,
> + struct v4l2_subdev_state *sd_state,
> + struct v4l2_subdev_format *format)
> +{
> + struct v4l2_mbus_framefmt *source_mbusformat;
> + struct v4l2_mbus_framefmt *mbusformat;
> + struct mei_csi *csi = sd_to_csi(sd);
> + struct media_pad *pad;
> +
> + mbusformat = mei_csi_get_pad_format(sd, sd_state, format->pad,
> + format->which);
> + if (!mbusformat)
> + return -EINVAL;
> +
> + source_mbusformat = mei_csi_get_pad_format(sd, sd_state, CSI_PAD_SOURCE,
> + format->which);
> + if (!source_mbusformat)
> + return -EINVAL;
> +
> + v4l_bound_align_image(&format->format.width, 1, 65536, 0,
> + &format->format.height, 1, 65536, 0, 0);
> +
> + switch (format->format.code) {
> + case MEDIA_BUS_FMT_RGB444_1X12:
> + case MEDIA_BUS_FMT_RGB444_2X8_PADHI_BE:
> + case MEDIA_BUS_FMT_RGB444_2X8_PADHI_LE:
> + case MEDIA_BUS_FMT_RGB555_2X8_PADHI_BE:
> + case MEDIA_BUS_FMT_RGB555_2X8_PADHI_LE:
> + case MEDIA_BUS_FMT_RGB565_1X16:
> + case MEDIA_BUS_FMT_BGR565_2X8_BE:
> + case MEDIA_BUS_FMT_BGR565_2X8_LE:
> + case MEDIA_BUS_FMT_RGB565_2X8_BE:
> + case MEDIA_BUS_FMT_RGB565_2X8_LE:
> + case MEDIA_BUS_FMT_RGB666_1X18:
> + case MEDIA_BUS_FMT_RBG888_1X24:
> + case MEDIA_BUS_FMT_RGB666_1X24_CPADHI:
> + case MEDIA_BUS_FMT_BGR888_1X24:
> + case MEDIA_BUS_FMT_GBR888_1X24:
> + case MEDIA_BUS_FMT_RGB888_1X24:
> + case MEDIA_BUS_FMT_RGB888_2X12_BE:
> + case MEDIA_BUS_FMT_RGB888_2X12_LE:
> + case MEDIA_BUS_FMT_ARGB8888_1X32:
> + case MEDIA_BUS_FMT_RGB888_1X32_PADHI:
> + case MEDIA_BUS_FMT_RGB101010_1X30:
> + case MEDIA_BUS_FMT_RGB121212_1X36:
> + case MEDIA_BUS_FMT_RGB161616_1X48:
> + case MEDIA_BUS_FMT_Y8_1X8:
> + case MEDIA_BUS_FMT_UV8_1X8:
> + case MEDIA_BUS_FMT_UYVY8_1_5X8:
> + case MEDIA_BUS_FMT_VYUY8_1_5X8:
> + case MEDIA_BUS_FMT_YUYV8_1_5X8:
> + case MEDIA_BUS_FMT_YVYU8_1_5X8:
> + case MEDIA_BUS_FMT_UYVY8_2X8:
> + case MEDIA_BUS_FMT_VYUY8_2X8:
> + case MEDIA_BUS_FMT_YUYV8_2X8:
> + case MEDIA_BUS_FMT_YVYU8_2X8:
> + case MEDIA_BUS_FMT_Y10_1X10:
> + case MEDIA_BUS_FMT_UYVY10_2X10:
> + case MEDIA_BUS_FMT_VYUY10_2X10:
> + case MEDIA_BUS_FMT_YUYV10_2X10:
> + case MEDIA_BUS_FMT_YVYU10_2X10:
> + case MEDIA_BUS_FMT_Y12_1X12:
> + case MEDIA_BUS_FMT_UYVY12_2X12:
> + case MEDIA_BUS_FMT_VYUY12_2X12:
> + case MEDIA_BUS_FMT_YUYV12_2X12:
> + case MEDIA_BUS_FMT_YVYU12_2X12:
> + case MEDIA_BUS_FMT_UYVY8_1X16:
> + case MEDIA_BUS_FMT_VYUY8_1X16:
> + case MEDIA_BUS_FMT_YUYV8_1X16:
> + case MEDIA_BUS_FMT_YVYU8_1X16:
> + case MEDIA_BUS_FMT_YDYUYDYV8_1X16:
> + case MEDIA_BUS_FMT_UYVY10_1X20:
> + case MEDIA_BUS_FMT_VYUY10_1X20:
> + case MEDIA_BUS_FMT_YUYV10_1X20:
> + case MEDIA_BUS_FMT_YVYU10_1X20:
> + case MEDIA_BUS_FMT_VUY8_1X24:
> + case MEDIA_BUS_FMT_YUV8_1X24:
> + case MEDIA_BUS_FMT_UYYVYY8_0_5X24:
> + case MEDIA_BUS_FMT_UYVY12_1X24:
> + case MEDIA_BUS_FMT_VYUY12_1X24:
> + case MEDIA_BUS_FMT_YUYV12_1X24:
> + case MEDIA_BUS_FMT_YVYU12_1X24:
> + case MEDIA_BUS_FMT_YUV10_1X30:
> + case MEDIA_BUS_FMT_UYYVYY10_0_5X30:
> + case MEDIA_BUS_FMT_AYUV8_1X32:
> + case MEDIA_BUS_FMT_UYYVYY12_0_5X36:
> + case MEDIA_BUS_FMT_YUV12_1X36:
> + case MEDIA_BUS_FMT_YUV16_1X48:
> + case MEDIA_BUS_FMT_UYYVYY16_0_5X48:
> + case MEDIA_BUS_FMT_JPEG_1X8:
> + case MEDIA_BUS_FMT_AHSV8888_1X32:
> + case MEDIA_BUS_FMT_SBGGR8_1X8:
> + case MEDIA_BUS_FMT_SGBRG8_1X8:
> + case MEDIA_BUS_FMT_SGRBG8_1X8:
> + case MEDIA_BUS_FMT_SRGGB8_1X8:
> + case MEDIA_BUS_FMT_SBGGR10_1X10:
> + case MEDIA_BUS_FMT_SGBRG10_1X10:
> + case MEDIA_BUS_FMT_SGRBG10_1X10:
> + case MEDIA_BUS_FMT_SRGGB10_1X10:
> + case MEDIA_BUS_FMT_SBGGR12_1X12:
> + case MEDIA_BUS_FMT_SGBRG12_1X12:
> + case MEDIA_BUS_FMT_SGRBG12_1X12:
> + case MEDIA_BUS_FMT_SRGGB12_1X12:
> + case MEDIA_BUS_FMT_SBGGR14_1X14:
> + case MEDIA_BUS_FMT_SGBRG14_1X14:
> + case MEDIA_BUS_FMT_SGRBG14_1X14:
> + case MEDIA_BUS_FMT_SRGGB14_1X14:
> + case MEDIA_BUS_FMT_SBGGR16_1X16:
> + case MEDIA_BUS_FMT_SGBRG16_1X16:
> + case MEDIA_BUS_FMT_SGRBG16_1X16:
> + case MEDIA_BUS_FMT_SRGGB16_1X16:
> + break;
> + default:
> + format->format.code = MEDIA_BUS_FMT_Y8_1X8;
> + break;
> + }
I wonder if we should simply accept all formats. The IVSC doesn't seem
to cara.
> +
> + if (format->format.field == V4L2_FIELD_ANY)
> + format->format.field = V4L2_FIELD_NONE;
> +
> + mutex_lock(&csi->lock);
> +
> + pad = &csi->pads[format->pad];
> + if (pad->flags & MEDIA_PAD_FL_SOURCE)
> + format->format = csi->format_mbus[CSI_PAD_SINK];
> +
> + *mbusformat = format->format;
> +
> + if (pad->flags & MEDIA_PAD_FL_SINK)
> + *source_mbusformat = format->format;
> +
> + mutex_unlock(&csi->lock);
> +
> + return 0;
> +}
> +
> +static int mei_csi_g_volatile_ctrl(struct v4l2_ctrl *ctrl)
> +{
> + struct mei_csi *csi = ctrl_to_csi(ctrl);
> + s64 freq;
> +
> + if (ctrl->id == V4L2_CID_LINK_FREQ) {
> + if (!csi->remote)
> + return -EINVAL;
> +
> + freq = v4l2_get_link_freq(csi->remote->ctrl_handler, 0, 0);
> + if (freq < 0) {
> + dev_err(&csi->cldev->dev,
> + "error %lld, invalid link_freq\n", freq);
> + return -EINVAL;
> + }
> +
> + link_freq_menu_items[0] = freq;
> + ctrl->val = 0;
> +
> + return 0;
> + }
> +
> + return -EINVAL;
> +}
> +
> +static const struct v4l2_ctrl_ops mei_csi_ctrl_ops = {
> + .g_volatile_ctrl = mei_csi_g_volatile_ctrl,
> +};
> +
> +static const struct v4l2_subdev_video_ops mei_csi_video_ops = {
> + .s_stream = mei_csi_set_stream,
> +};
> +
> +static const struct v4l2_subdev_pad_ops mei_csi_pad_ops = {
> + .init_cfg = mei_csi_init_cfg,
> + .get_fmt = mei_csi_get_fmt,
> + .set_fmt = mei_csi_set_fmt,
> +};
> +
> +static const struct v4l2_subdev_ops mei_csi_subdev_ops = {
> + .video = &mei_csi_video_ops,
> + .pad = &mei_csi_pad_ops,
> +};
> +
> +static const struct media_entity_operations mei_csi_entity_ops = {
> + .link_validate = v4l2_subdev_link_validate,
> +};
> +
> +static int mei_csi_notify_bound(struct v4l2_async_notifier *notifier,
> + struct v4l2_subdev *subdev,
> + struct v4l2_async_connection *asd)
> +{
> + struct mei_csi *csi = notifier_to_csi(notifier);
> + int pad;
> +
> + pad = media_entity_get_fwnode_pad(&subdev->entity, asd->match.fwnode,
> + MEDIA_PAD_FL_SOURCE);
> + if (pad < 0)
> + return pad;
> +
> + csi->remote = subdev;
> + csi->remote_pad = pad;
> +
> + return media_create_pad_link(&subdev->entity, pad,
> + &csi->subdev.entity, 1,
> + MEDIA_LNK_FL_ENABLED |
> + MEDIA_LNK_FL_IMMUTABLE);
> +}
> +
> +static void mei_csi_notify_unbind(struct v4l2_async_notifier *notifier,
> + struct v4l2_subdev *subdev,
> + struct v4l2_async_connection *asd)
> +{
> + struct mei_csi *csi = notifier_to_csi(notifier);
> +
> + csi->remote = NULL;
> +}
> +
> +static const struct v4l2_async_notifier_operations mei_csi_notify_ops = {
> + .bound = mei_csi_notify_bound,
> + .unbind = mei_csi_notify_unbind,
> +};
> +
> +static int mei_csi_init_controls(struct mei_csi *csi)
> +{
> + u32 max;
> + int ret;
> +
> + ret = v4l2_ctrl_handler_init(&csi->ctrl_handler, 2);
> + if (ret)
> + return ret;
> +
> + csi->ctrl_handler.lock = &csi->lock;
> +
> + max = ARRAY_SIZE(link_freq_menu_items) - 1;
> + csi->freq_ctrl = v4l2_ctrl_new_int_menu(&csi->ctrl_handler,
> + &mei_csi_ctrl_ops,
> + V4L2_CID_LINK_FREQ,
> + max,
> + 0,
> + link_freq_menu_items);
> + if (csi->freq_ctrl)
> + csi->freq_ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY |
> + V4L2_CTRL_FLAG_VOLATILE;
> +
> + csi->privacy_ctrl = v4l2_ctrl_new_std(&csi->ctrl_handler, NULL,
> + V4L2_CID_PRIVACY, 0, 1, 1, 0);
> + if (csi->privacy_ctrl)
> + csi->privacy_ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY;
> +
> + if (csi->ctrl_handler.error)
> + return csi->ctrl_handler.error;
> +
> + csi->subdev.ctrl_handler = &csi->ctrl_handler;
> +
> + return 0;
> +}
> +
> +static int mei_csi_parse_firmware(struct mei_csi *csi)
> +{
> + struct v4l2_fwnode_endpoint v4l2_ep = {
> + .bus_type = V4L2_MBUS_CSI2_DPHY,
> + };
> + struct device *dev = &csi->cldev->dev;
> + struct v4l2_async_connection *asd;
> + struct fwnode_handle *fwnode;
> + struct fwnode_handle *ep;
> + int ret;
> +
> + ep = fwnode_graph_get_endpoint_by_id(dev_fwnode(dev), 0, 0, 0);
> + if (!ep) {
> + dev_err(dev, "not connected to subdevice\n");
> + return -EINVAL;
> + }
> +
> + ret = v4l2_fwnode_endpoint_parse(ep, &v4l2_ep);
> + if (ret) {
> + dev_err(dev, "could not parse v4l2 endpoint\n");
> + fwnode_handle_put(ep);
> + return -EINVAL;
> + }
> +
> + fwnode = fwnode_graph_get_remote_endpoint(ep);
> + fwnode_handle_put(ep);
> +
> + v4l2_async_subdev_nf_init(&csi->notifier, &csi->subdev);
> + csi->notifier.ops = &mei_csi_notify_ops;
> +
> + asd = v4l2_async_nf_add_fwnode(&csi->notifier, fwnode,
> + struct v4l2_async_connection);
> + if (IS_ERR(asd)) {
> + fwnode_handle_put(fwnode);
> + return PTR_ERR(asd);
> + }
> +
> + ret = v4l2_fwnode_endpoint_alloc_parse(fwnode, &v4l2_ep);
> + fwnode_handle_put(fwnode);
> + if (ret)
> + return ret;
> + csi->nr_of_lanes = v4l2_ep.bus.mipi_csi2.num_data_lanes;
> +
> + ret = v4l2_async_nf_register(&csi->notifier);
> + if (ret)
> + v4l2_async_nf_cleanup(&csi->notifier);
> +
> + v4l2_fwnode_endpoint_free(&v4l2_ep);
> +
> + return ret;
> +}
> +
> +static int mei_csi_probe(struct mei_cl_device *cldev,
> + const struct mei_cl_device_id *id)
> +{
> + struct device *dev = &cldev->dev;
> + struct mei_csi *csi;
> + int ret;
> +
> + if (!dev_fwnode(dev))
> + return -EPROBE_DEFER;
> +
> + csi = devm_kzalloc(dev, sizeof(struct mei_csi), GFP_KERNEL);
> + if (!csi)
> + return -ENOMEM;
> +
> + csi->cldev = cldev;
> + mutex_init(&csi->lock);
> + init_completion(&csi->cmd_completion);
> +
> + mei_cldev_set_drvdata(cldev, csi);
> +
> + ret = mei_cldev_enable(cldev);
> + if (ret < 0) {
> + dev_err(dev, "mei_cldev_enable failed: %d\n", ret);
> + goto destroy_mutex;
> + }
> +
> + ret = mei_cldev_register_rx_cb(cldev, mei_csi_rx);
> + if (ret) {
> + dev_err(dev, "event cb registration failed: %d\n", ret);
> + goto err_disable;
> + }
> +
> + ret = mei_csi_parse_firmware(csi);
> + if (ret)
> + goto err_disable;
> +
> + csi->subdev.dev = &cldev->dev;
> + v4l2_subdev_init(&csi->subdev, &mei_csi_subdev_ops);
> + v4l2_set_subdevdata(&csi->subdev, csi);
> + csi->subdev.flags = V4L2_SUBDEV_FL_HAS_DEVNODE |
> + V4L2_SUBDEV_FL_HAS_EVENTS;
> + csi->subdev.entity.function = MEDIA_ENT_F_VID_IF_BRIDGE;
> + csi->subdev.entity.ops = &mei_csi_entity_ops;
> +
> + snprintf(csi->subdev.name, sizeof(csi->subdev.name),
> + MEI_CSI_ENTITY_NAME);
> +
> + ret = mei_csi_init_controls(csi);
> + if (ret)
> + goto err_ctrl_handler;
> +
> + csi->format_mbus[CSI_PAD_SOURCE] = mei_csi_format_mbus_default;
> + csi->format_mbus[CSI_PAD_SINK] = mei_csi_format_mbus_default;
> +
> + csi->pads[CSI_PAD_SOURCE].flags = MEDIA_PAD_FL_SOURCE;
> + csi->pads[CSI_PAD_SINK].flags = MEDIA_PAD_FL_SINK;
> + ret = media_entity_pads_init(&csi->subdev.entity, CSI_NUM_PADS,
> + csi->pads);
> + if (ret)
> + goto err_ctrl_handler;
> +
> + ret = v4l2_subdev_init_finalize(&csi->subdev);
> + if (ret < 0)
> + goto err_entity;
> +
> + ret = v4l2_async_register_subdev(&csi->subdev);
> + if (ret < 0)
> + goto err_subdev;
> +
> + pm_runtime_enable(&cldev->dev);
> +
> + return 0;
> +
> +err_subdev:
> + v4l2_subdev_cleanup(&csi->subdev);
> +
> +err_entity:
> + media_entity_cleanup(&csi->subdev.entity);
> +
> +err_ctrl_handler:
> + v4l2_ctrl_handler_free(&csi->ctrl_handler);
> + v4l2_async_nf_unregister(&csi->notifier);
> + v4l2_async_nf_cleanup(&csi->notifier);
> +
> +err_disable:
> + mei_cldev_disable(cldev);
> +
> +destroy_mutex:
> + mutex_destroy(&csi->lock);
> +
> + return ret;
> +}
> +
> +static void mei_csi_remove(struct mei_cl_device *cldev)
> +{
> + struct mei_csi *csi = mei_cldev_get_drvdata(cldev);
> +
> + v4l2_async_nf_unregister(&csi->notifier);
> + v4l2_async_nf_cleanup(&csi->notifier);
> + v4l2_ctrl_handler_free(&csi->ctrl_handler);
> + v4l2_async_unregister_subdev(&csi->subdev);
> + v4l2_subdev_cleanup(&csi->subdev);
> + media_entity_cleanup(&csi->subdev.entity);
> +
> + pm_runtime_disable(&cldev->dev);
> +
> + mutex_destroy(&csi->lock);
> +}
> +
> +#define MEI_CSI_UUID UUID_LE(0x92335FCF, 0x3203, 0x4472, \
> + 0xAF, 0x93, 0x7b, 0x44, 0x53, 0xAC, 0x29, 0xDA)
> +
> +static const struct mei_cl_device_id mei_csi_tbl[] = {
> + { MEI_CSI_DRIVER_NAME, MEI_CSI_UUID, MEI_CL_VERSION_ANY },
> + { /* sentinel */ }
> +};
> +MODULE_DEVICE_TABLE(mei, mei_csi_tbl);
> +
> +static struct mei_cl_driver mei_csi_driver = {
> + .id_table = mei_csi_tbl,
> + .name = MEI_CSI_DRIVER_NAME,
> +
> + .probe = mei_csi_probe,
> + .remove = mei_csi_remove,
> +};
> +
> +module_mei_cl_driver(mei_csi_driver);
> +
> +MODULE_AUTHOR("Wentong Wu <wentong.wu@intel.com>");
> +MODULE_AUTHOR("Zhifeng Wang <zhifeng.wang@intel.com>");
> +MODULE_DESCRIPTION("Device driver for IVSC CSI");
> +MODULE_LICENSE("GPL");
--
Regards,
Laurent Pinchart
next prev parent reply other threads:[~2023-08-03 21:58 UTC|newest]
Thread overview: 11+ messages / expand[flat|nested] mbox.gz Atom feed top
2023-08-03 11:55 [PATCH v11 0/2] media: pci: intel: ivsc: Add driver of Intel Visual Sensing Controller(IVSC) Sakari Ailus
2023-08-03 11:55 ` [PATCH v11 1/2] media: pci: intel: ivsc: Add CSI submodule Sakari Ailus
2023-08-03 21:58 ` Laurent Pinchart [this message]
2023-08-03 22:03 ` Sakari Ailus
2023-08-03 22:08 ` Laurent Pinchart
2023-08-04 6:20 ` Sakari Ailus
2023-08-04 10:32 ` Laurent Pinchart
2023-08-04 11:19 ` Sakari Ailus
2023-08-04 13:45 ` Laurent Pinchart
2023-08-07 9:33 ` Sakari Ailus
2023-08-03 11:55 ` [PATCH v11 2/2] media: pci: intel: ivsc: Add ACE submodule Sakari Ailus
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20230803215842.GJ27752@pendragon.ideasonboard.com \
--to=laurent.pinchart@ideasonboard.com \
--cc=bingbu.cao@linux.intel.com \
--cc=djrscally@gmail.com \
--cc=hdegoede@redhat.com \
--cc=linux-media@vger.kernel.org \
--cc=sakari.ailus@linux.intel.com \
--cc=tian.shu.qiu@intel.com \
--cc=wentong.wu@intel.com \
--cc=xiang.ye@intel.com \
--cc=zhifeng.wang@intel.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox