2005-07-07 17:58:10 -07:00
/* DVB USB compliant linux driver for Conexant USB reference design.
*
* The Conexant reference design I saw on their website was only for analogue
* capturing ( using the cx25842 ) . The box I took to write this driver ( reverse
* engineered ) is the one labeled Medion MD95700 . In addition to the cx25842
* for analogue capturing it also has a cx22702 DVB - T demodulator on the main
* board . Besides it has a atiremote ( X10 ) and a USB2 .0 hub onboard .
*
* Maybe it is a little bit premature to call this driver cxusb , but I assume
* the USB protocol is identical or at least inherited from the reference
* design , so it can be reused for the " analogue-only " device ( if it will
* appear at all ) .
*
2006-01-09 18:21:38 -02:00
* TODO : Use the cx25840 - driver for the analogue part
2005-07-07 17:58:10 -07:00
*
* Copyright ( C ) 2005 Patrick Boettcher ( patrick . boettcher @ desy . de )
2006-01-09 18:21:31 -02:00
* Copyright ( C ) 2005 Michael Krufky ( mkrufky @ m1k . net )
2006-01-09 18:21:29 -02:00
* Copyright ( C ) 2006 Chris Pascoe ( c . pascoe @ itee . uq . edu . au )
2005-07-07 17:58:10 -07:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the Free
* Software Foundation , version 2.
*
* see Documentation / dvb / README . dvb - usb for more information
*/
# include "cxusb.h"
# include "cx22702.h"
2006-01-09 15:25:47 -02:00
# include "lgdt330x.h"
2006-04-24 23:21:46 -03:00
# include "lg_h06xf.h"
2006-01-09 18:21:28 -02:00
# include "mt352.h"
# include "mt352_priv.h"
2006-06-11 04:24:31 -03:00
# include "zl10353.h"
2005-07-07 17:58:10 -07:00
/* debug */
int dvb_usb_cxusb_debug ;
module_param_named ( debug , dvb_usb_cxusb_debug , int , 0644 ) ;
MODULE_PARM_DESC ( debug , " set debugging level (1=rc (or-able)). " DVB_USB_DEBUG_STATUS ) ;
static int cxusb_ctrl_msg ( struct dvb_usb_device * d ,
u8 cmd , u8 * wbuf , int wlen , u8 * rbuf , int rlen )
{
int wo = ( rbuf = = NULL | | rlen = = 0 ) ; /* write-only */
u8 sndbuf [ 1 + wlen ] ;
memset ( sndbuf , 0 , 1 + wlen ) ;
sndbuf [ 0 ] = cmd ;
memcpy ( & sndbuf [ 1 ] , wbuf , wlen ) ;
if ( wo )
dvb_usb_generic_write ( d , sndbuf , 1 + wlen ) ;
else
dvb_usb_generic_rw ( d , sndbuf , 1 + wlen , rbuf , rlen , 0 ) ;
return 0 ;
}
2005-09-09 13:02:51 -07:00
/* GPIO */
static void cxusb_gpio_tuner ( struct dvb_usb_device * d , int onoff )
2005-07-07 17:58:10 -07:00
{
struct cxusb_state * st = d - > priv ;
u8 o [ 2 ] , i ;
2005-09-09 13:02:51 -07:00
if ( st - > gpio_write_state [ GPIO_TUNER ] = = onoff )
2005-07-07 17:58:10 -07:00
return ;
2005-09-09 13:02:51 -07:00
o [ 0 ] = GPIO_TUNER ;
o [ 1 ] = onoff ;
cxusb_ctrl_msg ( d , CMD_GPIO_WRITE , o , 2 , & i , 1 ) ;
2005-07-07 17:58:10 -07:00
if ( i ! = 0x01 )
2005-09-09 13:02:51 -07:00
deb_info ( " gpio_write failed. \n " ) ;
2005-07-07 17:58:10 -07:00
2005-09-09 13:02:51 -07:00
st - > gpio_write_state [ GPIO_TUNER ] = onoff ;
2005-07-07 17:58:10 -07:00
}
2005-09-09 13:02:51 -07:00
/* I2C */
2005-07-07 17:58:10 -07:00
static int cxusb_i2c_xfer ( struct i2c_adapter * adap , struct i2c_msg msg [ ] , int num )
{
struct dvb_usb_device * d = i2c_get_adapdata ( adap ) ;
int i ;
2006-02-07 06:49:14 -02:00
if ( mutex_lock_interruptible ( & d - > i2c_mutex ) < 0 )
2005-07-07 17:58:10 -07:00
return - EAGAIN ;
if ( num > 2 )
2006-03-23 00:55:23 -03:00
warn ( " more than two i2c messages at a time is not handled yet. TODO. " ) ;
2005-07-07 17:58:10 -07:00
for ( i = 0 ; i < num ; i + + ) {
2006-03-23 00:01:34 -03:00
if ( d - > udev - > descriptor . idVendor = = USB_VID_MEDION )
switch ( msg [ i ] . addr ) {
case 0x63 :
cxusb_gpio_tuner ( d , 0 ) ;
break ;
default :
cxusb_gpio_tuner ( d , 1 ) ;
break ;
}
2005-07-07 17:58:10 -07:00
/* read request */
if ( i + 1 < num & & ( msg [ i + 1 ] . flags & I2C_M_RD ) ) {
u8 obuf [ 3 + msg [ i ] . len ] , ibuf [ 1 + msg [ i + 1 ] . len ] ;
obuf [ 0 ] = msg [ i ] . len ;
obuf [ 1 ] = msg [ i + 1 ] . len ;
obuf [ 2 ] = msg [ i ] . addr ;
memcpy ( & obuf [ 3 ] , msg [ i ] . buf , msg [ i ] . len ) ;
if ( cxusb_ctrl_msg ( d , CMD_I2C_READ ,
obuf , 3 + msg [ i ] . len ,
ibuf , 1 + msg [ i + 1 ] . len ) < 0 )
break ;
if ( ibuf [ 0 ] ! = 0x08 )
2006-03-23 01:11:18 -03:00
deb_i2c ( " i2c read may have failed \n " ) ;
2005-07-07 17:58:10 -07:00
memcpy ( msg [ i + 1 ] . buf , & ibuf [ 1 ] , msg [ i + 1 ] . len ) ;
i + + ;
} else { /* write */
u8 obuf [ 2 + msg [ i ] . len ] , ibuf ;
obuf [ 0 ] = msg [ i ] . addr ;
obuf [ 1 ] = msg [ i ] . len ;
memcpy ( & obuf [ 2 ] , msg [ i ] . buf , msg [ i ] . len ) ;
if ( cxusb_ctrl_msg ( d , CMD_I2C_WRITE , obuf , 2 + msg [ i ] . len , & ibuf , 1 ) < 0 )
break ;
if ( ibuf ! = 0x08 )
2006-03-23 01:11:18 -03:00
deb_i2c ( " i2c write may have failed \n " ) ;
2005-07-07 17:58:10 -07:00
}
}
2006-02-07 06:49:14 -02:00
mutex_unlock ( & d - > i2c_mutex ) ;
2005-07-07 17:58:10 -07:00
return i ;
}
static u32 cxusb_i2c_func ( struct i2c_adapter * adapter )
{
return I2C_FUNC_I2C ;
}
static struct i2c_algorithm cxusb_i2c_algo = {
. master_xfer = cxusb_i2c_xfer ,
. functionality = cxusb_i2c_func ,
} ;
static int cxusb_power_ctrl ( struct dvb_usb_device * d , int onoff )
{
2005-09-09 13:02:51 -07:00
u8 b = 0 ;
if ( onoff )
return cxusb_ctrl_msg ( d , CMD_POWER_ON , & b , 1 , NULL , 0 ) ;
else
return cxusb_ctrl_msg ( d , CMD_POWER_OFF , & b , 1 , NULL , 0 ) ;
2005-07-07 17:58:10 -07:00
}
2006-04-19 20:40:01 -03:00
static int cxusb_bluebird_power_ctrl ( struct dvb_usb_device * d , int onoff )
{
u8 b = 0 ;
if ( onoff )
return cxusb_ctrl_msg ( d , CMD_POWER_ON , & b , 1 , NULL , 0 ) ;
else
return 0 ;
}
2005-07-07 17:58:10 -07:00
static int cxusb_streaming_ctrl ( struct dvb_usb_device * d , int onoff )
{
2005-07-07 17:58:15 -07:00
u8 buf [ 2 ] = { 0x03 , 0x00 } ;
if ( onoff )
2005-09-09 13:02:51 -07:00
cxusb_ctrl_msg ( d , CMD_STREAMING_ON , buf , 2 , NULL , 0 ) ;
2005-07-07 17:58:15 -07:00
else
2005-09-09 13:02:51 -07:00
cxusb_ctrl_msg ( d , CMD_STREAMING_OFF , NULL , 0 , NULL , 0 ) ;
2005-07-07 17:58:15 -07:00
2005-07-07 17:58:10 -07:00
return 0 ;
}
2006-01-09 18:21:29 -02:00
static int cxusb_rc_query ( struct dvb_usb_device * d , u32 * event , int * state )
{
struct dvb_usb_rc_key * keymap = d - > props . rc_key_map ;
2006-01-09 18:21:31 -02:00
u8 ircode [ 4 ] ;
2006-01-09 18:21:29 -02:00
int i ;
2006-01-09 18:21:31 -02:00
cxusb_ctrl_msg ( d , CMD_GET_IR_CODE , NULL , 0 , ircode , 4 ) ;
2006-01-09 18:21:29 -02:00
* event = 0 ;
* state = REMOTE_NO_KEY_PRESSED ;
for ( i = 0 ; i < d - > props . rc_key_map_size ; i + + ) {
2006-01-09 18:21:31 -02:00
if ( keymap [ i ] . custom = = ircode [ 2 ] & &
keymap [ i ] . data = = ircode [ 3 ] ) {
2006-01-09 18:21:29 -02:00
* event = keymap [ i ] . event ;
* state = REMOTE_KEY_PRESSED ;
return 0 ;
}
}
return 0 ;
}
2006-01-23 17:11:09 -02:00
static struct dvb_usb_rc_key dvico_mce_rc_keys [ ] = {
2006-01-09 18:21:31 -02:00
{ 0xfe , 0x02 , KEY_TV } ,
{ 0xfe , 0x0e , KEY_MP3 } ,
{ 0xfe , 0x1a , KEY_DVD } ,
{ 0xfe , 0x1e , KEY_FAVORITES } ,
{ 0xfe , 0x16 , KEY_SETUP } ,
{ 0xfe , 0x46 , KEY_POWER2 } ,
{ 0xfe , 0x0a , KEY_EPG } ,
{ 0xfe , 0x49 , KEY_BACK } ,
{ 0xfe , 0x4d , KEY_MENU } ,
{ 0xfe , 0x51 , KEY_UP } ,
{ 0xfe , 0x5b , KEY_LEFT } ,
{ 0xfe , 0x5f , KEY_RIGHT } ,
{ 0xfe , 0x53 , KEY_DOWN } ,
{ 0xfe , 0x5e , KEY_OK } ,
{ 0xfe , 0x59 , KEY_INFO } ,
{ 0xfe , 0x55 , KEY_TAB } ,
{ 0xfe , 0x0f , KEY_PREVIOUSSONG } , /* Replay */
{ 0xfe , 0x12 , KEY_NEXTSONG } , /* Skip */
{ 0xfe , 0x42 , KEY_ENTER } , /* Windows/Start */
{ 0xfe , 0x15 , KEY_VOLUMEUP } ,
{ 0xfe , 0x05 , KEY_VOLUMEDOWN } ,
{ 0xfe , 0x11 , KEY_CHANNELUP } ,
{ 0xfe , 0x09 , KEY_CHANNELDOWN } ,
{ 0xfe , 0x52 , KEY_CAMERA } ,
{ 0xfe , 0x5a , KEY_TUNER } , /* Live */
{ 0xfe , 0x19 , KEY_OPEN } ,
{ 0xfe , 0x0b , KEY_1 } ,
{ 0xfe , 0x17 , KEY_2 } ,
{ 0xfe , 0x1b , KEY_3 } ,
{ 0xfe , 0x07 , KEY_4 } ,
{ 0xfe , 0x50 , KEY_5 } ,
{ 0xfe , 0x54 , KEY_6 } ,
{ 0xfe , 0x48 , KEY_7 } ,
{ 0xfe , 0x4c , KEY_8 } ,
{ 0xfe , 0x58 , KEY_9 } ,
{ 0xfe , 0x13 , KEY_ANGLE } , /* Aspect */
{ 0xfe , 0x03 , KEY_0 } ,
{ 0xfe , 0x1f , KEY_ZOOM } ,
{ 0xfe , 0x43 , KEY_REWIND } ,
{ 0xfe , 0x47 , KEY_PLAYPAUSE } ,
{ 0xfe , 0x4f , KEY_FASTFORWARD } ,
{ 0xfe , 0x57 , KEY_MUTE } ,
{ 0xfe , 0x0d , KEY_STOP } ,
{ 0xfe , 0x01 , KEY_RECORD } ,
{ 0xfe , 0x4e , KEY_POWER } ,
} ;
2006-03-26 05:43:36 -03:00
static struct dvb_usb_rc_key dvico_portable_rc_keys [ ] = {
{ 0xfc , 0x02 , KEY_SETUP } , /* Profile */
{ 0xfc , 0x43 , KEY_POWER2 } ,
{ 0xfc , 0x06 , KEY_EPG } ,
{ 0xfc , 0x5a , KEY_BACK } ,
{ 0xfc , 0x05 , KEY_MENU } ,
{ 0xfc , 0x47 , KEY_INFO } ,
{ 0xfc , 0x01 , KEY_TAB } ,
{ 0xfc , 0x42 , KEY_PREVIOUSSONG } , /* Replay */
{ 0xfc , 0x49 , KEY_VOLUMEUP } ,
{ 0xfc , 0x09 , KEY_VOLUMEDOWN } ,
{ 0xfc , 0x54 , KEY_CHANNELUP } ,
{ 0xfc , 0x0b , KEY_CHANNELDOWN } ,
2006-03-26 18:59:45 -03:00
{ 0xfc , 0x16 , KEY_CAMERA } ,
2006-03-26 05:43:36 -03:00
{ 0xfc , 0x40 , KEY_TUNER } , /* ATV/DTV */
{ 0xfc , 0x45 , KEY_OPEN } ,
{ 0xfc , 0x19 , KEY_1 } ,
{ 0xfc , 0x18 , KEY_2 } ,
{ 0xfc , 0x1b , KEY_3 } ,
{ 0xfc , 0x1a , KEY_4 } ,
{ 0xfc , 0x58 , KEY_5 } ,
{ 0xfc , 0x59 , KEY_6 } ,
{ 0xfc , 0x15 , KEY_7 } ,
{ 0xfc , 0x14 , KEY_8 } ,
{ 0xfc , 0x17 , KEY_9 } ,
{ 0xfc , 0x44 , KEY_ANGLE } , /* Aspect */
{ 0xfc , 0x55 , KEY_0 } ,
{ 0xfc , 0x07 , KEY_ZOOM } ,
{ 0xfc , 0x0a , KEY_REWIND } ,
{ 0xfc , 0x08 , KEY_PLAYPAUSE } ,
{ 0xfc , 0x4b , KEY_FASTFORWARD } ,
{ 0xfc , 0x5b , KEY_MUTE } ,
{ 0xfc , 0x04 , KEY_STOP } ,
{ 0xfc , 0x56 , KEY_RECORD } ,
{ 0xfc , 0x57 , KEY_POWER } ,
{ 0xfc , 0x41 , KEY_UNKNOWN } , /* INPUT */
{ 0xfc , 0x00 , KEY_UNKNOWN } , /* HD */
} ;
2006-01-09 18:21:28 -02:00
static int cxusb_dee1601_demod_init ( struct dvb_frontend * fe )
{
2006-02-07 06:49:11 -02:00
static u8 clock_config [ ] = { CLOCK_CTL , 0x38 , 0x28 } ;
2006-01-09 18:21:28 -02:00
static u8 reset [ ] = { RESET , 0x80 } ;
static u8 adc_ctl_1_cfg [ ] = { ADC_CTL_1 , 0x40 } ;
static u8 agc_cfg [ ] = { AGC_TARGET , 0x28 , 0x20 } ;
static u8 gpp_ctl_cfg [ ] = { GPP_CTL , 0x33 } ;
static u8 capt_range_cfg [ ] = { CAPT_RANGE , 0x32 } ;
mt352_write ( fe , clock_config , sizeof ( clock_config ) ) ;
udelay ( 200 ) ;
mt352_write ( fe , reset , sizeof ( reset ) ) ;
mt352_write ( fe , adc_ctl_1_cfg , sizeof ( adc_ctl_1_cfg ) ) ;
mt352_write ( fe , agc_cfg , sizeof ( agc_cfg ) ) ;
mt352_write ( fe , gpp_ctl_cfg , sizeof ( gpp_ctl_cfg ) ) ;
mt352_write ( fe , capt_range_cfg , sizeof ( capt_range_cfg ) ) ;
return 0 ;
}
2006-01-11 19:40:33 -02:00
static int cxusb_mt352_demod_init ( struct dvb_frontend * fe )
{ /* used in both lgz201 and th7579 */
2006-02-07 06:49:12 -02:00
static u8 clock_config [ ] = { CLOCK_CTL , 0x38 , 0x29 } ;
2006-01-11 19:40:33 -02:00
static u8 reset [ ] = { RESET , 0x80 } ;
static u8 adc_ctl_1_cfg [ ] = { ADC_CTL_1 , 0x40 } ;
static u8 agc_cfg [ ] = { AGC_TARGET , 0x24 , 0x20 } ;
static u8 gpp_ctl_cfg [ ] = { GPP_CTL , 0x33 } ;
static u8 capt_range_cfg [ ] = { CAPT_RANGE , 0x32 } ;
mt352_write ( fe , clock_config , sizeof ( clock_config ) ) ;
udelay ( 200 ) ;
mt352_write ( fe , reset , sizeof ( reset ) ) ;
mt352_write ( fe , adc_ctl_1_cfg , sizeof ( adc_ctl_1_cfg ) ) ;
mt352_write ( fe , agc_cfg , sizeof ( agc_cfg ) ) ;
mt352_write ( fe , gpp_ctl_cfg , sizeof ( gpp_ctl_cfg ) ) ;
mt352_write ( fe , capt_range_cfg , sizeof ( capt_range_cfg ) ) ;
return 0 ;
}
2006-04-18 17:47:12 -03:00
static int cxusb_lgh064f_tuner_set_params ( struct dvb_frontend * fe ,
struct dvb_frontend_parameters * fep )
2006-04-16 13:19:24 -03:00
{
struct dvb_usb_device * d = fe - > dvb - > priv ;
2006-04-18 17:47:08 -03:00
return lg_h06xf_pll_set ( fe , & d - > i2c_adap , fep ) ;
2006-04-16 13:19:24 -03:00
}
2006-01-23 17:11:09 -02:00
static struct cx22702_config cxusb_cx22702_config = {
2005-07-07 17:58:10 -07:00
. demod_address = 0x63 ,
2005-07-07 17:58:15 -07:00
. output_mode = CX22702_PARALLEL_OUTPUT ,
2005-07-07 17:58:10 -07:00
} ;
2006-02-27 00:08:17 -03:00
static struct lgdt330x_config cxusb_lgdt3303_config = {
2006-01-09 15:25:47 -02:00
. demod_address = 0x0e ,
. demod_chip = LGDT3303 ,
} ;
2006-01-23 17:11:09 -02:00
static struct mt352_config cxusb_dee1601_config = {
2006-01-09 18:21:28 -02:00
. demod_address = 0x0f ,
. demod_init = cxusb_dee1601_demod_init ,
} ;
2006-06-11 04:24:31 -03:00
static struct zl10353_config cxusb_zl10353_dee1601_config = {
. demod_address = 0x0f ,
} ;
2006-04-19 20:49:28 -03:00
static struct mt352_config cxusb_mt352_config = {
2006-01-11 19:40:33 -02:00
/* used in both lgz201 and th7579 */
. demod_address = 0x0f ,
. demod_init = cxusb_mt352_demod_init ,
} ;
2005-07-07 17:58:10 -07:00
/* Callbacks for DVB USB */
2006-01-09 15:25:47 -02:00
static int cxusb_fmd1216me_tuner_attach ( struct dvb_usb_device * d )
2005-07-07 17:58:10 -07:00
{
u8 bpll [ 4 ] = { 0x0b , 0xdc , 0x9c , 0xa0 } ;
d - > pll_addr = 0x61 ;
2006-01-09 15:25:47 -02:00
memcpy ( d - > pll_init , bpll , 4 ) ;
2005-07-07 17:58:10 -07:00
d - > pll_desc = & dvb_pll_fmd1216me ;
2006-05-14 04:49:00 -03:00
2006-05-14 05:01:31 -03:00
d - > fe - > ops . tuner_ops . init = dvb_usb_tuner_init_i2c ;
d - > fe - > ops . tuner_ops . set_params = dvb_usb_tuner_set_params_i2c ;
2006-05-14 04:49:00 -03:00
2005-07-07 17:58:10 -07:00
return 0 ;
}
2006-01-09 18:21:28 -02:00
static int cxusb_dee1601_tuner_attach ( struct dvb_usb_device * d )
{
d - > pll_addr = 0x61 ;
d - > pll_desc = & dvb_pll_thomson_dtt7579 ;
2006-05-14 05:01:31 -03:00
d - > fe - > ops . tuner_ops . calc_regs = dvb_usb_tuner_calc_regs ;
2006-01-09 18:21:28 -02:00
return 0 ;
}
2006-01-11 19:40:33 -02:00
static int cxusb_lgz201_tuner_attach ( struct dvb_usb_device * d )
{
d - > pll_addr = 0x61 ;
d - > pll_desc = & dvb_pll_lg_z201 ;
2006-05-14 05:01:31 -03:00
d - > fe - > ops . tuner_ops . calc_regs = dvb_usb_tuner_calc_regs ;
2006-01-11 19:40:33 -02:00
return 0 ;
}
static int cxusb_dtt7579_tuner_attach ( struct dvb_usb_device * d )
{
d - > pll_addr = 0x60 ;
d - > pll_desc = & dvb_pll_thomson_dtt7579 ;
2006-05-14 05:01:31 -03:00
d - > fe - > ops . tuner_ops . calc_regs = dvb_usb_tuner_calc_regs ;
2006-05-14 04:49:00 -03:00
return 0 ;
}
static int cxusb_lgdt3303_tuner_attach ( struct dvb_usb_device * d )
{
2006-05-14 05:01:31 -03:00
d - > fe - > ops . tuner_ops . set_params = cxusb_lgh064f_tuner_set_params ;
2006-01-11 19:40:33 -02:00
return 0 ;
}
2006-01-09 15:25:47 -02:00
static int cxusb_cx22702_frontend_attach ( struct dvb_usb_device * d )
2005-07-07 17:58:10 -07:00
{
2005-09-09 13:02:51 -07:00
u8 b ;
2005-07-07 17:58:10 -07:00
if ( usb_set_interface ( d - > udev , 0 , 6 ) < 0 )
2005-07-07 17:58:15 -07:00
err ( " set interface failed " ) ;
2005-07-07 17:58:10 -07:00
2005-09-09 13:02:51 -07:00
cxusb_ctrl_msg ( d , CMD_DIGITAL , NULL , 0 , & b , 1 ) ;
2005-07-07 17:58:10 -07:00
2006-05-14 04:49:00 -03:00
if ( ( d - > fe = cx22702_attach ( & cxusb_cx22702_config , & d - > i2c_adap ) ) ! = NULL )
2005-07-07 17:58:10 -07:00
return 0 ;
return - EIO ;
}
2006-02-27 00:08:17 -03:00
static int cxusb_lgdt3303_frontend_attach ( struct dvb_usb_device * d )
2006-01-09 15:25:47 -02:00
{
2006-01-09 15:25:47 -02:00
if ( usb_set_interface ( d - > udev , 0 , 7 ) < 0 )
2006-01-09 15:25:47 -02:00
err ( " set interface failed " ) ;
cxusb_ctrl_msg ( d , CMD_DIGITAL , NULL , 0 , NULL , 0 ) ;
2006-05-14 04:49:00 -03:00
if ( ( d - > fe = lgdt330x_attach ( & cxusb_lgdt3303_config , & d - > i2c_adap ) ) ! = NULL )
2006-01-09 15:25:47 -02:00
return 0 ;
return - EIO ;
}
2006-01-11 19:40:33 -02:00
static int cxusb_mt352_frontend_attach ( struct dvb_usb_device * d )
{ /* used in both lgz201 and th7579 */
if ( usb_set_interface ( d - > udev , 0 , 0 ) < 0 )
err ( " set interface failed " ) ;
cxusb_ctrl_msg ( d , CMD_DIGITAL , NULL , 0 , NULL , 0 ) ;
2006-05-14 04:49:00 -03:00
if ( ( d - > fe = mt352_attach ( & cxusb_mt352_config , & d - > i2c_adap ) ) ! = NULL )
2006-01-11 19:40:33 -02:00
return 0 ;
return - EIO ;
}
2006-01-09 18:21:28 -02:00
static int cxusb_dee1601_frontend_attach ( struct dvb_usb_device * d )
{
if ( usb_set_interface ( d - > udev , 0 , 0 ) < 0 )
err ( " set interface failed " ) ;
cxusb_ctrl_msg ( d , CMD_DIGITAL , NULL , 0 , NULL , 0 ) ;
2006-06-11 04:24:31 -03:00
if ( ( ( d - > fe = mt352_attach ( & cxusb_dee1601_config , & d - > i2c_adap ) ) ! = NULL ) | |
( ( d - > fe = zl10353_attach ( & cxusb_zl10353_dee1601_config , & d - > i2c_adap ) ) ! = NULL ) )
2006-01-09 18:21:28 -02:00
return 0 ;
return - EIO ;
}
2006-01-09 18:21:38 -02:00
/*
* DViCO bluebird firmware needs the " warm " product ID to be patched into the
* firmware file before download .
*/
# define BLUEBIRD_01_ID_OFFSET 6638
static int bluebird_patch_dvico_firmware_download ( struct usb_device * udev , const struct firmware * fw )
{
if ( fw - > size < BLUEBIRD_01_ID_OFFSET + 4 )
return - EINVAL ;
if ( fw - > data [ BLUEBIRD_01_ID_OFFSET ] = = ( USB_VID_DVICO & 0xff ) & &
fw - > data [ BLUEBIRD_01_ID_OFFSET + 1 ] = = USB_VID_DVICO > > 8 ) {
fw - > data [ BLUEBIRD_01_ID_OFFSET + 2 ] = udev - > descriptor . idProduct + 1 ;
fw - > data [ BLUEBIRD_01_ID_OFFSET + 3 ] = udev - > descriptor . idProduct > > 8 ;
return usb_cypress_load_firmware ( udev , fw , CYPRESS_FX2 ) ;
}
return - EINVAL ;
}
2005-07-07 17:58:10 -07:00
/* DVB USB Driver stuff */
2006-01-09 15:25:47 -02:00
static struct dvb_usb_properties cxusb_medion_properties ;
2006-01-09 15:32:42 -02:00
static struct dvb_usb_properties cxusb_bluebird_lgh064f_properties ;
2006-01-09 18:21:28 -02:00
static struct dvb_usb_properties cxusb_bluebird_dee1601_properties ;
2006-01-11 19:40:33 -02:00
static struct dvb_usb_properties cxusb_bluebird_lgz201_properties ;
static struct dvb_usb_properties cxusb_bluebird_dtt7579_properties ;
2005-07-07 17:58:10 -07:00
static int cxusb_probe ( struct usb_interface * intf ,
const struct usb_device_id * id )
{
2006-01-09 15:25:47 -02:00
if ( dvb_usb_device_init ( intf , & cxusb_medion_properties , THIS_MODULE , NULL ) = = 0 | |
2006-01-09 18:21:28 -02:00
dvb_usb_device_init ( intf , & cxusb_bluebird_lgh064f_properties , THIS_MODULE , NULL ) = = 0 | |
2006-01-11 19:40:33 -02:00
dvb_usb_device_init ( intf , & cxusb_bluebird_dee1601_properties , THIS_MODULE , NULL ) = = 0 | |
dvb_usb_device_init ( intf , & cxusb_bluebird_lgz201_properties , THIS_MODULE , NULL ) = = 0 | |
dvb_usb_device_init ( intf , & cxusb_bluebird_dtt7579_properties , THIS_MODULE , NULL ) = = 0 ) {
2006-01-09 15:25:47 -02:00
return 0 ;
}
return - EINVAL ;
2005-07-07 17:58:10 -07:00
}
static struct usb_device_id cxusb_table [ ] = {
{ USB_DEVICE ( USB_VID_MEDION , USB_PID_MEDION_MD95700 ) } ,
2006-01-09 15:25:47 -02:00
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_LG064F_COLD ) } ,
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_LG064F_WARM ) } ,
2006-01-09 18:21:28 -02:00
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_DEE1601_COLD ) } ,
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_DEE1601_WARM ) } ,
2006-01-11 19:40:33 -02:00
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_LGZ201_COLD ) } ,
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_LGZ201_WARM ) } ,
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_TH7579_COLD ) } ,
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DVICO_BLUEBIRD_TH7579_WARM ) } ,
2006-01-11 23:21:00 -02:00
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DIGITALNOW_BLUEBIRD_DEE1601_COLD ) } ,
{ USB_DEVICE ( USB_VID_DVICO , USB_PID_DIGITALNOW_BLUEBIRD_DEE1601_WARM ) } ,
2005-07-07 17:58:10 -07:00
{ } /* Terminating entry */
} ;
MODULE_DEVICE_TABLE ( usb , cxusb_table ) ;
2006-01-09 15:25:47 -02:00
static struct dvb_usb_properties cxusb_medion_properties = {
2005-07-07 17:58:10 -07:00
. caps = DVB_USB_IS_AN_I2C_ADAPTER ,
. usb_ctrl = CYPRESS_FX2 ,
. size_of_priv = sizeof ( struct cxusb_state ) ,
. streaming_ctrl = cxusb_streaming_ctrl ,
. power_ctrl = cxusb_power_ctrl ,
2006-01-09 15:25:47 -02:00
. frontend_attach = cxusb_cx22702_frontend_attach ,
. tuner_attach = cxusb_fmd1216me_tuner_attach ,
2005-07-07 17:58:10 -07:00
. i2c_algo = & cxusb_i2c_algo ,
. generic_bulk_ctrl_endpoint = 0x01 ,
/* parameter for the MPEG2-data transfer */
. urb = {
2005-09-09 13:02:51 -07:00
. type = DVB_USB_BULK ,
2005-07-07 17:58:10 -07:00
. count = 5 ,
. endpoint = 0x02 ,
. u = {
2005-09-09 13:02:51 -07:00
. bulk = {
. buffersize = 8192 ,
2005-07-07 17:58:10 -07:00
}
}
} ,
. num_device_descs = 1 ,
. devices = {
{ " Medion MD95700 (MDUSBTV-HYBRID) " ,
{ NULL } ,
{ & cxusb_table [ 0 ] , NULL } ,
} ,
}
} ;
2006-01-09 15:32:42 -02:00
static struct dvb_usb_properties cxusb_bluebird_lgh064f_properties = {
2006-01-09 15:25:47 -02:00
. caps = DVB_USB_IS_AN_I2C_ADAPTER ,
2006-01-09 18:21:38 -02:00
. usb_ctrl = DEVICE_SPECIFIC ,
. firmware = " dvb-usb-bluebird-01.fw " ,
. download_firmware = bluebird_patch_dvico_firmware_download ,
2006-01-09 15:25:47 -02:00
/* use usb alt setting 0 for EP4 transfer (dvb-t),
use usb alt setting 7 for EP2 transfer ( atsc ) */
2006-01-09 15:25:47 -02:00
. size_of_priv = sizeof ( struct cxusb_state ) ,
. streaming_ctrl = cxusb_streaming_ctrl ,
2006-04-19 20:40:01 -03:00
. power_ctrl = cxusb_bluebird_power_ctrl ,
2006-02-27 00:08:17 -03:00
. frontend_attach = cxusb_lgdt3303_frontend_attach ,
2006-05-14 04:49:00 -03:00
. tuner_attach = cxusb_lgdt3303_tuner_attach ,
2006-01-09 15:25:47 -02:00
. i2c_algo = & cxusb_i2c_algo ,
2006-03-26 05:43:36 -03:00
. rc_interval = 100 ,
. rc_key_map = dvico_portable_rc_keys ,
. rc_key_map_size = ARRAY_SIZE ( dvico_portable_rc_keys ) ,
. rc_query = cxusb_rc_query ,
2006-01-09 15:25:47 -02:00
. generic_bulk_ctrl_endpoint = 0x01 ,
/* parameter for the MPEG2-data transfer */
. urb = {
. type = DVB_USB_BULK ,
. count = 5 ,
. endpoint = 0x02 ,
. u = {
. bulk = {
. buffersize = 8192 ,
}
}
} ,
. num_device_descs = 1 ,
. devices = {
{ " DViCO FusionHDTV5 USB Gold " ,
{ & cxusb_table [ 1 ] , NULL } ,
{ & cxusb_table [ 2 ] , NULL } ,
} ,
}
} ;
2006-01-09 18:21:28 -02:00
static struct dvb_usb_properties cxusb_bluebird_dee1601_properties = {
. caps = DVB_USB_IS_AN_I2C_ADAPTER ,
2006-01-09 18:21:38 -02:00
. usb_ctrl = DEVICE_SPECIFIC ,
. firmware = " dvb-usb-bluebird-01.fw " ,
. download_firmware = bluebird_patch_dvico_firmware_download ,
2006-01-09 18:21:28 -02:00
/* use usb alt setting 0 for EP4 transfer (dvb-t),
use usb alt setting 7 for EP2 transfer ( atsc ) */
. size_of_priv = sizeof ( struct cxusb_state ) ,
. streaming_ctrl = cxusb_streaming_ctrl ,
2006-04-19 20:40:01 -03:00
. power_ctrl = cxusb_bluebird_power_ctrl ,
2006-01-09 18:21:28 -02:00
. frontend_attach = cxusb_dee1601_frontend_attach ,
. tuner_attach = cxusb_dee1601_tuner_attach ,
. i2c_algo = & cxusb_i2c_algo ,
2006-01-09 18:21:29 -02:00
. rc_interval = 150 ,
. rc_key_map = dvico_mce_rc_keys ,
. rc_key_map_size = ARRAY_SIZE ( dvico_mce_rc_keys ) ,
. rc_query = cxusb_rc_query ,
2006-01-09 18:21:28 -02:00
. generic_bulk_ctrl_endpoint = 0x01 ,
/* parameter for the MPEG2-data transfer */
. urb = {
. type = DVB_USB_BULK ,
. count = 5 ,
. endpoint = 0x04 ,
. u = {
. bulk = {
. buffersize = 8192 ,
}
}
} ,
2006-01-11 23:21:00 -02:00
. num_device_descs = 2 ,
2006-01-09 18:21:28 -02:00
. devices = {
{ " DViCO FusionHDTV DVB-T Dual USB " ,
{ & cxusb_table [ 3 ] , NULL } ,
{ & cxusb_table [ 4 ] , NULL } ,
} ,
2006-01-11 23:21:00 -02:00
{ " DigitalNow DVB-T Dual USB " ,
{ & cxusb_table [ 9 ] , NULL } ,
{ & cxusb_table [ 10 ] , NULL } ,
} ,
2006-01-09 18:21:28 -02:00
}
} ;
2006-01-11 19:40:33 -02:00
static struct dvb_usb_properties cxusb_bluebird_lgz201_properties = {
. caps = DVB_USB_IS_AN_I2C_ADAPTER ,
. usb_ctrl = DEVICE_SPECIFIC ,
. firmware = " dvb-usb-bluebird-01.fw " ,
. download_firmware = bluebird_patch_dvico_firmware_download ,
/* use usb alt setting 0 for EP4 transfer (dvb-t),
use usb alt setting 7 for EP2 transfer ( atsc ) */
. size_of_priv = sizeof ( struct cxusb_state ) ,
. streaming_ctrl = cxusb_streaming_ctrl ,
2006-04-19 20:40:01 -03:00
. power_ctrl = cxusb_bluebird_power_ctrl ,
2006-01-11 19:40:33 -02:00
. frontend_attach = cxusb_mt352_frontend_attach ,
. tuner_attach = cxusb_lgz201_tuner_attach ,
. i2c_algo = & cxusb_i2c_algo ,
2006-03-26 05:43:36 -03:00
. rc_interval = 100 ,
. rc_key_map = dvico_portable_rc_keys ,
. rc_key_map_size = ARRAY_SIZE ( dvico_portable_rc_keys ) ,
. rc_query = cxusb_rc_query ,
2006-01-11 19:40:33 -02:00
. generic_bulk_ctrl_endpoint = 0x01 ,
/* parameter for the MPEG2-data transfer */
. urb = {
. type = DVB_USB_BULK ,
. count = 5 ,
. endpoint = 0x04 ,
. u = {
. bulk = {
. buffersize = 8192 ,
}
}
} ,
. num_device_descs = 1 ,
. devices = {
{ " DViCO FusionHDTV DVB-T USB (LGZ201) " ,
{ & cxusb_table [ 5 ] , NULL } ,
{ & cxusb_table [ 6 ] , NULL } ,
} ,
}
} ;
static struct dvb_usb_properties cxusb_bluebird_dtt7579_properties = {
. caps = DVB_USB_IS_AN_I2C_ADAPTER ,
. usb_ctrl = DEVICE_SPECIFIC ,
. firmware = " dvb-usb-bluebird-01.fw " ,
. download_firmware = bluebird_patch_dvico_firmware_download ,
/* use usb alt setting 0 for EP4 transfer (dvb-t),
use usb alt setting 7 for EP2 transfer ( atsc ) */
. size_of_priv = sizeof ( struct cxusb_state ) ,
. streaming_ctrl = cxusb_streaming_ctrl ,
2006-04-19 20:40:01 -03:00
. power_ctrl = cxusb_bluebird_power_ctrl ,
2006-01-11 19:40:33 -02:00
. frontend_attach = cxusb_mt352_frontend_attach ,
. tuner_attach = cxusb_dtt7579_tuner_attach ,
. i2c_algo = & cxusb_i2c_algo ,
2006-03-26 05:43:36 -03:00
. rc_interval = 100 ,
. rc_key_map = dvico_portable_rc_keys ,
. rc_key_map_size = ARRAY_SIZE ( dvico_portable_rc_keys ) ,
. rc_query = cxusb_rc_query ,
2006-01-11 19:40:33 -02:00
. generic_bulk_ctrl_endpoint = 0x01 ,
/* parameter for the MPEG2-data transfer */
. urb = {
. type = DVB_USB_BULK ,
. count = 5 ,
. endpoint = 0x04 ,
. u = {
. bulk = {
. buffersize = 8192 ,
}
}
} ,
. num_device_descs = 1 ,
. devices = {
{ " DViCO FusionHDTV DVB-T USB (TH7579) " ,
{ & cxusb_table [ 7 ] , NULL } ,
{ & cxusb_table [ 8 ] , NULL } ,
} ,
}
} ;
2005-07-07 17:58:10 -07:00
static struct usb_driver cxusb_driver = {
2005-07-07 17:58:30 -07:00
. name = " dvb_usb_cxusb " ,
2005-07-07 17:58:10 -07:00
. probe = cxusb_probe ,
. disconnect = dvb_usb_device_exit ,
. id_table = cxusb_table ,
} ;
/* module stuff */
static int __init cxusb_module_init ( void )
{
int result ;
if ( ( result = usb_register ( & cxusb_driver ) ) ) {
err ( " usb_register failed. Error number %d " , result ) ;
return result ;
}
return 0 ;
}
static void __exit cxusb_module_exit ( void )
{
/* deregister this driver from the USB subsystem */
usb_deregister ( & cxusb_driver ) ;
}
module_init ( cxusb_module_init ) ;
module_exit ( cxusb_module_exit ) ;
MODULE_AUTHOR ( " Patrick Boettcher <patrick.boettcher@desy.de> " ) ;
2006-01-13 14:10:25 -02:00
MODULE_AUTHOR ( " Michael Krufky <mkrufky@m1k.net> " ) ;
MODULE_AUTHOR ( " Chris Pascoe <c.pascoe@itee.uq.edu.au> " ) ;
2005-07-07 17:58:10 -07:00
MODULE_DESCRIPTION ( " Driver for Conexant USB2.0 hybrid reference design " ) ;
MODULE_VERSION ( " 1.0-alpha " ) ;
MODULE_LICENSE ( " GPL " ) ;