Message ID | 268161120906161305u24330118g2f9e35e65eecd04@mail.gmail.com (mailing list archive) |
---|---|
State | RFC |
Headers | show |
On Wed, Jun 17, 2009 at 12:05 AM, Edouard Lafargue<edouard@lafargue.name> wrote: > Hello Alexey, Hello Edouard > Many thanks for your comments. I have cleaned up my code/patch along > your guidelines, removed all the remaining code that was not used > anymore, checked that all buffers get deallocated properly - I hope. It looks better, thanks. > Review by Tobias is certainly necessary, he know his own driver best, > I might have broken things without knowing... Anyway it works fine on > the two 32 and 64bit machines where I have tested the dongle, both on > kernel 2.6.28. Review and testing are definitely necessary, of course. Let's wait his comments. > The main and very big improvement is the performance of RDS bitstream > reception, now Radiotext appears almost right away even on the most > difficult stations I can receive here in Paris, just like it should. > Performance is similar to 'real' radios, if you see what I mean. Audio > remains perfectly clear. Good to know that. I noticed one more possible thing, please see comment below. > When it comes to audio, it seems to me the "si470x_get_report" call > generates some serious clicking in the audio on my Silabs dongle. I > will investigate further, I wonder if other USB FM tokens have the > same issue. I fixed this problem on rdsd by adding a configuration > file option to prevent it constantly polling the dongle for its > current frequency... > > Comments are welcome! > > Best regards, > > Signed-off-by: Edouard Lafargue <edouard@lafargue.name> > > diff -Nur ../radio-si470x.c radio-si470x.c > --- ../radio-si470x.c  2008-12-25 00:26:37.000000000 +0100 > +++ radio-si470x.c    2009-06-16 21:47:58.000000000 +0200 > @@ -97,9 +97,13 @@ >  *       - add support for KWorld USB FM Radio FM700 >  *       - blacklisted KWorld radio in hid-core.c and hid-ids.h >  * > + * 2009-06-16  Edouard Lafargue <edouard@lafargue.name> > + *       Version 1.0.9 > + *       - add support for interrupt mode for RDS endpoint, instead of polling. > + *         Improves RDS reception significantly Hmmm. Did you make this patch against 2.6.28 kernel sources? Current version of driver is 1.0.9 already :) http://linuxtv.org/hg/v4l-dvb/file/45f342431f6e/linux/drivers/media/radio/ So, your change will make 1.1.0, right? Usually developers applied patches to the latest version of code. If you want to work/make patch with the latest version of v4l-dvb tree you can use v4l-dvb mercurial tree. Link that can help: http://linuxtv.org/wiki/index.php/How_to_Obtain,_Build_and_Install_V4L-DVB_Device_Drivers Actually, steps in general: 1) install mercurial 2) clone current v4l-dvb tree: hg clone http://linuxtv.org/hg/v4l-dvb 3) Add your changes in radio-si470x driver. 4) Command "hg diff" will show changes. You can save this changes and sent it to maillist. 5) Don't forget to run "make checkpatch" and "make whitespaces". It will check CodingStyle of your changes. 6) You can use commands like: make menuconfig, make, make install to select, compile and install v4l modules. This way you can update your patch and test it with latest si470x driver. Other way - you can wait Tobias comments and don't play with v4l-dvb tree. > + * >  * ToDo: >  * - add firmware download/update support > - * - RDS support: interrupt mode, instead of polling >  * - add LED status output (check if that's not already done in firmware) >  */ > > @@ -107,10 +111,10 @@ >  /* driver definitions */ >  #define DRIVER_AUTHOR "Tobias Lorenz <tobias.lorenz@gmx.net>" >  #define DRIVER_NAME "radio-si470x" > -#define DRIVER_KERNEL_VERSION KERNEL_VERSION(1, 0, 8) > +#define DRIVER_KERNEL_VERSION KERNEL_VERSION(1, 0, 9) >  #define DRIVER_CARD "Silicon Labs Si470x FM Radio Receiver" >  #define DRIVER_DESC "USB radio driver for Si470x FM Radio Receivers" > -#define DRIVER_VERSION "1.0.8" > +#define DRIVER_VERSION "1.0.9" > > >  /* kernel includes */ > @@ -206,16 +210,6 @@ >  module_param(max_rds_errors, ushort, 0644); >  MODULE_PARM_DESC(max_rds_errors, "RDS maximum block errors: *1*"); > > -/* RDS poll frequency */ > -static unsigned int rds_poll_time = 40; > -/* 40 is used by the original USBRadio.exe */ > -/* 50 is used by radio-cadet */ > -/* 75 should be okay */ > -/* 80 is the usual RDS receive interval */ > -module_param(rds_poll_time, uint, 0644); > -MODULE_PARM_DESC(rds_poll_time, "RDS poll time (ms): *40*"); > - > - > >  /************************************************************************** >  * Register Definitions > @@ -440,6 +434,12 @@ >     struct usb_interface *intf; >     struct video_device *videodev; > > +    /* Interrupt endpoint handling */ > +    char *int_in_buffer; > +    struct usb_endpoint_descriptor *int_in_endpoint; > +    struct urb *int_in_urb; > +    int int_in_running; > + >     /* driver management */ >     unsigned int users; >     unsigned char disconnected; > @@ -449,7 +449,6 @@ >     unsigned short registers[RADIO_REGISTER_NUM]; > >     /* RDS receive buffer */ > -    struct delayed_work work; >     wait_queue_head_t read_queue; >     struct mutex lock;        /* buffer locking */ >     unsigned char *buffer;      /* size is always multiple of three */ > @@ -578,37 +577,6 @@ >  } > > > -/* > - * si470x_get_rds_registers - read rds registers > - */ > -static int si470x_get_rds_registers(struct si470x_device *radio) > -{ > -    unsigned char buf[RDS_REPORT_SIZE]; > -    int retval; > -    int size; > -    unsigned char regnr; > - > -    buf[0] = RDS_REPORT; > - > -    retval = usb_interrupt_msg(radio->usbdev, > -        usb_rcvintpipe(radio->usbdev, 1), > -        (void *) &buf, sizeof(buf), &size, usb_timeout); > -    if (size != sizeof(buf)) > -        printk(KERN_WARNING DRIVER_NAME ": si470x_get_rds_registers: " > -            "return size differs: %d != %zu\n", size, sizeof(buf)); > -    if (retval < 0) > -        printk(KERN_WARNING DRIVER_NAME ": si470x_get_rds_registers: " > -            "usb_interrupt_msg returned %d\n", retval); > - > -    if (retval >= 0) > -        for (regnr = 0; regnr < RDS_REGISTER_NUM; regnr++) > -            radio->registers[STATUSRSSI + regnr] = > -                get_unaligned_be16( > -                &buf[regnr * RADIO_REGISTER_SIZE + 1]); > - > -    return (retval < 0) ? -EINVAL : 0; > -} > - > >  /* >  * si470x_set_chan - set the channel > @@ -882,105 +850,117 @@ >  **************************************************************************/ > >  /* > - * si470x_rds - rds processing function > + * si470x_int_in_callback - rds callback and processing function > + * > + * TODO: do we need to use mutex locks in some sections? >  */ > -static void si470x_rds(struct si470x_device *radio) > +static void si470x_int_in_callback(struct urb *urb) >  { > +    struct si470x_device *radio = urb->context; > +    unsigned char buf[RDS_REPORT_SIZE]; > +    int retval; > +    unsigned char regnr; >     unsigned char blocknum; >     unsigned short bler; /* rds block errors */ >     unsigned short rds; >     unsigned char tmpbuf[3]; > > -    /* get rds blocks */ > -    if (si470x_get_rds_registers(radio) < 0) > -        return; > -    if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSR) == 0) { > -        /* No RDS group ready */ > -        return; > -    } > -    if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSS) == 0) { > -        /* RDS decoder not synchronized */ > -        return; > -    } > - > -    /* copy all four RDS blocks to internal buffer */ > -    mutex_lock(&radio->lock); > -    for (blocknum = 0; blocknum < 4; blocknum++) { > -        switch (blocknum) { > -        default: > -            bler = (radio->registers[STATUSRSSI] & > -                    STATUSRSSI_BLERA) >> 9; > -            rds = radio->registers[RDSA]; > -            break; > -        case 1: > -            bler = (radio->registers[READCHAN] & > -                    READCHAN_BLERB) >> 14; > -            rds = radio->registers[RDSB]; > -            break; > -        case 2: > -            bler = (radio->registers[READCHAN] & > -                    READCHAN_BLERC) >> 12; > -            rds = radio->registers[RDSC]; > -            break; > -        case 3: > -            bler = (radio->registers[READCHAN] & > -                    READCHAN_BLERD) >> 10; > -            rds = radio->registers[RDSD]; > -            break; > -        }; > - > -        /* Fill the V4L2 RDS buffer */ > -        put_unaligned_le16(rds, &tmpbuf); > -        tmpbuf[2] = blocknum;      /* offset name */ > -        tmpbuf[2] |= blocknum << 3;   /* received offset */ > -        if (bler > max_rds_errors) > -            tmpbuf[2] |= 0x80; /* uncorrectable errors */ > -        else if (bler > 0) > -            tmpbuf[2] |= 0x40; /* corrected error(s) */ > - > -        /* copy RDS block to internal buffer */ > -        memcpy(&radio->buffer[radio->wr_index], &tmpbuf, 3); > -        radio->wr_index += 3; > - > -        /* wrap write pointer */ > -        if (radio->wr_index >= radio->buf_size) > -            radio->wr_index = 0; > - > -        /* check for overflow */ > -        if (radio->wr_index == radio->rd_index) { > -            /* increment and wrap read pointer */ > -            radio->rd_index += 3; > -            if (radio->rd_index >= radio->buf_size) > -                radio->rd_index = 0; > +    if (urb->status) { > +        if (urb->status == -ENOENT || > +                urb->status == -ECONNRESET || > +                urb->status == -ESHUTDOWN) { > +            return; > +        } else { > +            printk(KERN_WARNING DRIVER_NAME ": non-zero urb status (%d)\n", > urb->status); > +            goto resubmit; /* Maybe we can recover. */ >         } >     } > -    mutex_unlock(&radio->lock); > - > -    /* wake up read queue */ > -    if (radio->wr_index != radio->rd_index) > -        wake_up_interruptible(&radio->read_queue); > -} > - > - > -/* > - * si470x_work - rds work function > - */ > -static void si470x_work(struct work_struct *work) > -{ > -    struct si470x_device *radio = container_of(work, struct si470x_device, > -        work.work); > >     /* safety checks */ >     if (radio->disconnected) >         return; >     if ((radio->registers[SYSCONFIG1] & SYSCONFIG1_RDS) == 0) > -        return; > - > -    si470x_rds(radio); > -    schedule_delayed_work(&radio->work, msecs_to_jiffies(rds_poll_time)); > -} > +        goto resubmit; > > +    if (urb->actual_length > 0) { > +        /* Update RDS registers with URB data */ > +        buf[0] = RDS_REPORT; > +        for (regnr = 0; regnr < RDS_REGISTER_NUM; regnr++) > +            radio->registers[STATUSRSSI + regnr] = > +                get_unaligned_be16( > +                &radio->int_in_buffer[regnr * RADIO_REGISTER_SIZE + 1]); > +        /* get rds blocks */ > +        if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSR) == 0) { > +            /* No RDS group ready, better luck next time */ > +            goto resubmit; > +        } > +        if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSS) == 0) { > +            /* RDS decoder not synchronized */ > +            goto resubmit; > +        } > +        for (blocknum = 0; blocknum < 4; blocknum++) { > +            switch (blocknum) { > +            default: > +                bler = (radio->registers[STATUSRSSI] & > +                        STATUSRSSI_BLERA) >> 9; > +                rds = radio->registers[RDSA]; > +                break; > +            case 1: > +                bler = (radio->registers[READCHAN] & > +                        READCHAN_BLERB) >> 14; > +                rds = radio->registers[RDSB]; > +                break; > +            case 2: > +                bler = (radio->registers[READCHAN] & > +                        READCHAN_BLERC) >> 12; > +                rds = radio->registers[RDSC]; > +                break; > +            case 3: > +                bler = (radio->registers[READCHAN] & > +                        READCHAN_BLERD) >> 10; > +                rds = radio->registers[RDSD]; > +                break; > +            }; > + > +            /* Fill the V4L2 RDS buffer */ > +            put_unaligned_le16(rds, &tmpbuf); > +            tmpbuf[2] = blocknum;      /* offset name */ > +            tmpbuf[2] |= blocknum << 3;   /* received offset */ > +            if (bler > max_rds_errors) > +                tmpbuf[2] |= 0x80; /* uncorrectable errors */ > +            else if (bler > 0) > +                tmpbuf[2] |= 0x40; /* corrected error(s) */ > + > +            /* copy RDS block to internal buffer */ > +            memcpy(&radio->buffer[radio->wr_index], &tmpbuf, 3); > +            radio->wr_index += 3; > + > +            /* wrap write pointer */ > +            if (radio->wr_index >= radio->buf_size) > +                radio->wr_index = 0; > + > +            /* check for overflow */ > +            if (radio->wr_index == radio->rd_index) { > +                /* increment and wrap read pointer */ > +                radio->rd_index += 3; > +                if (radio->rd_index >= radio->buf_size) > +                    radio->rd_index = 0; > +            } > +        } > +        if (radio->wr_index != radio->rd_index) > +            wake_up_interruptible(&radio->read_queue); > +    } > > +resubmit: > +    /* Resubmit if we're still running. */ > +    if (radio->int_in_running && radio->usbdev) { > +        retval = usb_submit_urb(radio->int_in_urb, GFP_ATOMIC); > +        if (retval) { > +            printk(KERN_WARNING DRIVER_NAME ": resubmitting urb failed (%d)", retval); > +            radio->int_in_running = 0; > +        } > +    } > +} > >  /************************************************************************** >  * File Operations Interface > @@ -999,8 +979,6 @@ >     /* switch on rds reception */ >     if ((radio->registers[SYSCONFIG1] & SYSCONFIG1_RDS) == 0) { >         si470x_rds_on(radio); > -        schedule_delayed_work(&radio->work, > -            msecs_to_jiffies(rds_poll_time)); >     } > >     /* block if no new data available */ > @@ -1059,8 +1037,6 @@ >     /* switch on rds reception */ >     if ((radio->registers[SYSCONFIG1] & SYSCONFIG1_RDS) == 0) { >         si470x_rds_on(radio); > -        schedule_delayed_work(&radio->work, > -            msecs_to_jiffies(rds_poll_time)); >     } > >     poll_wait(file, &radio->read_queue, pts); > @@ -1090,10 +1066,34 @@ >         goto done; >     } > > +    printk(KERN_INFO DRIVER_NAME ": Opened radio (users now: %i)\n",radio->users); > + >     if (radio->users == 1) { >         retval = si470x_start(radio); > -        if (retval < 0) > +        if (retval < 0) { >             usb_autopm_put_interface(radio->intf); > +            goto done; > +        } > + > +        /* Initialize interrupt URB. */ > +        usb_fill_int_urb(radio->int_in_urb, radio->usbdev, > +                usb_rcvintpipe(radio->usbdev, radio->int_in_endpoint->bEndpointAddress), > +                radio->int_in_buffer, > +                le16_to_cpu(radio->int_in_endpoint->wMaxPacketSize), > +                si470x_int_in_callback, > +                radio, > +                radio->int_in_endpoint->bInterval); > + > +        radio->int_in_running = 1; > +        mb(); > + > +        retval = usb_submit_urb(radio->int_in_urb, GFP_KERNEL); > +        if (retval) { > +            printk(KERN_INFO DRIVER_NAME ": submitting int urb failed (%d)\n", retval); > +            radio->int_in_running = 0; > +            usb_autopm_put_interface(radio->intf); > +            goto done; > +        } >     } > >  done: > @@ -1118,16 +1118,22 @@ > >     mutex_lock(&radio->disconnect_lock); >     radio->users--; > +    printk(KERN_INFO DRIVER_NAME ": Closed radio (remaining > users:%i)\n",radio->users); >     if (radio->users == 0) { >         if (radio->disconnected) { >             video_unregister_device(radio->videodev); > +            kfree(radio->int_in_buffer); >             kfree(radio->buffer); >             kfree(radio); >             goto unlock; >         } > > -        /* stop rds reception */ > -        cancel_delayed_work_sync(&radio->work); > +        /* Shutdown Interrupt handler */ > +        if (radio->int_in_running) { > +            radio->int_in_running = 0; > +        if (radio->int_in_urb) > +            usb_kill_urb(radio->int_in_urb); > +        } > >         /* cancel read processes */ >         wake_up_interruptible(&radio->read_queue); > @@ -1580,7 +1586,9 @@ >         const struct usb_device_id *id) >  { >     struct si470x_device *radio; > -    int retval = 0; > +    struct usb_host_interface *iface_desc; > +    struct usb_endpoint_descriptor *endpoint; > +    int i,int_end_size,retval = 0; > >     /* private data allocation and initialization */ >     radio = kzalloc(sizeof(struct si470x_device), GFP_KERNEL); > @@ -1595,11 +1603,45 @@ >     mutex_init(&radio->disconnect_lock); >     mutex_init(&radio->lock); > > +    iface_desc = intf->cur_altsetting; > + > +    /* Set up interrupt endpoint information. */ > +    for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) { > +        endpoint = &iface_desc->endpoint[i].desc; > + > +        if (((endpoint->bEndpointAddress & USB_ENDPOINT_DIR_MASK) == USB_DIR_IN) > +                && ((endpoint->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) == > +                    USB_ENDPOINT_XFER_INT)) > +            radio->int_in_endpoint = endpoint; > + > +    } > +    if (! radio->int_in_endpoint) { > +        printk(KERN_INFO DRIVER_NAME ": could not find interrupt in endpoint\n"); > +        retval = -EIO; > +        goto err_radio; > +    } > + > +    int_end_size = le16_to_cpu(radio->int_in_endpoint->wMaxPacketSize); > + > +    radio->int_in_buffer = kmalloc(int_end_size, GFP_KERNEL); > +    if (! radio->int_in_buffer) { > +        printk(KERN_INFO DRIVER_NAME "could not allocate int_in_buffer"); > +        retval = -ENOMEM; > +        goto err_radio; > +    } > + > +    radio->int_in_urb = usb_alloc_urb(0, GFP_KERNEL); > +    if (! radio->int_in_urb) { > +        printk(KERN_INFO DRIVER_NAME "could not allocate int_in_urb"); > +        retval = -ENOMEM; > +        goto err_intbuffer; > +    } > + >     /* video device allocation and initialization */ >     radio->videodev = video_device_alloc(); >     if (!radio->videodev) { >         retval = -ENOMEM; > -        goto err_radio; > +        goto err_intbuffer; >     } >     memcpy(radio->videodev, &si470x_viddev_template, >             sizeof(si470x_viddev_template)); > @@ -1608,7 +1650,7 @@ >     /* show some infos about the specific device */ >     if (si470x_get_all_registers(radio) < 0) { >         retval = -EIO; > -        goto err_all; > +        goto err_video; >     } >     printk(KERN_INFO DRIVER_NAME ": DeviceID=0x%4.4hx ChipID=0x%4.4hx\n", >             radio->registers[DEVICEID], radio->registers[CHIPID]); > @@ -1636,7 +1678,7 @@ >     radio->buf_size = rds_buf * 3; >     radio->buffer = kmalloc(radio->buf_size, GFP_KERNEL); >     if (!radio->buffer) { > -        retval = -EIO; > +        retval = -ENOMEM; This change looks like fixing mistake with error code. Probably (not necessarily) it's better to move this change in separate patch. >         goto err_all; >     } > > @@ -1645,9 +1687,6 @@ >     radio->rd_index = 0; >     init_waitqueue_head(&radio->read_queue); > > -    /* prepare rds work function */ > -    INIT_DELAYED_WORK(&radio->work, si470x_work); > - >     /* register video device */ >     retval = video_register_device(radio->videodev, VFL_TYPE_RADIO, radio_nr); >     if (retval) { > @@ -1659,8 +1698,11 @@ > >     return 0; >  err_all: > -    video_device_release(radio->videodev); >     kfree(radio->buffer); > +err_video: > +    video_device_release(radio->videodev); > +err_intbuffer: > +    kfree(radio->int_in_buffer); >  err_radio: >     kfree(radio); >  err_initial: And also, cleanup was repaired in 2.6.29 :) > @@ -1674,12 +1716,8 @@ >  static int si470x_usb_driver_suspend(struct usb_interface *intf, >         pm_message_t message) >  { > -    struct si470x_device *radio = usb_get_intfdata(intf); > - >     printk(KERN_INFO DRIVER_NAME ": suspending now...\n"); > > -    cancel_delayed_work_sync(&radio->work); > - >     return 0; >  } > > @@ -1689,16 +1727,8 @@ >  */ >  static int si470x_usb_driver_resume(struct usb_interface *intf) >  { > -    struct si470x_device *radio = usb_get_intfdata(intf); > - >     printk(KERN_INFO DRIVER_NAME ": resuming now...\n"); > > -    mutex_lock(&radio->lock); > -    if (radio->users && radio->registers[SYSCONFIG1] & SYSCONFIG1_RDS) > -        schedule_delayed_work(&radio->work, > -            msecs_to_jiffies(rds_poll_time)); > -    mutex_unlock(&radio->lock); > - >     return 0; >  } > > @@ -1712,13 +1742,17 @@ > >     mutex_lock(&radio->disconnect_lock); >     radio->disconnected = 1; > -    cancel_delayed_work_sync(&radio->work); >     usb_set_intfdata(intf, NULL); > -    if (radio->users == 0) { > -        video_unregister_device(radio->videodev); > -        kfree(radio->buffer); > -        kfree(radio); > -    } > + > +    /* Free data structures. */ > +    if (radio->int_in_urb) > +        usb_free_urb(radio->int_in_urb); > + > +    kfree(radio->int_in_buffer); > +    video_unregister_device(radio->videodev); > +    kfree(radio->buffer); > +    kfree(radio); > + >     mutex_unlock(&radio->disconnect_lock); >  } >
difficult stations I can receive here in Paris, just like it should. Performance is similar to 'real' radios, if you see what I mean. Audio remains perfectly clear. When it comes to audio, it seems to me the "si470x_get_report" call generates some serious clicking in the audio on my Silabs dongle. I will investigate further, I wonder if other USB FM tokens have the same issue. I fixed this problem on rdsd by adding a configuration file option to prevent it constantly polling the dongle for its current frequency... Comments are welcome! Best regards, Signed-off-by: Edouard Lafargue <edouard@lafargue.name> diff -Nur ../radio-si470x.c radio-si470x.c --- ../radio-si470x.c 2008-12-25 00:26:37.000000000 +0100 +++ radio-si470x.c 2009-06-16 21:47:58.000000000 +0200 @@ -97,9 +97,13 @@ * - add support for KWorld USB FM Radio FM700 * - blacklisted KWorld radio in hid-core.c and hid-ids.h * + * 2009-06-16 Edouard Lafargue <edouard@lafargue.name> + * Version 1.0.9 + * - add support for interrupt mode for RDS endpoint, instead of polling. + * Improves RDS reception significantly + * * ToDo: * - add firmware download/update support - * - RDS support: interrupt mode, instead of polling * - add LED status output (check if that's not already done in firmware) */ @@ -107,10 +111,10 @@ /* driver definitions */ #define DRIVER_AUTHOR "Tobias Lorenz <tobias.lorenz@gmx.net>" #define DRIVER_NAME "radio-si470x" -#define DRIVER_KERNEL_VERSION KERNEL_VERSION(1, 0, 8) +#define DRIVER_KERNEL_VERSION KERNEL_VERSION(1, 0, 9) #define DRIVER_CARD "Silicon Labs Si470x FM Radio Receiver" #define DRIVER_DESC "USB radio driver for Si470x FM Radio Receivers" -#define DRIVER_VERSION "1.0.8" +#define DRIVER_VERSION "1.0.9" /* kernel includes */ @@ -206,16 +210,6 @@ module_param(max_rds_errors, ushort, 0644); MODULE_PARM_DESC(max_rds_errors, "RDS maximum block errors: *1*"); -/* RDS poll frequency */ -static unsigned int rds_poll_time = 40; -/* 40 is used by the original USBRadio.exe */ -/* 50 is used by radio-cadet */ -/* 75 should be okay */ -/* 80 is the usual RDS receive interval */ -module_param(rds_poll_time, uint, 0644); -MODULE_PARM_DESC(rds_poll_time, "RDS poll time (ms): *40*"); - - /************************************************************************** * Register Definitions @@ -440,6 +434,12 @@ struct usb_interface *intf; struct video_device *videodev; + /* Interrupt endpoint handling */ + char *int_in_buffer; + struct usb_endpoint_descriptor *int_in_endpoint; + struct urb *int_in_urb; + int int_in_running; + /* driver management */ unsigned int users; unsigned char disconnected; @@ -449,7 +449,6 @@ unsigned short registers[RADIO_REGISTER_NUM]; /* RDS receive buffer */ - struct delayed_work work; wait_queue_head_t read_queue; struct mutex lock; /* buffer locking */ unsigned char *buffer; /* size is always multiple of three */ @@ -578,37 +577,6 @@ } -/* - * si470x_get_rds_registers - read rds registers - */ -static int si470x_get_rds_registers(struct si470x_device *radio) -{ - unsigned char buf[RDS_REPORT_SIZE]; - int retval; - int size; - unsigned char regnr; - - buf[0] = RDS_REPORT; - - retval = usb_interrupt_msg(radio->usbdev, - usb_rcvintpipe(radio->usbdev, 1), - (void *) &buf, sizeof(buf), &size, usb_timeout); - if (size != sizeof(buf)) - printk(KERN_WARNING DRIVER_NAME ": si470x_get_rds_registers: " - "return size differs: %d != %zu\n", size, sizeof(buf)); - if (retval < 0) - printk(KERN_WARNING DRIVER_NAME ": si470x_get_rds_registers: " - "usb_interrupt_msg returned %d\n", retval); - - if (retval >= 0) - for (regnr = 0; regnr < RDS_REGISTER_NUM; regnr++) - radio->registers[STATUSRSSI + regnr] = - get_unaligned_be16( - &buf[regnr * RADIO_REGISTER_SIZE + 1]); - - return (retval < 0) ? -EINVAL : 0; -} - /* * si470x_set_chan - set the channel @@ -882,105 +850,117 @@ **************************************************************************/ /* - * si470x_rds - rds processing function + * si470x_int_in_callback - rds callback and processing function + * + * TODO: do we need to use mutex locks in some sections? */ -static void si470x_rds(struct si470x_device *radio) +static void si470x_int_in_callback(struct urb *urb) { + struct si470x_device *radio = urb->context; + unsigned char buf[RDS_REPORT_SIZE]; + int retval; + unsigned char regnr; unsigned char blocknum; unsigned short bler; /* rds block errors */ unsigned short rds; unsigned char tmpbuf[3]; - /* get rds blocks */ - if (si470x_get_rds_registers(radio) < 0) - return; - if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSR) == 0) { - /* No RDS group ready */ - return; - } - if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSS) == 0) { - /* RDS decoder not synchronized */ - return; - } - - /* copy all four RDS blocks to internal buffer */ - mutex_lock(&radio->lock); - for (blocknum = 0; blocknum < 4; blocknum++) { - switch (blocknum) { - default: - bler = (radio->registers[STATUSRSSI] & - STATUSRSSI_BLERA) >> 9; - rds = radio->registers[RDSA]; - break; - case 1: - bler = (radio->registers[READCHAN] & - READCHAN_BLERB) >> 14; - rds = radio->registers[RDSB]; - break; - case 2: - bler = (radio->registers[READCHAN] & - READCHAN_BLERC) >> 12; - rds = radio->registers[RDSC]; - break; - case 3: - bler = (radio->registers[READCHAN] & - READCHAN_BLERD) >> 10; - rds = radio->registers[RDSD]; - break; - }; - - /* Fill the V4L2 RDS buffer */ - put_unaligned_le16(rds, &tmpbuf); - tmpbuf[2] = blocknum; /* offset name */ - tmpbuf[2] |= blocknum << 3; /* received offset */ - if (bler > max_rds_errors) - tmpbuf[2] |= 0x80; /* uncorrectable errors */ - else if (bler > 0) - tmpbuf[2] |= 0x40; /* corrected error(s) */ - - /* copy RDS block to internal buffer */ - memcpy(&radio->buffer[radio->wr_index], &tmpbuf, 3); - radio->wr_index += 3; - - /* wrap write pointer */ - if (radio->wr_index >= radio->buf_size) - radio->wr_index = 0; - - /* check for overflow */ - if (radio->wr_index == radio->rd_index) { - /* increment and wrap read pointer */ - radio->rd_index += 3; - if (radio->rd_index >= radio->buf_size) - radio->rd_index = 0; + if (urb->status) { + if (urb->status == -ENOENT || + urb->status == -ECONNRESET || + urb->status == -ESHUTDOWN) { + return; + } else { + printk(KERN_WARNING DRIVER_NAME ": non-zero urb status (%d)\n", urb->status); + goto resubmit; /* Maybe we can recover. */ } } - mutex_unlock(&radio->lock); - - /* wake up read queue */ - if (radio->wr_index != radio->rd_index) - wake_up_interruptible(&radio->read_queue); -} - - -/* - * si470x_work - rds work function - */ -static void si470x_work(struct work_struct *work) -{ - struct si470x_device *radio = container_of(work, struct si470x_device, - work.work); /* safety checks */ if (radio->disconnected) return; if ((radio->registers[SYSCONFIG1] & SYSCONFIG1_RDS) == 0) - return; - - si470x_rds(radio); - schedule_delayed_work(&radio->work, msecs_to_jiffies(rds_poll_time)); -} + goto resubmit; + if (urb->actual_length > 0) { + /* Update RDS registers with URB data */ + buf[0] = RDS_REPORT; + for (regnr = 0; regnr < RDS_REGISTER_NUM; regnr++) + radio->registers[STATUSRSSI + regnr] = + get_unaligned_be16( + &radio->int_in_buffer[regnr * RADIO_REGISTER_SIZE + 1]); + /* get rds blocks */ + if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSR) == 0) { + /* No RDS group ready, better luck next time */ + goto resubmit; + } + if ((radio->registers[STATUSRSSI] & STATUSRSSI_RDSS) == 0) { + /* RDS decoder not synchronized */ + goto resubmit; + } + for (blocknum = 0; blocknum < 4; blocknum++) { + switch (blocknum) { + default: + bler = (radio->registers[STATUSRSSI] & + STATUSRSSI_BLERA) >> 9; + rds = radio->registers[RDSA]; + break; + case 1: + bler = (radio->registers[READCHAN] & + READCHAN_BLERB) >> 14; + rds = radio->registers[RDSB]; + break; + case 2: + bler = (radio->registers[READCHAN] & + READCHAN_BLERC) >> 12; + rds = radio->registers[RDSC]; + break; + case 3: + bler = (radio->registers[READCHAN] & + READCHAN_BLERD) >> 10; + rds = radio->registers[RDSD]; + break; + }; + + /* Fill the V4L2 RDS buffer */ + put_unaligned_le16(rds, &tmpbuf); + tmpbuf[2] = blocknum; /* offset name */ + tmpbuf[2] |= blocknum << 3; /* received offset */ + if (bler > max_rds_errors) + tmpbuf[2] |= 0x80; /* uncorrectable errors */ + else if (bler > 0) + tmpbuf[2] |= 0x40; /* corrected error(s) */ + + /* copy RDS block to internal buffer */ + memcpy(&radio->buffer[radio->wr_index], &tmpbuf, 3); + radio->wr_index += 3; + + /* wrap write pointer */ + if (radio->wr_index >= radio->buf_size) + radio->wr_index = 0; + + /* check for overflow */ + if (radio->wr_index == radio->rd_index) { + /* increment and wrap read pointer */ + radio->rd_index += 3; + if (radio->rd_index >= radio->buf_size) + radio->rd_index = 0; + } + } + if (radio->wr_index != radio->rd_index) + wake_up_interruptible(&radio->read_queue); + } +resubmit: + /* Resubmit if we're still running. */ + if (radio->int_in_running && radio->usbdev) { + retval = usb_submit_urb(radio->int_in_urb, GFP_ATOMIC); + if (retval) { + printk(KERN_WARNING DRIVER_NAME ": resubmitting urb failed (%d)", retval); + radio->int_in_running = 0; + } + } +} /**************************************************************************