blob: a69f7682295eac342dc77da79b2156cc33b72b4b [file] [log] [blame]
Pete Eberleinb11869d2008-10-30 12:56:41 -07001/*
2 * Copyright (C) 2008 Sensoray Company Inc.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License (Version 2) as
6 * published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program; if not, write to the Free Software Foundation,
15 * Inc., 59 Temple Place - Suite 330, Boston MA 02111-1307, USA.
16 */
17
18#include <linux/module.h>
19#include <linux/init.h>
20#include <linux/usb.h>
21#include <linux/i2c.h>
22#include <linux/videodev2.h>
Pete Eberlein832b6a82009-11-16 15:13:51 -030023#include <media/v4l2-device.h>
Pete Eberleinb11869d2008-10-30 12:56:41 -070024#include <media/v4l2-common.h>
Pete Eberlein832b6a82009-11-16 15:13:51 -030025#include <media/v4l2-i2c-drv.h>
Pete Eberleinb11869d2008-10-30 12:56:41 -070026#include "go7007-priv.h"
Pete Eberlein832b6a82009-11-16 15:13:51 -030027
28MODULE_DESCRIPTION("Sensoray 2250/2251 i2c v4l2 subdev driver");
29MODULE_LICENSE("GPL v2");
Pete Eberleinb11869d2008-10-30 12:56:41 -070030
Pete Eberleinb11869d2008-10-30 12:56:41 -070031#define TLV320_ADDRESS 0x34
Pete Eberleinb11869d2008-10-30 12:56:41 -070032#define VPX322_ADDR_ANALOGCONTROL1 0x02
33#define VPX322_ADDR_BRIGHTNESS0 0x0127
34#define VPX322_ADDR_BRIGHTNESS1 0x0131
35#define VPX322_ADDR_CONTRAST0 0x0128
36#define VPX322_ADDR_CONTRAST1 0x0132
37#define VPX322_ADDR_HUE 0x00dc
Mauro Carvalho Chehab028d4c92009-09-15 11:06:04 -030038#define VPX322_ADDR_SAT 0x0030
Pete Eberleinb11869d2008-10-30 12:56:41 -070039
40struct go7007_usb_board {
41 unsigned int flags;
42 struct go7007_board_info main_info;
43};
44
45struct go7007_usb {
46 struct go7007_usb_board *board;
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -030047 struct mutex i2c_lock;
Pete Eberleinb11869d2008-10-30 12:56:41 -070048 struct usb_device *usbdev;
49 struct urb *video_urbs[8];
50 struct urb *audio_urbs[8];
51 struct urb *intr_urb;
52};
53
54static unsigned char aud_regs[] = {
55 0x1e, 0x00,
56 0x00, 0x17,
57 0x02, 0x17,
58 0x04, 0xf9,
59 0x06, 0xf9,
60 0x08, 0x02,
61 0x0a, 0x00,
62 0x0c, 0x00,
63 0x0a, 0x00,
64 0x0c, 0x00,
65 0x0e, 0x02,
66 0x10, 0x00,
67 0x12, 0x01,
68 0x00, 0x00,
69};
70
71
72static unsigned char vid_regs[] = {
73 0xF2, 0x0f,
74 0xAA, 0x00,
75 0xF8, 0xff,
76 0x00, 0x00,
77};
78
79static u16 vid_regs_fp[] = {
80 0x028, 0x067,
81 0x120, 0x016,
82 0x121, 0xcF2,
83 0x122, 0x0F2,
84 0x123, 0x00c,
85 0x124, 0x2d0,
86 0x125, 0x2e0,
87 0x126, 0x004,
88 0x128, 0x1E0,
89 0x12A, 0x016,
90 0x12B, 0x0F2,
91 0x12C, 0x0F2,
92 0x12D, 0x00c,
93 0x12E, 0x2d0,
94 0x12F, 0x2e0,
95 0x130, 0x004,
96 0x132, 0x1E0,
97 0x140, 0x060,
98 0x153, 0x00C,
99 0x154, 0x200,
100 0x150, 0x801,
101 0x000, 0x000
102};
103
104/* PAL specific values */
105static u16 vid_regs_fp_pal[] =
106{
107 0x120, 0x017,
108 0x121, 0xd22,
109 0x122, 0x122,
110 0x12A, 0x017,
111 0x12B, 0x122,
112 0x12C, 0x122,
113 0x140, 0x060,
114 0x000, 0x000,
115};
116
117struct s2250 {
Pete Eberlein047efc72009-09-18 22:55:13 -0300118 v4l2_std_id std;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700119 int input;
120 int brightness;
121 int contrast;
122 int saturation;
123 int hue;
124 int reg12b_val;
125 int audio_input;
Jean Delvare74005162009-04-21 21:47:22 +0200126 struct i2c_client *audio;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700127};
128
129/* from go7007-usb.c which is Copyright (C) 2005-2006 Micronas USA Inc.*/
130static int go7007_usb_vendor_request(struct go7007 *go, u16 request,
131 u16 value, u16 index, void *transfer_buffer, int length, int in)
132{
133 struct go7007_usb *usb = go->hpi_context;
134 int timeout = 5000;
135
136 if (in) {
137 return usb_control_msg(usb->usbdev,
138 usb_rcvctrlpipe(usb->usbdev, 0), request,
139 USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
140 value, index, transfer_buffer, length, timeout);
141 } else {
142 return usb_control_msg(usb->usbdev,
143 usb_sndctrlpipe(usb->usbdev, 0), request,
144 USB_TYPE_VENDOR | USB_RECIP_DEVICE,
145 value, index, transfer_buffer, length, timeout);
146 }
147}
148/* end from go7007-usb.c which is Copyright (C) 2005-2006 Micronas USA Inc.*/
149
150static int write_reg(struct i2c_client *client, u8 reg, u8 value)
151{
152 struct go7007 *go = i2c_get_adapdata(client->adapter);
Julia Lawall41e84782008-12-19 18:11:25 +0100153 struct go7007_usb *usb;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700154 int rc;
155 int dev_addr = client->addr;
156 u8 *buf;
157
158 if (go == NULL)
159 return -ENODEV;
160
161 if (go->status == STATUS_SHUTDOWN)
162 return -EBUSY;
163
164 buf = kzalloc(16, GFP_KERNEL);
165 if (buf == NULL)
166 return -ENOMEM;
167
Julia Lawall41e84782008-12-19 18:11:25 +0100168 usb = go->hpi_context;
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -0300169 if (mutex_lock_interruptible(&usb->i2c_lock) != 0) {
Pete Eberleinb11869d2008-10-30 12:56:41 -0700170 printk(KERN_INFO "i2c lock failed\n");
Julia Lawall253d3b12008-12-25 21:09:57 +0100171 kfree(buf);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700172 return -EINTR;
173 }
174 rc = go7007_usb_vendor_request(go, 0x55, dev_addr,
175 (reg<<8 | value),
176 buf,
177 16, 1);
178
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -0300179 mutex_unlock(&usb->i2c_lock);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700180 kfree(buf);
181 return rc;
182}
183
184static int write_reg_fp(struct i2c_client *client, u16 addr, u16 val)
185{
186 struct go7007 *go = i2c_get_adapdata(client->adapter);
Julia Lawall41e84782008-12-19 18:11:25 +0100187 struct go7007_usb *usb;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700188 u8 *buf;
189 struct s2250 *dec = i2c_get_clientdata(client);
190
191 if (go == NULL)
192 return -ENODEV;
193
194 if (go->status == STATUS_SHUTDOWN)
195 return -EBUSY;
196
197 buf = kzalloc(16, GFP_KERNEL);
198
199 if (buf == NULL)
200 return -ENOMEM;
201
202
203
204 memset(buf, 0xcd, 6);
205
Julia Lawall41e84782008-12-19 18:11:25 +0100206 usb = go->hpi_context;
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -0300207 if (mutex_lock_interruptible(&usb->i2c_lock) != 0) {
Pete Eberleinb11869d2008-10-30 12:56:41 -0700208 printk(KERN_INFO "i2c lock failed\n");
Pete Eberleind66ddf22009-09-18 22:05:19 -0300209 kfree(buf);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700210 return -EINTR;
211 }
Pete Eberleind66ddf22009-09-18 22:05:19 -0300212 if (go7007_usb_vendor_request(go, 0x57, addr, val, buf, 16, 1) < 0) {
213 kfree(buf);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700214 return -EFAULT;
Pete Eberleind66ddf22009-09-18 22:05:19 -0300215 }
Pete Eberleinb11869d2008-10-30 12:56:41 -0700216
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -0300217 mutex_unlock(&usb->i2c_lock);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700218 if (buf[0] == 0) {
219 unsigned int subaddr, val_read;
220
221 subaddr = (buf[4] << 8) + buf[5];
222 val_read = (buf[2] << 8) + buf[3];
Pete Eberleind66ddf22009-09-18 22:05:19 -0300223 kfree(buf);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700224 if (val_read != val) {
225 printk(KERN_INFO "invalid fp write %x %x\n",
226 val_read, val);
227 return -EFAULT;
228 }
229 if (subaddr != addr) {
230 printk(KERN_INFO "invalid fp write addr %x %x\n",
231 subaddr, addr);
232 return -EFAULT;
233 }
Pete Eberleind66ddf22009-09-18 22:05:19 -0300234 } else {
235 kfree(buf);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700236 return -EFAULT;
Pete Eberleind66ddf22009-09-18 22:05:19 -0300237 }
Pete Eberleinb11869d2008-10-30 12:56:41 -0700238
239 /* save last 12b value */
240 if (addr == 0x12b)
241 dec->reg12b_val = val;
242
243 return 0;
244}
245
Pete Eberlein047efc72009-09-18 22:55:13 -0300246static int read_reg_fp(struct i2c_client *client, u16 addr, u16 *val)
247{
248 struct go7007 *go = i2c_get_adapdata(client->adapter);
249 struct go7007_usb *usb;
250 u8 *buf;
251
252 if (go == NULL)
253 return -ENODEV;
254
255 if (go->status == STATUS_SHUTDOWN)
256 return -EBUSY;
257
258 buf = kzalloc(16, GFP_KERNEL);
259
260 if (buf == NULL)
261 return -ENOMEM;
262
263
264
265 memset(buf, 0xcd, 6);
266 usb = go->hpi_context;
Pete Eberleine49bd722009-11-15 08:14:14 -0300267 if (mutex_lock_interruptible(&usb->i2c_lock) != 0) {
Pete Eberlein047efc72009-09-18 22:55:13 -0300268 printk(KERN_INFO "i2c lock failed\n");
269 kfree(buf);
270 return -EINTR;
271 }
272 if (go7007_usb_vendor_request(go, 0x58, addr, 0, buf, 16, 1) < 0) {
273 kfree(buf);
274 return -EFAULT;
275 }
Pete Eberleine49bd722009-11-15 08:14:14 -0300276 mutex_unlock(&usb->i2c_lock);
Pete Eberlein047efc72009-09-18 22:55:13 -0300277
278 *val = (buf[0] << 8) | buf[1];
279 kfree(buf);
280
281 return 0;
282}
283
284
Pete Eberleinb11869d2008-10-30 12:56:41 -0700285static int write_regs(struct i2c_client *client, u8 *regs)
286{
287 int i;
288
289 for (i = 0; !((regs[i] == 0x00) && (regs[i+1] == 0x00)); i += 2) {
290 if (write_reg(client, regs[i], regs[i+1]) < 0) {
291 printk(KERN_INFO "s2250: failed\n");
292 return -1;
293 }
294 }
295 return 0;
296}
297
298static int write_regs_fp(struct i2c_client *client, u16 *regs)
299{
300 int i;
301
302 for (i = 0; !((regs[i] == 0x00) && (regs[i+1] == 0x00)); i += 2) {
303 if (write_reg_fp(client, regs[i], regs[i+1]) < 0) {
304 printk(KERN_INFO "s2250: failed fp\n");
305 return -1;
306 }
307 }
308 return 0;
309}
310
311
312static int s2250_command(struct i2c_client *client,
313 unsigned int cmd, void *arg)
314{
315 struct s2250 *dec = i2c_get_clientdata(client);
316
317 switch (cmd) {
318 case VIDIOC_S_INPUT:
319 {
320 int vidsys;
321 int *input = arg;
322
323 vidsys = (dec->std == V4L2_STD_NTSC) ? 0x01 : 0x00;
324 if (*input == 0) {
325 /* composite */
326 write_reg_fp(client, 0x20, 0x020 | vidsys);
327 write_reg_fp(client, 0x21, 0x662);
328 write_reg_fp(client, 0x140, 0x060);
329 } else {
330 /* S-Video */
331 write_reg_fp(client, 0x20, 0x040 | vidsys);
332 write_reg_fp(client, 0x21, 0x666);
333 write_reg_fp(client, 0x140, 0x060);
334 }
335 dec->input = *input;
336 break;
337 }
338 case VIDIOC_S_STD:
339 {
340 v4l2_std_id *std = arg;
341 u16 vidsource;
342
343 vidsource = (dec->input == 1) ? 0x040 : 0x020;
344 dec->std = *std;
345 switch (dec->std) {
346 case V4L2_STD_NTSC:
347 write_regs_fp(client, vid_regs_fp);
348 write_reg_fp(client, 0x20, vidsource | 1);
349 break;
350 case V4L2_STD_PAL:
351 write_regs_fp(client, vid_regs_fp);
352 write_regs_fp(client, vid_regs_fp_pal);
353 write_reg_fp(client, 0x20, vidsource);
354 break;
355 default:
356 return -EINVAL;
357 }
358 break;
359 }
360 case VIDIOC_QUERYCTRL:
361 {
362 struct v4l2_queryctrl *ctrl = arg;
363 static const u32 user_ctrls[] = {
364 V4L2_CID_BRIGHTNESS,
365 V4L2_CID_CONTRAST,
366 V4L2_CID_SATURATION,
367 V4L2_CID_HUE,
368 0
369 };
370 static const u32 *ctrl_classes[] = {
371 user_ctrls,
372 NULL
373 };
374
375 ctrl->id = v4l2_ctrl_next(ctrl_classes, ctrl->id);
376 switch (ctrl->id) {
377 case V4L2_CID_BRIGHTNESS:
378 v4l2_ctrl_query_fill(ctrl, 0, 100, 1, 50);
379 break;
380 case V4L2_CID_CONTRAST:
381 v4l2_ctrl_query_fill(ctrl, 0, 100, 1, 50);
382 break;
383 case V4L2_CID_SATURATION:
384 v4l2_ctrl_query_fill(ctrl, 0, 100, 1, 50);
385 break;
386 case V4L2_CID_HUE:
387 v4l2_ctrl_query_fill(ctrl, -50, 50, 1, 0);
388 break;
389 default:
390 ctrl->name[0] = '\0';
391 return -EINVAL;
392 }
393 break;
394 }
395 case VIDIOC_S_CTRL:
396 {
397 struct v4l2_control *ctrl = arg;
398 int value1;
Pete Eberlein047efc72009-09-18 22:55:13 -0300399 u16 oldvalue;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700400
401 switch (ctrl->id) {
402 case V4L2_CID_BRIGHTNESS:
Pete Eberlein047efc72009-09-18 22:55:13 -0300403 if (ctrl->value > 100)
404 dec->brightness = 100;
405 else if (ctrl->value < 0)
406 dec->brightness = 0;
407 else
408 dec->brightness = ctrl->value;
409 value1 = (dec->brightness - 50) * 255 / 100;
410 read_reg_fp(client, VPX322_ADDR_BRIGHTNESS0, &oldvalue);
411 write_reg_fp(client, VPX322_ADDR_BRIGHTNESS0,
412 value1 | (oldvalue & ~0xff));
413 read_reg_fp(client, VPX322_ADDR_BRIGHTNESS1, &oldvalue);
414 write_reg_fp(client, VPX322_ADDR_BRIGHTNESS1,
415 value1 | (oldvalue & ~0xff));
416 write_reg_fp(client, 0x140, 0x60);
417 break;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700418 case V4L2_CID_CONTRAST:
Pete Eberlein047efc72009-09-18 22:55:13 -0300419 if (ctrl->value > 100)
420 dec->contrast = 100;
421 else if (ctrl->value < 0)
422 dec->contrast = 0;
423 else
424 dec->contrast = ctrl->value;
425 value1 = dec->contrast * 0x40 / 100;
426 if (value1 > 0x3f)
427 value1 = 0x3f; /* max */
428 read_reg_fp(client, VPX322_ADDR_CONTRAST0, &oldvalue);
429 write_reg_fp(client, VPX322_ADDR_CONTRAST0,
430 value1 | (oldvalue & ~0x3f));
431 read_reg_fp(client, VPX322_ADDR_CONTRAST1, &oldvalue);
432 write_reg_fp(client, VPX322_ADDR_CONTRAST1,
433 value1 | (oldvalue & ~0x3f));
434 write_reg_fp(client, 0x140, 0x60);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700435 break;
436 case V4L2_CID_SATURATION:
437 if (ctrl->value > 127)
438 dec->saturation = 127;
439 else if (ctrl->value < 0)
440 dec->saturation = 0;
441 else
442 dec->saturation = ctrl->value;
443
444 value1 = dec->saturation * 4140 / 100;
445 if (value1 > 4094)
446 value1 = 4094;
447 write_reg_fp(client, VPX322_ADDR_SAT, value1);
448 break;
449 case V4L2_CID_HUE:
450 if (ctrl->value > 50)
451 dec->hue = 50;
452 else if (ctrl->value < -50)
453 dec->hue = -50;
454 else
455 dec->hue = ctrl->value;
456 /* clamp the hue range */
457 value1 = dec->hue * 280 / 50;
458 write_reg_fp(client, VPX322_ADDR_HUE, value1);
459 break;
460 }
461 break;
462 }
463 case VIDIOC_G_CTRL:
464 {
465 struct v4l2_control *ctrl = arg;
466
467 switch (ctrl->id) {
468 case V4L2_CID_BRIGHTNESS:
469 ctrl->value = dec->brightness;
470 break;
471 case V4L2_CID_CONTRAST:
472 ctrl->value = dec->contrast;
473 break;
474 case V4L2_CID_SATURATION:
475 ctrl->value = dec->saturation;
476 break;
477 case V4L2_CID_HUE:
478 ctrl->value = dec->hue;
479 break;
480 }
481 break;
482 }
483 case VIDIOC_S_FMT:
484 {
485 struct v4l2_format *fmt = arg;
486 if (fmt->fmt.pix.height < 640) {
487 write_reg_fp(client, 0x12b, dec->reg12b_val | 0x400);
488 write_reg_fp(client, 0x140, 0x060);
489 } else {
490 write_reg_fp(client, 0x12b, dec->reg12b_val & ~0x400);
491 write_reg_fp(client, 0x140, 0x060);
492 }
493 return 0;
494 }
495 case VIDIOC_G_AUDIO:
496 {
497 struct v4l2_audio *audio = arg;
498
499 memset(audio, 0, sizeof(*audio));
500 audio->index = dec->audio_input;
501 /* fall through */
502 }
503 case VIDIOC_ENUMAUDIO:
504 {
505 struct v4l2_audio *audio = arg;
506
507 switch (audio->index) {
508 case 0:
509 strcpy(audio->name, "Line In");
510 break;
511 case 1:
512 strcpy(audio->name, "Mic");
513 break;
514 case 2:
515 strcpy(audio->name, "Mic Boost");
516 break;
517 default:
518 audio->name[0] = '\0';
519 return 0;
520 }
521 audio->capability = V4L2_AUDCAP_STEREO;
522 audio->mode = 0;
523 return 0;
524 }
525 case VIDIOC_S_AUDIO:
526 {
527 struct v4l2_audio *audio = arg;
528
Pete Eberleinb11869d2008-10-30 12:56:41 -0700529 switch (audio->index) {
530 case 0:
Jean Delvare74005162009-04-21 21:47:22 +0200531 write_reg(dec->audio, 0x08, 0x02); /* Line In */
Pete Eberleinb11869d2008-10-30 12:56:41 -0700532 break;
533 case 1:
Jean Delvare74005162009-04-21 21:47:22 +0200534 write_reg(dec->audio, 0x08, 0x04); /* Mic */
Pete Eberleinb11869d2008-10-30 12:56:41 -0700535 break;
536 case 2:
Jean Delvare74005162009-04-21 21:47:22 +0200537 write_reg(dec->audio, 0x08, 0x05); /* Mic Boost */
Pete Eberleinb11869d2008-10-30 12:56:41 -0700538 break;
539 default:
540 return -EINVAL;
541 }
542 dec->audio_input = audio->index;
543 return 0;
544 }
545
546 default:
547 printk(KERN_INFO "s2250: unknown command 0x%x\n", cmd);
548 break;
549 }
550 return 0;
551}
552
Jean Delvare74005162009-04-21 21:47:22 +0200553static int s2250_probe(struct i2c_client *client,
554 const struct i2c_device_id *id)
Pete Eberleinb11869d2008-10-30 12:56:41 -0700555{
Jean Delvare74005162009-04-21 21:47:22 +0200556 struct i2c_client *audio;
557 struct i2c_adapter *adapter = client->adapter;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700558 struct s2250 *dec;
559 u8 *data;
560 struct go7007 *go = i2c_get_adapdata(adapter);
561 struct go7007_usb *usb = go->hpi_context;
562
Jean Delvare74005162009-04-21 21:47:22 +0200563 audio = i2c_new_dummy(adapter, TLV320_ADDRESS >> 1);
564 if (audio == NULL)
Pete Eberleinb11869d2008-10-30 12:56:41 -0700565 return -ENOMEM;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700566
567 dec = kmalloc(sizeof(struct s2250), GFP_KERNEL);
568 if (dec == NULL) {
Jean Delvare74005162009-04-21 21:47:22 +0200569 i2c_unregister_device(audio);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700570 return -ENOMEM;
571 }
572
573 dec->std = V4L2_STD_NTSC;
574 dec->brightness = 50;
575 dec->contrast = 50;
576 dec->saturation = 50;
577 dec->hue = 0;
Jean Delvare74005162009-04-21 21:47:22 +0200578 dec->audio = audio;
Pete Eberleinb11869d2008-10-30 12:56:41 -0700579 i2c_set_clientdata(client, dec);
580
Pete Eberlein832b6a82009-11-16 15:13:51 -0300581 printk(KERN_INFO
Pete Eberleinb11869d2008-10-30 12:56:41 -0700582 "s2250: initializing video decoder on %s\n",
583 adapter->name);
584
585 /* initialize the audio */
Jean Delvare74005162009-04-21 21:47:22 +0200586 if (write_regs(audio, aud_regs) < 0) {
Pete Eberleinb11869d2008-10-30 12:56:41 -0700587 printk(KERN_ERR
588 "s2250: error initializing audio\n");
Jean Delvare74005162009-04-21 21:47:22 +0200589 i2c_unregister_device(audio);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700590 kfree(dec);
591 return 0;
592 }
Pete Eberleinb11869d2008-10-30 12:56:41 -0700593
594 if (write_regs(client, vid_regs) < 0) {
595 printk(KERN_ERR
596 "s2250: error initializing decoder\n");
Jean Delvare74005162009-04-21 21:47:22 +0200597 i2c_unregister_device(audio);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700598 kfree(dec);
599 return 0;
600 }
601 if (write_regs_fp(client, vid_regs_fp) < 0) {
602 printk(KERN_ERR
603 "s2250: error initializing decoder\n");
Jean Delvare74005162009-04-21 21:47:22 +0200604 i2c_unregister_device(audio);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700605 kfree(dec);
606 return 0;
607 }
608 /* set default channel */
609 /* composite */
610 write_reg_fp(client, 0x20, 0x020 | 1);
611 write_reg_fp(client, 0x21, 0x662);
612 write_reg_fp(client, 0x140, 0x060);
613
614 /* set default audio input */
615 dec->audio_input = 0;
616 write_reg(client, 0x08, 0x02); /* Line In */
617
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -0300618 if (mutex_lock_interruptible(&usb->i2c_lock) == 0) {
Pete Eberleinb11869d2008-10-30 12:56:41 -0700619 data = kzalloc(16, GFP_KERNEL);
620 if (data != NULL) {
621 int rc;
622 rc = go7007_usb_vendor_request(go, 0x41, 0, 0,
623 data, 16, 1);
624 if (rc > 0) {
625 u8 mask;
626 data[0] = 0;
627 mask = 1<<5;
628 data[0] &= ~mask;
629 data[1] |= mask;
630 go7007_usb_vendor_request(go, 0x40, 0,
631 (data[1]<<8)
632 + data[1],
633 data, 16, 0);
634 }
635 kfree(data);
636 }
Mauro Carvalho Chehabfd9a40d2009-09-15 11:07:59 -0300637 mutex_unlock(&usb->i2c_lock);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700638 }
639
Pete Eberleinb11869d2008-10-30 12:56:41 -0700640 printk("s2250: initialized successfully\n");
641 return 0;
642}
643
Jean Delvare74005162009-04-21 21:47:22 +0200644static int s2250_remove(struct i2c_client *client)
Pete Eberleinb11869d2008-10-30 12:56:41 -0700645{
646 struct s2250 *dec = i2c_get_clientdata(client);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700647
Jean Delvare74005162009-04-21 21:47:22 +0200648 i2c_set_clientdata(client, NULL);
649 i2c_unregister_device(dec->audio);
Pete Eberleinb11869d2008-10-30 12:56:41 -0700650 kfree(dec);
651 return 0;
652}
653
Jean Delvare74005162009-04-21 21:47:22 +0200654static struct i2c_device_id s2250_id[] = {
Pete Eberlein832b6a82009-11-16 15:13:51 -0300655 { "s2250", 0 },
Jean Delvare74005162009-04-21 21:47:22 +0200656 { }
657};
Pete Eberlein832b6a82009-11-16 15:13:51 -0300658MODULE_DEVICE_TABLE(i2c, s2250_id);
Jean Delvare74005162009-04-21 21:47:22 +0200659
Pete Eberlein832b6a82009-11-16 15:13:51 -0300660static struct v4l2_i2c_driver_data v4l2_i2c_data = {
661 .name = "s2250",
662 .probe = s2250_probe,
663 .remove = s2250_remove,
664 .command = s2250_command,
665 .id_table = s2250_id,
Pete Eberleinb11869d2008-10-30 12:56:41 -0700666};