4 * Copyright (C) 2010 Jean-Francois Moine (http://moinejf.free.fr)
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 *Notes: * t613 + tas5130A
17 * * Focus to light do not balance well as in win.
18 * Quality in win is not good, but its kinda better.
19 * * Fix some "extraneous bytes", most of apps will show the image anyway
20 * * Gamma table, is there, but its really doing something?
21 * * 7~8 Fps, its ok, max on win its 10.
25 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
27 #define MODULE_NAME "t613"
29 #include <linux/input.h>
30 #include <linux/slab.h>
33 MODULE_AUTHOR("Leandro Costantino <le_costantino@pixartargentina.com.ar>");
34 MODULE_DESCRIPTION("GSPCA/T613 (JPEG Compliance) USB Camera Driver");
35 MODULE_LICENSE("GPL");
38 struct gspca_dev gspca_dev
; /* !! must be the first item */
39 struct v4l2_ctrl
*freq
;
40 struct { /* awb / color gains control cluster */
41 struct v4l2_ctrl
*awb
;
42 struct v4l2_ctrl
*gain
;
43 struct v4l2_ctrl
*red_balance
;
44 struct v4l2_ctrl
*blue_balance
;
54 SENSOR_LT168G
, /* must verify if this is the actual model */
57 static const struct v4l2_pix_format vga_mode_t16
[] = {
58 {160, 120, V4L2_PIX_FMT_JPEG
, V4L2_FIELD_NONE
,
60 .sizeimage
= 160 * 120 * 4 / 8 + 590,
61 .colorspace
= V4L2_COLORSPACE_JPEG
,
63 #if 0 /* HDG: broken with my test cam, so lets disable it */
64 {176, 144, V4L2_PIX_FMT_JPEG
, V4L2_FIELD_NONE
,
66 .sizeimage
= 176 * 144 * 3 / 8 + 590,
67 .colorspace
= V4L2_COLORSPACE_JPEG
,
70 {320, 240, V4L2_PIX_FMT_JPEG
, V4L2_FIELD_NONE
,
72 .sizeimage
= 320 * 240 * 3 / 8 + 590,
73 .colorspace
= V4L2_COLORSPACE_JPEG
,
75 #if 0 /* HDG: broken with my test cam, so lets disable it */
76 {352, 288, V4L2_PIX_FMT_JPEG
, V4L2_FIELD_NONE
,
78 .sizeimage
= 352 * 288 * 3 / 8 + 590,
79 .colorspace
= V4L2_COLORSPACE_JPEG
,
82 {640, 480, V4L2_PIX_FMT_JPEG
, V4L2_FIELD_NONE
,
84 .sizeimage
= 640 * 480 * 3 / 8 + 590,
85 .colorspace
= V4L2_COLORSPACE_JPEG
,
89 /* sensor specific data */
90 struct additional_sensor_data
{
93 const u8 reg80
, reg8e
;
102 static const u8 n4_om6802
[] = {
103 0x09, 0x01, 0x12, 0x04, 0x66, 0x8a, 0x80, 0x3c,
104 0x81, 0x22, 0x84, 0x50, 0x8a, 0x78, 0x8b, 0x68,
105 0x8c, 0x88, 0x8e, 0x33, 0x8f, 0x24, 0xaa, 0xb1,
106 0xa2, 0x60, 0xa5, 0x30, 0xa6, 0x3a, 0xa8, 0xe8,
107 0xae, 0x05, 0xb1, 0x00, 0xbb, 0x04, 0xbc, 0x48,
108 0xbe, 0x36, 0xc6, 0x88, 0xe9, 0x00, 0xc5, 0xc0,
109 0x65, 0x0a, 0xbb, 0x86, 0xaf, 0x58, 0xb0, 0x68,
110 0x87, 0x40, 0x89, 0x2b, 0x8d, 0xff, 0x83, 0x40,
111 0xac, 0x84, 0xad, 0x86, 0xaf, 0x46
113 static const u8 n4_other
[] = {
114 0x66, 0x00, 0x7f, 0x00, 0x80, 0xac, 0x81, 0x69,
115 0x84, 0x40, 0x85, 0x70, 0x86, 0x20, 0x8a, 0x68,
116 0x8b, 0x58, 0x8c, 0x88, 0x8d, 0xff, 0x8e, 0xb8,
117 0x8f, 0x28, 0xa2, 0x60, 0xa5, 0x40, 0xa8, 0xa8,
118 0xac, 0x84, 0xad, 0x84, 0xae, 0x24, 0xaf, 0x56,
119 0xb0, 0x68, 0xb1, 0x00, 0xb2, 0x88, 0xbb, 0xc5,
120 0xbc, 0x4a, 0xbe, 0x36, 0xc2, 0x88, 0xc5, 0xc0,
121 0xc6, 0xda, 0xe9, 0x26, 0xeb, 0x00
123 static const u8 n4_tas5130a
[] = {
124 0x80, 0x3c, 0x81, 0x68, 0x83, 0xa0, 0x84, 0x20,
125 0x8a, 0x68, 0x8b, 0x58, 0x8c, 0x88, 0x8e, 0xb4,
126 0x8f, 0x24, 0xa1, 0xb1, 0xa2, 0x30, 0xa5, 0x10,
127 0xa6, 0x4a, 0xae, 0x03, 0xb1, 0x44, 0xb2, 0x08,
128 0xb7, 0x06, 0xb9, 0xe7, 0xbb, 0xc4, 0xbc, 0x4a,
129 0xbe, 0x36, 0xbf, 0xff, 0xc2, 0x88, 0xc5, 0xc8,
132 static const u8 n4_lt168g
[] = {
133 0x66, 0x01, 0x7f, 0x00, 0x80, 0x7c, 0x81, 0x28,
134 0x83, 0x44, 0x84, 0x20, 0x86, 0x20, 0x8a, 0x70,
135 0x8b, 0x58, 0x8c, 0x88, 0x8d, 0xa0, 0x8e, 0xb3,
136 0x8f, 0x24, 0xa1, 0xb0, 0xa2, 0x38, 0xa5, 0x20,
137 0xa6, 0x4a, 0xa8, 0xe8, 0xaf, 0x38, 0xb0, 0x68,
138 0xb1, 0x44, 0xb2, 0x88, 0xbb, 0x86, 0xbd, 0x40,
139 0xbe, 0x26, 0xc1, 0x05, 0xc2, 0x88, 0xc5, 0xc0,
140 0xda, 0x8e, 0xdb, 0xca, 0xdc, 0xa8, 0xdd, 0x8c,
141 0xde, 0x44, 0xdf, 0x0c, 0xe9, 0x80
144 static const struct additional_sensor_data sensor_data
[] = {
147 {0x61, 0x68, 0x65, 0x0a, 0x60, 0x04},
149 .n4sz
= sizeof n4_om6802
,
152 .nset8
= {0xa8, 0xf0, 0xc6, 0x88, 0xc0, 0x00},
154 {0xc2, 0x28, 0x0f, 0x22, 0xcd, 0x27, 0x2c, 0x06,
157 {0x80, 0xff, 0xff, 0x80, 0xff, 0xff, 0x80, 0xff,
160 {0x80, 0xff, 0xff, 0x80, 0xff, 0xff, 0x80, 0xff,
162 .data5
= /* this could be removed later */
163 {0x0c, 0x03, 0xab, 0x13, 0x81, 0x23},
165 {0x0b, 0x04, 0x0a, 0x78},
169 {0x61, 0xc2, 0x65, 0x88, 0x60, 0x00},
171 .n4sz
= sizeof n4_other
,
174 .nset8
= {0xa8, 0xa8, 0xc6, 0xda, 0xc0, 0x00},
176 {0xc1, 0x48, 0x04, 0x1b, 0xca, 0x2e, 0x33, 0x3a,
179 {0x4e, 0x9c, 0xec, 0x40, 0x80, 0xc0, 0x48, 0x96,
182 {0x4e, 0x9c, 0xec, 0x40, 0x80, 0xc0, 0x48, 0x96,
185 {0x0c, 0x03, 0xab, 0x29, 0x81, 0x69},
187 {0x0b, 0x04, 0x0a, 0x00},
189 [SENSOR_TAS5130A
] = {
191 {0x61, 0xc2, 0x65, 0x0d, 0x60, 0x08},
193 .n4sz
= sizeof n4_tas5130a
,
196 .nset8
= {0xa8, 0xf0, 0xc6, 0xda, 0xc0, 0x00},
198 {0xbb, 0x28, 0x10, 0x10, 0xbb, 0x28, 0x1e, 0x27,
201 {0x60, 0xa8, 0xe0, 0x60, 0xa8, 0xe0, 0x60, 0xa8,
204 {0x60, 0xa8, 0xe0, 0x60, 0xa8, 0xe0, 0x60, 0xa8,
207 {0x0c, 0x03, 0xab, 0x10, 0x81, 0x20},
209 {0x0b, 0x04, 0x0a, 0x40},
212 .n3
= {0x61, 0xc2, 0x65, 0x68, 0x60, 0x00},
214 .n4sz
= sizeof n4_lt168g
,
217 .nset8
= {0xa8, 0xf0, 0xc6, 0xba, 0xc0, 0x00},
218 .data1
= {0xc0, 0x38, 0x08, 0x10, 0xc0, 0x30, 0x10, 0x40,
220 .data2
= {0x40, 0x80, 0xc0, 0x50, 0xa0, 0xf0, 0x53, 0xa6,
222 .data3
= {0x40, 0x80, 0xc0, 0x50, 0xa0, 0xf0, 0x53, 0xa6,
224 .data5
= {0x0c, 0x03, 0xab, 0x4b, 0x81, 0x2b},
225 .stream
= {0x0b, 0x04, 0x0a, 0x28},
229 #define MAX_EFFECTS 7
230 static const u8 effects_table
[MAX_EFFECTS
][6] = {
231 {0xa8, 0xe8, 0xc6, 0xd2, 0xc0, 0x00}, /* Normal */
232 {0xa8, 0xc8, 0xc6, 0x52, 0xc0, 0x04}, /* Repujar */
233 {0xa8, 0xe8, 0xc6, 0xd2, 0xc0, 0x20}, /* Monochrome */
234 {0xa8, 0xe8, 0xc6, 0xd2, 0xc0, 0x80}, /* Sepia */
235 {0xa8, 0xc8, 0xc6, 0x52, 0xc0, 0x02}, /* Croquis */
236 {0xa8, 0xc8, 0xc6, 0xd2, 0xc0, 0x10}, /* Sun Effect */
237 {0xa8, 0xc8, 0xc6, 0xd2, 0xc0, 0x40}, /* Negative */
240 #define GAMMA_MAX (15)
241 static const u8 gamma_table
[GAMMA_MAX
+1][17] = {
242 /* gamma table from cam1690.ini */
243 {0x00, 0x00, 0x01, 0x04, 0x08, 0x0e, 0x16, 0x21, /* 0 */
244 0x2e, 0x3d, 0x50, 0x65, 0x7d, 0x99, 0xb8, 0xdb,
246 {0x00, 0x01, 0x03, 0x08, 0x0e, 0x16, 0x21, 0x2d, /* 1 */
247 0x3c, 0x4d, 0x60, 0x75, 0x8d, 0xa6, 0xc2, 0xe1,
249 {0x00, 0x01, 0x05, 0x0b, 0x12, 0x1c, 0x28, 0x35, /* 2 */
250 0x45, 0x56, 0x69, 0x7e, 0x95, 0xad, 0xc7, 0xe3,
252 {0x00, 0x02, 0x07, 0x0f, 0x18, 0x24, 0x30, 0x3f, /* 3 */
253 0x4f, 0x61, 0x73, 0x88, 0x9d, 0xb4, 0xcd, 0xe6,
255 {0x00, 0x04, 0x0b, 0x15, 0x20, 0x2d, 0x3b, 0x4a, /* 4 */
256 0x5b, 0x6c, 0x7f, 0x92, 0xa7, 0xbc, 0xd2, 0xe9,
258 {0x00, 0x07, 0x11, 0x15, 0x20, 0x2d, 0x48, 0x58, /* 5 */
259 0x68, 0x79, 0x8b, 0x9d, 0xb0, 0xc4, 0xd7, 0xec,
261 {0x00, 0x0c, 0x1a, 0x29, 0x38, 0x47, 0x57, 0x67, /* 6 */
262 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee,
264 {0x00, 0x10, 0x20, 0x30, 0x40, 0x50, 0x60, 0x70, /* 7 */
265 0x80, 0x90, 0xa0, 0xb0, 0xc0, 0xd0, 0xe0, 0xf0,
267 {0x00, 0x15, 0x27, 0x38, 0x49, 0x59, 0x69, 0x79, /* 8 */
268 0x88, 0x97, 0xa7, 0xb6, 0xc4, 0xd3, 0xe2, 0xf0,
270 {0x00, 0x1c, 0x30, 0x43, 0x54, 0x65, 0x75, 0x84, /* 9 */
271 0x93, 0xa1, 0xb0, 0xbd, 0xca, 0xd8, 0xe5, 0xf2,
273 {0x00, 0x24, 0x3b, 0x4f, 0x60, 0x70, 0x80, 0x8e, /* 10 */
274 0x9c, 0xaa, 0xb7, 0xc4, 0xd0, 0xdc, 0xe8, 0xf3,
276 {0x00, 0x2a, 0x3c, 0x5d, 0x6e, 0x7e, 0x8d, 0x9b, /* 11 */
277 0xa8, 0xb4, 0xc0, 0xcb, 0xd6, 0xe1, 0xeb, 0xf5,
279 {0x00, 0x3f, 0x5a, 0x6e, 0x7f, 0x8e, 0x9c, 0xa8, /* 12 */
280 0xb4, 0xbf, 0xc9, 0xd3, 0xdc, 0xe5, 0xee, 0xf6,
282 {0x00, 0x54, 0x6f, 0x83, 0x93, 0xa0, 0xad, 0xb7, /* 13 */
283 0xc2, 0xcb, 0xd4, 0xdc, 0xe4, 0xeb, 0xf2, 0xf9,
285 {0x00, 0x6e, 0x88, 0x9a, 0xa8, 0xb3, 0xbd, 0xc6, /* 14 */
286 0xcf, 0xd6, 0xdd, 0xe3, 0xe9, 0xef, 0xf4, 0xfa,
288 {0x00, 0x93, 0xa8, 0xb7, 0xc1, 0xca, 0xd2, 0xd8, /* 15 */
289 0xde, 0xe3, 0xe8, 0xed, 0xf1, 0xf5, 0xf8, 0xfc,
293 static const u8 tas5130a_sensor_init
[][8] = {
294 {0x62, 0x08, 0x63, 0x70, 0x64, 0x1d, 0x60, 0x09},
295 {0x62, 0x20, 0x63, 0x01, 0x64, 0x02, 0x60, 0x09},
296 {0x62, 0x07, 0x63, 0x03, 0x64, 0x00, 0x60, 0x09},
299 static u8 sensor_reset
[] = {0x61, 0x68, 0x62, 0xff, 0x60, 0x07};
302 static u8
reg_r(struct gspca_dev
*gspca_dev
,
305 usb_control_msg(gspca_dev
->dev
,
306 usb_rcvctrlpipe(gspca_dev
->dev
, 0),
308 USB_DIR_IN
| USB_TYPE_VENDOR
| USB_RECIP_DEVICE
,
311 gspca_dev
->usb_buf
, 1, 500);
312 return gspca_dev
->usb_buf
[0];
315 static void reg_w(struct gspca_dev
*gspca_dev
,
318 usb_control_msg(gspca_dev
->dev
,
319 usb_sndctrlpipe(gspca_dev
->dev
, 0),
321 USB_DIR_OUT
| USB_TYPE_VENDOR
| USB_RECIP_DEVICE
,
326 static void reg_w_buf(struct gspca_dev
*gspca_dev
,
327 const u8
*buffer
, u16 len
)
329 if (len
<= USB_BUF_SZ
) {
330 memcpy(gspca_dev
->usb_buf
, buffer
, len
);
331 usb_control_msg(gspca_dev
->dev
,
332 usb_sndctrlpipe(gspca_dev
->dev
, 0),
334 USB_DIR_OUT
| USB_TYPE_VENDOR
| USB_RECIP_DEVICE
,
336 gspca_dev
->usb_buf
, len
, 500);
340 tmpbuf
= kmemdup(buffer
, len
, GFP_KERNEL
);
342 pr_err("Out of memory\n");
345 usb_control_msg(gspca_dev
->dev
,
346 usb_sndctrlpipe(gspca_dev
->dev
, 0),
348 USB_DIR_OUT
| USB_TYPE_VENDOR
| USB_RECIP_DEVICE
,
355 /* write values to consecutive registers */
356 static void reg_w_ixbuf(struct gspca_dev
*gspca_dev
,
358 const u8
*buffer
, u16 len
)
363 if (len
* 2 <= USB_BUF_SZ
) {
364 p
= tmpbuf
= gspca_dev
->usb_buf
;
366 p
= tmpbuf
= kmalloc(len
* 2, GFP_KERNEL
);
368 pr_err("Out of memory\n");
377 usb_control_msg(gspca_dev
->dev
,
378 usb_sndctrlpipe(gspca_dev
->dev
, 0),
380 USB_DIR_OUT
| USB_TYPE_VENDOR
| USB_RECIP_DEVICE
,
382 tmpbuf
, len
* 2, 500);
383 if (len
* 2 > USB_BUF_SZ
)
387 static void om6802_sensor_init(struct gspca_dev
*gspca_dev
)
392 u8 val
[6] = {0x62, 0, 0x64, 0, 0x60, 0x05};
393 static const u8 sensor_init
[] = {
411 reg_w_buf(gspca_dev
, sensor_reset
, sizeof sensor_reset
);
415 byte
= reg_r(gspca_dev
, 0x0060);
420 byte
= reg_r(gspca_dev
, 0x0063);
422 pr_err("Bad sensor reset %02x\n", byte
);
431 reg_w(gspca_dev
, 0x3c80);
432 reg_w_buf(gspca_dev
, val
, sizeof val
);
436 byte
= reg_r(gspca_dev
, 0x60);
442 reg_w(gspca_dev
, 0x3c80);
445 /* this function is called at probe time */
446 static int sd_config(struct gspca_dev
*gspca_dev
,
447 const struct usb_device_id
*id
)
449 struct cam
*cam
= &gspca_dev
->cam
;
451 cam
->cam_mode
= vga_mode_t16
;
452 cam
->nmodes
= ARRAY_SIZE(vga_mode_t16
);
457 static void setbrightness(struct gspca_dev
*gspca_dev
, s32 brightness
)
459 u8 set6
[4] = { 0x8f, 0x24, 0xc3, 0x00 };
461 if (brightness
< 7) {
463 set6
[3] = 0x70 - brightness
* 0x10;
465 set6
[3] = 0x00 + ((brightness
- 7) * 0x10);
468 reg_w_buf(gspca_dev
, set6
, sizeof set6
);
471 static void setcontrast(struct gspca_dev
*gspca_dev
, s32 contrast
)
476 reg_to_write
= 0x8ea9 - contrast
* 0x200;
478 reg_to_write
= 0x00a9 + (contrast
- 7) * 0x200;
480 reg_w(gspca_dev
, reg_to_write
);
483 static void setcolors(struct gspca_dev
*gspca_dev
, s32 val
)
487 reg_to_write
= 0x80bb + val
* 0x100; /* was 0xc0 */
488 reg_w(gspca_dev
, reg_to_write
);
491 static void setgamma(struct gspca_dev
*gspca_dev
, s32 val
)
493 gspca_dbg(gspca_dev
, D_CONF
, "Gamma: %d\n", val
);
494 reg_w_ixbuf(gspca_dev
, 0x90,
495 gamma_table
[val
], sizeof gamma_table
[0]);
498 static void setawb_n_RGB(struct gspca_dev
*gspca_dev
)
500 struct sd
*sd
= (struct sd
*) gspca_dev
;
501 u8 all_gain_reg
[8] = {
502 0x87, 0x00, 0x88, 0x00, 0x89, 0x00, 0x80, 0x00 };
503 s32 red_gain
, blue_gain
, green_gain
;
505 green_gain
= sd
->gain
->val
;
507 red_gain
= green_gain
+ sd
->red_balance
->val
;
510 else if (red_gain
< 0x10)
513 blue_gain
= green_gain
+ sd
->blue_balance
->val
;
514 if (blue_gain
> 0x40)
516 else if (blue_gain
< 0x10)
519 all_gain_reg
[1] = red_gain
;
520 all_gain_reg
[3] = blue_gain
;
521 all_gain_reg
[5] = green_gain
;
522 all_gain_reg
[7] = sensor_data
[sd
->sensor
].reg80
;
524 all_gain_reg
[7] &= ~0x04; /* AWB off */
526 reg_w_buf(gspca_dev
, all_gain_reg
, sizeof all_gain_reg
);
529 static void setsharpness(struct gspca_dev
*gspca_dev
, s32 val
)
533 reg_to_write
= 0x0aa6 + 0x1000 * val
;
535 reg_w(gspca_dev
, reg_to_write
);
538 static void setfreq(struct gspca_dev
*gspca_dev
, s32 val
)
540 struct sd
*sd
= (struct sd
*) gspca_dev
;
542 u8 freq
[4] = { 0x66, 0x00, 0xa8, 0xe8 };
544 switch (sd
->sensor
) {
558 case 0: /* no flicker */
567 reg_w_buf(gspca_dev
, freq
, sizeof freq
);
570 /* this function is called at probe and resume time */
571 static int sd_init(struct gspca_dev
*gspca_dev
)
573 /* some of this registers are not really needed, because
574 * they are overridden by setbrigthness, setcontrast, etc.,
575 * but won't hurt anyway, and can help someone with similar webcam
576 * to see the initial parameters.*/
577 struct sd
*sd
= (struct sd
*) gspca_dev
;
578 const struct additional_sensor_data
*sensor
;
583 static const u8 read_indexs
[] =
584 { 0x0a, 0x0b, 0x66, 0x80, 0x81, 0x8e, 0x8f, 0xa5,
585 0xa6, 0xa8, 0xbb, 0xbc, 0xc6, 0x00 };
586 static const u8 n1
[] =
587 {0x08, 0x03, 0x09, 0x03, 0x12, 0x04};
588 static const u8 n2
[] =
591 sensor_id
= (reg_r(gspca_dev
, 0x06) << 8)
592 | reg_r(gspca_dev
, 0x07);
593 switch (sensor_id
& 0xff0f) {
595 gspca_dbg(gspca_dev
, D_PROBE
, "sensor tas5130a\n");
596 sd
->sensor
= SENSOR_TAS5130A
;
599 gspca_dbg(gspca_dev
, D_PROBE
, "sensor lt168g\n");
600 sd
->sensor
= SENSOR_LT168G
;
603 gspca_dbg(gspca_dev
, D_PROBE
, "sensor 'other'\n");
604 sd
->sensor
= SENSOR_OTHER
;
607 gspca_dbg(gspca_dev
, D_PROBE
, "sensor om6802\n");
608 sd
->sensor
= SENSOR_OM6802
;
611 pr_err("unknown sensor %04x\n", sensor_id
);
615 if (sd
->sensor
== SENSOR_OM6802
) {
616 reg_w_buf(gspca_dev
, n1
, sizeof n1
);
619 reg_w_buf(gspca_dev
, sensor_reset
, sizeof sensor_reset
);
620 test_byte
= reg_r(gspca_dev
, 0x0063);
622 if (test_byte
== 0x17)
626 pr_err("Bad sensor reset %02x\n", test_byte
);
629 reg_w_buf(gspca_dev
, n2
, sizeof n2
);
633 while (read_indexs
[i
] != 0x00) {
634 test_byte
= reg_r(gspca_dev
, read_indexs
[i
]);
635 gspca_dbg(gspca_dev
, D_STREAM
, "Reg 0x%02x = 0x%02x\n",
636 read_indexs
[i
], test_byte
);
640 sensor
= &sensor_data
[sd
->sensor
];
641 reg_w_buf(gspca_dev
, sensor
->n3
, sizeof sensor
->n3
);
642 reg_w_buf(gspca_dev
, sensor
->n4
, sensor
->n4sz
);
644 if (sd
->sensor
== SENSOR_LT168G
) {
645 test_byte
= reg_r(gspca_dev
, 0x80);
646 gspca_dbg(gspca_dev
, D_STREAM
, "Reg 0x%02x = 0x%02x\n", 0x80,
648 reg_w(gspca_dev
, 0x6c80);
651 reg_w_ixbuf(gspca_dev
, 0xd0, sensor
->data1
, sizeof sensor
->data1
);
652 reg_w_ixbuf(gspca_dev
, 0xc7, sensor
->data2
, sizeof sensor
->data2
);
653 reg_w_ixbuf(gspca_dev
, 0xe0, sensor
->data3
, sizeof sensor
->data3
);
655 reg_w(gspca_dev
, (sensor
->reg80
<< 8) + 0x80);
656 reg_w(gspca_dev
, (sensor
->reg80
<< 8) + 0x80);
657 reg_w(gspca_dev
, (sensor
->reg8e
<< 8) + 0x8e);
658 reg_w(gspca_dev
, (0x20 << 8) + 0x87);
659 reg_w(gspca_dev
, (0x20 << 8) + 0x88);
660 reg_w(gspca_dev
, (0x20 << 8) + 0x89);
662 reg_w_buf(gspca_dev
, sensor
->data5
, sizeof sensor
->data5
);
663 reg_w_buf(gspca_dev
, sensor
->nset8
, sizeof sensor
->nset8
);
664 reg_w_buf(gspca_dev
, sensor
->stream
, sizeof sensor
->stream
);
666 if (sd
->sensor
== SENSOR_LT168G
) {
667 test_byte
= reg_r(gspca_dev
, 0x80);
668 gspca_dbg(gspca_dev
, D_STREAM
, "Reg 0x%02x = 0x%02x\n", 0x80,
670 reg_w(gspca_dev
, 0x6c80);
673 reg_w_ixbuf(gspca_dev
, 0xd0, sensor
->data1
, sizeof sensor
->data1
);
674 reg_w_ixbuf(gspca_dev
, 0xc7, sensor
->data2
, sizeof sensor
->data2
);
675 reg_w_ixbuf(gspca_dev
, 0xe0, sensor
->data3
, sizeof sensor
->data3
);
680 static void setmirror(struct gspca_dev
*gspca_dev
, s32 val
)
683 {0x62, 0x07, 0x63, 0x03, 0x64, 0x00, 0x60, 0x09};
688 reg_w_buf(gspca_dev
, hflipcmd
, sizeof hflipcmd
);
691 static void seteffect(struct gspca_dev
*gspca_dev
, s32 val
)
696 case V4L2_COLORFX_NONE
:
698 case V4L2_COLORFX_BW
:
701 case V4L2_COLORFX_SEPIA
:
704 case V4L2_COLORFX_SKETCH
:
707 case V4L2_COLORFX_NEGATIVE
:
714 reg_w_buf(gspca_dev
, effects_table
[idx
],
715 sizeof effects_table
[0]);
717 if (val
== V4L2_COLORFX_SKETCH
)
718 reg_w(gspca_dev
, 0x4aa6);
720 reg_w(gspca_dev
, 0xfaa6);
723 /* Is this really needed?
724 * i added some module parameters for test with some users */
725 static void poll_sensor(struct gspca_dev
*gspca_dev
)
727 static const u8 poll1
[] =
728 {0x67, 0x05, 0x68, 0x81, 0x69, 0x80, 0x6a, 0x82,
729 0x6b, 0x68, 0x6c, 0x69, 0x72, 0xd9, 0x73, 0x34,
730 0x74, 0x32, 0x75, 0x92, 0x76, 0x00, 0x09, 0x01,
732 static const u8 poll2
[] =
733 {0x67, 0x02, 0x68, 0x71, 0x69, 0x72, 0x72, 0xa9,
734 0x73, 0x02, 0x73, 0x02, 0x60, 0x14};
735 static const u8 noise03
[] = /* (some differences / ms-drv) */
736 {0xa6, 0x0a, 0xea, 0xcf, 0xbe, 0x26, 0xb1, 0x5f,
737 0xa1, 0xb1, 0xda, 0x6b, 0xdb, 0x98, 0xdf, 0x0c,
738 0xc2, 0x80, 0xc3, 0x10};
740 gspca_dbg(gspca_dev
, D_STREAM
, "[Sensor requires polling]\n");
741 reg_w_buf(gspca_dev
, poll1
, sizeof poll1
);
742 reg_w_buf(gspca_dev
, poll2
, sizeof poll2
);
743 reg_w_buf(gspca_dev
, noise03
, sizeof noise03
);
746 static int sd_start(struct gspca_dev
*gspca_dev
)
748 struct sd
*sd
= (struct sd
*) gspca_dev
;
749 const struct additional_sensor_data
*sensor
;
751 u8 t2
[] = { 0x07, 0x00, 0x0d, 0x60, 0x0e, 0x80 };
752 static const u8 t3
[] =
753 { 0x07, 0x00, 0x88, 0x02, 0x06, 0x00, 0xe7, 0x01 };
755 mode
= gspca_dev
->cam
.cam_mode
[gspca_dev
->curr_mode
].priv
;
757 case 0: /* 640x480 (0x00) */
759 case 1: /* 352x288 */
762 case 2: /* 320x240 */
765 case 3: /* 176x144 */
769 /* case 4: * 160x120 */
774 switch (sd
->sensor
) {
776 om6802_sensor_init(gspca_dev
);
778 case SENSOR_TAS5130A
:
781 reg_w_buf(gspca_dev
, tas5130a_sensor_init
[i
],
782 sizeof tas5130a_sensor_init
[0]);
783 if (i
>= ARRAY_SIZE(tas5130a_sensor_init
) - 1)
787 reg_w(gspca_dev
, 0x3c80);
788 /* just in case and to keep sync with logs (for mine) */
789 reg_w_buf(gspca_dev
, tas5130a_sensor_init
[i
],
790 sizeof tas5130a_sensor_init
[0]);
791 reg_w(gspca_dev
, 0x3c80);
794 sensor
= &sensor_data
[sd
->sensor
];
795 setfreq(gspca_dev
, v4l2_ctrl_g_ctrl(sd
->freq
));
796 reg_r(gspca_dev
, 0x0012);
797 reg_w_buf(gspca_dev
, t2
, sizeof t2
);
798 reg_w_ixbuf(gspca_dev
, 0xb3, t3
, sizeof t3
);
799 reg_w(gspca_dev
, 0x0013);
801 reg_w_buf(gspca_dev
, sensor
->stream
, sizeof sensor
->stream
);
802 reg_w_buf(gspca_dev
, sensor
->stream
, sizeof sensor
->stream
);
804 if (sd
->sensor
== SENSOR_OM6802
)
805 poll_sensor(gspca_dev
);
810 static void sd_stopN(struct gspca_dev
*gspca_dev
)
812 struct sd
*sd
= (struct sd
*) gspca_dev
;
814 reg_w_buf(gspca_dev
, sensor_data
[sd
->sensor
].stream
,
815 sizeof sensor_data
[sd
->sensor
].stream
);
816 reg_w_buf(gspca_dev
, sensor_data
[sd
->sensor
].stream
,
817 sizeof sensor_data
[sd
->sensor
].stream
);
818 if (sd
->sensor
== SENSOR_OM6802
) {
820 reg_w(gspca_dev
, 0x0309);
822 #if IS_ENABLED(CONFIG_INPUT)
823 /* If the last button state is pressed, release it now! */
824 if (sd
->button_pressed
) {
825 input_report_key(gspca_dev
->input_dev
, KEY_CAMERA
, 0);
826 input_sync(gspca_dev
->input_dev
);
827 sd
->button_pressed
= 0;
832 static void sd_pkt_scan(struct gspca_dev
*gspca_dev
,
833 u8
*data
, /* isoc packet */
834 int len
) /* iso packet length */
836 struct sd
*sd __maybe_unused
= (struct sd
*) gspca_dev
;
839 if (data
[0] == 0x5a) {
840 #if IS_ENABLED(CONFIG_INPUT)
842 u8 state
= (data
[20] & 0x80) ? 1 : 0;
843 if (sd
->button_pressed
!= state
) {
844 input_report_key(gspca_dev
->input_dev
,
846 input_sync(gspca_dev
->input_dev
);
847 sd
->button_pressed
= state
;
851 /* Control Packet, after this came the header again,
852 * but extra bytes came in the packet before this,
853 * sometimes an EOF arrives, sometimes not... */
858 if (data
[0] == 0xff && data
[1] == 0xd8)
859 pkt_type
= FIRST_PACKET
;
860 else if (data
[len
- 2] == 0xff && data
[len
- 1] == 0xd9)
861 pkt_type
= LAST_PACKET
;
863 pkt_type
= INTER_PACKET
;
864 gspca_frame_add(gspca_dev
, pkt_type
, data
, len
);
867 static int sd_g_volatile_ctrl(struct v4l2_ctrl
*ctrl
)
869 struct gspca_dev
*gspca_dev
=
870 container_of(ctrl
->handler
, struct gspca_dev
, ctrl_handler
);
871 struct sd
*sd
= (struct sd
*)gspca_dev
;
872 s32 red_gain
, blue_gain
, green_gain
;
874 gspca_dev
->usb_err
= 0;
877 case V4L2_CID_AUTO_WHITE_BALANCE
:
878 red_gain
= reg_r(gspca_dev
, 0x0087);
881 else if (red_gain
< 0x10)
884 blue_gain
= reg_r(gspca_dev
, 0x0088);
885 if (blue_gain
> 0x40)
887 else if (blue_gain
< 0x10)
890 green_gain
= reg_r(gspca_dev
, 0x0089);
891 if (green_gain
> 0x40)
893 else if (green_gain
< 0x10)
896 sd
->gain
->val
= green_gain
;
897 sd
->red_balance
->val
= red_gain
- green_gain
;
898 sd
->blue_balance
->val
= blue_gain
- green_gain
;
904 static int sd_s_ctrl(struct v4l2_ctrl
*ctrl
)
906 struct gspca_dev
*gspca_dev
=
907 container_of(ctrl
->handler
, struct gspca_dev
, ctrl_handler
);
909 gspca_dev
->usb_err
= 0;
911 if (!gspca_dev
->streaming
)
915 case V4L2_CID_BRIGHTNESS
:
916 setbrightness(gspca_dev
, ctrl
->val
);
918 case V4L2_CID_CONTRAST
:
919 setcontrast(gspca_dev
, ctrl
->val
);
921 case V4L2_CID_SATURATION
:
922 setcolors(gspca_dev
, ctrl
->val
);
925 setgamma(gspca_dev
, ctrl
->val
);
928 setmirror(gspca_dev
, ctrl
->val
);
930 case V4L2_CID_SHARPNESS
:
931 setsharpness(gspca_dev
, ctrl
->val
);
933 case V4L2_CID_POWER_LINE_FREQUENCY
:
934 setfreq(gspca_dev
, ctrl
->val
);
936 case V4L2_CID_BACKLIGHT_COMPENSATION
:
937 reg_w(gspca_dev
, ctrl
->val
? 0xf48e : 0xb48e);
939 case V4L2_CID_AUTO_WHITE_BALANCE
:
940 setawb_n_RGB(gspca_dev
);
942 case V4L2_CID_COLORFX
:
943 seteffect(gspca_dev
, ctrl
->val
);
946 return gspca_dev
->usb_err
;
949 static const struct v4l2_ctrl_ops sd_ctrl_ops
= {
950 .g_volatile_ctrl
= sd_g_volatile_ctrl
,
954 static int sd_init_controls(struct gspca_dev
*gspca_dev
)
956 struct sd
*sd
= (struct sd
*)gspca_dev
;
957 struct v4l2_ctrl_handler
*hdl
= &gspca_dev
->ctrl_handler
;
959 gspca_dev
->vdev
.ctrl_handler
= hdl
;
960 v4l2_ctrl_handler_init(hdl
, 12);
961 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
962 V4L2_CID_BRIGHTNESS
, 0, 14, 1, 8);
963 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
964 V4L2_CID_CONTRAST
, 0, 0x0d, 1, 7);
965 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
966 V4L2_CID_SATURATION
, 0, 0xf, 1, 5);
967 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
968 V4L2_CID_GAMMA
, 0, GAMMA_MAX
, 1, 10);
969 /* Activate lowlight, some apps dont bring up the
970 backlight_compensation control) */
971 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
972 V4L2_CID_BACKLIGHT_COMPENSATION
, 0, 1, 1, 1);
973 if (sd
->sensor
== SENSOR_TAS5130A
)
974 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
975 V4L2_CID_HFLIP
, 0, 1, 1, 0);
976 sd
->awb
= v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
977 V4L2_CID_AUTO_WHITE_BALANCE
, 0, 1, 1, 1);
978 sd
->gain
= v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
979 V4L2_CID_GAIN
, 0x10, 0x40, 1, 0x20);
980 sd
->blue_balance
= v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
981 V4L2_CID_BLUE_BALANCE
, -0x30, 0x30, 1, 0);
982 sd
->red_balance
= v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
983 V4L2_CID_RED_BALANCE
, -0x30, 0x30, 1, 0);
984 v4l2_ctrl_new_std(hdl
, &sd_ctrl_ops
,
985 V4L2_CID_SHARPNESS
, 0, 15, 1, 6);
986 v4l2_ctrl_new_std_menu(hdl
, &sd_ctrl_ops
,
987 V4L2_CID_COLORFX
, V4L2_COLORFX_SKETCH
,
988 ~((1 << V4L2_COLORFX_NONE
) |
989 (1 << V4L2_COLORFX_BW
) |
990 (1 << V4L2_COLORFX_SEPIA
) |
991 (1 << V4L2_COLORFX_SKETCH
) |
992 (1 << V4L2_COLORFX_NEGATIVE
)),
994 sd
->freq
= v4l2_ctrl_new_std_menu(hdl
, &sd_ctrl_ops
,
995 V4L2_CID_POWER_LINE_FREQUENCY
,
996 V4L2_CID_POWER_LINE_FREQUENCY_60HZ
, 1,
997 V4L2_CID_POWER_LINE_FREQUENCY_50HZ
);
1000 pr_err("Could not initialize controls\n");
1004 v4l2_ctrl_auto_cluster(4, &sd
->awb
, 0, true);
1009 /* sub-driver description */
1010 static const struct sd_desc sd_desc
= {
1011 .name
= MODULE_NAME
,
1012 .config
= sd_config
,
1014 .init_controls
= sd_init_controls
,
1017 .pkt_scan
= sd_pkt_scan
,
1018 #if IS_ENABLED(CONFIG_INPUT)
1023 /* -- module initialisation -- */
1024 static const struct usb_device_id device_table
[] = {
1025 {USB_DEVICE(0x17a1, 0x0128)},
1028 MODULE_DEVICE_TABLE(usb
, device_table
);
1030 /* -- device connect -- */
1031 static int sd_probe(struct usb_interface
*intf
,
1032 const struct usb_device_id
*id
)
1034 return gspca_dev_probe(intf
, id
, &sd_desc
, sizeof(struct sd
),
1038 static struct usb_driver sd_driver
= {
1039 .name
= MODULE_NAME
,
1040 .id_table
= device_table
,
1042 .disconnect
= gspca_disconnect
,
1044 .suspend
= gspca_suspend
,
1045 .resume
= gspca_resume
,
1046 .reset_resume
= gspca_resume
,
1050 module_usb_driver(sd_driver
);