1 /* DVB USB compliant linux driver for MSI Mega Sky 580 DVB-T USB2.0 receiver
3 * Copyright (C) 2006 Aapo Tahkola (aet@rasterburn.org)
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License as published by the Free
7 * Software Foundation, version 2.
9 * see Documentation/dvb/README.dvb-usb for more information
15 #include "mt352_priv.h"
19 static int dvb_usb_m920x_debug
;
20 module_param_named(debug
,dvb_usb_m920x_debug
, int, 0644);
21 MODULE_PARM_DESC(debug
, "set debugging level (1=rc (or-able))." DVB_USB_DEBUG_STATUS
);
23 static struct dvb_usb_rc_key megasky_rc_keys
[] = {
24 { 0x0, 0x12, KEY_POWER
},
25 { 0x0, 0x1e, KEY_CYCLEWINDOWS
}, /* min/max */
26 { 0x0, 0x02, KEY_CHANNELUP
},
27 { 0x0, 0x05, KEY_CHANNELDOWN
},
28 { 0x0, 0x03, KEY_VOLUMEUP
},
29 { 0x0, 0x06, KEY_VOLUMEDOWN
},
30 { 0x0, 0x04, KEY_MUTE
},
31 { 0x0, 0x07, KEY_OK
}, /* TS */
32 { 0x0, 0x08, KEY_STOP
},
33 { 0x0, 0x09, KEY_MENU
}, /* swap */
34 { 0x0, 0x0a, KEY_REWIND
},
35 { 0x0, 0x1b, KEY_PAUSE
},
36 { 0x0, 0x1f, KEY_FASTFORWARD
},
37 { 0x0, 0x0c, KEY_RECORD
},
38 { 0x0, 0x0d, KEY_CAMERA
}, /* screenshot */
39 { 0x0, 0x0e, KEY_COFFEE
}, /* "MTS" */
42 static inline int m9206_read(struct usb_device
*udev
, u8 request
, u16 value
,\
43 u16 index
, void *data
, int size
)
47 ret
= usb_control_msg(udev
, usb_rcvctrlpipe(udev
, 0),
48 request
, USB_TYPE_VENDOR
| USB_DIR_IN
,
49 value
, index
, data
, size
, 2000);
59 static inline int m9206_write(struct usb_device
*udev
, u8 request
,
64 ret
= usb_control_msg(udev
, usb_sndctrlpipe(udev
, 0),
65 request
, USB_TYPE_VENDOR
| USB_DIR_OUT
,
66 value
, index
, NULL
, 0, 2000);
70 static int m9206_rc_init(struct usb_device
*udev
)
74 /* Remote controller init. */
75 if ((ret
= m9206_write(udev
, M9206_CORE
, 0xa8, M9206_RC_INIT2
)) != 0)
78 if ((ret
= m9206_write(udev
, M9206_CORE
, 0x51, M9206_RC_INIT1
)) != 0)
84 static int m9206_rc_query(struct dvb_usb_device
*d
, u32
*event
, int *state
)
86 struct m9206_state
*m
= d
->priv
;
91 if ((ret
= m9206_read(d
->udev
, M9206_CORE
, 0x0, M9206_RC_STATE
, rc_state
, 1)) != 0)
94 if ((ret
= m9206_read(d
->udev
, M9206_CORE
, 0x0, M9206_RC_KEY
, rc_state
+ 1, 1)) != 0)
97 for (i
= 0; i
< ARRAY_SIZE(megasky_rc_keys
); i
++)
98 if (megasky_rc_keys
[i
].data
== rc_state
[1]) {
99 *event
= megasky_rc_keys
[i
].event
;
101 switch(rc_state
[0]) {
103 *state
= REMOTE_NO_KEY_PRESSED
;
109 *state
= REMOTE_KEY_PRESSED
;
114 if (++m
->rep_count
> 2)
115 *state
= REMOTE_KEY_REPEAT
;
119 deb_rc("Unexpected rc response %x\n", rc_state
[0]);
120 *state
= REMOTE_NO_KEY_PRESSED
;
125 if (rc_state
[1] != 0)
126 deb_rc("Unknown rc key %x\n", rc_state
[1]);
128 *state
= REMOTE_NO_KEY_PRESSED
;
136 static int m9206_i2c_xfer(struct i2c_adapter
*adap
, struct i2c_msg msg
[],
139 struct dvb_usb_device
*d
= i2c_get_adapdata(adap
);
140 struct m9206_state
*m
= d
->priv
;
144 if (mutex_lock_interruptible(&d
->i2c_mutex
) < 0)
150 for (i
= 0; i
< num
; i
++) {
151 if ((ret
= m9206_write(d
->udev
, M9206_I2C
, msg
[i
].addr
, 0x80)) != 0)
154 if ((ret
= m9206_write(d
->udev
, M9206_I2C
, msg
[i
].buf
[0], 0x0)) != 0)
157 if (i
+ 1 < num
&& msg
[i
+ 1].flags
& I2C_M_RD
) {
160 for (i2c_i
= 0; i2c_i
< M9206_I2C_MAX
; i2c_i
++)
161 if (msg
[i
].addr
== m
->i2c_r
[i2c_i
].addr
)
164 if (i2c_i
>= M9206_I2C_MAX
) {
165 deb_rc("No magic for i2c addr!\n");
170 if ((ret
= m9206_write(d
->udev
, M9206_I2C
, m
->i2c_r
[i2c_i
].magic
, 0x80)) != 0)
173 if ((ret
= m9206_read(d
->udev
, M9206_I2C
, 0x0, 0x60, msg
[i
+ 1].buf
, msg
[i
+ 1].len
)) != 0)
181 if ((ret
= m9206_write(d
->udev
, M9206_I2C
, msg
[i
].buf
[1], 0x40)) != 0)
187 mutex_unlock(&d
->i2c_mutex
);
192 static u32
m9206_i2c_func(struct i2c_adapter
*adapter
)
197 static struct i2c_algorithm m9206_i2c_algo
= {
198 .master_xfer
= m9206_i2c_xfer
,
199 .functionality
= m9206_i2c_func
,
203 static int m9206_set_filter(struct dvb_usb_adapter
*adap
, int type
, int idx
,
213 if ((ret
= m9206_write(adap
->dev
->udev
, M9206_FILTER
, pid
, (type
<< 8) | (idx
* 4) )) != 0)
216 if ((ret
= m9206_write(adap
->dev
->udev
, M9206_FILTER
, 0, (type
<< 8) | (idx
* 4) )) != 0)
222 static int m9206_update_filters(struct dvb_usb_adapter
*adap
)
224 struct m9206_state
*m
= adap
->dev
->priv
;
225 int enabled
= m
->filtering_enabled
;
226 int i
, ret
= 0, filter
= 0;
228 for (i
= 0; i
< M9206_MAX_FILTERS
; i
++)
229 if (m
->filters
[i
] == 8192)
232 /* Disable all filters */
233 if ((ret
= m9206_set_filter(adap
, 0x81, 1, enabled
)) != 0)
236 for (i
= 0; i
< M9206_MAX_FILTERS
; i
++)
237 if ((ret
= m9206_set_filter(adap
, 0x81, i
+ 2, 0)) != 0)
240 if ((ret
= m9206_set_filter(adap
, 0x82, 0, 0x0)) != 0)
245 for (i
= 0; i
< M9206_MAX_FILTERS
; i
++) {
246 if (m
->filters
[i
] == 0)
249 if ((ret
= m9206_set_filter(adap
, 0x81, filter
+ 2, m
->filters
[i
])) != 0)
256 if ((ret
= m9206_set_filter(adap
, 0x82, 0, 0x02f5)) != 0)
262 static int m9206_pid_filter_ctrl(struct dvb_usb_adapter
*adap
, int onoff
)
264 struct m9206_state
*m
= adap
->dev
->priv
;
266 m
->filtering_enabled
= onoff
? 1 : 0;
268 return m9206_update_filters(adap
);
271 static int m9206_pid_filter(struct dvb_usb_adapter
*adap
, int index
, u16 pid
,
274 struct m9206_state
*m
= adap
->dev
->priv
;
276 m
->filters
[index
] = onoff
? pid
: 0;
278 return m9206_update_filters(adap
);
281 static int m9206_firmware_download(struct usb_device
*udev
,
282 const struct firmware
*fw
)
284 u16 value
, index
, size
;
286 int i
, pass
, ret
= 0;
288 buff
= kmalloc(65536, GFP_KERNEL
);
290 if ((ret
= m9206_read(udev
, M9206_FILTER
, 0x0, 0x8000, read
, 4)) != 0)
292 deb_rc("%x %x %x %x\n", read
[0], read
[1], read
[2], read
[3]);
294 if ((ret
= m9206_read(udev
, M9206_FW
, 0x0, 0x0, read
, 1)) != 0)
296 deb_rc("%x\n", read
[0]);
298 for (pass
= 0; pass
< 2; pass
++) {
299 for (i
= 0; i
+ (sizeof(u16
) * 3) < fw
->size
;) {
300 value
= le16_to_cpu(*(u16
*)(fw
->data
+ i
));
303 index
= le16_to_cpu(*(u16
*)(fw
->data
+ i
));
306 size
= le16_to_cpu(*(u16
*)(fw
->data
+ i
));
310 /* Will stall if using fw->data ... */
311 memcpy(buff
, fw
->data
+ i
, size
);
313 ret
= usb_control_msg(udev
, usb_sndctrlpipe(udev
,0),
315 USB_TYPE_VENDOR
| USB_DIR_OUT
,
316 value
, index
, buff
, size
, 20);
318 deb_rc("error while uploading fw!\n");
334 /* m9206 will disconnect itself from the bus after this. */
335 (void) m9206_write(udev
, M9206_CORE
, 0x01, M9206_FW_GO
);
336 deb_rc("firmware uploaded!\n");
344 /* Callbacks for DVB USB */
345 static int megasky_identify_state(struct usb_device
*udev
,
346 struct dvb_usb_device_properties
*props
,
347 struct dvb_usb_device_description
**desc
,
350 struct usb_host_interface
*alt
;
352 alt
= usb_altnum_to_altsetting(usb_ifnum_to_if(udev
, 0), 1);
353 *cold
= (alt
== NULL
) ? 1 : 0;
358 static int megasky_mt352_demod_init(struct dvb_frontend
*fe
)
360 u8 config
[] = { CONFIG
, 0x3d };
361 u8 clock
[] = { CLOCK_CTL
, 0x30 };
362 u8 reset
[] = { RESET
, 0x80 };
363 u8 adc_ctl
[] = { ADC_CTL_1
, 0x40 };
364 u8 agc
[] = { AGC_TARGET
, 0x1c, 0x20 };
365 u8 sec_agc
[] = { 0x69, 0x00, 0xff, 0xff, 0x40, 0xff, 0x00, 0x40, 0x40 };
366 u8 unk1
[] = { 0x93, 0x1a };
367 u8 unk2
[] = { 0xb5, 0x7a };
369 mt352_write(fe
, config
, ARRAY_SIZE(config
));
370 mt352_write(fe
, clock
, ARRAY_SIZE(clock
));
371 mt352_write(fe
, reset
, ARRAY_SIZE(reset
));
372 mt352_write(fe
, adc_ctl
, ARRAY_SIZE(adc_ctl
));
373 mt352_write(fe
, agc
, ARRAY_SIZE(agc
));
374 mt352_write(fe
, sec_agc
, ARRAY_SIZE(sec_agc
));
375 mt352_write(fe
, unk1
, ARRAY_SIZE(unk1
));
376 mt352_write(fe
, unk2
, ARRAY_SIZE(unk2
));
378 deb_rc("Demod init!\n");
383 static struct mt352_config megasky_mt352_config
= {
384 .demod_address
= 0x1e,
386 .demod_init
= megasky_mt352_demod_init
,
389 static int megasky_mt352_frontend_attach(struct dvb_usb_adapter
*adap
)
391 struct m9206_state
*m
= adap
->dev
->priv
;
393 deb_rc("megasky_frontend_attach!\n");
395 m
->i2c_r
[M9206_I2C_DEMOD
].addr
= megasky_mt352_config
.demod_address
;
396 m
->i2c_r
[M9206_I2C_DEMOD
].magic
= 0x1f;
398 if ((adap
->fe
= dvb_attach(mt352_attach
, &megasky_mt352_config
, &adap
->dev
->i2c_adap
)) == NULL
)
404 static struct qt1010_config megasky_qt1010_config
= {
408 static int megasky_qt1010_tuner_attach(struct dvb_usb_adapter
*adap
)
410 struct m9206_state
*m
= adap
->dev
->priv
;
412 m
->i2c_r
[M9206_I2C_TUNER
].addr
= megasky_qt1010_config
.i2c_address
;
413 m
->i2c_r
[M9206_I2C_TUNER
].magic
= 0xc5;
415 if (dvb_attach(qt1010_attach
, adap
->fe
, &adap
->dev
->i2c_adap
,
416 &megasky_qt1010_config
) == NULL
)
422 /* DVB USB Driver stuff */
423 static struct dvb_usb_device_properties megasky_properties
;
425 static int m920x_probe(struct usb_interface
*intf
,
426 const struct usb_device_id
*id
)
428 struct dvb_usb_device
*d
;
429 struct usb_host_interface
*alt
;
432 if ((ret
= dvb_usb_device_init(intf
, &megasky_properties
, THIS_MODULE
, &d
)) == 0) {
435 alt
= usb_altnum_to_altsetting(intf
, 1);
437 deb_rc("not alt found!\n");
441 ret
= usb_set_interface(d
->udev
, alt
->desc
.bInterfaceNumber
,
442 alt
->desc
.bAlternateSetting
);
446 deb_rc("Changed to alternate setting!\n");
448 if ((ret
= m9206_rc_init(d
->udev
)) != 0)
454 static struct usb_device_id m920x_table
[] = {
455 { USB_DEVICE(USB_VID_MSI
, USB_PID_MSI_MEGASKY580
) },
456 { } /* Terminating entry */
458 MODULE_DEVICE_TABLE (usb
, m920x_table
);
460 static struct dvb_usb_device_properties megasky_properties
= {
461 .caps
= DVB_USB_IS_AN_I2C_ADAPTER
,
463 .usb_ctrl
= DEVICE_SPECIFIC
,
464 .firmware
= "dvb-usb-megasky-02.fw",
465 .download_firmware
= m9206_firmware_download
,
468 .rc_key_map
= megasky_rc_keys
,
469 .rc_key_map_size
= ARRAY_SIZE(megasky_rc_keys
),
470 .rc_query
= m9206_rc_query
,
472 .size_of_priv
= sizeof(struct m9206_state
),
474 .identify_state
= megasky_identify_state
,
477 .caps
= DVB_USB_ADAP_HAS_PID_FILTER
|
478 DVB_USB_ADAP_PID_FILTER_CAN_BE_TURNED_OFF
,
480 .pid_filter_count
= 8,
481 .pid_filter
= m9206_pid_filter
,
482 .pid_filter_ctrl
= m9206_pid_filter_ctrl
,
484 .frontend_attach
= megasky_mt352_frontend_attach
,
485 .tuner_attach
= megasky_qt1010_tuner_attach
,
498 .i2c_algo
= &m9206_i2c_algo
,
500 .num_device_descs
= 1,
502 { "MSI Mega Sky 580 DVB-T USB2.0",
503 { &m920x_table
[0], NULL
},
509 static struct usb_driver m920x_driver
= {
510 .name
= "dvb_usb_m920x",
511 .probe
= m920x_probe
,
512 .disconnect
= dvb_usb_device_exit
,
513 .id_table
= m920x_table
,
517 static int __init
m920x_module_init(void)
521 if ((ret
= usb_register(&m920x_driver
))) {
522 err("usb_register failed. Error number %d", ret
);
529 static void __exit
m920x_module_exit(void)
531 /* deregister this driver from the USB subsystem */
532 usb_deregister(&m920x_driver
);
535 module_init (m920x_module_init
);
536 module_exit (m920x_module_exit
);
538 MODULE_AUTHOR("Aapo Tahkola <aet@rasterburn.org>");
539 MODULE_DESCRIPTION("Driver MSI Mega Sky 580 DVB-T USB2.0 / Uli m920x");
540 MODULE_VERSION("0.1");
541 MODULE_LICENSE("GPL");