linux-media.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH] media: imx208: Add imx208 camera sensor driver
@ 2018-07-10  7:15 Ping-chung Chen
  2018-07-10  7:37 ` Yeh, Andy
                   ` (3 more replies)
  0 siblings, 4 replies; 8+ messages in thread
From: Ping-chung Chen @ 2018-07-10  7:15 UTC (permalink / raw)
  To: linux-media; +Cc: sakari.ailus, andy.yeh, tfiga, Chen, Ping-chung

From: "Chen, Ping-chung" <ping-chung.chen@intel.com>

Add a V4L2 sub-device driver for the Sony IMX208 image sensor.
This is a camera sensor using the I2C bus for control and the
CSI-2 bus for data.

Signed-off-by: Ping-Chung Chen <ping-chung.chen@intel.com>
---
 drivers/media/i2c/Kconfig  |  11 +
 drivers/media/i2c/Makefile |   1 +
 drivers/media/i2c/imx208.c | 953 +++++++++++++++++++++++++++++++++++++++++++++
 3 files changed, 965 insertions(+)
 create mode 100644 drivers/media/i2c/imx208.c

diff --git a/drivers/media/i2c/Kconfig b/drivers/media/i2c/Kconfig
index 8669853..1c739f4 100644
--- a/drivers/media/i2c/Kconfig
+++ b/drivers/media/i2c/Kconfig
@@ -595,6 +595,17 @@ config VIDEO_APTINA_PLL
 config VIDEO_SMIAPP_PLL
 	tristate
 
+config VIDEO_IMX208
+	tristate "Sony IMX208 sensor support"
+	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API
+	depends on MEDIA_CAMERA_SUPPORT
+	---help---
+	  This is a Video4Linux2 sensor-level driver for the Sony
+	  IMX208 camera.
+
+          To compile this driver as a module, choose M here: the
+          module will be called imx208.
+
 config VIDEO_IMX258
 	tristate "Sony IMX258 sensor support"
 	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API
diff --git a/drivers/media/i2c/Makefile b/drivers/media/i2c/Makefile
index 837c428..47604c2 100644
--- a/drivers/media/i2c/Makefile
+++ b/drivers/media/i2c/Makefile
@@ -104,6 +104,7 @@ obj-$(CONFIG_VIDEO_I2C)		+= video-i2c.o
 obj-$(CONFIG_VIDEO_ML86V7667)	+= ml86v7667.o
 obj-$(CONFIG_VIDEO_OV2659)	+= ov2659.o
 obj-$(CONFIG_VIDEO_TC358743)	+= tc358743.o
+obj-$(CONFIG_VIDEO_IMX208)	+= imx208.o
 obj-$(CONFIG_VIDEO_IMX258)	+= imx258.o
 obj-$(CONFIG_VIDEO_IMX274)	+= imx274.o
 
