[PATCH v4 3/5] media: platform: mediatek: isp_30: add mediatek ISP3.0 sensor interface
AngeloGioacchino Del Regno
angelogioacchino.delregno at collabora.com
Wed Jun 12 01:06:35 PDT 2024
Il 10/06/24 16:39, Julien Stephan ha scritto:
> Le jeu. 11 janv. 2024 à 13:04, AngeloGioacchino Del Regno
> <angelogioacchino.delregno at collabora.com> a écrit :
>>
>> Il 10/01/24 15:14, Julien Stephan ha scritto:
>>> From: Louis Kuo <louis.kuo at mediatek.com>
>>>
>>> This will add the mediatek ISP3.0 seninf (sensor interface) driver found
>>> on several Mediatek SoCs such as the mt8365.
>>>
>>> Then seninf module has 4 physical CSI-2 inputs. Depending on the soc they
>>> may not be all connected.
>>>
>>> Signed-off-by: Louis Kuo <louis.kuo at mediatek.com>
>>> Signed-off-by: Phi-bang Nguyen <pnguyen at baylibre.com>
>>> Signed-off-by: Florian Sylvestre <fsylvestre at baylibre.com>
>>> Co-developed-by: Laurent Pinchart <laurent.pinchart at ideasonboard.com>
>>> Signed-off-by: Laurent Pinchart <laurent.pinchart at ideasonboard.com>
>>> Co-developed-by: Julien Stephan <jstephan at baylibre.com>
>>> Signed-off-by: Julien Stephan <jstephan at baylibre.com>
>>> ---
>>> MAINTAINERS | 1 +
>>> drivers/media/platform/mediatek/Kconfig | 1 +
>>> drivers/media/platform/mediatek/Makefile | 1 +
>>> drivers/media/platform/mediatek/isp/Kconfig | 2 +
>>> drivers/media/platform/mediatek/isp/Makefile | 3 +
>>> .../platform/mediatek/isp/isp_30/Kconfig | 16 +
>>> .../platform/mediatek/isp/isp_30/Makefile | 3 +
>>> .../mediatek/isp/isp_30/seninf/Makefile | 5 +
>>> .../mediatek/isp/isp_30/seninf/mtk_seninf.c | 1488 +++++++++++++++++
>>> .../isp/isp_30/seninf/mtk_seninf_reg.h | 112 ++
>>> 10 files changed, 1632 insertions(+)
>>> create mode 100644 drivers/media/platform/mediatek/isp/Kconfig
>>> create mode 100644 drivers/media/platform/mediatek/isp/Makefile
>>> create mode 100644 drivers/media/platform/mediatek/isp/isp_30/Kconfig
>>> create mode 100644 drivers/media/platform/mediatek/isp/isp_30/Makefile
>>> create mode 100644 drivers/media/platform/mediatek/isp/isp_30/seninf/Makefile
>>> create mode 100644 drivers/media/platform/mediatek/isp/isp_30/seninf/mtk_seninf.c
>>> create mode 100644 drivers/media/platform/mediatek/isp/isp_30/seninf/mtk_seninf_reg.h
>>>
>>> diff --git a/MAINTAINERS b/MAINTAINERS
>>> index 3ea2158864e1..52d200d5e36c 100644
>>> --- a/MAINTAINERS
>>> +++ b/MAINTAINERS
>>> @@ -13618,6 +13618,7 @@ M: Andy Hsieh <andy.hsieh at mediatek.com>
>>> S: Supported
>>> F: Documentation/devicetree/bindings/media/mediatek,mt8365-camsv.yaml
>>> F: Documentation/devicetree/bindings/media/mediatek,mt8365-seninf.yaml
>>> +F: drivers/media/platform/mediatek/isp/isp_30/seninf/*
>>>
>>> MEDIATEK SMI DRIVER
>>> M: Yong Wu <yong.wu at mediatek.com>
>>> diff --git a/drivers/media/platform/mediatek/Kconfig b/drivers/media/platform/mediatek/Kconfig
>>> index 84104e2cd024..4e0a5a43f35e 100644
>>> --- a/drivers/media/platform/mediatek/Kconfig
>>> +++ b/drivers/media/platform/mediatek/Kconfig
>>> @@ -7,3 +7,4 @@ source "drivers/media/platform/mediatek/mdp/Kconfig"
>>> source "drivers/media/platform/mediatek/vcodec/Kconfig"
>>> source "drivers/media/platform/mediatek/vpu/Kconfig"
>>> source "drivers/media/platform/mediatek/mdp3/Kconfig"
>>> +source "drivers/media/platform/mediatek/isp/Kconfig"
>>> diff --git a/drivers/media/platform/mediatek/Makefile b/drivers/media/platform/mediatek/Makefile
>>> index 38e6ba917fe5..695f05f525a6 100644
>>> --- a/drivers/media/platform/mediatek/Makefile
>>> +++ b/drivers/media/platform/mediatek/Makefile
>>> @@ -4,3 +4,4 @@ obj-y += mdp/
>>> obj-y += vcodec/
>>> obj-y += vpu/
>>> obj-y += mdp3/
>>> +obj-y += isp/
>>> diff --git a/drivers/media/platform/mediatek/isp/Kconfig b/drivers/media/platform/mediatek/isp/Kconfig
>>> new file mode 100644
>>> index 000000000000..708b9a6660d2
>>> --- /dev/null
>>> +++ b/drivers/media/platform/mediatek/isp/Kconfig
>>> @@ -0,0 +1,2 @@
>>> +# SPDX-License-Identifier: GPL-2.0-only
>>> +source "drivers/media/platform/mediatek/isp/isp_30/Kconfig"
>>> diff --git a/drivers/media/platform/mediatek/isp/Makefile b/drivers/media/platform/mediatek/isp/Makefile
>>> new file mode 100644
>>> index 000000000000..a81ab33d0dd3
>>> --- /dev/null
>>> +++ b/drivers/media/platform/mediatek/isp/Makefile
>>> @@ -0,0 +1,3 @@
>>> +# SPDX-License-Identifier: GPL-2.0-only
>>> +
>>> +obj-y += isp_30/
>>> diff --git a/drivers/media/platform/mediatek/isp/isp_30/Kconfig b/drivers/media/platform/mediatek/isp/isp_30/Kconfig
>>> new file mode 100644
>>> index 000000000000..9791312589fb
>>> --- /dev/null
>>> +++ b/drivers/media/platform/mediatek/isp/isp_30/Kconfig
>>> @@ -0,0 +1,16 @@
>>> +# SPDX-License-Identifier: GPL-2.0-only
>>> +config MTK_SENINF30
>>> + tristate "MediaTek ISP3.0 SENINF driver"
>>> + depends on VIDEO_V4L2_SUBDEV_API
>>> + depends on MEDIA_CAMERA_SUPPORT
>>> + depends on ARCH_MEDIATEK || COMPILE_TEST
>>> + depends on OF
>>> + select V4L2_FWNODE
>>> + default n
>>> + help
>>> + This driver provides a MIPI CSI-2 receiver interface to connect
>>> + an external camera module with MediaTek ISP3.0. It is able to handle
>>> + multiple cameras at the same time.
>>> +
>>> + To compile this driver as a module, choose M here: the
>>> + module will be called mtk-seninf.
>>> diff --git a/drivers/media/platform/mediatek/isp/isp_30/Makefile b/drivers/media/platform/mediatek/isp/isp_30/Makefile
>>> new file mode 100644
>>> index 000000000000..ac3142de4739
>>> --- /dev/null
>>> +++ b/drivers/media/platform/mediatek/isp/isp_30/Makefile
>>> @@ -0,0 +1,3 @@
>>> +# SPDX-License-Identifier: GPL-2.0
>>> +
>>> +obj-$(CONFIG_MTK_SENINF30) += seninf/
>>> diff --git a/drivers/media/platform/mediatek/isp/isp_30/seninf/Makefile b/drivers/media/platform/mediatek/isp/isp_30/seninf/Makefile
>>> new file mode 100644
>>> index 000000000000..f28480d6d6c3
>>> --- /dev/null
>>> +++ b/drivers/media/platform/mediatek/isp/isp_30/seninf/Makefile
>>> @@ -0,0 +1,5 @@
>>> +# SPDX-License-Identifier: GPL-2.0
>>> +
>>> +mtk-seninf-objs += mtk_seninf.o
>>> +
>>> +obj-$(CONFIG_MTK_SENINF30) += mtk-seninf.o
>>> diff --git a/drivers/media/platform/mediatek/isp/isp_30/seninf/mtk_seninf.c b/drivers/media/platform/mediatek/isp/isp_30/seninf/mtk_seninf.c
>>> new file mode 100644
>>> index 000000000000..67b2c697d9ca
>>> --- /dev/null
>>> +++ b/drivers/media/platform/mediatek/isp/isp_30/seninf/mtk_seninf.c
>>> @@ -0,0 +1,1488 @@
>>> +// SPDX-License-Identifier: GPL-2.0
>>> +/*
>>> + * Copyright (c) 2022 MediaTek Inc.
>>> + */
>>> +#include <linux/bitfield.h>
>>> +#include <linux/clk.h>
>>> +#include <linux/delay.h>
>>> +#include <linux/interrupt.h>
>>> +#include <linux/module.h>
>>> +#include <linux/of_graph.h>
>>> +#include <linux/of_platform.h>
>>> +#include <linux/phy/phy.h>
>>> +#include <linux/platform_device.h>
>>> +#include <linux/pm_runtime.h>
>>> +#include <linux/videodev2.h>
>>> +#include <media/media-device.h>
>>> +#include <media/media-entity.h>
>>> +#include <media/v4l2-async.h>
>>> +#include <media/v4l2-common.h>
>>> +#include <media/v4l2-ctrls.h>
>>> +#include <media/v4l2-dev.h>
>>> +#include <media/v4l2-device.h>
>>> +#include <media/v4l2-event.h>
>>> +#include <media/v4l2-fwnode.h>
>>> +#include <media/v4l2-mc.h>
>>> +#include <media/v4l2-subdev.h>
>>> +
>>> +#include "mtk_seninf_reg.h"
>>> +
>>> +#define SENINF_TIMESTAMP_STEP 0x67
>>> +#define SENINF_SETTLE_DELAY 0x15
>>> +#define SENINF_HS_TRAIL_PARAMETER 0x8
>>> +
>>> +#define SENINF_MAX_NUM_INPUTS 4
>>> +#define SENINF_MAX_NUM_OUTPUTS 6
>>> +#define SENINF_MAX_NUM_MUXES 6
>>> +#define SENINF_MAX_NUM_PADS (SENINF_MAX_NUM_INPUTS + \
>>> + SENINF_MAX_NUM_OUTPUTS)
>>> +
>>> +#define SENINF_DEFAULT_BUS_FMT MEDIA_BUS_FMT_SGRBG10_1X10
>>> +#define SENINF_DEFAULT_WIDTH 1920
>>> +#define SENINF_DEFAULT_HEIGHT 1080
>>> +
>>> +#define SENINF_PAD_10BIT 0
>>> +
>>> +#define SENINF_TEST_MODEL 0
>>> +#define SENINF_NORMAL_MODEL 1
>>> +#define SENINF_ALL_ERR_IRQ_EN 0x7f
>>> +#define SENINF_IRQ_CLR_SEL 0x80000000
>>> +
>>> +#define SENINF_MIPI_SENSOR 0x8
>>> +
>>> +#define MTK_CSI_MAX_LANES 4
>>> +
>>> +/* Port number in the device tree. */
>>> +enum mtk_seninf_port {
>>> + CSI_PORT_0 = 0, /* 4D1C or 2D1C */
>>> + CSI_PORT_1, /* 4D1C */
>>> + CSI_PORT_2, /* 4D1C */
>>> + CSI_PORT_0B, /* 2D1C */
>>> +};
>>> +
>>> +enum mtk_seninf_id {
>>> + SENINF_1 = 0,
>>> + SENINF_2 = 1,
>>> + SENINF_3 = 2,
>>> + SENINF_5 = 4,
>>> +};
>>> +
>>> +static const u32 port_to_seninf_id[] = {
>>> + [CSI_PORT_0] = SENINF_1,
>>> + [CSI_PORT_1] = SENINF_3,
>>> + [CSI_PORT_2] = SENINF_5,
>>> + [CSI_PORT_0B] = SENINF_2,
>>> +};
>>> +
>>> +enum mtk_seninf_phy_mode {
>>> + SENINF_PHY_MODE_NONE,
>>> + SENINF_PHY_MODE_4D1C,
>>> + SENINF_PHY_MODE_2D1C,
>>> +};
>>> +
>>> +enum mtk_seninf_format_flag {
>>> + MTK_SENINF_FORMAT_BAYER = BIT(0),
>>> + MTK_SENINF_FORMAT_DPCM = BIT(1),
>>> + MTK_SENINF_FORMAT_JPEG = BIT(2),
>>> + MTK_SENINF_FORMAT_INPUT_ONLY = BIT(3),
>>> +};
>>> +
>>> +/**
>>> + * struct mtk_seninf_conf - Model-specific SENINF parameters
>>> + * @model: Model description
>>> + * @nb_inputs: Number of SENINF inputs
>>> + * @nb_muxes: Number of SENINF MUX (FIFO) instances
>>> + * @nb_outputs: Number of outputs (to CAM and CAMSV instances)
>>> + */
>>> +struct mtk_seninf_conf {
>>> + const char *model;
>>> + u8 nb_inputs;
>>> + u8 nb_muxes;
>>> + u8 nb_outputs;
>>> +};
>>> +
>>> +/**
>>> + * struct mtk_seninf_format_info - Information about media bus formats
>>> + * @code: V4L2 media bus code
>>> + * @flags: Flags describing the format, as a combination of MTK_SENINF_FORMAT_*
>>> + */
>>> +struct mtk_seninf_format_info {
>>> + u32 code;
>>> + u32 flags;
>>> +};
>>> +
>>> +/**
>>> + * struct mtk_seninf_input - SENINF input block
>>> + * @pad: DT port and media entity pad number
>>> + * @seninf_id: SENINF hardware instance ID
>>> + * @base: Memory mapped I/O based address
>>> + * @seninf: Back pointer to the mtk_seninf
>>> + * @phy: PHY connected to the input
>>> + * @phy_mode: PHY operation mode (NONE when the input is not connected)
>>> + * @bus: CSI-2 bus configuration from DT
>>> + * @source_sd: Source subdev connected to the input
>>> + */
>>> +struct mtk_seninf_input {
>>> + enum mtk_seninf_port pad;
>>> + enum mtk_seninf_id seninf_id;
>>> + void __iomem *base;
>>> + struct mtk_seninf *seninf;
>>> +
>>> + struct phy *phy;
>>> + enum mtk_seninf_phy_mode phy_mode;
>>> +
>>> + struct v4l2_mbus_config_mipi_csi2 bus;
>>> +
>>> + struct v4l2_subdev *source_sd;
>>> +};
>>> +
>>> +/**
>>> + * struct mtk_seninf_mux - SENINF MUX channel
>>> + * @pad: DT port and media entity pad number
>>> + * @mux_id: MUX hardware instance ID
>>> + * @base: Memory mapped I/O based address
>>> + * @seninf: Back pointer to the mtk_seninf
>>> + */
>>> +struct mtk_seninf_mux {
>>> + unsigned int pad;
>>> + unsigned int mux_id;
>>> + void __iomem *base;
>>> + struct mtk_seninf *seninf;
>>> +};
>>> +
>>> +/**
>>> + * struct mtk_seninf - Top-level SENINF device
>>> + * @dev: The (platform) device
>>> + * @phy: PHYs at the SENINF inputs
>>> + * @num_clks: Number of clocks in the clks array
>>> + * @clks: Clocks
>>> + * @base: Memory mapped I/O base address
>>> + * @media_dev: Media controller device
>>> + * @v4l2_dev: V4L2 device
>>> + * @subdev: V4L2 subdevice
>>> + * @pads: Media entity pads
>>> + * @notifier: V4L2 async notifier for source subdevs
>>> + * @ctrl_handler: V4L2 controls handler
>>> + * @source_format: Active format on the source pad
>>> + * @inputs: Array of SENINF inputs
>>> + * @muxes: Array of MUXes
>>> + * @conf: Model-specific SENINF parameters
>>> + * @is_testmode: Whether or not the test pattern generator is enabled
>>> + */
>>> +struct mtk_seninf {
>>> + struct device *dev;
>>> + struct phy *phy[5];
>>> + unsigned int num_clks;
>>> + struct clk_bulk_data *clks;
>>> + void __iomem *base;
>>> +
>>> + struct media_device media_dev;
>>> + struct v4l2_device v4l2_dev;
>>> + struct v4l2_subdev subdev;
>>> + struct media_pad pads[SENINF_MAX_NUM_PADS];
>>> + struct v4l2_async_notifier notifier;
>>> + struct v4l2_ctrl_handler ctrl_handler;
>>> +
>>> + struct mtk_seninf_input inputs[SENINF_MAX_NUM_INPUTS];
>>> + struct mtk_seninf_mux muxes[SENINF_MAX_NUM_MUXES];
>>> +
>>> + const struct mtk_seninf_conf *conf;
>>> +
>>> + bool is_testmode;
>>> +};
>>> +
>>> +inline struct mtk_seninf *sd_to_mtk_seninf(struct v4l2_subdev *sd)
>>> +{
>>> + return container_of(sd, struct mtk_seninf, subdev);
>>> +}
>>> +
>>> +static inline bool mtk_seninf_pad_is_sink(struct mtk_seninf *priv,
>>> + unsigned int pad)
>>> +{
>>> + return pad < priv->conf->nb_inputs;
>>> +}
>>> +
>>> +static inline bool mtk_seninf_pad_is_source(struct mtk_seninf *priv,
>>> + unsigned int pad)
>>> +{
>>> + return !mtk_seninf_pad_is_sink(priv, pad);
>>> +}
>>> +
>>> +/* -----------------------------------------------------------------------------
>>> + * Formats
>>> + */
>>> +
>>> +static const struct mtk_seninf_format_info mtk_seninf_formats[] = {
>>> + {
>>> + .code = MEDIA_BUS_FMT_SBGGR8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGBRG8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGRBG8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SRGGB8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGRBG10_1X10,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SRGGB10_1X10,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SBGGR10_1X10,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGBRG10_1X10,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SBGGR12_1X12,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGBRG12_1X12,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGRBG12_1X12,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SRGGB12_1X12,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SBGGR14_1X14,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGBRG14_1X14,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGRBG14_1X14,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SRGGB14_1X14,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SBGGR16_1X16,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGBRG16_1X16,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGRBG16_1X16,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SRGGB16_1X16,
>>> + .flags = MTK_SENINF_FORMAT_BAYER,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_UYVY8_1X16,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_VYUY8_1X16,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_YUYV8_1X16,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_YVYU8_1X16,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_JPEG_1X8,
>>> + .flags = MTK_SENINF_FORMAT_JPEG,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_S5C_UYVY_JPEG_1X8,
>>> + .flags = MTK_SENINF_FORMAT_JPEG,
>>> + },
>>> + /* Keep the input-only formats last. */
>>
>> Your comment doesn't make me understand why input-only formats shall be
>> placed last - and makes me think that having two arrays (one for both
>> and one for input only) would be easier and less error prone, other than
>> making you able to drop the MTK_SENINF_FORMAT_INPUT_ONLY flag entirely.
>>
>>> + {
>>> + .code = MEDIA_BUS_FMT_SGRBG10_DPCM8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_DPCM | MTK_SENINF_FORMAT_INPUT_ONLY,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SRGGB10_DPCM8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_DPCM | MTK_SENINF_FORMAT_INPUT_ONLY,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SBGGR10_DPCM8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_DPCM | MTK_SENINF_FORMAT_INPUT_ONLY,
>>> + }, {
>>> + .code = MEDIA_BUS_FMT_SGBRG10_DPCM8_1X8,
>>> + .flags = MTK_SENINF_FORMAT_DPCM | MTK_SENINF_FORMAT_INPUT_ONLY,
>>> + }
>>> +};
>>> +
>>> +static const struct mtk_seninf_format_info *mtk_seninf_format_info(u32 code)
>>> +{
>>> + unsigned int i;
>>> +
>>> + for (i = 0; i < ARRAY_SIZE(mtk_seninf_formats); ++i) {
>>> + if (mtk_seninf_formats[i].code == code)
>>> + return &mtk_seninf_formats[i];
>>> + }
>>> +
>>> + return NULL;
>>> +}
>>> +
>>
>> ..snip..
>>
>>> +
>>> +static void mtk_seninf_input_setup_csi2(struct mtk_seninf_input *input,
>>> + struct v4l2_subdev_state *state)
>>> +{
>>> + const struct mtk_seninf_format_info *fmtinfo;
>>> + const struct v4l2_mbus_framefmt *format;
>>> + unsigned int num_data_lanes = input->bus.num_data_lanes;
>>> + unsigned int val = 0;
>>> +
>>> + format = v4l2_subdev_state_get_stream_format(state, input->pad, 0);
>>> + fmtinfo = mtk_seninf_format_info(format->code);
>>> +
>>> + /* Configure timestamp */
>>> + writel(SENINF_TIMESTAMP_STEP, input->base + SENINF_TG1_TM_STP);
>>> +
>>> + /* HQ */
>>> + writel(0x0, input->base + SENINF_TG1_PH_CNT);
>>
>> Zero means:
>> - Sensor master clock: ISP_CLK
>> - Sensor clock polarity: Rising edge
>> - Sensor reset deasserted
>> - Sensor powered up
>> - Pixel clock inversion disabled
>> - Sensor master clock polarity disabled
>> - Phase counter disabled
>>
>>> + writel(0x10001, input->base + SENINF_TG1_SEN_CK);
>>
>> Unroll this one... this is the TG1 sensor clock divider.
>>
>> CLKFL GENMASK(5, 0)
>> CLKRS GENMASK(13, 8)
>> CLKCNT GENMASK(21,16)
>>
>> Like this, I don't get what you're trying to set, because you're using a fixed
>> sensor clock rate, meaning that only a handful of camera sensors will be usable.
>>
>> Is this 8Mhz? 16? 24? what? :-)
>>
>> Two hints:
>> - sensor_clk = clk_get_rate(isp_clk) / (tg1_sen_ck_clkcnt + 1);
>> - int mtk_seninf_set_sensor_clk(u8 rate_mhz);
>>
>> Please :-)
>
> Hi Angelo,
>
> I think I get your point about not hardcoding the sensor rate, but I
> am not sure how to use
> a mtk_seninf_set_sensor_clk(u8 rate_mhz); function.
>
> Where would it be called? How is it exposed to the user?
>
As for where: setup, streaming start, resolution change (which may be covered
by streaming start anyway, as a change should be calling stop->start anyway).
And for the how is it exposed to the user - well, depends what you mean for user,
but it's all standard V4L2 API :-)
Last but not least, I can give you another hint....
struct media_entity *entity = (something_here);
struct media_pad *mpad;
struct v4l2_subdev *cam_subdev;
struct v4l2_ctrl *ctl;
s64 link_frequency, pixel_clock;
if (entity->pads[0].flags & MEDIA_PAD_FL_SINK)
return -E_NOT_A_CAMERA_SENSOR_WE_IGNORE_THIS_ONE;
pad = media_pad_remote_pad_first(&entity->pads[0]);
if (!pad)
return -ENOENT;
if (!is_media_entity_v4l2_subdev(pad->entity))
return -ENOENT;
if (pad->entity->function != MEDIA_ENT_F_CAM_SENSOR)
return -ENOENT;
cam_subdev = media_entity_to_v4l2_subdev(pad->entity);
ctl = v4l2_ctrl_find(subdev->ctrl_handler, V4L2_CID_PIXEL_RATE);
/* multiplier is usually bits per pixel, divider is usually num of lanes */
link_frequency = v4l2_get_link_freq(cam_subdev->ctrl_handler, multiplier, divider);
pixel_clock = v4l2_ctrl_g_ctrl_int64(ctl);
....now you know what the sensor wants, set the seninf sensor clock accordingly.
Cheers
Angelo
> Cheers
> Julien
>
>>
>>> +
>>> + /* First Enable Sensor interface and select pad (0x1a04_0200) */
>>> + mtk_seninf_input_update(input, SENINF_CTRL, SENINF_EN, 1);
>>> + mtk_seninf_input_update(input, SENINF_CTRL, PAD2CAM_DATA_SEL, SENINF_PAD_10BIT);
>>> + mtk_seninf_input_update(input, SENINF_CTRL, SENINF_SRC_SEL, 0);
>>> + mtk_seninf_input_update(input, SENINF_CTRL_EXT, SENINF_CSI2_IP_EN, 1);
>>> + mtk_seninf_input_update(input, SENINF_CTRL_EXT, SENINF_NCSI2_IP_EN, 0);
>>> +
>>> + /* DPCM Enable */
>>> + if (fmtinfo->flags & MTK_SENINF_FORMAT_DPCM)
>>> + val = SENINF_CSI2_DPCM_DI_2A_DPCM_EN;
>>> + else
>>> + val = SENINF_CSI2_DPCM_DI_30_DPCM_EN;
>>> + writel(val, input->base + SENINF_CSI2_DPCM);
>>> +
>>> + /* Settle delay */
>>> + mtk_seninf_input_update(input, SENINF_CSI2_LNRD_TIMING,
>>> + DATA_SETTLE_PARAMETER, SENINF_SETTLE_DELAY);
>>> +
>>> + /* HQ */
>>> + writel(0x10, input->base + SENINF_CSI2_LNRC_FSM);
>>
>> As far as I know, SENINF_CSI2_LNRC_FSM is a read-only register: this write will do
>> exactly nothing...
>>
>>> +
>>> + /* CSI2 control */
>>> + val = readl(input->base + SENINF_CSI2_CTL)
>>> + | (FIELD_PREP(SENINF_CSI2_CTL_ED_SEL, DATA_HEADER_ORDER_DI_WCL_WCH)
>>> + | SENINF_CSI2_CTL_CLOCK_LANE_EN | (BIT(num_data_lanes) - 1));
>>> + writel(val, input->base + SENINF_CSI2_CTL);
>>> +
>>> + mtk_seninf_input_update(input, SENINF_CSI2_RESYNC_MERGE_CTL,
>>> + BYPASS_LANE_RESYNC, 0);
>>
>> 93 columns: fits in one line (not only this one!).
>>
>>> + mtk_seninf_input_update(input, SENINF_CSI2_RESYNC_MERGE_CTL, CDPHY_SEL, 0);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_RESYNC_MERGE_CTL,
>>> + CPHY_LANE_RESYNC_CNT, 3);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_MODE, CSR_CSI2_MODE, 0);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_MODE, CSR_CSI2_HEADER_LEN, 0);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_DPHY_SYNC, SYNC_SEQ_MASK_0, 0xff00);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_DPHY_SYNC, SYNC_SEQ_PAT_0, 0x001d);
>>> +
>>> + mtk_seninf_input_update(input, SENINF_CSI2_CTL, CLOCK_HS_OPTION, 0);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_CTL, HSRX_DET_EN, 0);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_CTL, HS_TRAIL_EN, 1);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_HS_TRAIL, HS_TRAIL_PARAMETER,
>>> + SENINF_HS_TRAIL_PARAMETER);
>>> +
>>> + /* Set debug port to output packet number */
>>> + mtk_seninf_input_update(input, SENINF_CSI2_DGB_SEL, DEBUG_EN, 1);
>>> + mtk_seninf_input_update(input, SENINF_CSI2_DGB_SEL, DEBUG_SEL, 0x1a);
>>> +
>>> + /* HQ */
>>> + writel(0xfffffffe, input->base + SENINF_CSI2_SPARE0);
>>
>> I have no idea what this SPARE0 does, but I think that this is something that you
>> want to get from platform_data, as I guess this would be different on various SoCs?
>>
>>> +
>>> + /* Enable CSI2 IRQ mask */
>>> + /* Turn on all interrupt */
>>> + writel(0xffffffff, input->base + SENINF_CSI2_INT_EN);
>>> + /* Write clear CSI2 IRQ */
>>> + writel(0xffffffff, input->base + SENINF_CSI2_INT_STATUS);
>>> + /* Enable CSI2 Extend IRQ mask */
>>
>> You missed:
>> writel(0xffffffff, input->base + SENINF_CSI2_INT_EN_EXT);
>>
>> P.S.: #define SENINF_CSI2_INT_EN_EXT 0x0b10
>>
>>
>>> + /* Turn on all interrupt */
>>
>> /* Reset the CSI2 to commit changes */ <-- makes more sense, doesn't it?
>>
>>> + mtk_seninf_input_update(input, SENINF_CTRL, CSI2_SW_RST, 1);
>>> + udelay(1);
>>> + mtk_seninf_input_update(input, SENINF_CTRL, CSI2_SW_RST, 0);
>>> +}
>>> +
>>> +static void mtk_seninf_mux_setup(struct mtk_seninf_mux *mux,
>>> + struct mtk_seninf_input *input,
>>> + struct v4l2_subdev_state *state)
>>> +{
>>> + const struct mtk_seninf_format_info *fmtinfo;
>>> + const struct v4l2_mbus_framefmt *format;
>>> + unsigned int pix_sel_ext;
>>> + unsigned int pix_sel;
>>> + unsigned int hs_pol = 0;
>>> + unsigned int vs_pol = 0;
>>> + unsigned int val;
>>> + u32 rst_mask;
>>> +
>>> + format = v4l2_subdev_state_get_stream_format(state, input->pad, 0);
>>> + fmtinfo = mtk_seninf_format_info(format->code);
>>> +
>>> + /* Enable mux */
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_MUX_EN, 1);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_SRC_SEL, SENINF_MIPI_SENSOR);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL_EXT, SENINF_SRC_SEL_EXT, SENINF_NORMAL_MODEL);
>>> +
>>> + pix_sel_ext = 0;
>>> + pix_sel = 1;
>>
>>
>> pixels_per_cycle = 1;
>> bus_width = pixels_per_cycle >> 1;
>>
>> because: 0 == 1pix/cyc, 1 == 2pix/cyc, 2 == 4pix/cyc, 3 == 8pix... etc
>> ...but the width of this register depends on the SoC, so you also want to set
>> constraints to the bus width on a per-soc basis (platform data again, or at
>> least leave a comment here).
>>
>> mtk_seninf_mux_update(.... PIX_SEL_EXT, bus_width);
>> mtk_seninf_mux_update(.... PIX_SEL, bus_width);
>>
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL_EXT, SENINF_PIX_SEL_EXT, pix_sel_ext);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_PIX_SEL, pix_sel);
>>> +
>>> + if (fmtinfo->flags & MTK_SENINF_FORMAT_JPEG) {
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_FULL_WR_EN, 0);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_FLUSH_EN,
>>> + FIFO_FLUSH_EN_JPEG_2_PIXEL_MODE);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_PUSH_EN,
>>> + FIFO_PUSH_EN_JPEG_2_PIXEL_MODE);
>>> + } else {
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_FULL_WR_EN, 2);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_FLUSH_EN,
>>> + FIFO_FLUSH_EN_NORMAL_MODE);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_PUSH_EN,
>>> + FIFO_PUSH_EN_NORMAL_MODE);
>>> + }
>>> +
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_HSYNC_POL, hs_pol);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_VSYNC_POL, vs_pol);
>>> +
>>> + val = mtk_seninf_mux_read(mux, SENINF_MUX_CTRL);
>>> + rst_mask = SENINF_MUX_CTRL_SENINF_IRQ_SW_RST | SENINF_MUX_CTRL_SENINF_MUX_SW_RST;
>>> +
>>> + mtk_seninf_mux_write(mux, SENINF_MUX_CTRL, val | rst_mask);
>>
>> Are you sure that you don't need any wait between assertion and deassertion of RST?
>> Looks strange, but I don't really know then.
>>
>>> + mtk_seninf_mux_write(mux, SENINF_MUX_CTRL, val & ~rst_mask);
>>> +
>>> + /* HQ */
>>> + mtk_seninf_mux_write(mux, SENINF_MUX_SPARE, 0xc2000);
>>
>> val = SENINF_FIFO_FULL_SEL;
>>
>> /* SPARE field meaning is unknown */
>> val |= 0xc0000;
>>
>> mtk_seninf_mux_write(mux, SENINF_MUX_SPARE, val);
>>
>>> +}
>>> +
>>> +static void mtk_seninf_top_mux_setup(struct mtk_seninf *priv,
>>> + enum mtk_seninf_id seninf_id,
>>> + struct mtk_seninf_mux *mux)
>>> +{
>>> + unsigned int val;
>>> +
>>> + /*
>>> + * Use the top mux (from SENINF input to MUX) to configure routing, and
>>> + * hardcode a 1:1 mapping from the MUX instances to the SENINF outputs.
>>> + */
>>> + val = readl(priv->base + SENINF_TOP_MUX_CTRL)
>>> + & ~(0xf << (mux->mux_id * 4));
>>> + val |= (seninf_id & 0xf) << (mux->mux_id * 4);
>>> + writel(val, priv->base + SENINF_TOP_MUX_CTRL);
>>> +
>>> + writel(0x76541010, priv->base + SENINF_TOP_CAM_MUX_CTRL);
>>
>> Each four bits of TOP_CAM_MUX_CTRL selects between seninf1 to seninf8 muxes, and
>> TOP_MUX_CTRL is laid out in the very same way.
>>
>> This means that if you're calculating a value for TOP_MUX_CTRL, you can do exactly
>> the same for TOP_CAM_MUX_CTRL.
>>
>>> +}
>>> +
>>> +static void seninf_enable_test_pattern(struct mtk_seninf *priv,
>>> + struct v4l2_subdev_state *state)
>>> +{
>>> + struct mtk_seninf_input *input = &priv->inputs[CSI_PORT_0];
>>> + struct mtk_seninf_mux *mux = &priv->muxes[0];
>>> + const struct mtk_seninf_format_info *fmtinfo;
>>> + const struct v4l2_mbus_framefmt *format;
>>> + unsigned int val;
>>> + unsigned int pix_sel_ext;
>>> + unsigned int pix_sel;
>>> + unsigned int hs_pol = 0;
>>> + unsigned int vs_pol = 0;
>>> + unsigned int seninf = 0;
>>> + unsigned int tm_size = 0;
>>> + unsigned int mux_id = mux->mux_id;
>>> +
>>> + format = v4l2_subdev_state_get_stream_format(state, priv->conf->nb_inputs, 0);
>>> + fmtinfo = mtk_seninf_format_info(format->code);
>>> +
>>> + mtk_seninf_update(priv, SENINF_TOP_CTRL, MUX_LP_MODE, 0);
>>> +
>>> + mtk_seninf_update(priv, SENINF_TOP_CTRL, SENINF_PCLK_EN, 1);
>>> + mtk_seninf_update(priv, SENINF_TOP_CTRL, SENINF2_PCLK_EN, 1);
>>> +
>>> + mtk_seninf_input_update(input, SENINF_CTRL, SENINF_EN, 1);
>>> + mtk_seninf_input_update(input, SENINF_CTRL, SENINF_SRC_SEL, 1);
>>> + mtk_seninf_input_update(input, SENINF_CTRL_EXT,
>>> + SENINF_TESTMDL_IP_EN, 1);
>>> +
>>> + mtk_seninf_input_update(input, SENINF_TG1_TM_CTL, TM_EN, 1);
>>> + mtk_seninf_input_update(input, SENINF_TG1_TM_CTL, TM_PAT, 0xc);
>>> + mtk_seninf_input_update(input, SENINF_TG1_TM_CTL, TM_VSYNC, 4);
>>> + mtk_seninf_input_update(input, SENINF_TG1_TM_CTL, TM_DUMMYPXL, 0x28);
>>> +
>>> + if (fmtinfo->flags & MTK_SENINF_FORMAT_BAYER)
>>> + mtk_seninf_input_update(input, SENINF_TG1_TM_CTL, TM_FMT, 0x0);
>>> + else
>>> + mtk_seninf_input_update(input, SENINF_TG1_TM_CTL, TM_FMT, 0x1);
>>> +
>>> + tm_size = FIELD_PREP(SENINF_TG1_TM_SIZE_TM_LINE, format->height + 8);
>>> + switch (format->code) {
>>> + case MEDIA_BUS_FMT_UYVY8_1X16:
>>> + case MEDIA_BUS_FMT_VYUY8_1X16:
>>> + case MEDIA_BUS_FMT_YUYV8_1X16:
>>> + case MEDIA_BUS_FMT_YVYU8_1X16:
>>> + tm_size |= FIELD_PREP(SENINF_TG1_TM_SIZE_TM_PXL, format->width * 2);
>>> + break;
>>> + default:
>>> + tm_size |= FIELD_PREP(SENINF_TG1_TM_SIZE_TM_PXL, format->width);
>>> + break;
>>> + }
>>> + writel(tm_size, input->base + SENINF_TG1_TM_SIZE);
>>> +
>>> + writel(TEST_MODEL_CLK_DIVIDED_CNT, input->base + SENINF_TG1_TM_CLK);
>>> + writel(TIME_STAMP_DIVIDER, input->base + SENINF_TG1_TM_STP);
>>> +
>>> + /* Set top mux */
>>> + val = (readl(priv->base + SENINF_TOP_MUX_CTRL) & (~(0xf << (mux_id * 4)))) |
>>> + ((seninf & 0xf) << (mux_id * 4));
>>> + writel(val, priv->base + SENINF_TOP_MUX_CTRL);
>>
>> This is duplicated, and it is the same that you have in mtk_seninf_top_mux_setup()
>>
>>> +
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_MUX_EN, 1);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL_EXT,
>>> + SENINF_SRC_SEL_EXT, SENINF_TEST_MODEL);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_SRC_SEL, 1);
>>> +
>>> + pix_sel_ext = 0;
>>> + pix_sel = 1;
>>> +
>>
>> This is in mtk_seninf_mux_setup(), but if you apply my suggestion, it won't be in
>> there anymore, so you'll call a function here to set the right value :-)
>>
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL_EXT,
>>> + SENINF_PIX_SEL_EXT, pix_sel_ext);
>>> +
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_PIX_SEL, pix_sel);
>>> +
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_PUSH_EN, 0x1f);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_FLUSH_EN, 0x1b);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, FIFO_FULL_WR_EN, 2);
>>> +
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_HSYNC_POL, hs_pol);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_VSYNC_POL, vs_pol);
>>> + mtk_seninf_mux_update(mux, SENINF_MUX_CTRL, SENINF_HSYNC_MASK, 1);
>>> +
>>> + mtk_seninf_mux_write(mux, SENINF_MUX_INTEN,
>>> + SENINF_IRQ_CLR_SEL | SENINF_ALL_ERR_IRQ_EN);
>>> +
>>> + mtk_seninf_mux_write(mux, SENINF_MUX_CTRL,
>>> + mtk_seninf_mux_read(mux, SENINF_MUX_CTRL) |
>>> + SENINF_MUX_CTRL_SENINF_IRQ_SW_RST |
>>> + SENINF_MUX_CTRL_SENINF_MUX_SW_RST);
>>> + udelay(1);
>>> + mtk_seninf_mux_write(mux, SENINF_MUX_CTRL,
>>> + mtk_seninf_mux_read(mux, SENINF_MUX_CTRL) &
>>> + ~(SENINF_MUX_CTRL_SENINF_IRQ_SW_RST |
>>> + SENINF_MUX_CTRL_SENINF_MUX_SW_RST));
>>> +
>>> + //check this
>>> + writel(0x76540010, priv->base + SENINF_TOP_CAM_MUX_CTRL);
>>> +
>>> + dev_dbg(priv->dev, "%s: OK\n", __func__);
>>> +}
>>> +
>>
>> Cheers,
>> Angelo
>>
More information about the Linux-mediatek
mailing list