xref: /openbmc/linux/drivers/media/i2c/ov8856.c (revision aaeb31c0)
1879347f0SBen Kao // SPDX-License-Identifier: GPL-2.0
2879347f0SBen Kao // Copyright (c) 2019 Intel Corporation.
3879347f0SBen Kao 
4879347f0SBen Kao #include <asm/unaligned.h>
5879347f0SBen Kao #include <linux/acpi.h>
60c2c7a1eSRobert Foss #include <linux/clk.h>
7879347f0SBen Kao #include <linux/delay.h>
80c2c7a1eSRobert Foss #include <linux/gpio/consumer.h>
9879347f0SBen Kao #include <linux/i2c.h>
10879347f0SBen Kao #include <linux/module.h>
11879347f0SBen Kao #include <linux/pm_runtime.h>
120c2c7a1eSRobert Foss #include <linux/regulator/consumer.h>
13879347f0SBen Kao #include <media/v4l2-ctrls.h>
14879347f0SBen Kao #include <media/v4l2-device.h>
15879347f0SBen Kao #include <media/v4l2-fwnode.h>
16879347f0SBen Kao 
17879347f0SBen Kao #define OV8856_REG_VALUE_08BIT		1
18879347f0SBen Kao #define OV8856_REG_VALUE_16BIT		2
19879347f0SBen Kao #define OV8856_REG_VALUE_24BIT		3
20879347f0SBen Kao 
21879347f0SBen Kao #define OV8856_SCLK			144000000ULL
220c2c7a1eSRobert Foss #define OV8856_XVCLK_19_2		19200000
23879347f0SBen Kao #define OV8856_DATA_LANES		4
24879347f0SBen Kao #define OV8856_RGB_DEPTH		10
25879347f0SBen Kao 
26879347f0SBen Kao #define OV8856_REG_CHIP_ID		0x300a
27879347f0SBen Kao #define OV8856_CHIP_ID			0x00885a
28879347f0SBen Kao 
29879347f0SBen Kao #define OV8856_REG_MODE_SELECT		0x0100
30879347f0SBen Kao #define OV8856_MODE_STANDBY		0x00
31879347f0SBen Kao #define OV8856_MODE_STREAMING		0x01
32879347f0SBen Kao 
3396b5b11aSRobert Foss /* module revisions */
3496b5b11aSRobert Foss #define OV8856_2A_MODULE		0x01
3596b5b11aSRobert Foss #define OV8856_1B_MODULE		0x02
3696b5b11aSRobert Foss 
3796b5b11aSRobert Foss /* the OTP read-out buffer is at 0x7000 and 0xf is the offset
3896b5b11aSRobert Foss  * of the byte in the OTP that means the module revision
3996b5b11aSRobert Foss  */
4096b5b11aSRobert Foss #define OV8856_MODULE_REVISION		0x700f
4196b5b11aSRobert Foss #define OV8856_OTP_MODE_CTRL		0x3d84
4296b5b11aSRobert Foss #define OV8856_OTP_LOAD_CTRL		0x3d81
4396b5b11aSRobert Foss #define OV8856_OTP_MODE_AUTO		0x00
4496b5b11aSRobert Foss #define OV8856_OTP_LOAD_CTRL_ENABLE	BIT(0)
4596b5b11aSRobert Foss 
46879347f0SBen Kao /* vertical-timings from sensor */
47879347f0SBen Kao #define OV8856_REG_VTS			0x380e
48879347f0SBen Kao #define OV8856_VTS_MAX			0x7fff
49879347f0SBen Kao 
50879347f0SBen Kao /* horizontal-timings from sensor */
51879347f0SBen Kao #define OV8856_REG_HTS			0x380c
52879347f0SBen Kao 
53879347f0SBen Kao /* Exposure controls from sensor */
54879347f0SBen Kao #define OV8856_REG_EXPOSURE		0x3500
55879347f0SBen Kao #define	OV8856_EXPOSURE_MIN		6
56879347f0SBen Kao #define OV8856_EXPOSURE_MAX_MARGIN	6
57879347f0SBen Kao #define	OV8856_EXPOSURE_STEP		1
58879347f0SBen Kao 
59879347f0SBen Kao /* Analog gain controls from sensor */
60879347f0SBen Kao #define OV8856_REG_ANALOG_GAIN		0x3508
61879347f0SBen Kao #define	OV8856_ANAL_GAIN_MIN		128
62879347f0SBen Kao #define	OV8856_ANAL_GAIN_MAX		2047
63879347f0SBen Kao #define	OV8856_ANAL_GAIN_STEP		1
64879347f0SBen Kao 
65879347f0SBen Kao /* Digital gain controls from sensor */
66ba43392eSBingbu Cao #define OV8856_REG_DIGITAL_GAIN		0x350a
67879347f0SBen Kao #define OV8856_REG_MWB_R_GAIN		0x5019
68879347f0SBen Kao #define OV8856_REG_MWB_G_GAIN		0x501b
69879347f0SBen Kao #define OV8856_REG_MWB_B_GAIN		0x501d
70879347f0SBen Kao #define OV8856_DGTL_GAIN_MIN		0
71879347f0SBen Kao #define OV8856_DGTL_GAIN_MAX		4095
72879347f0SBen Kao #define OV8856_DGTL_GAIN_STEP		1
73879347f0SBen Kao #define OV8856_DGTL_GAIN_DEFAULT	1024
74879347f0SBen Kao 
75879347f0SBen Kao /* Test Pattern Control */
76879347f0SBen Kao #define OV8856_REG_TEST_PATTERN		0x5e00
77879347f0SBen Kao #define OV8856_TEST_PATTERN_ENABLE	BIT(7)
78879347f0SBen Kao #define OV8856_TEST_PATTERN_BAR_SHIFT	2
79879347f0SBen Kao 
80c492ec9aSShawn Tu #define NUM_REGS				7
81c492ec9aSShawn Tu #define NUM_MODE_REGS				187
82c492ec9aSShawn Tu #define NUM_MODE_REGS_2				200
83c492ec9aSShawn Tu 
84c19b93a6SShawn Tu /* Flip Mirror Controls from sensor */
85c19b93a6SShawn Tu #define OV8856_REG_FORMAT1			0x3820
86c19b93a6SShawn Tu #define OV8856_REG_FORMAT2			0x3821
87c19b93a6SShawn Tu #define OV8856_REG_FORMAT1_OP_1			BIT(1)
88c19b93a6SShawn Tu #define OV8856_REG_FORMAT1_OP_2			BIT(2)
89c19b93a6SShawn Tu #define OV8856_REG_FORMAT1_OP_3			BIT(6)
90c19b93a6SShawn Tu #define OV8856_REG_FORMAT2_OP_1			BIT(1)
91c19b93a6SShawn Tu #define OV8856_REG_FORMAT2_OP_2			BIT(2)
92c19b93a6SShawn Tu #define OV8856_REG_FORMAT2_OP_3			BIT(6)
93c19b93a6SShawn Tu #define OV8856_REG_FLIP_OPT_1			0x376b
94c19b93a6SShawn Tu #define OV8856_REG_FLIP_OPT_2			0x5001
95c19b93a6SShawn Tu #define OV8856_REG_FLIP_OPT_3			0x502e
96c19b93a6SShawn Tu #define OV8856_REG_MIRROR_OPT_1			0x5004
97c19b93a6SShawn Tu #define OV8856_REG_FLIP_OP_0			BIT(0)
98c19b93a6SShawn Tu #define OV8856_REG_FLIP_OP_1			BIT(1)
99c19b93a6SShawn Tu #define OV8856_REG_FLIP_OP_2			BIT(2)
100c19b93a6SShawn Tu #define OV8856_REG_MIRROR_OP_1			BIT(1)
101c19b93a6SShawn Tu #define OV8856_REG_MIRROR_OP_2			BIT(2)
102c19b93a6SShawn Tu 
103879347f0SBen Kao #define to_ov8856(_sd)			container_of(_sd, struct ov8856, sd)
104879347f0SBen Kao 
1050c2c7a1eSRobert Foss static const char * const ov8856_supply_names[] = {
1060c2c7a1eSRobert Foss 	"dovdd",	/* Digital I/O power */
1070c2c7a1eSRobert Foss 	"avdd",		/* Analog power */
1080c2c7a1eSRobert Foss 	"dvdd",		/* Digital core power */
1090c2c7a1eSRobert Foss };
1100c2c7a1eSRobert Foss 
111f1363166SHsin-Yi Wang enum {
112f1363166SHsin-Yi Wang 	OV8856_MEDIA_BUS_FMT_SBGGR10_1X10,
113f1363166SHsin-Yi Wang 	OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
114f1363166SHsin-Yi Wang };
115f1363166SHsin-Yi Wang 
116879347f0SBen Kao struct ov8856_reg {
117879347f0SBen Kao 	u16 address;
118879347f0SBen Kao 	u8 val;
119879347f0SBen Kao };
120879347f0SBen Kao 
121879347f0SBen Kao struct ov8856_reg_list {
122879347f0SBen Kao 	u32 num_of_regs;
123879347f0SBen Kao 	const struct ov8856_reg *regs;
124879347f0SBen Kao };
125879347f0SBen Kao 
126879347f0SBen Kao struct ov8856_link_freq_config {
127879347f0SBen Kao 	const struct ov8856_reg_list reg_list;
128879347f0SBen Kao };
129879347f0SBen Kao 
130879347f0SBen Kao struct ov8856_mode {
131879347f0SBen Kao 	/* Frame width in pixels */
132879347f0SBen Kao 	u32 width;
133879347f0SBen Kao 
134879347f0SBen Kao 	/* Frame height in pixels */
135879347f0SBen Kao 	u32 height;
136879347f0SBen Kao 
137879347f0SBen Kao 	/* Horizontal timining size */
138879347f0SBen Kao 	u32 hts;
139879347f0SBen Kao 
140879347f0SBen Kao 	/* Default vertical timining size */
141879347f0SBen Kao 	u32 vts_def;
142879347f0SBen Kao 
143879347f0SBen Kao 	/* Min vertical timining size */
144879347f0SBen Kao 	u32 vts_min;
145879347f0SBen Kao 
146879347f0SBen Kao 	/* Link frequency needed for this resolution */
147879347f0SBen Kao 	u32 link_freq_index;
148879347f0SBen Kao 
149879347f0SBen Kao 	/* Sensor register settings for this resolution */
150879347f0SBen Kao 	const struct ov8856_reg_list reg_list;
151c492ec9aSShawn Tu 
152c492ec9aSShawn Tu 	/* Number of data lanes */
153c492ec9aSShawn Tu 	u8 data_lanes;
154f1363166SHsin-Yi Wang 
155f1363166SHsin-Yi Wang 	/* Default MEDIA_BUS_FMT for this mode */
156f1363166SHsin-Yi Wang 	u32 default_mbus_index;
157879347f0SBen Kao };
158879347f0SBen Kao 
159c492ec9aSShawn Tu struct ov8856_mipi_data_rates {
160c492ec9aSShawn Tu 	const struct ov8856_reg regs_0[NUM_REGS];
161c492ec9aSShawn Tu 	const struct ov8856_reg regs_1[NUM_REGS];
162c492ec9aSShawn Tu };
163c492ec9aSShawn Tu 
164c492ec9aSShawn Tu static const struct ov8856_mipi_data_rates mipi_data_rate_lane_2 = {
165c492ec9aSShawn Tu 	//mipi_data_rate_1440mbps
166c492ec9aSShawn Tu 	{
167c492ec9aSShawn Tu 		{0x0103, 0x01},
168c492ec9aSShawn Tu 		{0x0100, 0x00},
169c492ec9aSShawn Tu 		{0x0302, 0x43},
170c492ec9aSShawn Tu 		{0x0303, 0x00},
171c492ec9aSShawn Tu 		{0x030b, 0x02},
172c492ec9aSShawn Tu 		{0x030d, 0x4b},
173c492ec9aSShawn Tu 		{0x031e, 0x0c}
174c492ec9aSShawn Tu 	},
175c492ec9aSShawn Tu 	//mipi_data_rate_720mbps
176c492ec9aSShawn Tu 	{
177879347f0SBen Kao 		{0x0103, 0x01},
178879347f0SBen Kao 		{0x0100, 0x00},
179879347f0SBen Kao 		{0x0302, 0x4b},
180879347f0SBen Kao 		{0x0303, 0x01},
181879347f0SBen Kao 		{0x030b, 0x02},
182879347f0SBen Kao 		{0x030d, 0x4b},
183c492ec9aSShawn Tu 		{0x031e, 0x0c}
184c492ec9aSShawn Tu 	}
185879347f0SBen Kao };
186879347f0SBen Kao 
187c492ec9aSShawn Tu static const struct ov8856_mipi_data_rates mipi_data_rate_lane_4 = {
188c492ec9aSShawn Tu 	//mipi_data_rate_720mbps
189c492ec9aSShawn Tu 	{
190c492ec9aSShawn Tu 		{0x0103, 0x01},
191c492ec9aSShawn Tu 		{0x0100, 0x00},
192c492ec9aSShawn Tu 		{0x0302, 0x4b},
193c492ec9aSShawn Tu 		{0x0303, 0x01},
194c492ec9aSShawn Tu 		{0x030b, 0x02},
195c492ec9aSShawn Tu 		{0x030d, 0x4b},
196c492ec9aSShawn Tu 		{0x031e, 0x0c}
197c492ec9aSShawn Tu 	},
198c492ec9aSShawn Tu 	//mipi_data_rate_360mbps
199c492ec9aSShawn Tu 	{
200879347f0SBen Kao 		{0x0103, 0x01},
201879347f0SBen Kao 		{0x0100, 0x00},
202879347f0SBen Kao 		{0x0302, 0x4b},
203879347f0SBen Kao 		{0x0303, 0x03},
204879347f0SBen Kao 		{0x030b, 0x02},
205879347f0SBen Kao 		{0x030d, 0x4b},
206c492ec9aSShawn Tu 		{0x031e, 0x0c}
207c492ec9aSShawn Tu 	}
208879347f0SBen Kao };
209879347f0SBen Kao 
210c492ec9aSShawn Tu static const struct ov8856_reg lane_2_mode_3280x2464[] = {
211c492ec9aSShawn Tu 	/* 3280x2464 resolution */
212c492ec9aSShawn Tu 		{0x3000, 0x20},
213c492ec9aSShawn Tu 		{0x3003, 0x08},
214c492ec9aSShawn Tu 		{0x300e, 0x20},
215c492ec9aSShawn Tu 		{0x3010, 0x00},
216c492ec9aSShawn Tu 		{0x3015, 0x84},
217c492ec9aSShawn Tu 		{0x3018, 0x32},
218c492ec9aSShawn Tu 		{0x3021, 0x23},
219c492ec9aSShawn Tu 		{0x3033, 0x24},
220c492ec9aSShawn Tu 		{0x3500, 0x00},
221c492ec9aSShawn Tu 		{0x3501, 0x9a},
222c492ec9aSShawn Tu 		{0x3502, 0x20},
223c492ec9aSShawn Tu 		{0x3503, 0x08},
224c492ec9aSShawn Tu 		{0x3505, 0x83},
225c492ec9aSShawn Tu 		{0x3508, 0x01},
226c492ec9aSShawn Tu 		{0x3509, 0x80},
227c492ec9aSShawn Tu 		{0x350c, 0x00},
228c492ec9aSShawn Tu 		{0x350d, 0x80},
229c492ec9aSShawn Tu 		{0x350e, 0x04},
230c492ec9aSShawn Tu 		{0x350f, 0x00},
231c492ec9aSShawn Tu 		{0x3510, 0x00},
232c492ec9aSShawn Tu 		{0x3511, 0x02},
233c492ec9aSShawn Tu 		{0x3512, 0x00},
234c492ec9aSShawn Tu 		{0x3600, 0x72},
235c492ec9aSShawn Tu 		{0x3601, 0x40},
236c492ec9aSShawn Tu 		{0x3602, 0x30},
237c492ec9aSShawn Tu 		{0x3610, 0xc5},
238c492ec9aSShawn Tu 		{0x3611, 0x58},
239c492ec9aSShawn Tu 		{0x3612, 0x5c},
240c492ec9aSShawn Tu 		{0x3613, 0xca},
241c492ec9aSShawn Tu 		{0x3614, 0x50},
242c492ec9aSShawn Tu 		{0x3628, 0xff},
243c492ec9aSShawn Tu 		{0x3629, 0xff},
244c492ec9aSShawn Tu 		{0x362a, 0xff},
245c492ec9aSShawn Tu 		{0x3633, 0x10},
246c492ec9aSShawn Tu 		{0x3634, 0x10},
247c492ec9aSShawn Tu 		{0x3635, 0x10},
248c492ec9aSShawn Tu 		{0x3636, 0x10},
249c492ec9aSShawn Tu 		{0x3663, 0x08},
250c492ec9aSShawn Tu 		{0x3669, 0x34},
251c492ec9aSShawn Tu 		{0x366e, 0x10},
252c492ec9aSShawn Tu 		{0x3706, 0x86},
253c492ec9aSShawn Tu 		{0x370b, 0x7e},
254c492ec9aSShawn Tu 		{0x3714, 0x23},
255c492ec9aSShawn Tu 		{0x3730, 0x12},
256c492ec9aSShawn Tu 		{0x3733, 0x10},
257c492ec9aSShawn Tu 		{0x3764, 0x00},
258c492ec9aSShawn Tu 		{0x3765, 0x00},
259c492ec9aSShawn Tu 		{0x3769, 0x62},
260c492ec9aSShawn Tu 		{0x376a, 0x2a},
261c492ec9aSShawn Tu 		{0x376b, 0x30},
262c492ec9aSShawn Tu 		{0x3780, 0x00},
263c492ec9aSShawn Tu 		{0x3781, 0x24},
264c492ec9aSShawn Tu 		{0x3782, 0x00},
265c492ec9aSShawn Tu 		{0x3783, 0x23},
266c492ec9aSShawn Tu 		{0x3798, 0x2f},
267c492ec9aSShawn Tu 		{0x37a1, 0x60},
268c492ec9aSShawn Tu 		{0x37a8, 0x6a},
269c492ec9aSShawn Tu 		{0x37ab, 0x3f},
270c492ec9aSShawn Tu 		{0x37c2, 0x04},
271c492ec9aSShawn Tu 		{0x37c3, 0xf1},
272c492ec9aSShawn Tu 		{0x37c9, 0x80},
273c492ec9aSShawn Tu 		{0x37cb, 0x16},
274c492ec9aSShawn Tu 		{0x37cc, 0x16},
275c492ec9aSShawn Tu 		{0x37cd, 0x16},
276c492ec9aSShawn Tu 		{0x37ce, 0x16},
277c492ec9aSShawn Tu 		{0x3800, 0x00},
278c492ec9aSShawn Tu 		{0x3801, 0x00},
279c492ec9aSShawn Tu 		{0x3802, 0x00},
280c492ec9aSShawn Tu 		{0x3803, 0x06},
281c492ec9aSShawn Tu 		{0x3804, 0x0c},
282c492ec9aSShawn Tu 		{0x3805, 0xdf},
283c492ec9aSShawn Tu 		{0x3806, 0x09},
284c492ec9aSShawn Tu 		{0x3807, 0xa7},
285c492ec9aSShawn Tu 		{0x3808, 0x0c},
286c492ec9aSShawn Tu 		{0x3809, 0xd0},
287c492ec9aSShawn Tu 		{0x380a, 0x09},
288c492ec9aSShawn Tu 		{0x380b, 0xa0},
289c492ec9aSShawn Tu 		{0x380c, 0x07},
290c492ec9aSShawn Tu 		{0x380d, 0x88},
291c492ec9aSShawn Tu 		{0x380e, 0x09},
292c492ec9aSShawn Tu 		{0x380f, 0xb8},
293c492ec9aSShawn Tu 		{0x3810, 0x00},
294c492ec9aSShawn Tu 		{0x3811, 0x00},
295c492ec9aSShawn Tu 		{0x3812, 0x00},
296c492ec9aSShawn Tu 		{0x3813, 0x01},
297c492ec9aSShawn Tu 		{0x3814, 0x01},
298c492ec9aSShawn Tu 		{0x3815, 0x01},
299c492ec9aSShawn Tu 		{0x3816, 0x00},
300c492ec9aSShawn Tu 		{0x3817, 0x00},
301c492ec9aSShawn Tu 		{0x3818, 0x00},
302c492ec9aSShawn Tu 		{0x3819, 0x00},
303c492ec9aSShawn Tu 		{0x3820, 0x80},
304c492ec9aSShawn Tu 		{0x3821, 0x46},
305c492ec9aSShawn Tu 		{0x382a, 0x01},
306c492ec9aSShawn Tu 		{0x382b, 0x01},
307c492ec9aSShawn Tu 		{0x3830, 0x06},
308c492ec9aSShawn Tu 		{0x3836, 0x02},
309c492ec9aSShawn Tu 		{0x3837, 0x10},
310c492ec9aSShawn Tu 		{0x3862, 0x04},
311c492ec9aSShawn Tu 		{0x3863, 0x08},
312c492ec9aSShawn Tu 		{0x3cc0, 0x33},
313c492ec9aSShawn Tu 		{0x3d85, 0x14},
314c492ec9aSShawn Tu 		{0x3d8c, 0x73},
315c492ec9aSShawn Tu 		{0x3d8d, 0xde},
316c492ec9aSShawn Tu 		{0x4001, 0xe0},
317c492ec9aSShawn Tu 		{0x4003, 0x40},
318c492ec9aSShawn Tu 		{0x4008, 0x00},
319c492ec9aSShawn Tu 		{0x4009, 0x0b},
320c492ec9aSShawn Tu 		{0x400a, 0x00},
321c492ec9aSShawn Tu 		{0x400b, 0x84},
322c492ec9aSShawn Tu 		{0x400f, 0x80},
323c492ec9aSShawn Tu 		{0x4010, 0xf0},
324c492ec9aSShawn Tu 		{0x4011, 0xff},
325c492ec9aSShawn Tu 		{0x4012, 0x02},
326c492ec9aSShawn Tu 		{0x4013, 0x01},
327c492ec9aSShawn Tu 		{0x4014, 0x01},
328c492ec9aSShawn Tu 		{0x4015, 0x01},
329c492ec9aSShawn Tu 		{0x4042, 0x00},
330c492ec9aSShawn Tu 		{0x4043, 0x80},
331c492ec9aSShawn Tu 		{0x4044, 0x00},
332c492ec9aSShawn Tu 		{0x4045, 0x80},
333c492ec9aSShawn Tu 		{0x4046, 0x00},
334c492ec9aSShawn Tu 		{0x4047, 0x80},
335c492ec9aSShawn Tu 		{0x4048, 0x00},
336c492ec9aSShawn Tu 		{0x4049, 0x80},
337c492ec9aSShawn Tu 		{0x4041, 0x03},
338c492ec9aSShawn Tu 		{0x404c, 0x20},
339c492ec9aSShawn Tu 		{0x404d, 0x00},
340c492ec9aSShawn Tu 		{0x404e, 0x20},
341c492ec9aSShawn Tu 		{0x4203, 0x80},
342c492ec9aSShawn Tu 		{0x4307, 0x30},
343c492ec9aSShawn Tu 		{0x4317, 0x00},
344c492ec9aSShawn Tu 		{0x4503, 0x08},
345c492ec9aSShawn Tu 		{0x4601, 0x80},
346c492ec9aSShawn Tu 		{0x4800, 0x44},
347c492ec9aSShawn Tu 		{0x4816, 0x53},
348c492ec9aSShawn Tu 		{0x481b, 0x58},
349c492ec9aSShawn Tu 		{0x481f, 0x27},
350c492ec9aSShawn Tu 		{0x4837, 0x0c},
351c492ec9aSShawn Tu 		{0x483c, 0x0f},
352c492ec9aSShawn Tu 		{0x484b, 0x05},
353c492ec9aSShawn Tu 		{0x5000, 0x57},
354c492ec9aSShawn Tu 		{0x5001, 0x0a},
355ba43392eSBingbu Cao 		{0x5004, 0x06},
356c492ec9aSShawn Tu 		{0x502e, 0x03},
357c492ec9aSShawn Tu 		{0x5030, 0x41},
358c492ec9aSShawn Tu 		{0x5795, 0x02},
359c492ec9aSShawn Tu 		{0x5796, 0x20},
360c492ec9aSShawn Tu 		{0x5797, 0x20},
361c492ec9aSShawn Tu 		{0x5798, 0xd5},
362c492ec9aSShawn Tu 		{0x5799, 0xd5},
363c492ec9aSShawn Tu 		{0x579a, 0x00},
364c492ec9aSShawn Tu 		{0x579b, 0x50},
365c492ec9aSShawn Tu 		{0x579c, 0x00},
366c492ec9aSShawn Tu 		{0x579d, 0x2c},
367c492ec9aSShawn Tu 		{0x579e, 0x0c},
368c492ec9aSShawn Tu 		{0x579f, 0x40},
369c492ec9aSShawn Tu 		{0x57a0, 0x09},
370c492ec9aSShawn Tu 		{0x57a1, 0x40},
371c492ec9aSShawn Tu 		{0x5780, 0x14},
372c492ec9aSShawn Tu 		{0x5781, 0x0f},
373c492ec9aSShawn Tu 		{0x5782, 0x44},
374c492ec9aSShawn Tu 		{0x5783, 0x02},
375c492ec9aSShawn Tu 		{0x5784, 0x01},
376c492ec9aSShawn Tu 		{0x5785, 0x01},
377c492ec9aSShawn Tu 		{0x5786, 0x00},
378c492ec9aSShawn Tu 		{0x5787, 0x04},
379c492ec9aSShawn Tu 		{0x5788, 0x02},
380c492ec9aSShawn Tu 		{0x5789, 0x0f},
381c492ec9aSShawn Tu 		{0x578a, 0xfd},
382c492ec9aSShawn Tu 		{0x578b, 0xf5},
383c492ec9aSShawn Tu 		{0x578c, 0xf5},
384c492ec9aSShawn Tu 		{0x578d, 0x03},
385c492ec9aSShawn Tu 		{0x578e, 0x08},
386c492ec9aSShawn Tu 		{0x578f, 0x0c},
387c492ec9aSShawn Tu 		{0x5790, 0x08},
388c492ec9aSShawn Tu 		{0x5791, 0x04},
389c492ec9aSShawn Tu 		{0x5792, 0x00},
390c492ec9aSShawn Tu 		{0x5793, 0x52},
391c492ec9aSShawn Tu 		{0x5794, 0xa3},
392c492ec9aSShawn Tu 		{0x59f8, 0x3d},
393c492ec9aSShawn Tu 		{0x5a08, 0x02},
394c492ec9aSShawn Tu 		{0x5b00, 0x02},
395c492ec9aSShawn Tu 		{0x5b01, 0x10},
396c492ec9aSShawn Tu 		{0x5b02, 0x03},
397c492ec9aSShawn Tu 		{0x5b03, 0xcf},
398c492ec9aSShawn Tu 		{0x5b05, 0x6c},
399c492ec9aSShawn Tu 		{0x5e00, 0x00}
400c492ec9aSShawn Tu };
401c492ec9aSShawn Tu 
402c492ec9aSShawn Tu static const struct ov8856_reg lane_2_mode_1640x1232[] = {
403c492ec9aSShawn Tu 	/* 1640x1232 resolution */
404c492ec9aSShawn Tu 		{0x3000, 0x20},
405c492ec9aSShawn Tu 		{0x3003, 0x08},
406c492ec9aSShawn Tu 		{0x300e, 0x20},
407c492ec9aSShawn Tu 		{0x3010, 0x00},
408c492ec9aSShawn Tu 		{0x3015, 0x84},
409c492ec9aSShawn Tu 		{0x3018, 0x32},
410c492ec9aSShawn Tu 		{0x3021, 0x23},
411c492ec9aSShawn Tu 		{0x3033, 0x24},
412c492ec9aSShawn Tu 		{0x3500, 0x00},
413c492ec9aSShawn Tu 		{0x3501, 0x4c},
414c492ec9aSShawn Tu 		{0x3502, 0xe0},
415c492ec9aSShawn Tu 		{0x3503, 0x08},
416c492ec9aSShawn Tu 		{0x3505, 0x83},
417c492ec9aSShawn Tu 		{0x3508, 0x01},
418c492ec9aSShawn Tu 		{0x3509, 0x80},
419c492ec9aSShawn Tu 		{0x350c, 0x00},
420c492ec9aSShawn Tu 		{0x350d, 0x80},
421c492ec9aSShawn Tu 		{0x350e, 0x04},
422c492ec9aSShawn Tu 		{0x350f, 0x00},
423c492ec9aSShawn Tu 		{0x3510, 0x00},
424c492ec9aSShawn Tu 		{0x3511, 0x02},
425c492ec9aSShawn Tu 		{0x3512, 0x00},
426c492ec9aSShawn Tu 		{0x3600, 0x72},
427c492ec9aSShawn Tu 		{0x3601, 0x40},
428c492ec9aSShawn Tu 		{0x3602, 0x30},
429c492ec9aSShawn Tu 		{0x3610, 0xc5},
430c492ec9aSShawn Tu 		{0x3611, 0x58},
431c492ec9aSShawn Tu 		{0x3612, 0x5c},
432c492ec9aSShawn Tu 		{0x3613, 0xca},
433c492ec9aSShawn Tu 		{0x3614, 0x50},
434c492ec9aSShawn Tu 		{0x3628, 0xff},
435c492ec9aSShawn Tu 		{0x3629, 0xff},
436c492ec9aSShawn Tu 		{0x362a, 0xff},
437c492ec9aSShawn Tu 		{0x3633, 0x10},
438c492ec9aSShawn Tu 		{0x3634, 0x10},
439c492ec9aSShawn Tu 		{0x3635, 0x10},
440c492ec9aSShawn Tu 		{0x3636, 0x10},
441c492ec9aSShawn Tu 		{0x3663, 0x08},
442c492ec9aSShawn Tu 		{0x3669, 0x34},
443c492ec9aSShawn Tu 		{0x366e, 0x08},
444c492ec9aSShawn Tu 		{0x3706, 0x86},
445c492ec9aSShawn Tu 		{0x370b, 0x7e},
446c492ec9aSShawn Tu 		{0x3714, 0x27},
447c492ec9aSShawn Tu 		{0x3730, 0x12},
448c492ec9aSShawn Tu 		{0x3733, 0x10},
449c492ec9aSShawn Tu 		{0x3764, 0x00},
450c492ec9aSShawn Tu 		{0x3765, 0x00},
451c492ec9aSShawn Tu 		{0x3769, 0x62},
452c492ec9aSShawn Tu 		{0x376a, 0x2a},
453c492ec9aSShawn Tu 		{0x376b, 0x30},
454c492ec9aSShawn Tu 		{0x3780, 0x00},
455c492ec9aSShawn Tu 		{0x3781, 0x24},
456c492ec9aSShawn Tu 		{0x3782, 0x00},
457c492ec9aSShawn Tu 		{0x3783, 0x23},
458c492ec9aSShawn Tu 		{0x3798, 0x2f},
459c492ec9aSShawn Tu 		{0x37a1, 0x60},
460c492ec9aSShawn Tu 		{0x37a8, 0x6a},
461c492ec9aSShawn Tu 		{0x37ab, 0x3f},
462c492ec9aSShawn Tu 		{0x37c2, 0x14},
463c492ec9aSShawn Tu 		{0x37c3, 0xf1},
464c492ec9aSShawn Tu 		{0x37c9, 0x80},
465c492ec9aSShawn Tu 		{0x37cb, 0x16},
466c492ec9aSShawn Tu 		{0x37cc, 0x16},
467c492ec9aSShawn Tu 		{0x37cd, 0x16},
468c492ec9aSShawn Tu 		{0x37ce, 0x16},
469c492ec9aSShawn Tu 		{0x3800, 0x00},
470c492ec9aSShawn Tu 		{0x3801, 0x00},
471c492ec9aSShawn Tu 		{0x3802, 0x00},
472c492ec9aSShawn Tu 		{0x3803, 0x00},
473c492ec9aSShawn Tu 		{0x3804, 0x0c},
474c492ec9aSShawn Tu 		{0x3805, 0xdf},
475c492ec9aSShawn Tu 		{0x3806, 0x09},
476c492ec9aSShawn Tu 		{0x3807, 0xaf},
477c492ec9aSShawn Tu 		{0x3808, 0x06},
478c492ec9aSShawn Tu 		{0x3809, 0x68},
479c492ec9aSShawn Tu 		{0x380a, 0x04},
480c492ec9aSShawn Tu 		{0x380b, 0xd0},
481c492ec9aSShawn Tu 		{0x380c, 0x0c},
482c492ec9aSShawn Tu 		{0x380d, 0x60},
483c492ec9aSShawn Tu 		{0x380e, 0x05},
484c492ec9aSShawn Tu 		{0x380f, 0xea},
485c492ec9aSShawn Tu 		{0x3810, 0x00},
486c492ec9aSShawn Tu 		{0x3811, 0x04},
487c492ec9aSShawn Tu 		{0x3812, 0x00},
488c492ec9aSShawn Tu 		{0x3813, 0x05},
489c492ec9aSShawn Tu 		{0x3814, 0x03},
490c492ec9aSShawn Tu 		{0x3815, 0x01},
491c492ec9aSShawn Tu 		{0x3816, 0x00},
492c492ec9aSShawn Tu 		{0x3817, 0x00},
493c492ec9aSShawn Tu 		{0x3818, 0x00},
494c492ec9aSShawn Tu 		{0x3819, 0x00},
495c492ec9aSShawn Tu 		{0x3820, 0x90},
496c492ec9aSShawn Tu 		{0x3821, 0x67},
497c492ec9aSShawn Tu 		{0x382a, 0x03},
498c492ec9aSShawn Tu 		{0x382b, 0x01},
499c492ec9aSShawn Tu 		{0x3830, 0x06},
500c492ec9aSShawn Tu 		{0x3836, 0x02},
501c492ec9aSShawn Tu 		{0x3837, 0x10},
502c492ec9aSShawn Tu 		{0x3862, 0x04},
503c492ec9aSShawn Tu 		{0x3863, 0x08},
504c492ec9aSShawn Tu 		{0x3cc0, 0x33},
505c492ec9aSShawn Tu 		{0x3d85, 0x14},
506c492ec9aSShawn Tu 		{0x3d8c, 0x73},
507c492ec9aSShawn Tu 		{0x3d8d, 0xde},
508c492ec9aSShawn Tu 		{0x4001, 0xe0},
509c492ec9aSShawn Tu 		{0x4003, 0x40},
510c492ec9aSShawn Tu 		{0x4008, 0x00},
511c492ec9aSShawn Tu 		{0x4009, 0x05},
512c492ec9aSShawn Tu 		{0x400a, 0x00},
513c492ec9aSShawn Tu 		{0x400b, 0x84},
514c492ec9aSShawn Tu 		{0x400f, 0x80},
515c492ec9aSShawn Tu 		{0x4010, 0xf0},
516c492ec9aSShawn Tu 		{0x4011, 0xff},
517c492ec9aSShawn Tu 		{0x4012, 0x02},
518c492ec9aSShawn Tu 		{0x4013, 0x01},
519c492ec9aSShawn Tu 		{0x4014, 0x01},
520c492ec9aSShawn Tu 		{0x4015, 0x01},
521c492ec9aSShawn Tu 		{0x4042, 0x00},
522c492ec9aSShawn Tu 		{0x4043, 0x80},
523c492ec9aSShawn Tu 		{0x4044, 0x00},
524c492ec9aSShawn Tu 		{0x4045, 0x80},
525c492ec9aSShawn Tu 		{0x4046, 0x00},
526c492ec9aSShawn Tu 		{0x4047, 0x80},
527c492ec9aSShawn Tu 		{0x4048, 0x00},
528c492ec9aSShawn Tu 		{0x4049, 0x80},
529c492ec9aSShawn Tu 		{0x4041, 0x03},
530c492ec9aSShawn Tu 		{0x404c, 0x20},
531c492ec9aSShawn Tu 		{0x404d, 0x00},
532c492ec9aSShawn Tu 		{0x404e, 0x20},
533c492ec9aSShawn Tu 		{0x4203, 0x80},
534c492ec9aSShawn Tu 		{0x4307, 0x30},
535c492ec9aSShawn Tu 		{0x4317, 0x00},
536c492ec9aSShawn Tu 		{0x4503, 0x08},
537c492ec9aSShawn Tu 		{0x4601, 0x80},
538c492ec9aSShawn Tu 		{0x4800, 0x44},
539c492ec9aSShawn Tu 		{0x4816, 0x53},
540c492ec9aSShawn Tu 		{0x481b, 0x58},
541c492ec9aSShawn Tu 		{0x481f, 0x27},
542c492ec9aSShawn Tu 		{0x4837, 0x16},
543c492ec9aSShawn Tu 		{0x483c, 0x0f},
544c492ec9aSShawn Tu 		{0x484b, 0x05},
545c492ec9aSShawn Tu 		{0x5000, 0x57},
546c492ec9aSShawn Tu 		{0x5001, 0x0a},
547ba43392eSBingbu Cao 		{0x5004, 0x06},
548c492ec9aSShawn Tu 		{0x502e, 0x03},
549c492ec9aSShawn Tu 		{0x5030, 0x41},
550c492ec9aSShawn Tu 		{0x5795, 0x00},
551c492ec9aSShawn Tu 		{0x5796, 0x10},
552c492ec9aSShawn Tu 		{0x5797, 0x10},
553c492ec9aSShawn Tu 		{0x5798, 0x73},
554c492ec9aSShawn Tu 		{0x5799, 0x73},
555c492ec9aSShawn Tu 		{0x579a, 0x00},
556c492ec9aSShawn Tu 		{0x579b, 0x28},
557c492ec9aSShawn Tu 		{0x579c, 0x00},
558c492ec9aSShawn Tu 		{0x579d, 0x16},
559c492ec9aSShawn Tu 		{0x579e, 0x06},
560c492ec9aSShawn Tu 		{0x579f, 0x20},
561c492ec9aSShawn Tu 		{0x57a0, 0x04},
562c492ec9aSShawn Tu 		{0x57a1, 0xa0},
563c492ec9aSShawn Tu 		{0x5780, 0x14},
564c492ec9aSShawn Tu 		{0x5781, 0x0f},
565c492ec9aSShawn Tu 		{0x5782, 0x44},
566c492ec9aSShawn Tu 		{0x5783, 0x02},
567c492ec9aSShawn Tu 		{0x5784, 0x01},
568c492ec9aSShawn Tu 		{0x5785, 0x01},
569c492ec9aSShawn Tu 		{0x5786, 0x00},
570c492ec9aSShawn Tu 		{0x5787, 0x04},
571c492ec9aSShawn Tu 		{0x5788, 0x02},
572c492ec9aSShawn Tu 		{0x5789, 0x0f},
573c492ec9aSShawn Tu 		{0x578a, 0xfd},
574c492ec9aSShawn Tu 		{0x578b, 0xf5},
575c492ec9aSShawn Tu 		{0x578c, 0xf5},
576c492ec9aSShawn Tu 		{0x578d, 0x03},
577c492ec9aSShawn Tu 		{0x578e, 0x08},
578c492ec9aSShawn Tu 		{0x578f, 0x0c},
579c492ec9aSShawn Tu 		{0x5790, 0x08},
580c492ec9aSShawn Tu 		{0x5791, 0x04},
581c492ec9aSShawn Tu 		{0x5792, 0x00},
582c492ec9aSShawn Tu 		{0x5793, 0x52},
583c492ec9aSShawn Tu 		{0x5794, 0xa3},
584c492ec9aSShawn Tu 		{0x59f8, 0x3d},
585c492ec9aSShawn Tu 		{0x5a08, 0x02},
586c492ec9aSShawn Tu 		{0x5b00, 0x02},
587c492ec9aSShawn Tu 		{0x5b01, 0x10},
588c492ec9aSShawn Tu 		{0x5b02, 0x03},
589c492ec9aSShawn Tu 		{0x5b03, 0xcf},
590c492ec9aSShawn Tu 		{0x5b05, 0x6c},
591c492ec9aSShawn Tu 		{0x5e00, 0x00}
592c492ec9aSShawn Tu };
593c492ec9aSShawn Tu 
594c492ec9aSShawn Tu static const struct ov8856_reg lane_4_mode_3280x2464[] = {
595c492ec9aSShawn Tu 	/* 3280x2464 resolution */
596879347f0SBen Kao 		{0x3000, 0x20},
597879347f0SBen Kao 		{0x3003, 0x08},
598879347f0SBen Kao 		{0x300e, 0x20},
599879347f0SBen Kao 		{0x3010, 0x00},
600879347f0SBen Kao 		{0x3015, 0x84},
601879347f0SBen Kao 		{0x3018, 0x72},
602879347f0SBen Kao 		{0x3021, 0x23},
603879347f0SBen Kao 		{0x3033, 0x24},
604879347f0SBen Kao 		{0x3500, 0x00},
605879347f0SBen Kao 		{0x3501, 0x9a},
606879347f0SBen Kao 		{0x3502, 0x20},
607879347f0SBen Kao 		{0x3503, 0x08},
608879347f0SBen Kao 		{0x3505, 0x83},
609879347f0SBen Kao 		{0x3508, 0x01},
610879347f0SBen Kao 		{0x3509, 0x80},
611879347f0SBen Kao 		{0x350c, 0x00},
612879347f0SBen Kao 		{0x350d, 0x80},
613879347f0SBen Kao 		{0x350e, 0x04},
614879347f0SBen Kao 		{0x350f, 0x00},
615879347f0SBen Kao 		{0x3510, 0x00},
616879347f0SBen Kao 		{0x3511, 0x02},
617879347f0SBen Kao 		{0x3512, 0x00},
618879347f0SBen Kao 		{0x3600, 0x72},
619879347f0SBen Kao 		{0x3601, 0x40},
620879347f0SBen Kao 		{0x3602, 0x30},
621879347f0SBen Kao 		{0x3610, 0xc5},
622879347f0SBen Kao 		{0x3611, 0x58},
623879347f0SBen Kao 		{0x3612, 0x5c},
624879347f0SBen Kao 		{0x3613, 0xca},
625879347f0SBen Kao 		{0x3614, 0x20},
626879347f0SBen Kao 		{0x3628, 0xff},
627879347f0SBen Kao 		{0x3629, 0xff},
628879347f0SBen Kao 		{0x362a, 0xff},
629879347f0SBen Kao 		{0x3633, 0x10},
630879347f0SBen Kao 		{0x3634, 0x10},
631879347f0SBen Kao 		{0x3635, 0x10},
632879347f0SBen Kao 		{0x3636, 0x10},
633879347f0SBen Kao 		{0x3663, 0x08},
634879347f0SBen Kao 		{0x3669, 0x34},
635879347f0SBen Kao 		{0x366e, 0x10},
636879347f0SBen Kao 		{0x3706, 0x86},
637879347f0SBen Kao 		{0x370b, 0x7e},
638879347f0SBen Kao 		{0x3714, 0x23},
639879347f0SBen Kao 		{0x3730, 0x12},
640879347f0SBen Kao 		{0x3733, 0x10},
641879347f0SBen Kao 		{0x3764, 0x00},
642879347f0SBen Kao 		{0x3765, 0x00},
643879347f0SBen Kao 		{0x3769, 0x62},
644879347f0SBen Kao 		{0x376a, 0x2a},
645879347f0SBen Kao 		{0x376b, 0x30},
646879347f0SBen Kao 		{0x3780, 0x00},
647879347f0SBen Kao 		{0x3781, 0x24},
648879347f0SBen Kao 		{0x3782, 0x00},
649879347f0SBen Kao 		{0x3783, 0x23},
650879347f0SBen Kao 		{0x3798, 0x2f},
651879347f0SBen Kao 		{0x37a1, 0x60},
652879347f0SBen Kao 		{0x37a8, 0x6a},
653879347f0SBen Kao 		{0x37ab, 0x3f},
654879347f0SBen Kao 		{0x37c2, 0x04},
655879347f0SBen Kao 		{0x37c3, 0xf1},
656879347f0SBen Kao 		{0x37c9, 0x80},
657879347f0SBen Kao 		{0x37cb, 0x16},
658879347f0SBen Kao 		{0x37cc, 0x16},
659879347f0SBen Kao 		{0x37cd, 0x16},
660879347f0SBen Kao 		{0x37ce, 0x16},
661879347f0SBen Kao 		{0x3800, 0x00},
662879347f0SBen Kao 		{0x3801, 0x00},
663879347f0SBen Kao 		{0x3802, 0x00},
664d3844b9dSShawn Tu 		{0x3803, 0x06},
665879347f0SBen Kao 		{0x3804, 0x0c},
666879347f0SBen Kao 		{0x3805, 0xdf},
667879347f0SBen Kao 		{0x3806, 0x09},
668d3844b9dSShawn Tu 		{0x3807, 0xa7},
669879347f0SBen Kao 		{0x3808, 0x0c},
670879347f0SBen Kao 		{0x3809, 0xd0},
671879347f0SBen Kao 		{0x380a, 0x09},
672879347f0SBen Kao 		{0x380b, 0xa0},
673879347f0SBen Kao 		{0x380c, 0x07},
674879347f0SBen Kao 		{0x380d, 0x88},
675879347f0SBen Kao 		{0x380e, 0x09},
676879347f0SBen Kao 		{0x380f, 0xb8},
677879347f0SBen Kao 		{0x3810, 0x00},
678879347f0SBen Kao 		{0x3811, 0x00},
679879347f0SBen Kao 		{0x3812, 0x00},
680d3844b9dSShawn Tu 		{0x3813, 0x01},
681879347f0SBen Kao 		{0x3814, 0x01},
682879347f0SBen Kao 		{0x3815, 0x01},
683879347f0SBen Kao 		{0x3816, 0x00},
684879347f0SBen Kao 		{0x3817, 0x00},
685879347f0SBen Kao 		{0x3818, 0x00},
686879347f0SBen Kao 		{0x3819, 0x10},
687879347f0SBen Kao 		{0x3820, 0x80},
688879347f0SBen Kao 		{0x3821, 0x46},
689879347f0SBen Kao 		{0x382a, 0x01},
690879347f0SBen Kao 		{0x382b, 0x01},
691879347f0SBen Kao 		{0x3830, 0x06},
692879347f0SBen Kao 		{0x3836, 0x02},
693879347f0SBen Kao 		{0x3862, 0x04},
694879347f0SBen Kao 		{0x3863, 0x08},
695879347f0SBen Kao 		{0x3cc0, 0x33},
696879347f0SBen Kao 		{0x3d85, 0x17},
697879347f0SBen Kao 		{0x3d8c, 0x73},
698879347f0SBen Kao 		{0x3d8d, 0xde},
699879347f0SBen Kao 		{0x4001, 0xe0},
700879347f0SBen Kao 		{0x4003, 0x40},
701879347f0SBen Kao 		{0x4008, 0x00},
702879347f0SBen Kao 		{0x4009, 0x0b},
703879347f0SBen Kao 		{0x400a, 0x00},
704879347f0SBen Kao 		{0x400b, 0x84},
705879347f0SBen Kao 		{0x400f, 0x80},
706879347f0SBen Kao 		{0x4010, 0xf0},
707879347f0SBen Kao 		{0x4011, 0xff},
708879347f0SBen Kao 		{0x4012, 0x02},
709879347f0SBen Kao 		{0x4013, 0x01},
710879347f0SBen Kao 		{0x4014, 0x01},
711879347f0SBen Kao 		{0x4015, 0x01},
712879347f0SBen Kao 		{0x4042, 0x00},
713879347f0SBen Kao 		{0x4043, 0x80},
714879347f0SBen Kao 		{0x4044, 0x00},
715879347f0SBen Kao 		{0x4045, 0x80},
716879347f0SBen Kao 		{0x4046, 0x00},
717879347f0SBen Kao 		{0x4047, 0x80},
718879347f0SBen Kao 		{0x4048, 0x00},
719879347f0SBen Kao 		{0x4049, 0x80},
720879347f0SBen Kao 		{0x4041, 0x03},
721879347f0SBen Kao 		{0x404c, 0x20},
722879347f0SBen Kao 		{0x404d, 0x00},
723879347f0SBen Kao 		{0x404e, 0x20},
724879347f0SBen Kao 		{0x4203, 0x80},
725879347f0SBen Kao 		{0x4307, 0x30},
726879347f0SBen Kao 		{0x4317, 0x00},
727879347f0SBen Kao 		{0x4503, 0x08},
728879347f0SBen Kao 		{0x4601, 0x80},
729879347f0SBen Kao 		{0x4800, 0x44},
730879347f0SBen Kao 		{0x4816, 0x53},
731879347f0SBen Kao 		{0x481b, 0x58},
732879347f0SBen Kao 		{0x481f, 0x27},
733879347f0SBen Kao 		{0x4837, 0x16},
734879347f0SBen Kao 		{0x483c, 0x0f},
735879347f0SBen Kao 		{0x484b, 0x05},
736879347f0SBen Kao 		{0x5000, 0x57},
737879347f0SBen Kao 		{0x5001, 0x0a},
738ba43392eSBingbu Cao 		{0x5004, 0x06},
739879347f0SBen Kao 		{0x502e, 0x03},
740879347f0SBen Kao 		{0x5030, 0x41},
741879347f0SBen Kao 		{0x5780, 0x14},
742879347f0SBen Kao 		{0x5781, 0x0f},
743879347f0SBen Kao 		{0x5782, 0x44},
744879347f0SBen Kao 		{0x5783, 0x02},
745879347f0SBen Kao 		{0x5784, 0x01},
746879347f0SBen Kao 		{0x5785, 0x01},
747879347f0SBen Kao 		{0x5786, 0x00},
748879347f0SBen Kao 		{0x5787, 0x04},
749879347f0SBen Kao 		{0x5788, 0x02},
750879347f0SBen Kao 		{0x5789, 0x0f},
751879347f0SBen Kao 		{0x578a, 0xfd},
752879347f0SBen Kao 		{0x578b, 0xf5},
753879347f0SBen Kao 		{0x578c, 0xf5},
754879347f0SBen Kao 		{0x578d, 0x03},
755879347f0SBen Kao 		{0x578e, 0x08},
756879347f0SBen Kao 		{0x578f, 0x0c},
757879347f0SBen Kao 		{0x5790, 0x08},
758879347f0SBen Kao 		{0x5791, 0x04},
759879347f0SBen Kao 		{0x5792, 0x00},
760879347f0SBen Kao 		{0x5793, 0x52},
761879347f0SBen Kao 		{0x5794, 0xa3},
762879347f0SBen Kao 		{0x5795, 0x02},
763879347f0SBen Kao 		{0x5796, 0x20},
764879347f0SBen Kao 		{0x5797, 0x20},
765879347f0SBen Kao 		{0x5798, 0xd5},
766879347f0SBen Kao 		{0x5799, 0xd5},
767879347f0SBen Kao 		{0x579a, 0x00},
768879347f0SBen Kao 		{0x579b, 0x50},
769879347f0SBen Kao 		{0x579c, 0x00},
770879347f0SBen Kao 		{0x579d, 0x2c},
771879347f0SBen Kao 		{0x579e, 0x0c},
772879347f0SBen Kao 		{0x579f, 0x40},
773879347f0SBen Kao 		{0x57a0, 0x09},
774879347f0SBen Kao 		{0x57a1, 0x40},
775879347f0SBen Kao 		{0x59f8, 0x3d},
776879347f0SBen Kao 		{0x5a08, 0x02},
777879347f0SBen Kao 		{0x5b00, 0x02},
778879347f0SBen Kao 		{0x5b01, 0x10},
779879347f0SBen Kao 		{0x5b02, 0x03},
780879347f0SBen Kao 		{0x5b03, 0xcf},
781879347f0SBen Kao 		{0x5b05, 0x6c},
782879347f0SBen Kao 		{0x5e00, 0x00}
783879347f0SBen Kao };
784879347f0SBen Kao 
785c492ec9aSShawn Tu static const struct ov8856_reg lane_4_mode_1640x1232[] = {
786c492ec9aSShawn Tu 	/* 1640x1232 resolution */
787c492ec9aSShawn Tu 		{0x3000, 0x20},
788c492ec9aSShawn Tu 		{0x3003, 0x08},
789c492ec9aSShawn Tu 		{0x300e, 0x20},
790c492ec9aSShawn Tu 		{0x3010, 0x00},
791c492ec9aSShawn Tu 		{0x3015, 0x84},
792c492ec9aSShawn Tu 		{0x3018, 0x72},
793c492ec9aSShawn Tu 		{0x3021, 0x23},
794c492ec9aSShawn Tu 		{0x3033, 0x24},
795c492ec9aSShawn Tu 		{0x3500, 0x00},
796c492ec9aSShawn Tu 		{0x3501, 0x4c},
797c492ec9aSShawn Tu 		{0x3502, 0xe0},
798c492ec9aSShawn Tu 		{0x3503, 0x08},
799c492ec9aSShawn Tu 		{0x3505, 0x83},
800c492ec9aSShawn Tu 		{0x3508, 0x01},
801c492ec9aSShawn Tu 		{0x3509, 0x80},
802c492ec9aSShawn Tu 		{0x350c, 0x00},
803c492ec9aSShawn Tu 		{0x350d, 0x80},
804c492ec9aSShawn Tu 		{0x350e, 0x04},
805c492ec9aSShawn Tu 		{0x350f, 0x00},
806c492ec9aSShawn Tu 		{0x3510, 0x00},
807c492ec9aSShawn Tu 		{0x3511, 0x02},
808c492ec9aSShawn Tu 		{0x3512, 0x00},
809c492ec9aSShawn Tu 		{0x3600, 0x72},
810c492ec9aSShawn Tu 		{0x3601, 0x40},
811c492ec9aSShawn Tu 		{0x3602, 0x30},
812c492ec9aSShawn Tu 		{0x3610, 0xc5},
813c492ec9aSShawn Tu 		{0x3611, 0x58},
814c492ec9aSShawn Tu 		{0x3612, 0x5c},
815c492ec9aSShawn Tu 		{0x3613, 0xca},
816c492ec9aSShawn Tu 		{0x3614, 0x20},
817c492ec9aSShawn Tu 		{0x3628, 0xff},
818c492ec9aSShawn Tu 		{0x3629, 0xff},
819c492ec9aSShawn Tu 		{0x362a, 0xff},
820c492ec9aSShawn Tu 		{0x3633, 0x10},
821c492ec9aSShawn Tu 		{0x3634, 0x10},
822c492ec9aSShawn Tu 		{0x3635, 0x10},
823c492ec9aSShawn Tu 		{0x3636, 0x10},
824c492ec9aSShawn Tu 		{0x3663, 0x08},
825c492ec9aSShawn Tu 		{0x3669, 0x34},
826c492ec9aSShawn Tu 		{0x366e, 0x08},
827c492ec9aSShawn Tu 		{0x3706, 0x86},
828c492ec9aSShawn Tu 		{0x370b, 0x7e},
829c492ec9aSShawn Tu 		{0x3714, 0x27},
830c492ec9aSShawn Tu 		{0x3730, 0x12},
831c492ec9aSShawn Tu 		{0x3733, 0x10},
832c492ec9aSShawn Tu 		{0x3764, 0x00},
833c492ec9aSShawn Tu 		{0x3765, 0x00},
834c492ec9aSShawn Tu 		{0x3769, 0x62},
835c492ec9aSShawn Tu 		{0x376a, 0x2a},
836c492ec9aSShawn Tu 		{0x376b, 0x30},
837c492ec9aSShawn Tu 		{0x3780, 0x00},
838c492ec9aSShawn Tu 		{0x3781, 0x24},
839c492ec9aSShawn Tu 		{0x3782, 0x00},
840c492ec9aSShawn Tu 		{0x3783, 0x23},
841c492ec9aSShawn Tu 		{0x3798, 0x2f},
842c492ec9aSShawn Tu 		{0x37a1, 0x60},
843c492ec9aSShawn Tu 		{0x37a8, 0x6a},
844c492ec9aSShawn Tu 		{0x37ab, 0x3f},
845c492ec9aSShawn Tu 		{0x37c2, 0x14},
846c492ec9aSShawn Tu 		{0x37c3, 0xf1},
847c492ec9aSShawn Tu 		{0x37c9, 0x80},
848c492ec9aSShawn Tu 		{0x37cb, 0x16},
849c492ec9aSShawn Tu 		{0x37cc, 0x16},
850c492ec9aSShawn Tu 		{0x37cd, 0x16},
851c492ec9aSShawn Tu 		{0x37ce, 0x16},
852c492ec9aSShawn Tu 		{0x3800, 0x00},
853c492ec9aSShawn Tu 		{0x3801, 0x00},
854c492ec9aSShawn Tu 		{0x3802, 0x00},
855c492ec9aSShawn Tu 		{0x3803, 0x00},
856c492ec9aSShawn Tu 		{0x3804, 0x0c},
857c492ec9aSShawn Tu 		{0x3805, 0xdf},
858c492ec9aSShawn Tu 		{0x3806, 0x09},
859c492ec9aSShawn Tu 		{0x3807, 0xaf},
860c492ec9aSShawn Tu 		{0x3808, 0x06},
861c492ec9aSShawn Tu 		{0x3809, 0x68},
862c492ec9aSShawn Tu 		{0x380a, 0x04},
863c492ec9aSShawn Tu 		{0x380b, 0xd0},
864c492ec9aSShawn Tu 		{0x380c, 0x0e},
865c492ec9aSShawn Tu 		{0x380d, 0xec},
866c492ec9aSShawn Tu 		{0x380e, 0x04},
867c492ec9aSShawn Tu 		{0x380f, 0xe8},
868c492ec9aSShawn Tu 		{0x3810, 0x00},
869c492ec9aSShawn Tu 		{0x3811, 0x04},
870c492ec9aSShawn Tu 		{0x3812, 0x00},
871c492ec9aSShawn Tu 		{0x3813, 0x05},
872c492ec9aSShawn Tu 		{0x3814, 0x03},
873c492ec9aSShawn Tu 		{0x3815, 0x01},
874c492ec9aSShawn Tu 		{0x3816, 0x00},
875c492ec9aSShawn Tu 		{0x3817, 0x00},
876c492ec9aSShawn Tu 		{0x3818, 0x00},
877c492ec9aSShawn Tu 		{0x3819, 0x10},
878c492ec9aSShawn Tu 		{0x3820, 0x90},
879c492ec9aSShawn Tu 		{0x3821, 0x67},
880c492ec9aSShawn Tu 		{0x382a, 0x03},
881c492ec9aSShawn Tu 		{0x382b, 0x01},
882c492ec9aSShawn Tu 		{0x3830, 0x06},
883c492ec9aSShawn Tu 		{0x3836, 0x02},
884c492ec9aSShawn Tu 		{0x3862, 0x04},
885c492ec9aSShawn Tu 		{0x3863, 0x08},
886c492ec9aSShawn Tu 		{0x3cc0, 0x33},
887c492ec9aSShawn Tu 		{0x3d85, 0x17},
888c492ec9aSShawn Tu 		{0x3d8c, 0x73},
889c492ec9aSShawn Tu 		{0x3d8d, 0xde},
890c492ec9aSShawn Tu 		{0x4001, 0xe0},
891c492ec9aSShawn Tu 		{0x4003, 0x40},
892c492ec9aSShawn Tu 		{0x4008, 0x00},
893c492ec9aSShawn Tu 		{0x4009, 0x05},
894c492ec9aSShawn Tu 		{0x400a, 0x00},
895c492ec9aSShawn Tu 		{0x400b, 0x84},
896c492ec9aSShawn Tu 		{0x400f, 0x80},
897c492ec9aSShawn Tu 		{0x4010, 0xf0},
898c492ec9aSShawn Tu 		{0x4011, 0xff},
899c492ec9aSShawn Tu 		{0x4012, 0x02},
900c492ec9aSShawn Tu 		{0x4013, 0x01},
901c492ec9aSShawn Tu 		{0x4014, 0x01},
902c492ec9aSShawn Tu 		{0x4015, 0x01},
903c492ec9aSShawn Tu 		{0x4042, 0x00},
904c492ec9aSShawn Tu 		{0x4043, 0x80},
905c492ec9aSShawn Tu 		{0x4044, 0x00},
906c492ec9aSShawn Tu 		{0x4045, 0x80},
907c492ec9aSShawn Tu 		{0x4046, 0x00},
908c492ec9aSShawn Tu 		{0x4047, 0x80},
909c492ec9aSShawn Tu 		{0x4048, 0x00},
910c492ec9aSShawn Tu 		{0x4049, 0x80},
911c492ec9aSShawn Tu 		{0x4041, 0x03},
912c492ec9aSShawn Tu 		{0x404c, 0x20},
913c492ec9aSShawn Tu 		{0x404d, 0x00},
914c492ec9aSShawn Tu 		{0x404e, 0x20},
915c492ec9aSShawn Tu 		{0x4203, 0x80},
916c492ec9aSShawn Tu 		{0x4307, 0x30},
917c492ec9aSShawn Tu 		{0x4317, 0x00},
918c492ec9aSShawn Tu 		{0x4503, 0x08},
919c492ec9aSShawn Tu 		{0x4601, 0x80},
920c492ec9aSShawn Tu 		{0x4800, 0x44},
921c492ec9aSShawn Tu 		{0x4816, 0x53},
922c492ec9aSShawn Tu 		{0x481b, 0x58},
923c492ec9aSShawn Tu 		{0x481f, 0x27},
924c492ec9aSShawn Tu 		{0x4837, 0x16},
925c492ec9aSShawn Tu 		{0x483c, 0x0f},
926c492ec9aSShawn Tu 		{0x484b, 0x05},
927c492ec9aSShawn Tu 		{0x5000, 0x57},
928c492ec9aSShawn Tu 		{0x5001, 0x0a},
929ba43392eSBingbu Cao 		{0x5004, 0x06},
930c492ec9aSShawn Tu 		{0x502e, 0x03},
931c492ec9aSShawn Tu 		{0x5030, 0x41},
932c492ec9aSShawn Tu 		{0x5780, 0x14},
933c492ec9aSShawn Tu 		{0x5781, 0x0f},
934c492ec9aSShawn Tu 		{0x5782, 0x44},
935c492ec9aSShawn Tu 		{0x5783, 0x02},
936c492ec9aSShawn Tu 		{0x5784, 0x01},
937c492ec9aSShawn Tu 		{0x5785, 0x01},
938c492ec9aSShawn Tu 		{0x5786, 0x00},
939c492ec9aSShawn Tu 		{0x5787, 0x04},
940c492ec9aSShawn Tu 		{0x5788, 0x02},
941c492ec9aSShawn Tu 		{0x5789, 0x0f},
942c492ec9aSShawn Tu 		{0x578a, 0xfd},
943c492ec9aSShawn Tu 		{0x578b, 0xf5},
944c492ec9aSShawn Tu 		{0x578c, 0xf5},
945c492ec9aSShawn Tu 		{0x578d, 0x03},
946c492ec9aSShawn Tu 		{0x578e, 0x08},
947c492ec9aSShawn Tu 		{0x578f, 0x0c},
948c492ec9aSShawn Tu 		{0x5790, 0x08},
949c492ec9aSShawn Tu 		{0x5791, 0x04},
950c492ec9aSShawn Tu 		{0x5792, 0x00},
951c492ec9aSShawn Tu 		{0x5793, 0x52},
952c492ec9aSShawn Tu 		{0x5794, 0xa3},
953c492ec9aSShawn Tu 		{0x5795, 0x00},
954c492ec9aSShawn Tu 		{0x5796, 0x10},
955c492ec9aSShawn Tu 		{0x5797, 0x10},
956c492ec9aSShawn Tu 		{0x5798, 0x73},
957c492ec9aSShawn Tu 		{0x5799, 0x73},
958c492ec9aSShawn Tu 		{0x579a, 0x00},
959c492ec9aSShawn Tu 		{0x579b, 0x28},
960c492ec9aSShawn Tu 		{0x579c, 0x00},
961c492ec9aSShawn Tu 		{0x579d, 0x16},
962c492ec9aSShawn Tu 		{0x579e, 0x06},
963c492ec9aSShawn Tu 		{0x579f, 0x20},
964c492ec9aSShawn Tu 		{0x57a0, 0x04},
965c492ec9aSShawn Tu 		{0x57a1, 0xa0},
966c492ec9aSShawn Tu 		{0x59f8, 0x3d},
967c492ec9aSShawn Tu 		{0x5a08, 0x02},
968c492ec9aSShawn Tu 		{0x5b00, 0x02},
969c492ec9aSShawn Tu 		{0x5b01, 0x10},
970c492ec9aSShawn Tu 		{0x5b02, 0x03},
971c492ec9aSShawn Tu 		{0x5b03, 0xcf},
972c492ec9aSShawn Tu 		{0x5b05, 0x6c},
973c492ec9aSShawn Tu 		{0x5e00, 0x00}
974c492ec9aSShawn Tu };
975c492ec9aSShawn Tu 
976c492ec9aSShawn Tu static const struct ov8856_reg lane_4_mode_3264x2448[] = {
977c492ec9aSShawn Tu 	/* 3264x2448 resolution */
97893c16fabSRobert Foss 		{0x0103, 0x01},
97993c16fabSRobert Foss 		{0x0302, 0x3c},
98093c16fabSRobert Foss 		{0x0303, 0x01},
98193c16fabSRobert Foss 		{0x031e, 0x0c},
98293c16fabSRobert Foss 		{0x3000, 0x20},
98393c16fabSRobert Foss 		{0x3003, 0x08},
98493c16fabSRobert Foss 		{0x300e, 0x20},
98593c16fabSRobert Foss 		{0x3010, 0x00},
98693c16fabSRobert Foss 		{0x3015, 0x84},
98793c16fabSRobert Foss 		{0x3018, 0x72},
98893c16fabSRobert Foss 		{0x3021, 0x23},
98993c16fabSRobert Foss 		{0x3033, 0x24},
99093c16fabSRobert Foss 		{0x3500, 0x00},
99193c16fabSRobert Foss 		{0x3501, 0x9a},
99293c16fabSRobert Foss 		{0x3502, 0x20},
99393c16fabSRobert Foss 		{0x3503, 0x08},
99493c16fabSRobert Foss 		{0x3505, 0x83},
99593c16fabSRobert Foss 		{0x3508, 0x01},
99693c16fabSRobert Foss 		{0x3509, 0x80},
99793c16fabSRobert Foss 		{0x350c, 0x00},
99893c16fabSRobert Foss 		{0x350d, 0x80},
99993c16fabSRobert Foss 		{0x350e, 0x04},
100093c16fabSRobert Foss 		{0x350f, 0x00},
100193c16fabSRobert Foss 		{0x3510, 0x00},
100293c16fabSRobert Foss 		{0x3511, 0x02},
100393c16fabSRobert Foss 		{0x3512, 0x00},
100493c16fabSRobert Foss 		{0x3600, 0x72},
100593c16fabSRobert Foss 		{0x3601, 0x40},
100693c16fabSRobert Foss 		{0x3602, 0x30},
100793c16fabSRobert Foss 		{0x3610, 0xc5},
100893c16fabSRobert Foss 		{0x3611, 0x58},
100993c16fabSRobert Foss 		{0x3612, 0x5c},
101093c16fabSRobert Foss 		{0x3613, 0xca},
101193c16fabSRobert Foss 		{0x3614, 0x60},
101293c16fabSRobert Foss 		{0x3628, 0xff},
101393c16fabSRobert Foss 		{0x3629, 0xff},
101493c16fabSRobert Foss 		{0x362a, 0xff},
101593c16fabSRobert Foss 		{0x3633, 0x10},
101693c16fabSRobert Foss 		{0x3634, 0x10},
101793c16fabSRobert Foss 		{0x3635, 0x10},
101893c16fabSRobert Foss 		{0x3636, 0x10},
101993c16fabSRobert Foss 		{0x3663, 0x08},
102093c16fabSRobert Foss 		{0x3669, 0x34},
102193c16fabSRobert Foss 		{0x366d, 0x00},
102293c16fabSRobert Foss 		{0x366e, 0x10},
102393c16fabSRobert Foss 		{0x3706, 0x86},
102493c16fabSRobert Foss 		{0x370b, 0x7e},
102593c16fabSRobert Foss 		{0x3714, 0x23},
102693c16fabSRobert Foss 		{0x3730, 0x12},
102793c16fabSRobert Foss 		{0x3733, 0x10},
102893c16fabSRobert Foss 		{0x3764, 0x00},
102993c16fabSRobert Foss 		{0x3765, 0x00},
103093c16fabSRobert Foss 		{0x3769, 0x62},
103193c16fabSRobert Foss 		{0x376a, 0x2a},
103293c16fabSRobert Foss 		{0x376b, 0x30},
103393c16fabSRobert Foss 		{0x3780, 0x00},
103493c16fabSRobert Foss 		{0x3781, 0x24},
103593c16fabSRobert Foss 		{0x3782, 0x00},
103693c16fabSRobert Foss 		{0x3783, 0x23},
103793c16fabSRobert Foss 		{0x3798, 0x2f},
103893c16fabSRobert Foss 		{0x37a1, 0x60},
103993c16fabSRobert Foss 		{0x37a8, 0x6a},
104093c16fabSRobert Foss 		{0x37ab, 0x3f},
104193c16fabSRobert Foss 		{0x37c2, 0x04},
104293c16fabSRobert Foss 		{0x37c3, 0xf1},
104393c16fabSRobert Foss 		{0x37c9, 0x80},
104493c16fabSRobert Foss 		{0x37cb, 0x16},
104593c16fabSRobert Foss 		{0x37cc, 0x16},
104693c16fabSRobert Foss 		{0x37cd, 0x16},
104793c16fabSRobert Foss 		{0x37ce, 0x16},
104893c16fabSRobert Foss 		{0x3800, 0x00},
104993c16fabSRobert Foss 		{0x3801, 0x00},
105093c16fabSRobert Foss 		{0x3802, 0x00},
105193c16fabSRobert Foss 		{0x3803, 0x0c},
105293c16fabSRobert Foss 		{0x3804, 0x0c},
105393c16fabSRobert Foss 		{0x3805, 0xdf},
105493c16fabSRobert Foss 		{0x3806, 0x09},
105593c16fabSRobert Foss 		{0x3807, 0xa3},
105693c16fabSRobert Foss 		{0x3808, 0x0c},
105793c16fabSRobert Foss 		{0x3809, 0xc0},
105893c16fabSRobert Foss 		{0x380a, 0x09},
105993c16fabSRobert Foss 		{0x380b, 0x90},
106093c16fabSRobert Foss 		{0x380c, 0x07},
106193c16fabSRobert Foss 		{0x380d, 0x8c},
106293c16fabSRobert Foss 		{0x380e, 0x09},
106393c16fabSRobert Foss 		{0x380f, 0xb2},
106493c16fabSRobert Foss 		{0x3810, 0x00},
106593c16fabSRobert Foss 		{0x3811, 0x04},
106693c16fabSRobert Foss 		{0x3812, 0x00},
1067f1363166SHsin-Yi Wang 		{0x3813, 0x02},
106893c16fabSRobert Foss 		{0x3814, 0x01},
106993c16fabSRobert Foss 		{0x3815, 0x01},
107093c16fabSRobert Foss 		{0x3816, 0x00},
107193c16fabSRobert Foss 		{0x3817, 0x00},
107293c16fabSRobert Foss 		{0x3818, 0x00},
107393c16fabSRobert Foss 		{0x3819, 0x10},
107493c16fabSRobert Foss 		{0x3820, 0x80},
107593c16fabSRobert Foss 		{0x3821, 0x46},
107693c16fabSRobert Foss 		{0x382a, 0x01},
107793c16fabSRobert Foss 		{0x382b, 0x01},
107893c16fabSRobert Foss 		{0x3830, 0x06},
107993c16fabSRobert Foss 		{0x3836, 0x02},
108093c16fabSRobert Foss 		{0x3862, 0x04},
108193c16fabSRobert Foss 		{0x3863, 0x08},
108293c16fabSRobert Foss 		{0x3cc0, 0x33},
108393c16fabSRobert Foss 		{0x3d85, 0x17},
108493c16fabSRobert Foss 		{0x3d8c, 0x73},
108593c16fabSRobert Foss 		{0x3d8d, 0xde},
108693c16fabSRobert Foss 		{0x4001, 0xe0},
108793c16fabSRobert Foss 		{0x4003, 0x40},
108893c16fabSRobert Foss 		{0x4008, 0x00},
108993c16fabSRobert Foss 		{0x4009, 0x0b},
109093c16fabSRobert Foss 		{0x400a, 0x00},
109193c16fabSRobert Foss 		{0x400b, 0x84},
109293c16fabSRobert Foss 		{0x400f, 0x80},
109393c16fabSRobert Foss 		{0x4010, 0xf0},
109493c16fabSRobert Foss 		{0x4011, 0xff},
109593c16fabSRobert Foss 		{0x4012, 0x02},
109693c16fabSRobert Foss 		{0x4013, 0x01},
109793c16fabSRobert Foss 		{0x4014, 0x01},
109893c16fabSRobert Foss 		{0x4015, 0x01},
109993c16fabSRobert Foss 		{0x4042, 0x00},
110093c16fabSRobert Foss 		{0x4043, 0x80},
110193c16fabSRobert Foss 		{0x4044, 0x00},
110293c16fabSRobert Foss 		{0x4045, 0x80},
110393c16fabSRobert Foss 		{0x4046, 0x00},
110493c16fabSRobert Foss 		{0x4047, 0x80},
110593c16fabSRobert Foss 		{0x4048, 0x00},
110693c16fabSRobert Foss 		{0x4049, 0x80},
110793c16fabSRobert Foss 		{0x4041, 0x03},
110893c16fabSRobert Foss 		{0x404c, 0x20},
110993c16fabSRobert Foss 		{0x404d, 0x00},
111093c16fabSRobert Foss 		{0x404e, 0x20},
111193c16fabSRobert Foss 		{0x4203, 0x80},
111293c16fabSRobert Foss 		{0x4307, 0x30},
111393c16fabSRobert Foss 		{0x4317, 0x00},
111493c16fabSRobert Foss 		{0x4502, 0x50},
111593c16fabSRobert Foss 		{0x4503, 0x08},
111693c16fabSRobert Foss 		{0x4601, 0x80},
111793c16fabSRobert Foss 		{0x4800, 0x44},
111893c16fabSRobert Foss 		{0x4816, 0x53},
111993c16fabSRobert Foss 		{0x481b, 0x50},
112093c16fabSRobert Foss 		{0x481f, 0x27},
112193c16fabSRobert Foss 		{0x4823, 0x3c},
112293c16fabSRobert Foss 		{0x482b, 0x00},
112393c16fabSRobert Foss 		{0x4831, 0x66},
112493c16fabSRobert Foss 		{0x4837, 0x16},
112593c16fabSRobert Foss 		{0x483c, 0x0f},
112693c16fabSRobert Foss 		{0x484b, 0x05},
112793c16fabSRobert Foss 		{0x5000, 0x77},
112893c16fabSRobert Foss 		{0x5001, 0x0a},
112993c16fabSRobert Foss 		{0x5003, 0xc8},
113093c16fabSRobert Foss 		{0x5004, 0x04},
113193c16fabSRobert Foss 		{0x5006, 0x00},
113293c16fabSRobert Foss 		{0x5007, 0x00},
113393c16fabSRobert Foss 		{0x502e, 0x03},
113493c16fabSRobert Foss 		{0x5030, 0x41},
113593c16fabSRobert Foss 		{0x5780, 0x14},
113693c16fabSRobert Foss 		{0x5781, 0x0f},
113793c16fabSRobert Foss 		{0x5782, 0x44},
113893c16fabSRobert Foss 		{0x5783, 0x02},
113993c16fabSRobert Foss 		{0x5784, 0x01},
114093c16fabSRobert Foss 		{0x5785, 0x01},
114193c16fabSRobert Foss 		{0x5786, 0x00},
114293c16fabSRobert Foss 		{0x5787, 0x04},
114393c16fabSRobert Foss 		{0x5788, 0x02},
114493c16fabSRobert Foss 		{0x5789, 0x0f},
114593c16fabSRobert Foss 		{0x578a, 0xfd},
114693c16fabSRobert Foss 		{0x578b, 0xf5},
114793c16fabSRobert Foss 		{0x578c, 0xf5},
114893c16fabSRobert Foss 		{0x578d, 0x03},
114993c16fabSRobert Foss 		{0x578e, 0x08},
115093c16fabSRobert Foss 		{0x578f, 0x0c},
115193c16fabSRobert Foss 		{0x5790, 0x08},
115293c16fabSRobert Foss 		{0x5791, 0x04},
115393c16fabSRobert Foss 		{0x5792, 0x00},
115493c16fabSRobert Foss 		{0x5793, 0x52},
115593c16fabSRobert Foss 		{0x5794, 0xa3},
115693c16fabSRobert Foss 		{0x5795, 0x02},
115793c16fabSRobert Foss 		{0x5796, 0x20},
115893c16fabSRobert Foss 		{0x5797, 0x20},
115993c16fabSRobert Foss 		{0x5798, 0xd5},
116093c16fabSRobert Foss 		{0x5799, 0xd5},
116193c16fabSRobert Foss 		{0x579a, 0x00},
116293c16fabSRobert Foss 		{0x579b, 0x50},
116393c16fabSRobert Foss 		{0x579c, 0x00},
116493c16fabSRobert Foss 		{0x579d, 0x2c},
116593c16fabSRobert Foss 		{0x579e, 0x0c},
116693c16fabSRobert Foss 		{0x579f, 0x40},
116793c16fabSRobert Foss 		{0x57a0, 0x09},
116893c16fabSRobert Foss 		{0x57a1, 0x40},
116993c16fabSRobert Foss 		{0x59f8, 0x3d},
117093c16fabSRobert Foss 		{0x5a08, 0x02},
117193c16fabSRobert Foss 		{0x5b00, 0x02},
117293c16fabSRobert Foss 		{0x5b01, 0x10},
117393c16fabSRobert Foss 		{0x5b02, 0x03},
117493c16fabSRobert Foss 		{0x5b03, 0xcf},
117593c16fabSRobert Foss 		{0x5b05, 0x6c},
117693c16fabSRobert Foss 		{0x5e00, 0x00},
117793c16fabSRobert Foss 		{0x5e10, 0xfc}
117893c16fabSRobert Foss };
117993c16fabSRobert Foss 
1180c492ec9aSShawn Tu static const struct ov8856_reg lane_4_mode_1632x1224[] = {
1181c492ec9aSShawn Tu 	/* 1632x1224 resolution */
1182e5a30635SRobert Foss 		{0x0103, 0x01},
1183e5a30635SRobert Foss 		{0x0302, 0x3c},
1184e5a30635SRobert Foss 		{0x0303, 0x01},
1185e5a30635SRobert Foss 		{0x031e, 0x0c},
1186e5a30635SRobert Foss 		{0x3000, 0x20},
1187e5a30635SRobert Foss 		{0x3003, 0x08},
1188e5a30635SRobert Foss 		{0x300e, 0x20},
1189e5a30635SRobert Foss 		{0x3010, 0x00},
1190e5a30635SRobert Foss 		{0x3015, 0x84},
1191e5a30635SRobert Foss 		{0x3018, 0x72},
1192e5a30635SRobert Foss 		{0x3021, 0x23},
1193e5a30635SRobert Foss 		{0x3033, 0x24},
1194e5a30635SRobert Foss 		{0x3500, 0x00},
1195e5a30635SRobert Foss 		{0x3501, 0x4c},
1196e5a30635SRobert Foss 		{0x3502, 0xe0},
1197e5a30635SRobert Foss 		{0x3503, 0x08},
1198e5a30635SRobert Foss 		{0x3505, 0x83},
1199e5a30635SRobert Foss 		{0x3508, 0x01},
1200e5a30635SRobert Foss 		{0x3509, 0x80},
1201e5a30635SRobert Foss 		{0x350c, 0x00},
1202e5a30635SRobert Foss 		{0x350d, 0x80},
1203e5a30635SRobert Foss 		{0x350e, 0x04},
1204e5a30635SRobert Foss 		{0x350f, 0x00},
1205e5a30635SRobert Foss 		{0x3510, 0x00},
1206e5a30635SRobert Foss 		{0x3511, 0x02},
1207e5a30635SRobert Foss 		{0x3512, 0x00},
1208e5a30635SRobert Foss 		{0x3600, 0x72},
1209e5a30635SRobert Foss 		{0x3601, 0x40},
1210e5a30635SRobert Foss 		{0x3602, 0x30},
1211e5a30635SRobert Foss 		{0x3610, 0xc5},
1212e5a30635SRobert Foss 		{0x3611, 0x58},
1213e5a30635SRobert Foss 		{0x3612, 0x5c},
1214e5a30635SRobert Foss 		{0x3613, 0xca},
1215e5a30635SRobert Foss 		{0x3614, 0x60},
1216e5a30635SRobert Foss 		{0x3628, 0xff},
1217e5a30635SRobert Foss 		{0x3629, 0xff},
1218e5a30635SRobert Foss 		{0x362a, 0xff},
1219e5a30635SRobert Foss 		{0x3633, 0x10},
1220e5a30635SRobert Foss 		{0x3634, 0x10},
1221e5a30635SRobert Foss 		{0x3635, 0x10},
1222e5a30635SRobert Foss 		{0x3636, 0x10},
1223e5a30635SRobert Foss 		{0x3663, 0x08},
1224e5a30635SRobert Foss 		{0x3669, 0x34},
1225e5a30635SRobert Foss 		{0x366d, 0x00},
1226e5a30635SRobert Foss 		{0x366e, 0x08},
1227e5a30635SRobert Foss 		{0x3706, 0x86},
1228e5a30635SRobert Foss 		{0x370b, 0x7e},
1229e5a30635SRobert Foss 		{0x3714, 0x27},
1230e5a30635SRobert Foss 		{0x3730, 0x12},
1231e5a30635SRobert Foss 		{0x3733, 0x10},
1232e5a30635SRobert Foss 		{0x3764, 0x00},
1233e5a30635SRobert Foss 		{0x3765, 0x00},
1234e5a30635SRobert Foss 		{0x3769, 0x62},
1235e5a30635SRobert Foss 		{0x376a, 0x2a},
1236e5a30635SRobert Foss 		{0x376b, 0x30},
1237e5a30635SRobert Foss 		{0x3780, 0x00},
1238e5a30635SRobert Foss 		{0x3781, 0x24},
1239e5a30635SRobert Foss 		{0x3782, 0x00},
1240e5a30635SRobert Foss 		{0x3783, 0x23},
1241e5a30635SRobert Foss 		{0x3798, 0x2f},
1242e5a30635SRobert Foss 		{0x37a1, 0x60},
1243e5a30635SRobert Foss 		{0x37a8, 0x6a},
1244e5a30635SRobert Foss 		{0x37ab, 0x3f},
1245e5a30635SRobert Foss 		{0x37c2, 0x14},
1246e5a30635SRobert Foss 		{0x37c3, 0xf1},
1247e5a30635SRobert Foss 		{0x37c9, 0x80},
1248e5a30635SRobert Foss 		{0x37cb, 0x16},
1249e5a30635SRobert Foss 		{0x37cc, 0x16},
1250e5a30635SRobert Foss 		{0x37cd, 0x16},
1251e5a30635SRobert Foss 		{0x37ce, 0x16},
1252e5a30635SRobert Foss 		{0x3800, 0x00},
1253e5a30635SRobert Foss 		{0x3801, 0x00},
1254e5a30635SRobert Foss 		{0x3802, 0x00},
1255e5a30635SRobert Foss 		{0x3803, 0x0c},
1256e5a30635SRobert Foss 		{0x3804, 0x0c},
1257e5a30635SRobert Foss 		{0x3805, 0xdf},
1258e5a30635SRobert Foss 		{0x3806, 0x09},
1259e5a30635SRobert Foss 		{0x3807, 0xa3},
1260e5a30635SRobert Foss 		{0x3808, 0x06},
1261e5a30635SRobert Foss 		{0x3809, 0x60},
1262e5a30635SRobert Foss 		{0x380a, 0x04},
1263e5a30635SRobert Foss 		{0x380b, 0xc8},
1264e5a30635SRobert Foss 		{0x380c, 0x07},
1265e5a30635SRobert Foss 		{0x380d, 0x8c},
1266e5a30635SRobert Foss 		{0x380e, 0x09},
1267e5a30635SRobert Foss 		{0x380f, 0xb2},
1268e5a30635SRobert Foss 		{0x3810, 0x00},
1269e5a30635SRobert Foss 		{0x3811, 0x02},
1270e5a30635SRobert Foss 		{0x3812, 0x00},
1271f1363166SHsin-Yi Wang 		{0x3813, 0x02},
1272e5a30635SRobert Foss 		{0x3814, 0x03},
1273e5a30635SRobert Foss 		{0x3815, 0x01},
1274e5a30635SRobert Foss 		{0x3816, 0x00},
1275e5a30635SRobert Foss 		{0x3817, 0x00},
1276e5a30635SRobert Foss 		{0x3818, 0x00},
1277e5a30635SRobert Foss 		{0x3819, 0x10},
1278e5a30635SRobert Foss 		{0x3820, 0x80},
1279e5a30635SRobert Foss 		{0x3821, 0x47},
1280e5a30635SRobert Foss 		{0x382a, 0x03},
1281e5a30635SRobert Foss 		{0x382b, 0x01},
1282e5a30635SRobert Foss 		{0x3830, 0x06},
1283e5a30635SRobert Foss 		{0x3836, 0x02},
1284e5a30635SRobert Foss 		{0x3862, 0x04},
1285e5a30635SRobert Foss 		{0x3863, 0x08},
1286e5a30635SRobert Foss 		{0x3cc0, 0x33},
1287e5a30635SRobert Foss 		{0x3d85, 0x17},
1288e5a30635SRobert Foss 		{0x3d8c, 0x73},
1289e5a30635SRobert Foss 		{0x3d8d, 0xde},
1290e5a30635SRobert Foss 		{0x4001, 0xe0},
1291e5a30635SRobert Foss 		{0x4003, 0x40},
1292e5a30635SRobert Foss 		{0x4008, 0x00},
1293e5a30635SRobert Foss 		{0x4009, 0x05},
1294e5a30635SRobert Foss 		{0x400a, 0x00},
1295e5a30635SRobert Foss 		{0x400b, 0x84},
1296e5a30635SRobert Foss 		{0x400f, 0x80},
1297e5a30635SRobert Foss 		{0x4010, 0xf0},
1298e5a30635SRobert Foss 		{0x4011, 0xff},
1299e5a30635SRobert Foss 		{0x4012, 0x02},
1300e5a30635SRobert Foss 		{0x4013, 0x01},
1301e5a30635SRobert Foss 		{0x4014, 0x01},
1302e5a30635SRobert Foss 		{0x4015, 0x01},
1303e5a30635SRobert Foss 		{0x4042, 0x00},
1304e5a30635SRobert Foss 		{0x4043, 0x80},
1305e5a30635SRobert Foss 		{0x4044, 0x00},
1306e5a30635SRobert Foss 		{0x4045, 0x80},
1307e5a30635SRobert Foss 		{0x4046, 0x00},
1308e5a30635SRobert Foss 		{0x4047, 0x80},
1309e5a30635SRobert Foss 		{0x4048, 0x00},
1310e5a30635SRobert Foss 		{0x4049, 0x80},
1311e5a30635SRobert Foss 		{0x4041, 0x03},
1312e5a30635SRobert Foss 		{0x404c, 0x20},
1313e5a30635SRobert Foss 		{0x404d, 0x00},
1314e5a30635SRobert Foss 		{0x404e, 0x20},
1315e5a30635SRobert Foss 		{0x4203, 0x80},
1316e5a30635SRobert Foss 		{0x4307, 0x30},
1317e5a30635SRobert Foss 		{0x4317, 0x00},
1318e5a30635SRobert Foss 		{0x4502, 0x50},
1319e5a30635SRobert Foss 		{0x4503, 0x08},
1320e5a30635SRobert Foss 		{0x4601, 0x80},
1321e5a30635SRobert Foss 		{0x4800, 0x44},
1322e5a30635SRobert Foss 		{0x4816, 0x53},
1323e5a30635SRobert Foss 		{0x481b, 0x50},
1324e5a30635SRobert Foss 		{0x481f, 0x27},
1325e5a30635SRobert Foss 		{0x4823, 0x3c},
1326e5a30635SRobert Foss 		{0x482b, 0x00},
1327e5a30635SRobert Foss 		{0x4831, 0x66},
1328e5a30635SRobert Foss 		{0x4837, 0x16},
1329e5a30635SRobert Foss 		{0x483c, 0x0f},
1330e5a30635SRobert Foss 		{0x484b, 0x05},
1331e5a30635SRobert Foss 		{0x5000, 0x77},
1332e5a30635SRobert Foss 		{0x5001, 0x0a},
1333e5a30635SRobert Foss 		{0x5003, 0xc8},
1334e5a30635SRobert Foss 		{0x5004, 0x04},
1335e5a30635SRobert Foss 		{0x5006, 0x00},
1336e5a30635SRobert Foss 		{0x5007, 0x00},
1337e5a30635SRobert Foss 		{0x502e, 0x03},
1338e5a30635SRobert Foss 		{0x5030, 0x41},
1339e5a30635SRobert Foss 		{0x5795, 0x00},
1340e5a30635SRobert Foss 		{0x5796, 0x10},
1341e5a30635SRobert Foss 		{0x5797, 0x10},
1342e5a30635SRobert Foss 		{0x5798, 0x73},
1343e5a30635SRobert Foss 		{0x5799, 0x73},
1344e5a30635SRobert Foss 		{0x579a, 0x00},
1345e5a30635SRobert Foss 		{0x579b, 0x28},
1346e5a30635SRobert Foss 		{0x579c, 0x00},
1347e5a30635SRobert Foss 		{0x579d, 0x16},
1348e5a30635SRobert Foss 		{0x579e, 0x06},
1349e5a30635SRobert Foss 		{0x579f, 0x20},
1350e5a30635SRobert Foss 		{0x57a0, 0x04},
1351e5a30635SRobert Foss 		{0x57a1, 0xa0},
1352e5a30635SRobert Foss 		{0x5780, 0x14},
1353e5a30635SRobert Foss 		{0x5781, 0x0f},
1354e5a30635SRobert Foss 		{0x5782, 0x44},
1355e5a30635SRobert Foss 		{0x5783, 0x02},
1356e5a30635SRobert Foss 		{0x5784, 0x01},
1357e5a30635SRobert Foss 		{0x5785, 0x01},
1358e5a30635SRobert Foss 		{0x5786, 0x00},
1359e5a30635SRobert Foss 		{0x5787, 0x04},
1360e5a30635SRobert Foss 		{0x5788, 0x02},
1361e5a30635SRobert Foss 		{0x5789, 0x0f},
1362e5a30635SRobert Foss 		{0x578a, 0xfd},
1363e5a30635SRobert Foss 		{0x578b, 0xf5},
1364e5a30635SRobert Foss 		{0x578c, 0xf5},
1365e5a30635SRobert Foss 		{0x578d, 0x03},
1366e5a30635SRobert Foss 		{0x578e, 0x08},
1367e5a30635SRobert Foss 		{0x578f, 0x0c},
1368e5a30635SRobert Foss 		{0x5790, 0x08},
1369e5a30635SRobert Foss 		{0x5791, 0x04},
1370e5a30635SRobert Foss 		{0x5792, 0x00},
1371e5a30635SRobert Foss 		{0x5793, 0x52},
1372e5a30635SRobert Foss 		{0x5794, 0xa3},
1373e5a30635SRobert Foss 		{0x59f8, 0x3d},
1374e5a30635SRobert Foss 		{0x5a08, 0x02},
1375e5a30635SRobert Foss 		{0x5b00, 0x02},
1376e5a30635SRobert Foss 		{0x5b01, 0x10},
1377e5a30635SRobert Foss 		{0x5b02, 0x03},
1378e5a30635SRobert Foss 		{0x5b03, 0xcf},
1379e5a30635SRobert Foss 		{0x5b05, 0x6c},
1380e5a30635SRobert Foss 		{0x5e00, 0x00},
1381e5a30635SRobert Foss 		{0x5e10, 0xfc}
1382e5a30635SRobert Foss };
1383e5a30635SRobert Foss 
1384f1363166SHsin-Yi Wang static const struct ov8856_reg mipi_data_mbus_sbggr10_1x10[] = {
1385f1363166SHsin-Yi Wang 	{0x3813, 0x02},
1386f1363166SHsin-Yi Wang };
1387f1363166SHsin-Yi Wang 
1388f1363166SHsin-Yi Wang static const struct ov8856_reg mipi_data_mbus_sgrbg10_1x10[] = {
1389f1363166SHsin-Yi Wang 	{0x3813, 0x01},
1390f1363166SHsin-Yi Wang };
1391f1363166SHsin-Yi Wang 
1392f1363166SHsin-Yi Wang static const u32 ov8856_mbus_codes[] = {
1393f1363166SHsin-Yi Wang 	MEDIA_BUS_FMT_SBGGR10_1X10,
1394f1363166SHsin-Yi Wang 	MEDIA_BUS_FMT_SGRBG10_1X10
1395f1363166SHsin-Yi Wang };
1396f1363166SHsin-Yi Wang 
1397879347f0SBen Kao static const char * const ov8856_test_pattern_menu[] = {
1398879347f0SBen Kao 	"Disabled",
1399879347f0SBen Kao 	"Standard Color Bar",
1400879347f0SBen Kao 	"Top-Bottom Darker Color Bar",
1401879347f0SBen Kao 	"Right-Left Darker Color Bar",
1402879347f0SBen Kao 	"Bottom-Top Darker Color Bar"
1403879347f0SBen Kao };
1404879347f0SBen Kao 
1405f1363166SHsin-Yi Wang static const struct ov8856_reg_list bayer_offset_configs[] = {
1406f1363166SHsin-Yi Wang 	[OV8856_MEDIA_BUS_FMT_SBGGR10_1X10] = {
1407f1363166SHsin-Yi Wang 		.num_of_regs = ARRAY_SIZE(mipi_data_mbus_sbggr10_1x10),
1408f1363166SHsin-Yi Wang 		.regs = mipi_data_mbus_sbggr10_1x10,
1409f1363166SHsin-Yi Wang 	},
1410f1363166SHsin-Yi Wang 	[OV8856_MEDIA_BUS_FMT_SGRBG10_1X10] = {
1411f1363166SHsin-Yi Wang 		.num_of_regs = ARRAY_SIZE(mipi_data_mbus_sgrbg10_1x10),
1412f1363166SHsin-Yi Wang 		.regs = mipi_data_mbus_sgrbg10_1x10,
1413f1363166SHsin-Yi Wang 	}
1414f1363166SHsin-Yi Wang };
1415f1363166SHsin-Yi Wang 
1416879347f0SBen Kao struct ov8856 {
1417879347f0SBen Kao 	struct v4l2_subdev sd;
1418879347f0SBen Kao 	struct media_pad pad;
1419879347f0SBen Kao 	struct v4l2_ctrl_handler ctrl_handler;
1420879347f0SBen Kao 
14210c2c7a1eSRobert Foss 	struct clk		*xvclk;
14220c2c7a1eSRobert Foss 	struct gpio_desc	*reset_gpio;
14230c2c7a1eSRobert Foss 	struct regulator_bulk_data supplies[ARRAY_SIZE(ov8856_supply_names)];
14240c2c7a1eSRobert Foss 
1425879347f0SBen Kao 	/* V4L2 Controls */
1426879347f0SBen Kao 	struct v4l2_ctrl *link_freq;
1427879347f0SBen Kao 	struct v4l2_ctrl *pixel_rate;
1428879347f0SBen Kao 	struct v4l2_ctrl *vblank;
1429879347f0SBen Kao 	struct v4l2_ctrl *hblank;
1430879347f0SBen Kao 	struct v4l2_ctrl *exposure;
1431879347f0SBen Kao 
1432879347f0SBen Kao 	/* Current mode */
1433879347f0SBen Kao 	const struct ov8856_mode *cur_mode;
1434879347f0SBen Kao 
1435f1363166SHsin-Yi Wang 	/* Application specified mbus format */
1436f1363166SHsin-Yi Wang 	u32 cur_mbus_index;
1437f1363166SHsin-Yi Wang 
1438879347f0SBen Kao 	/* To serialize asynchronus callbacks */
1439879347f0SBen Kao 	struct mutex mutex;
1440879347f0SBen Kao 
1441879347f0SBen Kao 	/* Streaming on/off */
1442879347f0SBen Kao 	bool streaming;
1443c492ec9aSShawn Tu 
1444c492ec9aSShawn Tu 	/* lanes index */
1445c492ec9aSShawn Tu 	u8 nlanes;
1446c492ec9aSShawn Tu 
1447c492ec9aSShawn Tu 	const struct ov8856_lane_cfg *priv_lane;
1448c492ec9aSShawn Tu 	u8 modes_size;
14490e014f1aSBingbu Cao 
14500e014f1aSBingbu Cao 	/* True if the device has been identified */
14510e014f1aSBingbu Cao 	bool identified;
1452879347f0SBen Kao };
1453879347f0SBen Kao 
1454c492ec9aSShawn Tu struct ov8856_lane_cfg {
1455c492ec9aSShawn Tu 	const s64 link_freq_menu_items[2];
1456c492ec9aSShawn Tu 	const struct ov8856_link_freq_config link_freq_configs[2];
1457c492ec9aSShawn Tu 	const struct ov8856_mode supported_modes[4];
1458c492ec9aSShawn Tu };
1459c492ec9aSShawn Tu 
1460c492ec9aSShawn Tu static const struct ov8856_lane_cfg lane_cfg_2 = {
1461879347f0SBen Kao 	{
1462c492ec9aSShawn Tu 		720000000,
1463c492ec9aSShawn Tu 		360000000,
1464c492ec9aSShawn Tu 	},
1465c492ec9aSShawn Tu 	{{
1466c492ec9aSShawn Tu 		.reg_list = {
1467c492ec9aSShawn Tu 			.num_of_regs =
1468c492ec9aSShawn Tu 			ARRAY_SIZE(mipi_data_rate_lane_2.regs_0),
1469c492ec9aSShawn Tu 			.regs = mipi_data_rate_lane_2.regs_0,
1470c492ec9aSShawn Tu 		}
1471c492ec9aSShawn Tu 	},
1472c492ec9aSShawn Tu 	{
1473c492ec9aSShawn Tu 		.reg_list = {
1474c492ec9aSShawn Tu 			.num_of_regs =
1475c492ec9aSShawn Tu 			ARRAY_SIZE(mipi_data_rate_lane_2.regs_1),
1476c492ec9aSShawn Tu 			.regs = mipi_data_rate_lane_2.regs_1,
1477c492ec9aSShawn Tu 		}
1478c492ec9aSShawn Tu 	}},
1479c492ec9aSShawn Tu 	{{
1480c492ec9aSShawn Tu 		.width = 3280,
1481c492ec9aSShawn Tu 		.height = 2464,
1482c492ec9aSShawn Tu 		.hts = 1928,
1483c492ec9aSShawn Tu 		.vts_def = 2488,
1484c492ec9aSShawn Tu 		.vts_min = 2488,
1485c492ec9aSShawn Tu 		.reg_list = {
1486c492ec9aSShawn Tu 			.num_of_regs =
1487c492ec9aSShawn Tu 			ARRAY_SIZE(lane_2_mode_3280x2464),
1488c492ec9aSShawn Tu 			.regs = lane_2_mode_3280x2464,
1489c492ec9aSShawn Tu 		},
1490c492ec9aSShawn Tu 		.link_freq_index = 0,
1491c492ec9aSShawn Tu 		.data_lanes = 2,
1492f1363166SHsin-Yi Wang 		.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
1493c492ec9aSShawn Tu 	},
1494c492ec9aSShawn Tu 	{
1495c492ec9aSShawn Tu 		.width = 1640,
1496c492ec9aSShawn Tu 		.height = 1232,
1497c492ec9aSShawn Tu 		.hts = 3168,
1498c492ec9aSShawn Tu 		.vts_def = 1514,
1499c492ec9aSShawn Tu 		.vts_min = 1514,
1500c492ec9aSShawn Tu 		.reg_list = {
1501c492ec9aSShawn Tu 			.num_of_regs =
1502c492ec9aSShawn Tu 			ARRAY_SIZE(lane_2_mode_1640x1232),
1503c492ec9aSShawn Tu 			.regs = lane_2_mode_1640x1232,
1504c492ec9aSShawn Tu 		},
1505c492ec9aSShawn Tu 		.link_freq_index = 1,
1506c492ec9aSShawn Tu 		.data_lanes = 2,
1507f1363166SHsin-Yi Wang 		.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
1508c492ec9aSShawn Tu 	}}
1509c492ec9aSShawn Tu };
1510c492ec9aSShawn Tu 
1511c492ec9aSShawn Tu static const struct ov8856_lane_cfg lane_cfg_4 = {
1512c492ec9aSShawn Tu 		{
1513c492ec9aSShawn Tu 			360000000,
1514c492ec9aSShawn Tu 			180000000,
1515c492ec9aSShawn Tu 		},
1516c492ec9aSShawn Tu 		{{
1517c492ec9aSShawn Tu 			.reg_list = {
1518c492ec9aSShawn Tu 				.num_of_regs =
1519c492ec9aSShawn Tu 				 ARRAY_SIZE(mipi_data_rate_lane_4.regs_0),
1520c492ec9aSShawn Tu 				.regs = mipi_data_rate_lane_4.regs_0,
1521c492ec9aSShawn Tu 			}
1522c492ec9aSShawn Tu 		},
1523c492ec9aSShawn Tu 		{
1524c492ec9aSShawn Tu 			.reg_list = {
1525c492ec9aSShawn Tu 				.num_of_regs =
1526c492ec9aSShawn Tu 				 ARRAY_SIZE(mipi_data_rate_lane_4.regs_1),
1527c492ec9aSShawn Tu 				.regs = mipi_data_rate_lane_4.regs_1,
1528c492ec9aSShawn Tu 			}
1529c492ec9aSShawn Tu 		}},
1530c492ec9aSShawn Tu 		{{
1531c492ec9aSShawn Tu 			.width = 3280,
1532c492ec9aSShawn Tu 			.height = 2464,
1533c492ec9aSShawn Tu 			.hts = 1928,
1534c492ec9aSShawn Tu 			.vts_def = 2488,
1535c492ec9aSShawn Tu 			.vts_min = 2488,
1536c492ec9aSShawn Tu 			.reg_list = {
1537c492ec9aSShawn Tu 				.num_of_regs =
1538c492ec9aSShawn Tu 				 ARRAY_SIZE(lane_4_mode_3280x2464),
1539c492ec9aSShawn Tu 				.regs = lane_4_mode_3280x2464,
1540c492ec9aSShawn Tu 			},
1541c492ec9aSShawn Tu 			.link_freq_index = 0,
1542c492ec9aSShawn Tu 			.data_lanes = 4,
1543f1363166SHsin-Yi Wang 			.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
1544c492ec9aSShawn Tu 		},
1545c492ec9aSShawn Tu 		{
1546c492ec9aSShawn Tu 			.width = 1640,
1547c492ec9aSShawn Tu 			.height = 1232,
1548c492ec9aSShawn Tu 			.hts = 3820,
1549c492ec9aSShawn Tu 			.vts_def = 1256,
1550c492ec9aSShawn Tu 			.vts_min = 1256,
1551c492ec9aSShawn Tu 			.reg_list = {
1552c492ec9aSShawn Tu 				.num_of_regs =
1553c492ec9aSShawn Tu 				 ARRAY_SIZE(lane_4_mode_1640x1232),
1554c492ec9aSShawn Tu 				.regs = lane_4_mode_1640x1232,
1555c492ec9aSShawn Tu 			},
1556c492ec9aSShawn Tu 			.link_freq_index = 1,
1557c492ec9aSShawn Tu 			.data_lanes = 4,
1558f1363166SHsin-Yi Wang 			.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
1559c492ec9aSShawn Tu 		},
1560c492ec9aSShawn Tu 		{
1561c492ec9aSShawn Tu 			.width = 3264,
1562c492ec9aSShawn Tu 			.height = 2448,
1563c492ec9aSShawn Tu 			.hts = 1932,
1564c492ec9aSShawn Tu 			.vts_def = 2482,
1565c492ec9aSShawn Tu 			.vts_min = 2482,
1566c492ec9aSShawn Tu 			.reg_list = {
1567c492ec9aSShawn Tu 				.num_of_regs =
1568c492ec9aSShawn Tu 				 ARRAY_SIZE(lane_4_mode_3264x2448),
1569c492ec9aSShawn Tu 				.regs = lane_4_mode_3264x2448,
1570c492ec9aSShawn Tu 			},
1571c492ec9aSShawn Tu 			.link_freq_index = 0,
1572c492ec9aSShawn Tu 			.data_lanes = 4,
1573f1363166SHsin-Yi Wang 			.default_mbus_index = OV8856_MEDIA_BUS_FMT_SBGGR10_1X10,
1574c492ec9aSShawn Tu 		},
1575c492ec9aSShawn Tu 		{
1576c492ec9aSShawn Tu 			.width = 1632,
1577c492ec9aSShawn Tu 			.height = 1224,
1578c492ec9aSShawn Tu 			.hts = 1932,
1579c492ec9aSShawn Tu 			.vts_def = 2482,
1580c492ec9aSShawn Tu 			.vts_min = 2482,
1581c492ec9aSShawn Tu 			.reg_list = {
1582c492ec9aSShawn Tu 				.num_of_regs =
1583c492ec9aSShawn Tu 				 ARRAY_SIZE(lane_4_mode_1632x1224),
1584c492ec9aSShawn Tu 				.regs = lane_4_mode_1632x1224,
1585c492ec9aSShawn Tu 			},
1586c492ec9aSShawn Tu 			.link_freq_index = 1,
1587c492ec9aSShawn Tu 			.data_lanes = 4,
1588f1363166SHsin-Yi Wang 			.default_mbus_index = OV8856_MEDIA_BUS_FMT_SBGGR10_1X10,
1589c492ec9aSShawn Tu 		}}
1590c492ec9aSShawn Tu };
1591c492ec9aSShawn Tu 
ov8856_modes_num(const struct ov8856 * ov8856)1592c492ec9aSShawn Tu static unsigned int ov8856_modes_num(const struct ov8856 *ov8856)
1593c492ec9aSShawn Tu {
1594c492ec9aSShawn Tu 	unsigned int i, count = 0;
1595c492ec9aSShawn Tu 
1596c492ec9aSShawn Tu 	for (i = 0; i < ARRAY_SIZE(ov8856->priv_lane->supported_modes); i++) {
1597c492ec9aSShawn Tu 		if (ov8856->priv_lane->supported_modes[i].width == 0)
1598c492ec9aSShawn Tu 			break;
1599c492ec9aSShawn Tu 		count++;
1600c492ec9aSShawn Tu 	}
1601c492ec9aSShawn Tu 
1602c492ec9aSShawn Tu 	return count;
1603c492ec9aSShawn Tu }
1604c492ec9aSShawn Tu 
to_rate(const s64 * link_freq_menu_items,u32 f_index,u8 nlanes)1605c492ec9aSShawn Tu static u64 to_rate(const s64 *link_freq_menu_items,
1606c492ec9aSShawn Tu 		   u32 f_index, u8 nlanes)
1607c492ec9aSShawn Tu {
1608c492ec9aSShawn Tu 	u64 pixel_rate = link_freq_menu_items[f_index] * 2 * nlanes;
1609879347f0SBen Kao 
1610879347f0SBen Kao 	do_div(pixel_rate, OV8856_RGB_DEPTH);
1611879347f0SBen Kao 
1612879347f0SBen Kao 	return pixel_rate;
1613879347f0SBen Kao }
1614879347f0SBen Kao 
to_pixels_per_line(const s64 * link_freq_menu_items,u32 hts,u32 f_index,u8 nlanes)1615c492ec9aSShawn Tu static u64 to_pixels_per_line(const s64 *link_freq_menu_items, u32 hts,
1616c492ec9aSShawn Tu 			      u32 f_index, u8 nlanes)
1617879347f0SBen Kao {
1618c492ec9aSShawn Tu 	u64 ppl = hts * to_rate(link_freq_menu_items, f_index, nlanes);
1619879347f0SBen Kao 
1620879347f0SBen Kao 	do_div(ppl, OV8856_SCLK);
1621879347f0SBen Kao 
1622879347f0SBen Kao 	return ppl;
1623879347f0SBen Kao }
1624879347f0SBen Kao 
ov8856_read_reg(struct ov8856 * ov8856,u16 reg,u16 len,u32 * val)1625879347f0SBen Kao static int ov8856_read_reg(struct ov8856 *ov8856, u16 reg, u16 len, u32 *val)
1626879347f0SBen Kao {
1627879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
1628879347f0SBen Kao 	struct i2c_msg msgs[2];
1629fa652734SBen Kao 	u8 addr_buf[2];
1630fa652734SBen Kao 	u8 data_buf[4] = {0};
1631879347f0SBen Kao 	int ret;
1632879347f0SBen Kao 
1633879347f0SBen Kao 	if (len > 4)
1634879347f0SBen Kao 		return -EINVAL;
1635879347f0SBen Kao 
1636fa652734SBen Kao 	put_unaligned_be16(reg, addr_buf);
1637879347f0SBen Kao 	msgs[0].addr = client->addr;
1638879347f0SBen Kao 	msgs[0].flags = 0;
1639fa652734SBen Kao 	msgs[0].len = sizeof(addr_buf);
1640879347f0SBen Kao 	msgs[0].buf = addr_buf;
1641879347f0SBen Kao 	msgs[1].addr = client->addr;
1642879347f0SBen Kao 	msgs[1].flags = I2C_M_RD;
1643879347f0SBen Kao 	msgs[1].len = len;
1644879347f0SBen Kao 	msgs[1].buf = &data_buf[4 - len];
1645879347f0SBen Kao 
1646879347f0SBen Kao 	ret = i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs));
1647879347f0SBen Kao 	if (ret != ARRAY_SIZE(msgs))
1648879347f0SBen Kao 		return -EIO;
1649879347f0SBen Kao 
1650879347f0SBen Kao 	*val = get_unaligned_be32(data_buf);
1651879347f0SBen Kao 
1652879347f0SBen Kao 	return 0;
1653879347f0SBen Kao }
1654879347f0SBen Kao 
ov8856_write_reg(struct ov8856 * ov8856,u16 reg,u16 len,u32 val)1655879347f0SBen Kao static int ov8856_write_reg(struct ov8856 *ov8856, u16 reg, u16 len, u32 val)
1656879347f0SBen Kao {
1657879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
1658879347f0SBen Kao 	u8 buf[6];
1659879347f0SBen Kao 
1660879347f0SBen Kao 	if (len > 4)
1661879347f0SBen Kao 		return -EINVAL;
1662879347f0SBen Kao 
1663879347f0SBen Kao 	put_unaligned_be16(reg, buf);
1664879347f0SBen Kao 	put_unaligned_be32(val << 8 * (4 - len), buf + 2);
1665879347f0SBen Kao 	if (i2c_master_send(client, buf, len + 2) != len + 2)
1666879347f0SBen Kao 		return -EIO;
1667879347f0SBen Kao 
1668879347f0SBen Kao 	return 0;
1669879347f0SBen Kao }
1670879347f0SBen Kao 
ov8856_write_reg_list(struct ov8856 * ov8856,const struct ov8856_reg_list * r_list)1671879347f0SBen Kao static int ov8856_write_reg_list(struct ov8856 *ov8856,
1672879347f0SBen Kao 				 const struct ov8856_reg_list *r_list)
1673879347f0SBen Kao {
1674879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
1675879347f0SBen Kao 	unsigned int i;
1676879347f0SBen Kao 	int ret;
1677879347f0SBen Kao 
1678879347f0SBen Kao 	for (i = 0; i < r_list->num_of_regs; i++) {
1679879347f0SBen Kao 		ret = ov8856_write_reg(ov8856, r_list->regs[i].address, 1,
1680879347f0SBen Kao 				       r_list->regs[i].val);
1681879347f0SBen Kao 		if (ret) {
1682879347f0SBen Kao 			dev_err_ratelimited(&client->dev,
1683879347f0SBen Kao 				    "failed to write reg 0x%4.4x. error = %d",
1684879347f0SBen Kao 				    r_list->regs[i].address, ret);
1685879347f0SBen Kao 			return ret;
1686879347f0SBen Kao 		}
1687879347f0SBen Kao 	}
1688879347f0SBen Kao 
1689879347f0SBen Kao 	return 0;
1690879347f0SBen Kao }
1691879347f0SBen Kao 
ov8856_identify_module(struct ov8856 * ov8856)16920e014f1aSBingbu Cao static int ov8856_identify_module(struct ov8856 *ov8856)
16930e014f1aSBingbu Cao {
16940e014f1aSBingbu Cao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
16950e014f1aSBingbu Cao 	int ret;
16960e014f1aSBingbu Cao 	u32 val;
16970e014f1aSBingbu Cao 
16980e014f1aSBingbu Cao 	if (ov8856->identified)
16990e014f1aSBingbu Cao 		return 0;
17000e014f1aSBingbu Cao 
17010e014f1aSBingbu Cao 	ret = ov8856_read_reg(ov8856, OV8856_REG_CHIP_ID,
17020e014f1aSBingbu Cao 			      OV8856_REG_VALUE_24BIT, &val);
17030e014f1aSBingbu Cao 	if (ret)
17040e014f1aSBingbu Cao 		return ret;
17050e014f1aSBingbu Cao 
17060e014f1aSBingbu Cao 	if (val != OV8856_CHIP_ID) {
17070e014f1aSBingbu Cao 		dev_err(&client->dev, "chip id mismatch: %x!=%x",
17080e014f1aSBingbu Cao 			OV8856_CHIP_ID, val);
17090e014f1aSBingbu Cao 		return -ENXIO;
17100e014f1aSBingbu Cao 	}
17110e014f1aSBingbu Cao 
17120e014f1aSBingbu Cao 	ov8856->identified = true;
17130e014f1aSBingbu Cao 
17140e014f1aSBingbu Cao 	return 0;
17150e014f1aSBingbu Cao }
17160e014f1aSBingbu Cao 
ov8856_update_digital_gain(struct ov8856 * ov8856,u32 d_gain)1717879347f0SBen Kao static int ov8856_update_digital_gain(struct ov8856 *ov8856, u32 d_gain)
1718879347f0SBen Kao {
1719ba43392eSBingbu Cao 	return ov8856_write_reg(ov8856, OV8856_REG_DIGITAL_GAIN,
1720879347f0SBen Kao 				OV8856_REG_VALUE_16BIT, d_gain);
1721879347f0SBen Kao }
1722879347f0SBen Kao 
ov8856_test_pattern(struct ov8856 * ov8856,u32 pattern)1723879347f0SBen Kao static int ov8856_test_pattern(struct ov8856 *ov8856, u32 pattern)
1724879347f0SBen Kao {
1725879347f0SBen Kao 	if (pattern)
1726879347f0SBen Kao 		pattern = (pattern - 1) << OV8856_TEST_PATTERN_BAR_SHIFT |
1727879347f0SBen Kao 			  OV8856_TEST_PATTERN_ENABLE;
1728879347f0SBen Kao 
1729879347f0SBen Kao 	return ov8856_write_reg(ov8856, OV8856_REG_TEST_PATTERN,
1730879347f0SBen Kao 				OV8856_REG_VALUE_08BIT, pattern);
1731879347f0SBen Kao }
1732879347f0SBen Kao 
ov8856_set_ctrl_hflip(struct ov8856 * ov8856,u32 ctrl_val)1733c19b93a6SShawn Tu static int ov8856_set_ctrl_hflip(struct ov8856 *ov8856, u32 ctrl_val)
1734c19b93a6SShawn Tu {
1735c19b93a6SShawn Tu 	int ret;
1736c19b93a6SShawn Tu 	u32 val;
1737c19b93a6SShawn Tu 
1738c19b93a6SShawn Tu 	ret = ov8856_read_reg(ov8856, OV8856_REG_MIRROR_OPT_1,
1739c19b93a6SShawn Tu 			      OV8856_REG_VALUE_08BIT, &val);
1740c19b93a6SShawn Tu 	if (ret)
1741c19b93a6SShawn Tu 		return ret;
1742c19b93a6SShawn Tu 
1743c19b93a6SShawn Tu 	ret = ov8856_write_reg(ov8856, OV8856_REG_MIRROR_OPT_1,
1744c19b93a6SShawn Tu 			       OV8856_REG_VALUE_08BIT,
1745c19b93a6SShawn Tu 			       ctrl_val ? val & ~OV8856_REG_MIRROR_OP_2 :
1746c19b93a6SShawn Tu 			       val | OV8856_REG_MIRROR_OP_2);
1747c19b93a6SShawn Tu 
1748c19b93a6SShawn Tu 	if (ret)
1749c19b93a6SShawn Tu 		return ret;
1750c19b93a6SShawn Tu 
1751c19b93a6SShawn Tu 	ret = ov8856_read_reg(ov8856, OV8856_REG_FORMAT2,
1752c19b93a6SShawn Tu 			      OV8856_REG_VALUE_08BIT, &val);
1753c19b93a6SShawn Tu 	if (ret)
1754c19b93a6SShawn Tu 		return ret;
1755c19b93a6SShawn Tu 
1756c19b93a6SShawn Tu 	return ov8856_write_reg(ov8856, OV8856_REG_FORMAT2,
1757c19b93a6SShawn Tu 				OV8856_REG_VALUE_08BIT,
1758c19b93a6SShawn Tu 				ctrl_val ? val & ~OV8856_REG_FORMAT2_OP_1 &
1759c19b93a6SShawn Tu 				~OV8856_REG_FORMAT2_OP_2 &
1760c19b93a6SShawn Tu 				~OV8856_REG_FORMAT2_OP_3 :
1761c19b93a6SShawn Tu 				val | OV8856_REG_FORMAT2_OP_1 |
1762c19b93a6SShawn Tu 				OV8856_REG_FORMAT2_OP_2 |
1763c19b93a6SShawn Tu 				OV8856_REG_FORMAT2_OP_3);
1764c19b93a6SShawn Tu }
1765c19b93a6SShawn Tu 
ov8856_set_ctrl_vflip(struct ov8856 * ov8856,u8 ctrl_val)1766c19b93a6SShawn Tu static int ov8856_set_ctrl_vflip(struct ov8856 *ov8856, u8 ctrl_val)
1767c19b93a6SShawn Tu {
1768c19b93a6SShawn Tu 	int ret;
1769c19b93a6SShawn Tu 	u32 val;
1770c19b93a6SShawn Tu 
1771c19b93a6SShawn Tu 	ret = ov8856_read_reg(ov8856, OV8856_REG_FLIP_OPT_1,
1772c19b93a6SShawn Tu 			      OV8856_REG_VALUE_08BIT, &val);
1773c19b93a6SShawn Tu 	if (ret)
1774c19b93a6SShawn Tu 		return ret;
1775c19b93a6SShawn Tu 
1776c19b93a6SShawn Tu 	ret = ov8856_write_reg(ov8856, OV8856_REG_FLIP_OPT_1,
1777c19b93a6SShawn Tu 			       OV8856_REG_VALUE_08BIT,
1778c19b93a6SShawn Tu 			       ctrl_val ? val | OV8856_REG_FLIP_OP_1 |
1779c19b93a6SShawn Tu 			       OV8856_REG_FLIP_OP_2 :
1780c19b93a6SShawn Tu 			       val & ~OV8856_REG_FLIP_OP_1 &
1781c19b93a6SShawn Tu 			       ~OV8856_REG_FLIP_OP_2);
1782c19b93a6SShawn Tu 
1783c19b93a6SShawn Tu 	ret = ov8856_read_reg(ov8856, OV8856_REG_FLIP_OPT_2,
1784c19b93a6SShawn Tu 			      OV8856_REG_VALUE_08BIT, &val);
1785c19b93a6SShawn Tu 	if (ret)
1786c19b93a6SShawn Tu 		return ret;
1787c19b93a6SShawn Tu 
1788c19b93a6SShawn Tu 	ret = ov8856_write_reg(ov8856, OV8856_REG_FLIP_OPT_2,
1789c19b93a6SShawn Tu 			       OV8856_REG_VALUE_08BIT,
1790c19b93a6SShawn Tu 			       ctrl_val ? val | OV8856_REG_FLIP_OP_2 :
1791c19b93a6SShawn Tu 			       val & ~OV8856_REG_FLIP_OP_2);
1792c19b93a6SShawn Tu 
1793c19b93a6SShawn Tu 	ret = ov8856_read_reg(ov8856, OV8856_REG_FLIP_OPT_3,
1794c19b93a6SShawn Tu 			      OV8856_REG_VALUE_08BIT, &val);
1795c19b93a6SShawn Tu 	if (ret)
1796c19b93a6SShawn Tu 		return ret;
1797c19b93a6SShawn Tu 
1798c19b93a6SShawn Tu 	ret = ov8856_write_reg(ov8856, OV8856_REG_FLIP_OPT_3,
1799c19b93a6SShawn Tu 			       OV8856_REG_VALUE_08BIT,
1800c19b93a6SShawn Tu 			       ctrl_val ? val & ~OV8856_REG_FLIP_OP_0 &
1801c19b93a6SShawn Tu 			       ~OV8856_REG_FLIP_OP_1 :
1802c19b93a6SShawn Tu 			       val | OV8856_REG_FLIP_OP_0 |
1803c19b93a6SShawn Tu 			       OV8856_REG_FLIP_OP_1);
1804c19b93a6SShawn Tu 
1805c19b93a6SShawn Tu 	ret = ov8856_read_reg(ov8856, OV8856_REG_FORMAT1,
1806c19b93a6SShawn Tu 			      OV8856_REG_VALUE_08BIT, &val);
1807c19b93a6SShawn Tu 	if (ret)
1808c19b93a6SShawn Tu 		return ret;
1809c19b93a6SShawn Tu 
1810c19b93a6SShawn Tu 	return ov8856_write_reg(ov8856, OV8856_REG_FORMAT1,
1811c19b93a6SShawn Tu 			       OV8856_REG_VALUE_08BIT,
1812c19b93a6SShawn Tu 			       ctrl_val ? val | OV8856_REG_FORMAT1_OP_1 |
1813c19b93a6SShawn Tu 			       OV8856_REG_FORMAT1_OP_3 |
1814c19b93a6SShawn Tu 			       OV8856_REG_FORMAT1_OP_2 :
1815c19b93a6SShawn Tu 			       val & ~OV8856_REG_FORMAT1_OP_1 &
1816c19b93a6SShawn Tu 			       ~OV8856_REG_FORMAT1_OP_3 &
1817c19b93a6SShawn Tu 			       ~OV8856_REG_FORMAT1_OP_2);
1818c19b93a6SShawn Tu }
1819c19b93a6SShawn Tu 
ov8856_set_ctrl(struct v4l2_ctrl * ctrl)1820879347f0SBen Kao static int ov8856_set_ctrl(struct v4l2_ctrl *ctrl)
1821879347f0SBen Kao {
1822879347f0SBen Kao 	struct ov8856 *ov8856 = container_of(ctrl->handler,
1823879347f0SBen Kao 					     struct ov8856, ctrl_handler);
1824879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
1825879347f0SBen Kao 	s64 exposure_max;
1826879347f0SBen Kao 	int ret = 0;
1827879347f0SBen Kao 
1828879347f0SBen Kao 	/* Propagate change of current control to all related controls */
1829879347f0SBen Kao 	if (ctrl->id == V4L2_CID_VBLANK) {
1830879347f0SBen Kao 		/* Update max exposure while meeting expected vblanking */
1831879347f0SBen Kao 		exposure_max = ov8856->cur_mode->height + ctrl->val -
1832879347f0SBen Kao 			       OV8856_EXPOSURE_MAX_MARGIN;
1833879347f0SBen Kao 		__v4l2_ctrl_modify_range(ov8856->exposure,
1834879347f0SBen Kao 					 ov8856->exposure->minimum,
1835879347f0SBen Kao 					 exposure_max, ov8856->exposure->step,
1836879347f0SBen Kao 					 exposure_max);
1837879347f0SBen Kao 	}
1838879347f0SBen Kao 
1839879347f0SBen Kao 	/* V4L2 controls values will be applied only when power is already up */
1840879347f0SBen Kao 	if (!pm_runtime_get_if_in_use(&client->dev))
1841879347f0SBen Kao 		return 0;
1842879347f0SBen Kao 
1843879347f0SBen Kao 	switch (ctrl->id) {
1844879347f0SBen Kao 	case V4L2_CID_ANALOGUE_GAIN:
1845879347f0SBen Kao 		ret = ov8856_write_reg(ov8856, OV8856_REG_ANALOG_GAIN,
1846879347f0SBen Kao 				       OV8856_REG_VALUE_16BIT, ctrl->val);
1847879347f0SBen Kao 		break;
1848879347f0SBen Kao 
1849879347f0SBen Kao 	case V4L2_CID_DIGITAL_GAIN:
1850879347f0SBen Kao 		ret = ov8856_update_digital_gain(ov8856, ctrl->val);
1851879347f0SBen Kao 		break;
1852879347f0SBen Kao 
1853879347f0SBen Kao 	case V4L2_CID_EXPOSURE:
1854879347f0SBen Kao 		/* 4 least significant bits of expsoure are fractional part */
1855879347f0SBen Kao 		ret = ov8856_write_reg(ov8856, OV8856_REG_EXPOSURE,
1856879347f0SBen Kao 				       OV8856_REG_VALUE_24BIT, ctrl->val << 4);
1857879347f0SBen Kao 		break;
1858879347f0SBen Kao 
1859879347f0SBen Kao 	case V4L2_CID_VBLANK:
1860879347f0SBen Kao 		ret = ov8856_write_reg(ov8856, OV8856_REG_VTS,
1861879347f0SBen Kao 				       OV8856_REG_VALUE_16BIT,
1862879347f0SBen Kao 				       ov8856->cur_mode->height + ctrl->val);
1863879347f0SBen Kao 		break;
1864879347f0SBen Kao 
1865879347f0SBen Kao 	case V4L2_CID_TEST_PATTERN:
1866879347f0SBen Kao 		ret = ov8856_test_pattern(ov8856, ctrl->val);
1867879347f0SBen Kao 		break;
1868879347f0SBen Kao 
1869c19b93a6SShawn Tu 	case V4L2_CID_HFLIP:
1870c19b93a6SShawn Tu 		ret = ov8856_set_ctrl_hflip(ov8856, ctrl->val);
1871c19b93a6SShawn Tu 		break;
1872c19b93a6SShawn Tu 
1873c19b93a6SShawn Tu 	case V4L2_CID_VFLIP:
1874c19b93a6SShawn Tu 		ret = ov8856_set_ctrl_vflip(ov8856, ctrl->val);
1875c19b93a6SShawn Tu 		break;
1876c19b93a6SShawn Tu 
1877879347f0SBen Kao 	default:
1878879347f0SBen Kao 		ret = -EINVAL;
1879879347f0SBen Kao 		break;
1880879347f0SBen Kao 	}
1881879347f0SBen Kao 
1882879347f0SBen Kao 	pm_runtime_put(&client->dev);
1883879347f0SBen Kao 
1884879347f0SBen Kao 	return ret;
1885879347f0SBen Kao }
1886879347f0SBen Kao 
1887879347f0SBen Kao static const struct v4l2_ctrl_ops ov8856_ctrl_ops = {
1888879347f0SBen Kao 	.s_ctrl = ov8856_set_ctrl,
1889879347f0SBen Kao };
1890879347f0SBen Kao 
ov8856_init_controls(struct ov8856 * ov8856)1891879347f0SBen Kao static int ov8856_init_controls(struct ov8856 *ov8856)
1892879347f0SBen Kao {
1893879347f0SBen Kao 	struct v4l2_ctrl_handler *ctrl_hdlr;
1894879347f0SBen Kao 	s64 exposure_max, h_blank;
1895879347f0SBen Kao 	int ret;
1896879347f0SBen Kao 
1897879347f0SBen Kao 	ctrl_hdlr = &ov8856->ctrl_handler;
1898879347f0SBen Kao 	ret = v4l2_ctrl_handler_init(ctrl_hdlr, 8);
1899879347f0SBen Kao 	if (ret)
1900879347f0SBen Kao 		return ret;
1901879347f0SBen Kao 
1902879347f0SBen Kao 	ctrl_hdlr->lock = &ov8856->mutex;
1903879347f0SBen Kao 	ov8856->link_freq = v4l2_ctrl_new_int_menu(ctrl_hdlr, &ov8856_ctrl_ops,
1904879347f0SBen Kao 					   V4L2_CID_LINK_FREQ,
1905c492ec9aSShawn Tu 					   ARRAY_SIZE
1906c492ec9aSShawn Tu 					   (ov8856->priv_lane->link_freq_menu_items)
1907c492ec9aSShawn Tu 					   - 1,
1908c492ec9aSShawn Tu 					   0, ov8856->priv_lane->link_freq_menu_items);
1909879347f0SBen Kao 	if (ov8856->link_freq)
1910879347f0SBen Kao 		ov8856->link_freq->flags |= V4L2_CTRL_FLAG_READ_ONLY;
1911879347f0SBen Kao 
1912879347f0SBen Kao 	ov8856->pixel_rate = v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops,
1913879347f0SBen Kao 				       V4L2_CID_PIXEL_RATE, 0,
1914c492ec9aSShawn Tu 				       to_rate(ov8856->priv_lane->link_freq_menu_items,
1915c492ec9aSShawn Tu 					       0,
1916c492ec9aSShawn Tu 					       ov8856->cur_mode->data_lanes), 1,
1917c492ec9aSShawn Tu 				       to_rate(ov8856->priv_lane->link_freq_menu_items,
1918c492ec9aSShawn Tu 					       0,
1919c492ec9aSShawn Tu 					       ov8856->cur_mode->data_lanes));
1920879347f0SBen Kao 	ov8856->vblank = v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops,
1921879347f0SBen Kao 			  V4L2_CID_VBLANK,
1922879347f0SBen Kao 			  ov8856->cur_mode->vts_min - ov8856->cur_mode->height,
1923879347f0SBen Kao 			  OV8856_VTS_MAX - ov8856->cur_mode->height, 1,
1924c492ec9aSShawn Tu 			  ov8856->cur_mode->vts_def -
1925c492ec9aSShawn Tu 			  ov8856->cur_mode->height);
1926c492ec9aSShawn Tu 	h_blank = to_pixels_per_line(ov8856->priv_lane->link_freq_menu_items,
1927c492ec9aSShawn Tu 				     ov8856->cur_mode->hts,
1928c492ec9aSShawn Tu 				     ov8856->cur_mode->link_freq_index,
1929c492ec9aSShawn Tu 				     ov8856->cur_mode->data_lanes) -
1930c492ec9aSShawn Tu 				     ov8856->cur_mode->width;
1931879347f0SBen Kao 	ov8856->hblank = v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops,
1932879347f0SBen Kao 					   V4L2_CID_HBLANK, h_blank, h_blank, 1,
1933879347f0SBen Kao 					   h_blank);
1934879347f0SBen Kao 	if (ov8856->hblank)
1935879347f0SBen Kao 		ov8856->hblank->flags |= V4L2_CTRL_FLAG_READ_ONLY;
1936879347f0SBen Kao 
1937879347f0SBen Kao 	v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops, V4L2_CID_ANALOGUE_GAIN,
1938879347f0SBen Kao 			  OV8856_ANAL_GAIN_MIN, OV8856_ANAL_GAIN_MAX,
1939879347f0SBen Kao 			  OV8856_ANAL_GAIN_STEP, OV8856_ANAL_GAIN_MIN);
1940879347f0SBen Kao 	v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops, V4L2_CID_DIGITAL_GAIN,
1941879347f0SBen Kao 			  OV8856_DGTL_GAIN_MIN, OV8856_DGTL_GAIN_MAX,
1942879347f0SBen Kao 			  OV8856_DGTL_GAIN_STEP, OV8856_DGTL_GAIN_DEFAULT);
1943879347f0SBen Kao 	exposure_max = ov8856->cur_mode->vts_def - OV8856_EXPOSURE_MAX_MARGIN;
1944879347f0SBen Kao 	ov8856->exposure = v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops,
1945879347f0SBen Kao 					     V4L2_CID_EXPOSURE,
1946879347f0SBen Kao 					     OV8856_EXPOSURE_MIN, exposure_max,
1947879347f0SBen Kao 					     OV8856_EXPOSURE_STEP,
1948879347f0SBen Kao 					     exposure_max);
1949879347f0SBen Kao 	v4l2_ctrl_new_std_menu_items(ctrl_hdlr, &ov8856_ctrl_ops,
1950879347f0SBen Kao 				     V4L2_CID_TEST_PATTERN,
1951879347f0SBen Kao 				     ARRAY_SIZE(ov8856_test_pattern_menu) - 1,
1952879347f0SBen Kao 				     0, 0, ov8856_test_pattern_menu);
1953c19b93a6SShawn Tu 	v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops,
1954c19b93a6SShawn Tu 			  V4L2_CID_HFLIP, 0, 1, 1, 0);
1955c19b93a6SShawn Tu 	v4l2_ctrl_new_std(ctrl_hdlr, &ov8856_ctrl_ops,
1956c19b93a6SShawn Tu 			  V4L2_CID_VFLIP, 0, 1, 1, 0);
1957879347f0SBen Kao 	if (ctrl_hdlr->error)
1958879347f0SBen Kao 		return ctrl_hdlr->error;
1959879347f0SBen Kao 
1960879347f0SBen Kao 	ov8856->sd.ctrl_handler = ctrl_hdlr;
1961879347f0SBen Kao 
1962879347f0SBen Kao 	return 0;
1963879347f0SBen Kao }
1964879347f0SBen Kao 
ov8856_update_pad_format(struct ov8856 * ov8856,const struct ov8856_mode * mode,struct v4l2_mbus_framefmt * fmt)1965f1363166SHsin-Yi Wang static void ov8856_update_pad_format(struct ov8856 *ov8856,
1966f1363166SHsin-Yi Wang 				     const struct ov8856_mode *mode,
1967879347f0SBen Kao 				     struct v4l2_mbus_framefmt *fmt)
1968879347f0SBen Kao {
1969f1363166SHsin-Yi Wang 	int index;
1970f1363166SHsin-Yi Wang 
1971879347f0SBen Kao 	fmt->width = mode->width;
1972879347f0SBen Kao 	fmt->height = mode->height;
1973f1363166SHsin-Yi Wang 	for (index = 0; index < ARRAY_SIZE(ov8856_mbus_codes); ++index)
1974f1363166SHsin-Yi Wang 		if (ov8856_mbus_codes[index] == fmt->code)
1975f1363166SHsin-Yi Wang 			break;
1976f1363166SHsin-Yi Wang 	if (index == ARRAY_SIZE(ov8856_mbus_codes))
1977f1363166SHsin-Yi Wang 		index = mode->default_mbus_index;
1978f1363166SHsin-Yi Wang 	fmt->code = ov8856_mbus_codes[index];
1979f1363166SHsin-Yi Wang 	ov8856->cur_mbus_index = index;
1980879347f0SBen Kao 	fmt->field = V4L2_FIELD_NONE;
1981879347f0SBen Kao }
1982879347f0SBen Kao 
ov8856_start_streaming(struct ov8856 * ov8856)1983879347f0SBen Kao static int ov8856_start_streaming(struct ov8856 *ov8856)
1984879347f0SBen Kao {
1985879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
1986879347f0SBen Kao 	const struct ov8856_reg_list *reg_list;
1987879347f0SBen Kao 	int link_freq_index, ret;
1988879347f0SBen Kao 
19890e014f1aSBingbu Cao 	ret = ov8856_identify_module(ov8856);
19900e014f1aSBingbu Cao 	if (ret)
19910e014f1aSBingbu Cao 		return ret;
19920e014f1aSBingbu Cao 
1993879347f0SBen Kao 	link_freq_index = ov8856->cur_mode->link_freq_index;
1994c492ec9aSShawn Tu 	reg_list = &ov8856->priv_lane->link_freq_configs[link_freq_index].reg_list;
1995c492ec9aSShawn Tu 
1996879347f0SBen Kao 	ret = ov8856_write_reg_list(ov8856, reg_list);
1997879347f0SBen Kao 	if (ret) {
1998879347f0SBen Kao 		dev_err(&client->dev, "failed to set plls");
1999879347f0SBen Kao 		return ret;
2000879347f0SBen Kao 	}
2001879347f0SBen Kao 
2002879347f0SBen Kao 	reg_list = &ov8856->cur_mode->reg_list;
2003879347f0SBen Kao 	ret = ov8856_write_reg_list(ov8856, reg_list);
2004879347f0SBen Kao 	if (ret) {
2005879347f0SBen Kao 		dev_err(&client->dev, "failed to set mode");
2006879347f0SBen Kao 		return ret;
2007879347f0SBen Kao 	}
2008879347f0SBen Kao 
2009f1363166SHsin-Yi Wang 	reg_list = &bayer_offset_configs[ov8856->cur_mbus_index];
2010f1363166SHsin-Yi Wang 	ret = ov8856_write_reg_list(ov8856, reg_list);
2011f1363166SHsin-Yi Wang 	if (ret) {
2012f1363166SHsin-Yi Wang 		dev_err(&client->dev, "failed to set mbus format");
2013f1363166SHsin-Yi Wang 		return ret;
2014f1363166SHsin-Yi Wang 	}
2015f1363166SHsin-Yi Wang 
2016879347f0SBen Kao 	ret = __v4l2_ctrl_handler_setup(ov8856->sd.ctrl_handler);
2017879347f0SBen Kao 	if (ret)
2018879347f0SBen Kao 		return ret;
2019879347f0SBen Kao 
2020879347f0SBen Kao 	ret = ov8856_write_reg(ov8856, OV8856_REG_MODE_SELECT,
2021879347f0SBen Kao 			       OV8856_REG_VALUE_08BIT, OV8856_MODE_STREAMING);
2022879347f0SBen Kao 	if (ret) {
2023879347f0SBen Kao 		dev_err(&client->dev, "failed to set stream");
2024879347f0SBen Kao 		return ret;
2025879347f0SBen Kao 	}
2026879347f0SBen Kao 
2027879347f0SBen Kao 	return 0;
2028879347f0SBen Kao }
2029879347f0SBen Kao 
ov8856_stop_streaming(struct ov8856 * ov8856)2030879347f0SBen Kao static void ov8856_stop_streaming(struct ov8856 *ov8856)
2031879347f0SBen Kao {
2032879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(&ov8856->sd);
2033879347f0SBen Kao 
2034879347f0SBen Kao 	if (ov8856_write_reg(ov8856, OV8856_REG_MODE_SELECT,
2035879347f0SBen Kao 			     OV8856_REG_VALUE_08BIT, OV8856_MODE_STANDBY))
2036879347f0SBen Kao 		dev_err(&client->dev, "failed to set stream");
2037879347f0SBen Kao }
2038879347f0SBen Kao 
ov8856_set_stream(struct v4l2_subdev * sd,int enable)2039879347f0SBen Kao static int ov8856_set_stream(struct v4l2_subdev *sd, int enable)
2040879347f0SBen Kao {
2041879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2042879347f0SBen Kao 	struct i2c_client *client = v4l2_get_subdevdata(sd);
2043879347f0SBen Kao 	int ret = 0;
2044879347f0SBen Kao 
2045879347f0SBen Kao 	if (ov8856->streaming == enable)
2046879347f0SBen Kao 		return 0;
2047879347f0SBen Kao 
2048879347f0SBen Kao 	mutex_lock(&ov8856->mutex);
2049879347f0SBen Kao 	if (enable) {
2050529e78deSMauro Carvalho Chehab 		ret = pm_runtime_resume_and_get(&client->dev);
2051879347f0SBen Kao 		if (ret < 0) {
2052879347f0SBen Kao 			mutex_unlock(&ov8856->mutex);
2053879347f0SBen Kao 			return ret;
2054879347f0SBen Kao 		}
2055879347f0SBen Kao 
2056879347f0SBen Kao 		ret = ov8856_start_streaming(ov8856);
2057879347f0SBen Kao 		if (ret) {
2058879347f0SBen Kao 			enable = 0;
2059879347f0SBen Kao 			ov8856_stop_streaming(ov8856);
2060879347f0SBen Kao 			pm_runtime_put(&client->dev);
2061879347f0SBen Kao 		}
2062879347f0SBen Kao 	} else {
2063879347f0SBen Kao 		ov8856_stop_streaming(ov8856);
2064879347f0SBen Kao 		pm_runtime_put(&client->dev);
2065879347f0SBen Kao 	}
2066879347f0SBen Kao 
2067879347f0SBen Kao 	ov8856->streaming = enable;
2068879347f0SBen Kao 	mutex_unlock(&ov8856->mutex);
2069879347f0SBen Kao 
2070879347f0SBen Kao 	return ret;
2071879347f0SBen Kao }
2072879347f0SBen Kao 
ov8856_power_on(struct device * dev)207308304923SHidenori Kobayashi static int ov8856_power_on(struct device *dev)
20740c2c7a1eSRobert Foss {
207508304923SHidenori Kobayashi 	struct v4l2_subdev *sd = dev_get_drvdata(dev);
207608304923SHidenori Kobayashi 	struct ov8856 *ov8856 = to_ov8856(sd);
20770c2c7a1eSRobert Foss 	int ret;
20780c2c7a1eSRobert Foss 
207908304923SHidenori Kobayashi 	if (is_acpi_node(dev_fwnode(dev)))
20800c2c7a1eSRobert Foss 		return 0;
20810c2c7a1eSRobert Foss 
20820c2c7a1eSRobert Foss 	ret = clk_prepare_enable(ov8856->xvclk);
20830c2c7a1eSRobert Foss 	if (ret < 0) {
208408304923SHidenori Kobayashi 		dev_err(dev, "failed to enable xvclk\n");
20850c2c7a1eSRobert Foss 		return ret;
20860c2c7a1eSRobert Foss 	}
20870c2c7a1eSRobert Foss 
20880c2c7a1eSRobert Foss 	if (ov8856->reset_gpio) {
20890c2c7a1eSRobert Foss 		gpiod_set_value_cansleep(ov8856->reset_gpio, 1);
20900c2c7a1eSRobert Foss 		usleep_range(1000, 2000);
20910c2c7a1eSRobert Foss 	}
20920c2c7a1eSRobert Foss 
20930c2c7a1eSRobert Foss 	ret = regulator_bulk_enable(ARRAY_SIZE(ov8856_supply_names),
20940c2c7a1eSRobert Foss 				    ov8856->supplies);
20950c2c7a1eSRobert Foss 	if (ret < 0) {
209608304923SHidenori Kobayashi 		dev_err(dev, "failed to enable regulators\n");
20970c2c7a1eSRobert Foss 		goto disable_clk;
20980c2c7a1eSRobert Foss 	}
20990c2c7a1eSRobert Foss 
21000c2c7a1eSRobert Foss 	gpiod_set_value_cansleep(ov8856->reset_gpio, 0);
21010c2c7a1eSRobert Foss 	usleep_range(1500, 1800);
21020c2c7a1eSRobert Foss 
21030c2c7a1eSRobert Foss 	return 0;
21040c2c7a1eSRobert Foss 
21050c2c7a1eSRobert Foss disable_clk:
21060c2c7a1eSRobert Foss 	gpiod_set_value_cansleep(ov8856->reset_gpio, 1);
21070c2c7a1eSRobert Foss 	clk_disable_unprepare(ov8856->xvclk);
21080c2c7a1eSRobert Foss 
21090c2c7a1eSRobert Foss 	return ret;
21100c2c7a1eSRobert Foss }
21110c2c7a1eSRobert Foss 
ov8856_power_off(struct device * dev)211208304923SHidenori Kobayashi static int ov8856_power_off(struct device *dev)
21130c2c7a1eSRobert Foss {
211408304923SHidenori Kobayashi 	struct v4l2_subdev *sd = dev_get_drvdata(dev);
211508304923SHidenori Kobayashi 	struct ov8856 *ov8856 = to_ov8856(sd);
21160c2c7a1eSRobert Foss 
211708304923SHidenori Kobayashi 	if (is_acpi_node(dev_fwnode(dev)))
211808304923SHidenori Kobayashi 		return 0;
21190c2c7a1eSRobert Foss 
21200c2c7a1eSRobert Foss 	gpiod_set_value_cansleep(ov8856->reset_gpio, 1);
21210c2c7a1eSRobert Foss 	regulator_bulk_disable(ARRAY_SIZE(ov8856_supply_names),
21220c2c7a1eSRobert Foss 			       ov8856->supplies);
21230c2c7a1eSRobert Foss 	clk_disable_unprepare(ov8856->xvclk);
212408304923SHidenori Kobayashi 
212508304923SHidenori Kobayashi 	return 0;
21260c2c7a1eSRobert Foss }
21270c2c7a1eSRobert Foss 
ov8856_suspend(struct device * dev)2128879347f0SBen Kao static int __maybe_unused ov8856_suspend(struct device *dev)
2129879347f0SBen Kao {
21309a855b73SKrzysztof Kozlowski 	struct v4l2_subdev *sd = dev_get_drvdata(dev);
2131879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2132879347f0SBen Kao 
2133879347f0SBen Kao 	mutex_lock(&ov8856->mutex);
2134879347f0SBen Kao 	if (ov8856->streaming)
2135879347f0SBen Kao 		ov8856_stop_streaming(ov8856);
2136879347f0SBen Kao 
213708304923SHidenori Kobayashi 	ov8856_power_off(dev);
2138879347f0SBen Kao 	mutex_unlock(&ov8856->mutex);
2139879347f0SBen Kao 
2140879347f0SBen Kao 	return 0;
2141879347f0SBen Kao }
2142879347f0SBen Kao 
ov8856_resume(struct device * dev)2143879347f0SBen Kao static int __maybe_unused ov8856_resume(struct device *dev)
2144879347f0SBen Kao {
21459a855b73SKrzysztof Kozlowski 	struct v4l2_subdev *sd = dev_get_drvdata(dev);
2146879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2147879347f0SBen Kao 	int ret;
2148879347f0SBen Kao 
2149879347f0SBen Kao 	mutex_lock(&ov8856->mutex);
21500c2c7a1eSRobert Foss 
215108304923SHidenori Kobayashi 	ov8856_power_on(dev);
2152879347f0SBen Kao 	if (ov8856->streaming) {
2153879347f0SBen Kao 		ret = ov8856_start_streaming(ov8856);
2154879347f0SBen Kao 		if (ret) {
2155879347f0SBen Kao 			ov8856->streaming = false;
2156879347f0SBen Kao 			ov8856_stop_streaming(ov8856);
2157879347f0SBen Kao 			mutex_unlock(&ov8856->mutex);
2158879347f0SBen Kao 			return ret;
2159879347f0SBen Kao 		}
2160879347f0SBen Kao 	}
2161879347f0SBen Kao 
2162879347f0SBen Kao 	mutex_unlock(&ov8856->mutex);
2163879347f0SBen Kao 
2164879347f0SBen Kao 	return 0;
2165879347f0SBen Kao }
2166879347f0SBen Kao 
ov8856_set_format(struct v4l2_subdev * sd,struct v4l2_subdev_state * sd_state,struct v4l2_subdev_format * fmt)2167879347f0SBen Kao static int ov8856_set_format(struct v4l2_subdev *sd,
21680d346d2aSTomi Valkeinen 			     struct v4l2_subdev_state *sd_state,
2169879347f0SBen Kao 			     struct v4l2_subdev_format *fmt)
2170879347f0SBen Kao {
2171879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2172879347f0SBen Kao 	const struct ov8856_mode *mode;
2173879347f0SBen Kao 	s32 vblank_def, h_blank;
2174879347f0SBen Kao 
2175c492ec9aSShawn Tu 	mode = v4l2_find_nearest_size(ov8856->priv_lane->supported_modes,
2176c492ec9aSShawn Tu 				      ov8856->modes_size,
2177c492ec9aSShawn Tu 				      width, height, fmt->format.width,
2178879347f0SBen Kao 				      fmt->format.height);
2179879347f0SBen Kao 
2180879347f0SBen Kao 	mutex_lock(&ov8856->mutex);
2181f1363166SHsin-Yi Wang 	ov8856_update_pad_format(ov8856, mode, &fmt->format);
2182879347f0SBen Kao 	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) {
21830d346d2aSTomi Valkeinen 		*v4l2_subdev_get_try_format(sd, sd_state, fmt->pad) = fmt->format;
2184879347f0SBen Kao 	} else {
2185879347f0SBen Kao 		ov8856->cur_mode = mode;
2186879347f0SBen Kao 		__v4l2_ctrl_s_ctrl(ov8856->link_freq, mode->link_freq_index);
2187879347f0SBen Kao 		__v4l2_ctrl_s_ctrl_int64(ov8856->pixel_rate,
2188c492ec9aSShawn Tu 					 to_rate(ov8856->priv_lane->link_freq_menu_items,
2189c492ec9aSShawn Tu 						 mode->link_freq_index,
2190c492ec9aSShawn Tu 						 ov8856->cur_mode->data_lanes));
2191879347f0SBen Kao 
2192879347f0SBen Kao 		/* Update limits and set FPS to default */
2193879347f0SBen Kao 		vblank_def = mode->vts_def - mode->height;
2194879347f0SBen Kao 		__v4l2_ctrl_modify_range(ov8856->vblank,
2195879347f0SBen Kao 					 mode->vts_min - mode->height,
2196879347f0SBen Kao 					 OV8856_VTS_MAX - mode->height, 1,
2197879347f0SBen Kao 					 vblank_def);
2198879347f0SBen Kao 		__v4l2_ctrl_s_ctrl(ov8856->vblank, vblank_def);
2199c492ec9aSShawn Tu 		h_blank = to_pixels_per_line(ov8856->priv_lane->link_freq_menu_items,
2200c492ec9aSShawn Tu 					     mode->hts,
2201c492ec9aSShawn Tu 					     mode->link_freq_index,
2202c492ec9aSShawn Tu 					     ov8856->cur_mode->data_lanes)
2203c492ec9aSShawn Tu 					     - mode->width;
2204879347f0SBen Kao 		__v4l2_ctrl_modify_range(ov8856->hblank, h_blank, h_blank, 1,
2205879347f0SBen Kao 					 h_blank);
2206879347f0SBen Kao 	}
2207879347f0SBen Kao 
2208879347f0SBen Kao 	mutex_unlock(&ov8856->mutex);
2209879347f0SBen Kao 
2210879347f0SBen Kao 	return 0;
2211879347f0SBen Kao }
2212879347f0SBen Kao 
ov8856_get_format(struct v4l2_subdev * sd,struct v4l2_subdev_state * sd_state,struct v4l2_subdev_format * fmt)2213879347f0SBen Kao static int ov8856_get_format(struct v4l2_subdev *sd,
22140d346d2aSTomi Valkeinen 			     struct v4l2_subdev_state *sd_state,
2215879347f0SBen Kao 			     struct v4l2_subdev_format *fmt)
2216879347f0SBen Kao {
2217879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2218879347f0SBen Kao 
2219879347f0SBen Kao 	mutex_lock(&ov8856->mutex);
2220879347f0SBen Kao 	if (fmt->which == V4L2_SUBDEV_FORMAT_TRY)
22210d346d2aSTomi Valkeinen 		fmt->format = *v4l2_subdev_get_try_format(&ov8856->sd,
22220d346d2aSTomi Valkeinen 							  sd_state,
2223879347f0SBen Kao 							  fmt->pad);
2224879347f0SBen Kao 	else
2225f1363166SHsin-Yi Wang 		ov8856_update_pad_format(ov8856, ov8856->cur_mode, &fmt->format);
2226879347f0SBen Kao 
2227879347f0SBen Kao 	mutex_unlock(&ov8856->mutex);
2228879347f0SBen Kao 
2229879347f0SBen Kao 	return 0;
2230879347f0SBen Kao }
2231879347f0SBen Kao 
ov8856_enum_mbus_code(struct v4l2_subdev * sd,struct v4l2_subdev_state * sd_state,struct v4l2_subdev_mbus_code_enum * code)2232879347f0SBen Kao static int ov8856_enum_mbus_code(struct v4l2_subdev *sd,
22330d346d2aSTomi Valkeinen 				 struct v4l2_subdev_state *sd_state,
2234879347f0SBen Kao 				 struct v4l2_subdev_mbus_code_enum *code)
2235879347f0SBen Kao {
2236f1363166SHsin-Yi Wang 	if (code->index >= ARRAY_SIZE(ov8856_mbus_codes))
2237879347f0SBen Kao 		return -EINVAL;
2238879347f0SBen Kao 
2239f1363166SHsin-Yi Wang 	code->code = ov8856_mbus_codes[code->index];
2240879347f0SBen Kao 
2241879347f0SBen Kao 	return 0;
2242879347f0SBen Kao }
2243879347f0SBen Kao 
ov8856_enum_frame_size(struct v4l2_subdev * sd,struct v4l2_subdev_state * sd_state,struct v4l2_subdev_frame_size_enum * fse)2244879347f0SBen Kao static int ov8856_enum_frame_size(struct v4l2_subdev *sd,
22450d346d2aSTomi Valkeinen 				  struct v4l2_subdev_state *sd_state,
2246879347f0SBen Kao 				  struct v4l2_subdev_frame_size_enum *fse)
2247879347f0SBen Kao {
2248c492ec9aSShawn Tu 	struct ov8856 *ov8856 = to_ov8856(sd);
2249f1363166SHsin-Yi Wang 	int index;
2250c492ec9aSShawn Tu 
2251c492ec9aSShawn Tu 	if (fse->index >= ov8856->modes_size)
2252879347f0SBen Kao 		return -EINVAL;
2253879347f0SBen Kao 
2254f1363166SHsin-Yi Wang 	for (index = 0; index < ARRAY_SIZE(ov8856_mbus_codes); ++index)
2255f1363166SHsin-Yi Wang 		if (fse->code == ov8856_mbus_codes[index])
2256f1363166SHsin-Yi Wang 			break;
2257f1363166SHsin-Yi Wang 	if (index == ARRAY_SIZE(ov8856_mbus_codes))
2258879347f0SBen Kao 		return -EINVAL;
2259879347f0SBen Kao 
2260c492ec9aSShawn Tu 	fse->min_width = ov8856->priv_lane->supported_modes[fse->index].width;
2261879347f0SBen Kao 	fse->max_width = fse->min_width;
2262c492ec9aSShawn Tu 	fse->min_height = ov8856->priv_lane->supported_modes[fse->index].height;
2263879347f0SBen Kao 	fse->max_height = fse->min_height;
2264879347f0SBen Kao 
2265879347f0SBen Kao 	return 0;
2266879347f0SBen Kao }
2267879347f0SBen Kao 
ov8856_open(struct v4l2_subdev * sd,struct v4l2_subdev_fh * fh)2268879347f0SBen Kao static int ov8856_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh)
2269879347f0SBen Kao {
2270879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2271879347f0SBen Kao 
2272879347f0SBen Kao 	mutex_lock(&ov8856->mutex);
2273f1363166SHsin-Yi Wang 	ov8856_update_pad_format(ov8856, &ov8856->priv_lane->supported_modes[0],
22740d346d2aSTomi Valkeinen 				 v4l2_subdev_get_try_format(sd, fh->state, 0));
2275879347f0SBen Kao 	mutex_unlock(&ov8856->mutex);
2276879347f0SBen Kao 
2277879347f0SBen Kao 	return 0;
2278879347f0SBen Kao }
2279879347f0SBen Kao 
2280879347f0SBen Kao static const struct v4l2_subdev_video_ops ov8856_video_ops = {
2281879347f0SBen Kao 	.s_stream = ov8856_set_stream,
2282879347f0SBen Kao };
2283879347f0SBen Kao 
2284879347f0SBen Kao static const struct v4l2_subdev_pad_ops ov8856_pad_ops = {
2285879347f0SBen Kao 	.set_fmt = ov8856_set_format,
2286879347f0SBen Kao 	.get_fmt = ov8856_get_format,
2287879347f0SBen Kao 	.enum_mbus_code = ov8856_enum_mbus_code,
2288879347f0SBen Kao 	.enum_frame_size = ov8856_enum_frame_size,
2289879347f0SBen Kao };
2290879347f0SBen Kao 
2291879347f0SBen Kao static const struct v4l2_subdev_ops ov8856_subdev_ops = {
2292879347f0SBen Kao 	.video = &ov8856_video_ops,
2293879347f0SBen Kao 	.pad = &ov8856_pad_ops,
2294879347f0SBen Kao };
2295879347f0SBen Kao 
2296879347f0SBen Kao static const struct media_entity_operations ov8856_subdev_entity_ops = {
2297879347f0SBen Kao 	.link_validate = v4l2_subdev_link_validate,
2298879347f0SBen Kao };
2299879347f0SBen Kao 
2300879347f0SBen Kao static const struct v4l2_subdev_internal_ops ov8856_internal_ops = {
2301879347f0SBen Kao 	.open = ov8856_open,
2302879347f0SBen Kao };
2303879347f0SBen Kao 
2304879347f0SBen Kao 
ov8856_get_hwcfg(struct ov8856 * ov8856,struct device * dev)23050c2c7a1eSRobert Foss static int ov8856_get_hwcfg(struct ov8856 *ov8856, struct device *dev)
2306879347f0SBen Kao {
2307879347f0SBen Kao 	struct fwnode_handle *ep;
2308879347f0SBen Kao 	struct fwnode_handle *fwnode = dev_fwnode(dev);
2309879347f0SBen Kao 	struct v4l2_fwnode_endpoint bus_cfg = {
2310879347f0SBen Kao 		.bus_type = V4L2_MBUS_CSI2_DPHY
2311879347f0SBen Kao 	};
23120c2c7a1eSRobert Foss 	u32 xvclk_rate;
2313879347f0SBen Kao 	int ret;
2314879347f0SBen Kao 	unsigned int i, j;
2315879347f0SBen Kao 
2316879347f0SBen Kao 	if (!fwnode)
2317879347f0SBen Kao 		return -ENXIO;
2318879347f0SBen Kao 
23190c2c7a1eSRobert Foss 	ret = fwnode_property_read_u32(fwnode, "clock-frequency", &xvclk_rate);
2320fa4bb7d3SSakari Ailus 	if (ret)
2321fa4bb7d3SSakari Ailus 		return ret;
2322fa4bb7d3SSakari Ailus 
23230c2c7a1eSRobert Foss 	if (!is_acpi_node(fwnode)) {
23240c2c7a1eSRobert Foss 		ov8856->xvclk = devm_clk_get(dev, "xvclk");
23250c2c7a1eSRobert Foss 		if (IS_ERR(ov8856->xvclk)) {
23260c2c7a1eSRobert Foss 			dev_err(dev, "could not get xvclk clock (%pe)\n",
23270c2c7a1eSRobert Foss 				ov8856->xvclk);
23280c2c7a1eSRobert Foss 			return PTR_ERR(ov8856->xvclk);
2329879347f0SBen Kao 		}
2330879347f0SBen Kao 
23310c2c7a1eSRobert Foss 		clk_set_rate(ov8856->xvclk, xvclk_rate);
23320c2c7a1eSRobert Foss 		xvclk_rate = clk_get_rate(ov8856->xvclk);
23330c2c7a1eSRobert Foss 
23340c2c7a1eSRobert Foss 		ov8856->reset_gpio = devm_gpiod_get_optional(dev, "reset",
23350c2c7a1eSRobert Foss 							     GPIOD_OUT_LOW);
23360c2c7a1eSRobert Foss 		if (IS_ERR(ov8856->reset_gpio))
23370c2c7a1eSRobert Foss 			return PTR_ERR(ov8856->reset_gpio);
23380c2c7a1eSRobert Foss 
23390c2c7a1eSRobert Foss 		for (i = 0; i < ARRAY_SIZE(ov8856_supply_names); i++)
23400c2c7a1eSRobert Foss 			ov8856->supplies[i].supply = ov8856_supply_names[i];
23410c2c7a1eSRobert Foss 
2342d84a2e49SBingbu Cao 		ret = devm_regulator_bulk_get(dev,
2343d84a2e49SBingbu Cao 					      ARRAY_SIZE(ov8856_supply_names),
23440c2c7a1eSRobert Foss 					      ov8856->supplies);
23450c2c7a1eSRobert Foss 		if (ret)
23460c2c7a1eSRobert Foss 			return ret;
2347d84a2e49SBingbu Cao 	}
2348d84a2e49SBingbu Cao 
2349d84a2e49SBingbu Cao 	if (xvclk_rate != OV8856_XVCLK_19_2)
2350d84a2e49SBingbu Cao 		dev_warn(dev, "external clock rate %u is unsupported",
2351d84a2e49SBingbu Cao 			 xvclk_rate);
23520c2c7a1eSRobert Foss 
2353879347f0SBen Kao 	ep = fwnode_graph_get_next_endpoint(fwnode, NULL);
2354879347f0SBen Kao 	if (!ep)
2355879347f0SBen Kao 		return -ENXIO;
2356879347f0SBen Kao 
2357879347f0SBen Kao 	ret = v4l2_fwnode_endpoint_alloc_parse(ep, &bus_cfg);
2358879347f0SBen Kao 	fwnode_handle_put(ep);
2359879347f0SBen Kao 	if (ret)
2360879347f0SBen Kao 		return ret;
2361879347f0SBen Kao 
2362c492ec9aSShawn Tu 	/* Get number of data lanes */
2363c492ec9aSShawn Tu 	if (bus_cfg.bus.mipi_csi2.num_data_lanes != 2 &&
2364c492ec9aSShawn Tu 	    bus_cfg.bus.mipi_csi2.num_data_lanes != 4) {
2365879347f0SBen Kao 		dev_err(dev, "number of CSI2 data lanes %d is not supported",
2366879347f0SBen Kao 			bus_cfg.bus.mipi_csi2.num_data_lanes);
2367879347f0SBen Kao 		ret = -EINVAL;
2368879347f0SBen Kao 		goto check_hwcfg_error;
2369879347f0SBen Kao 	}
2370879347f0SBen Kao 
2371c492ec9aSShawn Tu 	dev_dbg(dev, "Using %u data lanes\n", ov8856->cur_mode->data_lanes);
2372c492ec9aSShawn Tu 
2373c492ec9aSShawn Tu 	if (bus_cfg.bus.mipi_csi2.num_data_lanes == 2)
2374c492ec9aSShawn Tu 		ov8856->priv_lane = &lane_cfg_2;
2375c492ec9aSShawn Tu 	else
2376c492ec9aSShawn Tu 		ov8856->priv_lane = &lane_cfg_4;
2377c492ec9aSShawn Tu 
2378c492ec9aSShawn Tu 	ov8856->modes_size = ov8856_modes_num(ov8856);
2379c492ec9aSShawn Tu 
2380879347f0SBen Kao 	if (!bus_cfg.nr_of_link_frequencies) {
2381879347f0SBen Kao 		dev_err(dev, "no link frequencies defined");
2382879347f0SBen Kao 		ret = -EINVAL;
2383879347f0SBen Kao 		goto check_hwcfg_error;
2384879347f0SBen Kao 	}
2385879347f0SBen Kao 
2386c492ec9aSShawn Tu 	for (i = 0; i < ARRAY_SIZE(ov8856->priv_lane->link_freq_menu_items); i++) {
2387879347f0SBen Kao 		for (j = 0; j < bus_cfg.nr_of_link_frequencies; j++) {
2388c492ec9aSShawn Tu 			if (ov8856->priv_lane->link_freq_menu_items[i] ==
2389879347f0SBen Kao 			    bus_cfg.link_frequencies[j])
2390879347f0SBen Kao 				break;
2391879347f0SBen Kao 		}
2392879347f0SBen Kao 
2393879347f0SBen Kao 		if (j == bus_cfg.nr_of_link_frequencies) {
2394879347f0SBen Kao 			dev_err(dev, "no link frequency %lld supported",
2395c492ec9aSShawn Tu 				ov8856->priv_lane->link_freq_menu_items[i]);
2396879347f0SBen Kao 			ret = -EINVAL;
2397879347f0SBen Kao 			goto check_hwcfg_error;
2398879347f0SBen Kao 		}
2399879347f0SBen Kao 	}
2400879347f0SBen Kao 
2401879347f0SBen Kao check_hwcfg_error:
2402879347f0SBen Kao 	v4l2_fwnode_endpoint_free(&bus_cfg);
2403879347f0SBen Kao 
2404879347f0SBen Kao 	return ret;
2405879347f0SBen Kao }
2406879347f0SBen Kao 
ov8856_remove(struct i2c_client * client)2407ed5c2f5fSUwe Kleine-König static void ov8856_remove(struct i2c_client *client)
2408879347f0SBen Kao {
2409879347f0SBen Kao 	struct v4l2_subdev *sd = i2c_get_clientdata(client);
2410879347f0SBen Kao 	struct ov8856 *ov8856 = to_ov8856(sd);
2411879347f0SBen Kao 
2412879347f0SBen Kao 	v4l2_async_unregister_subdev(sd);
2413879347f0SBen Kao 	media_entity_cleanup(&sd->entity);
2414879347f0SBen Kao 	v4l2_ctrl_handler_free(sd->ctrl_handler);
2415879347f0SBen Kao 	pm_runtime_disable(&client->dev);
2416879347f0SBen Kao 	mutex_destroy(&ov8856->mutex);
2417879347f0SBen Kao 
241808304923SHidenori Kobayashi 	ov8856_power_off(&client->dev);
2419879347f0SBen Kao }
2420879347f0SBen Kao 
ov8856_probe(struct i2c_client * client)2421879347f0SBen Kao static int ov8856_probe(struct i2c_client *client)
2422879347f0SBen Kao {
2423879347f0SBen Kao 	struct ov8856 *ov8856;
2424879347f0SBen Kao 	int ret;
24250e014f1aSBingbu Cao 	bool full_power;
2426879347f0SBen Kao 
2427879347f0SBen Kao 	ov8856 = devm_kzalloc(&client->dev, sizeof(*ov8856), GFP_KERNEL);
2428879347f0SBen Kao 	if (!ov8856)
2429879347f0SBen Kao 		return -ENOMEM;
2430879347f0SBen Kao 
24310c2c7a1eSRobert Foss 	ret = ov8856_get_hwcfg(ov8856, &client->dev);
24320c2c7a1eSRobert Foss 	if (ret) {
24330c2c7a1eSRobert Foss 		dev_err(&client->dev, "failed to get HW configuration: %d",
24340c2c7a1eSRobert Foss 			ret);
24350c2c7a1eSRobert Foss 		return ret;
24360c2c7a1eSRobert Foss 	}
24370c2c7a1eSRobert Foss 
2438879347f0SBen Kao 	v4l2_i2c_subdev_init(&ov8856->sd, client, &ov8856_subdev_ops);
24390c2c7a1eSRobert Foss 
24400e014f1aSBingbu Cao 	full_power = acpi_dev_state_d0(&client->dev);
24410e014f1aSBingbu Cao 	if (full_power) {
244208304923SHidenori Kobayashi 		ret = ov8856_power_on(&client->dev);
24430c2c7a1eSRobert Foss 		if (ret) {
24440c2c7a1eSRobert Foss 			dev_err(&client->dev, "failed to power on\n");
24450c2c7a1eSRobert Foss 			return ret;
24460c2c7a1eSRobert Foss 		}
24470c2c7a1eSRobert Foss 
2448879347f0SBen Kao 		ret = ov8856_identify_module(ov8856);
2449879347f0SBen Kao 		if (ret) {
2450879347f0SBen Kao 			dev_err(&client->dev, "failed to find sensor: %d", ret);
24510c2c7a1eSRobert Foss 			goto probe_power_off;
2452879347f0SBen Kao 		}
24530e014f1aSBingbu Cao 	}
2454879347f0SBen Kao 
2455879347f0SBen Kao 	mutex_init(&ov8856->mutex);
2456c492ec9aSShawn Tu 	ov8856->cur_mode = &ov8856->priv_lane->supported_modes[0];
2457f1363166SHsin-Yi Wang 	ov8856->cur_mbus_index = ov8856->cur_mode->default_mbus_index;
2458879347f0SBen Kao 	ret = ov8856_init_controls(ov8856);
2459879347f0SBen Kao 	if (ret) {
2460879347f0SBen Kao 		dev_err(&client->dev, "failed to init controls: %d", ret);
2461879347f0SBen Kao 		goto probe_error_v4l2_ctrl_handler_free;
2462879347f0SBen Kao 	}
2463879347f0SBen Kao 
2464879347f0SBen Kao 	ov8856->sd.internal_ops = &ov8856_internal_ops;
2465879347f0SBen Kao 	ov8856->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
2466879347f0SBen Kao 	ov8856->sd.entity.ops = &ov8856_subdev_entity_ops;
2467879347f0SBen Kao 	ov8856->sd.entity.function = MEDIA_ENT_F_CAM_SENSOR;
2468879347f0SBen Kao 	ov8856->pad.flags = MEDIA_PAD_FL_SOURCE;
2469879347f0SBen Kao 	ret = media_entity_pads_init(&ov8856->sd.entity, 1, &ov8856->pad);
2470879347f0SBen Kao 	if (ret) {
2471879347f0SBen Kao 		dev_err(&client->dev, "failed to init entity pads: %d", ret);
2472879347f0SBen Kao 		goto probe_error_v4l2_ctrl_handler_free;
2473879347f0SBen Kao 	}
2474879347f0SBen Kao 
247515786f7bSSakari Ailus 	ret = v4l2_async_register_subdev_sensor(&ov8856->sd);
2476879347f0SBen Kao 	if (ret < 0) {
2477879347f0SBen Kao 		dev_err(&client->dev, "failed to register V4L2 subdev: %d",
2478879347f0SBen Kao 			ret);
2479879347f0SBen Kao 		goto probe_error_media_entity_cleanup;
2480879347f0SBen Kao 	}
2481879347f0SBen Kao 
24820e014f1aSBingbu Cao 	/* Set the device's state to active if it's in D0 state. */
24830e014f1aSBingbu Cao 	if (full_power)
2484879347f0SBen Kao 		pm_runtime_set_active(&client->dev);
2485879347f0SBen Kao 	pm_runtime_enable(&client->dev);
2486879347f0SBen Kao 	pm_runtime_idle(&client->dev);
2487879347f0SBen Kao 
2488879347f0SBen Kao 	return 0;
2489879347f0SBen Kao 
2490879347f0SBen Kao probe_error_media_entity_cleanup:
2491879347f0SBen Kao 	media_entity_cleanup(&ov8856->sd.entity);
2492879347f0SBen Kao 
2493879347f0SBen Kao probe_error_v4l2_ctrl_handler_free:
2494879347f0SBen Kao 	v4l2_ctrl_handler_free(ov8856->sd.ctrl_handler);
2495879347f0SBen Kao 	mutex_destroy(&ov8856->mutex);
2496879347f0SBen Kao 
24970c2c7a1eSRobert Foss probe_power_off:
249808304923SHidenori Kobayashi 	ov8856_power_off(&client->dev);
24990c2c7a1eSRobert Foss 
2500879347f0SBen Kao 	return ret;
2501879347f0SBen Kao }
2502879347f0SBen Kao 
2503879347f0SBen Kao static const struct dev_pm_ops ov8856_pm_ops = {
2504879347f0SBen Kao 	SET_SYSTEM_SLEEP_PM_OPS(ov8856_suspend, ov8856_resume)
250508304923SHidenori Kobayashi 	SET_RUNTIME_PM_OPS(ov8856_power_off, ov8856_power_on, NULL)
2506879347f0SBen Kao };
2507879347f0SBen Kao 
2508879347f0SBen Kao #ifdef CONFIG_ACPI
2509879347f0SBen Kao static const struct acpi_device_id ov8856_acpi_ids[] = {
2510879347f0SBen Kao 	{"OVTI8856"},
2511879347f0SBen Kao 	{}
2512879347f0SBen Kao };
2513879347f0SBen Kao 
2514879347f0SBen Kao MODULE_DEVICE_TABLE(acpi, ov8856_acpi_ids);
2515879347f0SBen Kao #endif
2516879347f0SBen Kao 
25170c2c7a1eSRobert Foss static const struct of_device_id ov8856_of_match[] = {
25180c2c7a1eSRobert Foss 	{ .compatible = "ovti,ov8856" },
25190c2c7a1eSRobert Foss 	{ /* sentinel */ }
25200c2c7a1eSRobert Foss };
25210c2c7a1eSRobert Foss MODULE_DEVICE_TABLE(of, ov8856_of_match);
25220c2c7a1eSRobert Foss 
2523879347f0SBen Kao static struct i2c_driver ov8856_i2c_driver = {
2524879347f0SBen Kao 	.driver = {
2525879347f0SBen Kao 		.name = "ov8856",
2526879347f0SBen Kao 		.pm = &ov8856_pm_ops,
2527879347f0SBen Kao 		.acpi_match_table = ACPI_PTR(ov8856_acpi_ids),
25280c2c7a1eSRobert Foss 		.of_match_table = ov8856_of_match,
2529879347f0SBen Kao 	},
2530*aaeb31c0SUwe Kleine-König 	.probe = ov8856_probe,
2531879347f0SBen Kao 	.remove = ov8856_remove,
25320e014f1aSBingbu Cao 	.flags = I2C_DRV_ACPI_WAIVE_D0_PROBE,
2533879347f0SBen Kao };
2534879347f0SBen Kao 
2535879347f0SBen Kao module_i2c_driver(ov8856_i2c_driver);
2536879347f0SBen Kao 
2537879347f0SBen Kao MODULE_AUTHOR("Ben Kao <ben.kao@intel.com>");
2538879347f0SBen Kao MODULE_DESCRIPTION("OmniVision OV8856 sensor driver");
2539879347f0SBen Kao MODULE_LICENSE("GPL v2");
2540