diff --git a/drivers/media/i2c/imx208.c b/drivers/media/i2c/imx208.c
new file mode 100644
index 0000000..9af9043
--- /dev/null
+++ b/drivers/media/i2c/imx208.c
@@ -0,0 +1,953 @@
+// SPDX-License-Identifier: GPL-2.0
+// Copyright (C) 2018 Intel Corporation
+
+#include <linux/acpi.h>
+#include <linux/delay.h>
+#include <linux/i2c.h>
+#include <linux/module.h>
+#include <linux/pm_runtime.h>
+#include <media/v4l2-ctrls.h>
+#include <media/v4l2-device.h>
+#include <asm/unaligned.h>
+
+#ifndef V4L2_CID_DIGITAL_GAIN
+#define V4L2_CID_DIGITAL_GAIN V4L2_CID_GAIN
+#endif
+
+#define IMX208_REG_VALUE_08BIT		1
+#define IMX208_REG_VALUE_16BIT		2
+#define IMX208_REG_VALUE_24BIT		3
+
+#define IMX208_REG_MODE_SELECT		0x0100
+#define IMX208_MODE_STANDBY		0x00
+#define IMX208_MODE_STREAMING		0x01
+
+/* Chip ID */
+#define IMX208_REG_CHIP_ID		0x0000
+#define IMX208_CHIP_ID			0x0208
+
+/* V_TIMING internal */
+#define IMX208_REG_VTS			0x0340
+#define IMX208_VTS_60FPS		0x0472
+#define IMX208_VTS_BINNING	0x0239
+#define IMX208_VTS_60FPS_MIN		0x0458
+#define IMX208_VTS_BINNING_MIN	0x0230
+#define IMX208_VTS_MAX			0xffff
+
+/* HBLANK control - read only */
+#define IMX208_PPL_384MHZ		1124
+#define IMX208_PPL_96MHZ		1124
+
+/* Exposure control */
+#define IMX208_REG_EXPOSURE		0x0202
+#define IMX208_EXPOSURE_MIN		4
+#define IMX208_EXPOSURE_STEP		1
+#define IMX208_EXPOSURE_DEFAULT		0x190
+#define IMX208_EXPOSURE_MAX		65535
+
+/* Analog gain control */
+#define IMX208_REG_ANALOG_GAIN		0x0204
+#define IMX208_ANA_GAIN_MIN		0
+#define IMX208_ANA_GAIN_MAX		0x00e0
+#define IMX208_ANA_GAIN_STEP		1
+#define IMX208_ANA_GAIN_DEFAULT		0x0
+
+/* Digital gain control */
+#define IMX208_REG_GR_DIGITAL_GAIN	0x020e
+#define IMX208_REG_R_DIGITAL_GAIN	0x0210
+#define IMX208_REG_B_DIGITAL_GAIN	0x0212
+#define IMX208_REG_GB_DIGITAL_GAIN	0x0214
+#define IMX208_DGTL_GAIN_MIN		0
+#define IMX208_DGTL_GAIN_MAX		4096   /* Max = 0xFFF */
+#define IMX208_DGTL_GAIN_DEFAULT	0x100
+#define IMX208_DGTL_GAIN_STEP           1
+
+/* Orientation */
+#define REG_MIRROR_FLIP_CONTROL	0x0101
+#define REG_CONFIG_MIRROR_FLIP		0x03
+
+#define IMX208_REG_TEST_PATTERN_MODE	0x0600
+
+struct imx208_reg {
+	u16 address;
+	u8 val;
+};
+
+struct imx208_reg_list {
+	u32 num_of_regs;
+	const struct imx208_reg *regs;
+};
+
+/* Link frequency config */
+struct imx208_link_freq_config {
+	u32 pixels_per_line;
+
+	/* PLL registers for this link frequency */
+	struct imx208_reg_list reg_list;
+};
+
+/* Mode : resolution and related config&values */
+struct imx208_mode {
+	/* Frame width */
+	u32 width;
+	/* Frame height */
+	u32 height;
+
+	/* V-timing */
+	u32 vts_def;
+	u32 vts_min;
+
+	/* Index of Link frequency config to be used */
+	u32 link_freq_index;
+	/* Default register values */
+	struct imx208_reg_list reg_list;
+};
+
+static const struct imx208_reg mipi_data_rate[] = {
+	{0x0305, 0x02},
+	{0x0307, 0x50},
+	{0x303C, 0x3C},
+};
+
+static const struct imx208_reg mode_1936x1096_60fps_regs[] = {
+	{0x0340, 0x04},
+	{0x0341, 0x72},
+	{0x0342, 0x04},
+	{0x0343, 0x64},
+	{0x034C, 0x07},
+	{0x034D, 0x90},
+	{0x034E, 0x04},
+	{0x034F, 0x48},
+	{0x0381, 0x01},
+	{0x0383, 0x01},
+	{0x0385, 0x01},
+	{0x0387, 0x01},
+	{0x3048, 0x00},
+	{0x3050, 0x01},
+	{0x30D5, 0x00},
+	{0x3301, 0x00},
+	{0x3318, 0x62},
+	{0x0202, 0x01},
+	{0x0203, 0x90},
+	{0x0205, 0x00},
+};
+
+static const struct imx208_reg mode_968_548_60fps_regs[] = {
+	{0x0340, 0x02},
+	{0x0341, 0x39},
+	{0x0342, 0x08},
+	{0x0343, 0xC8},
+	{0x034C, 0x03},
+	{0x034D, 0xC8},
+	{0x034E, 0x02},
+	{0x034F, 0x24},
+	{0x0381, 0x01},
+	{0x0383, 0x03},
+	{0x0385, 0x01},
+	{0x0387, 0x03},
+	{0x3048, 0x01},
+	{0x3050, 0x02},
+	{0x30D5, 0x03},
+	{0x3301, 0x10},
+	{0x3318, 0x75},
+	{0x0202, 0x01},
+	{0x0203, 0x90},
+	{0x0205, 0x00},
+};
+
+static const char * const imx208_test_pattern_menu[] = {
+	"Disabled",
+	"Solid Color",
+	"100% Color Bar",
+	"Fade to grey Color Bar",
+	"PN9",
+	"Fixed Pattern1",
+	"Fixed Pattern2",
+	"Fixed Pattern3",
+	"Fixed Pattern4",
+	"Fixed Pattern5",
+	"Fixed Pattern6"
+};
+
+/* Configurations for supported link frequencies */
+#define IMX208_LINK_FREQ_384MHZ	384000000ULL
+#define IMX208_LINK_FREQ_192MHZ	192000000ULL
+#define IMX208_LINK_FREQ_96MHZ  96000000ULL
+#define IMX208_LINK_FREQ_48MHZ  48000000ULL
+
+enum {
+	IMX208_LINK_FREQ_384MHZ_INDEX,
+	IMX208_LINK_FREQ_96MHZ_INDEX,
+};
+
+/*
+ * pixel_rate = link_freq * data-rate * nr_of_lanes / bits_per_sample
+ * data rate => double data rate; number of lanes => 2; bits per pixel => 10
+ */
+static u64 link_freq_to_pixel_rate(u64 f)
+{
+	f *= 2 * 2;
+	do_div(f, 10);
+
+	return f;
+}
+
+/* Menu items for LINK_FREQ V4L2 control */
+static const s64 link_freq_menu_items[] = {
+	IMX208_LINK_FREQ_384MHZ,
+	IMX208_LINK_FREQ_96MHZ,
+};
+
+/* Link frequency configs */
+static const struct imx208_link_freq_config link_freq_configs[] = {
+	{
+		.pixels_per_line = IMX208_PPL_384MHZ,
+		.reg_list = {
+			.num_of_regs = ARRAY_SIZE(mipi_data_rate),
+			.regs = mipi_data_rate,
+		}
+	},
+	{
+		.pixels_per_line = IMX208_PPL_96MHZ,
+		.reg_list = {
+			.num_of_regs = ARRAY_SIZE(mipi_data_rate),
+			.regs = mipi_data_rate,
+		}
+	},
+};
+
+/* Mode configs */
+static const struct imx208_mode supported_modes[] = {
+	{
+		.width = 1936,
+		.height = 1096,
+		.vts_def = IMX208_VTS_60FPS,
+		.vts_min = IMX208_VTS_60FPS_MIN,
+		.reg_list = {
+			.num_of_regs = ARRAY_SIZE(mode_1936x1096_60fps_regs),
+			.regs = mode_1936x1096_60fps_regs,
+		},
+		.link_freq_index = 0,
+	},
+	{
+		.width = 968,
+		.height = 548,
+		.vts_def = IMX208_VTS_BINNING,
+		.vts_min = IMX208_VTS_BINNING_MIN,
+		.reg_list = {
+			.num_of_regs = ARRAY_SIZE(mode_968_548_60fps_regs),
+			.regs = mode_968_548_60fps_regs,
+		},
+		.link_freq_index = IMX208_LINK_FREQ_96MHZ_INDEX,
+	},
+};
+
+struct imx208 {
+	struct v4l2_subdev sd;
+	struct media_pad pad;
+
+	struct v4l2_ctrl_handler ctrl_handler;
+	/* V4L2 Controls */
+	struct v4l2_ctrl *link_freq;
+	struct v4l2_ctrl *pixel_rate;
+	struct v4l2_ctrl *vblank;
+	struct v4l2_ctrl *hblank;
+	struct v4l2_ctrl *exposure;
+
+	/* Current mode */
+	const struct imx208_mode *cur_mode;
+
+	/* Mutex for serialized access */
+	struct mutex mutex;
+
+	/* Streaming on/off */
+	bool streaming;
+};
+
+static inline struct imx208 *to_imx208(struct v4l2_subdev *_sd)
+{
+	return container_of(_sd, struct imx208, sd);
+}
+
+/* Read registers up to 2 at a time */
+static int imx208_read_reg(struct imx208 *imx208, u16 reg, u32 len, u32 *val)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	struct i2c_msg msgs[2];
+	u8 addr_buf[2] = { reg >> 8, reg & 0xff };
+	u8 data_buf[4] = { 0, };
+	int ret;
+
+	if (len > 4)
+		return -EINVAL;
+
+	/* Write register address */
+	msgs[0].addr = client->addr;
+	msgs[0].flags = 0;
+	msgs[0].len = ARRAY_SIZE(addr_buf);
+	msgs[0].buf = addr_buf;
+
+	/* Read data from register */
+	msgs[1].addr = client->addr;
+	msgs[1].flags = I2C_M_RD;
+	msgs[1].len = len;
+	msgs[1].buf = &data_buf[4 - len];
+
+	ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
+	if (ret != ARRAY_SIZE(msgs))
+		return -EIO;
+
+	*val = get_unaligned_be32(data_buf);
+
+	return 0;
+}
+
+/* Write registers up to 4 at a time */
+static int imx208_write_reg(struct imx208 *imx208, u16 reg, u32 len, u32 val)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	u8 buf[6];
+
+	if (len > 4)
+		return -EINVAL;
+
+	put_unaligned_be16(reg, buf);
+	put_unaligned_be32(val << (8 * (4 - len)), buf + 2);
+	if (i2c_master_send(client, buf, len + 2) != len + 2)
+		return -EIO;
+
+	return 0;
+}
+
+/* Write a list of registers */
+static int imx208_write_regs(struct imx208 *imx208,
+			      const struct imx208_reg *regs, u32 len)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	unsigned int i;
+	int ret;
+
+	for (i = 0; i < len; i++) {
+		ret = imx208_write_reg(imx208, regs[i].address, 1,
+					regs[i].val);
+		if (ret) {
+			dev_err_ratelimited(
+				&client->dev,
+				"Failed to write reg 0x%4.4x. error = %d\n",
+				regs[i].address, ret);
+
+			return ret;
+		}
+	}
+
+	return 0;
+}
+
+/* Open sub-device */
+static int imx208_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh)
+{
+	struct v4l2_mbus_framefmt *try_fmt =
+		v4l2_subdev_get_try_format(sd, fh->pad, 0);
+
+	/* Initialize try_fmt */
+	try_fmt->width = supported_modes[0].width;
+	try_fmt->height = supported_modes[0].height;
+	try_fmt->code = MEDIA_BUS_FMT_SRGGB10_1X10;
+	try_fmt->field = V4L2_FIELD_NONE;
+
+	return 0;
+}
+
+
+static int imx208_update_digital_gain(struct imx208 *imx208, u32 len, u32 val)
+{
+	int ret;
+
+	ret = imx208_write_reg(imx208, IMX208_REG_GR_DIGITAL_GAIN,
+				IMX208_REG_VALUE_16BIT,
+				val);
+	if (ret)
+		return ret;
+	ret = imx208_write_reg(imx208, IMX208_REG_GB_DIGITAL_GAIN,
+				IMX208_REG_VALUE_16BIT,
+				val);
+	if (ret)
+		return ret;
+	ret = imx208_write_reg(imx208, IMX208_REG_R_DIGITAL_GAIN,
+				IMX208_REG_VALUE_16BIT,
+				val);
+	if (ret)
+		return ret;
+	ret = imx208_write_reg(imx208, IMX208_REG_B_DIGITAL_GAIN,
+				IMX208_REG_VALUE_16BIT,
+				val);
+	if (ret)
+		return ret;
+	return 0;
+}
+
+static int imx208_enable_test_pattern(struct imx208 *imx208, u32 pattern)
+{
+	u32 val;
+
+	val = (pattern <= 4) ? pattern : pattern + 0xFB;
+
+	return imx208_write_reg(imx208, IMX208_REG_TEST_PATTERN_MODE,
+				IMX208_REG_VALUE_16BIT, val);
+}
+
+static int imx208_set_ctrl(struct v4l2_ctrl *ctrl)
+{
+	struct imx208 *imx208 =
+		container_of(ctrl->handler, struct imx208, ctrl_handler);
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	int ret = 0;
+	/*
+	 * Applying V4L2 control value only happens
+	 * when power is up for streaming
+	 */
+	if (pm_runtime_get_if_in_use(&client->dev) <= 0)
+		return 0;
+
+	switch (ctrl->id) {
+	case V4L2_CID_ANALOGUE_GAIN:
+		ret = imx208_write_reg(imx208, IMX208_REG_ANALOG_GAIN,
+				IMX208_REG_VALUE_16BIT,
+				ctrl->val);
+		break;
+	case V4L2_CID_EXPOSURE:
+		ret = imx208_write_reg(imx208, IMX208_REG_EXPOSURE,
+				IMX208_REG_VALUE_16BIT,
+				ctrl->val);
+		break;
+	case V4L2_CID_DIGITAL_GAIN:
+		ret = imx208_update_digital_gain(imx208, IMX208_REG_VALUE_16BIT,
+				ctrl->val);
+		break;
+	case V4L2_CID_VBLANK:
+		/* Update VTS that meets expected vertical blanking */
+		ret = imx208_write_reg(imx208, IMX208_REG_VTS,
+				       IMX208_REG_VALUE_16BIT,
+				       imx208->cur_mode->height + ctrl->val);
+		break;
+	case V4L2_CID_TEST_PATTERN:
+		ret = imx208_enable_test_pattern(imx208, ctrl->val);
+		break;
+	default:
+		dev_info(&client->dev,
+			 "ctrl(id:0x%x,val:0x%x) is not handled\n",
+			 ctrl->id, ctrl->val);
+		break;
+	}
+
+	pm_runtime_put(&client->dev);
+
+	return ret;
+}
+
+static const struct v4l2_ctrl_ops imx208_ctrl_ops = {
+	.s_ctrl = imx208_set_ctrl,
+};
+
+static int imx208_enum_mbus_code(struct v4l2_subdev *sd,
+				  struct v4l2_subdev_pad_config *cfg,
+				  struct v4l2_subdev_mbus_code_enum *code)
+{
+	/* Only one bayer order(GRBG) is supported */
+	if (code->index > 0)
+		return -EINVAL;
+
+	code->code = MEDIA_BUS_FMT_SRGGB10_1X10;
+
+	return 0;
+}
+
+static int imx208_enum_frame_size(struct v4l2_subdev *sd,
+				   struct v4l2_subdev_pad_config *cfg,
+				   struct v4l2_subdev_frame_size_enum *fse)
+{
+	if (fse->index >= ARRAY_SIZE(supported_modes))
+		return -EINVAL;
+
+	if (fse->code != MEDIA_BUS_FMT_SRGGB10_1X10)
+		return -EINVAL;
+
+	fse->min_width = supported_modes[fse->index].width;
+	fse->max_width = fse->min_width;
+	fse->min_height = supported_modes[fse->index].height;
+	fse->max_height = fse->min_height;
+
+	return 0;
+}
+
+static void imx208_update_pad_format(const struct imx208_mode *mode,
+				      struct v4l2_subdev_format *fmt)
+{
+	fmt->format.width = mode->width;
+	fmt->format.height = mode->height;
+	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;
+	fmt->format.field = V4L2_FIELD_NONE;
+}
+
+static int __imx208_get_pad_format(struct imx208 *imx208,
+				     struct v4l2_subdev_pad_config *cfg,
+				     struct v4l2_subdev_format *fmt)
+{
+	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY)
+		fmt->format = *v4l2_subdev_get_try_format(&imx208->sd, cfg,
+							  fmt->pad);
+	else
+		imx208_update_pad_format(imx208->cur_mode, fmt);
+
+	return 0;
+}
+
+static int imx208_get_pad_format(struct v4l2_subdev *sd,
+				  struct v4l2_subdev_pad_config *cfg,
+				  struct v4l2_subdev_format *fmt)
+{
+	struct imx208 *imx208 = to_imx208(sd);
+	int ret;
+
+	mutex_lock(&imx208->mutex);
+	ret = __imx208_get_pad_format(imx208, cfg, fmt);
+	mutex_unlock(&imx208->mutex);
+
+	return ret;
+}
+
+static int imx208_set_pad_format(struct v4l2_subdev *sd,
+		       struct v4l2_subdev_pad_config *cfg,
+		       struct v4l2_subdev_format *fmt)
+{
+	struct imx208 *imx208 = to_imx208(sd);
+	const struct imx208_mode *mode;
+	struct v4l2_mbus_framefmt *framefmt;
+	s32 vblank_def;
+	s32 vblank_min;
+	s64 h_blank;
+	s64 pixel_rate;
+	s64 link_freq;
+
+	mutex_lock(&imx208->mutex);
+
+	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;
+
+	mode = v4l2_find_nearest_size(
+		supported_modes, ARRAY_SIZE(supported_modes), width, height,
+		fmt->format.width, fmt->format.height);
+	imx208_update_pad_format(mode, fmt);
+	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) {
+		framefmt = v4l2_subdev_get_try_format(sd, cfg, fmt->pad);
+		*framefmt = fmt->format;
+	} else {
+		imx208->cur_mode = mode;
+		__v4l2_ctrl_s_ctrl(imx208->link_freq, mode->link_freq_index);
+
+		link_freq = link_freq_menu_items[mode->link_freq_index];
+		pixel_rate = link_freq_to_pixel_rate(link_freq);
+		__v4l2_ctrl_s_ctrl_int64(imx208->pixel_rate, pixel_rate);
+		/* Update limits and set FPS to default */
+		vblank_def = imx208->cur_mode->vts_def -
+			     imx208->cur_mode->height;
+		vblank_min = imx208->cur_mode->vts_min -
+			     imx208->cur_mode->height;
+		__v4l2_ctrl_modify_range(
+			imx208->vblank, vblank_min,
+			IMX208_VTS_MAX - imx208->cur_mode->height, 1,
+			vblank_def);
+		__v4l2_ctrl_s_ctrl(imx208->vblank, vblank_def);
+		h_blank =
+			link_freq_configs[mode->link_freq_index].pixels_per_line
+			 - imx208->cur_mode->width;
+		__v4l2_ctrl_modify_range(imx208->hblank, h_blank,
+					 h_blank, 1, h_blank);
+	}
+
+	mutex_unlock(&imx208->mutex);
+
+	return 0;
+}
+
+/* Start streaming */
+static int imx208_start_streaming(struct imx208 *imx208)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	const struct imx208_reg_list *reg_list;
+	int ret, link_freq_index;
+
+	/* Setup PLL */
+	link_freq_index = imx208->cur_mode->link_freq_index;
+	reg_list = &link_freq_configs[link_freq_index].reg_list;
+	ret = imx208_write_regs(imx208, reg_list->regs, reg_list->num_of_regs);
+	if (ret) {
+		dev_err(&client->dev, "%s failed to set plls\n", __func__);
+		return ret;
+	}
+
+	/* Apply default values of current mode */
+	reg_list = &imx208->cur_mode->reg_list;
+	ret = imx208_write_regs(imx208, reg_list->regs, reg_list->num_of_regs);
+	if (ret) {
+		dev_err(&client->dev, "%s failed to set mode\n", __func__);
+		return ret;
+	}
+
+	/* Apply customized values from user */
+	ret =  __v4l2_ctrl_handler_setup(imx208->sd.ctrl_handler);
+	if (ret)
+		return ret;
+
+	/* set stream on register */
+	return imx208_write_reg(imx208, IMX208_REG_MODE_SELECT,
+				 IMX208_REG_VALUE_08BIT,
+				 IMX208_MODE_STREAMING);
+}
+
+/* Stop streaming */
+static int imx208_stop_streaming(struct imx208 *imx208)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	int ret;
+
+	/* set stream off register */
+	ret = imx208_write_reg(imx208, IMX208_REG_MODE_SELECT,
+		IMX208_REG_VALUE_08BIT, IMX208_MODE_STANDBY);
+	if (ret)
+		dev_err(&client->dev, "%s failed to set stream\n", __func__);
+
+	/* Return success even if it was an error, as there is nothing the
+	 * caller can do about it.
+	 */
+	return 0;
+}
+
+static int imx208_set_stream(struct v4l2_subdev *sd, int enable)
+{
+	struct imx208 *imx208 = to_imx208(sd);
+	struct i2c_client *client = v4l2_get_subdevdata(sd);
+	int ret = 0;
+
+	mutex_lock(&imx208->mutex);
+	if (imx208->streaming == enable) {
+		mutex_unlock(&imx208->mutex);
+		return 0;
+	}
+
+	if (enable) {
+		ret = pm_runtime_get_sync(&client->dev);
+		if (ret < 0) {
+			pm_runtime_put_noidle(&client->dev);
+			goto err_unlock;
+		}
+
+		/*
+		 * Apply default & customized values
+		 * and then start streaming.
+		 */
+		ret = imx208_start_streaming(imx208);
+		if (ret)
+			goto err_rpm_put;
+	} else {
+		imx208_stop_streaming(imx208);
+		pm_runtime_put(&client->dev);
+	}
+
+	imx208->streaming = enable;
+	mutex_unlock(&imx208->mutex);
+
+	return ret;
+
+err_rpm_put:
+	pm_runtime_put(&client->dev);
+err_unlock:
+	mutex_unlock(&imx208->mutex);
+
+	return ret;
+}
+
+static int __maybe_unused imx208_suspend(struct device *dev)
+{
+	struct i2c_client *client = to_i2c_client(dev);
+	struct v4l2_subdev *sd = i2c_get_clientdata(client);
+	struct imx208 *imx208 = to_imx208(sd);
+
+	if (imx208->streaming)
+		imx208_stop_streaming(imx208);
+
+	return 0;
+}
+
+static int __maybe_unused imx208_resume(struct device *dev)
+{
+	struct i2c_client *client = to_i2c_client(dev);
+	struct v4l2_subdev *sd = i2c_get_clientdata(client);
+	struct imx208 *imx208 = to_imx208(sd);
+	int ret;
+
+	if (imx208->streaming) {
+		ret = imx208_start_streaming(imx208);
+		if (ret)
+			goto error;
+	}
+
+	return 0;
+
+error:
+	imx208_stop_streaming(imx208);
+	imx208->streaming = 0;
+	return ret;
+}
+
+/* Verify chip ID */
+static int imx208_identify_module(struct imx208 *imx208)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	int ret;
+	u32 val;
+
+	ret = imx208_read_reg(imx208, IMX208_REG_CHIP_ID,
+			       IMX208_REG_VALUE_16BIT, &val);
+	if (ret) {
+		dev_err(&client->dev, "failed to read chip id %x\n",
+			IMX208_CHIP_ID);
+		return ret;
+	}
+
+	if (val != IMX208_CHIP_ID) {
+		dev_err(&client->dev, "chip id mismatch: %x!=%x\n",
+			IMX208_CHIP_ID, val);
+		return -EIO;
+	}
+
+	dev_err(&client->dev, "chip id!: %x %x\n", IMX208_CHIP_ID, val);
+
+	return 0;
+}
+
+static const struct v4l2_subdev_video_ops imx208_video_ops = {
+	.s_stream = imx208_set_stream,
+};
+
+static const struct v4l2_subdev_pad_ops imx208_pad_ops = {
+	.enum_mbus_code = imx208_enum_mbus_code,
+	.get_fmt = imx208_get_pad_format,
+	.set_fmt = imx208_set_pad_format,
+	.enum_frame_size = imx208_enum_frame_size,
+};
+
+static const struct v4l2_subdev_ops imx208_subdev_ops = {
+	.video = &imx208_video_ops,
+	.pad = &imx208_pad_ops,
+};
+
+static const struct v4l2_subdev_internal_ops imx208_internal_ops = {
+	.open = imx208_open,
+};
+
+/* Initialize control handlers */
+static int imx208_init_controls(struct imx208 *imx208)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
+	struct v4l2_ctrl_handler *ctrl_hdlr;
+	s64 exposure_max;
+	s64 vblank_def;
+	s64 vblank_min;
+	s64 pixel_rate_min;
+	s64 pixel_rate_max;
+	int ret;
+
+	ctrl_hdlr = &imx208->ctrl_handler;
+	ret = v4l2_ctrl_handler_init(ctrl_hdlr, 8);
+	if (ret)
+		return ret;
+
+	mutex_init(&imx208->mutex);
+	ctrl_hdlr->lock = &imx208->mutex;
+	imx208->link_freq = v4l2_ctrl_new_int_menu(ctrl_hdlr,
+				&imx208_ctrl_ops,
+				V4L2_CID_LINK_FREQ,
+				ARRAY_SIZE(link_freq_menu_items) - 1,
+				0,
+				link_freq_menu_items);
+
+	if (imx208->link_freq)
+		imx208->link_freq->flags |= V4L2_CTRL_FLAG_READ_ONLY;
+
+	pixel_rate_max = link_freq_to_pixel_rate(link_freq_menu_items[0]);
+	pixel_rate_min = link_freq_to_pixel_rate(link_freq_menu_items[1]);
+	/* By default, PIXEL_RATE is read only */
+	imx208->pixel_rate = v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops,
+					V4L2_CID_PIXEL_RATE,
+					pixel_rate_min, pixel_rate_max,
+					1, pixel_rate_max);
+
+
+	vblank_def = imx208->cur_mode->vts_def - imx208->cur_mode->height;
+	vblank_min = imx208->cur_mode->vts_min - imx208->cur_mode->height;
+	imx208->vblank = v4l2_ctrl_new_std(
+				ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_VBLANK,
+				vblank_min,
+				IMX208_VTS_MAX - imx208->cur_mode->height, 1,
+				vblank_def);
+
+	imx208->hblank = v4l2_ctrl_new_std(
+				ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_HBLANK,
+				IMX208_PPL_384MHZ - imx208->cur_mode->width,
+				IMX208_PPL_384MHZ - imx208->cur_mode->width,
+				1,
+				IMX208_PPL_384MHZ - imx208->cur_mode->width);
+
+	if (imx208->hblank)
+		imx208->hblank->flags |= V4L2_CTRL_FLAG_READ_ONLY;
+
+	exposure_max = imx208->cur_mode->vts_def - 8;
+	imx208->exposure = v4l2_ctrl_new_std(
+				ctrl_hdlr, &imx208_ctrl_ops,
+				V4L2_CID_EXPOSURE, IMX208_EXPOSURE_MIN,
+				IMX208_EXPOSURE_MAX, IMX208_EXPOSURE_STEP,
+				IMX208_EXPOSURE_DEFAULT);
+
+	v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_ANALOGUE_GAIN,
+				IMX208_ANA_GAIN_MIN, IMX208_ANA_GAIN_MAX,
+				IMX208_ANA_GAIN_STEP, IMX208_ANA_GAIN_DEFAULT);
+
+	v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_DIGITAL_GAIN,
+				IMX208_DGTL_GAIN_MIN, IMX208_DGTL_GAIN_MAX,
+				IMX208_DGTL_GAIN_STEP,
+				IMX208_DGTL_GAIN_DEFAULT);
+
+	v4l2_ctrl_new_std_menu_items(ctrl_hdlr, &imx208_ctrl_ops,
+				     V4L2_CID_TEST_PATTERN,
+				     ARRAY_SIZE(imx208_test_pattern_menu) - 1,
+				     0, 0, imx208_test_pattern_menu);
+
+	if (ctrl_hdlr->error) {
+		ret = ctrl_hdlr->error;
+		dev_err(&client->dev, "%s control init failed (%d)\n",
+			__func__, ret);
+		goto error;
+	}
+
+	imx208->sd.ctrl_handler = ctrl_hdlr;
+
+	return 0;
+
+error:
+	v4l2_ctrl_handler_free(ctrl_hdlr);
+	mutex_destroy(&imx208->mutex);
+
+	return ret;
+}
+
+static void imx208_free_controls(struct imx208 *imx208)
+{
+	v4l2_ctrl_handler_free(imx208->sd.ctrl_handler);
+	mutex_destroy(&imx208->mutex);
+}
+
+static int imx208_probe(struct i2c_client *client)
+{
+	struct imx208 *imx208;
+	int ret;
+	u32 val = 0;
+
+	device_property_read_u32(&client->dev, "clock-frequency", &val);
+	if (val != 19200000)
+		return -EINVAL;
+
+	imx208 = devm_kzalloc(&client->dev, sizeof(*imx208), GFP_KERNEL);
+	if (!imx208)
+		return -ENOMEM;
+
+	/* Initialize subdev */
+	v4l2_i2c_subdev_init(&imx208->sd, client, &imx208_subdev_ops);
+
+	/* Check module identity */
+	ret = imx208_identify_module(imx208);
+	if (ret)
+		return ret;
+
+	/* Set default mode to max resolution */
+	imx208->cur_mode = &supported_modes[0];
+
+	ret = imx208_init_controls(imx208);
+	if (ret)
+		return ret;
+
+	/* Initialize subdev */
+	imx208->sd.internal_ops = &imx208_internal_ops;
+	imx208->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
+	imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
+
+	/* Initialize source pad */
+	imx208->pad.flags = MEDIA_PAD_FL_SOURCE;
+	ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
+	if (ret) {
+		dev_err(&client->dev, "%s failed:%d\n", __func__, ret);
+		goto error_handler_free;
+	}
+
+	ret = v4l2_async_register_subdev_sensor_common(&imx208->sd);
+	if (ret < 0)
+		goto error_media_entity;
+
+	pm_runtime_set_active(&client->dev);
+	pm_runtime_enable(&client->dev);
+	pm_runtime_idle(&client->dev);
+
+	return 0;
+
+error_media_entity:
+	media_entity_cleanup(&imx208->sd.entity);
+
+error_handler_free:
+	imx208_free_controls(imx208);
+
+	return ret;
+}
+
+static int imx208_remove(struct i2c_client *client)
+{
+	struct v4l2_subdev *sd = i2c_get_clientdata(client);
+	struct imx208 *imx208 = to_imx208(sd);
+
+	v4l2_async_unregister_subdev(sd);
+	media_entity_cleanup(&sd->entity);
+	imx208_free_controls(imx208);
+
+	pm_runtime_disable(&client->dev);
+	pm_runtime_set_suspended(&client->dev);
+
+	return 0;
+}
+
+static const struct dev_pm_ops imx208_pm_ops = {
+	SET_SYSTEM_SLEEP_PM_OPS(imx208_suspend, imx208_resume)
+};
+
+#ifdef CONFIG_ACPI
+static const struct acpi_device_id imx208_acpi_ids[] = {
+	{ "INT3478" },
+	{ /* sentinel */ }
+};
+
+MODULE_DEVICE_TABLE(acpi, imx208_acpi_ids);
+#endif
+
+static struct i2c_driver imx208_i2c_driver = {
+	.driver = {
+		.name = "imx208",
+		.pm = &imx208_pm_ops,
+		.acpi_match_table = ACPI_PTR(imx208_acpi_ids),
+	},
+	.probe_new = imx208_probe,
+	.remove = imx208_remove,
+};
+
+module_i2c_driver(imx208_i2c_driver);
+
+MODULE_AUTHOR("Yeh, Andy <andy.yeh@intel.com>");
+MODULE_AUTHOR("Chen, Ping-chung <ping-chung.chen@intel.com>");
+MODULE_DESCRIPTION("Sony IMX208 sensor driver");
+MODULE_LICENSE("GPL v2");
-- 
1.9.1

