diff options
-rw-r--r-- | linux/drivers/media/dvb/ttusb-dec/Kconfig | 30 | ||||
-rw-r--r-- | linux/drivers/media/dvb/ttusb-dec/Makefile | 11 | ||||
-rw-r--r-- | linux/drivers/media/dvb/ttusb-dec/ttusb_dec.c | 269 |
3 files changed, 250 insertions, 60 deletions
diff --git a/linux/drivers/media/dvb/ttusb-dec/Kconfig b/linux/drivers/media/dvb/ttusb-dec/Kconfig index afcaf2923..5b17e103f 100644 --- a/linux/drivers/media/dvb/ttusb-dec/Kconfig +++ b/linux/drivers/media/dvb/ttusb-dec/Kconfig @@ -1,24 +1,34 @@ config DVB_TTUSB_DEC - tristate "Technotrend/Hauppauge USB DEC2000-T devices" + tristate "Technotrend/Hauppauge USB DEC devices" depends on DVB_CORE && USB && !STANDALONE help Support for external USB adapters designed by Technotrend and - produced by Hauppauge, shipped under the brand name 'DEC2000-T'. + produced by Hauppauge, shipped under the brand name 'DEC2000-t' + and 'DEC3000-s'. Even if these devices have a MPEG decoder built in, they transmit only compressed MPEG data over the USB bus, so you need an external software decoder to watch TV on your computer. + The DEC devices require firmware in order to boot into a mode in + which they are slaves to the PC. These are required at compile + time. The firmware can be obtained and put into the default + locations as follows: + + wget http://hauppauge.lightpath.net/de/dec215a.exe + unzip -j dec215a.exe Software/Oem/STB/App/Boot/STB_PC_T.bin + mv STB_PC_T.bin /etc/dvb/dec2000t.bin + unzip -j dec215a.exe Software/Oem/STB/App/Boot/STB_PC_S.bin + mv STB_PC_S.bin /etc/dvb/dec3000s.bin + Say Y if you own such a device and want to use it. -config DVB_TTUSB_DEC_FIRMWARE_FILE +config DVB_TTUSB_DEC2000T_FIRMWARE_FILE string "Full pathname of dec2000t.bin firmware file" depends on DVB_TTUSB_DEC default "/etc/dvb/dec2000t.bin" - help - The DEC2000-T requires a firmware in order to boot into a mode in - which it is a slave to the PC. The firmware file can obtained as - follows: - wget http://hauppauge.lightpath.net/de/dec215a.exe - unzip -j dec215a.exe Software/Oem/STB/App/Boot/STB_PC_T.bin - mv STB_PC_T.bin /etc/dvb/dec2000t.bin + +config DVB_TTUSB_DEC3000S_FIRMWARE_FILE + string "Full pathname of dec3000s.bin firmware file" + depends on DVB_TTUSB_DEC + default "/etc/dvb/dec3000s.bin" diff --git a/linux/drivers/media/dvb/ttusb-dec/Makefile b/linux/drivers/media/dvb/ttusb-dec/Makefile index daefade04..7e404d618 100644 --- a/linux/drivers/media/dvb/ttusb-dec/Makefile +++ b/linux/drivers/media/dvb/ttusb-dec/Makefile @@ -1,11 +1,14 @@ -obj-$(CONFIG_DVB_TTUSB_DEC) += ttusb_dec.o dec2000_frontend.o +obj-$(CONFIG_DVB_TTUSB_DEC) += ttusb_dec.o EXTRA_CFLAGS = -Idrivers/media/dvb/dvb-core/ host-progs := fdump -$(obj)/ttusb_dec.o: $(obj)/dsp_dec2000.h +$(obj)/ttusb_dec.o: $(obj)/dsp_dec2000t.h $(obj)/dsp_dec3000s.h -$(obj)/dsp_dec2000.h: $(patsubst "%", %, $(CONFIG_DVB_TTUSB_DEC_FIRMWARE_FILE)) $(obj)/fdump - $(obj)/fdump $< dsp_dec2000 > $@ +$(obj)/dsp_dec2000t.h: $(patsubst "%", %, $(CONFIG_DVB_TTUSB_DEC2000T_FIRMWARE_FILE)) $(obj)/fdump + $(obj)/fdump $< dsp_dec2000t > $@ + +$(obj)/dsp_dec3000s.h: $(patsubst "%", %, $(CONFIG_DVB_TTUSB_DEC3000S_FIRMWARE_FILE)) $(obj)/fdump + $(obj)/fdump $< dsp_dec3000s > $@ diff --git a/linux/drivers/media/dvb/ttusb-dec/ttusb_dec.c b/linux/drivers/media/dvb/ttusb-dec/ttusb_dec.c index 57fbc2d1f..1cc610c83 100644 --- a/linux/drivers/media/dvb/ttusb-dec/ttusb_dec.c +++ b/linux/drivers/media/dvb/ttusb-dec/ttusb_dec.c @@ -35,7 +35,6 @@ #include "dvb_filter.h" #include "dvb_frontend.h" #include "dvb_net.h" -#include "dvb_usb_compat.h" static int debug = 0; @@ -56,14 +55,24 @@ static int debug = 0; #define MAX_AV_PES_LENGTH 6144 +enum ttusb_model { + TTUSB_DEC2000T, + TTUSB_DEC3000S +}; + struct ttusb_dec { + enum ttusb_model model; + char *model_name; + /* DVB bits */ - struct dvb_adapter *adapter; - struct dmxdev dmxdev; - struct dvb_demux demux; - struct dmx_frontend frontend; - struct dvb_i2c_bus i2c_bus; - struct dvb_net dvb_net; + struct dvb_adapter *adapter; + struct dmxdev dmxdev; + struct dvb_demux demux; + struct dmx_frontend frontend; + struct dvb_i2c_bus i2c_bus; + struct dvb_net dvb_net; + struct dvb_frontend_info *frontend_info; + int (*frontend_ioctl) (struct dvb_frontend *, unsigned int, void *); u16 pid[DMX_PES_OTHER]; @@ -105,8 +114,8 @@ struct urb_frame { struct list_head urb_frame_list; }; -static struct dvb_frontend_info dec2000_frontend_info = { - .name = "TechnoTrend/Hauppauge DEC-2000-t Frontend", +static struct dvb_frontend_info dec2000t_frontend_info = { + .name = "TechnoTrend/Hauppauge DEC2000-t Frontend", .type = FE_OFDM, .frequency_min = 51000000, .frequency_max = 858000000, @@ -118,6 +127,19 @@ static struct dvb_frontend_info dec2000_frontend_info = { FE_CAN_HIERARCHY_AUTO, }; +static struct dvb_frontend_info dec3000s_frontend_info = { + .name = "TechnoTrend/Hauppauge DEC3000-s Frontend", + .type = FE_QPSK, + .frequency_min = 950000, + .frequency_max = 2150000, + .frequency_stepsize = 125, + .caps = FE_CAN_FEC_1_2 | FE_CAN_FEC_2_3 | FE_CAN_FEC_3_4 | + FE_CAN_FEC_5_6 | FE_CAN_FEC_7_8 | FE_CAN_FEC_AUTO | + FE_CAN_QAM_16 | FE_CAN_QAM_64 | FE_CAN_QAM_AUTO | + FE_CAN_TRANSMISSION_MODE_AUTO | FE_CAN_GUARD_INTERVAL_AUTO | + FE_CAN_HIERARCHY_AUTO, +}; + static int ttusb_dec_send_command(struct ttusb_dec *dec, const u8 command, int param_length, const u8 params[], int *result_length, u8 cmd_result[]) @@ -449,7 +471,11 @@ static void ttusb_dec_process_urb(struct urb *urb, struct pt_regs *ptregs) int i; for (i = 0; i < FRAMES_PER_ISO_BUF; i++) { +#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0) + struct iso_packet_descriptor *d; +#else struct usb_iso_packet_descriptor *d; +#endif u8 *b; int length; struct urb_frame *frame; @@ -484,7 +510,7 @@ static void ttusb_dec_process_urb(struct urb *urb, struct pt_regs *ptregs) #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0) if (dec->iso_stream_count) - usb_submit_urb(urb URB_MEM_FLAG); + usb_submit_urb(urb, GFP_ATOMIC); #endif } @@ -502,9 +528,11 @@ static void ttusb_dec_setup_urbs(struct ttusb_dec *dec) urb->context = dec; urb->complete = ttusb_dec_process_urb; urb->pipe = dec->stream_pipe; - urb->transfer_flags = URB_ISO_ASAP; #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0) + urb->transfer_flags = URB_ISO_ASAP; urb->interval = 1; +#else + urb->transfer_flags = USB_ISO_ASAP; #endif urb->number_of_packets = FRAMES_PER_ISO_BUF; urb->transfer_buffer_length = ISO_FRAME_SIZE * @@ -572,8 +600,12 @@ static int ttusb_dec_start_iso_xfer(struct ttusb_dec *dec) ttusb_dec_setup_urbs(dec); for (i = 0; i < ISO_BUF_COUNT; i++) { - if ((result = usb_submit_urb(dec->iso_urb[i] - URB_MEM_FLAG))) { +#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0) + if ((result = usb_submit_urb(dec->iso_urb[i], + GFP_ATOMIC))) { +#else + if ((result = usb_submit_urb(dec->iso_urb[i]))) { +#endif printk("%s: failed urb submission %d: " "error %d\n", __FUNCTION__, i, result); @@ -729,7 +761,11 @@ static int ttusb_dec_alloc_iso_urbs(struct ttusb_dec *dec) for (i = 0; i < ISO_BUF_COUNT; i++) { struct urb *urb; - if (!(urb = usb_alloc_urb(FRAMES_PER_ISO_BUF URB_MEM_FLAG))) { +#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0) + if (!(urb = usb_alloc_urb(FRAMES_PER_ISO_BUF, GFP_ATOMIC))) { +#else + if (!(urb = usb_alloc_urb(FRAMES_PER_ISO_BUF))) { +#endif ttusb_dec_free_iso_urbs(dec); return -ENOMEM; } @@ -781,20 +817,42 @@ static void ttusb_dec_init_usb(struct ttusb_dec *dec) ttusb_dec_alloc_iso_urbs(dec); } -#include "dsp_dec2000.h" +#include "dsp_dec2000t.h" +#include "dsp_dec3000s.h" static int ttusb_dec_boot_dsp(struct ttusb_dec *dec) { int i, j, actual_len, result, size, trans_count; - u8 b0[] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1b, 0xc8, 0x61, + u8 b0[] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; u8 b1[] = { 0x61 }; u8 b[ARM_PACKET_SIZE]; - u32 dsp_length = htonl(sizeof(dsp_dec2000)); + u8 *firmware = NULL; + size_t firmware_size = 0; + u32 firmware_csum = 0; + u32 firmware_size_nl; + u32 firmware_csum_nl; dprintk("%s\n", __FUNCTION__); - memcpy(b0, &dsp_length, 4); + switch (dec->model) { + case TTUSB_DEC2000T: + firmware = &dsp_dec2000t[0]; + firmware_size = sizeof(dsp_dec2000t); + firmware_csum = 0x1bc86100; + break; + + case TTUSB_DEC3000S: + firmware = &dsp_dec3000s[0]; + firmware_size = sizeof(dsp_dec3000s); + firmware_csum = 0x00000000; + break; + } + + firmware_size_nl = htonl(firmware_size); + memcpy(b0, &firmware_size_nl, 4); + firmware_csum_nl = htonl(firmware_csum); + memcpy(&b0[6], &firmware_csum_nl, 4); result = ttusb_dec_send_command(dec, 0x41, sizeof(b0), b0, NULL, NULL); @@ -804,8 +862,8 @@ static int ttusb_dec_boot_dsp(struct ttusb_dec *dec) trans_count = 0; j = 0; - for (i = 0; i < sizeof(dsp_dec2000); i += COMMAND_PACKET_SIZE) { - size = sizeof(dsp_dec2000) - i; + for (i = 0; i < firmware_size; i += COMMAND_PACKET_SIZE) { + size = firmware_size - i; if (size > COMMAND_PACKET_SIZE) size = COMMAND_PACKET_SIZE; @@ -813,7 +871,7 @@ static int ttusb_dec_boot_dsp(struct ttusb_dec *dec) b[j + 1] = trans_count++; b[j + 2] = 0xf0; b[j + 3] = size; - memcpy(&b[j + 4], &dsp_dec2000[i], size); + memcpy(&b[j + 4], &firmware[i], size); j += COMMAND_PACKET_SIZE + 4; @@ -855,7 +913,7 @@ static int ttusb_dec_init_dvb(struct ttusb_dec *dec) dprintk("%s\n", __FUNCTION__); - if ((result = dvb_register_adapter(&dec->adapter, "dec2000")) < 0) { + if ((result = dvb_register_adapter(&dec->adapter, dec->model_name)) < 0) { printk("%s: dvb_register_adapter failed: error %d\n", __FUNCTION__, result); @@ -966,7 +1024,7 @@ static void ttusb_dec_exit_tasklet(struct ttusb_dec *dec) } } -static int dec2000_frontend_ioctl(struct dvb_frontend *fe, unsigned int cmd, +static int ttusb_dec_2000t_frontend_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg) { struct ttusb_dec *dec = fe->data; @@ -977,7 +1035,7 @@ static int dec2000_frontend_ioctl(struct dvb_frontend *fe, unsigned int cmd, case FE_GET_INFO: dprintk("%s: FE_GET_INFO\n", __FUNCTION__); - memcpy(arg, &dec2000_frontend_info, + memcpy(arg, dec->frontend_info, sizeof (struct dvb_frontend_info)); break; @@ -1032,8 +1090,112 @@ static int dec2000_frontend_ioctl(struct dvb_frontend *fe, unsigned int cmd, dprintk(" inversion->%d\n", p->inversion); freq = htonl(p->frequency / 1000); - memcpy(&b[4], &freq, sizeof (int)); - ttusb_dec_send_command(dec, 0x71, 20, b, NULL, NULL); + memcpy(&b[4], &freq, sizeof (u32)); + ttusb_dec_send_command(dec, 0x71, sizeof(b), b, NULL, NULL); + + break; + } + + case FE_GET_FRONTEND: + dprintk("%s: FE_GET_FRONTEND\n", __FUNCTION__); + break; + + case FE_SLEEP: + dprintk("%s: FE_SLEEP\n", __FUNCTION__); + return -ENOSYS; + break; + + case FE_INIT: + dprintk("%s: FE_INIT\n", __FUNCTION__); + break; + + case FE_RESET: + dprintk("%s: FE_RESET\n", __FUNCTION__); + break; + + default: + dprintk("%s: unknown IOCTL (0x%X)\n", __FUNCTION__, cmd); + return -EINVAL; + + } + + return 0; +} + +static int ttusb_dec_3000s_frontend_ioctl(struct dvb_frontend *fe, unsigned int cmd, + void *arg) +{ + struct ttusb_dec *dec = fe->data; + + dprintk("%s\n", __FUNCTION__); + + switch (cmd) { + + case FE_GET_INFO: + dprintk("%s: FE_GET_INFO\n", __FUNCTION__); + memcpy(arg, dec->frontend_info, + sizeof (struct dvb_frontend_info)); + break; + + case FE_READ_STATUS: { + fe_status_t *status = (fe_status_t *)arg; + dprintk("%s: FE_READ_STATUS\n", __FUNCTION__); + *status = FE_HAS_SIGNAL | FE_HAS_VITERBI | + FE_HAS_SYNC | FE_HAS_CARRIER | FE_HAS_LOCK; + break; + } + + case FE_READ_BER: { + u32 *ber = (u32 *)arg; + dprintk("%s: FE_READ_BER\n", __FUNCTION__); + *ber = 0; + return -ENOSYS; + break; + } + + case FE_READ_SIGNAL_STRENGTH: { + dprintk("%s: FE_READ_SIGNAL_STRENGTH\n", __FUNCTION__); + *(s32 *)arg = 0xFF; + return -ENOSYS; + break; + } + + case FE_READ_SNR: + dprintk("%s: FE_READ_SNR\n", __FUNCTION__); + *(s32 *)arg = 0; + return -ENOSYS; + break; + + case FE_READ_UNCORRECTED_BLOCKS: + dprintk("%s: FE_READ_UNCORRECTED_BLOCKS\n", __FUNCTION__); + *(u32 *)arg = 0; + return -ENOSYS; + break; + + case FE_SET_FRONTEND:{ + struct dvb_frontend_parameters *p = + (struct dvb_frontend_parameters *)arg; + u8 b[] = { 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x0d, 0x00, 0x00, 0x00, + 0x00 }; + u32 freq; + u32 sym_rate; + + dprintk("%s: FE_SET_FRONTEND\n", __FUNCTION__); + + dprintk(" frequency->%d\n", p->frequency); + dprintk(" symbol_rate->%d\n", + p->u.qam.symbol_rate); + dprintk(" inversion->%d\n", p->inversion); + + freq = htonl(p->frequency / 1000); + memcpy(&b[4], &freq, sizeof(u32)); + sym_rate = htonl(p->u.qam.symbol_rate); + memcpy(&b[12], &sym_rate, sizeof(u32)); + ttusb_dec_send_command(dec, 0x71, sizeof(b), b, NULL, NULL); break; } @@ -1067,13 +1229,26 @@ static int dec2000_frontend_ioctl(struct dvb_frontend *fe, unsigned int cmd, static void ttusb_dec_init_frontend(struct ttusb_dec *dec) { dec->i2c_bus.adapter = dec->adapter; - dvb_register_frontend(dec2000_frontend_ioctl, &dec->i2c_bus, - (void *)dec, &dec2000_frontend_info); + + switch (dec->model) { + case TTUSB_DEC2000T: + dec->frontend_info = &dec2000t_frontend_info; + dec->frontend_ioctl = ttusb_dec_2000t_frontend_ioctl; + break; + + case TTUSB_DEC3000S: + dec->frontend_info = &dec3000s_frontend_info; + dec->frontend_ioctl = ttusb_dec_3000s_frontend_ioctl; + break; + } + + dvb_register_frontend(dec->frontend_ioctl, &dec->i2c_bus, (void *)dec, + dec->frontend_info); } static void ttusb_dec_exit_frontend(struct ttusb_dec *dec) { - dvb_unregister_frontend(dec2000_frontend_ioctl, &dec->i2c_bus); + dvb_unregister_frontend(dec->frontend_ioctl, &dec->i2c_bus); } #if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0) @@ -1091,20 +1266,6 @@ static void *ttusb_dec_probe(struct usb_device *udev, unsigned int ifnum, printk("%s: couldn't allocate memory.\n", __FUNCTION__); return NULL; } - - memset(dec, 0, sizeof(struct ttusb_dec)); - - dec->udev = udev; - - ttusb_dec_init_usb(dec); - ttusb_dec_init_stb(dec); - ttusb_dec_init_dvb(dec); - ttusb_dec_init_frontend(dec); - ttusb_dec_init_v_pes(dec); - ttusb_dec_init_tasklet(dec); - - return (void *)dec; -} #else static int ttusb_dec_probe(struct usb_interface *intf, const struct usb_device_id *id) @@ -1120,9 +1281,22 @@ static int ttusb_dec_probe(struct usb_interface *intf, printk("%s: couldn't allocate memory.\n", __FUNCTION__); return -ENOMEM; } +#endif memset(dec, 0, sizeof(struct ttusb_dec)); + switch (id->idProduct) { + case 0x1007: + dec->model = TTUSB_DEC3000S; + dec->model_name = "DEC3000-s"; + break; + + case 0x1008: + dec->model = TTUSB_DEC2000T; + dec->model_name = "DEC2000-t"; + break; + } + dec->udev = udev; ttusb_dec_init_usb(dec); @@ -1132,12 +1306,15 @@ static int ttusb_dec_probe(struct usb_interface *intf, ttusb_dec_init_v_pes(dec); ttusb_dec_init_tasklet(dec); +#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0) + return (void *)dec; +#else usb_set_intfdata(intf, (void *)dec); ttusb_dec_set_streaming_interface(dec); return 0; -} #endif +} #if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0) static void ttusb_dec_disconnect(struct usb_device *udev, void *data) @@ -1162,7 +1339,7 @@ static void ttusb_dec_disconnect(struct usb_interface *intf) } static struct usb_device_id ttusb_dec_table[] = { - {USB_DEVICE(0x0b48, 0x1006)}, /* Unconfirmed */ + /*{USB_DEVICE(0x0b48, 0x1006)}, Unconfirmed */ {USB_DEVICE(0x0b48, 0x1007)}, /* DEC3000-s */ {USB_DEVICE(0x0b48, 0x1008)}, /* DEC2000-t */ {} @@ -1172,7 +1349,7 @@ static struct usb_driver ttusb_dec_driver = { .name = DRIVER_NAME, .probe = ttusb_dec_probe, .disconnect = ttusb_dec_disconnect, - .id_table = ttusb_dec_table, + .id_table = ttusb_dec_table, }; static int __init ttusb_dec_init(void) |