1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /* Subdriver for the GL860 chip with the MI1320 sensor
3*4882a593Smuzhiyun * Author Olivier LORIN from own logs
4*4882a593Smuzhiyun */
5*4882a593Smuzhiyun
6*4882a593Smuzhiyun /* Sensor : MI1320 */
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun #include "gl860.h"
9*4882a593Smuzhiyun
10*4882a593Smuzhiyun static struct validx tbl_common[] = {
11*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba00, 0x00f1}, {0xba51, 0x0066}, {0xba02, 0x00f1},
12*4882a593Smuzhiyun {0xba05, 0x0067}, {0xba05, 0x00f1}, {0xbaa0, 0x0065}, {0xba00, 0x00f1},
13*4882a593Smuzhiyun {0xffff, 0xffff},
14*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba02, 0x00f1}, {0xbafa, 0x0028}, {0xba02, 0x00f1},
15*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba01, 0x00f1}, {0xbaf0, 0x0006}, {0xba0e, 0x00f1},
16*4882a593Smuzhiyun {0xba70, 0x0006}, {0xba0e, 0x00f1},
17*4882a593Smuzhiyun {0xffff, 0xffff},
18*4882a593Smuzhiyun {0xba74, 0x0006}, {0xba0e, 0x00f1},
19*4882a593Smuzhiyun {0xffff, 0xffff},
20*4882a593Smuzhiyun {0x0061, 0x0000}, {0x0068, 0x000d},
21*4882a593Smuzhiyun };
22*4882a593Smuzhiyun
23*4882a593Smuzhiyun static struct validx tbl_init_at_startup[] = {
24*4882a593Smuzhiyun {0x0000, 0x0000}, {0x0010, 0x0010},
25*4882a593Smuzhiyun {35, 0xffff},
26*4882a593Smuzhiyun {0x0008, 0x00c0}, {0x0001, 0x00c1}, {0x0001, 0x00c2}, {0x0020, 0x0006},
27*4882a593Smuzhiyun {0x006a, 0x000d},
28*4882a593Smuzhiyun };
29*4882a593Smuzhiyun
30*4882a593Smuzhiyun static struct validx tbl_sensor_settings_common[] = {
31*4882a593Smuzhiyun {0x0010, 0x0010}, {0x0003, 0x00c1}, {0x0042, 0x00c2}, {0x0040, 0x0000},
32*4882a593Smuzhiyun {0x006a, 0x0007}, {0x006a, 0x000d}, {0x0063, 0x0006},
33*4882a593Smuzhiyun };
34*4882a593Smuzhiyun static struct validx tbl_sensor_settings_1280[] = {
35*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba00, 0x00f1}, {0xba5a, 0x0066}, {0xba02, 0x00f1},
36*4882a593Smuzhiyun {0xba05, 0x0067}, {0xba05, 0x00f1}, {0xba20, 0x0065}, {0xba00, 0x00f1},
37*4882a593Smuzhiyun };
38*4882a593Smuzhiyun static struct validx tbl_sensor_settings_800[] = {
39*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba00, 0x00f1}, {0xba5a, 0x0066}, {0xba02, 0x00f1},
40*4882a593Smuzhiyun {0xba05, 0x0067}, {0xba05, 0x00f1}, {0xba20, 0x0065}, {0xba00, 0x00f1},
41*4882a593Smuzhiyun };
42*4882a593Smuzhiyun static struct validx tbl_sensor_settings_640[] = {
43*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba00, 0x00f1}, {0xbaa0, 0x0065}, {0xba00, 0x00f1},
44*4882a593Smuzhiyun {0xba51, 0x0066}, {0xba02, 0x00f1}, {0xba05, 0x0067}, {0xba05, 0x00f1},
45*4882a593Smuzhiyun {0xba20, 0x0065}, {0xba00, 0x00f1},
46*4882a593Smuzhiyun };
47*4882a593Smuzhiyun static struct validx tbl_post_unset_alt[] = {
48*4882a593Smuzhiyun {0xba00, 0x00f0}, {0xba00, 0x00f1}, {0xbaa0, 0x0065}, {0xba00, 0x00f1},
49*4882a593Smuzhiyun {0x0061, 0x0000}, {0x0068, 0x000d},
50*4882a593Smuzhiyun };
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun static u8 *tbl_1280[] = {
53*4882a593Smuzhiyun "\x0d\x80\xf1\x08\x03\x04\xf1\x00" "\x04\x05\xf1\x02\x05\x00\xf1\xf1"
54*4882a593Smuzhiyun "\x06\x00\xf1\x0d\x20\x01\xf1\x00" "\x21\x84\xf1\x00\x0d\x00\xf1\x08"
55*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x34\x00\xf1\x00" "\x9b\x43\xf1\x00\xa6\x05\xf1\x00"
56*4882a593Smuzhiyun "\xa9\x04\xf1\x00\xa1\x05\xf1\x00" "\xa4\x04\xf1\x00\xae\x0a\xf1\x08"
57*4882a593Smuzhiyun ,
58*4882a593Smuzhiyun "\xf0\x00\xf1\x02\x3a\x05\xf1\xf1" "\x3c\x05\xf1\xf1\x59\x01\xf1\x47"
59*4882a593Smuzhiyun "\x5a\x01\xf1\x88\x5c\x0a\xf1\x06" "\x5d\x0e\xf1\x0a\x64\x5e\xf1\x1c"
60*4882a593Smuzhiyun "\xd2\x00\xf1\xcf\xcb\x00\xf1\x01"
61*4882a593Smuzhiyun ,
62*4882a593Smuzhiyun "\xd3\x02\xd4\x28\xd5\x01\xd0\x02" "\xd1\x18\xd2\xc1"
63*4882a593Smuzhiyun };
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun static u8 *tbl_800[] = {
66*4882a593Smuzhiyun "\x0d\x80\xf1\x08\x03\x03\xf1\xc0" "\x04\x05\xf1\x02\x05\x00\xf1\xf1"
67*4882a593Smuzhiyun "\x06\x00\xf1\x0d\x20\x01\xf1\x00" "\x21\x84\xf1\x00\x0d\x00\xf1\x08"
68*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x34\x00\xf1\x00" "\x9b\x43\xf1\x00\xa6\x05\xf1\x00"
69*4882a593Smuzhiyun "\xa9\x03\xf1\xc0\xa1\x03\xf1\x20" "\xa4\x02\xf1\x5a\xae\x0a\xf1\x08"
70*4882a593Smuzhiyun ,
71*4882a593Smuzhiyun "\xf0\x00\xf1\x02\x3a\x05\xf1\xf1" "\x3c\x05\xf1\xf1\x59\x01\xf1\x47"
72*4882a593Smuzhiyun "\x5a\x01\xf1\x88\x5c\x0a\xf1\x06" "\x5d\x0e\xf1\x0a\x64\x5e\xf1\x1c"
73*4882a593Smuzhiyun "\xd2\x00\xf1\xcf\xcb\x00\xf1\x01"
74*4882a593Smuzhiyun ,
75*4882a593Smuzhiyun "\xd3\x02\xd4\x18\xd5\x21\xd0\x02" "\xd1\x10\xd2\x59"
76*4882a593Smuzhiyun };
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun static u8 *tbl_640[] = {
79*4882a593Smuzhiyun "\x0d\x80\xf1\x08\x03\x04\xf1\x04" "\x04\x05\xf1\x02\x07\x01\xf1\x7c"
80*4882a593Smuzhiyun "\x08\x00\xf1\x0e\x21\x80\xf1\x00" "\x0d\x00\xf1\x08\xf0\x00\xf1\x01"
81*4882a593Smuzhiyun "\x34\x10\xf1\x10\x3a\x43\xf1\x00" "\xa6\x05\xf1\x02\xa9\x04\xf1\x04"
82*4882a593Smuzhiyun "\xa7\x02\xf1\x81\xaa\x01\xf1\xe2" "\xae\x0c\xf1\x09"
83*4882a593Smuzhiyun ,
84*4882a593Smuzhiyun "\xf0\x00\xf1\x02\x39\x03\xf1\xfc" "\x3b\x04\xf1\x04\x57\x01\xf1\xb6"
85*4882a593Smuzhiyun "\x58\x02\xf1\x0d\x5c\x1f\xf1\x19" "\x5d\x24\xf1\x1e\x64\x5e\xf1\x1c"
86*4882a593Smuzhiyun "\xd2\x00\xf1\x00\xcb\x00\xf1\x01"
87*4882a593Smuzhiyun ,
88*4882a593Smuzhiyun "\xd3\x02\xd4\x10\xd5\x81\xd0\x02" "\xd1\x08\xd2\xe1"
89*4882a593Smuzhiyun };
90*4882a593Smuzhiyun
91*4882a593Smuzhiyun static s32 tbl_sat[] = {0x25, 0x1d, 0x15, 0x0d, 0x05, 0x4d, 0x55, 0x5d, 0x2d};
92*4882a593Smuzhiyun static s32 tbl_bright[] = {0, 8, 0x10, 0x20, 0x30, 0x40, 0x50, 0x60, 0x70};
93*4882a593Smuzhiyun static s32 tbl_backlight[] = {0x0e, 0x06, 0x02};
94*4882a593Smuzhiyun
95*4882a593Smuzhiyun static s32 tbl_cntr1[] = {
96*4882a593Smuzhiyun 0x90, 0x98, 0xa0, 0xa8, 0xb0, 0xb8, 0xc0, 0xc8, 0xd0, 0xe0, 0xf0};
97*4882a593Smuzhiyun static s32 tbl_cntr2[] = {
98*4882a593Smuzhiyun 0x70, 0x68, 0x60, 0x58, 0x50, 0x48, 0x40, 0x38, 0x30, 0x20, 0x10};
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun static u8 dat_wbalNL[] =
101*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x05\x00\xf1\x06" "\x3b\x04\xf1\x2a\x47\x10\xf1\x10"
102*4882a593Smuzhiyun "\x9d\x3c\xf1\xae\xaf\x10\xf1\x00" "\xf0\x00\xf1\x02\x2f\x91\xf1\x20"
103*4882a593Smuzhiyun "\x9c\x91\xf1\x20\x37\x03\xf1\x00" "\x9d\xc5\xf1\x0f\xf0\x00\xf1\x00";
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun static u8 dat_wbalLL[] =
106*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x05\x00\xf1\x0c" "\x3b\x04\xf1\x2a\x47\x40\xf1\x40"
107*4882a593Smuzhiyun "\x9d\x20\xf1\xae\xaf\x10\xf1\x00" "\xf0\x00\xf1\x02\x2f\xd1\xf1\x00"
108*4882a593Smuzhiyun "\x9c\xd1\xf1\x00\x37\x03\xf1\x00" "\x9d\xc5\xf1\x3f\xf0\x00\xf1\x00";
109*4882a593Smuzhiyun
110*4882a593Smuzhiyun static u8 dat_wbalBL[] =
111*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x05\x00\xf1\x06" "\x47\x10\xf1\x30\x9d\x3c\xf1\xae"
112*4882a593Smuzhiyun "\xaf\x10\xf1\x00\xf0\x00\xf1\x02" "\x2f\x91\xf1\x20\x9c\x91\xf1\x20"
113*4882a593Smuzhiyun "\x37\x03\xf1\x00\x9d\xc5\xf1\x2f" "\xf0\x00\xf1\x00";
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun static u8 dat_hvflip1[] = {0xf0, 0x00, 0xf1, 0x00};
116*4882a593Smuzhiyun
117*4882a593Smuzhiyun static u8 dat_common00[] =
118*4882a593Smuzhiyun "\x00\x01\x07\x6a\x06\x63\x0d\x6a" "\xc0\x00\x10\x10\xc1\x03\xc2\x42"
119*4882a593Smuzhiyun "\xd8\x04\x58\x00\x04\x02";
120*4882a593Smuzhiyun static u8 dat_common01[] =
121*4882a593Smuzhiyun "\x0d\x00\xf1\x0b\x0d\x00\xf1\x08" "\x35\x00\xf1\x22\x68\x00\xf1\x5d"
122*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x06\x70\xf1\x0e" "\xf0\x00\xf1\x02\xdd\x18\xf1\xe0";
123*4882a593Smuzhiyun static u8 dat_common02[] =
124*4882a593Smuzhiyun "\x05\x01\xf1\x84\x06\x00\xf1\x44" "\x07\x00\xf1\xbe\x08\x00\xf1\x1e"
125*4882a593Smuzhiyun "\x20\x01\xf1\x03\x21\x84\xf1\x00" "\x22\x0d\xf1\x0f\x24\x80\xf1\x00"
126*4882a593Smuzhiyun "\x34\x18\xf1\x2d\x35\x00\xf1\x22" "\x43\x83\xf1\x83\x59\x00\xf1\xff";
127*4882a593Smuzhiyun static u8 dat_common03[] =
128*4882a593Smuzhiyun "\xf0\x00\xf1\x02\x39\x06\xf1\x8c" "\x3a\x06\xf1\x8c\x3b\x03\xf1\xda"
129*4882a593Smuzhiyun "\x3c\x05\xf1\x30\x57\x01\xf1\x0c" "\x58\x01\xf1\x42\x59\x01\xf1\x0c"
130*4882a593Smuzhiyun "\x5a\x01\xf1\x42\x5c\x13\xf1\x0e" "\x5d\x17\xf1\x12\x64\x1e\xf1\x1c";
131*4882a593Smuzhiyun static u8 dat_common04[] =
132*4882a593Smuzhiyun "\xf0\x00\xf1\x02\x24\x5f\xf1\x20" "\x28\xea\xf1\x02\x5f\x41\xf1\x43";
133*4882a593Smuzhiyun static u8 dat_common05[] =
134*4882a593Smuzhiyun "\x02\x00\xf1\xee\x03\x29\xf1\x1a" "\x04\x02\xf1\xa4\x09\x00\xf1\x68"
135*4882a593Smuzhiyun "\x0a\x00\xf1\x2a\x0b\x00\xf1\x04" "\x0c\x00\xf1\x93\x0d\x00\xf1\x82"
136*4882a593Smuzhiyun "\x0e\x00\xf1\x40\x0f\x00\xf1\x5f" "\x10\x00\xf1\x4e\x11\x00\xf1\x5b";
137*4882a593Smuzhiyun static u8 dat_common06[] =
138*4882a593Smuzhiyun "\x15\x00\xf1\xc9\x16\x00\xf1\x5e" "\x17\x00\xf1\x9d\x18\x00\xf1\x06"
139*4882a593Smuzhiyun "\x19\x00\xf1\x89\x1a\x00\xf1\x12" "\x1b\x00\xf1\xa1\x1c\x00\xf1\xe4"
140*4882a593Smuzhiyun "\x1d\x00\xf1\x7a\x1e\x00\xf1\x64" "\xf6\x00\xf1\x5f";
141*4882a593Smuzhiyun static u8 dat_common07[] =
142*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x53\x09\xf1\x03" "\x54\x3d\xf1\x1c\x55\x99\xf1\x72"
143*4882a593Smuzhiyun "\x56\xc1\xf1\xb1\x57\xd8\xf1\xce" "\x58\xe0\xf1\x00\xdc\x0a\xf1\x03"
144*4882a593Smuzhiyun "\xdd\x45\xf1\x20\xde\xae\xf1\x82" "\xdf\xdc\xf1\xc9\xe0\xf6\xf1\xea"
145*4882a593Smuzhiyun "\xe1\xff\xf1\x00";
146*4882a593Smuzhiyun static u8 dat_common08[] =
147*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x80\x00\xf1\x06" "\x81\xf6\xf1\x08\x82\xfb\xf1\xf7"
148*4882a593Smuzhiyun "\x83\x00\xf1\xfe\xb6\x07\xf1\x03" "\xb7\x18\xf1\x0c\x84\xfb\xf1\x06"
149*4882a593Smuzhiyun "\x85\xfb\xf1\xf9\x86\x00\xf1\xff" "\xb8\x07\xf1\x04\xb9\x16\xf1\x0a";
150*4882a593Smuzhiyun static u8 dat_common09[] =
151*4882a593Smuzhiyun "\x87\xfa\xf1\x05\x88\xfc\xf1\xf9" "\x89\x00\xf1\xff\xba\x06\xf1\x03"
152*4882a593Smuzhiyun "\xbb\x17\xf1\x09\x8a\xe8\xf1\x14" "\x8b\xf7\xf1\xf0\x8c\xfd\xf1\xfa"
153*4882a593Smuzhiyun "\x8d\x00\xf1\x00\xbc\x05\xf1\x01" "\xbd\x0c\xf1\x08\xbe\x00\xf1\x14";
154*4882a593Smuzhiyun static u8 dat_common10[] =
155*4882a593Smuzhiyun "\x8e\xea\xf1\x13\x8f\xf7\xf1\xf2" "\x90\xfd\xf1\xfa\x91\x00\xf1\x00"
156*4882a593Smuzhiyun "\xbf\x05\xf1\x01\xc0\x0a\xf1\x08" "\xc1\x00\xf1\x0c\x92\xed\xf1\x0f"
157*4882a593Smuzhiyun "\x93\xf9\xf1\xf4\x94\xfe\xf1\xfb" "\x95\x00\xf1\x00\xc2\x04\xf1\x01"
158*4882a593Smuzhiyun "\xc3\x0a\xf1\x07\xc4\x00\xf1\x10";
159*4882a593Smuzhiyun static u8 dat_common11[] =
160*4882a593Smuzhiyun "\xf0\x00\xf1\x01\x05\x00\xf1\x06" "\x25\x00\xf1\x55\x34\x10\xf1\x10"
161*4882a593Smuzhiyun "\x35\xf0\xf1\x10\x3a\x02\xf1\x03" "\x3b\x04\xf1\x2a\x9b\x43\xf1\x00"
162*4882a593Smuzhiyun "\xa4\x03\xf1\xc0\xa7\x02\xf1\x81";
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun static int mi1320_init_at_startup(struct gspca_dev *gspca_dev);
165*4882a593Smuzhiyun static int mi1320_configure_alt(struct gspca_dev *gspca_dev);
166*4882a593Smuzhiyun static int mi1320_init_pre_alt(struct gspca_dev *gspca_dev);
167*4882a593Smuzhiyun static int mi1320_init_post_alt(struct gspca_dev *gspca_dev);
168*4882a593Smuzhiyun static void mi1320_post_unset_alt(struct gspca_dev *gspca_dev);
169*4882a593Smuzhiyun static int mi1320_sensor_settings(struct gspca_dev *gspca_dev);
170*4882a593Smuzhiyun static int mi1320_camera_settings(struct gspca_dev *gspca_dev);
171*4882a593Smuzhiyun /*==========================================================================*/
172*4882a593Smuzhiyun
mi1320_init_settings(struct gspca_dev * gspca_dev)173*4882a593Smuzhiyun void mi1320_init_settings(struct gspca_dev *gspca_dev)
174*4882a593Smuzhiyun {
175*4882a593Smuzhiyun struct sd *sd = (struct sd *) gspca_dev;
176*4882a593Smuzhiyun
177*4882a593Smuzhiyun sd->vcur.backlight = 0;
178*4882a593Smuzhiyun sd->vcur.brightness = 0;
179*4882a593Smuzhiyun sd->vcur.sharpness = 6;
180*4882a593Smuzhiyun sd->vcur.contrast = 10;
181*4882a593Smuzhiyun sd->vcur.gamma = 20;
182*4882a593Smuzhiyun sd->vcur.hue = 0;
183*4882a593Smuzhiyun sd->vcur.saturation = 6;
184*4882a593Smuzhiyun sd->vcur.whitebal = 0;
185*4882a593Smuzhiyun sd->vcur.mirror = 0;
186*4882a593Smuzhiyun sd->vcur.flip = 0;
187*4882a593Smuzhiyun sd->vcur.AC50Hz = 1;
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun sd->vmax.backlight = 2;
190*4882a593Smuzhiyun sd->vmax.brightness = 8;
191*4882a593Smuzhiyun sd->vmax.sharpness = 7;
192*4882a593Smuzhiyun sd->vmax.contrast = 0; /* 10 but not working with this driver */
193*4882a593Smuzhiyun sd->vmax.gamma = 40;
194*4882a593Smuzhiyun sd->vmax.hue = 5 + 1;
195*4882a593Smuzhiyun sd->vmax.saturation = 8;
196*4882a593Smuzhiyun sd->vmax.whitebal = 2;
197*4882a593Smuzhiyun sd->vmax.mirror = 1;
198*4882a593Smuzhiyun sd->vmax.flip = 1;
199*4882a593Smuzhiyun sd->vmax.AC50Hz = 1;
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun sd->dev_camera_settings = mi1320_camera_settings;
202*4882a593Smuzhiyun sd->dev_init_at_startup = mi1320_init_at_startup;
203*4882a593Smuzhiyun sd->dev_configure_alt = mi1320_configure_alt;
204*4882a593Smuzhiyun sd->dev_init_pre_alt = mi1320_init_pre_alt;
205*4882a593Smuzhiyun sd->dev_post_unset_alt = mi1320_post_unset_alt;
206*4882a593Smuzhiyun }
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun /*==========================================================================*/
209*4882a593Smuzhiyun
common(struct gspca_dev * gspca_dev)210*4882a593Smuzhiyun static void common(struct gspca_dev *gspca_dev)
211*4882a593Smuzhiyun {
212*4882a593Smuzhiyun s32 n; /* reserved for FETCH functions */
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 22, dat_common00);
215*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0x0041, 0x0000, 0, NULL);
216*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 32, dat_common01);
217*4882a593Smuzhiyun n = fetch_validx(gspca_dev, tbl_common, ARRAY_SIZE(tbl_common));
218*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 48, dat_common02);
219*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 48, dat_common03);
220*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 16, dat_common04);
221*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 48, dat_common05);
222*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 44, dat_common06);
223*4882a593Smuzhiyun keep_on_fetching_validx(gspca_dev, tbl_common,
224*4882a593Smuzhiyun ARRAY_SIZE(tbl_common), n);
225*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 52, dat_common07);
226*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 48, dat_common08);
227*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 48, dat_common09);
228*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 56, dat_common10);
229*4882a593Smuzhiyun keep_on_fetching_validx(gspca_dev, tbl_common,
230*4882a593Smuzhiyun ARRAY_SIZE(tbl_common), n);
231*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 40, dat_common11);
232*4882a593Smuzhiyun keep_on_fetching_validx(gspca_dev, tbl_common,
233*4882a593Smuzhiyun ARRAY_SIZE(tbl_common), n);
234*4882a593Smuzhiyun }
235*4882a593Smuzhiyun
mi1320_init_at_startup(struct gspca_dev * gspca_dev)236*4882a593Smuzhiyun static int mi1320_init_at_startup(struct gspca_dev *gspca_dev)
237*4882a593Smuzhiyun {
238*4882a593Smuzhiyun fetch_validx(gspca_dev, tbl_init_at_startup,
239*4882a593Smuzhiyun ARRAY_SIZE(tbl_init_at_startup));
240*4882a593Smuzhiyun
241*4882a593Smuzhiyun common(gspca_dev);
242*4882a593Smuzhiyun
243*4882a593Smuzhiyun /* ctrl_out(gspca_dev, 0x40, 11, 0x0000, 0x0000, 0, NULL); */
244*4882a593Smuzhiyun
245*4882a593Smuzhiyun return 0;
246*4882a593Smuzhiyun }
247*4882a593Smuzhiyun
mi1320_init_pre_alt(struct gspca_dev * gspca_dev)248*4882a593Smuzhiyun static int mi1320_init_pre_alt(struct gspca_dev *gspca_dev)
249*4882a593Smuzhiyun {
250*4882a593Smuzhiyun struct sd *sd = (struct sd *) gspca_dev;
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun sd->mirrorMask = 0;
253*4882a593Smuzhiyun
254*4882a593Smuzhiyun sd->vold.backlight = -1;
255*4882a593Smuzhiyun sd->vold.brightness = -1;
256*4882a593Smuzhiyun sd->vold.sharpness = -1;
257*4882a593Smuzhiyun sd->vold.contrast = -1;
258*4882a593Smuzhiyun sd->vold.saturation = -1;
259*4882a593Smuzhiyun sd->vold.gamma = -1;
260*4882a593Smuzhiyun sd->vold.hue = -1;
261*4882a593Smuzhiyun sd->vold.whitebal = -1;
262*4882a593Smuzhiyun sd->vold.mirror = -1;
263*4882a593Smuzhiyun sd->vold.flip = -1;
264*4882a593Smuzhiyun sd->vold.AC50Hz = -1;
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun common(gspca_dev);
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun mi1320_sensor_settings(gspca_dev);
269*4882a593Smuzhiyun
270*4882a593Smuzhiyun mi1320_init_post_alt(gspca_dev);
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun return 0;
273*4882a593Smuzhiyun }
274*4882a593Smuzhiyun
mi1320_init_post_alt(struct gspca_dev * gspca_dev)275*4882a593Smuzhiyun static int mi1320_init_post_alt(struct gspca_dev *gspca_dev)
276*4882a593Smuzhiyun {
277*4882a593Smuzhiyun mi1320_camera_settings(gspca_dev);
278*4882a593Smuzhiyun
279*4882a593Smuzhiyun return 0;
280*4882a593Smuzhiyun }
281*4882a593Smuzhiyun
mi1320_sensor_settings(struct gspca_dev * gspca_dev)282*4882a593Smuzhiyun static int mi1320_sensor_settings(struct gspca_dev *gspca_dev)
283*4882a593Smuzhiyun {
284*4882a593Smuzhiyun s32 reso = gspca_dev->cam.cam_mode[(s32) gspca_dev->curr_mode].priv;
285*4882a593Smuzhiyun
286*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 5, 0x0001, 0x0000, 0, NULL);
287*4882a593Smuzhiyun
288*4882a593Smuzhiyun fetch_validx(gspca_dev, tbl_sensor_settings_common,
289*4882a593Smuzhiyun ARRAY_SIZE(tbl_sensor_settings_common));
290*4882a593Smuzhiyun
291*4882a593Smuzhiyun switch (reso) {
292*4882a593Smuzhiyun case IMAGE_1280:
293*4882a593Smuzhiyun fetch_validx(gspca_dev, tbl_sensor_settings_1280,
294*4882a593Smuzhiyun ARRAY_SIZE(tbl_sensor_settings_1280));
295*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 64, tbl_1280[0]);
296*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 40, tbl_1280[1]);
297*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 12, tbl_1280[2]);
298*4882a593Smuzhiyun break;
299*4882a593Smuzhiyun
300*4882a593Smuzhiyun case IMAGE_800:
301*4882a593Smuzhiyun fetch_validx(gspca_dev, tbl_sensor_settings_800,
302*4882a593Smuzhiyun ARRAY_SIZE(tbl_sensor_settings_800));
303*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 64, tbl_800[0]);
304*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 40, tbl_800[1]);
305*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 12, tbl_800[2]);
306*4882a593Smuzhiyun break;
307*4882a593Smuzhiyun
308*4882a593Smuzhiyun default:
309*4882a593Smuzhiyun fetch_validx(gspca_dev, tbl_sensor_settings_640,
310*4882a593Smuzhiyun ARRAY_SIZE(tbl_sensor_settings_640));
311*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 60, tbl_640[0]);
312*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 40, tbl_640[1]);
313*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 12, tbl_640[2]);
314*4882a593Smuzhiyun break;
315*4882a593Smuzhiyun }
316*4882a593Smuzhiyun return 0;
317*4882a593Smuzhiyun }
318*4882a593Smuzhiyun
mi1320_configure_alt(struct gspca_dev * gspca_dev)319*4882a593Smuzhiyun static int mi1320_configure_alt(struct gspca_dev *gspca_dev)
320*4882a593Smuzhiyun {
321*4882a593Smuzhiyun s32 reso = gspca_dev->cam.cam_mode[(s32) gspca_dev->curr_mode].priv;
322*4882a593Smuzhiyun
323*4882a593Smuzhiyun switch (reso) {
324*4882a593Smuzhiyun case IMAGE_640:
325*4882a593Smuzhiyun gspca_dev->alt = 3 + 1;
326*4882a593Smuzhiyun break;
327*4882a593Smuzhiyun
328*4882a593Smuzhiyun case IMAGE_800:
329*4882a593Smuzhiyun case IMAGE_1280:
330*4882a593Smuzhiyun gspca_dev->alt = 1 + 1;
331*4882a593Smuzhiyun break;
332*4882a593Smuzhiyun }
333*4882a593Smuzhiyun return 0;
334*4882a593Smuzhiyun }
335*4882a593Smuzhiyun
mi1320_camera_settings(struct gspca_dev * gspca_dev)336*4882a593Smuzhiyun static int mi1320_camera_settings(struct gspca_dev *gspca_dev)
337*4882a593Smuzhiyun {
338*4882a593Smuzhiyun struct sd *sd = (struct sd *) gspca_dev;
339*4882a593Smuzhiyun
340*4882a593Smuzhiyun s32 backlight = sd->vcur.backlight;
341*4882a593Smuzhiyun s32 bright = sd->vcur.brightness;
342*4882a593Smuzhiyun s32 sharp = sd->vcur.sharpness;
343*4882a593Smuzhiyun s32 cntr = sd->vcur.contrast;
344*4882a593Smuzhiyun s32 gam = sd->vcur.gamma;
345*4882a593Smuzhiyun s32 hue = sd->vcur.hue;
346*4882a593Smuzhiyun s32 sat = sd->vcur.saturation;
347*4882a593Smuzhiyun s32 wbal = sd->vcur.whitebal;
348*4882a593Smuzhiyun s32 mirror = (((sd->vcur.mirror > 0) ^ sd->mirrorMask) > 0);
349*4882a593Smuzhiyun s32 flip = (((sd->vcur.flip > 0) ^ sd->mirrorMask) > 0);
350*4882a593Smuzhiyun s32 freq = (sd->vcur.AC50Hz > 0);
351*4882a593Smuzhiyun s32 i;
352*4882a593Smuzhiyun
353*4882a593Smuzhiyun if (freq != sd->vold.AC50Hz) {
354*4882a593Smuzhiyun sd->vold.AC50Hz = freq;
355*4882a593Smuzhiyun
356*4882a593Smuzhiyun freq = 2 * (freq == 0);
357*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
358*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba02, 0x00f1, 0, NULL);
359*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 , 0x005b, 0, NULL);
360*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01 + freq, 0x00f1, 0, NULL);
361*4882a593Smuzhiyun }
362*4882a593Smuzhiyun
363*4882a593Smuzhiyun if (wbal != sd->vold.whitebal) {
364*4882a593Smuzhiyun sd->vold.whitebal = wbal;
365*4882a593Smuzhiyun if (wbal < 0 || wbal > sd->vmax.whitebal)
366*4882a593Smuzhiyun wbal = 0;
367*4882a593Smuzhiyun
368*4882a593Smuzhiyun for (i = 0; i < 2; i++) {
369*4882a593Smuzhiyun if (wbal == 0) { /* Normal light */
370*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
371*4882a593Smuzhiyun 0x0010, 0x0010, 0, NULL);
372*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
373*4882a593Smuzhiyun 0x0003, 0x00c1, 0, NULL);
374*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
375*4882a593Smuzhiyun 0x0042, 0x00c2, 0, NULL);
376*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3,
377*4882a593Smuzhiyun 0xba00, 0x0200, 48, dat_wbalNL);
378*4882a593Smuzhiyun }
379*4882a593Smuzhiyun
380*4882a593Smuzhiyun if (wbal == 1) { /* Low light */
381*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
382*4882a593Smuzhiyun 0x0010, 0x0010, 0, NULL);
383*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
384*4882a593Smuzhiyun 0x0004, 0x00c1, 0, NULL);
385*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
386*4882a593Smuzhiyun 0x0043, 0x00c2, 0, NULL);
387*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3,
388*4882a593Smuzhiyun 0xba00, 0x0200, 48, dat_wbalLL);
389*4882a593Smuzhiyun }
390*4882a593Smuzhiyun
391*4882a593Smuzhiyun if (wbal == 2) { /* Back light */
392*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
393*4882a593Smuzhiyun 0x0010, 0x0010, 0, NULL);
394*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
395*4882a593Smuzhiyun 0x0003, 0x00c1, 0, NULL);
396*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1,
397*4882a593Smuzhiyun 0x0042, 0x00c2, 0, NULL);
398*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3,
399*4882a593Smuzhiyun 0xba00, 0x0200, 44, dat_wbalBL);
400*4882a593Smuzhiyun }
401*4882a593Smuzhiyun }
402*4882a593Smuzhiyun }
403*4882a593Smuzhiyun
404*4882a593Smuzhiyun if (bright != sd->vold.brightness) {
405*4882a593Smuzhiyun sd->vold.brightness = bright;
406*4882a593Smuzhiyun if (bright < 0 || bright > sd->vmax.brightness)
407*4882a593Smuzhiyun bright = 0;
408*4882a593Smuzhiyun
409*4882a593Smuzhiyun bright = tbl_bright[bright];
410*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
411*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
412*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + bright, 0x0034, 0, NULL);
413*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + bright, 0x00f1, 0, NULL);
414*4882a593Smuzhiyun }
415*4882a593Smuzhiyun
416*4882a593Smuzhiyun if (sat != sd->vold.saturation) {
417*4882a593Smuzhiyun sd->vold.saturation = sat;
418*4882a593Smuzhiyun if (sat < 0 || sat > sd->vmax.saturation)
419*4882a593Smuzhiyun sat = 0;
420*4882a593Smuzhiyun
421*4882a593Smuzhiyun sat = tbl_sat[sat];
422*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
423*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
424*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 , 0x0025, 0, NULL);
425*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + sat, 0x00f1, 0, NULL);
426*4882a593Smuzhiyun }
427*4882a593Smuzhiyun
428*4882a593Smuzhiyun if (sharp != sd->vold.sharpness) {
429*4882a593Smuzhiyun sd->vold.sharpness = sharp;
430*4882a593Smuzhiyun if (sharp < 0 || sharp > sd->vmax.sharpness)
431*4882a593Smuzhiyun sharp = 0;
432*4882a593Smuzhiyun
433*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
434*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
435*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 , 0x0005, 0, NULL);
436*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + sharp, 0x00f1, 0, NULL);
437*4882a593Smuzhiyun }
438*4882a593Smuzhiyun
439*4882a593Smuzhiyun if (hue != sd->vold.hue) {
440*4882a593Smuzhiyun /* 0=normal 1=NB 2="sepia" 3=negative 4=other 5=other2 */
441*4882a593Smuzhiyun if (hue < 0 || hue > sd->vmax.hue)
442*4882a593Smuzhiyun hue = 0;
443*4882a593Smuzhiyun if (hue == sd->vmax.hue)
444*4882a593Smuzhiyun sd->swapRB = 1;
445*4882a593Smuzhiyun else
446*4882a593Smuzhiyun sd->swapRB = 0;
447*4882a593Smuzhiyun
448*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
449*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
450*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba70, 0x00e2, 0, NULL);
451*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + hue * (hue < 6), 0x00f1,
452*4882a593Smuzhiyun 0, NULL);
453*4882a593Smuzhiyun }
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun if (backlight != sd->vold.backlight) {
456*4882a593Smuzhiyun sd->vold.backlight = backlight;
457*4882a593Smuzhiyun if (backlight < 0 || backlight > sd->vmax.backlight)
458*4882a593Smuzhiyun backlight = 0;
459*4882a593Smuzhiyun
460*4882a593Smuzhiyun backlight = tbl_backlight[backlight];
461*4882a593Smuzhiyun for (i = 0; i < 2; i++) {
462*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
463*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
464*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba74, 0x0006, 0, NULL);
465*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba80 + backlight, 0x00f1,
466*4882a593Smuzhiyun 0, NULL);
467*4882a593Smuzhiyun }
468*4882a593Smuzhiyun }
469*4882a593Smuzhiyun
470*4882a593Smuzhiyun if (hue != sd->vold.hue) {
471*4882a593Smuzhiyun sd->vold.hue = hue;
472*4882a593Smuzhiyun
473*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
474*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
475*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba70, 0x00e2, 0, NULL);
476*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + hue * (hue < 6), 0x00f1,
477*4882a593Smuzhiyun 0, NULL);
478*4882a593Smuzhiyun }
479*4882a593Smuzhiyun
480*4882a593Smuzhiyun if (mirror != sd->vold.mirror || flip != sd->vold.flip) {
481*4882a593Smuzhiyun u8 dat_hvflip2[4] = {0x20, 0x01, 0xf1, 0x00};
482*4882a593Smuzhiyun sd->vold.mirror = mirror;
483*4882a593Smuzhiyun sd->vold.flip = flip;
484*4882a593Smuzhiyun
485*4882a593Smuzhiyun dat_hvflip2[3] = flip + 2 * mirror;
486*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 4, dat_hvflip1);
487*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 3, 0xba00, 0x0200, 4, dat_hvflip2);
488*4882a593Smuzhiyun }
489*4882a593Smuzhiyun
490*4882a593Smuzhiyun if (gam != sd->vold.gamma) {
491*4882a593Smuzhiyun sd->vold.gamma = gam;
492*4882a593Smuzhiyun if (gam < 0 || gam > sd->vmax.gamma)
493*4882a593Smuzhiyun gam = 0;
494*4882a593Smuzhiyun
495*4882a593Smuzhiyun gam = 2 * gam;
496*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
497*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
498*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba04 , 0x003b, 0, NULL);
499*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba02 + gam, 0x00f1, 0, NULL);
500*4882a593Smuzhiyun }
501*4882a593Smuzhiyun
502*4882a593Smuzhiyun if (cntr != sd->vold.contrast) {
503*4882a593Smuzhiyun sd->vold.contrast = cntr;
504*4882a593Smuzhiyun if (cntr < 0 || cntr > sd->vmax.contrast)
505*4882a593Smuzhiyun cntr = 0;
506*4882a593Smuzhiyun
507*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00, 0x00f0, 0, NULL);
508*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba01, 0x00f1, 0, NULL);
509*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + tbl_cntr1[cntr], 0x0035,
510*4882a593Smuzhiyun 0, NULL);
511*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 1, 0xba00 + tbl_cntr2[cntr], 0x00f1,
512*4882a593Smuzhiyun 0, NULL);
513*4882a593Smuzhiyun }
514*4882a593Smuzhiyun
515*4882a593Smuzhiyun return 0;
516*4882a593Smuzhiyun }
517*4882a593Smuzhiyun
mi1320_post_unset_alt(struct gspca_dev * gspca_dev)518*4882a593Smuzhiyun static void mi1320_post_unset_alt(struct gspca_dev *gspca_dev)
519*4882a593Smuzhiyun {
520*4882a593Smuzhiyun ctrl_out(gspca_dev, 0x40, 5, 0x0000, 0x0000, 0, NULL);
521*4882a593Smuzhiyun
522*4882a593Smuzhiyun fetch_validx(gspca_dev, tbl_post_unset_alt,
523*4882a593Smuzhiyun ARRAY_SIZE(tbl_post_unset_alt));
524*4882a593Smuzhiyun }
525