^ permalink raw reply related	[flat|nested] 8+ messages in thread

* RE: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-10  7:15 [PATCH] media: imx208: Add imx208 camera sensor driver Ping-chung Chen
@ 2018-07-10  7:37 ` Yeh, Andy
  2018-07-16 16:03   ` sakari.ailus
  2018-07-10 11:53 ` kbuild test robot
                   ` (2 subsequent siblings)
  3 siblings, 1 reply; 8+ messages in thread
From: Yeh, Andy @ 2018-07-10  7:37 UTC (permalink / raw)
  To: Chen, Ping-chung, linux-media
  Cc: sakari.ailus, tfiga, grundler, Chen, JasonX Z, Lai, Jim

Hi PC,

Thanks for the patch.

Cc in Grant, and Intel Jim/Jason

> -----Original Message-----
> From: Chen, Ping-chung
> Sent: Tuesday, July 10, 2018 3:16 PM
> To: linux-media@vger.kernel.org
> Cc: sakari.ailus@linux.intel.com; Yeh, Andy <andy.yeh@intel.com>;
> tfiga@chromium.org; Chen, Ping-chung <ping-chung.chen@intel.com>
> Subject: [PATCH] media: imx208: Add imx208 camera sensor driver
> +};
> +
> +static int imx208_enum_mbus_code(struct v4l2_subdev *sd,
> +				  struct v4l2_subdev_pad_config *cfg,
> +				  struct v4l2_subdev_mbus_code_enum *code) {
> +	/* Only one bayer order(GRBG) is supported */
> +	if (code->index > 0)
> +		return -EINVAL;
> +

There is no limitation on using GRBG bayer order now. You can refer to imx355 driver as well.

	+static int imx355_enum_frame_size(struct v4l2_subdev *sd,
	+				   struct v4l2_subdev_pad_config *cfg,
	+				   struct v4l2_subdev_frame_size_enum *fse)
	+{	
	+	struct imx355 *imx355 = to_imx355(sd);

> +	code->code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +
> +	return 0;
> +}
> +
> +static int imx208_enum_frame_size(struct v4l2_subdev *sd,
> +				   struct v4l2_subdev_pad_config *cfg,
> +				   struct v4l2_subdev_frame_size_enum *fse) {
> +	if (fse->index >= ARRAY_SIZE(supported_modes))
> +		return -EINVAL;
> +
> +	if (fse->code != MEDIA_BUS_FMT_SRGGB10_1X10)
> +		return -EINVAL;
> +
> +	fse->min_width = supported_modes[fse->index].width;
> +	fse->max_width = fse->min_width;
> +	fse->min_height = supported_modes[fse->index].height;
> +	fse->max_height = fse->min_height;
> +
> +	return 0;
> +}
> +
> +static void imx208_update_pad_format(const struct imx208_mode *mode,
> +				      struct v4l2_subdev_format *fmt) {
> +	fmt->format.width = mode->width;
> +	fmt->format.height = mode->height;
> +	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +	fmt->format.field = V4L2_FIELD_NONE;
> +}
> +
> +
> +static int imx208_probe(struct i2c_client *client) {
> +	struct imx208 *imx208;
> +	int ret;
> +	u32 val = 0;
> +
> +	device_property_read_u32(&client->dev, "clock-frequency", &val);
> +	if (val != 19200000)
> +		return -EINVAL;
> +
> +	imx208 = devm_kzalloc(&client->dev, sizeof(*imx208), GFP_KERNEL);
> +	if (!imx208)
> +		return -ENOMEM;
> +
> +	/* Initialize subdev */
> +	v4l2_i2c_subdev_init(&imx208->sd, client, &imx208_subdev_ops);
> +
> +	/* Check module identity */
> +	ret = imx208_identify_module(imx208);
> +	if (ret)
> +		return ret;
> +
> +	/* Set default mode to max resolution */
> +	imx208->cur_mode = &supported_modes[0];
> +
> +	ret = imx208_init_controls(imx208);
> +	if (ret)
> +		return ret;
> +
> +	/* Initialize subdev */
> +	imx208->sd.internal_ops = &imx208_internal_ops;
> +	imx208->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
> +	imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
> +
> +	/* Initialize source pad */
> +	imx208->pad.flags = MEDIA_PAD_FL_SOURCE;

Please refer to IMX355 to change below code to new API on upstreamed kernel.
https://patchwork.linuxtv.org/patch/50028/

	+	/* Initialize subdev */
	+	imx355->sd.internal_ops = &imx355_internal_ops;
	+	imx355->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE |
	+		V4L2_SUBDEV_FL_HAS_EVENTS;
	+	imx355->sd.entity.ops = &imx355_subdev_entity_ops;
	+	imx355->sd.entity.function = MEDIA_ENT_F_CAM_SENSOR;
	+	ret = media_entity_pads_init(&imx355->sd.entity, 1, &imx355->pad);


> +	ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
> +	if (ret) {
> +		dev_err(&client->dev, "%s failed:%d\n", __func__, ret);
> +		goto error_handler_free;
> +	}
> +
> +	ret = v4l2_async_register_subdev_sensor_common(&imx208->sd);
> +	if (ret < 0)
> +		goto error_media_entity;
> +
> +	pm_runtime_set_active(&client->dev);
> +	pm_runtime_enable(&client->dev);
> +	pm_runtime_idle(&client->dev);
> +
> +	return 0;
> +
> +error_media_entity:
> +	media_entity_cleanup(&imx208->sd.entity);
> +
> +error_handler_free:
> +	imx208_free_controls(imx208);
> +
> +	return ret;
> +}
> +
> +static int imx208_remove(struct i2c_client *client) {
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +
> +	v4l2_async_unregister_subdev(sd);
> +	media_entity_cleanup(&sd->entity);
> +	imx208_free_controls(imx208);
> +
> +	pm_runtime_disable(&client->dev);
> +	pm_runtime_set_suspended(&client->dev);
> +
> +	return 0;
> +}
> +
> +static const struct dev_pm_ops imx208_pm_ops = {
> +	SET_SYSTEM_SLEEP_PM_OPS(imx208_suspend, imx208_resume) };
> +
> +#ifdef CONFIG_ACPI
> +static const struct acpi_device_id imx208_acpi_ids[] = {
> +	{ "INT3478" },
> +	{ /* sentinel */ }
> +};
> +
> +MODULE_DEVICE_TABLE(acpi, imx208_acpi_ids); #endif
> +
> +static struct i2c_driver imx208_i2c_driver = {
> +	.driver = {
> +		.name = "imx208",
> +		.pm = &imx208_pm_ops,
> +		.acpi_match_table = ACPI_PTR(imx208_acpi_ids),
> +	},
> +	.probe_new = imx208_probe,
> +	.remove = imx208_remove,
> +};
> +
> +module_i2c_driver(imx208_i2c_driver);
> +
> +MODULE_AUTHOR("Yeh, Andy <andy.yeh@intel.com>");
> MODULE_AUTHOR("Chen,
> +Ping-chung <ping-chung.chen@intel.com>"); MODULE_DESCRIPTION("Sony
> +IMX208 sensor driver"); MODULE_LICENSE("GPL v2");
> --
> 1.9.1

^ permalink raw reply	[flat|nested] 8+ messages in thread

* Re: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-10  7:15 [PATCH] media: imx208: Add imx208 camera sensor driver Ping-chung Chen
  2018-07-10  7:37 ` Yeh, Andy
@ 2018-07-10 11:53 ` kbuild test robot
  2018-07-10 12:12 ` kbuild test robot
  2018-07-16 15:58 ` Sakari Ailus
  3 siblings, 0 replies; 8+ messages in thread
From: kbuild test robot @ 2018-07-10 11:53 UTC (permalink / raw)
  To: Ping-chung Chen
  Cc: kbuild-all, linux-media, sakari.ailus, andy.yeh, tfiga, Chen, Ping-chung

[-- Attachment #1: Type: text/plain, Size: 5624 bytes --]

Hi Ping-chung,

Thank you for the patch! Yet something to improve:

[auto build test ERROR on linuxtv-media/master]
[also build test ERROR on v4.18-rc4 next-20180709]
[if your patch is applied to the wrong git tree, please drop us a note to help improve the system]

url:    https://github.com/0day-ci/linux/commits/Ping-chung-Chen/media-imx208-Add-imx208-camera-sensor-driver/20180710-153546
base:   git://linuxtv.org/media_tree.git master
config: x86_64-allmodconfig (attached as .config)
compiler: gcc-7 (Debian 7.3.0-16) 7.3.0
reproduce:
        # save the attached .config to linux build tree
        make ARCH=x86_64 

All errors (new ones prefixed by >>):

   drivers/media/i2c/imx208.c:881:26: sparse: no member 'type' in struct media_entity
   drivers/media/i2c/imx208.c:885:15: sparse: undefined identifier 'media_entity_init'
   drivers/media/i2c/imx208.c:881:26: sparse: generating address of non-lvalue (8)
   drivers/media/i2c/imx208.c:885:32: sparse: call with no type!
   drivers/media/i2c/imx208.c: In function 'imx208_probe':
>> drivers/media/i2c/imx208.c:881:20: error: 'struct media_entity' has no member named 'type'; did you mean 'pipe'?
     imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
                       ^~~~
                       pipe
>> drivers/media/i2c/imx208.c:881:27: error: 'MEDIA_ENT_T_V4L2_SUBDEV_SENSOR' undeclared (first use in this function); did you mean 'MEDIA_ENT_F_V4L2_SUBDEV_UNKNOWN'?
     imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
                              ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
                              MEDIA_ENT_F_V4L2_SUBDEV_UNKNOWN
   drivers/media/i2c/imx208.c:881:27: note: each undeclared identifier is reported only once for each function it appears in
>> drivers/media/i2c/imx208.c:885:8: error: implicit declaration of function 'media_entity_init'; did you mean 'media_entity_put'? [-Werror=implicit-function-declaration]
     ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
           ^~~~~~~~~~~~~~~~~
           media_entity_put
   cc1: some warnings being treated as errors

sparse warnings: (new ones prefixed by >>)

   drivers/media/i2c/imx208.c:881:26: sparse: no member 'type' in struct media_entity
   drivers/media/i2c/imx208.c:885:15: sparse: undefined identifier 'media_entity_init'
>> drivers/media/i2c/imx208.c:881:26: sparse: generating address of non-lvalue (8)
>> drivers/media/i2c/imx208.c:885:32: sparse: call with no type!
   drivers/media/i2c/imx208.c: In function 'imx208_probe':
   drivers/media/i2c/imx208.c:881:20: error: 'struct media_entity' has no member named 'type'; did you mean 'pipe'?
     imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
                       ^~~~
                       pipe
   drivers/media/i2c/imx208.c:881:27: error: 'MEDIA_ENT_T_V4L2_SUBDEV_SENSOR' undeclared (first use in this function); did you mean 'MEDIA_ENT_F_V4L2_SUBDEV_UNKNOWN'?
     imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
                              ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
                              MEDIA_ENT_F_V4L2_SUBDEV_UNKNOWN
   drivers/media/i2c/imx208.c:881:27: note: each undeclared identifier is reported only once for each function it appears in
   drivers/media/i2c/imx208.c:885:8: error: implicit declaration of function 'media_entity_init'; did you mean 'media_entity_put'? [-Werror=implicit-function-declaration]
     ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
           ^~~~~~~~~~~~~~~~~
           media_entity_put
   cc1: some warnings being treated as errors

vim +881 drivers/media/i2c/imx208.c

   848	
   849	static int imx208_probe(struct i2c_client *client)
   850	{
   851		struct imx208 *imx208;
   852		int ret;
   853		u32 val = 0;
   854	
   855		device_property_read_u32(&client->dev, "clock-frequency", &val);
   856		if (val != 19200000)
   857			return -EINVAL;
   858	
   859		imx208 = devm_kzalloc(&client->dev, sizeof(*imx208), GFP_KERNEL);
   860		if (!imx208)
   861			return -ENOMEM;
   862	
   863		/* Initialize subdev */
   864		v4l2_i2c_subdev_init(&imx208->sd, client, &imx208_subdev_ops);
   865	
   866		/* Check module identity */
   867		ret = imx208_identify_module(imx208);
   868		if (ret)
   869			return ret;
   870	
   871		/* Set default mode to max resolution */
   872		imx208->cur_mode = &supported_modes[0];
   873	
   874		ret = imx208_init_controls(imx208);
   875		if (ret)
   876			return ret;
   877	
   878		/* Initialize subdev */
   879		imx208->sd.internal_ops = &imx208_internal_ops;
   880		imx208->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
 > 881		imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
   882	
   883		/* Initialize source pad */
   884		imx208->pad.flags = MEDIA_PAD_FL_SOURCE;
 > 885		ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
   886		if (ret) {
   887			dev_err(&client->dev, "%s failed:%d\n", __func__, ret);
   888			goto error_handler_free;
   889		}
   890	
   891		ret = v4l2_async_register_subdev_sensor_common(&imx208->sd);
   892		if (ret < 0)
   893			goto error_media_entity;
   894	
   895		pm_runtime_set_active(&client->dev);
   896		pm_runtime_enable(&client->dev);
   897		pm_runtime_idle(&client->dev);
   898	
   899		return 0;
   900	
   901	error_media_entity:
   902		media_entity_cleanup(&imx208->sd.entity);
   903	
   904	error_handler_free:
   905		imx208_free_controls(imx208);
   906	
   907		return ret;
   908	}
   909	

---
0-DAY kernel test infrastructure                Open Source Technology Center
https://lists.01.org/pipermail/kbuild-all                   Intel Corporation

[-- Attachment #2: .config.gz --]
[-- Type: application/gzip, Size: 64681 bytes --]

^ permalink raw reply	[flat|nested] 8+ messages in thread

* Re: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-10  7:15 [PATCH] media: imx208: Add imx208 camera sensor driver Ping-chung Chen
  2018-07-10  7:37 ` Yeh, Andy
  2018-07-10 11:53 ` kbuild test robot
@ 2018-07-10 12:12 ` kbuild test robot
  2018-07-16 15:58 ` Sakari Ailus
  3 siblings, 0 replies; 8+ messages in thread
From: kbuild test robot @ 2018-07-10 12:12 UTC (permalink / raw)
  To: Ping-chung Chen
  Cc: kbuild-all, linux-media, sakari.ailus, andy.yeh, tfiga, Chen, Ping-chung

[-- Attachment #1: Type: text/plain, Size: 3815 bytes --]

Hi Ping-chung,

Thank you for the patch! Yet something to improve:

[auto build test ERROR on linuxtv-media/master]
[also build test ERROR on v4.18-rc4 next-20180709]
[if your patch is applied to the wrong git tree, please drop us a note to help improve the system]

url:    https://github.com/0day-ci/linux/commits/Ping-chung-Chen/media-imx208-Add-imx208-camera-sensor-driver/20180710-153546
base:   git://linuxtv.org/media_tree.git master
config: i386-allyesconfig (attached as .config)
compiler: gcc-7 (Debian 7.3.0-16) 7.3.0
reproduce:
        # save the attached .config to linux build tree
        make ARCH=i386 

All errors (new ones prefixed by >>):

   drivers/media/i2c/imx208.c: In function 'imx208_probe':
   drivers/media/i2c/imx208.c:881:20: error: 'struct media_entity' has no member named 'type'; did you mean 'pipe'?
     imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
                       ^~~~
                       pipe
>> drivers/media/i2c/imx208.c:881:27: error: 'MEDIA_ENT_T_V4L2_SUBDEV_SENSOR' undeclared (first use in this function); did you mean 'MEDIA_ENTITY_TYPE_V4L2_SUBDEV'?
     imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
                              ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
                              MEDIA_ENTITY_TYPE_V4L2_SUBDEV
   drivers/media/i2c/imx208.c:881:27: note: each undeclared identifier is reported only once for each function it appears in
   drivers/media/i2c/imx208.c:885:8: error: implicit declaration of function 'media_entity_init'; did you mean 'media_entity_put'? [-Werror=implicit-function-declaration]
     ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
           ^~~~~~~~~~~~~~~~~
           media_entity_put
   cc1: some warnings being treated as errors

vim +881 drivers/media/i2c/imx208.c

   848	
   849	static int imx208_probe(struct i2c_client *client)
   850	{
   851		struct imx208 *imx208;
   852		int ret;
   853		u32 val = 0;
   854	
   855		device_property_read_u32(&client->dev, "clock-frequency", &val);
   856		if (val != 19200000)
   857			return -EINVAL;
   858	
   859		imx208 = devm_kzalloc(&client->dev, sizeof(*imx208), GFP_KERNEL);
   860		if (!imx208)
   861			return -ENOMEM;
   862	
   863		/* Initialize subdev */
   864		v4l2_i2c_subdev_init(&imx208->sd, client, &imx208_subdev_ops);
   865	
   866		/* Check module identity */
   867		ret = imx208_identify_module(imx208);
   868		if (ret)
   869			return ret;
   870	
   871		/* Set default mode to max resolution */
   872		imx208->cur_mode = &supported_modes[0];
   873	
   874		ret = imx208_init_controls(imx208);
   875		if (ret)
   876			return ret;
   877	
   878		/* Initialize subdev */
   879		imx208->sd.internal_ops = &imx208_internal_ops;
   880		imx208->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
 > 881		imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;
   882	
   883		/* Initialize source pad */
   884		imx208->pad.flags = MEDIA_PAD_FL_SOURCE;
   885		ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);
   886		if (ret) {
   887			dev_err(&client->dev, "%s failed:%d\n", __func__, ret);
   888			goto error_handler_free;
   889		}
   890	
   891		ret = v4l2_async_register_subdev_sensor_common(&imx208->sd);
   892		if (ret < 0)
   893			goto error_media_entity;
   894	
   895		pm_runtime_set_active(&client->dev);
   896		pm_runtime_enable(&client->dev);
   897		pm_runtime_idle(&client->dev);
   898	
   899		return 0;
   900	
   901	error_media_entity:
   902		media_entity_cleanup(&imx208->sd.entity);
   903	
   904	error_handler_free:
   905		imx208_free_controls(imx208);
   906	
   907		return ret;
   908	}
   909	

---
0-DAY kernel test infrastructure                Open Source Technology Center
https://lists.01.org/pipermail/kbuild-all                   Intel Corporation

[-- Attachment #2: .config.gz --]
[-- Type: application/gzip, Size: 63436 bytes --]

^ permalink raw reply	[flat|nested] 8+ messages in thread

* Re: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-10  7:15 [PATCH] media: imx208: Add imx208 camera sensor driver Ping-chung Chen
                   ` (2 preceding siblings ...)
  2018-07-10 12:12 ` kbuild test robot
@ 2018-07-16 15:58 ` Sakari Ailus
  2018-07-17  6:53   ` Chen, Ping-chung
  3 siblings, 1 reply; 8+ messages in thread
From: Sakari Ailus @ 2018-07-16 15:58 UTC (permalink / raw)
  To: Ping-chung Chen; +Cc: linux-media, andy.yeh, tfiga

Hi Ping-chung,

Thanks for the patch. Please see my comments below.

On Tue, Jul 10, 2018 at 03:15:36PM +0800, Ping-chung Chen wrote:
> From: "Chen, Ping-chung" <ping-chung.chen@intel.com>
> 
> Add a V4L2 sub-device driver for the Sony IMX208 image sensor.
> This is a camera sensor using the I2C bus for control and the
> CSI-2 bus for data.
> 
> Signed-off-by: Ping-Chung Chen <ping-chung.chen@intel.com>
> ---
>  drivers/media/i2c/Kconfig  |  11 +
>  drivers/media/i2c/Makefile |   1 +
>  drivers/media/i2c/imx208.c | 953 +++++++++++++++++++++++++++++++++++++++++++++
>  3 files changed, 965 insertions(+)
>  create mode 100644 drivers/media/i2c/imx208.c
> 
> diff --git a/drivers/media/i2c/Kconfig b/drivers/media/i2c/Kconfig
> index 8669853..1c739f4 100644
> --- a/drivers/media/i2c/Kconfig
> +++ b/drivers/media/i2c/Kconfig
> @@ -595,6 +595,17 @@ config VIDEO_APTINA_PLL
>  config VIDEO_SMIAPP_PLL
>  	tristate
>  
> +config VIDEO_IMX208
> +	tristate "Sony IMX208 sensor support"
> +	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API
> +	depends on MEDIA_CAMERA_SUPPORT
> +	---help---
> +	  This is a Video4Linux2 sensor-level driver for the Sony

s/-level//

> +	  IMX208 camera.
> +
> +          To compile this driver as a module, choose M here: the
> +          module will be called imx208.
> +
>  config VIDEO_IMX258
>  	tristate "Sony IMX258 sensor support"
>  	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API
> diff --git a/drivers/media/i2c/Makefile b/drivers/media/i2c/Makefile
> index 837c428..47604c2 100644
> --- a/drivers/media/i2c/Makefile
> +++ b/drivers/media/i2c/Makefile
> @@ -104,6 +104,7 @@ obj-$(CONFIG_VIDEO_I2C)		+= video-i2c.o
>  obj-$(CONFIG_VIDEO_ML86V7667)	+= ml86v7667.o
>  obj-$(CONFIG_VIDEO_OV2659)	+= ov2659.o
>  obj-$(CONFIG_VIDEO_TC358743)	+= tc358743.o
> +obj-$(CONFIG_VIDEO_IMX208)	+= imx208.o
>  obj-$(CONFIG_VIDEO_IMX258)	+= imx258.o
>  obj-$(CONFIG_VIDEO_IMX274)	+= imx274.o
>  
> diff --git a/drivers/media/i2c/imx208.c b/drivers/media/i2c/imx208.c
> new file mode 100644
> index 0000000..9af9043
> --- /dev/null
> +++ b/drivers/media/i2c/imx208.c
> @@ -0,0 +1,953 @@
> +// SPDX-License-Identifier: GPL-2.0
> +// Copyright (C) 2018 Intel Corporation
> +
> +#include <linux/acpi.h>
> +#include <linux/delay.h>
> +#include <linux/i2c.h>
> +#include <linux/module.h>
> +#include <linux/pm_runtime.h>
> +#include <media/v4l2-ctrls.h>
> +#include <media/v4l2-device.h>
> +#include <asm/unaligned.h>
> +
> +#ifndef V4L2_CID_DIGITAL_GAIN
> +#define V4L2_CID_DIGITAL_GAIN V4L2_CID_GAIN
> +#endif

Please drop this.

> +
> +#define IMX208_REG_VALUE_08BIT		1
> +#define IMX208_REG_VALUE_16BIT		2
> +#define IMX208_REG_VALUE_24BIT		3

The last one is unused.

> +
> +#define IMX208_REG_MODE_SELECT		0x0100
> +#define IMX208_MODE_STANDBY		0x00
> +#define IMX208_MODE_STREAMING		0x01
> +
> +/* Chip ID */
> +#define IMX208_REG_CHIP_ID		0x0000
> +#define IMX208_CHIP_ID			0x0208
> +
> +/* V_TIMING internal */
> +#define IMX208_REG_VTS			0x0340
> +#define IMX208_VTS_60FPS		0x0472
> +#define IMX208_VTS_BINNING	0x0239
> +#define IMX208_VTS_60FPS_MIN		0x0458
> +#define IMX208_VTS_BINNING_MIN	0x0230
> +#define IMX208_VTS_MAX			0xffff
> +
> +/* HBLANK control - read only */
> +#define IMX208_PPL_384MHZ		1124
> +#define IMX208_PPL_96MHZ		1124
> +
> +/* Exposure control */
> +#define IMX208_REG_EXPOSURE		0x0202
> +#define IMX208_EXPOSURE_MIN		4
> +#define IMX208_EXPOSURE_STEP		1
> +#define IMX208_EXPOSURE_DEFAULT		0x190
> +#define IMX208_EXPOSURE_MAX		65535
> +
> +/* Analog gain control */
> +#define IMX208_REG_ANALOG_GAIN		0x0204
> +#define IMX208_ANA_GAIN_MIN		0
> +#define IMX208_ANA_GAIN_MAX		0x00e0
> +#define IMX208_ANA_GAIN_STEP		1
> +#define IMX208_ANA_GAIN_DEFAULT		0x0
> +
> +/* Digital gain control */
> +#define IMX208_REG_GR_DIGITAL_GAIN	0x020e
> +#define IMX208_REG_R_DIGITAL_GAIN	0x0210
> +#define IMX208_REG_B_DIGITAL_GAIN	0x0212
> +#define IMX208_REG_GB_DIGITAL_GAIN	0x0214
> +#define IMX208_DGTL_GAIN_MIN		0
> +#define IMX208_DGTL_GAIN_MAX		4096   /* Max = 0xFFF */
> +#define IMX208_DGTL_GAIN_DEFAULT	0x100
> +#define IMX208_DGTL_GAIN_STEP           1
> +
> +/* Orientation */
> +#define REG_MIRROR_FLIP_CONTROL	0x0101
> +#define REG_CONFIG_MIRROR_FLIP		0x03
> +
> +#define IMX208_REG_TEST_PATTERN_MODE	0x0600
> +
> +struct imx208_reg {
> +	u16 address;
> +	u8 val;
> +};
> +
> +struct imx208_reg_list {
> +	u32 num_of_regs;
> +	const struct imx208_reg *regs;
> +};
> +
> +/* Link frequency config */
> +struct imx208_link_freq_config {
> +	u32 pixels_per_line;
> +
> +	/* PLL registers for this link frequency */
> +	struct imx208_reg_list reg_list;
> +};
> +
> +/* Mode : resolution and related config&values */
> +struct imx208_mode {
> +	/* Frame width */
> +	u32 width;
> +	/* Frame height */
> +	u32 height;
> +
> +	/* V-timing */
> +	u32 vts_def;
> +	u32 vts_min;
> +
> +	/* Index of Link frequency config to be used */
> +	u32 link_freq_index;
> +	/* Default register values */
> +	struct imx208_reg_list reg_list;
> +};
> +
> +static const struct imx208_reg mipi_data_rate[] = {
> +	{0x0305, 0x02},
> +	{0x0307, 0x50},
> +	{0x303C, 0x3C},
> +};
> +
> +static const struct imx208_reg mode_1936x1096_60fps_regs[] = {
> +	{0x0340, 0x04},
> +	{0x0341, 0x72},
> +	{0x0342, 0x04},
> +	{0x0343, 0x64},
> +	{0x034C, 0x07},
> +	{0x034D, 0x90},
> +	{0x034E, 0x04},
> +	{0x034F, 0x48},
> +	{0x0381, 0x01},
> +	{0x0383, 0x01},
> +	{0x0385, 0x01},
> +	{0x0387, 0x01},
> +	{0x3048, 0x00},
> +	{0x3050, 0x01},
> +	{0x30D5, 0x00},
> +	{0x3301, 0x00},
> +	{0x3318, 0x62},
> +	{0x0202, 0x01},
> +	{0x0203, 0x90},
> +	{0x0205, 0x00},
> +};
> +
> +static const struct imx208_reg mode_968_548_60fps_regs[] = {
> +	{0x0340, 0x02},
> +	{0x0341, 0x39},
> +	{0x0342, 0x08},
> +	{0x0343, 0xC8},
> +	{0x034C, 0x03},
> +	{0x034D, 0xC8},
> +	{0x034E, 0x02},
> +	{0x034F, 0x24},
> +	{0x0381, 0x01},
> +	{0x0383, 0x03},
> +	{0x0385, 0x01},
> +	{0x0387, 0x03},
> +	{0x3048, 0x01},
> +	{0x3050, 0x02},
> +	{0x30D5, 0x03},
> +	{0x3301, 0x10},
> +	{0x3318, 0x75},
> +	{0x0202, 0x01},
> +	{0x0203, 0x90},
> +	{0x0205, 0x00},
> +};
> +
> +static const char * const imx208_test_pattern_menu[] = {
> +	"Disabled",
> +	"Solid Color",
> +	"100% Color Bar",
> +	"Fade to grey Color Bar",

s/g/G/

> +	"PN9",
> +	"Fixed Pattern1",
> +	"Fixed Pattern2",
> +	"Fixed Pattern3",
> +	"Fixed Pattern4",
> +	"Fixed Pattern5",
> +	"Fixed Pattern6"
> +};
> +
> +/* Configurations for supported link frequencies */
> +#define IMX208_LINK_FREQ_384MHZ	384000000ULL
> +#define IMX208_LINK_FREQ_192MHZ	192000000ULL
> +#define IMX208_LINK_FREQ_96MHZ  96000000ULL
> +#define IMX208_LINK_FREQ_48MHZ  48000000ULL

You may put the link frequencies here, too, but they should be checked
against the endpoint configuration ("link-frequencies" property; the
V4L2 fwnode framework parses them).

> +
> +enum {
> +	IMX208_LINK_FREQ_384MHZ_INDEX,
> +	IMX208_LINK_FREQ_96MHZ_INDEX,
> +};
> +
> +/*
> + * pixel_rate = link_freq * data-rate * nr_of_lanes / bits_per_sample
> + * data rate => double data rate; number of lanes => 2; bits per pixel => 10
> + */
> +static u64 link_freq_to_pixel_rate(u64 f)
> +{
> +	f *= 2 * 2;
> +	do_div(f, 10);
> +
> +	return f;
> +}
> +
> +/* Menu items for LINK_FREQ V4L2 control */
> +static const s64 link_freq_menu_items[] = {
> +	IMX208_LINK_FREQ_384MHZ,
> +	IMX208_LINK_FREQ_96MHZ,
> +};
> +
> +/* Link frequency configs */
> +static const struct imx208_link_freq_config link_freq_configs[] = {
> +	{
> +		.pixels_per_line = IMX208_PPL_384MHZ,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mipi_data_rate),
> +			.regs = mipi_data_rate,
> +		}
> +	},
> +	{
> +		.pixels_per_line = IMX208_PPL_96MHZ,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mipi_data_rate),
> +			.regs = mipi_data_rate,
> +		}
> +	},
> +};
> +
> +/* Mode configs */
> +static const struct imx208_mode supported_modes[] = {
> +	{
> +		.width = 1936,
> +		.height = 1096,
> +		.vts_def = IMX208_VTS_60FPS,
> +		.vts_min = IMX208_VTS_60FPS_MIN,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mode_1936x1096_60fps_regs),
> +			.regs = mode_1936x1096_60fps_regs,
> +		},
> +		.link_freq_index = 0,
> +	},
> +	{
> +		.width = 968,
> +		.height = 548,
> +		.vts_def = IMX208_VTS_BINNING,
> +		.vts_min = IMX208_VTS_BINNING_MIN,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mode_968_548_60fps_regs),
> +			.regs = mode_968_548_60fps_regs,
> +		},
> +		.link_freq_index = IMX208_LINK_FREQ_96MHZ_INDEX,
> +	},
> +};
> +
> +struct imx208 {
> +	struct v4l2_subdev sd;
> +	struct media_pad pad;
> +
> +	struct v4l2_ctrl_handler ctrl_handler;
> +	/* V4L2 Controls */
> +	struct v4l2_ctrl *link_freq;
> +	struct v4l2_ctrl *pixel_rate;
> +	struct v4l2_ctrl *vblank;
> +	struct v4l2_ctrl *hblank;
> +	struct v4l2_ctrl *exposure;
> +
> +	/* Current mode */
> +	const struct imx208_mode *cur_mode;
> +
> +	/* Mutex for serialized access */
> +	struct mutex mutex;
> +
> +	/* Streaming on/off */
> +	bool streaming;
> +};
> +
> +static inline struct imx208 *to_imx208(struct v4l2_subdev *_sd)

