2 * device driver for Conexant 2388x based TV cards
5 * (c) 2003 Gerd Knorr <kraxel@bytesex.org> [SuSE Labs]
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 #include <linux/init.h>
23 #include <linux/module.h>
24 #include <linux/pci.h>
28 /* ------------------------------------------------------------------ */
29 /* board config info */
31 struct cx88_board cx88_boards[] = {
32 [CX88_BOARD_UNKNOWN] = {
33 .name = "UNKNOWN/GENERIC",
36 .type = CX88_VMUX_COMPOSITE1,
39 .type = CX88_VMUX_COMPOSITE2,
42 .type = CX88_VMUX_COMPOSITE3,
45 .type = CX88_VMUX_COMPOSITE4,
49 [CX88_BOARD_HAUPPAUGE] = {
50 .name = "Hauppauge WinTV 34xxx models",
53 .type = CX88_VMUX_TELEVISION,
55 .gpio0 = 0xff00, // internal decoder
57 .type = CX88_VMUX_DEBUG,
59 .gpio0 = 0xff01, // mono from tuner chip
61 .type = CX88_VMUX_COMPOSITE1,
65 .type = CX88_VMUX_SVIDEO,
75 .name = "GDI Black Gold",
78 .type = CX88_VMUX_TELEVISION,
82 [CX88_BOARD_PIXELVIEW] = {
86 .type = CX88_VMUX_TELEVISION,
89 .type = CX88_VMUX_COMPOSITE1,
92 .type = CX88_VMUX_SVIDEO,
96 [CX88_BOARD_ATI_WONDER_PRO] = {
97 .name = "ATI TV Wonder Pro",
100 .type = CX88_VMUX_TELEVISION,
103 .type = CX88_VMUX_COMPOSITE1,
106 .type = CX88_VMUX_SVIDEO,
111 [CX88_BOARD_WINFAST2000XP] = {
112 .name = "Leadtek Winfast 2000XP Expert",
115 .type = CX88_VMUX_TELEVISION,
122 .type = CX88_VMUX_COMPOSITE1,
129 .type = CX88_VMUX_SVIDEO,
144 [CX88_BOARD_AVERTV_303] = {
145 .name = "AverTV Studio 303 (M126)",
146 .tuner_type = TUNER_PHILIPS_PAL_DK,
148 .type = CX88_VMUX_TELEVISION,
152 [CX88_BOARD_MSI_TVANYWHERE] = {
153 .name = "MSI TV-@nywhere Master",
156 .type = CX88_VMUX_TELEVISION,
159 .type = CX88_VMUX_COMPOSITE1,
162 // temporarly for testing ...
163 .type = CX88_VMUX_COMPOSITE2,
166 .type = CX88_VMUX_SVIDEO,
173 [CX88_BOARD_WINFAST_DV2000] = {
174 .name = "Leadtek Winfast DV2000",
178 .type = CX88_VMUX_TELEVISION,
185 [CX88_BOARD_LEADTEK_PVR2000] = {
186 .name = "Leadtek PVR 2000",
189 .type = CX88_VMUX_TELEVISION,
192 .type = CX88_VMUX_COMPOSITE1,
195 .type = CX88_VMUX_SVIDEO,
205 const unsigned int cx88_bcount = ARRAY_SIZE(cx88_boards);
207 /* ------------------------------------------------------------------ */
208 /* PCI subsystem IDs */
210 struct cx88_subid cx88_subids[] = {
214 .card = CX88_BOARD_HAUPPAUGE,
218 .card = CX88_BOARD_HAUPPAUGE,
222 .card = CX88_BOARD_GDI,
225 .subdevice = 0x0107, /* with mpeg encoder */
226 .card = CX88_BOARD_GDI,
228 .subvendor = PCI_VENDOR_ID_ATI,
230 .card = CX88_BOARD_ATI_WONDER_PRO,
234 .card = CX88_BOARD_WINFAST2000XP,
237 .subdevice = 0x6613, /* NTSC */
238 .card = CX88_BOARD_WINFAST2000XP,
242 .card = CX88_BOARD_WINFAST_DV2000,
246 .card = CX88_BOARD_LEADTEK_PVR2000,
250 .card = CX88_BOARD_AVERTV_303,
254 .card = CX88_BOARD_MSI_TVANYWHERE,
257 const unsigned int cx88_idcount = ARRAY_SIZE(cx88_subids);
260 /* ----------------------------------------------------------------------- */
261 /* some leadtek specific stuff */
263 static void __devinit leadtek_eeprom(struct cx8800_dev *dev, u8 *eeprom_data)
265 /* This is just for the Winfast 2000 XP board ATM; I don't have data on
268 * Byte 0 is 1 on the NTSC board.
271 if (eeprom_data[4] != 0x7d ||
272 eeprom_data[5] != 0x10 ||
273 eeprom_data[7] != 0x66) {
274 printk(KERN_WARNING "%s Leadtek eeprom invalid.\n", dev->name);
279 dev->tuner_type = (eeprom_data[6] == 0x13) ? 43 : 38;
281 printk(KERN_INFO "%s: Leadtek Winfast 2000 XP config: "
282 "tuner=%d, eeprom[0]=0x%02x\n",
283 dev->name, dev->tuner_type, eeprom_data[0]);
287 /* ----------------------------------------------------------------------- */
288 /* some hauppauge specific stuff */
293 } hauppauge_tuner[] __devinitdata = {
294 { TUNER_ABSENT, "" },
295 { TUNER_ABSENT, "External" },
296 { TUNER_ABSENT, "Unspecified" },
297 { TUNER_PHILIPS_PAL, "Philips FI1216" },
298 { TUNER_PHILIPS_SECAM, "Philips FI1216MF" },
299 { TUNER_PHILIPS_NTSC, "Philips FI1236" },
300 { TUNER_PHILIPS_PAL_I, "Philips FI1246" },
301 { TUNER_PHILIPS_PAL_DK,"Philips FI1256" },
302 { TUNER_PHILIPS_PAL, "Philips FI1216 MK2" },
303 { TUNER_PHILIPS_SECAM, "Philips FI1216MF MK2" },
304 { TUNER_PHILIPS_NTSC, "Philips FI1236 MK2" },
305 { TUNER_PHILIPS_PAL_I, "Philips FI1246 MK2" },
306 { TUNER_PHILIPS_PAL_DK,"Philips FI1256 MK2" },
307 { TUNER_TEMIC_NTSC, "Temic 4032FY5" },
308 { TUNER_TEMIC_PAL, "Temic 4002FH5" },
309 { TUNER_TEMIC_PAL_I, "Temic 4062FY5" },
310 { TUNER_PHILIPS_PAL, "Philips FR1216 MK2" },
311 { TUNER_PHILIPS_SECAM, "Philips FR1216MF MK2" },
312 { TUNER_PHILIPS_NTSC, "Philips FR1236 MK2" },
313 { TUNER_PHILIPS_PAL_I, "Philips FR1246 MK2" },
314 { TUNER_PHILIPS_PAL_DK,"Philips FR1256 MK2" },
315 { TUNER_PHILIPS_PAL, "Philips FM1216" },
316 { TUNER_PHILIPS_SECAM, "Philips FM1216MF" },
317 { TUNER_PHILIPS_NTSC, "Philips FM1236" },
318 { TUNER_PHILIPS_PAL_I, "Philips FM1246" },
319 { TUNER_PHILIPS_PAL_DK,"Philips FM1256" },
320 { TUNER_TEMIC_4036FY5_NTSC, "Temic 4036FY5" },
321 { TUNER_ABSENT, "Samsung TCPN9082D" },
322 { TUNER_ABSENT, "Samsung TCPM9092P" },
323 { TUNER_TEMIC_4006FH5_PAL, "Temic 4006FH5" },
324 { TUNER_ABSENT, "Samsung TCPN9085D" },
325 { TUNER_ABSENT, "Samsung TCPB9085P" },
326 { TUNER_ABSENT, "Samsung TCPL9091P" },
327 { TUNER_TEMIC_4039FR5_NTSC, "Temic 4039FR5" },
328 { TUNER_PHILIPS_FQ1216ME, "Philips FQ1216 ME" },
329 { TUNER_TEMIC_4066FY5_PAL_I, "Temic 4066FY5" },
330 { TUNER_PHILIPS_NTSC, "Philips TD1536" },
331 { TUNER_PHILIPS_NTSC, "Philips TD1536D" },
332 { TUNER_PHILIPS_NTSC, "Philips FMR1236" }, /* mono radio */
333 { TUNER_ABSENT, "Philips FI1256MP" },
334 { TUNER_ABSENT, "Samsung TCPQ9091P" },
335 { TUNER_TEMIC_4006FN5_MULTI_PAL, "Temic 4006FN5" },
336 { TUNER_TEMIC_4009FR5_PAL, "Temic 4009FR5" },
337 { TUNER_TEMIC_4046FM5, "Temic 4046FM5" },
338 { TUNER_TEMIC_4009FN5_MULTI_PAL_FM, "Temic 4009FN5" },
339 { TUNER_ABSENT, "Philips TD1536D_FH_44"},
340 { TUNER_LG_NTSC_FM, "LG TPI8NSR01F"},
341 { TUNER_LG_PAL_FM, "LG TPI8PSB01D"},
342 { TUNER_LG_PAL, "LG TPI8PSB11D"},
343 { TUNER_LG_PAL_I_FM, "LG TAPC-I001D"},
344 { TUNER_LG_PAL_I, "LG TAPC-I701D"}
347 static void __devinit hauppauge_eeprom(struct cx8800_dev *dev, u8 *eeprom_data)
349 unsigned int blk2,tuner,radio,model;
351 if (eeprom_data[0] != 0x84 || eeprom_data[2] != 0) {
352 printk(KERN_WARNING "%s: Hauppauge eeprom: invalid\n",
357 /* Block 2 starts after len+3 bytes header */
358 blk2 = eeprom_data[1] + 3;
360 /* decode + use some config infos */
361 model = eeprom_data[12] << 8 | eeprom_data[11];
362 tuner = eeprom_data[9];
363 radio = eeprom_data[blk2-1] & 0x01;
365 if (tuner < ARRAY_SIZE(hauppauge_tuner))
366 dev->tuner_type = hauppauge_tuner[tuner].id;
370 printk(KERN_INFO "%s: hauppauge eeprom: model=%d, "
371 "tuner=%s (%d), radio=%s\n",
372 dev->name, model, hauppauge_tuner[tuner].name,
373 dev->tuner_type, radio ? "yes" : "no");
376 /* ----------------------------------------------------------------------- */
377 /* some GDI (was: Modular Technology) specific stuff */
384 [ 0x01 ] = { .id = TUNER_ABSENT,
386 [ 0x02 ] = { .id = TUNER_ABSENT,
388 [ 0x03 ] = { .id = TUNER_ABSENT,
390 [ 0x04 ] = { .id = TUNER_ABSENT,
392 [ 0x05 ] = { .id = TUNER_ABSENT,
395 [ 0x10 ] = { .id = TUNER_ABSENT, .fm = 1,
396 .name = "TEMIC_4049" },
397 [ 0x11 ] = { .id = TUNER_TEMIC_4136FY5,
398 .name = "TEMIC_4136" },
399 [ 0x12 ] = { .id = TUNER_ABSENT,
400 .name = "TEMIC_4146" },
402 [ 0x20 ] = { .id = TUNER_PHILIPS_FQ1216ME, .fm = 1,
403 .name = "PHILIPS_FQ1216_MK3" },
404 [ 0x21 ] = { .id = TUNER_ABSENT, .fm = 1,
405 .name = "PHILIPS_FQ1236_MK3" },
406 [ 0x22 ] = { .id = TUNER_ABSENT,
407 .name = "PHILIPS_FI1236_MK3" },
408 [ 0x23 ] = { .id = TUNER_ABSENT,
409 .name = "PHILIPS_FI1216_MK3" },
412 static void __devinit gdi_eeprom(struct cx8800_dev *dev, u8 *eeprom_data)
414 char *name = (eeprom_data[0x0d] < ARRAY_SIZE(gdi_tuner))
415 ? gdi_tuner[eeprom_data[0x0d]].name : NULL;
417 printk(KERN_INFO "%s: GDI: tuner=%s\n", dev->name,
418 name ? name : "unknown");
421 dev->tuner_type = gdi_tuner[eeprom_data[0x0d]].id;
422 dev->has_radio = gdi_tuner[eeprom_data[0x0d]].fm;
425 /* ----------------------------------------------------------------------- */
428 i2c_eeprom(struct i2c_client *c, unsigned char *eedata, int len)
435 if (1 != (err = i2c_master_send(c,&buf,1))) {
436 printk(KERN_INFO "cx88: Huh, no eeprom present (err=%d)?\n",
440 if (len != (err = i2c_master_recv(c,eedata,len))) {
441 printk(KERN_WARNING "cx88: i2c eeprom read error (err=%d)\n",
446 for (i = 0; i < len; i++) {
448 printk(KERN_INFO "cx88 ee: %02x:",i);
449 printk(" %02x",eedata[i]);
457 void __devinit cx88_card_setup(struct cx8800_dev *dev)
459 static u8 eeprom[128];
461 switch (dev->board) {
462 case CX88_BOARD_HAUPPAUGE:
463 if (0 == dev->i2c_rc)
464 i2c_eeprom(&dev->i2c_client,eeprom,sizeof(eeprom));
465 hauppauge_eeprom(dev,eeprom+8);
468 if (0 == dev->i2c_rc)
469 i2c_eeprom(&dev->i2c_client,eeprom,sizeof(eeprom));
470 gdi_eeprom(dev,eeprom);
472 case CX88_BOARD_WINFAST2000XP:
473 if (0 == dev->i2c_rc)
474 i2c_eeprom(&dev->i2c_client,eeprom,sizeof(eeprom));
475 leadtek_eeprom(dev,eeprom);
480 /* ------------------------------------------------------------------ */
482 EXPORT_SYMBOL(cx88_boards);
483 EXPORT_SYMBOL(cx88_bcount);
484 EXPORT_SYMBOL(cx88_subids);
485 EXPORT_SYMBOL(cx88_idcount);
486 EXPORT_SYMBOL(cx88_card_setup);