drm/tests: hdmi: Fix memory leaks in drm_display_mode_from_cea_vic()
[drm/drm-misc.git] / drivers / media / usb / gspca / gl860 / gl860-ov9655.c
blob766677ebcb3465757b734c0bc33ad273aa067b9a
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /* Subdriver for the GL860 chip with the OV9655 sensor
3 * Author Olivier LORIN, from logs done by Simon (Sur3) and Almighurt
4 * on dsd's weblog
5 */
7 /* Sensor : OV9655 */
9 #include "gl860.h"
11 static struct validx tbl_init_at_startup[] = {
12 {0x0000, 0x0000}, {0x0010, 0x0010}, {0x0008, 0x00c0}, {0x0001, 0x00c1},
13 {0x0001, 0x00c2}, {0x0020, 0x0006}, {0x006a, 0x000d},
15 {0x0040, 0x0000},
18 static struct validx tbl_commmon[] = {
19 {0x0041, 0x0000}, {0x006a, 0x0007}, {0x0063, 0x0006}, {0x006a, 0x000d},
20 {0x0000, 0x00c0}, {0x0010, 0x0010}, {0x0001, 0x00c1}, {0x0041, 0x00c2},
21 {0x0004, 0x00d8}, {0x0012, 0x0004}, {0x0000, 0x0058}, {0x0040, 0x0000},
22 {0x00f3, 0x0006}, {0x0058, 0x0000}, {0x0048, 0x0000}, {0x0061, 0x0000},
25 static s32 tbl_length[] = {12, 56, 52, 54, 56, 42, 32, 12};
27 static u8 *tbl_640[] = {
28 (u8[]){
29 0x00, 0x40, 0x07, 0x6a, 0x06, 0xf3, 0x0d, 0x6a,
30 0x10, 0x10, 0xc1, 0x01
31 }, (u8[]){
32 0x12, 0x80, 0x00, 0x00, 0x01, 0x98, 0x02, 0x80,
33 0x03, 0x12, 0x04, 0x03, 0x0b, 0x57, 0x0e, 0x61,
34 0x0f, 0x42, 0x11, 0x01, 0x12, 0x60, 0x13, 0x00,
35 0x14, 0x3a, 0x16, 0x24, 0x17, 0x14, 0x18, 0x00,
36 0x19, 0x01, 0x1a, 0x3d, 0x1e, 0x04, 0x24, 0x3c,
37 0x25, 0x36, 0x26, 0x72, 0x27, 0x08, 0x28, 0x08,
38 0x29, 0x15, 0x2a, 0x00, 0x2b, 0x00, 0x2c, 0x08
39 }, (u8[]){
40 0x32, 0xff, 0x33, 0x00, 0x34, 0x3d, 0x35, 0x00,
41 0x36, 0xfa, 0x38, 0x72, 0x39, 0x57, 0x3a, 0x00,
42 0x3b, 0x0c, 0x3d, 0x99, 0x3e, 0x0c, 0x3f, 0xc1,
43 0x40, 0xc0, 0x41, 0x00, 0x42, 0xc0, 0x43, 0x0a,
44 0x44, 0xf0, 0x45, 0x46, 0x46, 0x62, 0x47, 0x2a,
45 0x48, 0x3c, 0x4a, 0xee, 0x4b, 0xe7, 0x4c, 0xe7,
46 0x4d, 0xe7, 0x4e, 0xe7
47 }, (u8[]){
48 0x4f, 0x98, 0x50, 0x98, 0x51, 0x00, 0x52, 0x28,
49 0x53, 0x70, 0x54, 0x98, 0x58, 0x1a, 0x59, 0x85,
50 0x5a, 0xa9, 0x5b, 0x64, 0x5c, 0x84, 0x5d, 0x53,
51 0x5e, 0x0e, 0x5f, 0xf0, 0x60, 0xf0, 0x61, 0xf0,
52 0x62, 0x00, 0x63, 0x00, 0x64, 0x02, 0x65, 0x20,
53 0x66, 0x00, 0x69, 0x0a, 0x6b, 0x5a, 0x6c, 0x04,
54 0x6d, 0x55, 0x6e, 0x00, 0x6f, 0x9d
55 }, (u8[]){
56 0x70, 0x15, 0x71, 0x78, 0x72, 0x00, 0x73, 0x00,
57 0x74, 0x3a, 0x75, 0x35, 0x76, 0x01, 0x77, 0x02,
58 0x7a, 0x24, 0x7b, 0x04, 0x7c, 0x07, 0x7d, 0x10,
59 0x7e, 0x28, 0x7f, 0x36, 0x80, 0x44, 0x81, 0x52,
60 0x82, 0x60, 0x83, 0x6c, 0x84, 0x78, 0x85, 0x8c,
61 0x86, 0x9e, 0x87, 0xbb, 0x88, 0xd2, 0x89, 0xe5,
62 0x8a, 0x23, 0x8c, 0x8d, 0x90, 0x7c, 0x91, 0x7b
63 }, (u8[]){
64 0x9d, 0x02, 0x9e, 0x02, 0x9f, 0x74, 0xa0, 0x73,
65 0xa1, 0x40, 0xa4, 0x50, 0xa5, 0x68, 0xa6, 0x70,
66 0xa8, 0xc1, 0xa9, 0xef, 0xaa, 0x92, 0xab, 0x04,
67 0xac, 0x80, 0xad, 0x80, 0xae, 0x80, 0xaf, 0x80,
68 0xb2, 0xf2, 0xb3, 0x20, 0xb4, 0x20, 0xb5, 0x00,
69 0xb6, 0xaf
70 }, (u8[]){
71 0xbb, 0xae, 0xbc, 0x4f, 0xbd, 0x4e, 0xbe, 0x6a,
72 0xbf, 0x68, 0xc0, 0xaa, 0xc1, 0xc0, 0xc2, 0x01,
73 0xc3, 0x4e, 0xc6, 0x85, 0xc7, 0x81, 0xc9, 0xe0,
74 0xca, 0xe8, 0xcb, 0xf0, 0xcc, 0xd8, 0xcd, 0x93
75 }, (u8[]){
76 0xd0, 0x01, 0xd1, 0x08, 0xd2, 0xe0, 0xd3, 0x01,
77 0xd4, 0x10, 0xd5, 0x80
81 static u8 *tbl_1280[] = {
82 (u8[]){
83 0x00, 0x40, 0x07, 0x6a, 0x06, 0xf3, 0x0d, 0x6a,
84 0x10, 0x10, 0xc1, 0x01
86 (u8[]){
87 0x12, 0x80, 0x00, 0x00, 0x01, 0x98, 0x02, 0x80,
88 0x03, 0x12, 0x04, 0x01, 0x0b, 0x57, 0x0e, 0x61,
89 0x0f, 0x42, 0x11, 0x00, 0x12, 0x00, 0x13, 0x00,
90 0x14, 0x3a, 0x16, 0x24, 0x17, 0x1b, 0x18, 0xbb,
91 0x19, 0x01, 0x1a, 0x81, 0x1e, 0x04, 0x24, 0x3c,
92 0x25, 0x36, 0x26, 0x72, 0x27, 0x08, 0x28, 0x08,
93 0x29, 0x15, 0x2a, 0x00, 0x2b, 0x00, 0x2c, 0x08
95 (u8[]){
96 0x32, 0xa4, 0x33, 0x00, 0x34, 0x3d, 0x35, 0x00,
97 0x36, 0xf8, 0x38, 0x72, 0x39, 0x57, 0x3a, 0x00,
98 0x3b, 0x0c, 0x3d, 0x99, 0x3e, 0x0c, 0x3f, 0xc2,
99 0x40, 0xc0, 0x41, 0x00, 0x42, 0xc0, 0x43, 0x0a,
100 0x44, 0xf0, 0x45, 0x46, 0x46, 0x62, 0x47, 0x2a,
101 0x48, 0x3c, 0x4a, 0xec, 0x4b, 0xe8, 0x4c, 0xe8,
102 0x4d, 0xe8, 0x4e, 0xe8
104 (u8[]){
105 0x4f, 0x98, 0x50, 0x98, 0x51, 0x00, 0x52, 0x28,
106 0x53, 0x70, 0x54, 0x98, 0x58, 0x1a, 0x59, 0x85,
107 0x5a, 0xa9, 0x5b, 0x64, 0x5c, 0x84, 0x5d, 0x53,
108 0x5e, 0x0e, 0x5f, 0xf0, 0x60, 0xf0, 0x61, 0xf0,
109 0x62, 0x00, 0x63, 0x00, 0x64, 0x02, 0x65, 0x20,
110 0x66, 0x00, 0x69, 0x02, 0x6b, 0x5a, 0x6c, 0x04,
111 0x6d, 0x55, 0x6e, 0x00, 0x6f, 0x9d
113 (u8[]){
114 0x70, 0x08, 0x71, 0x78, 0x72, 0x00, 0x73, 0x01,
115 0x74, 0x3a, 0x75, 0x35, 0x76, 0x01, 0x77, 0x02,
116 0x7a, 0x24, 0x7b, 0x04, 0x7c, 0x07, 0x7d, 0x10,
117 0x7e, 0x28, 0x7f, 0x36, 0x80, 0x44, 0x81, 0x52,
118 0x82, 0x60, 0x83, 0x6c, 0x84, 0x78, 0x85, 0x8c,
119 0x86, 0x9e, 0x87, 0xbb, 0x88, 0xd2, 0x89, 0xe5,
120 0x8a, 0x23, 0x8c, 0x0d, 0x90, 0x90, 0x91, 0x90
122 (u8[]){
123 0x9d, 0x02, 0x9e, 0x02, 0x9f, 0x94, 0xa0, 0x94,
124 0xa1, 0x01, 0xa4, 0x50, 0xa5, 0x68, 0xa6, 0x70,
125 0xa8, 0xc1, 0xa9, 0xef, 0xaa, 0x92, 0xab, 0x04,
126 0xac, 0x80, 0xad, 0x80, 0xae, 0x80, 0xaf, 0x80,
127 0xb2, 0xf2, 0xb3, 0x20, 0xb4, 0x20, 0xb5, 0x00,
128 0xb6, 0xaf
130 (u8[]){
131 0xbb, 0xae, 0xbc, 0x38, 0xbd, 0x39, 0xbe, 0x01,
132 0xbf, 0x01, 0xc0, 0xe2, 0xc1, 0xc0, 0xc2, 0x01,
133 0xc3, 0x4e, 0xc6, 0x85, 0xc7, 0x81, 0xc9, 0xe0,
134 0xca, 0xe8, 0xcb, 0xf0, 0xcc, 0xd8, 0xcd, 0x93
136 (u8[]){
137 0xd0, 0x21, 0xd1, 0x18, 0xd2, 0xe0, 0xd3, 0x01,
138 0xd4, 0x28, 0xd5, 0x00
142 static u8 c04[] = {0x04};
143 static u8 dat_post1[] = "\x04\x00\x10\x20\xa1\x00\x00\x02";
144 static u8 dat_post2[] = "\x10\x10\xc1\x02";
145 static u8 dat_post3[] = "\x04\x00\x10\x7c\xa1\x00\x00\x04";
146 static u8 dat_post4[] = "\x10\x02\xc1\x06";
147 static u8 dat_post5[] = "\x04\x00\x10\x7b\xa1\x00\x00\x08";
148 static u8 dat_post6[] = "\x10\x10\xc1\x05";
149 static u8 dat_post7[] = "\x04\x00\x10\x7c\xa1\x00\x00\x08";
150 static u8 dat_post8[] = "\x04\x00\x10\x7c\xa1\x00\x00\x09";
152 static struct validx tbl_init_post_alt[] = {
153 {0x6032, 0x00ff}, {0x6032, 0x00ff}, {0x6032, 0x00ff}, {0x603c, 0x00ff},
154 {0x6003, 0x00ff}, {0x6032, 0x00ff}, {0x6032, 0x00ff}, {0x6001, 0x00ff},
155 {0x6000, 0x801e},
156 {0xffff, 0xffff},
157 {0x6004, 0x001e}, {0x6000, 0x801e},
158 {0xffff, 0xffff},
159 {0x6004, 0x001e}, {0x6012, 0x0003}, {0x6000, 0x801e},
160 {0xffff, 0xffff},
161 {0x6004, 0x001e}, {0x6000, 0x801e},
162 {0xffff, 0xffff},
163 {0x6004, 0x001e}, {0x6012, 0x0003},
164 {0xffff, 0xffff},
165 {0x6000, 0x801e},
166 {0xffff, 0xffff},
167 {0x6004, 0x001e}, {0x6000, 0x801e},
168 {0xffff, 0xffff},
169 {0x6004, 0x001e}, {0x6012, 0x0003}, {0x6000, 0x801e},
170 {0xffff, 0xffff},
171 {0x6004, 0x001e}, {0x6000, 0x801e},
172 {0xffff, 0xffff},
173 {0x6004, 0x001e}, {0x6012, 0x0003},
174 {0xffff, 0xffff},
175 {0x6000, 0x801e},
176 {0xffff, 0xffff},
177 {0x6004, 0x001e}, {0x6000, 0x801e},
178 {0xffff, 0xffff},
179 {0x6004, 0x001e}, {0x6012, 0x0003},
182 static int ov9655_init_at_startup(struct gspca_dev *gspca_dev);
183 static int ov9655_configure_alt(struct gspca_dev *gspca_dev);
184 static int ov9655_init_pre_alt(struct gspca_dev *gspca_dev);
185 static int ov9655_init_post_alt(struct gspca_dev *gspca_dev);
186 static void ov9655_post_unset_alt(struct gspca_dev *gspca_dev);
187 static int ov9655_camera_settings(struct gspca_dev *gspca_dev);
188 /*==========================================================================*/
190 void ov9655_init_settings(struct gspca_dev *gspca_dev)
192 struct sd *sd = (struct sd *) gspca_dev;
194 sd->vcur.backlight = 0;
195 sd->vcur.brightness = 128;
196 sd->vcur.sharpness = 0;
197 sd->vcur.contrast = 0;
198 sd->vcur.gamma = 0;
199 sd->vcur.hue = 0;
200 sd->vcur.saturation = 0;
201 sd->vcur.whitebal = 0;
203 sd->vmax.backlight = 0;
204 sd->vmax.brightness = 255;
205 sd->vmax.sharpness = 0;
206 sd->vmax.contrast = 0;
207 sd->vmax.gamma = 0;
208 sd->vmax.hue = 0 + 1;
209 sd->vmax.saturation = 0;
210 sd->vmax.whitebal = 0;
211 sd->vmax.mirror = 0;
212 sd->vmax.flip = 0;
213 sd->vmax.AC50Hz = 0;
215 sd->dev_camera_settings = ov9655_camera_settings;
216 sd->dev_init_at_startup = ov9655_init_at_startup;
217 sd->dev_configure_alt = ov9655_configure_alt;
218 sd->dev_init_pre_alt = ov9655_init_pre_alt;
219 sd->dev_post_unset_alt = ov9655_post_unset_alt;
222 /*==========================================================================*/
224 static int ov9655_init_at_startup(struct gspca_dev *gspca_dev)
226 fetch_validx(gspca_dev, tbl_init_at_startup,
227 ARRAY_SIZE(tbl_init_at_startup));
228 fetch_validx(gspca_dev, tbl_commmon, ARRAY_SIZE(tbl_commmon));
229 /* ctrl_out(gspca_dev, 0x40, 11, 0x0000, 0x0000, 0, NULL);*/
231 return 0;
234 static int ov9655_init_pre_alt(struct gspca_dev *gspca_dev)
236 struct sd *sd = (struct sd *) gspca_dev;
238 sd->vold.brightness = -1;
239 sd->vold.hue = -1;
241 fetch_validx(gspca_dev, tbl_commmon, ARRAY_SIZE(tbl_commmon));
243 ov9655_init_post_alt(gspca_dev);
245 return 0;
248 static int ov9655_init_post_alt(struct gspca_dev *gspca_dev)
250 s32 reso = gspca_dev->cam.cam_mode[(s32) gspca_dev->curr_mode].priv;
251 s32 n; /* reserved for FETCH functions */
252 s32 i;
253 u8 **tbl;
255 ctrl_out(gspca_dev, 0x40, 5, 0x0001, 0x0000, 0, NULL);
257 tbl = (reso == IMAGE_640) ? tbl_640 : tbl_1280;
259 ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200,
260 tbl_length[0], tbl[0]);
261 for (i = 1; i < 7; i++)
262 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200,
263 tbl_length[i], tbl[i]);
264 ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200,
265 tbl_length[7], tbl[7]);
267 n = fetch_validx(gspca_dev, tbl_init_post_alt,
268 ARRAY_SIZE(tbl_init_post_alt));
270 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
271 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
272 ARRAY_SIZE(tbl_init_post_alt), n);
273 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
274 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
275 ARRAY_SIZE(tbl_init_post_alt), n);
276 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
277 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
278 ARRAY_SIZE(tbl_init_post_alt), n);
279 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
280 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
281 ARRAY_SIZE(tbl_init_post_alt), n);
282 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post1);
283 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
284 ARRAY_SIZE(tbl_init_post_alt), n);
286 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
287 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
288 ARRAY_SIZE(tbl_init_post_alt), n);
289 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
290 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
291 ARRAY_SIZE(tbl_init_post_alt), n);
292 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
293 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
294 ARRAY_SIZE(tbl_init_post_alt), n);
295 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
296 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
297 ARRAY_SIZE(tbl_init_post_alt), n);
298 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post1);
299 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
300 ARRAY_SIZE(tbl_init_post_alt), n);
302 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
303 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
304 ARRAY_SIZE(tbl_init_post_alt), n);
305 ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
306 keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
307 ARRAY_SIZE(tbl_init_post_alt), n);
309 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post1);
311 ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 4, dat_post2);
312 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post3);
314 ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 4, dat_post4);
315 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post5);
317 ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 4, dat_post6);
318 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post7);
320 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post8);
322 ov9655_camera_settings(gspca_dev);
324 return 0;
327 static int ov9655_configure_alt(struct gspca_dev *gspca_dev)
329 s32 reso = gspca_dev->cam.cam_mode[(s32) gspca_dev->curr_mode].priv;
331 switch (reso) {
332 case IMAGE_640:
333 gspca_dev->alt = 1 + 1;
334 break;
336 default:
337 gspca_dev->alt = 1 + 1;
338 break;
340 return 0;
343 static int ov9655_camera_settings(struct gspca_dev *gspca_dev)
345 struct sd *sd = (struct sd *) gspca_dev;
347 u8 dat_bright[] = "\x04\x00\x10\x7c\xa1\x00\x00\x70";
349 s32 bright = sd->vcur.brightness;
350 s32 hue = sd->vcur.hue;
352 if (bright != sd->vold.brightness) {
353 sd->vold.brightness = bright;
354 if (bright < 0 || bright > sd->vmax.brightness)
355 bright = 0;
357 dat_bright[3] = bright;
358 ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_bright);
361 if (hue != sd->vold.hue) {
362 sd->vold.hue = hue;
363 sd->swapRB = (hue != 0);
366 return 0;
369 static void ov9655_post_unset_alt(struct gspca_dev *gspca_dev)
371 ctrl_out(gspca_dev, 0x40, 5, 0x0000, 0x0000, 0, NULL);
372 ctrl_out(gspca_dev, 0x40, 1, 0x0061, 0x0000, 0, NULL);