No need for an underscore as this isn't a macro; up to you...

> +{
> +	return container_of(_sd, struct imx208, sd);
> +}
> +
> +/* Read registers up to 2 at a time */
> +static int imx208_read_reg(struct imx208 *imx208, u16 reg, u32 len, u32 *val)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	struct i2c_msg msgs[2];
> +	u8 addr_buf[2] = { reg >> 8, reg & 0xff };
> +	u8 data_buf[4] = { 0, };
> +	int ret;
> +
> +	if (len > 4)
> +		return -EINVAL;
> +
> +	/* Write register address */
> +	msgs[0].addr = client->addr;
> +	msgs[0].flags = 0;
> +	msgs[0].len = ARRAY_SIZE(addr_buf);
> +	msgs[0].buf = addr_buf;
> +
> +	/* Read data from register */
> +	msgs[1].addr = client->addr;
> +	msgs[1].flags = I2C_M_RD;
> +	msgs[1].len = len;
> +	msgs[1].buf = &data_buf[4 - len];
> +
> +	ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
> +	if (ret != ARRAY_SIZE(msgs))
> +		return -EIO;
> +
> +	*val = get_unaligned_be32(data_buf);
> +
> +	return 0;
> +}
> +
> +/* Write registers up to 4 at a time */
> +static int imx208_write_reg(struct imx208 *imx208, u16 reg, u32 len, u32 val)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	u8 buf[6];
> +
> +	if (len > 4)
> +		return -EINVAL;
> +
> +	put_unaligned_be16(reg, buf);
> +	put_unaligned_be32(val << (8 * (4 - len)), buf + 2);
> +	if (i2c_master_send(client, buf, len + 2) != len + 2)
> +		return -EIO;
> +
> +	return 0;
> +}
> +
> +/* Write a list of registers */
> +static int imx208_write_regs(struct imx208 *imx208,
> +			      const struct imx208_reg *regs, u32 len)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	unsigned int i;
> +	int ret;
> +
> +	for (i = 0; i < len; i++) {
> +		ret = imx208_write_reg(imx208, regs[i].address, 1,
> +					regs[i].val);
> +		if (ret) {
> +			dev_err_ratelimited(
> +				&client->dev,
> +				"Failed to write reg 0x%4.4x. error = %d\n",
> +				regs[i].address, ret);
> +
> +			return ret;
> +		}
> +	}
> +
> +	return 0;
> +}
> +
> +/* Open sub-device */
> +static int imx208_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh)
> +{
> +	struct v4l2_mbus_framefmt *try_fmt =
> +		v4l2_subdev_get_try_format(sd, fh->pad, 0);
> +
> +	/* Initialize try_fmt */
> +	try_fmt->width = supported_modes[0].width;
> +	try_fmt->height = supported_modes[0].height;
> +	try_fmt->code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +	try_fmt->field = V4L2_FIELD_NONE;
> +
> +	return 0;
> +}
> +
> +
> +static int imx208_update_digital_gain(struct imx208 *imx208, u32 len, u32 val)
> +{
> +	int ret;
> +
> +	ret = imx208_write_reg(imx208, IMX208_REG_GR_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);
> +	if (ret)
> +		return ret;
> +	ret = imx208_write_reg(imx208, IMX208_REG_GB_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);
> +	if (ret)
> +		return ret;
> +	ret = imx208_write_reg(imx208, IMX208_REG_R_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);
> +	if (ret)
> +		return ret;
> +	ret = imx208_write_reg(imx208, IMX208_REG_B_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);

return imx208...();

> +	if (ret)
> +		return ret;
> +	return 0;
> +}
> +
> +static int imx208_enable_test_pattern(struct imx208 *imx208, u32 pattern)
> +{
> +	u32 val;
> +
> +	val = (pattern <= 4) ? pattern : pattern + 0xFB;
> +
> +	return imx208_write_reg(imx208, IMX208_REG_TEST_PATTERN_MODE,
> +				IMX208_REG_VALUE_16BIT, val);
> +}
> +
> +static int imx208_set_ctrl(struct v4l2_ctrl *ctrl)
> +{
> +	struct imx208 *imx208 =
> +		container_of(ctrl->handler, struct imx208, ctrl_handler);
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	int ret = 0;

An empty line after variable declarations, please.

> +	/*
> +	 * Applying V4L2 control value only happens
> +	 * when power is up for streaming
> +	 */
> +	if (pm_runtime_get_if_in_use(&client->dev) <= 0)
> +		return 0;
> +
> +	switch (ctrl->id) {
> +	case V4L2_CID_ANALOGUE_GAIN:
> +		ret = imx208_write_reg(imx208, IMX208_REG_ANALOG_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				ctrl->val);
> +		break;
> +	case V4L2_CID_EXPOSURE:
> +		ret = imx208_write_reg(imx208, IMX208_REG_EXPOSURE,
> +				IMX208_REG_VALUE_16BIT,
> +				ctrl->val);
> +		break;
> +	case V4L2_CID_DIGITAL_GAIN:
> +		ret = imx208_update_digital_gain(imx208, IMX208_REG_VALUE_16BIT,
> +				ctrl->val);
> +		break;
> +	case V4L2_CID_VBLANK:
> +		/* Update VTS that meets expected vertical blanking */
> +		ret = imx208_write_reg(imx208, IMX208_REG_VTS,
> +				       IMX208_REG_VALUE_16BIT,
> +				       imx208->cur_mode->height + ctrl->val);
> +		break;
> +	case V4L2_CID_TEST_PATTERN:
> +		ret = imx208_enable_test_pattern(imx208, ctrl->val);
> +		break;
> +	default:
> +		dev_info(&client->dev,
> +			 "ctrl(id:0x%x,val:0x%x) is not handled\n",
> +			 ctrl->id, ctrl->val);
> +		break;
> +	}
> +
> +	pm_runtime_put(&client->dev);
> +
> +	return ret;
> +}
> +
> +static const struct v4l2_ctrl_ops imx208_ctrl_ops = {
> +	.s_ctrl = imx208_set_ctrl,
> +};
> +
> +static int imx208_enum_mbus_code(struct v4l2_subdev *sd,
> +				  struct v4l2_subdev_pad_config *cfg,
> +				  struct v4l2_subdev_mbus_code_enum *code)
> +{
> +	/* Only one bayer order(GRBG) is supported */
> +	if (code->index > 0)
> +		return -EINVAL;
> +
> +	code->code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +
> +	return 0;
> +}
> +
> +static int imx208_enum_frame_size(struct v4l2_subdev *sd,
> +				   struct v4l2_subdev_pad_config *cfg,
> +				   struct v4l2_subdev_frame_size_enum *fse)
> +{
> +	if (fse->index >= ARRAY_SIZE(supported_modes))
> +		return -EINVAL;
> +
> +	if (fse->code != MEDIA_BUS_FMT_SRGGB10_1X10)
> +		return -EINVAL;
> +
> +	fse->min_width = supported_modes[fse->index].width;
> +	fse->max_width = fse->min_width;
> +	fse->min_height = supported_modes[fse->index].height;
> +	fse->max_height = fse->min_height;
> +
> +	return 0;
> +}
> +
> +static void imx208_update_pad_format(const struct imx208_mode *mode,
> +				      struct v4l2_subdev_format *fmt)

