[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <20210116174030.GS850@valkosipuli.retiisi.org.uk>
Date: Sat, 16 Jan 2021 19:40:30 +0200
From: Sakari Ailus <sakari.ailus@....fi>
To: AngeloGioacchino Del Regno <kholk11@...il.com>
Cc: Mauro Carvalho Chehab <mchehab@...nel.org>,
Rob Herring <robh+dt@...nel.org>, marijns95@...il.com,
Konrad Dybcio <konradybcio@...il.com>, martin.botka1@...il.com,
"open list:OPEN FIRMWARE AND FLATTENED DEVICE TREE BINDINGS"
<devicetree@...r.kernel.org>,
linux-media <linux-media@...r.kernel.org>,
phone-devel@...r.kernel.org,
linux-kernel <linux-kernel@...r.kernel.org>
Subject: Re: [PATCH v2 1/2] media: i2c: Add driver for the Sony Exmor-RS
IMX300 camera sensor
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@....fi> ha scritto:
> >
> > Hi AngeloGioacchino,
> >
> > Thanks for the patchset.
> >
> > On Thu, Oct 29, 2020 at 06:29:46PM +0100, kholk11@...il.com wrote:
> > > From: AngeloGioacchino Del Regno <kholk11@...il.com>
> > >
> > > 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@...il.com>
> > > ---
> > > 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@...il.com>
> > > + *
> > > + * 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@...il.com>");
> > > +MODULE_DESCRIPTION("Sony Exmor-RS IMX300 camera sensor driver");
> > > +MODULE_LICENSE("GPL v2");
--
Regards,
Sakari Ailus
Powered by blists - more mailing lists