162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * ov534-ov9xxx gspca driver
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2009-2011 Jean-Francois Moine http://moinejf.free.fr
662306a36Sopenharmony_ci * Copyright (C) 2008 Antonio Ospite <ospite@studenti.unina.it>
762306a36Sopenharmony_ci * Copyright (C) 2008 Jim Paris <jim@jtan.com>
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * Based on a prototype written by Mark Ferrell <majortrips@gmail.com>
1062306a36Sopenharmony_ci * USB protocol reverse engineered by Jim Paris <jim@jtan.com>
1162306a36Sopenharmony_ci * https://jim.sh/svn/jim/devl/playstation/ps3/eye/test/
1262306a36Sopenharmony_ci */
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
1562306a36Sopenharmony_ci
1662306a36Sopenharmony_ci#define MODULE_NAME "ov534_9"
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ci#include "gspca.h"
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#define OV534_REG_ADDRESS	0xf1	/* sensor address */
2162306a36Sopenharmony_ci#define OV534_REG_SUBADDR	0xf2
2262306a36Sopenharmony_ci#define OV534_REG_WRITE		0xf3
2362306a36Sopenharmony_ci#define OV534_REG_READ		0xf4
2462306a36Sopenharmony_ci#define OV534_REG_OPERATION	0xf5
2562306a36Sopenharmony_ci#define OV534_REG_STATUS	0xf6
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci#define OV534_OP_WRITE_3	0x37
2862306a36Sopenharmony_ci#define OV534_OP_WRITE_2	0x33
2962306a36Sopenharmony_ci#define OV534_OP_READ_2		0xf9
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci#define CTRL_TIMEOUT 500
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ciMODULE_AUTHOR("Jean-Francois Moine <moinejf@free.fr>");
3462306a36Sopenharmony_ciMODULE_DESCRIPTION("GSPCA/OV534_9 USB Camera Driver");
3562306a36Sopenharmony_ciMODULE_LICENSE("GPL");
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci/* specific webcam descriptor */
3862306a36Sopenharmony_cistruct sd {
3962306a36Sopenharmony_ci	struct gspca_dev gspca_dev;	/* !! must be the first item */
4062306a36Sopenharmony_ci	__u32 last_pts;
4162306a36Sopenharmony_ci	u8 last_fid;
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	u8 sensor;
4462306a36Sopenharmony_ci};
4562306a36Sopenharmony_cienum sensors {
4662306a36Sopenharmony_ci	SENSOR_OV965x,		/* ov9657 */
4762306a36Sopenharmony_ci	SENSOR_OV971x,		/* ov9712 */
4862306a36Sopenharmony_ci	SENSOR_OV562x,		/* ov5621 */
4962306a36Sopenharmony_ci	SENSOR_OV361x,		/* ov3610 */
5062306a36Sopenharmony_ci	NSENSORS
5162306a36Sopenharmony_ci};
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_cistatic const struct v4l2_pix_format ov965x_mode[] = {
5462306a36Sopenharmony_ci#define QVGA_MODE 0
5562306a36Sopenharmony_ci	{320, 240, V4L2_PIX_FMT_JPEG, V4L2_FIELD_NONE,
5662306a36Sopenharmony_ci		.bytesperline = 320,
5762306a36Sopenharmony_ci		.sizeimage = 320 * 240 * 3 / 8 + 590,
5862306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_JPEG},
5962306a36Sopenharmony_ci#define VGA_MODE 1
6062306a36Sopenharmony_ci	{640, 480, V4L2_PIX_FMT_JPEG, V4L2_FIELD_NONE,
6162306a36Sopenharmony_ci		.bytesperline = 640,
6262306a36Sopenharmony_ci		.sizeimage = 640 * 480 * 3 / 8 + 590,
6362306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_JPEG},
6462306a36Sopenharmony_ci#define SVGA_MODE 2
6562306a36Sopenharmony_ci	{800, 600, V4L2_PIX_FMT_JPEG, V4L2_FIELD_NONE,
6662306a36Sopenharmony_ci		.bytesperline = 800,
6762306a36Sopenharmony_ci		.sizeimage = 800 * 600 * 3 / 8 + 590,
6862306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_JPEG},
6962306a36Sopenharmony_ci#define XGA_MODE 3
7062306a36Sopenharmony_ci	{1024, 768, V4L2_PIX_FMT_JPEG, V4L2_FIELD_NONE,
7162306a36Sopenharmony_ci		.bytesperline = 1024,
7262306a36Sopenharmony_ci		.sizeimage = 1024 * 768 * 3 / 8 + 590,
7362306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_JPEG},
7462306a36Sopenharmony_ci#define SXGA_MODE 4
7562306a36Sopenharmony_ci	{1280, 1024, V4L2_PIX_FMT_JPEG, V4L2_FIELD_NONE,
7662306a36Sopenharmony_ci		.bytesperline = 1280,
7762306a36Sopenharmony_ci		.sizeimage = 1280 * 1024 * 3 / 8 + 590,
7862306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_JPEG},
7962306a36Sopenharmony_ci};
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_cistatic const struct v4l2_pix_format ov971x_mode[] = {
8262306a36Sopenharmony_ci	{640, 480, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
8362306a36Sopenharmony_ci		.bytesperline = 640,
8462306a36Sopenharmony_ci		.sizeimage = 640 * 480,
8562306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB
8662306a36Sopenharmony_ci	}
8762306a36Sopenharmony_ci};
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cistatic const struct v4l2_pix_format ov562x_mode[] = {
9062306a36Sopenharmony_ci	{2592, 1680, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
9162306a36Sopenharmony_ci		.bytesperline = 2592,
9262306a36Sopenharmony_ci		.sizeimage = 2592 * 1680,
9362306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB
9462306a36Sopenharmony_ci	}
9562306a36Sopenharmony_ci};
9662306a36Sopenharmony_ci
9762306a36Sopenharmony_cienum ov361x {
9862306a36Sopenharmony_ci	ov361x_2048 = 0,
9962306a36Sopenharmony_ci	ov361x_1600,
10062306a36Sopenharmony_ci	ov361x_1024,
10162306a36Sopenharmony_ci	ov361x_640,
10262306a36Sopenharmony_ci	ov361x_320,
10362306a36Sopenharmony_ci	ov361x_160,
10462306a36Sopenharmony_ci	ov361x_last
10562306a36Sopenharmony_ci};
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_cistatic const struct v4l2_pix_format ov361x_mode[] = {
10862306a36Sopenharmony_ci	{0x800, 0x600, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
10962306a36Sopenharmony_ci		.bytesperline = 0x800,
11062306a36Sopenharmony_ci		.sizeimage = 0x800 * 0x600,
11162306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB},
11262306a36Sopenharmony_ci	{1600, 1200, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
11362306a36Sopenharmony_ci		.bytesperline = 1600,
11462306a36Sopenharmony_ci		.sizeimage = 1600 * 1200,
11562306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB},
11662306a36Sopenharmony_ci	{1024, 768, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
11762306a36Sopenharmony_ci		.bytesperline = 768,
11862306a36Sopenharmony_ci		.sizeimage = 1024 * 768,
11962306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB},
12062306a36Sopenharmony_ci	{640, 480, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
12162306a36Sopenharmony_ci		.bytesperline = 640,
12262306a36Sopenharmony_ci		.sizeimage = 640 * 480,
12362306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB},
12462306a36Sopenharmony_ci	{320, 240, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
12562306a36Sopenharmony_ci		.bytesperline = 320,
12662306a36Sopenharmony_ci		.sizeimage = 320 * 240,
12762306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB},
12862306a36Sopenharmony_ci	{160, 120, V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
12962306a36Sopenharmony_ci		.bytesperline = 160,
13062306a36Sopenharmony_ci		.sizeimage = 160 * 120,
13162306a36Sopenharmony_ci		.colorspace = V4L2_COLORSPACE_SRGB}
13262306a36Sopenharmony_ci};
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_cistatic const u8 ov361x_start_2048[][2] = {
13562306a36Sopenharmony_ci	{0x12, 0x80},
13662306a36Sopenharmony_ci	{0x13, 0xcf},
13762306a36Sopenharmony_ci	{0x14, 0x40},
13862306a36Sopenharmony_ci	{0x15, 0x00},
13962306a36Sopenharmony_ci	{0x01, 0x80},
14062306a36Sopenharmony_ci	{0x02, 0x80},
14162306a36Sopenharmony_ci	{0x04, 0x70},
14262306a36Sopenharmony_ci	{0x0d, 0x40},
14362306a36Sopenharmony_ci	{0x0f, 0x47},
14462306a36Sopenharmony_ci	{0x11, 0x81},
14562306a36Sopenharmony_ci	{0x32, 0x36},
14662306a36Sopenharmony_ci	{0x33, 0x0c},
14762306a36Sopenharmony_ci	{0x34, 0x00},
14862306a36Sopenharmony_ci	{0x35, 0x90},
14962306a36Sopenharmony_ci	{0x12, 0x00},
15062306a36Sopenharmony_ci	{0x17, 0x10},
15162306a36Sopenharmony_ci	{0x18, 0x90},
15262306a36Sopenharmony_ci	{0x19, 0x00},
15362306a36Sopenharmony_ci	{0x1a, 0xc0},
15462306a36Sopenharmony_ci};
15562306a36Sopenharmony_cistatic const u8 ov361x_bridge_start_2048[][2] = {
15662306a36Sopenharmony_ci	{0xf1, 0x60},
15762306a36Sopenharmony_ci	{0x88, 0x00},
15862306a36Sopenharmony_ci	{0x89, 0x08},
15962306a36Sopenharmony_ci	{0x8a, 0x00},
16062306a36Sopenharmony_ci	{0x8b, 0x06},
16162306a36Sopenharmony_ci	{0x8c, 0x01},
16262306a36Sopenharmony_ci	{0x8d, 0x10},
16362306a36Sopenharmony_ci	{0x1c, 0x00},
16462306a36Sopenharmony_ci	{0x1d, 0x48},
16562306a36Sopenharmony_ci	{0x1d, 0x00},
16662306a36Sopenharmony_ci	{0x1d, 0xff},
16762306a36Sopenharmony_ci	{0x1c, 0x0a},
16862306a36Sopenharmony_ci	{0x1d, 0x2e},
16962306a36Sopenharmony_ci	{0x1d, 0x1e},
17062306a36Sopenharmony_ci};
17162306a36Sopenharmony_ci
17262306a36Sopenharmony_cistatic const u8 ov361x_start_1600[][2] = {
17362306a36Sopenharmony_ci	{0x12, 0x80},
17462306a36Sopenharmony_ci	{0x13, 0xcf},
17562306a36Sopenharmony_ci	{0x14, 0x40},
17662306a36Sopenharmony_ci	{0x15, 0x00},
17762306a36Sopenharmony_ci	{0x01, 0x80},
17862306a36Sopenharmony_ci	{0x02, 0x80},
17962306a36Sopenharmony_ci	{0x04, 0x70},
18062306a36Sopenharmony_ci	{0x0d, 0x40},
18162306a36Sopenharmony_ci	{0x0f, 0x47},
18262306a36Sopenharmony_ci	{0x11, 0x81},
18362306a36Sopenharmony_ci	{0x32, 0x36},
18462306a36Sopenharmony_ci	{0x33, 0x0C},
18562306a36Sopenharmony_ci	{0x34, 0x00},
18662306a36Sopenharmony_ci	{0x35, 0x90},
18762306a36Sopenharmony_ci	{0x12, 0x00},
18862306a36Sopenharmony_ci	{0x17, 0x10},
18962306a36Sopenharmony_ci	{0x18, 0x90},
19062306a36Sopenharmony_ci	{0x19, 0x00},
19162306a36Sopenharmony_ci	{0x1a, 0xc0},
19262306a36Sopenharmony_ci};
19362306a36Sopenharmony_cistatic const u8 ov361x_bridge_start_1600[][2] = {
19462306a36Sopenharmony_ci	{0xf1, 0x60},  /* Hsize[7:0] */
19562306a36Sopenharmony_ci	{0x88, 0x00},  /* Hsize[15:8] Write Only, can't read */
19662306a36Sopenharmony_ci	{0x89, 0x08},  /* Vsize[7:0] */
19762306a36Sopenharmony_ci	{0x8a, 0x00},  /* Vsize[15:8] Write Only, can't read */
19862306a36Sopenharmony_ci	{0x8b, 0x06},  /* for Iso */
19962306a36Sopenharmony_ci	{0x8c, 0x01},  /* RAW input */
20062306a36Sopenharmony_ci	{0x8d, 0x10},
20162306a36Sopenharmony_ci	{0x1c, 0x00},  /* RAW output, Iso transfer */
20262306a36Sopenharmony_ci	{0x1d, 0x48},
20362306a36Sopenharmony_ci	{0x1d, 0x00},
20462306a36Sopenharmony_ci	{0x1d, 0xff},
20562306a36Sopenharmony_ci	{0x1c, 0x0a},  /* turn off JPEG, Iso mode */
20662306a36Sopenharmony_ci	{0x1d, 0x2e},  /* for Iso */
20762306a36Sopenharmony_ci	{0x1d, 0x1e},
20862306a36Sopenharmony_ci};
20962306a36Sopenharmony_ci
21062306a36Sopenharmony_cistatic const u8 ov361x_start_1024[][2] = {
21162306a36Sopenharmony_ci	{0x12, 0x80},
21262306a36Sopenharmony_ci	{0x13, 0xcf},
21362306a36Sopenharmony_ci	{0x14, 0x40},
21462306a36Sopenharmony_ci	{0x15, 0x00},
21562306a36Sopenharmony_ci	{0x01, 0x80},
21662306a36Sopenharmony_ci	{0x02, 0x80},
21762306a36Sopenharmony_ci	{0x04, 0x70},
21862306a36Sopenharmony_ci	{0x0d, 0x40},
21962306a36Sopenharmony_ci	{0x0f, 0x47},
22062306a36Sopenharmony_ci	{0x11, 0x81},
22162306a36Sopenharmony_ci	{0x32, 0x36},
22262306a36Sopenharmony_ci	{0x33, 0x0C},
22362306a36Sopenharmony_ci	{0x34, 0x00},
22462306a36Sopenharmony_ci	{0x35, 0x90},
22562306a36Sopenharmony_ci	{0x12, 0x40},
22662306a36Sopenharmony_ci	{0x17, 0x1f},
22762306a36Sopenharmony_ci	{0x18, 0x5f},
22862306a36Sopenharmony_ci	{0x19, 0x00},
22962306a36Sopenharmony_ci	{0x1a, 0x68},
23062306a36Sopenharmony_ci};
23162306a36Sopenharmony_cistatic const u8 ov361x_bridge_start_1024[][2] = {
23262306a36Sopenharmony_ci	{0xf1, 0x60},  /* Hsize[7:0] */
23362306a36Sopenharmony_ci	{0x88, 0x00},  /* Hsize[15:8] Write Only, can't read */
23462306a36Sopenharmony_ci	{0x89, 0x04},  /* Vsize[7:0] */
23562306a36Sopenharmony_ci	{0x8a, 0x00},  /* Vsize[15:8] Write Only, can't read */
23662306a36Sopenharmony_ci	{0x8b, 0x03},  /* for Iso */
23762306a36Sopenharmony_ci	{0x8c, 0x01},  /* RAW input  */
23862306a36Sopenharmony_ci	{0x8d, 0x10},
23962306a36Sopenharmony_ci	{0x1c, 0x00},  /* RAW output, Iso transfer */
24062306a36Sopenharmony_ci	{0x1d, 0x48},
24162306a36Sopenharmony_ci	{0x1d, 0x00},
24262306a36Sopenharmony_ci	{0x1d, 0xff},
24362306a36Sopenharmony_ci	{0x1c, 0x0a},  /* turn off JPEG, Iso mode */
24462306a36Sopenharmony_ci	{0x1d, 0x2e},  /* for Iso */
24562306a36Sopenharmony_ci	{0x1d, 0x1e},
24662306a36Sopenharmony_ci};
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_cistatic const u8 ov361x_start_640[][2] = {
24962306a36Sopenharmony_ci	{0x12, 0x80},
25062306a36Sopenharmony_ci	{0x13, 0xcf},
25162306a36Sopenharmony_ci	{0x14, 0x40},
25262306a36Sopenharmony_ci	{0x15, 0x00},
25362306a36Sopenharmony_ci	{0x01, 0x80},
25462306a36Sopenharmony_ci	{0x02, 0x80},
25562306a36Sopenharmony_ci	{0x04, 0x70},
25662306a36Sopenharmony_ci	{0x0d, 0x40},
25762306a36Sopenharmony_ci	{0x0f, 0x47},
25862306a36Sopenharmony_ci	{0x11, 0x81},
25962306a36Sopenharmony_ci	{0x32, 0x36},
26062306a36Sopenharmony_ci	{0x33, 0x0C},
26162306a36Sopenharmony_ci	{0x34, 0x00},
26262306a36Sopenharmony_ci	{0x35, 0x90},
26362306a36Sopenharmony_ci	{0x12, 0x40},
26462306a36Sopenharmony_ci	{0x17, 0x1f},
26562306a36Sopenharmony_ci	{0x18, 0x5f},
26662306a36Sopenharmony_ci	{0x19, 0x00},
26762306a36Sopenharmony_ci	{0x1a, 0x68},
26862306a36Sopenharmony_ci};
26962306a36Sopenharmony_ci
27062306a36Sopenharmony_cistatic const u8 ov361x_bridge_start_640[][2] = {
27162306a36Sopenharmony_ci	{0xf1, 0x60},  /* Hsize[7:0]*/
27262306a36Sopenharmony_ci	{0x88, 0x00},  /* Hsize[15:8] Write Only, can't read */
27362306a36Sopenharmony_ci	{0x89, 0x04},  /* Vsize[7:0] */
27462306a36Sopenharmony_ci	{0x8a, 0x00},  /* Vsize[15:8] Write Only, can't read */
27562306a36Sopenharmony_ci	{0x8b, 0x03},  /* for Iso */
27662306a36Sopenharmony_ci	{0x8c, 0x01},  /* RAW input */
27762306a36Sopenharmony_ci	{0x8d, 0x10},
27862306a36Sopenharmony_ci	{0x1c, 0x00},  /* RAW output, Iso transfer */
27962306a36Sopenharmony_ci	{0x1d, 0x48},
28062306a36Sopenharmony_ci	{0x1d, 0x00},
28162306a36Sopenharmony_ci	{0x1d, 0xff},
28262306a36Sopenharmony_ci	{0x1c, 0x0a},  /* turn off JPEG, Iso mode */
28362306a36Sopenharmony_ci	{0x1d, 0x2e},  /* for Iso */
28462306a36Sopenharmony_ci	{0x1d, 0x1e},
28562306a36Sopenharmony_ci};
28662306a36Sopenharmony_ci
28762306a36Sopenharmony_cistatic const u8 ov361x_start_320[][2] = {
28862306a36Sopenharmony_ci	{0x12, 0x80},
28962306a36Sopenharmony_ci	{0x13, 0xcf},
29062306a36Sopenharmony_ci	{0x14, 0x40},
29162306a36Sopenharmony_ci	{0x15, 0x00},
29262306a36Sopenharmony_ci	{0x01, 0x80},
29362306a36Sopenharmony_ci	{0x02, 0x80},
29462306a36Sopenharmony_ci	{0x04, 0x70},
29562306a36Sopenharmony_ci	{0x0d, 0x40},
29662306a36Sopenharmony_ci	{0x0f, 0x47},
29762306a36Sopenharmony_ci	{0x11, 0x81},
29862306a36Sopenharmony_ci	{0x32, 0x36},
29962306a36Sopenharmony_ci	{0x33, 0x0C},
30062306a36Sopenharmony_ci	{0x34, 0x00},
30162306a36Sopenharmony_ci	{0x35, 0x90},
30262306a36Sopenharmony_ci	{0x12, 0x40},
30362306a36Sopenharmony_ci	{0x17, 0x1f},
30462306a36Sopenharmony_ci	{0x18, 0x5f},
30562306a36Sopenharmony_ci	{0x19, 0x00},
30662306a36Sopenharmony_ci	{0x1a, 0x68},
30762306a36Sopenharmony_ci};
30862306a36Sopenharmony_ci
30962306a36Sopenharmony_cistatic const u8 ov361x_bridge_start_320[][2] = {
31062306a36Sopenharmony_ci	{0xf1, 0x60},  /* Hsize[7:0] */
31162306a36Sopenharmony_ci	{0x88, 0x00},  /* Hsize[15:8] Write Only, can't read */
31262306a36Sopenharmony_ci	{0x89, 0x04},  /* Vsize[7:0] */
31362306a36Sopenharmony_ci	{0x8a, 0x00},  /* Vsize[15:8] Write Only, can't read */
31462306a36Sopenharmony_ci	{0x8b, 0x03},  /* for Iso */
31562306a36Sopenharmony_ci	{0x8c, 0x01},  /* RAW input */
31662306a36Sopenharmony_ci	{0x8d, 0x10},
31762306a36Sopenharmony_ci	{0x1c, 0x00},  /* RAW output, Iso transfer; */
31862306a36Sopenharmony_ci	{0x1d, 0x48},
31962306a36Sopenharmony_ci	{0x1d, 0x00},
32062306a36Sopenharmony_ci	{0x1d, 0xff},
32162306a36Sopenharmony_ci	{0x1c, 0x0a},  /* turn off JPEG, Iso mode */
32262306a36Sopenharmony_ci	{0x1d, 0x2e},  /* for Iso */
32362306a36Sopenharmony_ci	{0x1d, 0x1e},
32462306a36Sopenharmony_ci};
32562306a36Sopenharmony_ci
32662306a36Sopenharmony_cistatic const u8 ov361x_start_160[][2] = {
32762306a36Sopenharmony_ci	{0x12, 0x80},
32862306a36Sopenharmony_ci	{0x13, 0xcf},
32962306a36Sopenharmony_ci	{0x14, 0x40},
33062306a36Sopenharmony_ci	{0x15, 0x00},
33162306a36Sopenharmony_ci	{0x01, 0x80},
33262306a36Sopenharmony_ci	{0x02, 0x80},
33362306a36Sopenharmony_ci	{0x04, 0x70},
33462306a36Sopenharmony_ci	{0x0d, 0x40},
33562306a36Sopenharmony_ci	{0x0f, 0x47},
33662306a36Sopenharmony_ci	{0x11, 0x81},
33762306a36Sopenharmony_ci	{0x32, 0x36},
33862306a36Sopenharmony_ci	{0x33, 0x0C},
33962306a36Sopenharmony_ci	{0x34, 0x00},
34062306a36Sopenharmony_ci	{0x35, 0x90},
34162306a36Sopenharmony_ci	{0x12, 0x40},
34262306a36Sopenharmony_ci	{0x17, 0x1f},
34362306a36Sopenharmony_ci	{0x18, 0x5f},
34462306a36Sopenharmony_ci	{0x19, 0x00},
34562306a36Sopenharmony_ci	{0x1a, 0x68},
34662306a36Sopenharmony_ci};
34762306a36Sopenharmony_ci
34862306a36Sopenharmony_cistatic const u8 ov361x_bridge_start_160[][2] = {
34962306a36Sopenharmony_ci	{0xf1, 0x60},  /* Hsize[7:0] */
35062306a36Sopenharmony_ci	{0x88, 0x00},  /* Hsize[15:8] Write Only, can't read */
35162306a36Sopenharmony_ci	{0x89, 0x04},  /* Vsize[7:0] */
35262306a36Sopenharmony_ci	{0x8a, 0x00},  /* Vsize[15:8] Write Only, can't read */
35362306a36Sopenharmony_ci	{0x8b, 0x03},  /* for Iso */
35462306a36Sopenharmony_ci	{0x8c, 0x01},  /* RAW input */
35562306a36Sopenharmony_ci	{0x8d, 0x10},
35662306a36Sopenharmony_ci	{0x1c, 0x00},  /* RAW output, Iso transfer */
35762306a36Sopenharmony_ci	{0x1d, 0x48},
35862306a36Sopenharmony_ci	{0x1d, 0x00},
35962306a36Sopenharmony_ci	{0x1d, 0xff},
36062306a36Sopenharmony_ci	{0x1c, 0x0a},  /* turn off JPEG, Iso mode */
36162306a36Sopenharmony_ci	{0x1d, 0x2e},  /* for Iso */
36262306a36Sopenharmony_ci	{0x1d, 0x1e},
36362306a36Sopenharmony_ci};
36462306a36Sopenharmony_ci
36562306a36Sopenharmony_cistatic const u8 bridge_init[][2] = {
36662306a36Sopenharmony_ci	{0x88, 0xf8},
36762306a36Sopenharmony_ci	{0x89, 0xff},
36862306a36Sopenharmony_ci	{0x76, 0x03},
36962306a36Sopenharmony_ci	{0x92, 0x03},
37062306a36Sopenharmony_ci	{0x95, 0x10},
37162306a36Sopenharmony_ci	{0xe2, 0x00},
37262306a36Sopenharmony_ci	{0xe7, 0x3e},
37362306a36Sopenharmony_ci	{0x8d, 0x1c},
37462306a36Sopenharmony_ci	{0x8e, 0x00},
37562306a36Sopenharmony_ci	{0x8f, 0x00},
37662306a36Sopenharmony_ci	{0x1f, 0x00},
37762306a36Sopenharmony_ci	{0xc3, 0xf9},
37862306a36Sopenharmony_ci	{0x89, 0xff},
37962306a36Sopenharmony_ci	{0x88, 0xf8},
38062306a36Sopenharmony_ci	{0x76, 0x03},
38162306a36Sopenharmony_ci	{0x92, 0x01},
38262306a36Sopenharmony_ci	{0x93, 0x18},
38362306a36Sopenharmony_ci	{0x1c, 0x0a},
38462306a36Sopenharmony_ci	{0x1d, 0x48},
38562306a36Sopenharmony_ci	{0xc0, 0x50},
38662306a36Sopenharmony_ci	{0xc1, 0x3c},
38762306a36Sopenharmony_ci	{0x34, 0x05},
38862306a36Sopenharmony_ci	{0xc2, 0x0c},
38962306a36Sopenharmony_ci	{0xc3, 0xf9},
39062306a36Sopenharmony_ci	{0x34, 0x05},
39162306a36Sopenharmony_ci	{0xe7, 0x2e},
39262306a36Sopenharmony_ci	{0x31, 0xf9},
39362306a36Sopenharmony_ci	{0x35, 0x02},
39462306a36Sopenharmony_ci	{0xd9, 0x10},
39562306a36Sopenharmony_ci	{0x25, 0x42},
39662306a36Sopenharmony_ci	{0x94, 0x11},
39762306a36Sopenharmony_ci};
39862306a36Sopenharmony_ci
39962306a36Sopenharmony_cistatic const u8 ov965x_init[][2] = {
40062306a36Sopenharmony_ci	{0x12, 0x80},	/* com7 - SSCB reset */
40162306a36Sopenharmony_ci	{0x00, 0x00},	/* gain */
40262306a36Sopenharmony_ci	{0x01, 0x80},	/* blue */
40362306a36Sopenharmony_ci	{0x02, 0x80},	/* red */
40462306a36Sopenharmony_ci	{0x03, 0x1b},	/* vref */
40562306a36Sopenharmony_ci	{0x04, 0x03},	/* com1 - exposure low bits */
40662306a36Sopenharmony_ci	{0x0b, 0x57},	/* ver */
40762306a36Sopenharmony_ci	{0x0e, 0x61},	/* com5 */
40862306a36Sopenharmony_ci	{0x0f, 0x42},	/* com6 */
40962306a36Sopenharmony_ci	{0x11, 0x00},	/* clkrc */
41062306a36Sopenharmony_ci	{0x12, 0x02},	/* com7 - 15fps VGA YUYV */
41162306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
41262306a36Sopenharmony_ci	{0x14, 0x28},	/* com9 */
41362306a36Sopenharmony_ci	{0x16, 0x24},	/* reg16 */
41462306a36Sopenharmony_ci	{0x17, 0x1d},	/* hstart*/
41562306a36Sopenharmony_ci	{0x18, 0xbd},	/* hstop */
41662306a36Sopenharmony_ci	{0x19, 0x01},	/* vstrt */
41762306a36Sopenharmony_ci	{0x1a, 0x81},	/* vstop*/
41862306a36Sopenharmony_ci	{0x1e, 0x04},	/* mvfp */
41962306a36Sopenharmony_ci	{0x24, 0x3c},	/* aew */
42062306a36Sopenharmony_ci	{0x25, 0x36},	/* aeb */
42162306a36Sopenharmony_ci	{0x26, 0x71},	/* vpt */
42262306a36Sopenharmony_ci	{0x27, 0x08},	/* bbias */
42362306a36Sopenharmony_ci	{0x28, 0x08},	/* gbbias */
42462306a36Sopenharmony_ci	{0x29, 0x15},	/* gr com */
42562306a36Sopenharmony_ci	{0x2a, 0x00},	/* exhch */
42662306a36Sopenharmony_ci	{0x2b, 0x00},	/* exhcl */
42762306a36Sopenharmony_ci	{0x2c, 0x08},	/* rbias */
42862306a36Sopenharmony_ci	{0x32, 0xff},	/* href */
42962306a36Sopenharmony_ci	{0x33, 0x00},	/* chlf */
43062306a36Sopenharmony_ci	{0x34, 0x3f},	/* aref1 */
43162306a36Sopenharmony_ci	{0x35, 0x00},	/* aref2 */
43262306a36Sopenharmony_ci	{0x36, 0xf8},	/* aref3 */
43362306a36Sopenharmony_ci	{0x38, 0x72},	/* adc2 */
43462306a36Sopenharmony_ci	{0x39, 0x57},	/* aref4 */
43562306a36Sopenharmony_ci	{0x3a, 0x80},	/* tslb - yuyv */
43662306a36Sopenharmony_ci	{0x3b, 0xc4},	/* com11 - night mode 1/4 frame rate */
43762306a36Sopenharmony_ci	{0x3d, 0x99},	/* com13 */
43862306a36Sopenharmony_ci	{0x3f, 0xc1},	/* edge */
43962306a36Sopenharmony_ci	{0x40, 0xc0},	/* com15 */
44062306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
44162306a36Sopenharmony_ci	{0x42, 0xc0},	/* com17 */
44262306a36Sopenharmony_ci	{0x43, 0x0a},	/* rsvd */
44362306a36Sopenharmony_ci	{0x44, 0xf0},
44462306a36Sopenharmony_ci	{0x45, 0x46},
44562306a36Sopenharmony_ci	{0x46, 0x62},
44662306a36Sopenharmony_ci	{0x47, 0x2a},
44762306a36Sopenharmony_ci	{0x48, 0x3c},
44862306a36Sopenharmony_ci	{0x4a, 0xfc},
44962306a36Sopenharmony_ci	{0x4b, 0xfc},
45062306a36Sopenharmony_ci	{0x4c, 0x7f},
45162306a36Sopenharmony_ci	{0x4d, 0x7f},
45262306a36Sopenharmony_ci	{0x4e, 0x7f},
45362306a36Sopenharmony_ci	{0x4f, 0x98},	/* matrix */
45462306a36Sopenharmony_ci	{0x50, 0x98},
45562306a36Sopenharmony_ci	{0x51, 0x00},
45662306a36Sopenharmony_ci	{0x52, 0x28},
45762306a36Sopenharmony_ci	{0x53, 0x70},
45862306a36Sopenharmony_ci	{0x54, 0x98},
45962306a36Sopenharmony_ci	{0x58, 0x1a},	/* matrix coef sign */
46062306a36Sopenharmony_ci	{0x59, 0x85},	/* AWB control */
46162306a36Sopenharmony_ci	{0x5a, 0xa9},
46262306a36Sopenharmony_ci	{0x5b, 0x64},
46362306a36Sopenharmony_ci	{0x5c, 0x84},
46462306a36Sopenharmony_ci	{0x5d, 0x53},
46562306a36Sopenharmony_ci	{0x5e, 0x0e},
46662306a36Sopenharmony_ci	{0x5f, 0xf0},	/* AWB blue limit */
46762306a36Sopenharmony_ci	{0x60, 0xf0},	/* AWB red limit */
46862306a36Sopenharmony_ci	{0x61, 0xf0},	/* AWB green limit */
46962306a36Sopenharmony_ci	{0x62, 0x00},	/* lcc1 */
47062306a36Sopenharmony_ci	{0x63, 0x00},	/* lcc2 */
47162306a36Sopenharmony_ci	{0x64, 0x02},	/* lcc3 */
47262306a36Sopenharmony_ci	{0x65, 0x16},	/* lcc4 */
47362306a36Sopenharmony_ci	{0x66, 0x01},	/* lcc5 */
47462306a36Sopenharmony_ci	{0x69, 0x02},	/* hv */
47562306a36Sopenharmony_ci	{0x6b, 0x5a},	/* dbvl */
47662306a36Sopenharmony_ci	{0x6c, 0x04},
47762306a36Sopenharmony_ci	{0x6d, 0x55},
47862306a36Sopenharmony_ci	{0x6e, 0x00},
47962306a36Sopenharmony_ci	{0x6f, 0x9d},
48062306a36Sopenharmony_ci	{0x70, 0x21},	/* dnsth */
48162306a36Sopenharmony_ci	{0x71, 0x78},
48262306a36Sopenharmony_ci	{0x72, 0x00},	/* poidx */
48362306a36Sopenharmony_ci	{0x73, 0x01},	/* pckdv */
48462306a36Sopenharmony_ci	{0x74, 0x3a},	/* xindx */
48562306a36Sopenharmony_ci	{0x75, 0x35},	/* yindx */
48662306a36Sopenharmony_ci	{0x76, 0x01},
48762306a36Sopenharmony_ci	{0x77, 0x02},
48862306a36Sopenharmony_ci	{0x7a, 0x12},	/* gamma curve */
48962306a36Sopenharmony_ci	{0x7b, 0x08},
49062306a36Sopenharmony_ci	{0x7c, 0x16},
49162306a36Sopenharmony_ci	{0x7d, 0x30},
49262306a36Sopenharmony_ci	{0x7e, 0x5e},
49362306a36Sopenharmony_ci	{0x7f, 0x72},
49462306a36Sopenharmony_ci	{0x80, 0x82},
49562306a36Sopenharmony_ci	{0x81, 0x8e},
49662306a36Sopenharmony_ci	{0x82, 0x9a},
49762306a36Sopenharmony_ci	{0x83, 0xa4},
49862306a36Sopenharmony_ci	{0x84, 0xac},
49962306a36Sopenharmony_ci	{0x85, 0xb8},
50062306a36Sopenharmony_ci	{0x86, 0xc3},
50162306a36Sopenharmony_ci	{0x87, 0xd6},
50262306a36Sopenharmony_ci	{0x88, 0xe6},
50362306a36Sopenharmony_ci	{0x89, 0xf2},
50462306a36Sopenharmony_ci	{0x8a, 0x03},
50562306a36Sopenharmony_ci	{0x8c, 0x89},	/* com19 */
50662306a36Sopenharmony_ci	{0x14, 0x28},	/* com9 */
50762306a36Sopenharmony_ci	{0x90, 0x7d},
50862306a36Sopenharmony_ci	{0x91, 0x7b},
50962306a36Sopenharmony_ci	{0x9d, 0x03},	/* lcc6 */
51062306a36Sopenharmony_ci	{0x9e, 0x04},	/* lcc7 */
51162306a36Sopenharmony_ci	{0x9f, 0x7a},
51262306a36Sopenharmony_ci	{0xa0, 0x79},
51362306a36Sopenharmony_ci	{0xa1, 0x40},	/* aechm */
51462306a36Sopenharmony_ci	{0xa4, 0x50},	/* com21 */
51562306a36Sopenharmony_ci	{0xa5, 0x68},	/* com26 */
51662306a36Sopenharmony_ci	{0xa6, 0x4a},	/* AWB green */
51762306a36Sopenharmony_ci	{0xa8, 0xc1},	/* refa8 */
51862306a36Sopenharmony_ci	{0xa9, 0xef},	/* refa9 */
51962306a36Sopenharmony_ci	{0xaa, 0x92},
52062306a36Sopenharmony_ci	{0xab, 0x04},
52162306a36Sopenharmony_ci	{0xac, 0x80},	/* black level control */
52262306a36Sopenharmony_ci	{0xad, 0x80},
52362306a36Sopenharmony_ci	{0xae, 0x80},
52462306a36Sopenharmony_ci	{0xaf, 0x80},
52562306a36Sopenharmony_ci	{0xb2, 0xf2},
52662306a36Sopenharmony_ci	{0xb3, 0x20},
52762306a36Sopenharmony_ci	{0xb4, 0x20},	/* ctrlb4 */
52862306a36Sopenharmony_ci	{0xb5, 0x00},
52962306a36Sopenharmony_ci	{0xb6, 0xaf},
53062306a36Sopenharmony_ci	{0xbb, 0xae},
53162306a36Sopenharmony_ci	{0xbc, 0x7f},	/* ADC channel offsets */
53262306a36Sopenharmony_ci	{0xdb, 0x7f},
53362306a36Sopenharmony_ci	{0xbe, 0x7f},
53462306a36Sopenharmony_ci	{0xbf, 0x7f},
53562306a36Sopenharmony_ci	{0xc0, 0xe2},
53662306a36Sopenharmony_ci	{0xc1, 0xc0},
53762306a36Sopenharmony_ci	{0xc2, 0x01},
53862306a36Sopenharmony_ci	{0xc3, 0x4e},
53962306a36Sopenharmony_ci	{0xc6, 0x85},
54062306a36Sopenharmony_ci	{0xc7, 0x80},	/* com24 */
54162306a36Sopenharmony_ci	{0xc9, 0xe0},
54262306a36Sopenharmony_ci	{0xca, 0xe8},
54362306a36Sopenharmony_ci	{0xcb, 0xf0},
54462306a36Sopenharmony_ci	{0xcc, 0xd8},
54562306a36Sopenharmony_ci	{0xcd, 0xf1},
54662306a36Sopenharmony_ci	{0x4f, 0x98},	/* matrix */
54762306a36Sopenharmony_ci	{0x50, 0x98},
54862306a36Sopenharmony_ci	{0x51, 0x00},
54962306a36Sopenharmony_ci	{0x52, 0x28},
55062306a36Sopenharmony_ci	{0x53, 0x70},
55162306a36Sopenharmony_ci	{0x54, 0x98},
55262306a36Sopenharmony_ci	{0x58, 0x1a},
55362306a36Sopenharmony_ci	{0xff, 0x41},	/* read 41, write ff 00 */
55462306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
55562306a36Sopenharmony_ci
55662306a36Sopenharmony_ci	{0xc5, 0x03},	/* 60 Hz banding filter */
55762306a36Sopenharmony_ci	{0x6a, 0x02},	/* 50 Hz banding filter */
55862306a36Sopenharmony_ci
55962306a36Sopenharmony_ci	{0x12, 0x62},	/* com7 - 30fps VGA YUV */
56062306a36Sopenharmony_ci	{0x36, 0xfa},	/* aref3 */
56162306a36Sopenharmony_ci	{0x69, 0x0a},	/* hv */
56262306a36Sopenharmony_ci	{0x8c, 0x89},	/* com22 */
56362306a36Sopenharmony_ci	{0x14, 0x28},	/* com9 */
56462306a36Sopenharmony_ci	{0x3e, 0x0c},
56562306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
56662306a36Sopenharmony_ci	{0x72, 0x00},
56762306a36Sopenharmony_ci	{0x73, 0x00},
56862306a36Sopenharmony_ci	{0x74, 0x3a},
56962306a36Sopenharmony_ci	{0x75, 0x35},
57062306a36Sopenharmony_ci	{0x76, 0x01},
57162306a36Sopenharmony_ci	{0xc7, 0x80},
57262306a36Sopenharmony_ci	{0x03, 0x12},	/* vref */
57362306a36Sopenharmony_ci	{0x17, 0x16},	/* hstart */
57462306a36Sopenharmony_ci	{0x18, 0x02},	/* hstop */
57562306a36Sopenharmony_ci	{0x19, 0x01},	/* vstrt */
57662306a36Sopenharmony_ci	{0x1a, 0x3d},	/* vstop */
57762306a36Sopenharmony_ci	{0x32, 0xff},	/* href */
57862306a36Sopenharmony_ci	{0xc0, 0xaa},
57962306a36Sopenharmony_ci};
58062306a36Sopenharmony_ci
58162306a36Sopenharmony_cistatic const u8 bridge_init_2[][2] = {
58262306a36Sopenharmony_ci	{0x94, 0xaa},
58362306a36Sopenharmony_ci	{0xf1, 0x60},
58462306a36Sopenharmony_ci	{0xe5, 0x04},
58562306a36Sopenharmony_ci	{0xc0, 0x50},
58662306a36Sopenharmony_ci	{0xc1, 0x3c},
58762306a36Sopenharmony_ci	{0x8c, 0x00},
58862306a36Sopenharmony_ci	{0x8d, 0x1c},
58962306a36Sopenharmony_ci	{0x34, 0x05},
59062306a36Sopenharmony_ci
59162306a36Sopenharmony_ci	{0xc2, 0x0c},
59262306a36Sopenharmony_ci	{0xc3, 0xf9},
59362306a36Sopenharmony_ci	{0xda, 0x01},
59462306a36Sopenharmony_ci	{0x50, 0x00},
59562306a36Sopenharmony_ci	{0x51, 0xa0},
59662306a36Sopenharmony_ci	{0x52, 0x3c},
59762306a36Sopenharmony_ci	{0x53, 0x00},
59862306a36Sopenharmony_ci	{0x54, 0x00},
59962306a36Sopenharmony_ci	{0x55, 0x00},
60062306a36Sopenharmony_ci	{0x57, 0x00},
60162306a36Sopenharmony_ci	{0x5c, 0x00},
60262306a36Sopenharmony_ci	{0x5a, 0xa0},
60362306a36Sopenharmony_ci	{0x5b, 0x78},
60462306a36Sopenharmony_ci	{0x35, 0x02},
60562306a36Sopenharmony_ci	{0xd9, 0x10},
60662306a36Sopenharmony_ci	{0x94, 0x11},
60762306a36Sopenharmony_ci};
60862306a36Sopenharmony_ci
60962306a36Sopenharmony_cistatic const u8 ov965x_init_2[][2] = {
61062306a36Sopenharmony_ci	{0x3b, 0xc4},
61162306a36Sopenharmony_ci	{0x1e, 0x04},	/* mvfp */
61262306a36Sopenharmony_ci	{0x13, 0xe0},	/* com8 */
61362306a36Sopenharmony_ci	{0x00, 0x00},	/* gain */
61462306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
61562306a36Sopenharmony_ci	{0x11, 0x03},	/* clkrc */
61662306a36Sopenharmony_ci	{0x6b, 0x5a},	/* dblv */
61762306a36Sopenharmony_ci	{0x6a, 0x05},
61862306a36Sopenharmony_ci	{0xc5, 0x07},
61962306a36Sopenharmony_ci	{0xa2, 0x4b},
62062306a36Sopenharmony_ci	{0xa3, 0x3e},
62162306a36Sopenharmony_ci	{0x2d, 0x00},
62262306a36Sopenharmony_ci	{0xff, 0x42},	/* read 42, write ff 00 */
62362306a36Sopenharmony_ci	{0x42, 0xc0},	/* com17 */
62462306a36Sopenharmony_ci	{0x2d, 0x00},
62562306a36Sopenharmony_ci	{0xff, 0x42},	/* read 42, write ff 00 */
62662306a36Sopenharmony_ci	{0x42, 0xc1},	/* com17 */
62762306a36Sopenharmony_ci/* sharpness */
62862306a36Sopenharmony_ci	{0x3f, 0x01},
62962306a36Sopenharmony_ci	{0xff, 0x42},	/* read 42, write ff 00 */
63062306a36Sopenharmony_ci	{0x42, 0xc1},	/* com17 */
63162306a36Sopenharmony_ci/* saturation */
63262306a36Sopenharmony_ci	{0x4f, 0x98},	/* matrix */
63362306a36Sopenharmony_ci	{0x50, 0x98},
63462306a36Sopenharmony_ci	{0x51, 0x00},
63562306a36Sopenharmony_ci	{0x52, 0x28},
63662306a36Sopenharmony_ci	{0x53, 0x70},
63762306a36Sopenharmony_ci	{0x54, 0x98},
63862306a36Sopenharmony_ci	{0x58, 0x1a},
63962306a36Sopenharmony_ci	{0xff, 0x41},	/* read 41, write ff 00 */
64062306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
64162306a36Sopenharmony_ci/* contrast */
64262306a36Sopenharmony_ci	{0x56, 0x40},
64362306a36Sopenharmony_ci/* brightness */
64462306a36Sopenharmony_ci	{0x55, 0x8f},
64562306a36Sopenharmony_ci/* expo */
64662306a36Sopenharmony_ci	{0x10, 0x25},	/* aech - exposure high bits */
64762306a36Sopenharmony_ci	{0xff, 0x13},	/* read 13, write ff 00 */
64862306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
64962306a36Sopenharmony_ci};
65062306a36Sopenharmony_ci
65162306a36Sopenharmony_cistatic const u8 ov971x_init[][2] = {
65262306a36Sopenharmony_ci	{0x12, 0x80},
65362306a36Sopenharmony_ci	{0x09, 0x10},
65462306a36Sopenharmony_ci	{0x1e, 0x07},
65562306a36Sopenharmony_ci	{0x5f, 0x18},
65662306a36Sopenharmony_ci	{0x69, 0x04},
65762306a36Sopenharmony_ci	{0x65, 0x2a},
65862306a36Sopenharmony_ci	{0x68, 0x0a},
65962306a36Sopenharmony_ci	{0x39, 0x28},
66062306a36Sopenharmony_ci	{0x4d, 0x90},
66162306a36Sopenharmony_ci	{0xc1, 0x80},
66262306a36Sopenharmony_ci	{0x0c, 0x30},
66362306a36Sopenharmony_ci	{0x6d, 0x02},
66462306a36Sopenharmony_ci	{0x96, 0xf1},
66562306a36Sopenharmony_ci	{0xbc, 0x68},
66662306a36Sopenharmony_ci	{0x12, 0x00},
66762306a36Sopenharmony_ci	{0x3b, 0x00},
66862306a36Sopenharmony_ci	{0x97, 0x80},
66962306a36Sopenharmony_ci	{0x17, 0x25},
67062306a36Sopenharmony_ci	{0x18, 0xa2},
67162306a36Sopenharmony_ci	{0x19, 0x01},
67262306a36Sopenharmony_ci	{0x1a, 0xca},
67362306a36Sopenharmony_ci	{0x03, 0x0a},
67462306a36Sopenharmony_ci	{0x32, 0x07},
67562306a36Sopenharmony_ci	{0x98, 0x40},	/*{0x98, 0x00},*/
67662306a36Sopenharmony_ci	{0x99, 0xA0},	/*{0x99, 0x00},*/
67762306a36Sopenharmony_ci	{0x9a, 0x01},	/*{0x9a, 0x00},*/
67862306a36Sopenharmony_ci	{0x57, 0x00},
67962306a36Sopenharmony_ci	{0x58, 0x78},	/*{0x58, 0xc8},*/
68062306a36Sopenharmony_ci	{0x59, 0x50},	/*{0x59, 0xa0},*/
68162306a36Sopenharmony_ci	{0x4c, 0x13},
68262306a36Sopenharmony_ci	{0x4b, 0x36},
68362306a36Sopenharmony_ci	{0x3d, 0x3c},
68462306a36Sopenharmony_ci	{0x3e, 0x03},
68562306a36Sopenharmony_ci	{0xbd, 0x50},	/*{0xbd, 0xa0},*/
68662306a36Sopenharmony_ci	{0xbe, 0x78},	/*{0xbe, 0xc8},*/
68762306a36Sopenharmony_ci	{0x4e, 0x55},
68862306a36Sopenharmony_ci	{0x4f, 0x55},
68962306a36Sopenharmony_ci	{0x50, 0x55},
69062306a36Sopenharmony_ci	{0x51, 0x55},
69162306a36Sopenharmony_ci	{0x24, 0x55},
69262306a36Sopenharmony_ci	{0x25, 0x40},
69362306a36Sopenharmony_ci	{0x26, 0xa1},
69462306a36Sopenharmony_ci	{0x5c, 0x59},
69562306a36Sopenharmony_ci	{0x5d, 0x00},
69662306a36Sopenharmony_ci	{0x11, 0x00},
69762306a36Sopenharmony_ci	{0x2a, 0x98},
69862306a36Sopenharmony_ci	{0x2b, 0x06},
69962306a36Sopenharmony_ci	{0x2d, 0x00},
70062306a36Sopenharmony_ci	{0x2e, 0x00},
70162306a36Sopenharmony_ci	{0x13, 0xa5},
70262306a36Sopenharmony_ci	{0x14, 0x40},
70362306a36Sopenharmony_ci	{0x4a, 0x00},
70462306a36Sopenharmony_ci	{0x49, 0xce},
70562306a36Sopenharmony_ci	{0x22, 0x03},
70662306a36Sopenharmony_ci	{0x09, 0x00}
70762306a36Sopenharmony_ci};
70862306a36Sopenharmony_ci
70962306a36Sopenharmony_cistatic const u8 ov965x_start_1_vga[][2] = {	/* same for qvga */
71062306a36Sopenharmony_ci	{0x12, 0x62},	/* com7 - 30fps VGA YUV */
71162306a36Sopenharmony_ci	{0x36, 0xfa},	/* aref3 */
71262306a36Sopenharmony_ci	{0x69, 0x0a},	/* hv */
71362306a36Sopenharmony_ci	{0x8c, 0x89},	/* com22 */
71462306a36Sopenharmony_ci	{0x14, 0x28},	/* com9 */
71562306a36Sopenharmony_ci	{0x3e, 0x0c},	/* com14 */
71662306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
71762306a36Sopenharmony_ci	{0x72, 0x00},
71862306a36Sopenharmony_ci	{0x73, 0x00},
71962306a36Sopenharmony_ci	{0x74, 0x3a},
72062306a36Sopenharmony_ci	{0x75, 0x35},
72162306a36Sopenharmony_ci	{0x76, 0x01},
72262306a36Sopenharmony_ci	{0xc7, 0x80},	/* com24 */
72362306a36Sopenharmony_ci	{0x03, 0x12},	/* vref */
72462306a36Sopenharmony_ci	{0x17, 0x16},	/* hstart */
72562306a36Sopenharmony_ci	{0x18, 0x02},	/* hstop */
72662306a36Sopenharmony_ci	{0x19, 0x01},	/* vstrt */
72762306a36Sopenharmony_ci	{0x1a, 0x3d},	/* vstop */
72862306a36Sopenharmony_ci	{0x32, 0xff},	/* href */
72962306a36Sopenharmony_ci	{0xc0, 0xaa},
73062306a36Sopenharmony_ci};
73162306a36Sopenharmony_ci
73262306a36Sopenharmony_cistatic const u8 ov965x_start_1_svga[][2] = {
73362306a36Sopenharmony_ci	{0x12, 0x02},	/* com7 - YUYV - VGA 15 full resolution */
73462306a36Sopenharmony_ci	{0x36, 0xf8},	/* aref3 */
73562306a36Sopenharmony_ci	{0x69, 0x02},	/* hv */
73662306a36Sopenharmony_ci	{0x8c, 0x0d},	/* com22 */
73762306a36Sopenharmony_ci	{0x3e, 0x0c},	/* com14 */
73862306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
73962306a36Sopenharmony_ci	{0x72, 0x00},
74062306a36Sopenharmony_ci	{0x73, 0x01},
74162306a36Sopenharmony_ci	{0x74, 0x3a},
74262306a36Sopenharmony_ci	{0x75, 0x35},
74362306a36Sopenharmony_ci	{0x76, 0x01},
74462306a36Sopenharmony_ci	{0xc7, 0x80},	/* com24 */
74562306a36Sopenharmony_ci	{0x03, 0x1b},	/* vref */
74662306a36Sopenharmony_ci	{0x17, 0x1d},	/* hstart */
74762306a36Sopenharmony_ci	{0x18, 0xbd},	/* hstop */
74862306a36Sopenharmony_ci	{0x19, 0x01},	/* vstrt */
74962306a36Sopenharmony_ci	{0x1a, 0x81},	/* vstop */
75062306a36Sopenharmony_ci	{0x32, 0xff},	/* href */
75162306a36Sopenharmony_ci	{0xc0, 0xe2},
75262306a36Sopenharmony_ci};
75362306a36Sopenharmony_ci
75462306a36Sopenharmony_cistatic const u8 ov965x_start_1_xga[][2] = {
75562306a36Sopenharmony_ci	{0x12, 0x02},	/* com7 */
75662306a36Sopenharmony_ci	{0x36, 0xf8},	/* aref3 */
75762306a36Sopenharmony_ci	{0x69, 0x02},	/* hv */
75862306a36Sopenharmony_ci	{0x8c, 0x89},	/* com22 */
75962306a36Sopenharmony_ci	{0x14, 0x28},	/* com9 */
76062306a36Sopenharmony_ci	{0x3e, 0x0c},	/* com14 */
76162306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
76262306a36Sopenharmony_ci	{0x72, 0x00},
76362306a36Sopenharmony_ci	{0x73, 0x01},
76462306a36Sopenharmony_ci	{0x74, 0x3a},
76562306a36Sopenharmony_ci	{0x75, 0x35},
76662306a36Sopenharmony_ci	{0x76, 0x01},
76762306a36Sopenharmony_ci	{0xc7, 0x80},	/* com24 */
76862306a36Sopenharmony_ci	{0x03, 0x1b},	/* vref */
76962306a36Sopenharmony_ci	{0x17, 0x1d},	/* hstart */
77062306a36Sopenharmony_ci	{0x18, 0xbd},	/* hstop */
77162306a36Sopenharmony_ci	{0x19, 0x01},	/* vstrt */
77262306a36Sopenharmony_ci	{0x1a, 0x81},	/* vstop */
77362306a36Sopenharmony_ci	{0x32, 0xff},	/* href */
77462306a36Sopenharmony_ci	{0xc0, 0xe2},
77562306a36Sopenharmony_ci};
77662306a36Sopenharmony_ci
77762306a36Sopenharmony_cistatic const u8 ov965x_start_1_sxga[][2] = {
77862306a36Sopenharmony_ci	{0x12, 0x02},	/* com7 */
77962306a36Sopenharmony_ci	{0x36, 0xf8},	/* aref3 */
78062306a36Sopenharmony_ci	{0x69, 0x02},	/* hv */
78162306a36Sopenharmony_ci	{0x8c, 0x89},	/* com22 */
78262306a36Sopenharmony_ci	{0x14, 0x28},	/* com9 */
78362306a36Sopenharmony_ci	{0x3e, 0x0c},	/* com14 */
78462306a36Sopenharmony_ci	{0x41, 0x40},	/* com16 */
78562306a36Sopenharmony_ci	{0x72, 0x00},
78662306a36Sopenharmony_ci	{0x73, 0x01},
78762306a36Sopenharmony_ci	{0x74, 0x3a},
78862306a36Sopenharmony_ci	{0x75, 0x35},
78962306a36Sopenharmony_ci	{0x76, 0x01},
79062306a36Sopenharmony_ci	{0xc7, 0x80},	/* com24 */
79162306a36Sopenharmony_ci	{0x03, 0x1b},	/* vref */
79262306a36Sopenharmony_ci	{0x17, 0x1d},	/* hstart */
79362306a36Sopenharmony_ci	{0x18, 0x02},	/* hstop */
79462306a36Sopenharmony_ci	{0x19, 0x01},	/* vstrt */
79562306a36Sopenharmony_ci	{0x1a, 0x81},	/* vstop */
79662306a36Sopenharmony_ci	{0x32, 0xff},	/* href */
79762306a36Sopenharmony_ci	{0xc0, 0xe2},
79862306a36Sopenharmony_ci};
79962306a36Sopenharmony_ci
80062306a36Sopenharmony_cistatic const u8 bridge_start_qvga[][2] = {
80162306a36Sopenharmony_ci	{0x94, 0xaa},
80262306a36Sopenharmony_ci	{0xf1, 0x60},
80362306a36Sopenharmony_ci	{0xe5, 0x04},
80462306a36Sopenharmony_ci	{0xc0, 0x50},
80562306a36Sopenharmony_ci	{0xc1, 0x3c},
80662306a36Sopenharmony_ci	{0x8c, 0x00},
80762306a36Sopenharmony_ci	{0x8d, 0x1c},
80862306a36Sopenharmony_ci	{0x34, 0x05},
80962306a36Sopenharmony_ci
81062306a36Sopenharmony_ci	{0xc2, 0x4c},
81162306a36Sopenharmony_ci	{0xc3, 0xf9},
81262306a36Sopenharmony_ci	{0xda, 0x00},
81362306a36Sopenharmony_ci	{0x50, 0x00},
81462306a36Sopenharmony_ci	{0x51, 0xa0},
81562306a36Sopenharmony_ci	{0x52, 0x78},
81662306a36Sopenharmony_ci	{0x53, 0x00},
81762306a36Sopenharmony_ci	{0x54, 0x00},
81862306a36Sopenharmony_ci	{0x55, 0x00},
81962306a36Sopenharmony_ci	{0x57, 0x00},
82062306a36Sopenharmony_ci	{0x5c, 0x00},
82162306a36Sopenharmony_ci	{0x5a, 0x50},
82262306a36Sopenharmony_ci	{0x5b, 0x3c},
82362306a36Sopenharmony_ci	{0x35, 0x02},
82462306a36Sopenharmony_ci	{0xd9, 0x10},
82562306a36Sopenharmony_ci	{0x94, 0x11},
82662306a36Sopenharmony_ci};
82762306a36Sopenharmony_ci
82862306a36Sopenharmony_cistatic const u8 bridge_start_vga[][2] = {
82962306a36Sopenharmony_ci	{0x94, 0xaa},
83062306a36Sopenharmony_ci	{0xf1, 0x60},
83162306a36Sopenharmony_ci	{0xe5, 0x04},
83262306a36Sopenharmony_ci	{0xc0, 0x50},
83362306a36Sopenharmony_ci	{0xc1, 0x3c},
83462306a36Sopenharmony_ci	{0x8c, 0x00},
83562306a36Sopenharmony_ci	{0x8d, 0x1c},
83662306a36Sopenharmony_ci	{0x34, 0x05},
83762306a36Sopenharmony_ci	{0xc2, 0x0c},
83862306a36Sopenharmony_ci	{0xc3, 0xf9},
83962306a36Sopenharmony_ci	{0xda, 0x01},
84062306a36Sopenharmony_ci	{0x50, 0x00},
84162306a36Sopenharmony_ci	{0x51, 0xa0},
84262306a36Sopenharmony_ci	{0x52, 0x3c},
84362306a36Sopenharmony_ci	{0x53, 0x00},
84462306a36Sopenharmony_ci	{0x54, 0x00},
84562306a36Sopenharmony_ci	{0x55, 0x00},
84662306a36Sopenharmony_ci	{0x57, 0x00},
84762306a36Sopenharmony_ci	{0x5c, 0x00},
84862306a36Sopenharmony_ci	{0x5a, 0xa0},
84962306a36Sopenharmony_ci	{0x5b, 0x78},
85062306a36Sopenharmony_ci	{0x35, 0x02},
85162306a36Sopenharmony_ci	{0xd9, 0x10},
85262306a36Sopenharmony_ci	{0x94, 0x11},
85362306a36Sopenharmony_ci};
85462306a36Sopenharmony_ci
85562306a36Sopenharmony_cistatic const u8 bridge_start_svga[][2] = {
85662306a36Sopenharmony_ci	{0x94, 0xaa},
85762306a36Sopenharmony_ci	{0xf1, 0x60},
85862306a36Sopenharmony_ci	{0xe5, 0x04},
85962306a36Sopenharmony_ci	{0xc0, 0xa0},
86062306a36Sopenharmony_ci	{0xc1, 0x80},
86162306a36Sopenharmony_ci	{0x8c, 0x00},
86262306a36Sopenharmony_ci	{0x8d, 0x1c},
86362306a36Sopenharmony_ci	{0x34, 0x05},
86462306a36Sopenharmony_ci	{0xc2, 0x4c},
86562306a36Sopenharmony_ci	{0xc3, 0xf9},
86662306a36Sopenharmony_ci	{0x50, 0x00},
86762306a36Sopenharmony_ci	{0x51, 0x40},
86862306a36Sopenharmony_ci	{0x52, 0x00},
86962306a36Sopenharmony_ci	{0x53, 0x00},
87062306a36Sopenharmony_ci	{0x54, 0x00},
87162306a36Sopenharmony_ci	{0x55, 0x88},
87262306a36Sopenharmony_ci	{0x57, 0x00},
87362306a36Sopenharmony_ci	{0x5c, 0x00},
87462306a36Sopenharmony_ci	{0x5a, 0xc8},
87562306a36Sopenharmony_ci	{0x5b, 0x96},
87662306a36Sopenharmony_ci	{0x35, 0x02},
87762306a36Sopenharmony_ci	{0xd9, 0x10},
87862306a36Sopenharmony_ci	{0xda, 0x00},
87962306a36Sopenharmony_ci	{0x94, 0x11},
88062306a36Sopenharmony_ci};
88162306a36Sopenharmony_ci
88262306a36Sopenharmony_cistatic const u8 bridge_start_xga[][2] = {
88362306a36Sopenharmony_ci	{0x94, 0xaa},
88462306a36Sopenharmony_ci	{0xf1, 0x60},
88562306a36Sopenharmony_ci	{0xe5, 0x04},
88662306a36Sopenharmony_ci	{0xc0, 0xa0},
88762306a36Sopenharmony_ci	{0xc1, 0x80},
88862306a36Sopenharmony_ci	{0x8c, 0x00},
88962306a36Sopenharmony_ci	{0x8d, 0x1c},
89062306a36Sopenharmony_ci	{0x34, 0x05},
89162306a36Sopenharmony_ci	{0xc2, 0x4c},
89262306a36Sopenharmony_ci	{0xc3, 0xf9},
89362306a36Sopenharmony_ci	{0x50, 0x00},
89462306a36Sopenharmony_ci	{0x51, 0x40},
89562306a36Sopenharmony_ci	{0x52, 0x00},
89662306a36Sopenharmony_ci	{0x53, 0x00},
89762306a36Sopenharmony_ci	{0x54, 0x00},
89862306a36Sopenharmony_ci	{0x55, 0x88},
89962306a36Sopenharmony_ci	{0x57, 0x00},
90062306a36Sopenharmony_ci	{0x5c, 0x01},
90162306a36Sopenharmony_ci	{0x5a, 0x00},
90262306a36Sopenharmony_ci	{0x5b, 0xc0},
90362306a36Sopenharmony_ci	{0x35, 0x02},
90462306a36Sopenharmony_ci	{0xd9, 0x10},
90562306a36Sopenharmony_ci	{0xda, 0x01},
90662306a36Sopenharmony_ci	{0x94, 0x11},
90762306a36Sopenharmony_ci};
90862306a36Sopenharmony_ci
90962306a36Sopenharmony_cistatic const u8 bridge_start_sxga[][2] = {
91062306a36Sopenharmony_ci	{0x94, 0xaa},
91162306a36Sopenharmony_ci	{0xf1, 0x60},
91262306a36Sopenharmony_ci	{0xe5, 0x04},
91362306a36Sopenharmony_ci	{0xc0, 0xa0},
91462306a36Sopenharmony_ci	{0xc1, 0x80},
91562306a36Sopenharmony_ci	{0x8c, 0x00},
91662306a36Sopenharmony_ci	{0x8d, 0x1c},
91762306a36Sopenharmony_ci	{0x34, 0x05},
91862306a36Sopenharmony_ci	{0xc2, 0x0c},
91962306a36Sopenharmony_ci	{0xc3, 0xf9},
92062306a36Sopenharmony_ci	{0xda, 0x00},
92162306a36Sopenharmony_ci	{0x35, 0x02},
92262306a36Sopenharmony_ci	{0xd9, 0x10},
92362306a36Sopenharmony_ci	{0x94, 0x11},
92462306a36Sopenharmony_ci};
92562306a36Sopenharmony_ci
92662306a36Sopenharmony_cistatic const u8 ov965x_start_2_qvga[][2] = {
92762306a36Sopenharmony_ci	{0x3b, 0xe4},	/* com11 - night mode 1/4 frame rate */
92862306a36Sopenharmony_ci	{0x1e, 0x04},	/* mvfp */
92962306a36Sopenharmony_ci	{0x13, 0xe0},	/* com8 */
93062306a36Sopenharmony_ci	{0x00, 0x00},
93162306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
93262306a36Sopenharmony_ci	{0x11, 0x01},	/* clkrc */
93362306a36Sopenharmony_ci	{0x6b, 0x5a},	/* dblv */
93462306a36Sopenharmony_ci	{0x6a, 0x02},	/* 50 Hz banding filter */
93562306a36Sopenharmony_ci	{0xc5, 0x03},	/* 60 Hz banding filter */
93662306a36Sopenharmony_ci	{0xa2, 0x96},	/* bd50 */
93762306a36Sopenharmony_ci	{0xa3, 0x7d},	/* bd60 */
93862306a36Sopenharmony_ci
93962306a36Sopenharmony_ci	{0xff, 0x13},	/* read 13, write ff 00 */
94062306a36Sopenharmony_ci	{0x13, 0xe7},
94162306a36Sopenharmony_ci	{0x3a, 0x80},	/* tslb - yuyv */
94262306a36Sopenharmony_ci};
94362306a36Sopenharmony_ci
94462306a36Sopenharmony_cistatic const u8 ov965x_start_2_vga[][2] = {
94562306a36Sopenharmony_ci	{0x3b, 0xc4},	/* com11 - night mode 1/4 frame rate */
94662306a36Sopenharmony_ci	{0x1e, 0x04},	/* mvfp */
94762306a36Sopenharmony_ci	{0x13, 0xe0},	/* com8 */
94862306a36Sopenharmony_ci	{0x00, 0x00},
94962306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
95062306a36Sopenharmony_ci	{0x11, 0x03},	/* clkrc */
95162306a36Sopenharmony_ci	{0x6b, 0x5a},	/* dblv */
95262306a36Sopenharmony_ci	{0x6a, 0x05},	/* 50 Hz banding filter */
95362306a36Sopenharmony_ci	{0xc5, 0x07},	/* 60 Hz banding filter */
95462306a36Sopenharmony_ci	{0xa2, 0x4b},	/* bd50 */
95562306a36Sopenharmony_ci	{0xa3, 0x3e},	/* bd60 */
95662306a36Sopenharmony_ci
95762306a36Sopenharmony_ci	{0x2d, 0x00},	/* advfl */
95862306a36Sopenharmony_ci};
95962306a36Sopenharmony_ci
96062306a36Sopenharmony_cistatic const u8 ov965x_start_2_svga[][2] = {	/* same for xga */
96162306a36Sopenharmony_ci	{0x3b, 0xc4},	/* com11 - night mode 1/4 frame rate */
96262306a36Sopenharmony_ci	{0x1e, 0x04},	/* mvfp */
96362306a36Sopenharmony_ci	{0x13, 0xe0},	/* com8 */
96462306a36Sopenharmony_ci	{0x00, 0x00},
96562306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
96662306a36Sopenharmony_ci	{0x11, 0x01},	/* clkrc */
96762306a36Sopenharmony_ci	{0x6b, 0x5a},	/* dblv */
96862306a36Sopenharmony_ci	{0x6a, 0x0c},	/* 50 Hz banding filter */
96962306a36Sopenharmony_ci	{0xc5, 0x0f},	/* 60 Hz banding filter */
97062306a36Sopenharmony_ci	{0xa2, 0x4e},	/* bd50 */
97162306a36Sopenharmony_ci	{0xa3, 0x41},	/* bd60 */
97262306a36Sopenharmony_ci};
97362306a36Sopenharmony_ci
97462306a36Sopenharmony_cistatic const u8 ov965x_start_2_sxga[][2] = {
97562306a36Sopenharmony_ci	{0x13, 0xe0},	/* com8 */
97662306a36Sopenharmony_ci	{0x00, 0x00},
97762306a36Sopenharmony_ci	{0x13, 0xe7},	/* com8 - everything (AGC, AWB and AEC) */
97862306a36Sopenharmony_ci	{0x3b, 0xc4},	/* com11 - night mode 1/4 frame rate */
97962306a36Sopenharmony_ci	{0x1e, 0x04},	/* mvfp */
98062306a36Sopenharmony_ci	{0x11, 0x01},	/* clkrc */
98162306a36Sopenharmony_ci	{0x6b, 0x5a},	/* dblv */
98262306a36Sopenharmony_ci	{0x6a, 0x0c},	/* 50 Hz banding filter */
98362306a36Sopenharmony_ci	{0xc5, 0x0f},	/* 60 Hz banding filter */
98462306a36Sopenharmony_ci	{0xa2, 0x4e},	/* bd50 */
98562306a36Sopenharmony_ci	{0xa3, 0x41},	/* bd60 */
98662306a36Sopenharmony_ci};
98762306a36Sopenharmony_ci
98862306a36Sopenharmony_cistatic const u8 ov562x_init[][2] = {
98962306a36Sopenharmony_ci	{0x88, 0x20},
99062306a36Sopenharmony_ci	{0x89, 0x0a},
99162306a36Sopenharmony_ci	{0x8a, 0x90},
99262306a36Sopenharmony_ci	{0x8b, 0x06},
99362306a36Sopenharmony_ci	{0x8c, 0x01},
99462306a36Sopenharmony_ci	{0x8d, 0x10},
99562306a36Sopenharmony_ci	{0x1c, 0x00},
99662306a36Sopenharmony_ci	{0x1d, 0x48},
99762306a36Sopenharmony_ci	{0x1d, 0x00},
99862306a36Sopenharmony_ci	{0x1d, 0xff},
99962306a36Sopenharmony_ci	{0x1c, 0x0a},
100062306a36Sopenharmony_ci	{0x1d, 0x2e},
100162306a36Sopenharmony_ci	{0x1d, 0x1e},
100262306a36Sopenharmony_ci};
100362306a36Sopenharmony_ci
100462306a36Sopenharmony_cistatic const u8 ov562x_init_2[][2] = {
100562306a36Sopenharmony_ci	{0x12, 0x80},
100662306a36Sopenharmony_ci	{0x11, 0x41},
100762306a36Sopenharmony_ci	{0x13, 0x00},
100862306a36Sopenharmony_ci	{0x10, 0x1e},
100962306a36Sopenharmony_ci	{0x3b, 0x07},
101062306a36Sopenharmony_ci	{0x5b, 0x40},
101162306a36Sopenharmony_ci	{0x39, 0x07},
101262306a36Sopenharmony_ci	{0x53, 0x02},
101362306a36Sopenharmony_ci	{0x54, 0x60},
101462306a36Sopenharmony_ci	{0x04, 0x20},
101562306a36Sopenharmony_ci	{0x27, 0x04},
101662306a36Sopenharmony_ci	{0x3d, 0x40},
101762306a36Sopenharmony_ci	{0x36, 0x00},
101862306a36Sopenharmony_ci	{0xc5, 0x04},
101962306a36Sopenharmony_ci	{0x4e, 0x00},
102062306a36Sopenharmony_ci	{0x4f, 0x93},
102162306a36Sopenharmony_ci	{0x50, 0x7b},
102262306a36Sopenharmony_ci	{0xca, 0x0c},
102362306a36Sopenharmony_ci	{0xcb, 0x0f},
102462306a36Sopenharmony_ci	{0x39, 0x07},
102562306a36Sopenharmony_ci	{0x4a, 0x10},
102662306a36Sopenharmony_ci	{0x3e, 0x0a},
102762306a36Sopenharmony_ci	{0x3d, 0x00},
102862306a36Sopenharmony_ci	{0x0c, 0x38},
102962306a36Sopenharmony_ci	{0x38, 0x90},
103062306a36Sopenharmony_ci	{0x46, 0x30},
103162306a36Sopenharmony_ci	{0x4f, 0x93},
103262306a36Sopenharmony_ci	{0x50, 0x7b},
103362306a36Sopenharmony_ci	{0xab, 0x00},
103462306a36Sopenharmony_ci	{0xca, 0x0c},
103562306a36Sopenharmony_ci	{0xcb, 0x0f},
103662306a36Sopenharmony_ci	{0x37, 0x02},
103762306a36Sopenharmony_ci	{0x44, 0x48},
103862306a36Sopenharmony_ci	{0x8d, 0x44},
103962306a36Sopenharmony_ci	{0x2a, 0x00},
104062306a36Sopenharmony_ci	{0x2b, 0x00},
104162306a36Sopenharmony_ci	{0x32, 0x00},
104262306a36Sopenharmony_ci	{0x38, 0x90},
104362306a36Sopenharmony_ci	{0x53, 0x02},
104462306a36Sopenharmony_ci	{0x54, 0x60},
104562306a36Sopenharmony_ci	{0x12, 0x00},
104662306a36Sopenharmony_ci	{0x17, 0x12},
104762306a36Sopenharmony_ci	{0x18, 0xb4},
104862306a36Sopenharmony_ci	{0x19, 0x0c},
104962306a36Sopenharmony_ci	{0x1a, 0xf4},
105062306a36Sopenharmony_ci	{0x03, 0x4a},
105162306a36Sopenharmony_ci	{0x89, 0x20},
105262306a36Sopenharmony_ci	{0x83, 0x80},
105362306a36Sopenharmony_ci	{0xb7, 0x9d},
105462306a36Sopenharmony_ci	{0xb6, 0x11},
105562306a36Sopenharmony_ci	{0xb5, 0x55},
105662306a36Sopenharmony_ci	{0xb4, 0x00},
105762306a36Sopenharmony_ci	{0xa9, 0xf0},
105862306a36Sopenharmony_ci	{0xa8, 0x0a},
105962306a36Sopenharmony_ci	{0xb8, 0xf0},
106062306a36Sopenharmony_ci	{0xb9, 0xf0},
106162306a36Sopenharmony_ci	{0xba, 0xf0},
106262306a36Sopenharmony_ci	{0x81, 0x07},
106362306a36Sopenharmony_ci	{0x63, 0x44},
106462306a36Sopenharmony_ci	{0x13, 0xc7},
106562306a36Sopenharmony_ci	{0x14, 0x60},
106662306a36Sopenharmony_ci	{0x33, 0x75},
106762306a36Sopenharmony_ci	{0x2c, 0x00},
106862306a36Sopenharmony_ci	{0x09, 0x00},
106962306a36Sopenharmony_ci	{0x35, 0x30},
107062306a36Sopenharmony_ci	{0x27, 0x04},
107162306a36Sopenharmony_ci	{0x3c, 0x07},
107262306a36Sopenharmony_ci	{0x3a, 0x0a},
107362306a36Sopenharmony_ci	{0x3b, 0x07},
107462306a36Sopenharmony_ci	{0x01, 0x40},
107562306a36Sopenharmony_ci	{0x02, 0x40},
107662306a36Sopenharmony_ci	{0x16, 0x40},
107762306a36Sopenharmony_ci	{0x52, 0xb0},
107862306a36Sopenharmony_ci	{0x51, 0x83},
107962306a36Sopenharmony_ci	{0x21, 0xbb},
108062306a36Sopenharmony_ci	{0x22, 0x10},
108162306a36Sopenharmony_ci	{0x23, 0x03},
108262306a36Sopenharmony_ci	{0x35, 0x38},
108362306a36Sopenharmony_ci	{0x20, 0x90},
108462306a36Sopenharmony_ci	{0x28, 0x30},
108562306a36Sopenharmony_ci	{0x73, 0xe1},
108662306a36Sopenharmony_ci	{0x6c, 0x00},
108762306a36Sopenharmony_ci	{0x6d, 0x80},
108862306a36Sopenharmony_ci	{0x6e, 0x00},
108962306a36Sopenharmony_ci	{0x70, 0x04},
109062306a36Sopenharmony_ci	{0x71, 0x00},
109162306a36Sopenharmony_ci	{0x8d, 0x04},
109262306a36Sopenharmony_ci	{0x64, 0x00},
109362306a36Sopenharmony_ci	{0x65, 0x00},
109462306a36Sopenharmony_ci	{0x66, 0x00},
109562306a36Sopenharmony_ci	{0x67, 0x00},
109662306a36Sopenharmony_ci	{0x68, 0x00},
109762306a36Sopenharmony_ci	{0x69, 0x00},
109862306a36Sopenharmony_ci	{0x6a, 0x00},
109962306a36Sopenharmony_ci	{0x6b, 0x00},
110062306a36Sopenharmony_ci	{0x71, 0x94},
110162306a36Sopenharmony_ci	{0x74, 0x20},
110262306a36Sopenharmony_ci	{0x80, 0x09},
110362306a36Sopenharmony_ci	{0x85, 0xc0},
110462306a36Sopenharmony_ci};
110562306a36Sopenharmony_ci
110662306a36Sopenharmony_cistatic void reg_w_i(struct gspca_dev *gspca_dev, u16 reg, u8 val)
110762306a36Sopenharmony_ci{
110862306a36Sopenharmony_ci	struct usb_device *udev = gspca_dev->dev;
110962306a36Sopenharmony_ci	int ret;
111062306a36Sopenharmony_ci
111162306a36Sopenharmony_ci	if (gspca_dev->usb_err < 0)
111262306a36Sopenharmony_ci		return;
111362306a36Sopenharmony_ci	gspca_dev->usb_buf[0] = val;
111462306a36Sopenharmony_ci	ret = usb_control_msg(udev,
111562306a36Sopenharmony_ci			      usb_sndctrlpipe(udev, 0),
111662306a36Sopenharmony_ci			      0x01,
111762306a36Sopenharmony_ci			      USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
111862306a36Sopenharmony_ci			      0x00, reg, gspca_dev->usb_buf, 1, CTRL_TIMEOUT);
111962306a36Sopenharmony_ci	if (ret < 0) {
112062306a36Sopenharmony_ci		pr_err("reg_w failed %d\n", ret);
112162306a36Sopenharmony_ci		gspca_dev->usb_err = ret;
112262306a36Sopenharmony_ci	}
112362306a36Sopenharmony_ci}
112462306a36Sopenharmony_ci
112562306a36Sopenharmony_cistatic void reg_w(struct gspca_dev *gspca_dev, u16 reg, u8 val)
112662306a36Sopenharmony_ci{
112762306a36Sopenharmony_ci	gspca_dbg(gspca_dev, D_USBO, "reg_w [%04x] = %02x\n", reg, val);
112862306a36Sopenharmony_ci	reg_w_i(gspca_dev, reg, val);
112962306a36Sopenharmony_ci}
113062306a36Sopenharmony_ci
113162306a36Sopenharmony_cistatic u8 reg_r(struct gspca_dev *gspca_dev, u16 reg)
113262306a36Sopenharmony_ci{
113362306a36Sopenharmony_ci	struct usb_device *udev = gspca_dev->dev;
113462306a36Sopenharmony_ci	int ret;
113562306a36Sopenharmony_ci
113662306a36Sopenharmony_ci	if (gspca_dev->usb_err < 0)
113762306a36Sopenharmony_ci		return 0;
113862306a36Sopenharmony_ci	ret = usb_control_msg(udev,
113962306a36Sopenharmony_ci			      usb_rcvctrlpipe(udev, 0),
114062306a36Sopenharmony_ci			      0x01,
114162306a36Sopenharmony_ci			      USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
114262306a36Sopenharmony_ci			      0x00, reg, gspca_dev->usb_buf, 1, CTRL_TIMEOUT);
114362306a36Sopenharmony_ci	gspca_dbg(gspca_dev, D_USBI, "reg_r [%04x] -> %02x\n",
114462306a36Sopenharmony_ci		  reg, gspca_dev->usb_buf[0]);
114562306a36Sopenharmony_ci	if (ret < 0) {
114662306a36Sopenharmony_ci		pr_err("reg_r err %d\n", ret);
114762306a36Sopenharmony_ci		gspca_dev->usb_err = ret;
114862306a36Sopenharmony_ci		return 0;
114962306a36Sopenharmony_ci	}
115062306a36Sopenharmony_ci	return gspca_dev->usb_buf[0];
115162306a36Sopenharmony_ci}
115262306a36Sopenharmony_ci
115362306a36Sopenharmony_cistatic int sccb_check_status(struct gspca_dev *gspca_dev)
115462306a36Sopenharmony_ci{
115562306a36Sopenharmony_ci	u8 data;
115662306a36Sopenharmony_ci	int i;
115762306a36Sopenharmony_ci
115862306a36Sopenharmony_ci	for (i = 0; i < 5; i++) {
115962306a36Sopenharmony_ci		msleep(20);
116062306a36Sopenharmony_ci		data = reg_r(gspca_dev, OV534_REG_STATUS);
116162306a36Sopenharmony_ci
116262306a36Sopenharmony_ci		switch (data) {
116362306a36Sopenharmony_ci		case 0x00:
116462306a36Sopenharmony_ci			return 1;
116562306a36Sopenharmony_ci		case 0x04:
116662306a36Sopenharmony_ci			return 0;
116762306a36Sopenharmony_ci		case 0x03:
116862306a36Sopenharmony_ci			break;
116962306a36Sopenharmony_ci		default:
117062306a36Sopenharmony_ci			gspca_dbg(gspca_dev, D_USBI|D_USBO,
117162306a36Sopenharmony_ci				  "sccb status 0x%02x, attempt %d/5\n",
117262306a36Sopenharmony_ci				  data, i + 1);
117362306a36Sopenharmony_ci		}
117462306a36Sopenharmony_ci	}
117562306a36Sopenharmony_ci	return 0;
117662306a36Sopenharmony_ci}
117762306a36Sopenharmony_ci
117862306a36Sopenharmony_cistatic void sccb_write(struct gspca_dev *gspca_dev, u8 reg, u8 val)
117962306a36Sopenharmony_ci{
118062306a36Sopenharmony_ci	gspca_dbg(gspca_dev, D_USBO, "sccb_write [%02x] = %02x\n", reg, val);
118162306a36Sopenharmony_ci	reg_w_i(gspca_dev, OV534_REG_SUBADDR, reg);
118262306a36Sopenharmony_ci	reg_w_i(gspca_dev, OV534_REG_WRITE, val);
118362306a36Sopenharmony_ci	reg_w_i(gspca_dev, OV534_REG_OPERATION, OV534_OP_WRITE_3);
118462306a36Sopenharmony_ci
118562306a36Sopenharmony_ci	if (!sccb_check_status(gspca_dev))
118662306a36Sopenharmony_ci		pr_err("sccb_write failed\n");
118762306a36Sopenharmony_ci}
118862306a36Sopenharmony_ci
118962306a36Sopenharmony_cistatic u8 sccb_read(struct gspca_dev *gspca_dev, u16 reg)
119062306a36Sopenharmony_ci{
119162306a36Sopenharmony_ci	reg_w(gspca_dev, OV534_REG_SUBADDR, reg);
119262306a36Sopenharmony_ci	reg_w(gspca_dev, OV534_REG_OPERATION, OV534_OP_WRITE_2);
119362306a36Sopenharmony_ci	if (!sccb_check_status(gspca_dev))
119462306a36Sopenharmony_ci		pr_err("sccb_read failed 1\n");
119562306a36Sopenharmony_ci
119662306a36Sopenharmony_ci	reg_w(gspca_dev, OV534_REG_OPERATION, OV534_OP_READ_2);
119762306a36Sopenharmony_ci	if (!sccb_check_status(gspca_dev))
119862306a36Sopenharmony_ci		pr_err("sccb_read failed 2\n");
119962306a36Sopenharmony_ci
120062306a36Sopenharmony_ci	return reg_r(gspca_dev, OV534_REG_READ);
120162306a36Sopenharmony_ci}
120262306a36Sopenharmony_ci
120362306a36Sopenharmony_ci/* output a bridge sequence (reg - val) */
120462306a36Sopenharmony_cistatic void reg_w_array(struct gspca_dev *gspca_dev,
120562306a36Sopenharmony_ci			const u8 (*data)[2], int len)
120662306a36Sopenharmony_ci{
120762306a36Sopenharmony_ci	while (--len >= 0) {
120862306a36Sopenharmony_ci		reg_w(gspca_dev, (*data)[0], (*data)[1]);
120962306a36Sopenharmony_ci		data++;
121062306a36Sopenharmony_ci	}
121162306a36Sopenharmony_ci}
121262306a36Sopenharmony_ci
121362306a36Sopenharmony_ci/* output a sensor sequence (reg - val) */
121462306a36Sopenharmony_cistatic void sccb_w_array(struct gspca_dev *gspca_dev,
121562306a36Sopenharmony_ci			const u8 (*data)[2], int len)
121662306a36Sopenharmony_ci{
121762306a36Sopenharmony_ci	while (--len >= 0) {
121862306a36Sopenharmony_ci		if ((*data)[0] != 0xff) {
121962306a36Sopenharmony_ci			sccb_write(gspca_dev, (*data)[0], (*data)[1]);
122062306a36Sopenharmony_ci		} else {
122162306a36Sopenharmony_ci			sccb_read(gspca_dev, (*data)[1]);
122262306a36Sopenharmony_ci			sccb_write(gspca_dev, 0xff, 0x00);
122362306a36Sopenharmony_ci		}
122462306a36Sopenharmony_ci		data++;
122562306a36Sopenharmony_ci	}
122662306a36Sopenharmony_ci}
122762306a36Sopenharmony_ci
122862306a36Sopenharmony_ci/* Two bits control LED: 0x21 bit 7 and 0x23 bit 7.
122962306a36Sopenharmony_ci * (direction and output)? */
123062306a36Sopenharmony_cistatic void set_led(struct gspca_dev *gspca_dev, int status)
123162306a36Sopenharmony_ci{
123262306a36Sopenharmony_ci	u8 data;
123362306a36Sopenharmony_ci
123462306a36Sopenharmony_ci	gspca_dbg(gspca_dev, D_CONF, "led status: %d\n", status);
123562306a36Sopenharmony_ci
123662306a36Sopenharmony_ci	data = reg_r(gspca_dev, 0x21);
123762306a36Sopenharmony_ci	data |= 0x80;
123862306a36Sopenharmony_ci	reg_w(gspca_dev, 0x21, data);
123962306a36Sopenharmony_ci
124062306a36Sopenharmony_ci	data = reg_r(gspca_dev, 0x23);
124162306a36Sopenharmony_ci	if (status)
124262306a36Sopenharmony_ci		data |= 0x80;
124362306a36Sopenharmony_ci	else
124462306a36Sopenharmony_ci		data &= ~0x80;
124562306a36Sopenharmony_ci
124662306a36Sopenharmony_ci	reg_w(gspca_dev, 0x23, data);
124762306a36Sopenharmony_ci
124862306a36Sopenharmony_ci	if (!status) {
124962306a36Sopenharmony_ci		data = reg_r(gspca_dev, 0x21);
125062306a36Sopenharmony_ci		data &= ~0x80;
125162306a36Sopenharmony_ci		reg_w(gspca_dev, 0x21, data);
125262306a36Sopenharmony_ci	}
125362306a36Sopenharmony_ci}
125462306a36Sopenharmony_ci
125562306a36Sopenharmony_cistatic void setbrightness(struct gspca_dev *gspca_dev, s32 brightness)
125662306a36Sopenharmony_ci{
125762306a36Sopenharmony_ci	struct sd *sd = (struct sd *) gspca_dev;
125862306a36Sopenharmony_ci	u8 val;
125962306a36Sopenharmony_ci	s8 sval;
126062306a36Sopenharmony_ci
126162306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV562x) {
126262306a36Sopenharmony_ci		sval = brightness;
126362306a36Sopenharmony_ci		val = 0x76;
126462306a36Sopenharmony_ci		val += sval;
126562306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x24, val);
126662306a36Sopenharmony_ci		val = 0x6a;
126762306a36Sopenharmony_ci		val += sval;
126862306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x25, val);
126962306a36Sopenharmony_ci		if (sval < -40)
127062306a36Sopenharmony_ci			val = 0x71;
127162306a36Sopenharmony_ci		else if (sval < 20)
127262306a36Sopenharmony_ci			val = 0x94;
127362306a36Sopenharmony_ci		else
127462306a36Sopenharmony_ci			val = 0xe6;
127562306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x26, val);
127662306a36Sopenharmony_ci	} else {
127762306a36Sopenharmony_ci		val = brightness;
127862306a36Sopenharmony_ci		if (val < 8)
127962306a36Sopenharmony_ci			val = 15 - val;		/* f .. 8 */
128062306a36Sopenharmony_ci		else
128162306a36Sopenharmony_ci			val = val - 8;		/* 0 .. 7 */
128262306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x55,	/* brtn - brightness adjustment */
128362306a36Sopenharmony_ci				0x0f | (val << 4));
128462306a36Sopenharmony_ci	}
128562306a36Sopenharmony_ci}
128662306a36Sopenharmony_ci
128762306a36Sopenharmony_cistatic void setcontrast(struct gspca_dev *gspca_dev, s32 val)
128862306a36Sopenharmony_ci{
128962306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x56,	/* cnst1 - contrast 1 ctrl coeff */
129062306a36Sopenharmony_ci			val << 4);
129162306a36Sopenharmony_ci}
129262306a36Sopenharmony_ci
129362306a36Sopenharmony_cistatic void setautogain(struct gspca_dev *gspca_dev, s32 autogain)
129462306a36Sopenharmony_ci{
129562306a36Sopenharmony_ci	u8 val;
129662306a36Sopenharmony_ci
129762306a36Sopenharmony_ci/*fixme: should adjust agc/awb/aec by different controls */
129862306a36Sopenharmony_ci	val = sccb_read(gspca_dev, 0x13);		/* com8 */
129962306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
130062306a36Sopenharmony_ci	if (autogain)
130162306a36Sopenharmony_ci		val |= 0x05;		/* agc & aec */
130262306a36Sopenharmony_ci	else
130362306a36Sopenharmony_ci		val &= 0xfa;
130462306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x13, val);
130562306a36Sopenharmony_ci}
130662306a36Sopenharmony_ci
130762306a36Sopenharmony_cistatic void setexposure(struct gspca_dev *gspca_dev, s32 exposure)
130862306a36Sopenharmony_ci{
130962306a36Sopenharmony_ci	static const u8 expo[4] = {0x00, 0x25, 0x38, 0x5e};
131062306a36Sopenharmony_ci	u8 val;
131162306a36Sopenharmony_ci
131262306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x10, expo[exposure]);	/* aec[9:2] */
131362306a36Sopenharmony_ci
131462306a36Sopenharmony_ci	val = sccb_read(gspca_dev, 0x13);		/* com8 */
131562306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
131662306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x13, val);
131762306a36Sopenharmony_ci
131862306a36Sopenharmony_ci	val = sccb_read(gspca_dev, 0xa1);		/* aech */
131962306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
132062306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xa1, val & 0xe0);	/* aec[15:10] = 0 */
132162306a36Sopenharmony_ci}
132262306a36Sopenharmony_ci
132362306a36Sopenharmony_cistatic void setsharpness(struct gspca_dev *gspca_dev, s32 val)
132462306a36Sopenharmony_ci{
132562306a36Sopenharmony_ci	if (val < 0) {				/* auto */
132662306a36Sopenharmony_ci		val = sccb_read(gspca_dev, 0x42);	/* com17 */
132762306a36Sopenharmony_ci		sccb_write(gspca_dev, 0xff, 0x00);
132862306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x42, val | 0x40);
132962306a36Sopenharmony_ci				/* Edge enhancement strength auto adjust */
133062306a36Sopenharmony_ci		return;
133162306a36Sopenharmony_ci	}
133262306a36Sopenharmony_ci	if (val != 0)
133362306a36Sopenharmony_ci		val = 1 << (val - 1);
133462306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x3f,	/* edge - edge enhance. factor */
133562306a36Sopenharmony_ci			val);
133662306a36Sopenharmony_ci	val = sccb_read(gspca_dev, 0x42);		/* com17 */
133762306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
133862306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x42, val & 0xbf);
133962306a36Sopenharmony_ci}
134062306a36Sopenharmony_ci
134162306a36Sopenharmony_cistatic void setsatur(struct gspca_dev *gspca_dev, s32 val)
134262306a36Sopenharmony_ci{
134362306a36Sopenharmony_ci	u8 val1, val2, val3;
134462306a36Sopenharmony_ci	static const u8 matrix[5][2] = {
134562306a36Sopenharmony_ci		{0x14, 0x38},
134662306a36Sopenharmony_ci		{0x1e, 0x54},
134762306a36Sopenharmony_ci		{0x28, 0x70},
134862306a36Sopenharmony_ci		{0x32, 0x8c},
134962306a36Sopenharmony_ci		{0x48, 0x90}
135062306a36Sopenharmony_ci	};
135162306a36Sopenharmony_ci
135262306a36Sopenharmony_ci	val1 = matrix[val][0];
135362306a36Sopenharmony_ci	val2 = matrix[val][1];
135462306a36Sopenharmony_ci	val3 = val1 + val2;
135562306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x4f, val3);	/* matrix coeff */
135662306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x50, val3);
135762306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x51, 0x00);
135862306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x52, val1);
135962306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x53, val2);
136062306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x54, val3);
136162306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x58, 0x1a);	/* mtxs - coeff signs */
136262306a36Sopenharmony_ci
136362306a36Sopenharmony_ci	val1 = sccb_read(gspca_dev, 0x41);	/* com16 */
136462306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
136562306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x41, val1);
136662306a36Sopenharmony_ci}
136762306a36Sopenharmony_ci
136862306a36Sopenharmony_cistatic void setlightfreq(struct gspca_dev *gspca_dev, s32 freq)
136962306a36Sopenharmony_ci{
137062306a36Sopenharmony_ci	u8 val;
137162306a36Sopenharmony_ci
137262306a36Sopenharmony_ci	val = sccb_read(gspca_dev, 0x13);		/* com8 */
137362306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
137462306a36Sopenharmony_ci	if (freq == 0) {
137562306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x13, val & 0xdf);
137662306a36Sopenharmony_ci		return;
137762306a36Sopenharmony_ci	}
137862306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x13, val | 0x20);
137962306a36Sopenharmony_ci
138062306a36Sopenharmony_ci	val = sccb_read(gspca_dev, 0x42);		/* com17 */
138162306a36Sopenharmony_ci	sccb_write(gspca_dev, 0xff, 0x00);
138262306a36Sopenharmony_ci	if (freq == 1)
138362306a36Sopenharmony_ci		val |= 0x01;
138462306a36Sopenharmony_ci	else
138562306a36Sopenharmony_ci		val &= 0xfe;
138662306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x42, val);
138762306a36Sopenharmony_ci}
138862306a36Sopenharmony_ci
138962306a36Sopenharmony_ci/* this function is called at probe time */
139062306a36Sopenharmony_cistatic int sd_config(struct gspca_dev *gspca_dev,
139162306a36Sopenharmony_ci		     const struct usb_device_id *id)
139262306a36Sopenharmony_ci{
139362306a36Sopenharmony_ci	return 0;
139462306a36Sopenharmony_ci}
139562306a36Sopenharmony_ci
139662306a36Sopenharmony_ci/* this function is called at probe and resume time */
139762306a36Sopenharmony_cistatic int sd_init(struct gspca_dev *gspca_dev)
139862306a36Sopenharmony_ci{
139962306a36Sopenharmony_ci	struct sd *sd = (struct sd *) gspca_dev;
140062306a36Sopenharmony_ci	u16 sensor_id;
140162306a36Sopenharmony_ci
140262306a36Sopenharmony_ci	/* reset bridge */
140362306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe7, 0x3a);
140462306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe0, 0x08);
140562306a36Sopenharmony_ci	msleep(100);
140662306a36Sopenharmony_ci
140762306a36Sopenharmony_ci	/* initialize the sensor address */
140862306a36Sopenharmony_ci	reg_w(gspca_dev, OV534_REG_ADDRESS, 0x60);
140962306a36Sopenharmony_ci
141062306a36Sopenharmony_ci	/* reset sensor */
141162306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x12, 0x80);
141262306a36Sopenharmony_ci	msleep(10);
141362306a36Sopenharmony_ci
141462306a36Sopenharmony_ci	/* probe the sensor */
141562306a36Sopenharmony_ci	sccb_read(gspca_dev, 0x0a);
141662306a36Sopenharmony_ci	sensor_id = sccb_read(gspca_dev, 0x0a) << 8;
141762306a36Sopenharmony_ci	sccb_read(gspca_dev, 0x0b);
141862306a36Sopenharmony_ci	sensor_id |= sccb_read(gspca_dev, 0x0b);
141962306a36Sopenharmony_ci	gspca_dbg(gspca_dev, D_PROBE, "Sensor ID: %04x\n", sensor_id);
142062306a36Sopenharmony_ci
142162306a36Sopenharmony_ci	/* initialize */
142262306a36Sopenharmony_ci	if ((sensor_id & 0xfff0) == 0x9650) {
142362306a36Sopenharmony_ci		sd->sensor = SENSOR_OV965x;
142462306a36Sopenharmony_ci
142562306a36Sopenharmony_ci		gspca_dev->cam.cam_mode = ov965x_mode;
142662306a36Sopenharmony_ci		gspca_dev->cam.nmodes = ARRAY_SIZE(ov965x_mode);
142762306a36Sopenharmony_ci
142862306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_init,
142962306a36Sopenharmony_ci				ARRAY_SIZE(bridge_init));
143062306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_init,
143162306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_init));
143262306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_init_2,
143362306a36Sopenharmony_ci				ARRAY_SIZE(bridge_init_2));
143462306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_init_2,
143562306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_init_2));
143662306a36Sopenharmony_ci		reg_w(gspca_dev, 0xe0, 0x00);
143762306a36Sopenharmony_ci		reg_w(gspca_dev, 0xe0, 0x01);
143862306a36Sopenharmony_ci		set_led(gspca_dev, 0);
143962306a36Sopenharmony_ci		reg_w(gspca_dev, 0xe0, 0x00);
144062306a36Sopenharmony_ci	} else if ((sensor_id & 0xfff0) == 0x9710) {
144162306a36Sopenharmony_ci		const char *p;
144262306a36Sopenharmony_ci		int l;
144362306a36Sopenharmony_ci
144462306a36Sopenharmony_ci		sd->sensor = SENSOR_OV971x;
144562306a36Sopenharmony_ci
144662306a36Sopenharmony_ci		gspca_dev->cam.cam_mode = ov971x_mode;
144762306a36Sopenharmony_ci		gspca_dev->cam.nmodes = ARRAY_SIZE(ov971x_mode);
144862306a36Sopenharmony_ci
144962306a36Sopenharmony_ci		gspca_dev->cam.bulk = 1;
145062306a36Sopenharmony_ci		gspca_dev->cam.bulk_size = 16384;
145162306a36Sopenharmony_ci		gspca_dev->cam.bulk_nurbs = 2;
145262306a36Sopenharmony_ci
145362306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov971x_init,
145462306a36Sopenharmony_ci				ARRAY_SIZE(ov971x_init));
145562306a36Sopenharmony_ci
145662306a36Sopenharmony_ci		/* set video format on bridge processor */
145762306a36Sopenharmony_ci		/* access bridge processor's video format registers at: 0x00 */
145862306a36Sopenharmony_ci		reg_w(gspca_dev, 0x1c, 0x00);
145962306a36Sopenharmony_ci		/*set register: 0x00 is 'RAW8', 0x40 is 'YUV422' (YUYV?)*/
146062306a36Sopenharmony_ci		reg_w(gspca_dev, 0x1d, 0x00);
146162306a36Sopenharmony_ci
146262306a36Sopenharmony_ci		/* Will W. specific stuff
146362306a36Sopenharmony_ci		 * set VSYNC to
146462306a36Sopenharmony_ci		 *	output (0x1f) if first webcam
146562306a36Sopenharmony_ci		 *	input (0x17) if 2nd or 3rd webcam */
146662306a36Sopenharmony_ci		p = video_device_node_name(&gspca_dev->vdev);
146762306a36Sopenharmony_ci		l = strlen(p) - 1;
146862306a36Sopenharmony_ci		if (p[l] == '0')
146962306a36Sopenharmony_ci			reg_w(gspca_dev, 0x56, 0x1f);
147062306a36Sopenharmony_ci		else
147162306a36Sopenharmony_ci			reg_w(gspca_dev, 0x56, 0x17);
147262306a36Sopenharmony_ci	} else if ((sensor_id & 0xfff0) == 0x5620) {
147362306a36Sopenharmony_ci		sd->sensor = SENSOR_OV562x;
147462306a36Sopenharmony_ci		gspca_dev->cam.cam_mode = ov562x_mode;
147562306a36Sopenharmony_ci		gspca_dev->cam.nmodes = ARRAY_SIZE(ov562x_mode);
147662306a36Sopenharmony_ci
147762306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov562x_init,
147862306a36Sopenharmony_ci				ARRAY_SIZE(ov562x_init));
147962306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov562x_init_2,
148062306a36Sopenharmony_ci				ARRAY_SIZE(ov562x_init_2));
148162306a36Sopenharmony_ci		reg_w(gspca_dev, 0xe0, 0x00);
148262306a36Sopenharmony_ci	} else if ((sensor_id & 0xfff0) == 0x3610) {
148362306a36Sopenharmony_ci		sd->sensor = SENSOR_OV361x;
148462306a36Sopenharmony_ci		gspca_dev->cam.cam_mode = ov361x_mode;
148562306a36Sopenharmony_ci		gspca_dev->cam.nmodes = ARRAY_SIZE(ov361x_mode);
148662306a36Sopenharmony_ci		reg_w(gspca_dev, 0xe7, 0x3a);
148762306a36Sopenharmony_ci		reg_w(gspca_dev, 0xf1, 0x60);
148862306a36Sopenharmony_ci		sccb_write(gspca_dev, 0x12, 0x80);
148962306a36Sopenharmony_ci	} else {
149062306a36Sopenharmony_ci		pr_err("Unknown sensor %04x", sensor_id);
149162306a36Sopenharmony_ci		return -EINVAL;
149262306a36Sopenharmony_ci	}
149362306a36Sopenharmony_ci
149462306a36Sopenharmony_ci	return gspca_dev->usb_err;
149562306a36Sopenharmony_ci}
149662306a36Sopenharmony_ci
149762306a36Sopenharmony_cistatic int sd_start_ov361x(struct gspca_dev *gspca_dev)
149862306a36Sopenharmony_ci{
149962306a36Sopenharmony_ci	sccb_write(gspca_dev, 0x12, 0x80);
150062306a36Sopenharmony_ci	msleep(20);
150162306a36Sopenharmony_ci	switch (gspca_dev->curr_mode % (ov361x_last)) {
150262306a36Sopenharmony_ci	case ov361x_2048:
150362306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov361x_bridge_start_2048,
150462306a36Sopenharmony_ci			    ARRAY_SIZE(ov361x_bridge_start_2048));
150562306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov361x_start_2048,
150662306a36Sopenharmony_ci			     ARRAY_SIZE(ov361x_start_2048));
150762306a36Sopenharmony_ci		break;
150862306a36Sopenharmony_ci	case ov361x_1600:
150962306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov361x_bridge_start_1600,
151062306a36Sopenharmony_ci			    ARRAY_SIZE(ov361x_bridge_start_1600));
151162306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov361x_start_1600,
151262306a36Sopenharmony_ci			     ARRAY_SIZE(ov361x_start_1600));
151362306a36Sopenharmony_ci		break;
151462306a36Sopenharmony_ci	case ov361x_1024:
151562306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov361x_bridge_start_1024,
151662306a36Sopenharmony_ci			    ARRAY_SIZE(ov361x_bridge_start_1024));
151762306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov361x_start_1024,
151862306a36Sopenharmony_ci			     ARRAY_SIZE(ov361x_start_1024));
151962306a36Sopenharmony_ci		break;
152062306a36Sopenharmony_ci	case ov361x_640:
152162306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov361x_bridge_start_640,
152262306a36Sopenharmony_ci			    ARRAY_SIZE(ov361x_bridge_start_640));
152362306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov361x_start_640,
152462306a36Sopenharmony_ci			     ARRAY_SIZE(ov361x_start_640));
152562306a36Sopenharmony_ci		break;
152662306a36Sopenharmony_ci	case ov361x_320:
152762306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov361x_bridge_start_320,
152862306a36Sopenharmony_ci			    ARRAY_SIZE(ov361x_bridge_start_320));
152962306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov361x_start_320,
153062306a36Sopenharmony_ci			     ARRAY_SIZE(ov361x_start_320));
153162306a36Sopenharmony_ci		break;
153262306a36Sopenharmony_ci	case ov361x_160:
153362306a36Sopenharmony_ci		reg_w_array(gspca_dev, ov361x_bridge_start_160,
153462306a36Sopenharmony_ci			    ARRAY_SIZE(ov361x_bridge_start_160));
153562306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov361x_start_160,
153662306a36Sopenharmony_ci			     ARRAY_SIZE(ov361x_start_160));
153762306a36Sopenharmony_ci		break;
153862306a36Sopenharmony_ci	}
153962306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe0, 0x00); /* start transfer */
154062306a36Sopenharmony_ci
154162306a36Sopenharmony_ci	return gspca_dev->usb_err;
154262306a36Sopenharmony_ci}
154362306a36Sopenharmony_ci
154462306a36Sopenharmony_cistatic int sd_start(struct gspca_dev *gspca_dev)
154562306a36Sopenharmony_ci{
154662306a36Sopenharmony_ci	struct sd *sd = (struct sd *) gspca_dev;
154762306a36Sopenharmony_ci
154862306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV971x)
154962306a36Sopenharmony_ci		return gspca_dev->usb_err;
155062306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV562x)
155162306a36Sopenharmony_ci		return gspca_dev->usb_err;
155262306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV361x)
155362306a36Sopenharmony_ci		return sd_start_ov361x(gspca_dev);
155462306a36Sopenharmony_ci
155562306a36Sopenharmony_ci	switch (gspca_dev->curr_mode) {
155662306a36Sopenharmony_ci	case QVGA_MODE:			/* 320x240 */
155762306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_1_vga,
155862306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_1_vga));
155962306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_start_qvga,
156062306a36Sopenharmony_ci				ARRAY_SIZE(bridge_start_qvga));
156162306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_2_qvga,
156262306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_2_qvga));
156362306a36Sopenharmony_ci		break;
156462306a36Sopenharmony_ci	case VGA_MODE:			/* 640x480 */
156562306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_1_vga,
156662306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_1_vga));
156762306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_start_vga,
156862306a36Sopenharmony_ci				ARRAY_SIZE(bridge_start_vga));
156962306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_2_vga,
157062306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_2_vga));
157162306a36Sopenharmony_ci		break;
157262306a36Sopenharmony_ci	case SVGA_MODE:			/* 800x600 */
157362306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_1_svga,
157462306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_1_svga));
157562306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_start_svga,
157662306a36Sopenharmony_ci				ARRAY_SIZE(bridge_start_svga));
157762306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_2_svga,
157862306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_2_svga));
157962306a36Sopenharmony_ci		break;
158062306a36Sopenharmony_ci	case XGA_MODE:			/* 1024x768 */
158162306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_1_xga,
158262306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_1_xga));
158362306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_start_xga,
158462306a36Sopenharmony_ci				ARRAY_SIZE(bridge_start_xga));
158562306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_2_svga,
158662306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_2_svga));
158762306a36Sopenharmony_ci		break;
158862306a36Sopenharmony_ci	default:
158962306a36Sopenharmony_ci/*	case SXGA_MODE:			 * 1280x1024 */
159062306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_1_sxga,
159162306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_1_sxga));
159262306a36Sopenharmony_ci		reg_w_array(gspca_dev, bridge_start_sxga,
159362306a36Sopenharmony_ci				ARRAY_SIZE(bridge_start_sxga));
159462306a36Sopenharmony_ci		sccb_w_array(gspca_dev, ov965x_start_2_sxga,
159562306a36Sopenharmony_ci				ARRAY_SIZE(ov965x_start_2_sxga));
159662306a36Sopenharmony_ci		break;
159762306a36Sopenharmony_ci	}
159862306a36Sopenharmony_ci
159962306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe0, 0x00);
160062306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe0, 0x00);
160162306a36Sopenharmony_ci	set_led(gspca_dev, 1);
160262306a36Sopenharmony_ci	return gspca_dev->usb_err;
160362306a36Sopenharmony_ci}
160462306a36Sopenharmony_ci
160562306a36Sopenharmony_cistatic void sd_stopN(struct gspca_dev *gspca_dev)
160662306a36Sopenharmony_ci{
160762306a36Sopenharmony_ci	if (((struct sd *)gspca_dev)->sensor == SENSOR_OV361x) {
160862306a36Sopenharmony_ci		reg_w(gspca_dev, 0xe0, 0x01); /* stop transfer */
160962306a36Sopenharmony_ci		/* reg_w(gspca_dev, 0x31, 0x09); */
161062306a36Sopenharmony_ci		return;
161162306a36Sopenharmony_ci	}
161262306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe0, 0x01);
161362306a36Sopenharmony_ci	set_led(gspca_dev, 0);
161462306a36Sopenharmony_ci	reg_w(gspca_dev, 0xe0, 0x00);
161562306a36Sopenharmony_ci}
161662306a36Sopenharmony_ci
161762306a36Sopenharmony_ci/* Values for bmHeaderInfo (Video and Still Image Payload Headers, 2.4.3.3) */
161862306a36Sopenharmony_ci#define UVC_STREAM_EOH	(1 << 7)
161962306a36Sopenharmony_ci#define UVC_STREAM_ERR	(1 << 6)
162062306a36Sopenharmony_ci#define UVC_STREAM_STI	(1 << 5)
162162306a36Sopenharmony_ci#define UVC_STREAM_RES	(1 << 4)
162262306a36Sopenharmony_ci#define UVC_STREAM_SCR	(1 << 3)
162362306a36Sopenharmony_ci#define UVC_STREAM_PTS	(1 << 2)
162462306a36Sopenharmony_ci#define UVC_STREAM_EOF	(1 << 1)
162562306a36Sopenharmony_ci#define UVC_STREAM_FID	(1 << 0)
162662306a36Sopenharmony_ci
162762306a36Sopenharmony_cistatic void sd_pkt_scan(struct gspca_dev *gspca_dev,
162862306a36Sopenharmony_ci			u8 *data, int len)
162962306a36Sopenharmony_ci{
163062306a36Sopenharmony_ci	struct sd *sd = (struct sd *) gspca_dev;
163162306a36Sopenharmony_ci	__u32 this_pts;
163262306a36Sopenharmony_ci	u8 this_fid;
163362306a36Sopenharmony_ci	int remaining_len = len;
163462306a36Sopenharmony_ci	int payload_len;
163562306a36Sopenharmony_ci
163662306a36Sopenharmony_ci	payload_len = gspca_dev->cam.bulk ? 2048 : 2040;
163762306a36Sopenharmony_ci	do {
163862306a36Sopenharmony_ci		len = min(remaining_len, payload_len);
163962306a36Sopenharmony_ci
164062306a36Sopenharmony_ci		/* Payloads are prefixed with a UVC-style header.  We
164162306a36Sopenharmony_ci		   consider a frame to start when the FID toggles, or the PTS
164262306a36Sopenharmony_ci		   changes.  A frame ends when EOF is set, and we've received
164362306a36Sopenharmony_ci		   the correct number of bytes. */
164462306a36Sopenharmony_ci
164562306a36Sopenharmony_ci		/* Verify UVC header.  Header length is always 12 */
164662306a36Sopenharmony_ci		if (data[0] != 12 || len < 12) {
164762306a36Sopenharmony_ci			gspca_dbg(gspca_dev, D_PACK, "bad header\n");
164862306a36Sopenharmony_ci			goto discard;
164962306a36Sopenharmony_ci		}
165062306a36Sopenharmony_ci
165162306a36Sopenharmony_ci		/* Check errors */
165262306a36Sopenharmony_ci		if (data[1] & UVC_STREAM_ERR) {
165362306a36Sopenharmony_ci			gspca_dbg(gspca_dev, D_PACK, "payload error\n");
165462306a36Sopenharmony_ci			goto discard;
165562306a36Sopenharmony_ci		}
165662306a36Sopenharmony_ci
165762306a36Sopenharmony_ci		/* Extract PTS and FID */
165862306a36Sopenharmony_ci		if (!(data[1] & UVC_STREAM_PTS)) {
165962306a36Sopenharmony_ci			gspca_dbg(gspca_dev, D_PACK, "PTS not present\n");
166062306a36Sopenharmony_ci			goto discard;
166162306a36Sopenharmony_ci		}
166262306a36Sopenharmony_ci		this_pts = (data[5] << 24) | (data[4] << 16)
166362306a36Sopenharmony_ci						| (data[3] << 8) | data[2];
166462306a36Sopenharmony_ci		this_fid = data[1] & UVC_STREAM_FID;
166562306a36Sopenharmony_ci
166662306a36Sopenharmony_ci		/* If PTS or FID has changed, start a new frame. */
166762306a36Sopenharmony_ci		if (this_pts != sd->last_pts || this_fid != sd->last_fid) {
166862306a36Sopenharmony_ci			if (gspca_dev->last_packet_type == INTER_PACKET)
166962306a36Sopenharmony_ci				gspca_frame_add(gspca_dev, LAST_PACKET,
167062306a36Sopenharmony_ci						NULL, 0);
167162306a36Sopenharmony_ci			sd->last_pts = this_pts;
167262306a36Sopenharmony_ci			sd->last_fid = this_fid;
167362306a36Sopenharmony_ci			gspca_frame_add(gspca_dev, FIRST_PACKET,
167462306a36Sopenharmony_ci					data + 12, len - 12);
167562306a36Sopenharmony_ci		/* If this packet is marked as EOF, end the frame */
167662306a36Sopenharmony_ci		} else if (data[1] & UVC_STREAM_EOF) {
167762306a36Sopenharmony_ci			sd->last_pts = 0;
167862306a36Sopenharmony_ci			gspca_frame_add(gspca_dev, LAST_PACKET,
167962306a36Sopenharmony_ci					data + 12, len - 12);
168062306a36Sopenharmony_ci		} else {
168162306a36Sopenharmony_ci
168262306a36Sopenharmony_ci			/* Add the data from this payload */
168362306a36Sopenharmony_ci			gspca_frame_add(gspca_dev, INTER_PACKET,
168462306a36Sopenharmony_ci					data + 12, len - 12);
168562306a36Sopenharmony_ci		}
168662306a36Sopenharmony_ci
168762306a36Sopenharmony_ci		/* Done this payload */
168862306a36Sopenharmony_ci		goto scan_next;
168962306a36Sopenharmony_ci
169062306a36Sopenharmony_cidiscard:
169162306a36Sopenharmony_ci		/* Discard data until a new frame starts. */
169262306a36Sopenharmony_ci		gspca_dev->last_packet_type = DISCARD_PACKET;
169362306a36Sopenharmony_ci
169462306a36Sopenharmony_ciscan_next:
169562306a36Sopenharmony_ci		remaining_len -= len;
169662306a36Sopenharmony_ci		data += len;
169762306a36Sopenharmony_ci	} while (remaining_len > 0);
169862306a36Sopenharmony_ci}
169962306a36Sopenharmony_ci
170062306a36Sopenharmony_cistatic int sd_s_ctrl(struct v4l2_ctrl *ctrl)
170162306a36Sopenharmony_ci{
170262306a36Sopenharmony_ci	struct gspca_dev *gspca_dev =
170362306a36Sopenharmony_ci		container_of(ctrl->handler, struct gspca_dev, ctrl_handler);
170462306a36Sopenharmony_ci
170562306a36Sopenharmony_ci	gspca_dev->usb_err = 0;
170662306a36Sopenharmony_ci
170762306a36Sopenharmony_ci	if (!gspca_dev->streaming)
170862306a36Sopenharmony_ci		return 0;
170962306a36Sopenharmony_ci
171062306a36Sopenharmony_ci	switch (ctrl->id) {
171162306a36Sopenharmony_ci	case V4L2_CID_BRIGHTNESS:
171262306a36Sopenharmony_ci		setbrightness(gspca_dev, ctrl->val);
171362306a36Sopenharmony_ci		break;
171462306a36Sopenharmony_ci	case V4L2_CID_CONTRAST:
171562306a36Sopenharmony_ci		setcontrast(gspca_dev, ctrl->val);
171662306a36Sopenharmony_ci		break;
171762306a36Sopenharmony_ci	case V4L2_CID_SATURATION:
171862306a36Sopenharmony_ci		setsatur(gspca_dev, ctrl->val);
171962306a36Sopenharmony_ci		break;
172062306a36Sopenharmony_ci	case V4L2_CID_POWER_LINE_FREQUENCY:
172162306a36Sopenharmony_ci		setlightfreq(gspca_dev, ctrl->val);
172262306a36Sopenharmony_ci		break;
172362306a36Sopenharmony_ci	case V4L2_CID_SHARPNESS:
172462306a36Sopenharmony_ci		setsharpness(gspca_dev, ctrl->val);
172562306a36Sopenharmony_ci		break;
172662306a36Sopenharmony_ci	case V4L2_CID_AUTOGAIN:
172762306a36Sopenharmony_ci		if (ctrl->is_new)
172862306a36Sopenharmony_ci			setautogain(gspca_dev, ctrl->val);
172962306a36Sopenharmony_ci		if (!ctrl->val && gspca_dev->exposure->is_new)
173062306a36Sopenharmony_ci			setexposure(gspca_dev, gspca_dev->exposure->val);
173162306a36Sopenharmony_ci		break;
173262306a36Sopenharmony_ci	}
173362306a36Sopenharmony_ci	return gspca_dev->usb_err;
173462306a36Sopenharmony_ci}
173562306a36Sopenharmony_ci
173662306a36Sopenharmony_cistatic const struct v4l2_ctrl_ops sd_ctrl_ops = {
173762306a36Sopenharmony_ci	.s_ctrl = sd_s_ctrl,
173862306a36Sopenharmony_ci};
173962306a36Sopenharmony_ci
174062306a36Sopenharmony_cistatic int sd_init_controls(struct gspca_dev *gspca_dev)
174162306a36Sopenharmony_ci{
174262306a36Sopenharmony_ci	struct sd *sd = (struct sd *)gspca_dev;
174362306a36Sopenharmony_ci	struct v4l2_ctrl_handler *hdl = &gspca_dev->ctrl_handler;
174462306a36Sopenharmony_ci
174562306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV971x)
174662306a36Sopenharmony_ci		return 0;
174762306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV361x)
174862306a36Sopenharmony_ci		return 0;
174962306a36Sopenharmony_ci	gspca_dev->vdev.ctrl_handler = hdl;
175062306a36Sopenharmony_ci	v4l2_ctrl_handler_init(hdl, 7);
175162306a36Sopenharmony_ci	if (sd->sensor == SENSOR_OV562x) {
175262306a36Sopenharmony_ci		v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
175362306a36Sopenharmony_ci			V4L2_CID_BRIGHTNESS, -90, 90, 1, 0);
175462306a36Sopenharmony_ci	} else {
175562306a36Sopenharmony_ci		v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
175662306a36Sopenharmony_ci			V4L2_CID_BRIGHTNESS, 0, 15, 1, 7);
175762306a36Sopenharmony_ci		v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
175862306a36Sopenharmony_ci			V4L2_CID_CONTRAST, 0, 15, 1, 3);
175962306a36Sopenharmony_ci		v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
176062306a36Sopenharmony_ci			V4L2_CID_SATURATION, 0, 4, 1, 2);
176162306a36Sopenharmony_ci		/* -1 = auto */
176262306a36Sopenharmony_ci		v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
176362306a36Sopenharmony_ci			V4L2_CID_SHARPNESS, -1, 4, 1, -1);
176462306a36Sopenharmony_ci		gspca_dev->autogain = v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
176562306a36Sopenharmony_ci			V4L2_CID_AUTOGAIN, 0, 1, 1, 1);
176662306a36Sopenharmony_ci		gspca_dev->exposure = v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
176762306a36Sopenharmony_ci			V4L2_CID_EXPOSURE, 0, 3, 1, 0);
176862306a36Sopenharmony_ci		v4l2_ctrl_new_std_menu(hdl, &sd_ctrl_ops,
176962306a36Sopenharmony_ci			V4L2_CID_POWER_LINE_FREQUENCY,
177062306a36Sopenharmony_ci			V4L2_CID_POWER_LINE_FREQUENCY_60HZ, 0, 0);
177162306a36Sopenharmony_ci		v4l2_ctrl_auto_cluster(3, &gspca_dev->autogain, 0, false);
177262306a36Sopenharmony_ci	}
177362306a36Sopenharmony_ci
177462306a36Sopenharmony_ci	if (hdl->error) {
177562306a36Sopenharmony_ci		pr_err("Could not initialize controls\n");
177662306a36Sopenharmony_ci		return hdl->error;
177762306a36Sopenharmony_ci	}
177862306a36Sopenharmony_ci	return 0;
177962306a36Sopenharmony_ci}
178062306a36Sopenharmony_ci
178162306a36Sopenharmony_ci/* sub-driver description */
178262306a36Sopenharmony_cistatic const struct sd_desc sd_desc = {
178362306a36Sopenharmony_ci	.name     = MODULE_NAME,
178462306a36Sopenharmony_ci	.config   = sd_config,
178562306a36Sopenharmony_ci	.init     = sd_init,
178662306a36Sopenharmony_ci	.init_controls = sd_init_controls,
178762306a36Sopenharmony_ci	.start    = sd_start,
178862306a36Sopenharmony_ci	.stopN    = sd_stopN,
178962306a36Sopenharmony_ci	.pkt_scan = sd_pkt_scan,
179062306a36Sopenharmony_ci};
179162306a36Sopenharmony_ci
179262306a36Sopenharmony_ci/* -- module initialisation -- */
179362306a36Sopenharmony_cistatic const struct usb_device_id device_table[] = {
179462306a36Sopenharmony_ci	{USB_DEVICE(0x05a9, 0x8065)},
179562306a36Sopenharmony_ci	{USB_DEVICE(0x06f8, 0x3003)},
179662306a36Sopenharmony_ci	{USB_DEVICE(0x05a9, 0x1550)},
179762306a36Sopenharmony_ci	{}
179862306a36Sopenharmony_ci};
179962306a36Sopenharmony_ci
180062306a36Sopenharmony_ciMODULE_DEVICE_TABLE(usb, device_table);
180162306a36Sopenharmony_ci
180262306a36Sopenharmony_ci/* -- device connect -- */
180362306a36Sopenharmony_cistatic int sd_probe(struct usb_interface *intf, const struct usb_device_id *id)
180462306a36Sopenharmony_ci{
180562306a36Sopenharmony_ci	return gspca_dev_probe(intf, id, &sd_desc, sizeof(struct sd),
180662306a36Sopenharmony_ci				THIS_MODULE);
180762306a36Sopenharmony_ci}
180862306a36Sopenharmony_ci
180962306a36Sopenharmony_cistatic struct usb_driver sd_driver = {
181062306a36Sopenharmony_ci	.name       = MODULE_NAME,
181162306a36Sopenharmony_ci	.id_table   = device_table,
181262306a36Sopenharmony_ci	.probe      = sd_probe,
181362306a36Sopenharmony_ci	.disconnect = gspca_disconnect,
181462306a36Sopenharmony_ci#ifdef CONFIG_PM
181562306a36Sopenharmony_ci	.suspend    = gspca_suspend,
181662306a36Sopenharmony_ci	.resume     = gspca_resume,
181762306a36Sopenharmony_ci	.reset_resume = gspca_resume,
181862306a36Sopenharmony_ci#endif
181962306a36Sopenharmony_ci};
182062306a36Sopenharmony_ci
182162306a36Sopenharmony_cimodule_usb_driver(sd_driver);
1822