How about calling this e.g. imx208_mode_to_pad_format?

> +{
> +	fmt->format.width = mode->width;
> +	fmt->format.height = mode->height;
> +	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +	fmt->format.field = V4L2_FIELD_NONE;
> +}
> +
> +static int __imx208_get_pad_format(struct imx208 *imx208,
> +				     struct v4l2_subdev_pad_config *cfg,
> +				     struct v4l2_subdev_format *fmt)
> +{
> +	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY)
> +		fmt->format = *v4l2_subdev_get_try_format(&imx208->sd, cfg,
> +							  fmt->pad);
> +	else
> +		imx208_update_pad_format(imx208->cur_mode, fmt);
> +
> +	return 0;
> +}
> +
> +static int imx208_get_pad_format(struct v4l2_subdev *sd,
> +				  struct v4l2_subdev_pad_config *cfg,
> +				  struct v4l2_subdev_format *fmt)
> +{
> +	struct imx208 *imx208 = to_imx208(sd);
> +	int ret;
> +
> +	mutex_lock(&imx208->mutex);
> +	ret = __imx208_get_pad_format(imx208, cfg, fmt);
> +	mutex_unlock(&imx208->mutex);
> +
> +	return ret;
> +}
> +
> +static int imx208_set_pad_format(struct v4l2_subdev *sd,
> +		       struct v4l2_subdev_pad_config *cfg,
> +		       struct v4l2_subdev_format *fmt)
> +{
> +	struct imx208 *imx208 = to_imx208(sd);
> +	const struct imx208_mode *mode;
> +	struct v4l2_mbus_framefmt *framefmt;
> +	s32 vblank_def;
> +	s32 vblank_min;
> +	s64 h_blank;
> +	s64 pixel_rate;
> +	s64 link_freq;
> +
> +	mutex_lock(&imx208->mutex);
> +
> +	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;

This line seems redundant.

> +
> +	mode = v4l2_find_nearest_size(
> +		supported_modes, ARRAY_SIZE(supported_modes), width, height,
> +		fmt->format.width, fmt->format.height);
> +	imx208_update_pad_format(mode, fmt);
> +	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) {
> +		framefmt = v4l2_subdev_get_try_format(sd, cfg, fmt->pad);

You could move the declaration of framefmt here.

> +		*framefmt = fmt->format;
> +	} else {
> +		imx208->cur_mode = mode;
> +		__v4l2_ctrl_s_ctrl(imx208->link_freq, mode->link_freq_index);
> +
> +		link_freq = link_freq_menu_items[mode->link_freq_index];
> +		pixel_rate = link_freq_to_pixel_rate(link_freq);
> +		__v4l2_ctrl_s_ctrl_int64(imx208->pixel_rate, pixel_rate);
> +		/* Update limits and set FPS to default */
> +		vblank_def = imx208->cur_mode->vts_def -
> +			     imx208->cur_mode->height;
> +		vblank_min = imx208->cur_mode->vts_min -
> +			     imx208->cur_mode->height;
> +		__v4l2_ctrl_modify_range(
> +			imx208->vblank, vblank_min,
> +			IMX208_VTS_MAX - imx208->cur_mode->height, 1,
> +			vblank_def);
> +		__v4l2_ctrl_s_ctrl(imx208->vblank, vblank_def);
> +		h_blank =
> +			link_freq_configs[mode->link_freq_index].pixels_per_line
> +			 - imx208->cur_mode->width;
> +		__v4l2_ctrl_modify_range(imx208->hblank, h_blank,
> +					 h_blank, 1, h_blank);
> +	}
> +
> +	mutex_unlock(&imx208->mutex);
> +
> +	return 0;
> +}
> +
> +/* Start streaming */
> +static int imx208_start_streaming(struct imx208 *imx208)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	const struct imx208_reg_list *reg_list;
> +	int ret, link_freq_index;
> +
> +	/* Setup PLL */
> +	link_freq_index = imx208->cur_mode->link_freq_index;
> +	reg_list = &link_freq_configs[link_freq_index].reg_list;
> +	ret = imx208_write_regs(imx208, reg_list->regs, reg_list->num_of_regs);
> +	if (ret) {
> +		dev_err(&client->dev, "%s failed to set plls\n", __func__);
> +		return ret;
> +	}
> +
> +	/* Apply default values of current mode */
> +	reg_list = &imx208->cur_mode->reg_list;
> +	ret = imx208_write_regs(imx208, reg_list->regs, reg_list->num_of_regs);
> +	if (ret) {
> +		dev_err(&client->dev, "%s failed to set mode\n", __func__);
> +		return ret;
> +	}
> +
> +	/* Apply customized values from user */
> +	ret =  __v4l2_ctrl_handler_setup(imx208->sd.ctrl_handler);
> +	if (ret)
> +		return ret;
> +
> +	/* set stream on register */
> +	return imx208_write_reg(imx208, IMX208_REG_MODE_SELECT,
> +				 IMX208_REG_VALUE_08BIT,
> +				 IMX208_MODE_STREAMING);

Fits on previous line.

