Hi AngeloGioacchino, Apologies for the late reply. Please see my comments below... On Fri, Nov 27, 2020 at 10:34:27PM +0100, AngeloGioacchino Del Regno wrote: > Il giorno ven 27 nov 2020 alle ore 00:43 Sakari Ailus > <sakari.ailus@xxxxxx> ha scritto: > > > > 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? > > > A copy-paste from the cover letter for this patch series that I have > sent already (but I can understand: you weren't in the cc list): > > "This camera sensor driver was tested with all the resolutions declared in > it on two phones: Sony Xperia XA2 and XA2 Ultra, on a SDM630 SoC (camss > patches for this SoC will come in a later series) and is working great" Ack. > > > > > > > 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)? > > > I have checked from 0x0 to 0x100 and from 0x904 to the register space end, > which I don't really remember what it was right now. > Unfortunately, as I wrote in that comment, there are no meaningful values to > compare. Even trying to compare the registers at power-on is of no use: I > have verified that they change on at least a MSM8998 phone from Sony that > uses the same camera sensor. > I am a bit disappointed with Sony about this.. but there is no other way. > > After all, that's a "custom" sensor that is supposed to be used only from them, > and only on their smartphones, so they may have a point for not adding any > ID information in it. Ack. I don't know how they work, but omitting something like that seems odd. > > > > + * > > > + * 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. > > > I know, I know... But I did it like that to avoid doing bit shifting > and such, so to > simplify reading the already over-complicated table. > Otherwise I would have to define a EXCK_FREQ_24MHZ and a > EXCK_FREQ_8MHZ, then I would be out of the 80 cols, having to wrap-around > and impact on the readability. I'm ok with either. > > > > + { 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? > > > If by initial you mean the initial value that the registers have at > the first sensor > power-on, then... it's like 95% different. I tried to yeet out that 5% > but it seems > that the sensor will lock up if I do. Some register is resetting some > other as soon > as it gets written, most probably. Writing some registers also may have side effects. Would it be somehow different if you always wrote the two in series after power-up? > > I am sure that I can remove something like 3-4 entries from that magic sequence, > but researching about this for days didn't seem to be worth it. > I mean, if there was any way to shrink that list by something like even 25%, I > would've lost more time on it but.. from my findings, that's a big no... :( > > > > + { 0x0138, 0x01, IMX300_REG_VALUE_08BIT }, > > > > And this one enables the temperature sensor. > > > Are you *really* sure? Unless you have a datasheet for this specific sensor, > I would be cautious about this. I have tried to cross-check this with others and > sometimes they change, even if for just one bit. > I have no way of testing the temperature sensor in there for the moment, so > I also cannot verify that setting this to zero will stop the reading. > > I propose to leave it like this and get back to it when I will find a way (other > than when I will find the time to - as that's obviously the main issue > -) to test > the temp sensor and eventually even implement it.... Works for me. > > > > + { 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 ... > > > I really don't like ending functions like that, as any possible > addition at the end > of that would then result in more line changes than necessary. > But anyway, if it's *really* a necessity, I guess I can change it....... Please do. It makes these functions cleaner. > > > > + 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. > > > This is actually also selecting 8/10bits codes, as there you have both > supported. > ...So, should it? Yes, on both. > > > > + > > > + 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. > > > Let's remove it then :) > > > > + 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? > > > Depending on the target camera subsystem implementation, having a lower > pixel rate will allow for scaling clocks lower, thus saving power. > In any case, even if your camera subsystem does not allow it, then having > this sensor outputting at a lower pixel rate will reduce the produced heat: > you definitely know that more heat means more power going through. > > So, well, here you get power saving twice, if your platform allows. I wanted to say that this just updates the control value, it does not actually change the configuration. So I think the comment should be removed. > > > > + __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. > > > Yes but I would have to rewrite the setting tables or cripple the output > resolution if a device does not support the highest one. > I am mostly sure that all modern SoCs (or even less modern) have the > ability to use one link frequency at a time for a specific output mode. Note that this is board specific, not SoC. > > But then, that's also done to save power and to help preserving the > camera sensor by lowering the heat (nah, it's never overheating but > apparently if it stays hotter it's more likely to develop "black spots" > sooner than a sensor that's kept "colder). > > Not to mention that I haven't got any datasheet for this sensor (but from > the comments in the code .. I'm sure that's pretty clear) and rewriting > the setup is basically impossible like this. > I am totally sure you that you agree with me on that. I don't argue that, but should we have a board that only supports one of the frequencies, the driver should be able to work with that, shouldn't it? IOW, that looks like a bug (or at least an odd limitation) to me. > > > > + 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? > > > I would have to de-constify the struct, as the cur_bps parameter is > reassigned at every BPS change. It's also true that I can check the > fmt.code instead, but doing one check instead of four everytime looks > better to me. > If you really wish to see such a check, then... even if it looks bad to > me, I guess my keyboard won't refuse to... Ah, I see now that you can use the same mode on both bpps, unlike virtually all other register list based drivers. Feel free to keep it as-is. > > > > + > > > + /* > > > + * 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. > > > Sorry, I have copied that from the IMX219 driver, assuming that it was right. > If you're saying that's not, then we should fix that one as well some day... > But hey, that's not a bad thing at all! I have some imx219 "things" here that > I was planning to push as soon as possible. I will include that fix there too! > > > > + > > > + 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(). > > > I chose to use dev_info because of the sensor ID issue. So I would be > happy to keep it as dev_info. Is it that bad? I can surely change it, let > me know. Most other sensor drivers are silent if all goes well, just saying something if there's an error. > > > + > > > + 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"); -- Regards, Sakari Ailus