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