> +}
> +
> +/* Stop streaming */
> +static int imx208_stop_streaming(struct imx208 *imx208)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	int ret;
> +
> +	/* set stream off register */
> +	ret = imx208_write_reg(imx208, IMX208_REG_MODE_SELECT,
> +		IMX208_REG_VALUE_08BIT, IMX208_MODE_STANDBY);
> +	if (ret)
> +		dev_err(&client->dev, "%s failed to set stream\n", __func__);
> +
> +	/* Return success even if it was an error, as there is nothing the
> +	 * caller can do about it.
> +	 */
> +	return 0;
> +}
> +
> +static int imx208_set_stream(struct v4l2_subdev *sd, int enable)
> +{
> +	struct imx208 *imx208 = to_imx208(sd);
> +	struct i2c_client *client = v4l2_get_subdevdata(sd);
> +	int ret = 0;
> +
> +	mutex_lock(&imx208->mutex);
> +	if (imx208->streaming == enable) {
> +		mutex_unlock(&imx208->mutex);
> +		return 0;
> +	}
> +
> +	if (enable) {
> +		ret = pm_runtime_get_sync(&client->dev);
> +		if (ret < 0) {
> +			pm_runtime_put_noidle(&client->dev);
> +			goto err_unlock;
> +		}
> +
> +		/*
> +		 * Apply default & customized values
> +		 * and then start streaming.
> +		 */
> +		ret = imx208_start_streaming(imx208);
> +		if (ret)
> +			goto err_rpm_put;
> +	} else {
> +		imx208_stop_streaming(imx208);
> +		pm_runtime_put(&client->dev);
> +	}
> +
> +	imx208->streaming = enable;
> +	mutex_unlock(&imx208->mutex);
> +
> +	return ret;
> +
> +err_rpm_put:
> +	pm_runtime_put(&client->dev);
> +err_unlock:
> +	mutex_unlock(&imx208->mutex);
> +
> +	return ret;
> +}
> +
> +static int __maybe_unused imx208_suspend(struct device *dev)
> +{
> +	struct i2c_client *client = to_i2c_client(dev);
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +
> +	if (imx208->streaming)
> +		imx208_stop_streaming(imx208);
> +
> +	return 0;
> +}
> +
> +static int __maybe_unused imx208_resume(struct device *dev)
> +{
> +	struct i2c_client *client = to_i2c_client(dev);
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +	int ret;
> +
> +	if (imx208->streaming) {
> +		ret = imx208_start_streaming(imx208);
> +		if (ret)
> +			goto error;
> +	}
> +
> +	return 0;
> +
> +error:
> +	imx208_stop_streaming(imx208);
> +	imx208->streaming = 0;
> +	return ret;
> +}
> +
> +/* Verify chip ID */
> +static int imx208_identify_module(struct imx208 *imx208)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	int ret;
> +	u32 val;
> +
> +	ret = imx208_read_reg(imx208, IMX208_REG_CHIP_ID,
> +			       IMX208_REG_VALUE_16BIT, &val);
> +	if (ret) {
> +		dev_err(&client->dev, "failed to read chip id %x\n",
> +			IMX208_CHIP_ID);
> +		return ret;
> +	}
> +
> +	if (val != IMX208_CHIP_ID) {
> +		dev_err(&client->dev, "chip id mismatch: %x!=%x\n",
> +			IMX208_CHIP_ID, val);
> +		return -EIO;
> +	}
> +
> +	dev_err(&client->dev, "chip id!: %x %x\n", IMX208_CHIP_ID, val);
> +
> +	return 0;
> +}
> +
> +static const struct v4l2_subdev_video_ops imx208_video_ops = {
> +	.s_stream = imx208_set_stream,
> +};
> +
> +static const struct v4l2_subdev_pad_ops imx208_pad_ops = {
> +	.enum_mbus_code = imx208_enum_mbus_code,
> +	.get_fmt = imx208_get_pad_format,
> +	.set_fmt = imx208_set_pad_format,
> +	.enum_frame_size = imx208_enum_frame_size,
> +};
> +
> +static const struct v4l2_subdev_ops imx208_subdev_ops = {
> +	.video = &imx208_video_ops,
> +	.pad = &imx208_pad_ops,
> +};
> +
> +static const struct v4l2_subdev_internal_ops imx208_internal_ops = {
> +	.open = imx208_open,
> +};
> +
> +/* Initialize control handlers */
> +static int imx208_init_controls(struct imx208 *imx208)
> +{
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	struct v4l2_ctrl_handler *ctrl_hdlr;
> +	s64 exposure_max;
> +	s64 vblank_def;
> +	s64 vblank_min;
> +	s64 pixel_rate_min;
> +	s64 pixel_rate_max;
> +	int ret;
> +
> +	ctrl_hdlr = &imx208->ctrl_handler;

Please move the assignment to the declaration. And perhaps call it e.g. hdl
--- the latter is up to you.

> +	ret = v4l2_ctrl_handler_init(ctrl_hdlr, 8);
> +	if (ret)
> +		return ret;
> +
> +	mutex_init(&imx208->mutex);
> +	ctrl_hdlr->lock = &imx208->mutex;
> +	imx208->link_freq = v4l2_ctrl_new_int_menu(ctrl_hdlr,
> +				&imx208_ctrl_ops,
> +				V4L2_CID_LINK_FREQ,
> +				ARRAY_SIZE(link_freq_menu_items) - 1,
> +				0,
> +				link_freq_menu_items);
> +
> +	if (imx208->link_freq)
> +		imx208->link_freq->flags |= V4L2_CTRL_FLAG_READ_ONLY;
> +
> +	pixel_rate_max = link_freq_to_pixel_rate(link_freq_menu_items[0]);
> +	pixel_rate_min = link_freq_to_pixel_rate(link_freq_menu_items[1]);
> +	/* By default, PIXEL_RATE is read only */
> +	imx208->pixel_rate = v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops,
> +					V4L2_CID_PIXEL_RATE,
> +					pixel_rate_min, pixel_rate_max,
> +					1, pixel_rate_max);
> +
> +
> +	vblank_def = imx208->cur_mode->vts_def - imx208->cur_mode->height;
> +	vblank_min = imx208->cur_mode->vts_min - imx208->cur_mode->height;
> +	imx208->vblank = v4l2_ctrl_new_std(
> +				ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_VBLANK,
> +				vblank_min,
> +				IMX208_VTS_MAX - imx208->cur_mode->height, 1,
> +				vblank_def);
> +
> +	imx208->hblank = v4l2_ctrl_new_std(
> +				ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_HBLANK,
> +				IMX208_PPL_384MHZ - imx208->cur_mode->width,
> +				IMX208_PPL_384MHZ - imx208->cur_mode->width,
> +				1,
> +				IMX208_PPL_384MHZ - imx208->cur_mode->width);
> +
> +	if (imx208->hblank)
> +		imx208->hblank->flags |= V4L2_CTRL_FLAG_READ_ONLY;
> +
> +	exposure_max = imx208->cur_mode->vts_def - 8;
> +	imx208->exposure = v4l2_ctrl_new_std(

imx208->exposure is unused. Please remove it.

> +				ctrl_hdlr, &imx208_ctrl_ops,
> +				V4L2_CID_EXPOSURE, IMX208_EXPOSURE_MIN,
> +				IMX208_EXPOSURE_MAX, IMX208_EXPOSURE_STEP,
> +				IMX208_EXPOSURE_DEFAULT);
> +
> +	v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_ANALOGUE_GAIN,
> +				IMX208_ANA_GAIN_MIN, IMX208_ANA_GAIN_MAX,
> +				IMX208_ANA_GAIN_STEP, IMX208_ANA_GAIN_DEFAULT);
> +
> +	v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_DIGITAL_GAIN,
> +				IMX208_DGTL_GAIN_MIN, IMX208_DGTL_GAIN_MAX,
> +				IMX208_DGTL_GAIN_STEP,
> +				IMX208_DGTL_GAIN_DEFAULT);
> +
> +	v4l2_ctrl_new_std_menu_items(ctrl_hdlr, &imx208_ctrl_ops,
> +				     V4L2_CID_TEST_PATTERN,
> +				     ARRAY_SIZE(imx208_test_pattern_menu) - 1,
> +				     0, 0, imx208_test_pattern_menu);
> +
> +	if (ctrl_hdlr->error) {
> +		ret = ctrl_hdlr->error;
> +		dev_err(&client->dev, "%s control init failed (%d)\n",
> +			__func__, ret);
> +		goto error;
> +	}
> +
> +	imx208->sd.ctrl_handler = ctrl_hdlr;
> +
> +	return 0;
> +
> +error:
> +	v4l2_ctrl_handler_free(ctrl_hdlr);
> +	mutex_destroy(&imx208->mutex);
> +
> +	return ret;
> +}
> +
> +static void imx208_free_controls(struct imx208 *imx208)
> +{
> +	v4l2_ctrl_handler_free(imx208->sd.ctrl_handler);
> +	mutex_destroy(&imx208->mutex);
> +}
> +
> +static int imx208_probe(struct i2c_client *client)
> +{
> +	struct imx208 *imx208;
> +	int ret;
> +	u32 val = 0;
> +
> +	device_property_read_u32(&client->dev, "clock-frequency", &val);
> +	if (val != 19200000)
> +		return -EINVAL;
> +
> +	imx208 = devm_kzalloc(&client->dev, sizeof(*imx208), GFP_KERNEL);
> +	if (!imx208)
> +		return -ENOMEM;
> +
> +	/* Initialize subdev */
> +	v4l2_i2c_subdev_init(&imx208->sd, client, &imx208_subdev_ops);
> +
> +	/* Check module identity */
> +	ret = imx208_identify_module(imx208);
> +	if (ret)
> +		return ret;
> +
> +	/* Set default mode to max resolution */
> +	imx208->cur_mode = &supported_modes[0];
> +
> +	ret = imx208_init_controls(imx208);
> +	if (ret)
> +		return ret;
> +
> +	/* Initialize subdev */
> +	imx208->sd.internal_ops = &imx208_internal_ops;
> +	imx208->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
> +	imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;

We no longer have types, but functions. I.e. MEDIA_ENT_F_CAM_SENSOR . The
type field no longer exists. (See below.)

> +
> +	/* Initialize source pad */
> +	imx208->pad.flags = MEDIA_PAD_FL_SOURCE;
> +	ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);

This function got renamed quite some time ago as media_entity_pads_init().
Have you compile tested this against the current media tree?

> +	if (ret) {
> +		dev_err(&client->dev, "%s failed:%d\n", __func__, ret);
> +		goto error_handler_free;
> +	}
> +
> +	ret = v4l2_async_register_subdev_sensor_common(&imx208->sd);
> +	if (ret < 0)
> +		goto error_media_entity;
> +
> +	pm_runtime_set_active(&client->dev);
> +	pm_runtime_enable(&client->dev);
> +	pm_runtime_idle(&client->dev);
> +
> +	return 0;
> +
> +error_media_entity:
> +	media_entity_cleanup(&imx208->sd.entity);
> +
> +error_handler_free:
> +	imx208_free_controls(imx208);
> +
> +	return ret;
> +}
> +
> +static int imx208_remove(struct i2c_client *client)
> +{
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +
> +	v4l2_async_unregister_subdev(sd);
> +	media_entity_cleanup(&sd->entity);
> +	imx208_free_controls(imx208);
> +
> +	pm_runtime_disable(&client->dev);
> +	pm_runtime_set_suspended(&client->dev);
> +
> +	return 0;
> +}
> +
> +static const struct dev_pm_ops imx208_pm_ops = {
> +	SET_SYSTEM_SLEEP_PM_OPS(imx208_suspend, imx208_resume)
> +};
> +
> +#ifdef CONFIG_ACPI
> +static const struct acpi_device_id imx208_acpi_ids[] = {
> +	{ "INT3478" },
> +	{ /* sentinel */ }
> +};
> +
> +MODULE_DEVICE_TABLE(acpi, imx208_acpi_ids);
> +#endif
> +
> +static struct i2c_driver imx208_i2c_driver = {
> +	.driver = {
> +		.name = "imx208",
> +		.pm = &imx208_pm_ops,
> +		.acpi_match_table = ACPI_PTR(imx208_acpi_ids),
> +	},
> +	.probe_new = imx208_probe,
> +	.remove = imx208_remove,
> +};
> +
> +module_i2c_driver(imx208_i2c_driver);
> +
> +MODULE_AUTHOR("Yeh, Andy <andy.yeh@intel.com>");
> +MODULE_AUTHOR("Chen, Ping-chung <ping-chung.chen@intel.com>");
> +MODULE_DESCRIPTION("Sony IMX208 sensor driver");
> +MODULE_LICENSE("GPL v2");

-- 
Kind regards,

Sakari Ailus
sakari.ailus@linux.intel.com

^ permalink raw reply	[flat|nested] 8+ messages in thread

* Re: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-10  7:37 ` Yeh, Andy
@ 2018-07-16 16:03   ` sakari.ailus
  0 siblings, 0 replies; 8+ messages in thread
From: sakari.ailus @ 2018-07-16 16:03 UTC (permalink / raw)
  To: Yeh, Andy
  Cc: Chen, Ping-chung, linux-media, tfiga, grundler, Chen, JasonX Z, Lai, Jim

