Hi AngeloGioacchino, Thanks for the patchset. On Thu, Oct 29, 2020 at 06:29:46PM +0100, kholk11@xxxxxxxxx wrote: > From: AngeloGioacchino Del Regno <kholk11@xxxxxxxxx> > > This is a custom multi-aspect 25MegaPixels sensor from Sony, > found in many Sony Xperia smartphones from various eras. > > The camera assembly for this sensor usually (at least in Xperia > phones) has a lens that does not cover the entire sensor area, > which means that the real corners are blind and that, in many > lighting conditions, some more pixels in the corners are very > getting obscured (as no decent amount of light can get in)... > so, the maximum resolution that can produce a good image is: > - In 4:3 aspect ratio, 5520x4160 (23.0MP) > - In 16:9 aspect ratio, 5984x3392 (20.3MP). > > This sensor supports high frame rates (>=60FPS) when in binning > mode and both RAW8 and RAW10 output modes. > In this version of the driver, support has been provided for the > following resolutions: > W x H SZ MAX_FPS BINNING > - 5520x4160 23.0MP 23 No > - 5984x3392 20.3MP 26 No > - 2992x1696 3.8MP 60 Yes > - 1424x800 1.2MP 120 Yes > > Note 1: The "standard" camera assy for IMX300 also contains an > actuator (to focus the image), but this driver only manages the > actual image sensor. > > Note 2: The command tables for this sensor were reverse > engineered from a downstream "userspace driver" that has been > released in various versions on various Xperia smartphones. > Register layout seems to be only vaguely similar to IMX219, > which has a public datasheet from where some names for the > figured out registers were taken and added to the driver: > these names are probably not the right ones, but they surely > represent the intended thing. Can I ask you have you tested the driver? > > Signed-off-by: AngeloGioacchino Del Regno <kholk11@xxxxxxxxx> > --- > drivers/media/i2c/Kconfig | 13 + > drivers/media/i2c/Makefile | 1 + > drivers/media/i2c/imx300.c | 3089 ++++++++++++++++++++++++++++++++++++ > 3 files changed, 3103 insertions(+) > create mode 100644 drivers/media/i2c/imx300.c > > diff --git a/drivers/media/i2c/Kconfig b/drivers/media/i2c/Kconfig > index 878f66ef2719..032f45dfed16 100644 > --- a/drivers/media/i2c/Kconfig > +++ b/drivers/media/i2c/Kconfig > @@ -801,6 +801,19 @@ config VIDEO_IMX290 > To compile this driver as a module, choose M here: the > module will be called imx290. > > +config VIDEO_IMX300 > + tristate "Sony IMX300 Exmor RS sensor support" > + depends on I2C && VIDEO_V4L2 > + select MEDIA_CONTROLLER > + select VIDEO_V4L2_SUBDEV_API > + select V4L2_FWNODE > + help > + This is a Video4Linux2 sensor driver for the Sony > + IMX300 Exmor RS multi-aspect sensor. > + > + To compile this driver as a module, choose M here: the > + module will be called imx300. > + > config VIDEO_IMX319 > tristate "Sony IMX319 sensor support" > depends on I2C && VIDEO_V4L2 > diff --git a/drivers/media/i2c/Makefile b/drivers/media/i2c/Makefile > index f0a77473979d..8a3e003dea45 100644 > --- a/drivers/media/i2c/Makefile > +++ b/drivers/media/i2c/Makefile > @@ -117,6 +117,7 @@ obj-$(CONFIG_VIDEO_IMX219) += imx219.o > obj-$(CONFIG_VIDEO_IMX258) += imx258.o > obj-$(CONFIG_VIDEO_IMX274) += imx274.o > obj-$(CONFIG_VIDEO_IMX290) += imx290.o > +obj-$(CONFIG_VIDEO_IMX300) += imx300.o > obj-$(CONFIG_VIDEO_IMX319) += imx319.o > obj-$(CONFIG_VIDEO_IMX355) += imx355.o > obj-$(CONFIG_VIDEO_MAX9286) += max9286.o > diff --git a/drivers/media/i2c/imx300.c b/drivers/media/i2c/imx300.c > new file mode 100644 > index 000000000000..47de5fa3329c > --- /dev/null > +++ b/drivers/media/i2c/imx300.c > @@ -0,0 +1,3089 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * A V4L2 driver for Sony IMX300 Exmor RS multi-aspect image sensors. > + * Copyright (C) 2020, AngeloGioacchino Del Regno <kholk11@xxxxxxxxx> > + * > + * Based on Sony imx219 camera driver > + */ > + > +#include <linux/clk.h> > +#include <linux/delay.h> > +#include <linux/gpio/consumer.h> > +#include <linux/i2c.h> > +#include <linux/module.h> > +#include <linux/pm_runtime.h> > +#include <linux/regulator/consumer.h> > +#include <media/v4l2-ctrls.h> > +#include <media/v4l2-device.h> > +#include <media/v4l2-event.h> > +#include <media/v4l2-fwnode.h> > +#include <media/v4l2-mediabus.h> > +#include <asm/unaligned.h> > + > +#define IMX300_REG_VALUE_08BIT 1 > +#define IMX300_REG_VALUE_16BIT 2 > + > +/* > + * Supported external clock frequency is from (around) 6 to 26MHz > + * but there is no information about how to configure this sensor > + * for anything else but 24MHz, since there is no datasheet... > + */ > +#define IMX300_XCLK_FREQ_24M 24000000 > + > +/* Delay after XCLK/RESET during power up for sensor boot/stabilization */ > +#define IMX300_XCLK_STABLE_DELAY_US 10000 > +#define IMX300_XCLK_DELAY_RANGE_US 1000 > +#define IMX300_XCLR_MIN_DELAY_US 25000 > +#define IMX300_XCLR_DELAY_RANGE_US 1000 > + > +/* > + * Pixel rates: max resolution + max FPS uses high bw; low resolution > + * can use low bw in order to save power and limit sensor heating > + */ > +#define IMX300_HIGH_BW_PIXEL_RATE 624000000 > +#define IMX300_LOW_BW_PIXEL_RATE 384000000 > +#define IMX300_HIGH_BW_LINK_FREQ 780000000 > +#define IMX300_LOW_BW_LINK_FREQ 480000000 > + > +/* > + * About the Chip ID: > + * > + * IMX300 seems to be sort of flawed... scanning the registers reveals > + * that there's no reg having the expected 0x300 ChipID, like literally > + * all of the other Sony IMX sensors. > + * There seem to be no meaningful values and, even during reverse > + * engineering of three "userspace drivers" for this sensor, found on > + * three different series of smartphones, there is no trace of any ChipID. > + * These userspace drivers seem to be reading the Sensor ID at register > + * 0x0016 and comparing it to zero: it looks like being a dummy and not > + * reliable at all, since I imagine that *many* camera sensors may contain > + * zero at that register address, but that's still better than nothing... Have you checked what's at 0x0 (16 bits)? > + * > + * After all, if you don't have this camera sensor attached to your board, > + * you shouldn't be adding it in your DT... :( > + */ > +#define IMX300_REG_CHIP_ID 0x0016 > +#define IMX300_CHIP_ID 0x0 > + > +#define IMX300_REG_MODE_SELECT 0x0100 > +#define IMX300_MODE_STANDBY 0x00 > +#define IMX300_MODE_STREAMING 0x01 > + > +/* Orientation: changes bayer order */ > +#define IMX300_REG_ORIENTATION 0x0101 > + > +/* SW Reset: set to 1 to reset. Gets automatically set back to 0. */ > +#define IMX300_REG_SW_RESET 0x0103 > + > +/* Data format */ > +#define IMX300_REG_CSI_DATA_FORMAT_HI 0x0112 > +#define IMX300_REG_CSI_DATA_FORMAT_LO 0x0113 > +#define IMX300_CSI_DATA_FMT_8BIT 8 > +#define IMX300_CSI_DATA_FMT_10BIT 10 > + > +/* Output Set-up */ > +#define IMX300_REG_CSI_LANE_MODE 0x0114 > +#define IMX300_CSI_LANE_MODE_2LANE BIT(0) > +#define IMX300_CSI_LANE_MODE_4LANE (BIT(0) | BIT(1)) > + > +/* EXCK: Min 6MHz, Max 27MHz */ > +#define IMX300_REG_EXCK_FREQ_MHZ 0x0136 > + > +/* Exposure control */ > +#define IMX300_REG_EXPOSURE 0x0202 > +#define IMX300_EXPOSURE_MIN 4 > +#define IMX300_EXPOSURE_STEP 1 > +#define IMX300_EXPOSURE_DEFAULT 3556 > +#define IMX300_EXPOSURE_MAX 65535 > + > +/* Analog gain control */ > +#define IMX300_REG_ANALOG_GAIN 0x0204 > +#define IMX300_ANA_GAIN_MIN 0 > +#define IMX300_ANA_GAIN_MAX 0x1fff > +#define IMX300_ANA_GAIN_STEP 1 > +#define IMX300_ANA_GAIN_DEFAULT 0x0 > + > +/* Digital gain control */ > +#define IMX300_REG_GR_DIGITAL_GAIN 0x020e > +#define IMX300_REG_R_DIGITAL_GAIN 0x0210 > +#define IMX300_REG_B_DIGITAL_GAIN 0x0212 > +#define IMX300_REG_GB_DIGITAL_GAIN 0x0214 > +#define IMX300_DGTL_GAIN_MIN 0 > +#define IMX300_DGTL_GAIN_MAX 4095 > +#define IMX300_DGTL_GAIN_DEFAULT 1024 > +#define IMX300_DGTL_GAIN_STEP 1 > + > +/* Clock generator */ > +#define IMX300_REG_OP_PIX_CLK_DIV 0x0309 > +#define IMX300_REG_OP_SYS_CLK_DIV 0x030b > + > +/* V_TIMING internal */ > +#define IMX300_REG_VTS 0x0340 > +#define IMX300_VTS_MIN 41 > +#define IMX300_VTS_MAX (131071 - IMX300_VTS_MIN) > + > +/* Line Length */ > +#define IMX300_REG_LINE_LEN_PCK 0x0342 > +#define IMX300_LINELEN_MIN 7000 > +#define IMX300_LINELEN_MAX 8230 > +#define IMX300_LINELEN_STEP 1 > +#define IMX300_LINELEN_DEFAULT 7096 > + > +/* Output configuration */ > +#define IMX300_REG_X_ADDR_START 0x0344 > +#define IMX300_REG_Y_ADDR_START 0x0346 > +#define IMX300_REG_X_ADDR_END 0x0348 > +#define IMX300_REG_Y_ADDR_END 0x034a > +#define IMX300_REG_X_OUTPUT_SIZE 0x034c > +#define IMX300_REG_Y_OUTPUT_SIZE 0x034e > +#define IMX300_REG_X_ODD_INC 0x0383 > +#define IMX300_REG_Y_ODD_INC 0x0387 > + > +/* HBLANK control - read only */ > +#define IMX300_PPL_DEFAULT 8224 > + > +/* Test Pattern Control */ > +#define IMX300_REG_TEST_PATTERN 0x0600 > +#define IMX300_REG_TEST_PATTERN_WIDTH 0x0624 > +#define IMX300_REG_TEST_PATTERN_HEIGHT 0x0626 > +#define IMX300_TEST_PATTERN_DISABLE 0 > +#define IMX300_TEST_PATTERN_SOLID_COLOR 1 > +#define IMX300_TEST_PATTERN_COLOR_BARS 2 > +#define IMX300_TEST_PATTERN_GREY_COLOR 3 > +#define IMX300_TEST_PATTERN_PN9 4 > + > +/* Test pattern colour components */ > +#define IMX300_REG_TESTP_RED 0x0602 > +#define IMX300_REG_TESTP_GREENR 0x0604 > +#define IMX300_REG_TESTP_BLUE 0x0606 > +#define IMX300_REG_TESTP_GREENB 0x0608 > +#define IMX300_TESTP_COLOUR_MIN 0 > +#define IMX300_TESTP_COLOUR_MAX 0x03ff > +#define IMX300_TESTP_COLOUR_STEP 1 > +#define IMX300_TESTP_RED_DEFAULT IMX300_TESTP_COLOUR_MAX > +#define IMX300_TESTP_GREENR_DEFAULT 0 > +#define IMX300_TESTP_BLUE_DEFAULT 0 > +#define IMX300_TESTP_GREENB_DEFAULT 0 > + > +/* Binning */ > +#define IMX300_REG_BINNING_EN 0x0900 > +#define IMX300_REG_BINNING_TYPE 0x0902 > + > +/* > + * ** IMX300 native and active pixel array size ** > + * > + * Being this a multi-aspect sensor, the following native W/H apply, but > + * beware: the module assembly usually has a (round) lens that is shadowing > + * or covering the corners of the (square) image sensor, so the maximum > + * output resolution must be lower than the maximum sensor resolution > + * otherwise we get something like a view from a porthole... :) > + * > + * For 4:3 aspect ratio, max is: 5984x4140 (25MP) > + * For 16:9 aspect ratio, max is: 5984x3392 (20.3MP) > + */ > +#define IMX300_NATIVE_WIDTH 5980U > +#define IMX300_NATIVE_HEIGHT 4140U > +#define IMX300_PIXEL_ARRAY_LEFT 0U > +#define IMX300_PIXEL_ARRAY_TOP 0U > +#define IMX300_PIXEL_ARRAY_WIDTH 5520U > +#define IMX300_PIXEL_ARRAY_HEIGHT 4160U > + > +struct imx300_reg { > + u16 address; > + u16 val; > + u8 reg_len; > +}; > + > +struct imx300_reg_list { > + unsigned int num_of_regs; > + const struct imx300_reg *regs; > +}; > + > +/* Mode : resolution and related config&values */ > +struct imx300_mode { > + /* Frame width */ > + unsigned int width; > + /* Frame height */ > + unsigned int height; > + /* Maximum achievable FPS */ > + unsigned int max_fps; > + /* Needs high data rate */ > + bool high_bw; > + /* Needs binning setup */ > + bool binned; > + > + /* Analog crop rectangle. */ > + struct v4l2_rect crop; > + > + /* V-timing default */ > + unsigned int vts_def; > + > + /* Default register values */ > + struct imx300_reg_list reg_list; > +}; > + > +/* > + * I have no idea what this very long initialization sequence is for... > + * but missing writes in this makes the sensor to output corrupted > + * frames or nothing at all... > + */ > +static const struct imx300_reg init_sequence[] = { > + { IMX300_REG_EXCK_FREQ_MHZ, 24, IMX300_REG_VALUE_08BIT }, > + { 0x0137, 0x00, IMX300_REG_VALUE_08BIT }, This is part of the same 16-bit register. > + { 0x0101, 0x03, IMX300_REG_VALUE_08BIT }, This enables horizontal and vertical flip. You could use the same macro for the register address as in the common list. What's the difference between the initial and common registers? > + { 0x0138, 0x01, IMX300_REG_VALUE_08BIT }, And this one enables the temperature sensor. > + { 0x3154, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3155, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x3156, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3157, 0x66, IMX300_REG_VALUE_08BIT }, > + { 0x305d, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x31b0, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3208, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3210, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x406c, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x406d, 0x39, IMX300_REG_VALUE_08BIT }, > + { 0x430d, 0xaa, IMX300_REG_VALUE_08BIT }, > + { 0x4313, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x431e, 0x88, IMX300_REG_VALUE_08BIT }, > + { 0x431f, 0x8f, IMX300_REG_VALUE_08BIT }, > + { 0x4321, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x4324, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x4325, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x4326, 0x28, IMX300_REG_VALUE_08BIT }, > + { 0x4328, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0x4329, 0x5b, IMX300_REG_VALUE_08BIT }, > + { 0x432c, 0x06, IMX300_REG_VALUE_08BIT }, > + { 0x432d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x432e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x4554, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x4811, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4819, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x481d, 0x9d, IMX300_REG_VALUE_08BIT }, > + { 0x481f, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4838, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x4839, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x483a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x483b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x4871, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4877, 0x9b, IMX300_REG_VALUE_08BIT }, > + { 0x487d, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4883, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4891, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4893, 0x9d, IMX300_REG_VALUE_08BIT }, > + { 0x48ad, 0x9d, IMX300_REG_VALUE_08BIT }, > + { 0x48af, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x48b1, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x48b5, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x48bb, 0x9b, IMX300_REG_VALUE_08BIT }, > + { 0x48c1, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x48c7, 0x9b, IMX300_REG_VALUE_08BIT }, > + { 0x48c9, 0x95, IMX300_REG_VALUE_08BIT }, > + { 0x48cb, 0x94, IMX300_REG_VALUE_08BIT }, > + { 0x4bd7, 0x24, IMX300_REG_VALUE_08BIT }, > + { 0x48dd, 0x95, IMX300_REG_VALUE_08BIT }, > + { 0x48e5, 0x97, IMX300_REG_VALUE_08BIT }, > + { 0x49d1, 0x9a, IMX300_REG_VALUE_08BIT }, > + { 0x4a00, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x4a01, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x4a02, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x4a03, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x4a1d, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4a1f, 0x9b, IMX300_REG_VALUE_08BIT }, > + { 0x4a27, 0x97, IMX300_REG_VALUE_08BIT }, > + { 0x4a29, 0x12, IMX300_REG_VALUE_08BIT }, > + { 0x4a2f, 0x97, IMX300_REG_VALUE_08BIT }, > + { 0x4a4d, 0x9d, IMX300_REG_VALUE_08BIT }, > + { 0x4a4f, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4a51, 0x9d, IMX300_REG_VALUE_08BIT }, > + { 0x4a53, 0x99, IMX300_REG_VALUE_08BIT }, > + { 0x4a6d, 0x9e, IMX300_REG_VALUE_08BIT }, > + { 0x4a6f, 0x9d, IMX300_REG_VALUE_08BIT }, > + { 0x4be7, 0x24, IMX300_REG_VALUE_08BIT }, > + { 0x4c07, 0x24, IMX300_REG_VALUE_08BIT }, > + { 0x4c17, 0x24, IMX300_REG_VALUE_08BIT }, > + { 0x4c97, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x4ca7, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x4cc7, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x4cd7, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x4d57, 0x12, IMX300_REG_VALUE_08BIT }, > + { 0x4d67, 0x12, IMX300_REG_VALUE_08BIT }, > + { 0x4d87, 0x12, IMX300_REG_VALUE_08BIT }, > + { 0x4d97, 0x12, IMX300_REG_VALUE_08BIT }, > + { 0x4db7, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x4dc7, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x4de7, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x4df7, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x4e15, 0x43, IMX300_REG_VALUE_08BIT }, > + { 0x4e25, 0x43, IMX300_REG_VALUE_08BIT }, > + { 0x4e45, 0x43, IMX300_REG_VALUE_08BIT }, > + { 0x4e55, 0x43, IMX300_REG_VALUE_08BIT }, > + { 0x4f40, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x4f42, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x4f48, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x4f4e, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x505c, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x505d, 0x36, IMX300_REG_VALUE_08BIT }, > + { 0x505f, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x5069, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x5e6a, 0xfe, IMX300_REG_VALUE_08BIT }, > + { 0x5e70, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6153, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x6156, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x656a, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x656b, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x7300, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9006, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0xb000, 0x6e, IMX300_REG_VALUE_08BIT }, > + { 0xb001, 0xfd, IMX300_REG_VALUE_08BIT }, > + { 0xb002, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0xb003, 0x2a, IMX300_REG_VALUE_08BIT }, > + { 0xb004, 0x83, IMX300_REG_VALUE_08BIT }, > + { 0xb005, 0x41, IMX300_REG_VALUE_08BIT }, > + { 0xb006, 0x83, IMX300_REG_VALUE_08BIT }, > + { 0xb007, 0x43, IMX300_REG_VALUE_08BIT }, > + { 0xb008, 0x82, IMX300_REG_VALUE_08BIT }, > + { 0xb009, 0x49, IMX300_REG_VALUE_08BIT }, > + { 0xb00a, 0x82, IMX300_REG_VALUE_08BIT }, > + { 0xb00b, 0x4a, IMX300_REG_VALUE_08BIT }, > + { 0xb00c, 0x6e, IMX300_REG_VALUE_08BIT }, > + { 0xb00d, 0x93, IMX300_REG_VALUE_08BIT }, > + { 0xb00e, 0x82, IMX300_REG_VALUE_08BIT }, > + { 0xb00f, 0x41, IMX300_REG_VALUE_08BIT }, > + { 0xb010, 0x82, IMX300_REG_VALUE_08BIT }, > + { 0xb011, 0x4b, IMX300_REG_VALUE_08BIT }, > + { 0xb100, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0xb101, 0x5e, IMX300_REG_VALUE_08BIT }, > + { 0xb102, 0x80, IMX300_REG_VALUE_08BIT }, > + { 0xb103, 0x9a, IMX300_REG_VALUE_08BIT }, > + { 0xb104, 0xaf, IMX300_REG_VALUE_08BIT }, > + { 0xb105, 0xc0, IMX300_REG_VALUE_08BIT }, > + { 0xb106, 0xcd, IMX300_REG_VALUE_08BIT }, > + { 0xb107, 0xd7, IMX300_REG_VALUE_08BIT }, > + { 0xb123, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xb125, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xb127, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xb129, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xb12b, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xb136, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb137, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb138, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb139, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb13a, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb13b, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb13c, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb13d, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb13e, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb13f, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb140, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xb141, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xb142, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xb143, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb144, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb145, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb146, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb147, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb148, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb149, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb14a, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb14b, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb14c, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb14d, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb14e, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb14f, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb150, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb151, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb152, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb153, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xb154, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0xb155, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0xb156, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0xb157, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb158, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb159, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb15a, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb15b, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb15c, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb15d, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0xb210, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x313c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3198, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x31a0, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x31a1, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x31a2, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x31a3, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x31a8, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x3290, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6d20, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6d21, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x6d22, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x6d23, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x6d26, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6d27, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6e07, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x6e35, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6e55, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6e7c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6e93, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6efb, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x7c68, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x845c, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x8469, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9257, 0x96, IMX300_REG_VALUE_08BIT }, > + { 0x9258, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x933a, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x933b, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x933d, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x933e, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x933f, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x934b, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x934c, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9356, 0x8c, IMX300_REG_VALUE_08BIT }, > + { 0x9357, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9358, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9359, 0x8c, IMX300_REG_VALUE_08BIT }, > + { 0x935a, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x935b, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9360, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9361, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9362, 0x8c, IMX300_REG_VALUE_08BIT }, > + { 0x9363, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9364, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9365, 0x8c, IMX300_REG_VALUE_08BIT }, > + { 0x9366, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9367, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x940d, 0x07, IMX300_REG_VALUE_08BIT }, > + { 0x940e, 0x07, IMX300_REG_VALUE_08BIT }, > + { 0x9414, 0x06, IMX300_REG_VALUE_08BIT }, > + { 0x942b, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x942c, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x942d, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x942e, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x942f, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9430, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9431, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9432, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9433, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9434, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9435, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9436, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9437, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9438, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9439, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x943b, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x943d, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x943f, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9441, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9443, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9445, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9447, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9449, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x944b, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x944d, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x944f, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9451, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9452, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9454, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9456, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9457, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9458, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9459, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x945a, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x945b, 0x07, IMX300_REG_VALUE_08BIT }, > + { 0x945c, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x945d, 0x07, IMX300_REG_VALUE_08BIT }, > + { 0x945e, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x945f, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9460, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9461, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9462, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9463, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9464, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9465, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9466, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x947c, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x947d, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9480, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9481, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9503, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x9504, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x9505, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x9506, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9507, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9508, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9526, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9527, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9528, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9619, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x961b, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x961d, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x961f, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9621, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9623, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9625, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9627, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9629, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x962b, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x962d, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x962f, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9719, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x971b, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x971d, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x971f, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9721, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9723, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9725, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9727, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9729, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x972b, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x972d, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x972f, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9901, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9903, 0x23, IMX300_REG_VALUE_08BIT }, > + { 0x9905, 0x23, IMX300_REG_VALUE_08BIT }, > + { 0x9906, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9907, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9908, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9909, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x990a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x990b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x990d, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x990f, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9911, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9913, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9915, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9917, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9919, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x991b, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x991d, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x991f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9921, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9923, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9925, 0x23, IMX300_REG_VALUE_08BIT }, > + { 0x9927, 0x23, IMX300_REG_VALUE_08BIT }, > + { 0x9929, 0x23, IMX300_REG_VALUE_08BIT }, > + { 0x992b, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x992d, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x992f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9931, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9933, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9935, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9937, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9939, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x993b, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9943, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9945, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9947, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9949, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x994b, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x994d, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x994f, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9951, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9953, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9955, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9957, 0x13, IMX300_REG_VALUE_08BIT }, > + { 0x9959, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x995a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x995b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x995c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x996b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x996d, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x996f, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9971, 0xc8, IMX300_REG_VALUE_08BIT }, > + { 0x9973, 0x32, IMX300_REG_VALUE_08BIT }, > + { 0x9975, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9976, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9979, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x997a, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x997b, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9981, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9983, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9985, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x99a4, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x99a5, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x99a6, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x99a7, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x99a8, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x99a9, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x99aa, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x99ab, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x99ac, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x99ad, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x99ae, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x99af, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x99b0, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x99b1, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x99b2, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x99b3, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x99b4, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x99b5, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x99bb, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x99bd, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x99bf, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x99c0, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x99c1, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x99c2, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x99c6, 0x3c, IMX300_REG_VALUE_08BIT }, > + { 0x99c7, 0x3c, IMX300_REG_VALUE_08BIT }, > + { 0x99c8, 0x3c, IMX300_REG_VALUE_08BIT }, > + { 0x99c9, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x99ca, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x99cb, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9a01, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9a03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a05, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a07, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9a09, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9a0b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9a0d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9a0f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9a11, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9a13, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9a15, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9a17, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9a19, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9a1b, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9a1d, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9a1f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9a21, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9a23, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9a25, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a27, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a29, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a2b, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9a2d, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9a2f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a31, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9a33, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9a35, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9a37, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9a39, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9a3b, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9a3d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a3f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a41, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a4f, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9a51, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9a53, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9a55, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9a57, 0x13, IMX300_REG_VALUE_08BIT }, > + { 0x9a59, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x9a5a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a5b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a5c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a6b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9a6d, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9a6f, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9a71, 0xc8, IMX300_REG_VALUE_08BIT }, > + { 0x9a73, 0x32, IMX300_REG_VALUE_08BIT }, > + { 0x9a75, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9a79, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9a7a, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9a7b, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9a81, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a83, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9a85, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9aa4, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9aa5, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9aa6, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9aa7, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9aa8, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9aa9, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9aaa, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9aab, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9aac, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9aad, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9aae, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9aaf, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9ab0, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9ab1, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9ab2, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9ab3, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x9ab4, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x9ab5, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x9ab6, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9ab7, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9ab8, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9abb, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9abd, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9abf, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9ac0, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9ac1, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9ac2, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9ac6, 0x2d, IMX300_REG_VALUE_08BIT }, > + { 0x9ac7, 0x2d, IMX300_REG_VALUE_08BIT }, > + { 0x9ac8, 0x2d, IMX300_REG_VALUE_08BIT }, > + { 0x9ac9, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9aca, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9acb, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9b01, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9b03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b05, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b07, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9b08, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9b09, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9b0a, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9b0b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9b0d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9b0f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9b11, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9b13, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9b15, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9b17, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9b19, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9b1b, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9b1d, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9b1f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9b21, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9b23, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9b25, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b27, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b29, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b2b, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9b2d, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9b2f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b31, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9b33, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9b35, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9b37, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9b39, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9b3b, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9b43, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9b45, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9b47, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9b49, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9b4b, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9b4d, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9b4f, 0x2d, IMX300_REG_VALUE_08BIT }, > + { 0x9b51, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9b53, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9b55, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9b57, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0x9b59, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9b5a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9b5b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9b5c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9b5d, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9b5e, 0x0e, IMX300_REG_VALUE_08BIT }, > + { 0x9b60, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9b61, 0x0e, IMX300_REG_VALUE_08BIT }, > + { 0x9b6b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9b6d, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9b6f, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9b71, 0xc8, IMX300_REG_VALUE_08BIT }, > + { 0x9b73, 0x32, IMX300_REG_VALUE_08BIT }, > + { 0x9b75, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9b76, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9b79, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9b7a, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9b7b, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9b81, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b83, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9b85, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9bb0, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9bb1, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9bb2, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9bb3, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x9bb4, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x9bb5, 0x30, IMX300_REG_VALUE_08BIT }, > + { 0x9bbb, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9bbd, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9bbf, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9bc0, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9bc1, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9bc2, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9bc6, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9bc7, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9bc8, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9bc9, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9bca, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9bcb, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9bcc, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9bcd, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9bce, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9c01, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9c03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c05, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c07, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9c09, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9c0b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9c0d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9c0f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9c11, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9c13, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9c15, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9c17, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9c19, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9c1b, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9c1d, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9c1f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9c21, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9c23, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9c25, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c27, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c29, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c2b, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9c2d, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9c2f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c31, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9c33, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9c35, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9c37, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9c39, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9c3b, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9c3d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c3f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c41, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c4f, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9c51, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9c53, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9c55, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9c57, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0x9c59, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9c5a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c5b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c5c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c6b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9c6d, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9c6f, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9c71, 0xc8, IMX300_REG_VALUE_08BIT }, > + { 0x9c73, 0x32, IMX300_REG_VALUE_08BIT }, > + { 0x9c75, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9c79, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9c7a, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9c7b, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9c81, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c83, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c85, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9c87, 0x48, IMX300_REG_VALUE_08BIT }, > + { 0x9c89, 0x48, IMX300_REG_VALUE_08BIT }, > + { 0x9c8b, 0x48, IMX300_REG_VALUE_08BIT }, > + { 0x9ca4, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9ca5, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9ca6, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9ca7, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9ca8, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9ca9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9caa, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9cab, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9cac, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9cad, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9cae, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9caf, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9cb0, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9cb1, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9cb2, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9cb3, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9cb4, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9cb5, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x9cbb, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9cbd, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9cbf, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9cc0, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9cc1, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9cc2, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x9cc6, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9cc7, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9cc8, 0x18, IMX300_REG_VALUE_08BIT }, > + { 0x9cc9, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9cca, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9ccb, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x9d01, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9d03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d05, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d07, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9d09, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9d0b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9d0d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d0f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d11, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d13, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d15, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d17, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d19, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9d1b, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9d1d, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9d1f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d21, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d23, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d25, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d27, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d29, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d2b, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9d2d, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9d2f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d31, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d33, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d35, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d37, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9d39, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9d3b, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9d3d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9d3f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9d41, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9d4f, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9d50, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9d51, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9d53, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9d55, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9d57, 0x13, IMX300_REG_VALUE_08BIT }, > + { 0x9d59, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x9d5b, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9d5d, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d5f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d61, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9d63, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d65, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9d67, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9d69, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9d6b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9d6d, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9d6f, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9d71, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9d73, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d75, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d77, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d79, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d7b, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d7d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9d7f, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d81, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d83, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d85, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d87, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d89, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9d8b, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9d8d, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9d8f, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9d91, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9d93, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9d95, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9d97, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d99, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d9b, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d9d, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9d9f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9da1, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e01, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9e03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e05, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e07, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9e09, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9e0b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9e0d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e0f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e11, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e13, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e15, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e17, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e19, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9e1b, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9e1d, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9e1f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e21, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e23, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e25, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e27, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e29, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e2b, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9e2d, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9e2f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e31, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e33, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e35, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e37, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9e39, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9e3b, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9e3d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9e3f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9e41, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9e4f, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9e51, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9e53, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9e55, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9e57, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0x9e59, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x9e5b, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9e5d, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e5f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e61, 0x35, IMX300_REG_VALUE_08BIT }, > + { 0x9e63, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e65, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9e67, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9e69, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9e6b, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9e6d, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0x9e6f, 0x1b, IMX300_REG_VALUE_08BIT }, > + { 0x9e71, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x9e73, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e75, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e77, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e79, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e7b, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e7d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9e7f, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e81, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e83, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e85, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e87, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e89, 0x64, IMX300_REG_VALUE_08BIT }, > + { 0x9e8b, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9e8d, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9e8f, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9e91, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x9e93, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x9e95, 0x65, IMX300_REG_VALUE_08BIT }, > + { 0x9e97, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e99, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e9b, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e9d, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9e9f, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9ea1, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x9f01, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f05, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f07, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f09, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f0b, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f0d, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9f0f, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9f11, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f13, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0x9f15, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x9f17, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f19, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9f1b, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9f1d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9f1f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9f21, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9f23, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x9f25, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9f27, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9f29, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9f2b, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0x9f2d, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0x9f2f, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0x9f31, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f33, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f35, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f37, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f39, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f3b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f3c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f3d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f3e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f41, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f43, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9f45, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9f47, 0xc8, IMX300_REG_VALUE_08BIT }, > + { 0x9f49, 0x32, IMX300_REG_VALUE_08BIT }, > + { 0x9f4b, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9f4d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f4f, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9f51, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9f53, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f55, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9f57, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x9f59, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9f5b, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9f5d, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9f5f, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x9f61, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9f63, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x9f64, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9f65, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9f66, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0x9f6a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f6b, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f6c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f6d, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f6e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f6f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x9f77, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9f78, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f79, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9f7b, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9f7d, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0x9f7e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9f7f, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x9f81, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0x9f83, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9f85, 0x13, IMX300_REG_VALUE_08BIT }, > + { 0x9f87, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x9f89, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0x9f8b, 0x13, IMX300_REG_VALUE_08BIT }, > + { 0x9f8d, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x9f8f, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0x9fa6, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9fa7, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9fa8, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9fa9, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9faa, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9fab, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x9fac, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9fad, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9fae, 0x3f, IMX300_REG_VALUE_08BIT }, > + { 0x9faf, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9fb0, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x9fb1, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa001, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa003, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa005, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa007, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa009, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa00b, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa00d, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0xa00f, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0xa011, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa013, 0x2f, IMX300_REG_VALUE_08BIT }, > + { 0xa015, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0xa017, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa019, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0xa01b, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0xa01d, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0xa01f, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0xa021, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0xa023, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0xa025, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0xa027, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0xa029, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0xa02b, 0x6b, IMX300_REG_VALUE_08BIT }, > + { 0xa02d, 0x7c, IMX300_REG_VALUE_08BIT }, > + { 0xa02f, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0xa031, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa033, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa035, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa037, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa039, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa03b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa03c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa03d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa03e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa041, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa043, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0xa045, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0xa047, 0xc8, IMX300_REG_VALUE_08BIT }, > + { 0xa049, 0x32, IMX300_REG_VALUE_08BIT }, > + { 0xa04b, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xa04d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa04f, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0xa051, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0xa053, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa055, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0xa057, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0xa059, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0xa05b, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xa05d, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xa05f, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0xa061, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xa063, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0xa064, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0xa065, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0xa066, 0xe0, IMX300_REG_VALUE_08BIT }, > + { 0xa067, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0xa068, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0xa069, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0xa06b, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa06d, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa06f, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xa071, 0x48, IMX300_REG_VALUE_08BIT }, > + { 0xa073, 0x48, IMX300_REG_VALUE_08BIT }, > + { 0xa075, 0x48, IMX300_REG_VALUE_08BIT }, > + { 0xa077, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0xa079, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0xa07b, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0xa07d, 0x42, IMX300_REG_VALUE_08BIT }, > + { 0xa07f, 0x0b, IMX300_REG_VALUE_08BIT }, > + { 0xa081, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0xa083, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0xa085, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0xa087, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0xa089, 0x5a, IMX300_REG_VALUE_08BIT }, > + { 0xa08b, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0xa08d, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0xa08f, 0xa0, IMX300_REG_VALUE_08BIT }, > + { 0xa091, 0x3a, IMX300_REG_VALUE_08BIT }, > + { 0xa093, 0x3a, IMX300_REG_VALUE_08BIT }, > + { 0xa095, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xa097, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xa099, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0xa0a9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa0aa, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa0ab, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa0af, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa0b0, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xa0b1, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xf800, 0x5c, IMX300_REG_VALUE_08BIT }, > + { 0xf801, 0x5c, IMX300_REG_VALUE_08BIT }, > + { 0xf802, 0x92, IMX300_REG_VALUE_08BIT }, > + { 0xf803, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xf804, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xf805, 0xbc, IMX300_REG_VALUE_08BIT }, > + { 0xf806, 0x22, IMX300_REG_VALUE_08BIT }, > + { 0xf807, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xf808, 0x1c, IMX300_REG_VALUE_08BIT }, > + { 0xf809, 0x80, IMX300_REG_VALUE_08BIT }, > + { 0xf80a, 0xfa, IMX300_REG_VALUE_08BIT }, > + { 0xf80b, 0x21, IMX300_REG_VALUE_08BIT }, > + { 0xf80c, 0x55, IMX300_REG_VALUE_08BIT }, > + { 0xf80d, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0xf80e, 0xba, IMX300_REG_VALUE_08BIT }, > + { 0xf80f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0xf810, 0x81, IMX300_REG_VALUE_08BIT }, > + { 0xf811, 0xd1, IMX300_REG_VALUE_08BIT }, > + { 0xf812, 0x31, IMX300_REG_VALUE_08BIT }, > + { 0xf813, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x5869, 0x01, IMX300_REG_VALUE_08BIT }, > +}; > + > +/* > + * Magic registers sequence, common for all of the specified resolutions. > + * > + * Note: Binaries seem to send data to 0x0220, 0x0221 when enabling HDR mode... > + */ > +static const struct imx300_reg mode_common_regs[] = { > + { IMX300_REG_ORIENTATION, 0x03, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_CSI_DATA_FORMAT_HI, 10, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_CSI_DATA_FORMAT_LO, 10, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_CSI_LANE_MODE, IMX300_CSI_LANE_MODE_4LANE, > + IMX300_REG_VALUE_08BIT }, > + { 0x0221, 0x11, IMX300_REG_VALUE_08BIT }, > + { 0x0381, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0383, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0385, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0387, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0401, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0404, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0405, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0408, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0409, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040c, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x040d, 0x90, IMX300_REG_VALUE_08BIT }, > + { 0x040e, 0x10, IMX300_REG_VALUE_08BIT }, > + { 0x040f, 0x40, IMX300_REG_VALUE_08BIT }, > + { 0x30e4, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x30e5, 0x90, IMX300_REG_VALUE_08BIT }, > + { 0x30e6, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x30e7, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x30e8, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x30e9, 0x90, IMX300_REG_VALUE_08BIT }, > + { 0x30ea, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x30eb, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x30ec, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x30ed, 0x90, IMX300_REG_VALUE_08BIT }, > + { 0x30ee, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x30ef, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x30f4, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x30f5, 0x90, IMX300_REG_VALUE_08BIT }, > + { 0x30f6, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x30f7, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3294, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3295, 0xe8, IMX300_REG_VALUE_08BIT }, > + { 0x3296, 0x16, IMX300_REG_VALUE_08BIT }, > + { 0x3297, 0x77, IMX300_REG_VALUE_08BIT }, > + { 0x0210, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0211, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0212, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0213, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0214, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0215, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0216, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0217, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3220, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3006, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3007, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x31e0, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x31e1, 0xff, IMX300_REG_VALUE_08BIT }, > + { 0x31e4, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x4301, 0x94, IMX300_REG_VALUE_08BIT }, > + { 0x4302, 0x15, IMX300_REG_VALUE_08BIT }, > + { 0x4303, 0x44, IMX300_REG_VALUE_08BIT }, > + { 0x4304, 0x09, IMX300_REG_VALUE_08BIT }, > + { 0x4305, 0x07, IMX300_REG_VALUE_08BIT }, > + { 0x4306, 0x87, IMX300_REG_VALUE_08BIT }, > + { 0x4307, 0x88, IMX300_REG_VALUE_08BIT }, > + { 0x4308, 0xba, IMX300_REG_VALUE_08BIT }, > + { 0x4309, 0x21, IMX300_REG_VALUE_08BIT }, > + { 0x4f7e, 0x12, IMX300_REG_VALUE_08BIT }, > + { 0x0220, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x0222, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0224, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0225, 0xf4, IMX300_REG_VALUE_08BIT }, > + { 0x3000, 0x54, IMX300_REG_VALUE_08BIT }, > + { 0x3001, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0600, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0601, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3022, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0b05, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0b06, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3018, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3019, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x301a, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3025, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3130, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3004, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x30a2, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0b00, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3250, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3251, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3011, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3013, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3129, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3125, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3127, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3140, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3141, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3142, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x314f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0b8e, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0b8f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0b90, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0b91, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0b92, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0b93, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0b94, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x0b95, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3121, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3123, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31b0, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3158, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3159, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x315a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x315b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x315c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x315d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x315e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x315f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3160, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3161, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3162, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3163, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3164, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3165, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3166, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3167, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3168, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3169, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x316a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x316b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x316c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x316d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x316e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x316f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3170, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3171, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3172, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3173, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3174, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3175, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3176, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3177, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3178, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3179, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x317a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x317b, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x317c, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x317d, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x317e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x317f, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3180, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3181, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3182, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3183, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3184, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3185, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3186, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3187, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31b4, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31b5, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31b6, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31b7, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31b8, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31b9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31ba, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31bb, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3291, 0x01, IMX300_REG_VALUE_08BIT }, > +}; > + > +/* Data rate configuration: HIGH (780MHz) */ > +static const struct imx300_reg mipi_data_rate_1560mbps[] = { > + { 0x0301, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x0303, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x0305, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x0306, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0307, 0xbe, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_OP_PIX_CLK_DIV, 10, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_OP_SYS_CLK_DIV, 1, IMX300_REG_VALUE_08BIT }, > + { 0x030d, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x030e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x030f, 0xbb, IMX300_REG_VALUE_08BIT }, > +}; > + > +/* Data rate configuration: LOW (480MHz) */ > +static const struct imx300_reg mipi_data_rate_960mbps[] = { > + { 0x0301, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x0303, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x0305, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x0306, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0307, 0xbe, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_OP_PIX_CLK_DIV, 10, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_OP_SYS_CLK_DIV, 1, IMX300_REG_VALUE_08BIT }, > + { 0x030d, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x030e, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x030f, 0x4b, IMX300_REG_VALUE_08BIT }, > +}; > + > +static const struct imx300_reg binning_mode_off[] = { > + { IMX300_REG_BINNING_EN, 0, IMX300_REG_VALUE_08BIT }, > + { 0x0901, 0x11, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_BINNING_TYPE, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a00, 0x19, IMX300_REG_VALUE_08BIT }, > + { 0x3a01, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a02, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a03, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x3a04, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a05, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a06, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a07, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a08, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6dc2, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x6dc3, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x6dc4, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x6dc7, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x6dca, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6dcb, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6dcc, 0x01, IMX300_REG_VALUE_08BIT }, > +}; > + > +static const struct imx300_reg binning_mode_2x2[] = { > + { IMX300_REG_BINNING_EN, 1, IMX300_REG_VALUE_08BIT }, > + { 0x0901, 0x22, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_BINNING_TYPE, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x3a00, 0x19, IMX300_REG_VALUE_08BIT }, > + { 0x3a01, 0x0a, IMX300_REG_VALUE_08BIT }, > + { 0x3a02, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x3a03, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x3a04, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a05, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a06, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3a07, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3a08, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6dc2, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x6dc3, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x6dc4, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x6dc7, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x6dca, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6dcb, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x6dcc, 0x01, IMX300_REG_VALUE_08BIT }, > +}; > + > + > +/* Output resolution configuration */ > +static const struct imx300_reg mode_5520x4160[] = { > + { IMX300_REG_EXPOSURE, 4230, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_ANALOG_GAIN, 0, IMX300_REG_VALUE_16BIT }, > + > + { IMX300_REG_VTS, 4250, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_LINE_LEN_PCK, 8008, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_START, 232, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_START, 0, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_END, 5751, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_END, 4159, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_OUTPUT_SIZE, 5520, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_OUTPUT_SIZE, 4160, IMX300_REG_VALUE_16BIT }, > + > + { 0x0408, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0409, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040b, 0x00, IMX300_REG_VALUE_08BIT }, > + > + { 0x040c, 5520, IMX300_REG_VALUE_16BIT }, > + { 0x040e, 4160, IMX300_REG_VALUE_16BIT }, > + > + { 0x3150, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3151, 0x49, IMX300_REG_VALUE_08BIT }, > + { 0x3152, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3153, 0x20, IMX300_REG_VALUE_08BIT }, > + { 0x3154, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x3155, 0x9f, IMX300_REG_VALUE_08BIT }, > + { 0x3156, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x3157, 0xaa, IMX300_REG_VALUE_08BIT }, > + > + { 0x31e8, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31e9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31ec, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31f0, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x31f1, 0xbc, IMX300_REG_VALUE_08BIT }, > + { 0x3221, 0x01, IMX300_REG_VALUE_08BIT }, > + > + { 0x7ea0, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea3, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x7ea5, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x7ea6, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x7ea8, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea9, 0x78, IMX300_REG_VALUE_08BIT }, > + { 0x7eac, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb3, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb4, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb5, 0x00, IMX300_REG_VALUE_08BIT }, > +}; > + > + > +static const struct imx300_reg mode_5984x3392[] = { > + { IMX300_REG_EXPOSURE, 3556, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_ANALOG_GAIN, 0, IMX300_REG_VALUE_16BIT }, > + > + { IMX300_REG_VTS, 3576, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_LINE_LEN_PCK, 8224, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_START, 0, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_START, 384, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_END, 5751, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_END, 3775, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_OUTPUT_SIZE, 5984, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_OUTPUT_SIZE, 3392, IMX300_REG_VALUE_16BIT }, > + > + { 0x0408, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0409, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040b, 0x00, IMX300_REG_VALUE_08BIT }, > + > + { 0x040c, 5984, IMX300_REG_VALUE_16BIT }, > + { 0x040e, 3392, IMX300_REG_VALUE_16BIT }, > + > + { 0x3150, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3151, 0x50, IMX300_REG_VALUE_08BIT }, > + { 0x3152, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3153, 0x1a, IMX300_REG_VALUE_08BIT }, > + { 0x3154, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x3155, 0xd8, IMX300_REG_VALUE_08BIT }, > + { 0x3156, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x3157, 0x2c, IMX300_REG_VALUE_08BIT }, > + > + { 0x31e8, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31e9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31ec, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31f0, 0x03, IMX300_REG_VALUE_08BIT }, > + { 0x31f1, 0x84, IMX300_REG_VALUE_08BIT }, > + { 0x3221, 0x00, IMX300_REG_VALUE_08BIT }, > + > + { 0x7ea0, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea3, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x7ea5, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x7ea6, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x7ea8, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea9, 0x78, IMX300_REG_VALUE_08BIT }, > + { 0x7eac, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb3, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb4, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb5, 0x00, IMX300_REG_VALUE_08BIT }, > +}; > + > +static const struct imx300_reg mode_2992x1696[] = { > + { IMX300_REG_EXPOSURE, 3566, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_ANALOG_GAIN, 0, IMX300_REG_VALUE_16BIT }, > + > + { IMX300_REG_VTS, 1772, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_LINE_LEN_PCK, 7064, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_START, 0, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_START, 384, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_END, 5983, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_END, 3775, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_OUTPUT_SIZE, 2992, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_OUTPUT_SIZE, 1696, IMX300_REG_VALUE_16BIT }, > + > + { 0x0408, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0409, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040b, 0x00, IMX300_REG_VALUE_08BIT }, > + > + { 0x040c, 2992, IMX300_REG_VALUE_16BIT }, > + { 0x040e, 1696, IMX300_REG_VALUE_16BIT }, > + > + { 0x3150, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3151, 0x28, IMX300_REG_VALUE_08BIT }, > + { 0x3152, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3153, 0x0d, IMX300_REG_VALUE_08BIT }, > + { 0x3154, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3155, 0x6c, IMX300_REG_VALUE_08BIT }, > + { 0x3156, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x3157, 0x16, IMX300_REG_VALUE_08BIT }, > + > + { 0x31e8, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31e9, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31ec, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31f0, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31f1, 0x14, IMX300_REG_VALUE_08BIT }, > + { 0x3221, 0x00, IMX300_REG_VALUE_08BIT }, > + > + { 0x7ea0, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x7ea3, 0x05, IMX300_REG_VALUE_08BIT }, > + { 0x7ea5, 0x1e, IMX300_REG_VALUE_08BIT }, > + { 0x7ea6, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x7ea8, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eac, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x7eb3, 0x04, IMX300_REG_VALUE_08BIT }, > + { 0x7eb4, 0x0c, IMX300_REG_VALUE_08BIT }, > + { 0x7eb5, 0x10, IMX300_REG_VALUE_08BIT }, > +}; > + > +static const struct imx300_reg mode_1424x800[] = { > + { IMX300_REG_EXPOSURE, 3692, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_ANALOG_GAIN, 0, IMX300_REG_VALUE_16BIT }, > + > + { IMX300_REG_VTS, 3712, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_LINE_LEN_PCK, 7064, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_START, 1568, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_START, 1280, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_ADDR_END, 4415, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_ADDR_END, 2879, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_X_OUTPUT_SIZE, 1424, IMX300_REG_VALUE_16BIT }, > + { IMX300_REG_Y_OUTPUT_SIZE, 800, IMX300_REG_VALUE_16BIT }, > + > + { 0x0408, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x0409, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040a, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x040b, 0x00, IMX300_REG_VALUE_08BIT }, > + > + { 0x040c, 1424, IMX300_REG_VALUE_16BIT }, > + { 0x040e, 800, IMX300_REG_VALUE_16BIT }, > + > + { 0x3150, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3151, 0x13, IMX300_REG_VALUE_08BIT }, > + { 0x3152, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3153, 0x06, IMX300_REG_VALUE_08BIT }, > + { 0x3154, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3155, 0xad, IMX300_REG_VALUE_08BIT }, > + { 0x3156, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x3157, 0x83, IMX300_REG_VALUE_08BIT }, > + > + { 0x31e8, 0x01, IMX300_REG_VALUE_08BIT }, > + { 0x31e9, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31ec, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x31f0, 0x02, IMX300_REG_VALUE_08BIT }, > + { 0x31f1, 0xbc, IMX300_REG_VALUE_08BIT }, > + { 0x3221, 0x01, IMX300_REG_VALUE_08BIT }, > + > + { 0x7ea0, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea3, 0x08, IMX300_REG_VALUE_08BIT }, > + { 0x7ea5, 0x0f, IMX300_REG_VALUE_08BIT }, > + { 0x7ea6, 0x60, IMX300_REG_VALUE_08BIT }, > + { 0x7ea8, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7ea9, 0x78, IMX300_REG_VALUE_08BIT }, > + { 0x7eac, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb3, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb4, 0x00, IMX300_REG_VALUE_08BIT }, > + { 0x7eb5, 0x00, IMX300_REG_VALUE_08BIT }, > +}; > + > +/* Output bits per sample: This sensor support RAW8 and RAW10 */ > +static const struct imx300_reg raw8_framefmt_regs[] = { > + { IMX300_REG_CSI_DATA_FORMAT_HI, 8, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_CSI_DATA_FORMAT_LO, 8, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_OP_PIX_CLK_DIV, 8, IMX300_REG_VALUE_08BIT }, > +}; > + > +static const struct imx300_reg raw10_framefmt_regs[] = { > + { IMX300_REG_CSI_DATA_FORMAT_HI, 10, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_CSI_DATA_FORMAT_LO, 10, IMX300_REG_VALUE_08BIT }, > + { IMX300_REG_OP_PIX_CLK_DIV, 10, IMX300_REG_VALUE_08BIT }, > +}; > + > +static const char * const imx300_test_pattern_menu[] = { > + "Disabled", > + "Solid Color", > + "Color Bars Solid", > + "Color Bars Fade To Gray", > + "Pseudorandom Sequence (PN9)" > +}; > + > +static const int imx300_test_pattern_val[] = { > + IMX300_TEST_PATTERN_DISABLE, > + IMX300_TEST_PATTERN_SOLID_COLOR, > + IMX300_TEST_PATTERN_COLOR_BARS, > + IMX300_TEST_PATTERN_GREY_COLOR, > + IMX300_TEST_PATTERN_PN9, > +}; > + > +/* regulator supplies */ > +static const char * const imx300_supply_name[] = { > + /* Supplies can be enabled in any order */ > + "vana", /* Analog (2.2V) supply */ > + "vdig", /* Digital Core (1.15-1.20V) supply */ > + "vddl", /* IF (1.8V) supply */ > +}; > + > +#define IMX300_NUM_SUPPLIES ARRAY_SIZE(imx300_supply_name) > + > +/* > + * The supported formats. > + * This table MUST contain 4 entries per format, to cover the various flip > + * combinations in the order > + * - no flip > + * - h flip > + * - v flip > + * - h&v flips > + */ > +static const u32 codes[] = { > + MEDIA_BUS_FMT_SRGGB10_1X10, > + MEDIA_BUS_FMT_SGRBG10_1X10, > + MEDIA_BUS_FMT_SGBRG10_1X10, > + MEDIA_BUS_FMT_SBGGR10_1X10, > + > + MEDIA_BUS_FMT_SRGGB8_1X8, > + MEDIA_BUS_FMT_SGRBG8_1X8, > + MEDIA_BUS_FMT_SGBRG8_1X8, > + MEDIA_BUS_FMT_SBGGR8_1X8, > +}; > + > +/* Mode configs */ > +static const struct imx300_mode supported_modes[] = { > + { > + /* 23MP 23fps mode */ > + .width = 5520, > + .height = 4160, > + .max_fps = 23, > + .crop = { > + .left = 0, > + .top = 0, > + .width = 5520, > + .height = 4160 > + }, > + .vts_def = 4250, > + .high_bw = true, > + .binned = false, > + .reg_list = { > + .num_of_regs = ARRAY_SIZE(mode_5520x4160), > + .regs = mode_5520x4160, > + }, > + }, > + { > + /* 20.3MP 26fps mode */ > + .width = 5984, > + .height = 3392, > + .max_fps = 26, > + .crop = { > + .left = 0, > + .top = 384, > + .width = 5984, > + .height = 3392 > + }, > + .vts_def = 3576, > + .high_bw = true, > + .binned = false, > + .reg_list = { > + .num_of_regs = ARRAY_SIZE(mode_5984x3392), > + .regs = mode_5984x3392, > + }, > + }, > + { > + /* 3.8MP 60fps 2x2 binning */ > + .width = 2992, > + .height = 1696, > + .max_fps = 60, > + .crop = { > + .left = 0, > + .top = 384, > + .width = 2292, > + .height = 1696 > + }, > + .vts_def = 1792, > + .high_bw = true, > + .binned = true, > + .reg_list = { > + .num_of_regs = > + ARRAY_SIZE(mode_2992x1696), > + .regs = mode_2992x1696, > + }, > + }, > + { > + /* 1.2MP binned 120fps mode */ > + .width = 1424, > + .height = 800, > + .max_fps = 120, > + .crop = { > + .left = 0, > + .top = 384, > + .width = 1424, > + .height = 800, > + }, > + .vts_def = 896, > + .high_bw = false, > + .binned = true, > + .reg_list = { > + .num_of_regs = > + ARRAY_SIZE(mode_1424x800), > + .regs = mode_1424x800, > + }, > + }, > +}; > + > +struct imx300 { > + struct v4l2_subdev sd; > + struct media_pad pad; > + > + struct v4l2_mbus_framefmt fmt; > + > + struct clk *xclk; /* system clock to IMX300 */ > + u32 xclk_freq; > + > + struct gpio_desc *reset_gpio; > + struct regulator_bulk_data supplies[IMX300_NUM_SUPPLIES]; > + > + struct v4l2_ctrl_handler ctrl_handler; > + /* V4L2 Controls */ > + struct v4l2_ctrl *pixel_rate; > + struct v4l2_ctrl *exposure; > + struct v4l2_ctrl *vflip; > + struct v4l2_ctrl *hflip; > + struct v4l2_ctrl *vblank; > + struct v4l2_ctrl *hblank; > + > + /* Frame rate */ > + struct v4l2_fract frame_rate; > + > + /* Current mode */ > + const struct imx300_mode *mode; > + u32 cur_bps; > + > + /* > + * Mutex for serialized access: > + * Protect sensor module set pad format and start/stop streaming safely. > + */ > + struct mutex mutex; > + > + /* Streaming on/off */ > + bool streaming; > +}; > + > +static inline struct imx300 *to_imx300(struct v4l2_subdev *_sd) > +{ > + return container_of(_sd, struct imx300, sd); > +} > + > +static s64 get_pixel_rate(struct imx300 *imx300) > +{ > + s64 prate; > + > + if (imx300->mode->high_bw) > + prate = IMX300_HIGH_BW_PIXEL_RATE; > + else > + prate = IMX300_LOW_BW_PIXEL_RATE; > + > + /* Satisfy the settle time for 8bits */ > + if (imx300->cur_bps == 8) { > + do_div(prate, 10); > + prate *= 12; > + } > + > + return prate; > +} > + > +/* Read registers up to 2 at a time */ > +static int imx300_read_reg(struct imx300 *imx300, u16 reg, u32 len, u32 *val) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->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 imx300_write_reg(struct imx300 *imx300, u16 reg, u32 len, u32 val) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->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 imx300_write_regs(struct imx300 *imx300, > + const struct imx300_reg *regs, u32 len) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + unsigned int i; > + int ret; > + > + for (i = 0; i < len; i++) { > + ret = imx300_write_reg(imx300, regs[i].address, > + regs[i].reg_len, > + regs[i].val); > + if (ret) { > + dev_err_ratelimited(&client->dev, > + "Cannot write reg 0x%4.4x. (%d)\n", > + regs[i].address, ret); > + > + return ret; > + } > + } > + > + return 0; > +} > + > +/* Get bayer order based on flip setting. */ > +static u32 imx300_get_format_code(struct imx300 *imx300, u32 code) > +{ > + unsigned int i; > + > + lockdep_assert_held(&imx300->mutex); > + > + for (i = 0; i < ARRAY_SIZE(codes); i++) > + if (codes[i] == code) > + break; > + > + if (i >= ARRAY_SIZE(codes)) > + i = 0; > + > + i = (i & ~3) | (imx300->vflip->val ? 2 : 0) | > + (imx300->hflip->val ? 1 : 0); > + > + return codes[i]; > +} > + > +static void imx300_set_default_format(struct imx300 *imx300) > +{ > + struct v4l2_mbus_framefmt *fmt; > + > + fmt = &imx300->fmt; > + fmt->code = MEDIA_BUS_FMT_SRGGB10_1X10; > + fmt->colorspace = V4L2_COLORSPACE_SRGB; > + fmt->ycbcr_enc = V4L2_MAP_YCBCR_ENC_DEFAULT(fmt->colorspace); > + fmt->quantization = V4L2_MAP_QUANTIZATION_DEFAULT(true, > + fmt->colorspace, > + fmt->ycbcr_enc); > + fmt->xfer_func = V4L2_MAP_XFER_FUNC_DEFAULT(fmt->colorspace); > + fmt->width = supported_modes[0].width; > + fmt->height = supported_modes[0].height; > + fmt->field = V4L2_FIELD_NONE; > +} > + > +static int imx300_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + struct v4l2_mbus_framefmt *try_fmt = > + v4l2_subdev_get_try_format(sd, fh->pad, 0); > + struct v4l2_rect *try_crop; > + > + mutex_lock(&imx300->mutex); > + > + /* Initialize try_fmt */ > + try_fmt->width = supported_modes[0].width; > + try_fmt->height = supported_modes[0].height; > + try_fmt->code = imx300_get_format_code(imx300, > + MEDIA_BUS_FMT_SRGGB10_1X10); > + try_fmt->field = V4L2_FIELD_NONE; > + > + /* Initialize try_crop rectangle. */ > + try_crop = v4l2_subdev_get_try_crop(sd, fh->pad, 0); > + try_crop->top = IMX300_PIXEL_ARRAY_TOP; > + try_crop->left = IMX300_PIXEL_ARRAY_LEFT; > + try_crop->width = IMX300_PIXEL_ARRAY_WIDTH; > + try_crop->height = IMX300_PIXEL_ARRAY_HEIGHT; > + > + mutex_unlock(&imx300->mutex); > + > + return 0; > +} > + > +static int imx300_update_digital_gain(struct imx300 *imx300, u32 gain) > +{ > + int ret; > + > + ret = imx300_write_reg(imx300, IMX300_REG_GR_DIGITAL_GAIN, > + IMX300_REG_VALUE_16BIT, > + gain); > + if (ret) > + return ret; > + > + ret = imx300_write_reg(imx300, IMX300_REG_GB_DIGITAL_GAIN, > + IMX300_REG_VALUE_16BIT, > + gain); > + if (ret) > + return ret; > + > + ret = imx300_write_reg(imx300, IMX300_REG_R_DIGITAL_GAIN, > + IMX300_REG_VALUE_16BIT, > + gain); > + if (ret) > + return ret; > + > + ret = imx300_write_reg(imx300, IMX300_REG_B_DIGITAL_GAIN, return ... > + IMX300_REG_VALUE_16BIT, > + gain); > + if (ret) > + return ret; > + > + return 0; > +} > + > +static int imx300_set_ctrl(struct v4l2_ctrl *ctrl) > +{ > + struct imx300 *imx300 = > + container_of(ctrl->handler, struct imx300, ctrl_handler); > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + int ret; > + > + if (ctrl->id == V4L2_CID_VBLANK) { > + int exposure_max, exposure_def; > + > + /* Update max exposure while meeting expected vblanking */ > + exposure_max = imx300->mode->height + ctrl->val - 10; > + exposure_def = (exposure_max < IMX300_EXPOSURE_DEFAULT) ? > + exposure_max : IMX300_EXPOSURE_DEFAULT; > + __v4l2_ctrl_modify_range(imx300->exposure, > + imx300->exposure->minimum, > + exposure_max, imx300->exposure->step, > + exposure_def); > + } > + > + /* > + * 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 = imx300_write_reg(imx300, IMX300_REG_ANALOG_GAIN, > + IMX300_REG_VALUE_16BIT, > + ctrl->val); > + break; > + case V4L2_CID_EXPOSURE: > + ret = imx300_write_reg(imx300, IMX300_REG_EXPOSURE, > + IMX300_REG_VALUE_16BIT, > + ctrl->val); > + break; > + case V4L2_CID_DIGITAL_GAIN: > + ret = imx300_update_digital_gain(imx300, ctrl->val); > + break; > + case V4L2_CID_TEST_PATTERN: > + ret = imx300_write_reg(imx300, IMX300_REG_TEST_PATTERN, > + IMX300_REG_VALUE_16BIT, > + imx300_test_pattern_val[ctrl->val]); > + break; > + case V4L2_CID_HFLIP: > + case V4L2_CID_VFLIP: > + ret = imx300_write_reg(imx300, IMX300_REG_ORIENTATION, > + IMX300_REG_VALUE_08BIT, > + imx300->hflip->val | > + imx300->vflip->val << 1); > + break; > + case V4L2_CID_VBLANK: > + ret = imx300_write_reg(imx300, IMX300_REG_VTS, > + IMX300_REG_VALUE_16BIT, > + imx300->mode->height + ctrl->val); > + break; > + case V4L2_CID_TEST_PATTERN_RED: > + ret = imx300_write_reg(imx300, IMX300_REG_TESTP_RED, > + IMX300_REG_VALUE_16BIT, ctrl->val); > + break; > + case V4L2_CID_TEST_PATTERN_GREENR: > + ret = imx300_write_reg(imx300, IMX300_REG_TESTP_GREENR, > + IMX300_REG_VALUE_16BIT, ctrl->val); > + break; > + case V4L2_CID_TEST_PATTERN_BLUE: > + ret = imx300_write_reg(imx300, IMX300_REG_TESTP_BLUE, > + IMX300_REG_VALUE_16BIT, ctrl->val); > + break; > + case V4L2_CID_TEST_PATTERN_GREENB: > + ret = imx300_write_reg(imx300, IMX300_REG_TESTP_GREENB, > + IMX300_REG_VALUE_16BIT, ctrl->val); > + break; > + default: > + dev_info(&client->dev, > + "ctrl(id:0x%x,val:0x%x) is not handled\n", > + ctrl->id, ctrl->val); > + ret = -EINVAL; > + break; > + } > + > + pm_runtime_put(&client->dev); > + > + return ret; > +} > + > +static const struct v4l2_ctrl_ops imx300_ctrl_ops = { > + .s_ctrl = imx300_set_ctrl, > +}; > + > +static int imx300_enum_mbus_code(struct v4l2_subdev *sd, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_mbus_code_enum *code) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + > + if (code->index >= (ARRAY_SIZE(codes) / 4)) > + return -EINVAL; > + > + code->code = imx300_get_format_code(imx300, codes[code->index * 4]); The should depend on the flip controls. > + > + return 0; > +} > + > +static int imx300_enum_frame_size(struct v4l2_subdev *sd, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_frame_size_enum *fse) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + > + if (fse->index >= ARRAY_SIZE(supported_modes)) > + return -EINVAL; > + > + if (fse->code != imx300_get_format_code(imx300, fse->code)) > + 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 int imx300_enum_frame_interval(struct v4l2_subdev *sd, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_frame_interval_enum *fie) > +{ > + unsigned int i; > + > + if (fie->pad || fie->index >= ARRAY_SIZE(supported_modes)) > + return -EINVAL; > + > + for (i = 0; i < ARRAY_SIZE(supported_modes); i++) > + if (fie->width == supported_modes[i].width && > + fie->height == supported_modes[i].height) > + break; > + > + if (i == ARRAY_SIZE(supported_modes)) > + return -EINVAL; > + > + fie->interval.numerator = 1; > + fie->interval.denominator = supported_modes[i].max_fps; > + > + return 0; > +} > + > +static int imx300_g_frame_interval(struct v4l2_subdev *sd, > + struct v4l2_subdev_frame_interval *ival) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + > + ival->interval.numerator = imx300->frame_rate.denominator; > + ival->interval.denominator = imx300->frame_rate.numerator; > + > + return 0; > +} > + > +static int imx300_s_frame_interval(struct v4l2_subdev *sd, > + struct v4l2_subdev_frame_interval *ival) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + const struct imx300_mode *cur_mode = imx300->mode; > + struct v4l2_fract *tpf = &ival->interval; > + int exposure_max, exposure_def; > + u32 new_vts; > + u32 vblank = 0; > + > + if (tpf->numerator == 0 || tpf->denominator == 0 || > + (tpf->denominator > tpf->numerator * cur_mode->max_fps)) { > + /* reset to max frame rate */ > + tpf->numerator = 1; > + tpf->denominator = cur_mode->max_fps; > + new_vts = cur_mode->vts_def; > + } else { > + /* Approximation of new VTS: recalculate default vblank */ > + vblank = cur_mode->vts_def - cur_mode->height; > + > + /* Avoid floating point */ > + new_vts = vblank * 1000; > + new_vts = new_vts / cur_mode->max_fps; > + new_vts = (new_vts * tpf->denominator) / 1000; > + new_vts += vblank + cur_mode->height; > + } > + > + imx300->frame_rate.numerator = tpf->numerator; > + imx300->frame_rate.denominator = tpf->denominator; > + > + /* > + * Note: VTS cannot be less than cur_mode->height, but that's useless > + * to check at this point, since we are surely complying here. > + * > + * Now that we've got a new VTS, let's update the exposure control > + * min/max in order to avoid impossible and/or useless combinations. > + */ > + exposure_max = new_vts - 4; > + exposure_def = (exposure_max < IMX300_EXPOSURE_DEFAULT) ? > + exposure_max : IMX300_EXPOSURE_DEFAULT; > + __v4l2_ctrl_modify_range(imx300->exposure, > + imx300->exposure->minimum, > + exposure_max, imx300->exposure->step, > + exposure_def); > + > + return imx300_write_reg(imx300, IMX300_REG_VTS, > + IMX300_REG_VALUE_16BIT, > + new_vts); > +} > + > +static void imx300_reset_colorspace(struct v4l2_mbus_framefmt *fmt) > +{ > + fmt->colorspace = V4L2_COLORSPACE_SRGB; > + fmt->ycbcr_enc = V4L2_MAP_YCBCR_ENC_DEFAULT(fmt->colorspace); > + fmt->quantization = V4L2_MAP_QUANTIZATION_DEFAULT(true, > + fmt->colorspace, > + fmt->ycbcr_enc); > + fmt->xfer_func = V4L2_MAP_XFER_FUNC_DEFAULT(fmt->colorspace); > +} > + > +static void imx300_update_pad_format(struct imx300 *imx300, > + const struct imx300_mode *mode, > + struct v4l2_subdev_format *fmt) > +{ > + fmt->format.width = mode->width; > + fmt->format.height = mode->height; > + fmt->format.field = V4L2_FIELD_NONE; > + imx300_reset_colorspace(&fmt->format); > +} > + > +static int __imx300_get_pad_format(struct imx300 *imx300, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_format *fmt) > +{ > + if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) { > + struct v4l2_mbus_framefmt *try_fmt = > + v4l2_subdev_get_try_format(&imx300->sd, cfg, fmt->pad); > + /* update the code which could change due to vflip or hflip: */ > + try_fmt->code = imx300_get_format_code(imx300, try_fmt->code); > + fmt->format = *try_fmt; > + } else { > + imx300_update_pad_format(imx300, imx300->mode, fmt); > + fmt->format.code = imx300_get_format_code(imx300, > + imx300->fmt.code); > + } > + > + return 0; > +} > + > +static int imx300_get_pad_format(struct v4l2_subdev *sd, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_format *fmt) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + int ret; > + > + mutex_lock(&imx300->mutex); > + ret = __imx300_get_pad_format(imx300, cfg, fmt); > + mutex_unlock(&imx300->mutex); > + > + return ret; > +} > + > +static int imx300_set_pad_format(struct v4l2_subdev *sd, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_format *fmt) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + const struct imx300_mode *mode; > + struct v4l2_mbus_framefmt *framefmt; > + int exposure_max, exposure_def, hblank; > + unsigned int i; > + > + mutex_lock(&imx300->mutex); > + > + for (i = 0; i < ARRAY_SIZE(codes); i++) > + if (codes[i] == fmt->format.code) > + break; > + if (i >= ARRAY_SIZE(codes)) > + i = 0; > + > + /* Bayer order varies with flips */ > + fmt->format.code = imx300_get_format_code(imx300, codes[i]); > + > + mode = v4l2_find_nearest_size(supported_modes, > + ARRAY_SIZE(supported_modes), > + width, height, > + fmt->format.width, fmt->format.height); > + imx300_update_pad_format(imx300, mode, fmt); > + if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) { > + framefmt = v4l2_subdev_get_try_format(sd, cfg, fmt->pad); > + *framefmt = fmt->format; > + } else if (imx300->mode != mode || > + imx300->fmt.code != fmt->format.code) { > + imx300->fmt = fmt->format; > + imx300->mode = mode; > + /* Update limits and set FPS to default */ > + __v4l2_ctrl_modify_range(imx300->vblank, IMX300_VTS_MIN, > + IMX300_VTS_MAX - mode->height, 1, > + mode->vts_def - mode->height); > + __v4l2_ctrl_s_ctrl(imx300->vblank, > + mode->vts_def - mode->height); > + /* Update max exposure while meeting expected vblanking */ > + exposure_max = mode->vts_def - 4; > + exposure_def = (exposure_max < IMX300_EXPOSURE_DEFAULT) ? > + exposure_max : IMX300_EXPOSURE_DEFAULT; > + __v4l2_ctrl_modify_range(imx300->exposure, > + imx300->exposure->minimum, > + exposure_max, imx300->exposure->step, > + exposure_def); > + /* > + * Currently PPL is fixed to IMX300_PPL_DEFAULT, so hblank > + * depends on mode->width only, and is not changeble in any > + * way other than changing the mode. > + */ > + hblank = IMX300_PPL_DEFAULT - mode->width; > + __v4l2_ctrl_modify_range(imx300->hblank, hblank, hblank, 1, > + hblank); > + } > + > + mutex_unlock(&imx300->mutex); > + > + return 0; > +} > + > +static int imx300_set_framefmt(struct imx300 *imx300) > +{ > + int ret; > + > + switch (imx300->fmt.code) { > + case MEDIA_BUS_FMT_SRGGB8_1X8: > + fallthrough; You can omit fallthrough here and below. > + case MEDIA_BUS_FMT_SGRBG8_1X8: > + fallthrough; > + case MEDIA_BUS_FMT_SGBRG8_1X8: > + fallthrough; > + case MEDIA_BUS_FMT_SBGGR8_1X8: > + ret = imx300_write_regs(imx300, raw8_framefmt_regs, > + ARRAY_SIZE(raw8_framefmt_regs)); > + if (ret) > + return ret; > + imx300->cur_bps = 8; > + break; > + case MEDIA_BUS_FMT_SRGGB10_1X10: > + fallthrough; > + case MEDIA_BUS_FMT_SGRBG10_1X10: > + fallthrough; > + case MEDIA_BUS_FMT_SGBRG10_1X10: > + fallthrough; > + case MEDIA_BUS_FMT_SBGGR10_1X10: > + ret = imx300_write_regs(imx300, raw10_framefmt_regs, > + ARRAY_SIZE(raw10_framefmt_regs)); > + if (ret) > + return ret; > + imx300->cur_bps = 10; > + break; > + default: > + return -EINVAL; > + } > + > + /* Update the pixel rate to eventually save some power */ How is this related to any possible power savings? > + __v4l2_ctrl_s_ctrl_int64(imx300->pixel_rate, get_pixel_rate(imx300)); > + > + return ret; > +} > + > +static const struct v4l2_rect * > +__imx300_get_pad_crop(struct imx300 *imx300, struct v4l2_subdev_pad_config *cfg, > + unsigned int pad, enum v4l2_subdev_format_whence which) > +{ > + switch (which) { > + case V4L2_SUBDEV_FORMAT_TRY: > + return v4l2_subdev_get_try_crop(&imx300->sd, cfg, pad); > + case V4L2_SUBDEV_FORMAT_ACTIVE: > + return &imx300->mode->crop; > + } > + > + return NULL; > +} > + > +static int imx300_get_selection(struct v4l2_subdev *sd, > + struct v4l2_subdev_pad_config *cfg, > + struct v4l2_subdev_selection *sel) > +{ > + switch (sel->target) { > + case V4L2_SEL_TGT_CROP: { > + struct imx300 *imx300 = to_imx300(sd); > + > + mutex_lock(&imx300->mutex); > + sel->r = *__imx300_get_pad_crop(imx300, cfg, sel->pad, > + sel->which); > + mutex_unlock(&imx300->mutex); > + > + return 0; > + } > + > + case V4L2_SEL_TGT_NATIVE_SIZE: > + sel->r.top = 0; > + sel->r.left = 0; > + sel->r.width = IMX300_NATIVE_WIDTH; > + sel->r.height = IMX300_NATIVE_HEIGHT; > + > + return 0; > + > + case V4L2_SEL_TGT_CROP_DEFAULT: > + sel->r.top = IMX300_PIXEL_ARRAY_TOP; > + sel->r.left = IMX300_PIXEL_ARRAY_LEFT; > + sel->r.width = IMX300_PIXEL_ARRAY_WIDTH; > + sel->r.height = IMX300_PIXEL_ARRAY_HEIGHT; > + > + return 0; > + } > + > + return -EINVAL; > +} > + > +static int imx300_start_streaming(struct imx300 *imx300) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + const struct imx300_reg_list *reg_list; > + int ret; > + > + ret = imx300_write_regs(imx300, init_sequence, > + ARRAY_SIZE(init_sequence)); > + if (ret) { > + dev_err(&client->dev, > + "Cannot write init sequence\n"); > + return ret; > + } > + > + if (imx300->mode->high_bw) > + ret = imx300_write_regs(imx300, mipi_data_rate_1560mbps, > + ARRAY_SIZE(mipi_data_rate_1560mbps)); > + else > + ret = imx300_write_regs(imx300, mipi_data_rate_960mbps, > + ARRAY_SIZE(mipi_data_rate_960mbps)); > + if (ret) { > + dev_err(&client->dev, "Failed to set MIPI data rate\n"); > + return ret; > + } > + > + /* Common between all resolutions */ > + ret = imx300_write_regs(imx300, mode_common_regs, > + ARRAY_SIZE(mode_common_regs)); > + if (ret) { > + dev_err(&client->dev, > + "Cannot write init sequence\n"); > + return ret; > + } > + > + /* Apply default values of the selected mode */ > + reg_list = &imx300->mode->reg_list; > + ret = imx300_write_regs(imx300, reg_list->regs, reg_list->num_of_regs); > + if (ret) { > + dev_err(&client->dev, "Failed to set resolution\n"); > + return ret; > + } > + > + if (imx300->mode->binned) > + ret = imx300_write_regs(imx300, binning_mode_2x2, > + ARRAY_SIZE(binning_mode_2x2)); > + else > + ret = imx300_write_regs(imx300, binning_mode_off, > + ARRAY_SIZE(binning_mode_off)); > + if (ret) { > + dev_err(&client->dev, "Failed to set binning mode\n"); > + return ret; > + } > + > + ret = imx300_set_framefmt(imx300); > + if (ret) { > + dev_err(&client->dev, "%s failed to set frame format: %d\n", > + __func__, ret); > + return ret; > + } > + > + /* Apply customized values from user */ > + ret = __v4l2_ctrl_handler_setup(imx300->sd.ctrl_handler); > + if (ret) > + return ret; > + > + /* set stream on register */ > + return imx300_write_reg(imx300, IMX300_REG_MODE_SELECT, > + IMX300_REG_VALUE_08BIT, IMX300_MODE_STREAMING); > +} > + > +static void imx300_stop_streaming(struct imx300 *imx300) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + int ret; > + > + /* set stream off register */ > + ret = imx300_write_reg(imx300, IMX300_REG_MODE_SELECT, > + IMX300_REG_VALUE_08BIT, IMX300_MODE_STANDBY); > + if (ret) > + dev_err(&client->dev, "%s failed to set stream\n", __func__); > +} > + > +static int imx300_set_stream(struct v4l2_subdev *sd, int enable) > +{ > + struct imx300 *imx300 = to_imx300(sd); > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + int ret = 0; > + > + mutex_lock(&imx300->mutex); > + if (imx300->streaming == enable) { > + mutex_unlock(&imx300->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 = imx300_start_streaming(imx300); > + if (ret) > + goto err_rpm_put; > + } else { > + imx300_stop_streaming(imx300); > + pm_runtime_put(&client->dev); > + } > + > + imx300->streaming = enable; > + > + /* vflip and hflip cannot change during streaming */ > + __v4l2_ctrl_grab(imx300->vflip, enable); > + __v4l2_ctrl_grab(imx300->hflip, enable); > + > + mutex_unlock(&imx300->mutex); > + > + return ret; > + > +err_rpm_put: > + pm_runtime_put(&client->dev); > +err_unlock: > + mutex_unlock(&imx300->mutex); > + > + return ret; > +} > + > +/* Power/clock management functions */ > +static int imx300_power_on(struct device *dev) > +{ > + struct i2c_client *client = to_i2c_client(dev); > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct imx300 *imx300 = to_imx300(sd); > + int ret; > + > + ret = regulator_bulk_enable(IMX300_NUM_SUPPLIES, > + imx300->supplies); > + if (ret) { > + dev_err(&client->dev, "%s: failed to enable regulators\n", > + __func__); > + return ret; > + } > + > + ret = clk_prepare_enable(imx300->xclk); > + if (ret) { > + dev_err(&client->dev, "%s: failed to enable clock\n", > + __func__); > + goto reg_off; > + } > + > + /* Wait for the internal PLLs stabilization time */ > + usleep_range(IMX300_XCLK_STABLE_DELAY_US, > + IMX300_XCLK_STABLE_DELAY_US + IMX300_XCLK_DELAY_RANGE_US); > + > + /* PLLs are stable now: get out of reset! */ > + gpiod_set_value_cansleep(imx300->reset_gpio, 1); > + usleep_range(IMX300_XCLR_MIN_DELAY_US, > + IMX300_XCLR_MIN_DELAY_US + IMX300_XCLR_DELAY_RANGE_US); > + > + return 0; > + > +reg_off: > + regulator_bulk_disable(IMX300_NUM_SUPPLIES, imx300->supplies); > + > + return ret; > +} > + > +static int imx300_power_off(struct device *dev) > +{ > + struct i2c_client *client = to_i2c_client(dev); > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct imx300 *imx300 = to_imx300(sd); > + > + gpiod_set_value_cansleep(imx300->reset_gpio, 0); > + regulator_bulk_disable(IMX300_NUM_SUPPLIES, imx300->supplies); > + clk_disable_unprepare(imx300->xclk); > + > + return 0; > +} > + > +static int __maybe_unused imx300_suspend(struct device *dev) > +{ > + struct i2c_client *client = to_i2c_client(dev); > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct imx300 *imx300 = to_imx300(sd); > + > + if (imx300->streaming) > + imx300_stop_streaming(imx300); > + > + return 0; > +} > + > +static int __maybe_unused imx300_resume(struct device *dev) > +{ > + struct i2c_client *client = to_i2c_client(dev); > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct imx300 *imx300 = to_imx300(sd); > + int ret; > + > + if (imx300->streaming) { > + ret = imx300_start_streaming(imx300); > + if (ret) > + goto error; > + } > + > + return 0; > + > +error: > + imx300_stop_streaming(imx300); > + imx300->streaming = false; > + > + return ret; > +} > + > +static int imx300_get_regulators(struct imx300 *imx300) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + unsigned int i; > + > + for (i = 0; i < IMX300_NUM_SUPPLIES; i++) > + imx300->supplies[i].supply = imx300_supply_name[i]; > + > + return devm_regulator_bulk_get(&client->dev, > + IMX300_NUM_SUPPLIES, > + imx300->supplies); > +} > + > +/* Verify chip ID */ > +static int imx300_identify_module(struct imx300 *imx300) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + int ret; > + u32 val; > + > + ret = imx300_read_reg(imx300, IMX300_REG_CHIP_ID, > + IMX300_REG_VALUE_16BIT, &val); > + if (ret) { > + dev_err(&client->dev, "failed to read chip id %x\n", > + IMX300_CHIP_ID); > + return ret; > + } > + > + if (val != IMX300_CHIP_ID) { > + dev_err(&client->dev, "chip id mismatch: %x!=%x\n", > + IMX300_CHIP_ID, val); > + return -EIO; > + } > + > + return 0; > +} > + > +static const struct v4l2_subdev_core_ops imx300_core_ops = { > + .subscribe_event = v4l2_ctrl_subdev_subscribe_event, > + .unsubscribe_event = v4l2_event_subdev_unsubscribe, > +}; > + > +static const struct v4l2_subdev_video_ops imx300_video_ops = { > + .s_stream = imx300_set_stream, > + .g_frame_interval = imx300_g_frame_interval, > + .s_frame_interval = imx300_s_frame_interval, > +}; > + > +static const struct v4l2_subdev_pad_ops imx300_pad_ops = { > + .enum_mbus_code = imx300_enum_mbus_code, > + .get_fmt = imx300_get_pad_format, > + .set_fmt = imx300_set_pad_format, > + .get_selection = imx300_get_selection, > + .enum_frame_size = imx300_enum_frame_size, > + .enum_frame_interval = imx300_enum_frame_interval, > +}; > + > +static const struct v4l2_subdev_ops imx300_subdev_ops = { > + .core = &imx300_core_ops, > + .video = &imx300_video_ops, > + .pad = &imx300_pad_ops, > +}; > + > +static const struct v4l2_subdev_internal_ops imx300_internal_ops = { > + .open = imx300_open, > +}; > + > +/* Initialize control handlers */ > +static int imx300_init_controls(struct imx300 *imx300) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&imx300->sd); > + struct v4l2_ctrl_handler *ctrl_hdlr; > + unsigned int height = imx300->mode->height; > + struct v4l2_fwnode_device_properties props; > + int exposure_max, exposure_def, hblank; > + int i, ret; > + > + ctrl_hdlr = &imx300->ctrl_handler; > + ret = v4l2_ctrl_handler_init(ctrl_hdlr, 11); > + if (ret) > + return ret; > + > + mutex_init(&imx300->mutex); > + ctrl_hdlr->lock = &imx300->mutex; > + > + /* By default, PIXEL_RATE is read only */ > + imx300->pixel_rate = v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_PIXEL_RATE, > + IMX300_LOW_BW_PIXEL_RATE, > + IMX300_HIGH_BW_PIXEL_RATE, 1, > + IMX300_HIGH_BW_PIXEL_RATE); > + > + /* Initial vblank/hblank/exposure parameters based on current mode */ > + imx300->vblank = v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_VBLANK, IMX300_VTS_MIN, > + IMX300_VTS_MAX - height, 1, > + imx300->mode->vts_def - height); > + hblank = IMX300_PPL_DEFAULT - imx300->mode->width; > + imx300->hblank = v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_HBLANK, hblank, hblank, > + 1, hblank); > + if (imx300->hblank) > + imx300->hblank->flags |= V4L2_CTRL_FLAG_READ_ONLY; > + exposure_max = imx300->mode->vts_def - 4; > + exposure_def = (exposure_max < IMX300_EXPOSURE_DEFAULT) ? > + exposure_max : IMX300_EXPOSURE_DEFAULT; > + imx300->exposure = v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_EXPOSURE, > + IMX300_EXPOSURE_MIN, exposure_max, > + IMX300_EXPOSURE_STEP, > + exposure_def); > + > + v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, V4L2_CID_ANALOGUE_GAIN, > + IMX300_ANA_GAIN_MIN, IMX300_ANA_GAIN_MAX, > + IMX300_ANA_GAIN_STEP, IMX300_ANA_GAIN_DEFAULT); > + > + v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, V4L2_CID_DIGITAL_GAIN, > + IMX300_DGTL_GAIN_MIN, IMX300_DGTL_GAIN_MAX, > + IMX300_DGTL_GAIN_STEP, IMX300_DGTL_GAIN_DEFAULT); > + > + imx300->hflip = v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_HFLIP, 0, 1, 1, 0); > + if (imx300->hflip) > + imx300->hflip->flags |= V4L2_CTRL_FLAG_MODIFY_LAYOUT; > + > + imx300->vflip = v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_VFLIP, 0, 1, 1, 0); > + if (imx300->vflip) > + imx300->vflip->flags |= V4L2_CTRL_FLAG_MODIFY_LAYOUT; > + > + v4l2_ctrl_new_std_menu_items(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_TEST_PATTERN, > + ARRAY_SIZE(imx300_test_pattern_menu) - 1, > + 0, 0, imx300_test_pattern_menu); > + for (i = 0; i < 4; i++) { > + /* > + * The assumption is that > + * V4L2_CID_TEST_PATTERN_GREENR == V4L2_CID_TEST_PATTERN_RED + 1 > + * V4L2_CID_TEST_PATTERN_BLUE == V4L2_CID_TEST_PATTERN_RED + 2 > + * V4L2_CID_TEST_PATTERN_GREENB == V4L2_CID_TEST_PATTERN_RED + 3 > + */ > + v4l2_ctrl_new_std(ctrl_hdlr, &imx300_ctrl_ops, > + V4L2_CID_TEST_PATTERN_RED + i, > + IMX300_TESTP_COLOUR_MIN, > + IMX300_TESTP_COLOUR_MAX, > + IMX300_TESTP_COLOUR_STEP, > + IMX300_TESTP_COLOUR_MAX); > + /* The "Solid color" pattern is white by default */ > + } > + > + if (ctrl_hdlr->error) { > + ret = ctrl_hdlr->error; > + dev_err(&client->dev, "%s control init failed (%d)\n", > + __func__, ret); > + goto error; > + } > + > + ret = v4l2_fwnode_device_parse(&client->dev, &props); > + if (ret) > + goto error; > + > + ret = v4l2_ctrl_new_fwnode_properties(ctrl_hdlr, &imx300_ctrl_ops, > + &props); > + if (ret) > + goto error; > + > + imx300->sd.ctrl_handler = ctrl_hdlr; > + > + return 0; > + > +error: > + v4l2_ctrl_handler_free(ctrl_hdlr); > + mutex_destroy(&imx300->mutex); > + > + return ret; > +} > + > +static void imx300_free_controls(struct imx300 *imx300) > +{ > + v4l2_ctrl_handler_free(imx300->sd.ctrl_handler); > + mutex_destroy(&imx300->mutex); > +} > + > +static int imx300_match_link_freq(u64 link_freq) > +{ > + if (link_freq == IMX300_HIGH_BW_LINK_FREQ || > + link_freq == IMX300_LOW_BW_LINK_FREQ) > + return 0; > + > + return -EINVAL; > +} > + > +static int imx300_check_hwcfg(struct device *dev, struct imx300 *imx300) > +{ > + struct fwnode_handle *endpoint; > + struct v4l2_fwnode_endpoint ep_cfg = { > + .bus_type = V4L2_MBUS_CSI2_DPHY > + }; > + int ret = -EINVAL; > + > + endpoint = fwnode_graph_get_next_endpoint(dev_fwnode(dev), NULL); > + if (!endpoint) { > + dev_err(dev, "Endpoint node not found\n"); > + return -EINVAL; > + } > + > + if (v4l2_fwnode_endpoint_alloc_parse(endpoint, &ep_cfg)) { > + dev_err(dev, "Cannot parse endpoint\n"); > + goto error_out; > + } > + > + /* Check the link frequency set in device tree */ > + if (ep_cfg.nr_of_link_frequencies != 2) { > + dev_err(dev, "This sensor uses two link frequencies.\n"); You could allow still using one frequency, assuming it's supported, but that's checked below. > + goto error_out; > + } > + > + /* Check the number of MIPI CSI2 data lanes */ > + if (ep_cfg.bus.mipi_csi2.num_data_lanes != 4) { > + dev_err(dev, "This sensor needs 4 MIPI Lanes!\n"); > + goto error_out; > + } > + > + if (imx300_match_link_freq(ep_cfg.link_frequencies[0]) || > + imx300_match_link_freq(ep_cfg.link_frequencies[1])) { > + dev_err(dev, "Unsupported link frequencies.\n"); > + goto error_out; > + } > + > + ret = 0; > + > +error_out: > + v4l2_fwnode_endpoint_free(&ep_cfg); > + fwnode_handle_put(endpoint); > + > + return ret; > +} > + > +static int imx300_probe(struct i2c_client *client) > +{ > + struct device *dev = &client->dev; > + struct imx300 *imx300; > + int ret; > + > + imx300 = devm_kzalloc(&client->dev, sizeof(*imx300), GFP_KERNEL); > + if (!imx300) > + return -ENOMEM; > + > + v4l2_i2c_subdev_init(&imx300->sd, client, &imx300_subdev_ops); > + > + /* Check the hardware configuration in device tree */ > + if (imx300_check_hwcfg(dev, imx300)) > + return -EINVAL; > + > + /* Get system clock (xclk) */ > + imx300->xclk = devm_clk_get(dev, NULL); > + if (IS_ERR(imx300->xclk)) { > + dev_err(dev, "failed to get xclk\n"); > + return PTR_ERR(imx300->xclk); > + } > + > + imx300->xclk_freq = clk_get_rate(imx300->xclk); > + if (imx300->xclk_freq != IMX300_XCLK_FREQ_24M) { > + dev_err(dev, "xclk frequency not supported: %d Hz\n", > + imx300->xclk_freq); > + return -EINVAL; > + } > + > + ret = imx300_get_regulators(imx300); > + if (ret) { > + dev_err(dev, "failed to get regulators\n"); > + return ret; > + } > + > + /* Request optional enable pin */ > + imx300->reset_gpio = devm_gpiod_get_optional(dev, "reset", > + GPIOD_OUT_HIGH); > + > + /* > + * The sensor must be powered for imx300_identify_module() > + * to be able to read the CHIP_ID register > + */ > + ret = imx300_power_on(dev); > + if (ret) > + return ret; > + > + ret = imx300_identify_module(imx300); > + if (ret) > + goto error_power_off; > + > + /* Set default mode to max resolution */ > + imx300->mode = &supported_modes[0]; > + > + /* Default is 10 bits per sample */ > + imx300->cur_bps = 10; Could you move this information to the mode definition? > + > + /* > + * Sensor doesn't enter LP-11 state upon power up until and unless > + * streaming is started, so upon power up switch the modes to: > + * streaming -> standby > + */ > + ret = imx300_write_reg(imx300, IMX300_REG_MODE_SELECT, > + IMX300_REG_VALUE_08BIT, IMX300_MODE_STREAMING); > + if (ret < 0) > + goto error_power_off; > + > + /* put sensor back to standby mode */ > + ret = imx300_write_reg(imx300, IMX300_REG_MODE_SELECT, > + IMX300_REG_VALUE_08BIT, IMX300_MODE_STANDBY); > + if (ret < 0) > + goto error_power_off; > + > + ret = imx300_init_controls(imx300); > + if (ret) > + goto error_power_off; > + > + /* Initialize subdev */ > + imx300->sd.internal_ops = &imx300_internal_ops; > + imx300->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE; > + imx300->sd.entity.function = MEDIA_ENT_F_CAM_SENSOR; > + > + /* Initialize source pad */ > + imx300->pad.flags = MEDIA_PAD_FL_SOURCE; > + > + /* Initialize default format */ > + imx300_set_default_format(imx300); This is related to the mode set earlier. So please use either. > + > + ret = media_entity_pads_init(&imx300->sd.entity, 1, &imx300->pad); > + if (ret) { > + dev_err(dev, "failed to init entity pads: %d\n", ret); > + goto error_handler_free; > + } > + > + ret = v4l2_async_register_subdev_sensor_common(&imx300->sd); > + if (ret < 0) { > + dev_err(dev, "failed to register sensor subdevice: %d\n", ret); > + goto error_media_entity; > + } > + > + /* Enable runtime PM and turn off the device */ > + pm_runtime_set_active(dev); > + pm_runtime_enable(dev); > + pm_runtime_idle(dev); > + > + /* Due to the sensor id being odd, let's inform about us... */ > + dev_info(dev, "Sony Exmor-RS IMX300 camera sensor is up.\n"); I'd use dev_dbg(). > + > + return 0; > + > +error_media_entity: > + media_entity_cleanup(&imx300->sd.entity); > + > +error_handler_free: > + imx300_free_controls(imx300); > + > +error_power_off: > + imx300_power_off(dev); > + > + return ret; > +} > + > +static int imx300_remove(struct i2c_client *client) > +{ > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct imx300 *imx300 = to_imx300(sd); > + > + v4l2_async_unregister_subdev(sd); > + media_entity_cleanup(&sd->entity); > + imx300_free_controls(imx300); > + > + pm_runtime_disable(&client->dev); > + if (!pm_runtime_status_suspended(&client->dev)) > + imx300_power_off(&client->dev); > + pm_runtime_set_suspended(&client->dev); > + > + return 0; > +} > + > +static const struct of_device_id imx300_dt_ids[] = { > + { .compatible = "sony,imx300" }, > + { /* sentinel */ } > +}; > +MODULE_DEVICE_TABLE(of, imx300_dt_ids); > + > +static const struct dev_pm_ops imx300_pm_ops = { > + SET_SYSTEM_SLEEP_PM_OPS(imx300_suspend, imx300_resume) > + SET_RUNTIME_PM_OPS(imx300_power_off, imx300_power_on, NULL) > +}; > + > +static struct i2c_driver imx300_i2c_driver = { > + .driver = { > + .name = "imx300", > + .of_match_table = imx300_dt_ids, > + .pm = &imx300_pm_ops, > + }, > + .probe_new = imx300_probe, > + .remove = imx300_remove, > +}; > + > +module_i2c_driver(imx300_i2c_driver); > + > +MODULE_AUTHOR("AngeloGioacchino Del Regno <kholk11@xxxxxxxxx>"); > +MODULE_DESCRIPTION("Sony Exmor-RS IMX300 camera sensor driver"); > +MODULE_LICENSE("GPL v2"); -- Kind regards, Sakari Ailus