1 // SPDX-License-Identifier: GPL-2.0-only
2 /* DVB USB compliant linux driver for MSI Mega Sky 580 DVB-T USB2.0 receiver
4 * Copyright (C) 2006 Aapo Tahkola (aet@rasterburn.org)
6 * see Documentation/driver-api/media/drivers/dvb-usb.rst for more information
12 #include "mt352_priv.h"
18 #include <media/tuner.h>
19 #include "tuner-simple.h"
20 #include <linux/unaligned.h>
23 static int dvb_usb_m920x_debug
;
24 module_param_named(debug
,dvb_usb_m920x_debug
, int, 0644);
25 MODULE_PARM_DESC(debug
, "set debugging level (1=rc (or-able))." DVB_USB_DEBUG_STATUS
);
27 DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr
);
29 static int m920x_set_filter(struct dvb_usb_device
*d
, int type
, int idx
, int pid
);
31 static inline int m920x_read(struct usb_device
*udev
, u8 request
, u16 value
,
32 u16 index
, void *data
, int size
)
36 ret
= usb_control_msg(udev
, usb_rcvctrlpipe(udev
, 0),
37 request
, USB_TYPE_VENDOR
| USB_DIR_IN
,
38 value
, index
, data
, size
, 2000);
40 printk(KERN_INFO
"m920x_read = error: %d\n", ret
);
45 deb("m920x_read = no data\n");
52 static inline int m920x_write(struct usb_device
*udev
, u8 request
,
55 return usb_control_msg(udev
, usb_sndctrlpipe(udev
, 0), request
,
56 USB_TYPE_VENDOR
| USB_DIR_OUT
, value
, index
,
60 static inline int m920x_write_seq(struct usb_device
*udev
, u8 request
,
61 struct m920x_inits
*seq
)
65 ret
= m920x_write(udev
, request
, seq
->data
, seq
->address
);
70 } while (seq
->address
);
75 static int m920x_init(struct dvb_usb_device
*d
, struct m920x_inits
*rc_seq
)
77 int ret
, i
, epi
, flags
= 0;
78 int adap_enabled
[M9206_MAX_ADAPTERS
] = { 0 };
80 /* Remote controller init. */
81 if (d
->props
.rc
.legacy
.rc_query
|| d
->props
.rc
.core
.rc_query
) {
82 deb("Initialising remote control\n");
83 ret
= m920x_write_seq(d
->udev
, M9206_CORE
, rc_seq
);
85 deb("Initialising remote control failed\n");
89 deb("Initialising remote control success\n");
92 for (i
= 0; i
< d
->props
.num_adapters
; i
++)
93 flags
|= d
->adapter
[i
].props
.fe
[0].caps
;
95 /* Some devices(Dposh) might crash if we attempt touch at all. */
96 if (flags
& DVB_USB_ADAP_HAS_PID_FILTER
) {
97 for (i
= 0; i
< d
->props
.num_adapters
; i
++) {
98 epi
= d
->adapter
[i
].props
.fe
[0].stream
.endpoint
- 0x81;
100 if (epi
< 0 || epi
>= M9206_MAX_ADAPTERS
) {
101 printk(KERN_INFO
"m920x: Unexpected adapter endpoint!\n");
105 adap_enabled
[epi
] = 1;
108 for (i
= 0; i
< M9206_MAX_ADAPTERS
; i
++) {
112 if ((ret
= m920x_set_filter(d
, 0x81 + i
, 0, 0x0)) != 0)
115 if ((ret
= m920x_set_filter(d
, 0x81 + i
, 0, 0x02f5)) != 0)
123 static int m920x_init_ep(struct usb_interface
*intf
)
125 struct usb_device
*udev
= interface_to_usbdev(intf
);
126 struct usb_host_interface
*alt
;
128 if ((alt
= usb_altnum_to_altsetting(intf
, 1)) == NULL
) {
129 deb("No alt found!\n");
133 return usb_set_interface(udev
, alt
->desc
.bInterfaceNumber
,
134 alt
->desc
.bAlternateSetting
);
137 static inline void m920x_parse_rc_state(struct dvb_usb_device
*d
, u8 rc_state
,
140 struct m920x_state
*m
= d
->priv
;
144 *state
= REMOTE_NO_KEY_PRESSED
;
147 case 0x88: /* framing error or "invalid code" */
151 *state
= REMOTE_NO_KEY_PRESSED
;
157 case 0x83: /* pinnacle PCTV310e */
160 *state
= REMOTE_KEY_PRESSED
;
164 case 0x81: /* pinnacle PCTV310e */
165 /* prevent immediate auto-repeat */
166 if (++m
->rep_count
> 2)
167 *state
= REMOTE_KEY_REPEAT
;
169 *state
= REMOTE_NO_KEY_PRESSED
;
173 deb("Unexpected rc state %02x\n", rc_state
);
174 *state
= REMOTE_NO_KEY_PRESSED
;
179 static int m920x_rc_query(struct dvb_usb_device
*d
, u32
*event
, int *state
)
184 rc_state
= kmalloc(2, GFP_KERNEL
);
188 ret
= m920x_read(d
->udev
, M9206_CORE
, 0x0, M9206_RC_STATE
,
193 ret
= m920x_read(d
->udev
, M9206_CORE
, 0x0, M9206_RC_KEY
,
198 m920x_parse_rc_state(d
, rc_state
[0], state
);
200 for (i
= 0; i
< d
->props
.rc
.legacy
.rc_map_size
; i
++)
201 if (rc5_data(&d
->props
.rc
.legacy
.rc_map_table
[i
]) == rc_state
[1]) {
202 *event
= d
->props
.rc
.legacy
.rc_map_table
[i
].keycode
;
206 if (rc_state
[1] != 0)
207 deb("Unknown rc key %02x\n", rc_state
[1]);
209 *state
= REMOTE_NO_KEY_PRESSED
;
216 static int m920x_rc_core_query(struct dvb_usb_device
*d
)
222 rc_state
= kmalloc(2, GFP_KERNEL
);
226 if ((ret
= m920x_read(d
->udev
, M9206_CORE
, 0x0, M9206_RC_STATE
, &rc_state
[0], 1)) != 0)
229 if ((ret
= m920x_read(d
->udev
, M9206_CORE
, 0x0, M9206_RC_KEY
, &rc_state
[1], 1)) != 0)
232 deb("state=0x%02x keycode=0x%02x\n", rc_state
[0], rc_state
[1]);
234 m920x_parse_rc_state(d
, rc_state
[0], &state
);
236 if (state
== REMOTE_NO_KEY_PRESSED
)
238 else if (state
== REMOTE_KEY_REPEAT
)
239 rc_repeat(d
->rc_dev
);
241 rc_keydown(d
->rc_dev
, RC_PROTO_UNKNOWN
, rc_state
[1], 0);
249 static int m920x_i2c_xfer(struct i2c_adapter
*adap
, struct i2c_msg msg
[], int num
)
251 struct dvb_usb_device
*d
= i2c_get_adapdata(adap
);
255 if (mutex_lock_interruptible(&d
->i2c_mutex
) < 0)
258 for (i
= 0; i
< num
; i
++) {
259 if (msg
[i
].flags
& (I2C_M_NO_RD_ACK
| I2C_M_IGNORE_NAK
| I2C_M_TEN
) || msg
[i
].len
== 0) {
260 /* For a 0 byte message, I think sending the address
261 * to index 0x80|0x40 would be the correct thing to
262 * do. However, zero byte messages are only used for
263 * probing, and since we don't know how to get the
264 * slave's ack, we can't probe. */
268 /* Send START & address/RW bit */
269 if (!(msg
[i
].flags
& I2C_M_NOSTART
)) {
270 if ((ret
= m920x_write(d
->udev
, M9206_I2C
,
272 (msg
[i
].flags
& I2C_M_RD
? 0x01 : 0), 0x80)) != 0)
274 /* Should check for ack here, if we knew how. */
276 if (msg
[i
].flags
& I2C_M_RD
) {
277 char *read
= kmalloc(1, GFP_KERNEL
);
283 for (j
= 0; j
< msg
[i
].len
; j
++) {
284 /* Last byte of transaction?
285 * Send STOP, otherwise send ACK. */
286 int stop
= (i
+1 == num
&& j
+1 == msg
[i
].len
) ? 0x40 : 0x01;
288 if ((ret
= m920x_read(d
->udev
, M9206_I2C
, 0x0,
294 msg
[i
].buf
[j
] = read
[0];
299 for (j
= 0; j
< msg
[i
].len
; j
++) {
300 /* Last byte of transaction? Then send STOP. */
301 int stop
= (i
+1 == num
&& j
+1 == msg
[i
].len
) ? 0x40 : 0x00;
303 if ((ret
= m920x_write(d
->udev
, M9206_I2C
, msg
[i
].buf
[j
], stop
)) != 0)
305 /* Should check for ack here too. */
312 mutex_unlock(&d
->i2c_mutex
);
317 static u32
m920x_i2c_func(struct i2c_adapter
*adapter
)
322 static struct i2c_algorithm m920x_i2c_algo
= {
323 .master_xfer
= m920x_i2c_xfer
,
324 .functionality
= m920x_i2c_func
,
328 static int m920x_set_filter(struct dvb_usb_device
*d
, int type
, int idx
, int pid
)
337 if ((ret
= m920x_write(d
->udev
, M9206_FILTER
, pid
, (type
<< 8) | (idx
* 4) )) != 0)
340 if ((ret
= m920x_write(d
->udev
, M9206_FILTER
, 0, (type
<< 8) | (idx
* 4) )) != 0)
346 static int m920x_update_filters(struct dvb_usb_adapter
*adap
)
348 struct m920x_state
*m
= adap
->dev
->priv
;
349 int enabled
= m
->filtering_enabled
[adap
->id
];
350 int i
, ret
= 0, filter
= 0;
351 int ep
= adap
->props
.fe
[0].stream
.endpoint
;
353 for (i
= 0; i
< M9206_MAX_FILTERS
; i
++)
354 if (m
->filters
[adap
->id
][i
] == 8192)
357 /* Disable all filters */
358 if ((ret
= m920x_set_filter(adap
->dev
, ep
, 1, enabled
)) != 0)
361 for (i
= 0; i
< M9206_MAX_FILTERS
; i
++)
362 if ((ret
= m920x_set_filter(adap
->dev
, ep
, i
+ 2, 0)) != 0)
367 for (i
= 0; i
< M9206_MAX_FILTERS
; i
++) {
368 if (m
->filters
[adap
->id
][i
] == 0)
371 if ((ret
= m920x_set_filter(adap
->dev
, ep
, filter
+ 2, m
->filters
[adap
->id
][i
])) != 0)
381 static int m920x_pid_filter_ctrl(struct dvb_usb_adapter
*adap
, int onoff
)
383 struct m920x_state
*m
= adap
->dev
->priv
;
385 m
->filtering_enabled
[adap
->id
] = onoff
? 1 : 0;
387 return m920x_update_filters(adap
);
390 static int m920x_pid_filter(struct dvb_usb_adapter
*adap
, int index
, u16 pid
, int onoff
)
392 struct m920x_state
*m
= adap
->dev
->priv
;
394 m
->filters
[adap
->id
][index
] = onoff
? pid
: 0;
396 return m920x_update_filters(adap
);
399 static int m920x_firmware_download(struct usb_device
*udev
, const struct firmware
*fw
)
401 u16 value
, index
, size
;
403 int i
, pass
, ret
= 0;
405 buff
= kmalloc(65536, GFP_KERNEL
);
409 read
= kmalloc(4, GFP_KERNEL
);
415 if ((ret
= m920x_read(udev
, M9206_FILTER
, 0x0, 0x8000, read
, 4)) != 0)
417 deb("%*ph\n", 4, read
);
419 if ((ret
= m920x_read(udev
, M9206_FW
, 0x0, 0x0, read
, 1)) != 0)
421 deb("%x\n", read
[0]);
423 for (pass
= 0; pass
< 2; pass
++) {
424 for (i
= 0; i
+ (sizeof(u16
) * 3) < fw
->size
;) {
425 value
= get_unaligned_le16(fw
->data
+ i
);
428 index
= get_unaligned_le16(fw
->data
+ i
);
431 size
= get_unaligned_le16(fw
->data
+ i
);
435 /* Will stall if using fw->data ... */
436 memcpy(buff
, fw
->data
+ i
, size
);
438 ret
= usb_control_msg(udev
, usb_sndctrlpipe(udev
,0),
440 USB_TYPE_VENDOR
| USB_DIR_OUT
,
441 value
, index
, buff
, size
, 20);
443 deb("error while uploading fw!\n");
452 deb("bad firmware file!\n");
460 /* m920x will disconnect itself from the bus after this. */
461 (void) m920x_write(udev
, M9206_CORE
, 0x01, M9206_FW_GO
);
462 deb("firmware uploaded!\n");
471 /* Callbacks for DVB USB */
472 static int m920x_identify_state(struct usb_device
*udev
,
473 const struct dvb_usb_device_properties
*props
,
474 const struct dvb_usb_device_description
**desc
,
477 struct usb_host_interface
*alt
;
479 alt
= usb_altnum_to_altsetting(usb_ifnum_to_if(udev
, 0), 1);
480 *cold
= (alt
== NULL
) ? 1 : 0;
485 /* demod configurations */
486 static int m920x_mt352_demod_init(struct dvb_frontend
*fe
)
489 static const u8 config
[] = { CONFIG
, 0x3d };
490 static const u8 clock
[] = { CLOCK_CTL
, 0x30 };
491 static const u8 reset
[] = { RESET
, 0x80 };
492 static const u8 adc_ctl
[] = { ADC_CTL_1
, 0x40 };
493 static const u8 agc
[] = { AGC_TARGET
, 0x1c, 0x20 };
494 static const u8 sec_agc
[] = { 0x69, 0x00, 0xff, 0xff, 0x40, 0xff, 0x00, 0x40, 0x40 };
495 static const u8 unk1
[] = { 0x93, 0x1a };
496 static const u8 unk2
[] = { 0xb5, 0x7a };
498 deb("Demod init!\n");
500 if ((ret
= mt352_write(fe
, config
, ARRAY_SIZE(config
))) != 0)
502 if ((ret
= mt352_write(fe
, clock
, ARRAY_SIZE(clock
))) != 0)
504 if ((ret
= mt352_write(fe
, reset
, ARRAY_SIZE(reset
))) != 0)
506 if ((ret
= mt352_write(fe
, adc_ctl
, ARRAY_SIZE(adc_ctl
))) != 0)
508 if ((ret
= mt352_write(fe
, agc
, ARRAY_SIZE(agc
))) != 0)
510 if ((ret
= mt352_write(fe
, sec_agc
, ARRAY_SIZE(sec_agc
))) != 0)
512 if ((ret
= mt352_write(fe
, unk1
, ARRAY_SIZE(unk1
))) != 0)
514 if ((ret
= mt352_write(fe
, unk2
, ARRAY_SIZE(unk2
))) != 0)
520 static struct mt352_config m920x_mt352_config
= {
521 .demod_address
= 0x0f,
523 .demod_init
= m920x_mt352_demod_init
,
526 static struct tda1004x_config m920x_tda10046_08_config
= {
527 .demod_address
= 0x08,
530 .ts_mode
= TDA10046_TS_SERIAL
,
531 .xtal_freq
= TDA10046_XTAL_16M
,
532 .if_freq
= TDA10046_FREQ_045
,
533 .agc_config
= TDA10046_AGC_TDA827X
,
534 .gpio_config
= TDA10046_GPTRI
,
535 .request_firmware
= NULL
,
538 static struct tda1004x_config m920x_tda10046_0b_config
= {
539 .demod_address
= 0x0b,
542 .ts_mode
= TDA10046_TS_SERIAL
,
543 .xtal_freq
= TDA10046_XTAL_16M
,
544 .if_freq
= TDA10046_FREQ_045
,
545 .agc_config
= TDA10046_AGC_TDA827X
,
546 .gpio_config
= TDA10046_GPTRI
,
547 .request_firmware
= NULL
, /* uses firmware EEPROM */
550 /* tuner configurations */
551 static struct qt1010_config m920x_qt1010_config
= {
555 static struct mt2060_config m920x_mt2060_config
= {
556 .i2c_address
= 0x60, /* 0xc0 */
561 /* Callbacks for DVB USB */
562 static int m920x_mt352_frontend_attach(struct dvb_usb_adapter
*adap
)
564 deb("%s\n",__func__
);
566 adap
->fe_adap
[0].fe
= dvb_attach(mt352_attach
,
568 &adap
->dev
->i2c_adap
);
569 if ((adap
->fe_adap
[0].fe
) == NULL
)
575 static int m920x_mt352_frontend_attach_vp7049(struct dvb_usb_adapter
*adap
)
577 struct m920x_inits vp7049_fe_init_seq
[] = {
578 /* XXX without these commands the frontend cannot be detected,
579 * they must be sent BEFORE the frontend is attached */
591 { } /* terminating entry */
595 deb("%s\n", __func__
);
597 ret
= m920x_write_seq(adap
->dev
->udev
, M9206_CORE
, vp7049_fe_init_seq
);
599 deb("Initialization of vp7049 frontend failed.");
603 return m920x_mt352_frontend_attach(adap
);
606 static int m920x_tda10046_08_frontend_attach(struct dvb_usb_adapter
*adap
)
608 deb("%s\n",__func__
);
610 adap
->fe_adap
[0].fe
= dvb_attach(tda10046_attach
,
611 &m920x_tda10046_08_config
,
612 &adap
->dev
->i2c_adap
);
613 if ((adap
->fe_adap
[0].fe
) == NULL
)
619 static int m920x_tda10046_0b_frontend_attach(struct dvb_usb_adapter
*adap
)
621 deb("%s\n",__func__
);
623 adap
->fe_adap
[0].fe
= dvb_attach(tda10046_attach
,
624 &m920x_tda10046_0b_config
,
625 &adap
->dev
->i2c_adap
);
626 if ((adap
->fe_adap
[0].fe
) == NULL
)
632 static int m920x_qt1010_tuner_attach(struct dvb_usb_adapter
*adap
)
634 deb("%s\n",__func__
);
636 if (dvb_attach(qt1010_attach
, adap
->fe_adap
[0].fe
, &adap
->dev
->i2c_adap
, &m920x_qt1010_config
) == NULL
)
642 static int m920x_tda8275_60_tuner_attach(struct dvb_usb_adapter
*adap
)
644 deb("%s\n",__func__
);
646 if (dvb_attach(tda827x_attach
, adap
->fe_adap
[0].fe
, 0x60, &adap
->dev
->i2c_adap
, NULL
) == NULL
)
652 static int m920x_tda8275_61_tuner_attach(struct dvb_usb_adapter
*adap
)
654 deb("%s\n",__func__
);
656 if (dvb_attach(tda827x_attach
, adap
->fe_adap
[0].fe
, 0x61, &adap
->dev
->i2c_adap
, NULL
) == NULL
)
662 static int m920x_fmd1216me_tuner_attach(struct dvb_usb_adapter
*adap
)
664 dvb_attach(simple_tuner_attach
, adap
->fe_adap
[0].fe
,
665 &adap
->dev
->i2c_adap
, 0x61,
666 TUNER_PHILIPS_FMD1216ME_MK3
);
670 static int m920x_mt2060_tuner_attach(struct dvb_usb_adapter
*adap
)
672 deb("%s\n", __func__
);
674 if (dvb_attach(mt2060_attach
, adap
->fe_adap
[0].fe
, &adap
->dev
->i2c_adap
,
675 &m920x_mt2060_config
, 1220) == NULL
)
682 /* device-specific initialization */
683 static struct m920x_inits megasky_rc_init
[] = {
684 { M9206_RC_INIT2
, 0xa8 },
685 { M9206_RC_INIT1
, 0x51 },
686 { } /* terminating entry */
689 static struct m920x_inits tvwalkertwin_rc_init
[] = {
690 { M9206_RC_INIT2
, 0x00 },
691 { M9206_RC_INIT1
, 0xef },
695 { } /* terminating entry */
698 static struct m920x_inits pinnacle310e_init
[] = {
699 /* without these the tuner doesn't work */
705 { M9206_RC_INIT1
, 0x00 },
706 { M9206_RC_INIT2
, 0xff },
707 { } /* terminating entry */
710 static struct m920x_inits vp7049_rc_init
[] = {
714 { M9206_RC_INIT2
, 0x00 },
715 { M9206_RC_INIT1
, 0xff },
716 { } /* terminating entry */
720 static struct rc_map_table rc_map_megasky_table
[] = {
721 { 0x0012, KEY_POWER
},
722 { 0x001e, KEY_CYCLEWINDOWS
}, /* min/max */
723 { 0x0002, KEY_CHANNELUP
},
724 { 0x0005, KEY_CHANNELDOWN
},
725 { 0x0003, KEY_VOLUMEUP
},
726 { 0x0006, KEY_VOLUMEDOWN
},
727 { 0x0004, KEY_MUTE
},
728 { 0x0007, KEY_OK
}, /* TS */
729 { 0x0008, KEY_STOP
},
730 { 0x0009, KEY_MENU
}, /* swap */
731 { 0x000a, KEY_REWIND
},
732 { 0x001b, KEY_PAUSE
},
733 { 0x001f, KEY_FASTFORWARD
},
734 { 0x000c, KEY_RECORD
},
735 { 0x000d, KEY_CAMERA
}, /* screenshot */
736 { 0x000e, KEY_COFFEE
}, /* "MTS" */
739 static struct rc_map_table rc_map_tvwalkertwin_table
[] = {
740 { 0x0001, KEY_ZOOM
}, /* Full Screen */
741 { 0x0002, KEY_CAMERA
}, /* snapshot */
742 { 0x0003, KEY_MUTE
},
743 { 0x0004, KEY_REWIND
},
744 { 0x0005, KEY_PLAYPAUSE
}, /* Play/Pause */
745 { 0x0006, KEY_FASTFORWARD
},
746 { 0x0007, KEY_RECORD
},
747 { 0x0008, KEY_STOP
},
748 { 0x0009, KEY_TIME
}, /* Timeshift */
749 { 0x000c, KEY_COFFEE
}, /* Recall */
750 { 0x000e, KEY_CHANNELUP
},
751 { 0x0012, KEY_POWER
},
752 { 0x0015, KEY_MENU
}, /* source */
753 { 0x0018, KEY_CYCLEWINDOWS
}, /* TWIN PIP */
754 { 0x001a, KEY_CHANNELDOWN
},
755 { 0x001b, KEY_VOLUMEDOWN
},
756 { 0x001e, KEY_VOLUMEUP
},
759 static struct rc_map_table rc_map_pinnacle310e_table
[] = {
761 { 0x17, KEY_FAVORITES
},
763 { 0x48, KEY_PROGRAM
}, /* preview */
765 { 0x04, KEY_LIST
}, /* record list */
776 { 0x0c, KEY_CANCEL
},
784 { 0x4f, KEY_ENTER
}, /* could also be KEY_OK */
785 { 0x1e, KEY_VOLUMEUP
},
786 { 0x0a, KEY_VOLUMEDOWN
},
787 { 0x05, KEY_CHANNELUP
},
788 { 0x02, KEY_CHANNELDOWN
},
789 { 0x11, KEY_RECORD
},
793 { 0x40, KEY_REWIND
},
794 { 0x12, KEY_FASTFORWARD
},
795 { 0x41, KEY_PREVIOUSSONG
}, /* Replay */
796 { 0x42, KEY_NEXTSONG
}, /* Skip */
797 { 0x54, KEY_CAMERA
}, /* Capture */
798 /* { 0x50, KEY_SAP }, */ /* Sap */
799 { 0x47, KEY_CYCLEWINDOWS
}, /* Pip */
800 { 0x4d, KEY_SCREEN
}, /* FullScreen */
801 { 0x08, KEY_SUBTITLE
},
803 /* { 0x49, KEY_LR }, */ /* L/R */
804 { 0x07, KEY_SLEEP
}, /* Hibernate */
805 { 0x08, KEY_VIDEO
}, /* A/V */
806 { 0x0e, KEY_MENU
}, /* Recall */
807 { 0x45, KEY_ZOOMIN
},
808 { 0x46, KEY_ZOOMOUT
},
809 { 0x18, KEY_RED
}, /* Red */
810 { 0x53, KEY_GREEN
}, /* Green */
811 { 0x5e, KEY_YELLOW
}, /* Yellow */
812 { 0x5f, KEY_BLUE
}, /* Blue */
815 /* DVB USB Driver stuff */
816 static struct dvb_usb_device_properties megasky_properties
;
817 static struct dvb_usb_device_properties digivox_mini_ii_properties
;
818 static struct dvb_usb_device_properties tvwalkertwin_properties
;
819 static struct dvb_usb_device_properties dposh_properties
;
820 static struct dvb_usb_device_properties pinnacle_pctv310e_properties
;
821 static struct dvb_usb_device_properties vp7049_properties
;
823 static int m920x_probe(struct usb_interface
*intf
,
824 const struct usb_device_id
*id
)
826 struct dvb_usb_device
*d
= NULL
;
828 struct m920x_inits
*rc_init_seq
= NULL
;
829 int bInterfaceNumber
= intf
->cur_altsetting
->desc
.bInterfaceNumber
;
831 deb("Probing for m920x device at interface %d\n", bInterfaceNumber
);
833 if (bInterfaceNumber
== 0) {
834 /* Single-tuner device, or first interface on
838 ret
= dvb_usb_device_init(intf
, &megasky_properties
,
839 THIS_MODULE
, &d
, adapter_nr
);
841 rc_init_seq
= megasky_rc_init
;
845 ret
= dvb_usb_device_init(intf
, &digivox_mini_ii_properties
,
846 THIS_MODULE
, &d
, adapter_nr
);
848 /* No remote control, so no rc_init_seq */
852 /* This configures both tuners on the TV Walker Twin */
853 ret
= dvb_usb_device_init(intf
, &tvwalkertwin_properties
,
854 THIS_MODULE
, &d
, adapter_nr
);
856 rc_init_seq
= tvwalkertwin_rc_init
;
860 ret
= dvb_usb_device_init(intf
, &dposh_properties
,
861 THIS_MODULE
, &d
, adapter_nr
);
863 /* Remote controller not supported yet. */
867 ret
= dvb_usb_device_init(intf
, &pinnacle_pctv310e_properties
,
868 THIS_MODULE
, &d
, adapter_nr
);
870 rc_init_seq
= pinnacle310e_init
;
874 ret
= dvb_usb_device_init(intf
, &vp7049_properties
,
875 THIS_MODULE
, &d
, adapter_nr
);
877 rc_init_seq
= vp7049_rc_init
;
883 /* Another interface on a multi-tuner device */
885 /* The LifeView TV Walker Twin gets here, but struct
886 * tvwalkertwin_properties already configured both
887 * tuners, so there is nothing for us to do here
892 if ((ret
= m920x_init_ep(intf
)) < 0)
895 if (d
&& (ret
= m920x_init(d
, rc_init_seq
)) != 0)
903 ANUBIS_MSI_DIGI_VOX_MINI_II
,
904 ANUBIS_LIFEVIEW_TV_WALKER_TWIN_COLD
,
905 ANUBIS_LIFEVIEW_TV_WALKER_TWIN_WARM
,
908 VISIONPLUS_PINNACLE_PCTV310E
,
909 AZUREWAVE_TWINHAN_VP7049
,
912 static struct usb_device_id m920x_table
[] = {
913 DVB_USB_DEV(MSI
, MSI_MEGASKY580
),
914 DVB_USB_DEV(ANUBIS_ELECTRONIC
, ANUBIS_MSI_DIGI_VOX_MINI_II
),
915 DVB_USB_DEV(ANUBIS_ELECTRONIC
, ANUBIS_LIFEVIEW_TV_WALKER_TWIN_COLD
),
916 DVB_USB_DEV(ANUBIS_ELECTRONIC
, ANUBIS_LIFEVIEW_TV_WALKER_TWIN_WARM
),
917 DVB_USB_DEV(DPOSH
, DPOSH_M9206_COLD
),
918 DVB_USB_DEV(DPOSH
, DPOSH_M9206_WARM
),
919 DVB_USB_DEV(VISIONPLUS
, VISIONPLUS_PINNACLE_PCTV310E
),
920 DVB_USB_DEV(AZUREWAVE
, AZUREWAVE_TWINHAN_VP7049
),
924 MODULE_DEVICE_TABLE (usb
, m920x_table
);
926 static struct dvb_usb_device_properties megasky_properties
= {
927 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
929 .usb_ctrl
= DEVICE_SPECIFIC
,
930 .firmware
= "dvb-usb-megasky-02.fw",
931 .download_firmware
= m920x_firmware_download
,
935 .rc_map_table
= rc_map_megasky_table
,
936 .rc_map_size
= ARRAY_SIZE(rc_map_megasky_table
),
937 .rc_query
= m920x_rc_query
,
940 .size_of_priv
= sizeof(struct m920x_state
),
942 .identify_state
= m920x_identify_state
,
948 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
949 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
951 .pid_filter_count
= 8,
952 .pid_filter
= m920x_pid_filter
,
953 .pid_filter_ctrl
= m920x_pid_filter_ctrl
,
955 .frontend_attach
= m920x_mt352_frontend_attach
,
956 .tuner_attach
= m920x_qt1010_tuner_attach
,
970 .i2c_algo
= &m920x_i2c_algo
,
972 .num_device_descs
= 1,
974 { "MSI Mega Sky 580 DVB-T USB2.0",
975 { &m920x_table
[MSI_MEGASKY580
], NULL
},
981 static struct dvb_usb_device_properties digivox_mini_ii_properties
= {
982 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
984 .usb_ctrl
= DEVICE_SPECIFIC
,
985 .firmware
= "dvb-usb-digivox-02.fw",
986 .download_firmware
= m920x_firmware_download
,
988 .size_of_priv
= sizeof(struct m920x_state
),
990 .identify_state
= m920x_identify_state
,
996 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
997 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
999 .pid_filter_count
= 8,
1000 .pid_filter
= m920x_pid_filter
,
1001 .pid_filter_ctrl
= m920x_pid_filter_ctrl
,
1003 .frontend_attach
= m920x_tda10046_08_frontend_attach
,
1004 .tuner_attach
= m920x_tda8275_60_tuner_attach
,
1012 .buffersize
= 0x4000,
1018 .i2c_algo
= &m920x_i2c_algo
,
1020 .num_device_descs
= 1,
1022 { "MSI DIGI VOX mini II DVB-T USB2.0",
1023 { &m920x_table
[ANUBIS_MSI_DIGI_VOX_MINI_II
], NULL
},
1029 /* LifeView TV Walker Twin support by Nick Andrew <nick@nick-andrew.net>
1031 * LifeView TV Walker Twin has 1 x M9206, 2 x TDA10046, 2 x TDA8275A
1032 * TDA10046 #0 is located at i2c address 0x08
1033 * TDA10046 #1 is located at i2c address 0x0b
1034 * TDA8275A #0 is located at i2c address 0x60
1035 * TDA8275A #1 is located at i2c address 0x61
1037 static struct dvb_usb_device_properties tvwalkertwin_properties
= {
1038 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
1040 .usb_ctrl
= DEVICE_SPECIFIC
,
1041 .firmware
= "dvb-usb-tvwalkert.fw",
1042 .download_firmware
= m920x_firmware_download
,
1046 .rc_map_table
= rc_map_tvwalkertwin_table
,
1047 .rc_map_size
= ARRAY_SIZE(rc_map_tvwalkertwin_table
),
1048 .rc_query
= m920x_rc_query
,
1051 .size_of_priv
= sizeof(struct m920x_state
),
1053 .identify_state
= m920x_identify_state
,
1059 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
1060 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
1062 .pid_filter_count
= 8,
1063 .pid_filter
= m920x_pid_filter
,
1064 .pid_filter_ctrl
= m920x_pid_filter_ctrl
,
1066 .frontend_attach
= m920x_tda10046_08_frontend_attach
,
1067 .tuner_attach
= m920x_tda8275_60_tuner_attach
,
1083 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
1084 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
1086 .pid_filter_count
= 8,
1087 .pid_filter
= m920x_pid_filter
,
1088 .pid_filter_ctrl
= m920x_pid_filter_ctrl
,
1090 .frontend_attach
= m920x_tda10046_0b_frontend_attach
,
1091 .tuner_attach
= m920x_tda8275_61_tuner_attach
,
1105 .i2c_algo
= &m920x_i2c_algo
,
1107 .num_device_descs
= 1,
1109 { .name
= "LifeView TV Walker Twin DVB-T USB2.0",
1110 .cold_ids
= { &m920x_table
[ANUBIS_LIFEVIEW_TV_WALKER_TWIN_COLD
], NULL
},
1111 .warm_ids
= { &m920x_table
[ANUBIS_LIFEVIEW_TV_WALKER_TWIN_WARM
], NULL
},
1116 static struct dvb_usb_device_properties dposh_properties
= {
1117 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
1119 .usb_ctrl
= DEVICE_SPECIFIC
,
1120 .firmware
= "dvb-usb-dposh-01.fw",
1121 .download_firmware
= m920x_firmware_download
,
1123 .size_of_priv
= sizeof(struct m920x_state
),
1125 .identify_state
= m920x_identify_state
,
1130 /* Hardware pid filters don't work with this device/firmware */
1132 .frontend_attach
= m920x_mt352_frontend_attach
,
1133 .tuner_attach
= m920x_qt1010_tuner_attach
,
1147 .i2c_algo
= &m920x_i2c_algo
,
1149 .num_device_descs
= 1,
1151 { .name
= "Dposh DVB-T USB2.0",
1152 .cold_ids
= { &m920x_table
[DPOSH_M9206_COLD
], NULL
},
1153 .warm_ids
= { &m920x_table
[DPOSH_M9206_WARM
], NULL
},
1158 static struct dvb_usb_device_properties pinnacle_pctv310e_properties
= {
1159 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
1161 .usb_ctrl
= DEVICE_SPECIFIC
,
1162 .download_firmware
= NULL
,
1166 .rc_map_table
= rc_map_pinnacle310e_table
,
1167 .rc_map_size
= ARRAY_SIZE(rc_map_pinnacle310e_table
),
1168 .rc_query
= m920x_rc_query
,
1171 .size_of_priv
= sizeof(struct m920x_state
),
1173 .identify_state
= m920x_identify_state
,
1179 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
1180 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
1182 .pid_filter_count
= 8,
1183 .pid_filter
= m920x_pid_filter
,
1184 .pid_filter_ctrl
= m920x_pid_filter_ctrl
,
1186 .frontend_attach
= m920x_mt352_frontend_attach
,
1187 .tuner_attach
= m920x_fmd1216me_tuner_attach
,
1195 .framesperurb
= 128,
1203 .i2c_algo
= &m920x_i2c_algo
,
1205 .num_device_descs
= 1,
1207 { "Pinnacle PCTV 310e",
1208 { &m920x_table
[VISIONPLUS_PINNACLE_PCTV310E
], NULL
},
1214 static struct dvb_usb_device_properties vp7049_properties
= {
1215 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
1217 .usb_ctrl
= DEVICE_SPECIFIC
,
1218 .firmware
= "dvb-usb-vp7049-0.95.fw",
1219 .download_firmware
= m920x_firmware_download
,
1223 .rc_codes
= RC_MAP_TWINHAN_VP1027_DVBS
,
1224 .rc_query
= m920x_rc_core_query
,
1225 .allowed_protos
= RC_PROTO_BIT_UNKNOWN
,
1228 .size_of_priv
= sizeof(struct m920x_state
),
1230 .identify_state
= m920x_identify_state
,
1236 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
1237 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
1239 .pid_filter_count
= 8,
1240 .pid_filter
= m920x_pid_filter
,
1241 .pid_filter_ctrl
= m920x_pid_filter_ctrl
,
1243 .frontend_attach
= m920x_mt352_frontend_attach_vp7049
,
1244 .tuner_attach
= m920x_mt2060_tuner_attach
,
1258 .i2c_algo
= &m920x_i2c_algo
,
1260 .num_device_descs
= 1,
1262 { "DTV-DVB UDTT7049",
1263 { &m920x_table
[AZUREWAVE_TWINHAN_VP7049
], NULL
},
1269 static struct usb_driver m920x_driver
= {
1270 .name
= "dvb_usb_m920x",
1271 .probe
= m920x_probe
,
1272 .disconnect
= dvb_usb_device_exit
,
1273 .id_table
= m920x_table
,
1276 module_usb_driver(m920x_driver
);
1278 MODULE_AUTHOR("Aapo Tahkola <aet@rasterburn.org>");
1279 MODULE_DESCRIPTION("DVB Driver for ULI M920x");
1280 MODULE_VERSION("0.1");
1281 MODULE_LICENSE("GPL");