On Tue, Jul 10, 2018 at 07:37:54AM +0000, Yeh, Andy wrote:
> Hi PC,
> 
> Thanks for the patch.
> 
> Cc in Grant, and Intel Jim/Jason
> 
> > -----Original Message-----
> > From: Chen, Ping-chung
> > Sent: Tuesday, July 10, 2018 3:16 PM
> > To: linux-media@vger.kernel.org
> > Cc: sakari.ailus@linux.intel.com; Yeh, Andy <andy.yeh@intel.com>;
> > tfiga@chromium.org; Chen, Ping-chung <ping-chung.chen@intel.com>
> > Subject: [PATCH] media: imx208: Add imx208 camera sensor driver
> > +};
> > +
> > +static int imx208_enum_mbus_code(struct v4l2_subdev *sd,
> > +				  struct v4l2_subdev_pad_config *cfg,
> > +				  struct v4l2_subdev_mbus_code_enum *code) {
> > +	/* Only one bayer order(GRBG) is supported */
> > +	if (code->index > 0)
> > +		return -EINVAL;
> > +
> 
> There is no limitation on using GRBG bayer order now. You can refer to imx355 driver as well.

It seems the rest of the driver uses RGGB.

The enumeration should only contain what is possible using the current
flipping configuration.

-- 
Sakari Ailus
sakari.ailus@linux.intel.com

^ permalink raw reply	[flat|nested] 8+ messages in thread

* RE: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-16 15:58 ` Sakari Ailus
@ 2018-07-17  6:53   ` Chen, Ping-chung
  2018-07-19 13:22     ` Tomasz Figa
  0 siblings, 1 reply; 8+ messages in thread
From: Chen, Ping-chung @ 2018-07-17  6:53 UTC (permalink / raw)
  To: Sakari Ailus; +Cc: linux-media, Yeh, Andy, tfiga

Hi Sakari,

Some of suggestions below has been added in to [PATCH v2] or [PATCH v3].
We will fix others in PATCH v4 soon.

Thanks,
PC 
-----Original Message-----
From: Sakari Ailus [mailto:sakari.ailus@linux.intel.com] 
Sent: Monday, July 16, 2018 11:59 PM
To: Chen, Ping-chung <ping-chung.chen@intel.com>
Cc: linux-media@vger.kernel.org; Yeh, Andy <andy.yeh@intel.com>; tfiga@chromium.org
Subject: Re: [PATCH] media: imx208: Add imx208 camera sensor driver

Hi Ping-chung,

Thanks for the patch. Please see my comments below.

On Tue, Jul 10, 2018 at 03:15:36PM +0800, Ping-chung Chen wrote:
> From: "Chen, Ping-chung" <ping-chung.chen@intel.com>
> 
> Add a V4L2 sub-device driver for the Sony IMX208 image sensor.
> This is a camera sensor using the I2C bus for control and the
> CSI-2 bus for data.
> 
> Signed-off-by: Ping-Chung Chen <ping-chung.chen@intel.com>
> ---
>  drivers/media/i2c/Kconfig  |  11 +
>  drivers/media/i2c/Makefile |   1 +
>  drivers/media/i2c/imx208.c | 953 
> +++++++++++++++++++++++++++++++++++++++++++++
>  3 files changed, 965 insertions(+)
>  create mode 100644 drivers/media/i2c/imx208.c
> 
> diff --git a/drivers/media/i2c/Kconfig b/drivers/media/i2c/Kconfig 
> index 8669853..1c739f4 100644
> --- a/drivers/media/i2c/Kconfig
> +++ b/drivers/media/i2c/Kconfig
> @@ -595,6 +595,17 @@ config VIDEO_APTINA_PLL  config VIDEO_SMIAPP_PLL
>  	tristate
>  
> +config VIDEO_IMX208
> +	tristate "Sony IMX208 sensor support"
> +	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API
> +	depends on MEDIA_CAMERA_SUPPORT
> +	---help---
> +	  This is a Video4Linux2 sensor-level driver for the Sony

s/-level//

> +	  IMX208 camera.
> +
> +          To compile this driver as a module, choose M here: the
> +          module will be called imx208.
> +
>  config VIDEO_IMX258
>  	tristate "Sony IMX258 sensor support"
>  	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API diff --git 
> a/drivers/media/i2c/Makefile b/drivers/media/i2c/Makefile index 
> 837c428..47604c2 100644
> --- a/drivers/media/i2c/Makefile
> +++ b/drivers/media/i2c/Makefile
> @@ -104,6 +104,7 @@ obj-$(CONFIG_VIDEO_I2C)		+= video-i2c.o
>  obj-$(CONFIG_VIDEO_ML86V7667)	+= ml86v7667.o
>  obj-$(CONFIG_VIDEO_OV2659)	+= ov2659.o
>  obj-$(CONFIG_VIDEO_TC358743)	+= tc358743.o
> +obj-$(CONFIG_VIDEO_IMX208)	+= imx208.o
>  obj-$(CONFIG_VIDEO_IMX258)	+= imx258.o
>  obj-$(CONFIG_VIDEO_IMX274)	+= imx274.o
>  
> diff --git a/drivers/media/i2c/imx208.c b/drivers/media/i2c/imx208.c 
> new file mode 100644 index 0000000..9af9043
> --- /dev/null
> +++ b/drivers/media/i2c/imx208.c
> @@ -0,0 +1,953 @@
> +// SPDX-License-Identifier: GPL-2.0
> +// Copyright (C) 2018 Intel Corporation
> +
> +#include <linux/acpi.h>
> +#include <linux/delay.h>
> +#include <linux/i2c.h>
> +#include <linux/module.h>
> +#include <linux/pm_runtime.h>
> +#include <media/v4l2-ctrls.h>
> +#include <media/v4l2-device.h>
> +#include <asm/unaligned.h>
> +
> +#ifndef V4L2_CID_DIGITAL_GAIN
> +#define V4L2_CID_DIGITAL_GAIN V4L2_CID_GAIN #endif

Please drop this.

> +
> +#define IMX208_REG_VALUE_08BIT		1
> +#define IMX208_REG_VALUE_16BIT		2
> +#define IMX208_REG_VALUE_24BIT		3

The last one is unused.

> +
> +#define IMX208_REG_MODE_SELECT		0x0100
> +#define IMX208_MODE_STANDBY		0x00
> +#define IMX208_MODE_STREAMING		0x01
> +
> +/* Chip ID */
> +#define IMX208_REG_CHIP_ID		0x0000
> +#define IMX208_CHIP_ID			0x0208
> +
> +/* V_TIMING internal */
> +#define IMX208_REG_VTS			0x0340
> +#define IMX208_VTS_60FPS		0x0472
> +#define IMX208_VTS_BINNING	0x0239
> +#define IMX208_VTS_60FPS_MIN		0x0458
> +#define IMX208_VTS_BINNING_MIN	0x0230
> +#define IMX208_VTS_MAX			0xffff
> +
> +/* HBLANK control - read only */
> +#define IMX208_PPL_384MHZ		1124
> +#define IMX208_PPL_96MHZ		1124
> +
> +/* Exposure control */
> +#define IMX208_REG_EXPOSURE		0x0202
> +#define IMX208_EXPOSURE_MIN		4
> +#define IMX208_EXPOSURE_STEP		1
> +#define IMX208_EXPOSURE_DEFAULT		0x190
> +#define IMX208_EXPOSURE_MAX		65535
> +
> +/* Analog gain control */
> +#define IMX208_REG_ANALOG_GAIN		0x0204
> +#define IMX208_ANA_GAIN_MIN		0
> +#define IMX208_ANA_GAIN_MAX		0x00e0
> +#define IMX208_ANA_GAIN_STEP		1
> +#define IMX208_ANA_GAIN_DEFAULT		0x0
> +
> +/* Digital gain control */
> +#define IMX208_REG_GR_DIGITAL_GAIN	0x020e
> +#define IMX208_REG_R_DIGITAL_GAIN	0x0210
> +#define IMX208_REG_B_DIGITAL_GAIN	0x0212
> +#define IMX208_REG_GB_DIGITAL_GAIN	0x0214
> +#define IMX208_DGTL_GAIN_MIN		0
> +#define IMX208_DGTL_GAIN_MAX		4096   /* Max = 0xFFF */
> +#define IMX208_DGTL_GAIN_DEFAULT	0x100
> +#define IMX208_DGTL_GAIN_STEP           1
> +
> +/* Orientation */
> +#define REG_MIRROR_FLIP_CONTROL	0x0101
> +#define REG_CONFIG_MIRROR_FLIP		0x03
> +
> +#define IMX208_REG_TEST_PATTERN_MODE	0x0600
> +
> +struct imx208_reg {
> +	u16 address;
> +	u8 val;
> +};
> +
> +struct imx208_reg_list {
> +	u32 num_of_regs;
> +	const struct imx208_reg *regs;
> +};
> +
> +/* Link frequency config */
> +struct imx208_link_freq_config {
> +	u32 pixels_per_line;
> +
> +	/* PLL registers for this link frequency */
> +	struct imx208_reg_list reg_list;
> +};
> +
> +/* Mode : resolution and related config&values */ struct imx208_mode 
> +{
> +	/* Frame width */
> +	u32 width;
> +	/* Frame height */
> +	u32 height;
> +
> +	/* V-timing */
> +	u32 vts_def;
> +	u32 vts_min;
> +
> +	/* Index of Link frequency config to be used */
> +	u32 link_freq_index;
> +	/* Default register values */
> +	struct imx208_reg_list reg_list;
> +};
> +
> +static const struct imx208_reg mipi_data_rate[] = {
> +	{0x0305, 0x02},
> +	{0x0307, 0x50},
> +	{0x303C, 0x3C},
> +};
> +
> +static const struct imx208_reg mode_1936x1096_60fps_regs[] = {
> +	{0x0340, 0x04},
> +	{0x0341, 0x72},
> +	{0x0342, 0x04},
> +	{0x0343, 0x64},
> +	{0x034C, 0x07},
> +	{0x034D, 0x90},
> +	{0x034E, 0x04},
> +	{0x034F, 0x48},
> +	{0x0381, 0x01},
> +	{0x0383, 0x01},
> +	{0x0385, 0x01},
> +	{0x0387, 0x01},
> +	{0x3048, 0x00},
> +	{0x3050, 0x01},
> +	{0x30D5, 0x00},
> +	{0x3301, 0x00},
> +	{0x3318, 0x62},
> +	{0x0202, 0x01},
> +	{0x0203, 0x90},
> +	{0x0205, 0x00},
> +};
> +
> +static const struct imx208_reg mode_968_548_60fps_regs[] = {
> +	{0x0340, 0x02},
> +	{0x0341, 0x39},
> +	{0x0342, 0x08},
> +	{0x0343, 0xC8},
> +	{0x034C, 0x03},
> +	{0x034D, 0xC8},
> +	{0x034E, 0x02},
> +	{0x034F, 0x24},
> +	{0x0381, 0x01},
> +	{0x0383, 0x03},
> +	{0x0385, 0x01},
> +	{0x0387, 0x03},
> +	{0x3048, 0x01},
> +	{0x3050, 0x02},
> +	{0x30D5, 0x03},
> +	{0x3301, 0x10},
> +	{0x3318, 0x75},
> +	{0x0202, 0x01},
> +	{0x0203, 0x90},
> +	{0x0205, 0x00},
> +};
> +
> +static const char * const imx208_test_pattern_menu[] = {
> +	"Disabled",
> +	"Solid Color",
> +	"100% Color Bar",
> +	"Fade to grey Color Bar",

s/g/G/

> +	"PN9",
> +	"Fixed Pattern1",
> +	"Fixed Pattern2",
> +	"Fixed Pattern3",
> +	"Fixed Pattern4",
> +	"Fixed Pattern5",
> +	"Fixed Pattern6"
> +};
> +
> +/* Configurations for supported link frequencies */
> +#define IMX208_LINK_FREQ_384MHZ	384000000ULL
> +#define IMX208_LINK_FREQ_192MHZ	192000000ULL
> +#define IMX208_LINK_FREQ_96MHZ  96000000ULL #define 
> +IMX208_LINK_FREQ_48MHZ  48000000ULL

You may put the link frequencies here, too, but they should be checked against the endpoint configuration ("link-frequencies" property; the
V4L2 fwnode framework parses them).

> +
> +enum {
> +	IMX208_LINK_FREQ_384MHZ_INDEX,
> +	IMX208_LINK_FREQ_96MHZ_INDEX,
> +};
> +
> +/*
> + * pixel_rate = link_freq * data-rate * nr_of_lanes / bits_per_sample
> + * data rate => double data rate; number of lanes => 2; bits per 
> +pixel => 10  */ static u64 link_freq_to_pixel_rate(u64 f) {
> +	f *= 2 * 2;
> +	do_div(f, 10);
> +
> +	return f;
> +}
> +
> +/* Menu items for LINK_FREQ V4L2 control */ static const s64 
> +link_freq_menu_items[] = {
> +	IMX208_LINK_FREQ_384MHZ,
> +	IMX208_LINK_FREQ_96MHZ,
> +};
> +
> +/* Link frequency configs */
> +static const struct imx208_link_freq_config link_freq_configs[] = {
> +	{
> +		.pixels_per_line = IMX208_PPL_384MHZ,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mipi_data_rate),
> +			.regs = mipi_data_rate,
> +		}
> +	},
> +	{
> +		.pixels_per_line = IMX208_PPL_96MHZ,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mipi_data_rate),
> +			.regs = mipi_data_rate,
> +		}
> +	},
> +};
> +
> +/* Mode configs */
> +static const struct imx208_mode supported_modes[] = {
> +	{
> +		.width = 1936,
> +		.height = 1096,
> +		.vts_def = IMX208_VTS_60FPS,
> +		.vts_min = IMX208_VTS_60FPS_MIN,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mode_1936x1096_60fps_regs),
> +			.regs = mode_1936x1096_60fps_regs,
> +		},
> +		.link_freq_index = 0,
> +	},
> +	{
> +		.width = 968,
> +		.height = 548,
> +		.vts_def = IMX208_VTS_BINNING,
> +		.vts_min = IMX208_VTS_BINNING_MIN,
> +		.reg_list = {
> +			.num_of_regs = ARRAY_SIZE(mode_968_548_60fps_regs),
> +			.regs = mode_968_548_60fps_regs,
> +		},
> +		.link_freq_index = IMX208_LINK_FREQ_96MHZ_INDEX,
> +	},
> +};
> +
> +struct imx208 {
> +	struct v4l2_subdev sd;
> +	struct media_pad pad;
> +
> +	struct v4l2_ctrl_handler ctrl_handler;
> +	/* V4L2 Controls */
> +	struct v4l2_ctrl *link_freq;
> +	struct v4l2_ctrl *pixel_rate;
> +	struct v4l2_ctrl *vblank;
> +	struct v4l2_ctrl *hblank;
> +	struct v4l2_ctrl *exposure;
> +
> +	/* Current mode */
> +	const struct imx208_mode *cur_mode;
> +
> +	/* Mutex for serialized access */
> +	struct mutex mutex;
> +
> +	/* Streaming on/off */
> +	bool streaming;
> +};
> +
> +static inline struct imx208 *to_imx208(struct v4l2_subdev *_sd)

No need for an underscore as this isn't a macro; up to you...

> +{
> +	return container_of(_sd, struct imx208, sd); }
> +
> +/* Read registers up to 2 at a time */ static int 
> +imx208_read_reg(struct imx208 *imx208, u16 reg, u32 len, u32 *val) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	struct i2c_msg msgs[2];
> +	u8 addr_buf[2] = { reg >> 8, reg & 0xff };
> +	u8 data_buf[4] = { 0, };
> +	int ret;
> +
> +	if (len > 4)
> +		return -EINVAL;
> +
> +	/* Write register address */
> +	msgs[0].addr = client->addr;
> +	msgs[0].flags = 0;
> +	msgs[0].len = ARRAY_SIZE(addr_buf);
> +	msgs[0].buf = addr_buf;
> +
> +	/* Read data from register */
> +	msgs[1].addr = client->addr;
> +	msgs[1].flags = I2C_M_RD;
> +	msgs[1].len = len;
> +	msgs[1].buf = &data_buf[4 - len];
> +
> +	ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
> +	if (ret != ARRAY_SIZE(msgs))
> +		return -EIO;
> +
> +	*val = get_unaligned_be32(data_buf);
> +
> +	return 0;
> +}
> +
> +/* Write registers up to 4 at a time */ static int 
> +imx208_write_reg(struct imx208 *imx208, u16 reg, u32 len, u32 val) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	u8 buf[6];
> +
> +	if (len > 4)
> +		return -EINVAL;
> +
> +	put_unaligned_be16(reg, buf);
> +	put_unaligned_be32(val << (8 * (4 - len)), buf + 2);
> +	if (i2c_master_send(client, buf, len + 2) != len + 2)
> +		return -EIO;
> +
> +	return 0;
> +}
> +
> +/* Write a list of registers */
> +static int imx208_write_regs(struct imx208 *imx208,
> +			      const struct imx208_reg *regs, u32 len) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	unsigned int i;
> +	int ret;
> +
> +	for (i = 0; i < len; i++) {
> +		ret = imx208_write_reg(imx208, regs[i].address, 1,
> +					regs[i].val);
> +		if (ret) {
> +			dev_err_ratelimited(
> +				&client->dev,
> +				"Failed to write reg 0x%4.4x. error = %d\n",
> +				regs[i].address, ret);
> +
> +			return ret;
> +		}
> +	}
> +
> +	return 0;
> +}
> +
> +/* Open sub-device */
> +static int imx208_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh 
> +*fh) {
> +	struct v4l2_mbus_framefmt *try_fmt =
> +		v4l2_subdev_get_try_format(sd, fh->pad, 0);
> +
> +	/* Initialize try_fmt */
> +	try_fmt->width = supported_modes[0].width;
> +	try_fmt->height = supported_modes[0].height;
> +	try_fmt->code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +	try_fmt->field = V4L2_FIELD_NONE;
> +
> +	return 0;
> +}
> +
> +
> +static int imx208_update_digital_gain(struct imx208 *imx208, u32 len, 
> +u32 val) {
> +	int ret;
> +
> +	ret = imx208_write_reg(imx208, IMX208_REG_GR_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);
> +	if (ret)
> +		return ret;
> +	ret = imx208_write_reg(imx208, IMX208_REG_GB_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);
> +	if (ret)
> +		return ret;
> +	ret = imx208_write_reg(imx208, IMX208_REG_R_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);
> +	if (ret)
> +		return ret;
> +	ret = imx208_write_reg(imx208, IMX208_REG_B_DIGITAL_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				val);

return imx208...();

> +	if (ret)
> +		return ret;
> +	return 0;
> +}
> +
> +static int imx208_enable_test_pattern(struct imx208 *imx208, u32 
> +pattern) {
> +	u32 val;
> +
> +	val = (pattern <= 4) ? pattern : pattern + 0xFB;
> +
> +	return imx208_write_reg(imx208, IMX208_REG_TEST_PATTERN_MODE,
> +				IMX208_REG_VALUE_16BIT, val);
> +}
> +
> +static int imx208_set_ctrl(struct v4l2_ctrl *ctrl) {
> +	struct imx208 *imx208 =
> +		container_of(ctrl->handler, struct imx208, ctrl_handler);
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	int ret = 0;

An empty line after variable declarations, please.

> +	/*
> +	 * Applying V4L2 control value only happens
> +	 * when power is up for streaming
> +	 */
> +	if (pm_runtime_get_if_in_use(&client->dev) <= 0)
> +		return 0;
> +
> +	switch (ctrl->id) {
> +	case V4L2_CID_ANALOGUE_GAIN:
> +		ret = imx208_write_reg(imx208, IMX208_REG_ANALOG_GAIN,
> +				IMX208_REG_VALUE_16BIT,
> +				ctrl->val);
> +		break;
> +	case V4L2_CID_EXPOSURE:
> +		ret = imx208_write_reg(imx208, IMX208_REG_EXPOSURE,
> +				IMX208_REG_VALUE_16BIT,
> +				ctrl->val);
> +		break;
> +	case V4L2_CID_DIGITAL_GAIN:
> +		ret = imx208_update_digital_gain(imx208, IMX208_REG_VALUE_16BIT,
> +				ctrl->val);
> +		break;
> +	case V4L2_CID_VBLANK:
> +		/* Update VTS that meets expected vertical blanking */
> +		ret = imx208_write_reg(imx208, IMX208_REG_VTS,
> +				       IMX208_REG_VALUE_16BIT,
> +				       imx208->cur_mode->height + ctrl->val);
> +		break;
> +	case V4L2_CID_TEST_PATTERN:
> +		ret = imx208_enable_test_pattern(imx208, ctrl->val);
> +		break;
> +	default:
> +		dev_info(&client->dev,
> +			 "ctrl(id:0x%x,val:0x%x) is not handled\n",
> +			 ctrl->id, ctrl->val);
> +		break;
> +	}
> +
> +	pm_runtime_put(&client->dev);
> +
> +	return ret;
> +}
> +
> +static const struct v4l2_ctrl_ops imx208_ctrl_ops = {
> +	.s_ctrl = imx208_set_ctrl,
> +};
> +
> +static int imx208_enum_mbus_code(struct v4l2_subdev *sd,
> +				  struct v4l2_subdev_pad_config *cfg,
> +				  struct v4l2_subdev_mbus_code_enum *code) {
> +	/* Only one bayer order(GRBG) is supported */
> +	if (code->index > 0)
> +		return -EINVAL;
> +
> +	code->code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +
> +	return 0;
> +}
> +
> +static int imx208_enum_frame_size(struct v4l2_subdev *sd,
> +				   struct v4l2_subdev_pad_config *cfg,
> +				   struct v4l2_subdev_frame_size_enum *fse) {
> +	if (fse->index >= ARRAY_SIZE(supported_modes))
> +		return -EINVAL;
> +
> +	if (fse->code != MEDIA_BUS_FMT_SRGGB10_1X10)
> +		return -EINVAL;
> +
> +	fse->min_width = supported_modes[fse->index].width;
> +	fse->max_width = fse->min_width;
> +	fse->min_height = supported_modes[fse->index].height;
> +	fse->max_height = fse->min_height;
> +
> +	return 0;
> +}
> +
> +static void imx208_update_pad_format(const struct imx208_mode *mode,
> +				      struct v4l2_subdev_format *fmt)

How about calling this e.g. imx208_mode_to_pad_format?

> +{
> +	fmt->format.width = mode->width;
> +	fmt->format.height = mode->height;
> +	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;
> +	fmt->format.field = V4L2_FIELD_NONE; }
> +
> +static int __imx208_get_pad_format(struct imx208 *imx208,
> +				     struct v4l2_subdev_pad_config *cfg,
> +				     struct v4l2_subdev_format *fmt) {
> +	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY)
> +		fmt->format = *v4l2_subdev_get_try_format(&imx208->sd, cfg,
> +							  fmt->pad);
> +	else
> +		imx208_update_pad_format(imx208->cur_mode, fmt);
> +
> +	return 0;
> +}
> +
> +static int imx208_get_pad_format(struct v4l2_subdev *sd,
> +				  struct v4l2_subdev_pad_config *cfg,
> +				  struct v4l2_subdev_format *fmt) {
> +	struct imx208 *imx208 = to_imx208(sd);
> +	int ret;
> +
> +	mutex_lock(&imx208->mutex);
> +	ret = __imx208_get_pad_format(imx208, cfg, fmt);
> +	mutex_unlock(&imx208->mutex);
> +
> +	return ret;
> +}
> +
> +static int imx208_set_pad_format(struct v4l2_subdev *sd,
> +		       struct v4l2_subdev_pad_config *cfg,
> +		       struct v4l2_subdev_format *fmt) {
> +	struct imx208 *imx208 = to_imx208(sd);
> +	const struct imx208_mode *mode;
> +	struct v4l2_mbus_framefmt *framefmt;
> +	s32 vblank_def;
> +	s32 vblank_min;
> +	s64 h_blank;
> +	s64 pixel_rate;
> +	s64 link_freq;
> +
> +	mutex_lock(&imx208->mutex);
> +
> +	fmt->format.code = MEDIA_BUS_FMT_SRGGB10_1X10;

This line seems redundant.

> +
> +	mode = v4l2_find_nearest_size(
> +		supported_modes, ARRAY_SIZE(supported_modes), width, height,
> +		fmt->format.width, fmt->format.height);
> +	imx208_update_pad_format(mode, fmt);
> +	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) {
> +		framefmt = v4l2_subdev_get_try_format(sd, cfg, fmt->pad);

You could move the declaration of framefmt here.

> +		*framefmt = fmt->format;
> +	} else {
> +		imx208->cur_mode = mode;
> +		__v4l2_ctrl_s_ctrl(imx208->link_freq, mode->link_freq_index);
> +
> +		link_freq = link_freq_menu_items[mode->link_freq_index];
> +		pixel_rate = link_freq_to_pixel_rate(link_freq);
> +		__v4l2_ctrl_s_ctrl_int64(imx208->pixel_rate, pixel_rate);
> +		/* Update limits and set FPS to default */
> +		vblank_def = imx208->cur_mode->vts_def -
> +			     imx208->cur_mode->height;
> +		vblank_min = imx208->cur_mode->vts_min -
> +			     imx208->cur_mode->height;
> +		__v4l2_ctrl_modify_range(
> +			imx208->vblank, vblank_min,
> +			IMX208_VTS_MAX - imx208->cur_mode->height, 1,
> +			vblank_def);
> +		__v4l2_ctrl_s_ctrl(imx208->vblank, vblank_def);
> +		h_blank =
> +			link_freq_configs[mode->link_freq_index].pixels_per_line
> +			 - imx208->cur_mode->width;
> +		__v4l2_ctrl_modify_range(imx208->hblank, h_blank,
> +					 h_blank, 1, h_blank);
> +	}
> +
> +	mutex_unlock(&imx208->mutex);
> +
> +	return 0;
> +}
> +
> +/* Start streaming */
> +static int imx208_start_streaming(struct imx208 *imx208) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	const struct imx208_reg_list *reg_list;
> +	int ret, link_freq_index;
> +
> +	/* Setup PLL */
> +	link_freq_index = imx208->cur_mode->link_freq_index;
> +	reg_list = &link_freq_configs[link_freq_index].reg_list;
> +	ret = imx208_write_regs(imx208, reg_list->regs, reg_list->num_of_regs);
> +	if (ret) {
> +		dev_err(&client->dev, "%s failed to set plls\n", __func__);
> +		return ret;
> +	}
> +
> +	/* Apply default values of current mode */
> +	reg_list = &imx208->cur_mode->reg_list;
> +	ret = imx208_write_regs(imx208, reg_list->regs, reg_list->num_of_regs);
> +	if (ret) {
> +		dev_err(&client->dev, "%s failed to set mode\n", __func__);
> +		return ret;
> +	}
> +
> +	/* Apply customized values from user */
> +	ret =  __v4l2_ctrl_handler_setup(imx208->sd.ctrl_handler);
> +	if (ret)
> +		return ret;
> +
> +	/* set stream on register */
> +	return imx208_write_reg(imx208, IMX208_REG_MODE_SELECT,
> +				 IMX208_REG_VALUE_08BIT,
> +				 IMX208_MODE_STREAMING);

Fits on previous line.

> +}
> +
> +/* Stop streaming */
> +static int imx208_stop_streaming(struct imx208 *imx208) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	int ret;
> +
> +	/* set stream off register */
> +	ret = imx208_write_reg(imx208, IMX208_REG_MODE_SELECT,
> +		IMX208_REG_VALUE_08BIT, IMX208_MODE_STANDBY);
> +	if (ret)
> +		dev_err(&client->dev, "%s failed to set stream\n", __func__);
> +
> +	/* Return success even if it was an error, as there is nothing the
> +	 * caller can do about it.
> +	 */
> +	return 0;
> +}
> +
> +static int imx208_set_stream(struct v4l2_subdev *sd, int enable) {
> +	struct imx208 *imx208 = to_imx208(sd);
> +	struct i2c_client *client = v4l2_get_subdevdata(sd);
> +	int ret = 0;
> +
> +	mutex_lock(&imx208->mutex);
> +	if (imx208->streaming == enable) {
> +		mutex_unlock(&imx208->mutex);
> +		return 0;
> +	}
> +
> +	if (enable) {
> +		ret = pm_runtime_get_sync(&client->dev);
> +		if (ret < 0) {
> +			pm_runtime_put_noidle(&client->dev);
> +			goto err_unlock;
> +		}
> +
> +		/*
> +		 * Apply default & customized values
> +		 * and then start streaming.
> +		 */
> +		ret = imx208_start_streaming(imx208);
> +		if (ret)
> +			goto err_rpm_put;
> +	} else {
> +		imx208_stop_streaming(imx208);
> +		pm_runtime_put(&client->dev);
> +	}
> +
> +	imx208->streaming = enable;
> +	mutex_unlock(&imx208->mutex);
> +
> +	return ret;
> +
> +err_rpm_put:
> +	pm_runtime_put(&client->dev);
> +err_unlock:
> +	mutex_unlock(&imx208->mutex);
> +
> +	return ret;
> +}
> +
> +static int __maybe_unused imx208_suspend(struct device *dev) {
> +	struct i2c_client *client = to_i2c_client(dev);
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +
> +	if (imx208->streaming)
> +		imx208_stop_streaming(imx208);
> +
> +	return 0;
> +}
> +
> +static int __maybe_unused imx208_resume(struct device *dev) {
> +	struct i2c_client *client = to_i2c_client(dev);
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +	int ret;
> +
> +	if (imx208->streaming) {
> +		ret = imx208_start_streaming(imx208);
> +		if (ret)
> +			goto error;
> +	}
> +
> +	return 0;
> +
> +error:
> +	imx208_stop_streaming(imx208);
> +	imx208->streaming = 0;
> +	return ret;
> +}
> +
> +/* Verify chip ID */
> +static int imx208_identify_module(struct imx208 *imx208) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	int ret;
> +	u32 val;
> +
> +	ret = imx208_read_reg(imx208, IMX208_REG_CHIP_ID,
> +			       IMX208_REG_VALUE_16BIT, &val);
> +	if (ret) {
> +		dev_err(&client->dev, "failed to read chip id %x\n",
> +			IMX208_CHIP_ID);
> +		return ret;
> +	}
> +
> +	if (val != IMX208_CHIP_ID) {
> +		dev_err(&client->dev, "chip id mismatch: %x!=%x\n",
> +			IMX208_CHIP_ID, val);
> +		return -EIO;
> +	}
> +
> +	dev_err(&client->dev, "chip id!: %x %x\n", IMX208_CHIP_ID, val);
> +
> +	return 0;
> +}
> +
> +static const struct v4l2_subdev_video_ops imx208_video_ops = {
> +	.s_stream = imx208_set_stream,
> +};
> +
> +static const struct v4l2_subdev_pad_ops imx208_pad_ops = {
> +	.enum_mbus_code = imx208_enum_mbus_code,
> +	.get_fmt = imx208_get_pad_format,
> +	.set_fmt = imx208_set_pad_format,
> +	.enum_frame_size = imx208_enum_frame_size, };
> +
> +static const struct v4l2_subdev_ops imx208_subdev_ops = {
> +	.video = &imx208_video_ops,
> +	.pad = &imx208_pad_ops,
> +};
> +
> +static const struct v4l2_subdev_internal_ops imx208_internal_ops = {
> +	.open = imx208_open,
> +};
> +
> +/* Initialize control handlers */
> +static int imx208_init_controls(struct imx208 *imx208) {
> +	struct i2c_client *client = v4l2_get_subdevdata(&imx208->sd);
> +	struct v4l2_ctrl_handler *ctrl_hdlr;
> +	s64 exposure_max;
> +	s64 vblank_def;
> +	s64 vblank_min;
> +	s64 pixel_rate_min;
> +	s64 pixel_rate_max;
> +	int ret;
> +
> +	ctrl_hdlr = &imx208->ctrl_handler;

Please move the assignment to the declaration. And perhaps call it e.g. hdl
--- the latter is up to you.

> +	ret = v4l2_ctrl_handler_init(ctrl_hdlr, 8);
> +	if (ret)
> +		return ret;
> +
> +	mutex_init(&imx208->mutex);
> +	ctrl_hdlr->lock = &imx208->mutex;
> +	imx208->link_freq = v4l2_ctrl_new_int_menu(ctrl_hdlr,
> +				&imx208_ctrl_ops,
> +				V4L2_CID_LINK_FREQ,
> +				ARRAY_SIZE(link_freq_menu_items) - 1,
> +				0,
> +				link_freq_menu_items);
> +
> +	if (imx208->link_freq)
> +		imx208->link_freq->flags |= V4L2_CTRL_FLAG_READ_ONLY;
> +
> +	pixel_rate_max = link_freq_to_pixel_rate(link_freq_menu_items[0]);
> +	pixel_rate_min = link_freq_to_pixel_rate(link_freq_menu_items[1]);
> +	/* By default, PIXEL_RATE is read only */
> +	imx208->pixel_rate = v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops,
> +					V4L2_CID_PIXEL_RATE,
> +					pixel_rate_min, pixel_rate_max,
> +					1, pixel_rate_max);
> +
> +
> +	vblank_def = imx208->cur_mode->vts_def - imx208->cur_mode->height;
> +	vblank_min = imx208->cur_mode->vts_min - imx208->cur_mode->height;
> +	imx208->vblank = v4l2_ctrl_new_std(
> +				ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_VBLANK,
> +				vblank_min,
> +				IMX208_VTS_MAX - imx208->cur_mode->height, 1,
> +				vblank_def);
> +
> +	imx208->hblank = v4l2_ctrl_new_std(
> +				ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_HBLANK,
> +				IMX208_PPL_384MHZ - imx208->cur_mode->width,
> +				IMX208_PPL_384MHZ - imx208->cur_mode->width,
> +				1,
> +				IMX208_PPL_384MHZ - imx208->cur_mode->width);
> +
> +	if (imx208->hblank)
> +		imx208->hblank->flags |= V4L2_CTRL_FLAG_READ_ONLY;
> +
> +	exposure_max = imx208->cur_mode->vts_def - 8;
> +	imx208->exposure = v4l2_ctrl_new_std(

imx208->exposure is unused. Please remove it.

> +				ctrl_hdlr, &imx208_ctrl_ops,
> +				V4L2_CID_EXPOSURE, IMX208_EXPOSURE_MIN,
> +				IMX208_EXPOSURE_MAX, IMX208_EXPOSURE_STEP,
> +				IMX208_EXPOSURE_DEFAULT);
> +
> +	v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_ANALOGUE_GAIN,
> +				IMX208_ANA_GAIN_MIN, IMX208_ANA_GAIN_MAX,
> +				IMX208_ANA_GAIN_STEP, IMX208_ANA_GAIN_DEFAULT);
> +
> +	v4l2_ctrl_new_std(ctrl_hdlr, &imx208_ctrl_ops, V4L2_CID_DIGITAL_GAIN,
> +				IMX208_DGTL_GAIN_MIN, IMX208_DGTL_GAIN_MAX,
> +				IMX208_DGTL_GAIN_STEP,
> +				IMX208_DGTL_GAIN_DEFAULT);
> +
> +	v4l2_ctrl_new_std_menu_items(ctrl_hdlr, &imx208_ctrl_ops,
> +				     V4L2_CID_TEST_PATTERN,
> +				     ARRAY_SIZE(imx208_test_pattern_menu) - 1,
> +				     0, 0, imx208_test_pattern_menu);
> +
> +	if (ctrl_hdlr->error) {
> +		ret = ctrl_hdlr->error;
> +		dev_err(&client->dev, "%s control init failed (%d)\n",
> +			__func__, ret);
> +		goto error;
> +	}
> +
> +	imx208->sd.ctrl_handler = ctrl_hdlr;
> +
> +	return 0;
> +
> +error:
> +	v4l2_ctrl_handler_free(ctrl_hdlr);
> +	mutex_destroy(&imx208->mutex);
> +
> +	return ret;
> +}
> +
> +static void imx208_free_controls(struct imx208 *imx208) {
> +	v4l2_ctrl_handler_free(imx208->sd.ctrl_handler);
> +	mutex_destroy(&imx208->mutex);
> +}
> +
> +static int imx208_probe(struct i2c_client *client) {
> +	struct imx208 *imx208;
> +	int ret;
> +	u32 val = 0;
> +
> +	device_property_read_u32(&client->dev, "clock-frequency", &val);
> +	if (val != 19200000)
> +		return -EINVAL;
> +
> +	imx208 = devm_kzalloc(&client->dev, sizeof(*imx208), GFP_KERNEL);
> +	if (!imx208)
> +		return -ENOMEM;
> +
> +	/* Initialize subdev */
> +	v4l2_i2c_subdev_init(&imx208->sd, client, &imx208_subdev_ops);
> +
> +	/* Check module identity */
> +	ret = imx208_identify_module(imx208);
> +	if (ret)
> +		return ret;
> +
> +	/* Set default mode to max resolution */
> +	imx208->cur_mode = &supported_modes[0];
> +
> +	ret = imx208_init_controls(imx208);
> +	if (ret)
> +		return ret;
> +
> +	/* Initialize subdev */
> +	imx208->sd.internal_ops = &imx208_internal_ops;
> +	imx208->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
> +	imx208->sd.entity.type = MEDIA_ENT_T_V4L2_SUBDEV_SENSOR;

We no longer have types, but functions. I.e. MEDIA_ENT_F_CAM_SENSOR . The type field no longer exists. (See below.)

> +
> +	/* Initialize source pad */
> +	imx208->pad.flags = MEDIA_PAD_FL_SOURCE;
> +	ret = media_entity_init(&imx208->sd.entity, 1, &imx208->pad, 0);

This function got renamed quite some time ago as media_entity_pads_init().
Have you compile tested this against the current media tree?

> +	if (ret) {
> +		dev_err(&client->dev, "%s failed:%d\n", __func__, ret);
> +		goto error_handler_free;
> +	}
> +
> +	ret = v4l2_async_register_subdev_sensor_common(&imx208->sd);
> +	if (ret < 0)
> +		goto error_media_entity;
> +
> +	pm_runtime_set_active(&client->dev);
> +	pm_runtime_enable(&client->dev);
> +	pm_runtime_idle(&client->dev);
> +
> +	return 0;
> +
> +error_media_entity:
> +	media_entity_cleanup(&imx208->sd.entity);
> +
> +error_handler_free:
> +	imx208_free_controls(imx208);
> +
> +	return ret;
> +}
> +
> +static int imx208_remove(struct i2c_client *client) {
> +	struct v4l2_subdev *sd = i2c_get_clientdata(client);
> +	struct imx208 *imx208 = to_imx208(sd);
> +
> +	v4l2_async_unregister_subdev(sd);
> +	media_entity_cleanup(&sd->entity);
> +	imx208_free_controls(imx208);
> +
> +	pm_runtime_disable(&client->dev);
> +	pm_runtime_set_suspended(&client->dev);
> +
> +	return 0;
> +}
> +
> +static const struct dev_pm_ops imx208_pm_ops = {
> +	SET_SYSTEM_SLEEP_PM_OPS(imx208_suspend, imx208_resume) };
> +
> +#ifdef CONFIG_ACPI
> +static const struct acpi_device_id imx208_acpi_ids[] = {
> +	{ "INT3478" },
> +	{ /* sentinel */ }
> +};
> +
> +MODULE_DEVICE_TABLE(acpi, imx208_acpi_ids); #endif
> +
> +static struct i2c_driver imx208_i2c_driver = {
> +	.driver = {
> +		.name = "imx208",
> +		.pm = &imx208_pm_ops,
> +		.acpi_match_table = ACPI_PTR(imx208_acpi_ids),
> +	},
> +	.probe_new = imx208_probe,
> +	.remove = imx208_remove,
> +};
> +
> +module_i2c_driver(imx208_i2c_driver);
> +
> +MODULE_AUTHOR("Yeh, Andy <andy.yeh@intel.com>"); MODULE_AUTHOR("Chen, 
> +Ping-chung <ping-chung.chen@intel.com>"); MODULE_DESCRIPTION("Sony 
> +IMX208 sensor driver"); MODULE_LICENSE("GPL v2");

--
Kind regards,

Sakari Ailus
sakari.ailus@linux.intel.com

^ permalink raw reply	[flat|nested] 8+ messages in thread

* Re: [PATCH] media: imx208: Add imx208 camera sensor driver
  2018-07-17  6:53   ` Chen, Ping-chung
@ 2018-07-19 13:22     ` Tomasz Figa
  0 siblings, 0 replies; 8+ messages in thread
From: Tomasz Figa @ 2018-07-19 13:22 UTC (permalink / raw)
  To: ping-chung.chen; +Cc: Sakari Ailus, Linux Media Mailing List, Yeh, Andy

Hi Ping-chung,

On Tue, Jul 17, 2018 at 3:53 PM Chen, Ping-chung
<ping-chung.chen@intel.com> wrote:
>
> Hi Sakari,
>
> Some of suggestions below has been added in to [PATCH v2] or [PATCH v3].
> We will fix others in PATCH v4 soon.

I don't see v2 or v3 on linux-media (or my inbox). Where were they sent?

I'd like to review the driver, but if there is already v3, it would
make sense to review that one.

Also, please refrain from top-posting on mailing lists, it's
considered bad manner (and makes threads difficult to read).

Best regards,
Tomasz

^ permalink raw reply	[flat|nested] 8+ messages in thread

end of thread, other threads:[~2018-07-19 14:05 UTC | newest]

Thread overview: 8+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2018-07-10  7:15 [PATCH] media: imx208: Add imx208 camera sensor driver Ping-chung Chen
2018-07-10  7:37 ` Yeh, Andy
2018-07-16 16:03   ` sakari.ailus
2018-07-10 11:53 ` kbuild test robot
2018-07-10 12:12 ` kbuild test robot
2018-07-16 15:58 ` Sakari Ailus
2018-07-17  6:53   ` Chen, Ping-chung
2018-07-19 13:22     ` Tomasz Figa

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).