Commit faf8a6b6 authored by Markus Demleitner's avatar Markus Demleitner Committed by Greg Kroah-Hartman

[PATCH] USB: DSBR-100 tiny patch

On Fri, Feb 06, 2004 at 10:17:32AM -0800, Greg KH wrote:
> On Fri, Feb 06, 2004 at 05:06:01PM +0100, Markus Demleitner wrote:
> > Since I finally switched over to 2.6 I noticed that my dsbr100 driver
> > produces a warning to the effect that I should provide a release
> > callback.  After a quick google on the issue I came to the conclusion
>
> No, you will have to fix up your driver to work properly, sorry.  It's
> due to the changes to the v4l layer to handle removable devices much
> better (and to tie it into the driver model.)

I didn't get around to doing real work on this until now, but finally
in the attachment there's my stab at bringing dsbr100 up to kernel 2.6.
I'm not really comfortable with the release callback issues (I've yet
to find some HOWTO-like documentation on this...) on the v4l side,
so I'd be grateful if you could have a look at it.  I've basically
tried to copy what stv680 does, which may or may not have been a
good idea (in particular see the comment above the disconnect
function).

I've used the opportunity for some code beautyfing, which of course
makes the patch a bit of a mess.  I hope you won't mind too much
-- as you can see, it would have been pretty messy anyway.
parent 3861df46
...@@ -33,6 +33,9 @@ ...@@ -33,6 +33,9 @@
History: History:
Version 0.40:
Markus: Updates for 2.6.x kernels, code layout changes, name sanitizing
Version 0.30: Version 0.30:
Markus: Updates for 2.5.x kernel and more ISO compliant source Markus: Updates for 2.5.x kernel and more ISO compliant source
...@@ -75,13 +78,17 @@ ...@@ -75,13 +78,17 @@
/* /*
* Version Information * Version Information
*/ */
#define DRIVER_VERSION "v0.30" #define DRIVER_VERSION "v0.40"
#define DRIVER_AUTHOR "Markus Demleitner <msdemlei@tucana.harvard.edu>" #define DRIVER_AUTHOR "Markus Demleitner <msdemlei@tucana.harvard.edu>"
#define DRIVER_DESC "D-Link DSB-R100 USB FM radio driver" #define DRIVER_DESC "D-Link DSB-R100 USB FM radio driver"
#define DSB100_VENDOR 0x04b4 #define DSB100_VENDOR 0x04b4
#define DSB100_PRODUCT 0x1002 #define DSB100_PRODUCT 0x1002
/* Commands the device appears to understand */
#define DSB100_TUNE 1
#define DSB100_ONOFF 2
#define TB_LEN 16 #define TB_LEN 16
/* Frequency limits in MHz -- these are European values. For Japanese /* Frequency limits in MHz -- these are European values. For Japanese
...@@ -102,15 +109,19 @@ static int usb_dsbr100_close(struct inode *inode, struct file *file); ...@@ -102,15 +109,19 @@ static int usb_dsbr100_close(struct inode *inode, struct file *file);
static int radio_nr = -1; static int radio_nr = -1;
MODULE_PARM(radio_nr, "i"); MODULE_PARM(radio_nr, "i");
typedef struct /* Data for one (physical) device */
{ typedef struct {
struct usb_device *dev; struct usb_device *usbdev;
struct video_device *videodev;
unsigned char transfer_buffer[TB_LEN]; unsigned char transfer_buffer[TB_LEN];
int curfreq; int curfreq;
int stereo; int stereo;
} usb_dsbr100; int users;
int removed;
} dsbr100_device;
/* File system interface */
static struct file_operations usb_dsbr100_fops = { static struct file_operations usb_dsbr100_fops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.open = usb_dsbr100_open, .open = usb_dsbr100_open,
...@@ -118,65 +129,84 @@ static struct file_operations usb_dsbr100_fops = { ...@@ -118,65 +129,84 @@ static struct file_operations usb_dsbr100_fops = {
.ioctl = usb_dsbr100_ioctl, .ioctl = usb_dsbr100_ioctl,
.llseek = no_llseek, .llseek = no_llseek,
}; };
static struct video_device usb_dsbr100_radio=
/* V4L interface */
static struct video_device dsbr100_videodev_template=
{ {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.name = "D-Link DSB-R 100", .name = "D-Link DSB-R 100",
.type = VID_TYPE_TUNER, .type = VID_TYPE_TUNER,
.hardware = VID_HARDWARE_AZTECH, .hardware = VID_HARDWARE_AZTECH,
.fops = &usb_dsbr100_fops, .fops = &usb_dsbr100_fops,
.release = video_device_release,
}; };
static int users = 0; static struct usb_device_id usb_dsbr100_device_table [] = {
static struct usb_device_id usb_dsbr100_table [] = {
{ USB_DEVICE(DSB100_VENDOR, DSB100_PRODUCT) }, { USB_DEVICE(DSB100_VENDOR, DSB100_PRODUCT) },
{ } /* Terminating entry */ { } /* Terminating entry */
}; };
MODULE_DEVICE_TABLE (usb, usb_dsbr100_table); MODULE_DEVICE_TABLE (usb, usb_dsbr100_device_table);
/* USB subsystem interface */
static struct usb_driver usb_dsbr100_driver = { static struct usb_driver usb_dsbr100_driver = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.name = "dsbr100", .name = "dsbr100",
.probe = usb_dsbr100_probe, .probe = usb_dsbr100_probe,
.disconnect = usb_dsbr100_disconnect, .disconnect = usb_dsbr100_disconnect,
.id_table = usb_dsbr100_table, .id_table = usb_dsbr100_device_table,
}; };
/* Low-level device interface begins here */
static int dsbr100_start(usb_dsbr100 *radio) /* switch on radio */
static int dsbr100_start(dsbr100_device *radio)
{ {
if (usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), if (usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
0x00, 0xC0, 0x00, 0xC7, radio->transfer_buffer, 8, 300)<0 || USB_REQ_GET_STATUS,
usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x02, 0xC0, 0x01, 0x00, radio->transfer_buffer, 8, 300)<0) 0x00, 0xC7, radio->transfer_buffer, 8, 300)<0 ||
usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
DSB100_ONOFF,
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x01, 0x00, radio->transfer_buffer, 8, 300)<0)
return -1; return -1;
return (radio->transfer_buffer)[0]; return (radio->transfer_buffer)[0];
} }
static int dsbr100_stop(usb_dsbr100 *radio) /* switch off radio */
static int dsbr100_stop(dsbr100_device *radio)
{ {
if (usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), if (usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
0x00, 0xC0, 0x16, 0x1C, radio->transfer_buffer, 8, 300)<0 || USB_REQ_GET_STATUS,
usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x02, 0xC0, 0x00, 0x00, radio->transfer_buffer, 8, 300)<0) 0x16, 0x1C, radio->transfer_buffer, 8, 300)<0 ||
usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
DSB100_ONOFF,
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x00, 0x00, radio->transfer_buffer, 8, 300)<0)
return -1; return -1;
return (radio->transfer_buffer)[0]; return (radio->transfer_buffer)[0];
} }
/* set a frequency, freq is defined by v4l's TUNER_LOW, i.e. 1/16th kHz */
static int dsbr100_setfreq(usb_dsbr100 *radio, int freq) static int dsbr100_setfreq(dsbr100_device *radio, int freq)
{ {
freq = (freq/16*80)/1000+856; freq = (freq/16*80)/1000+856;
if (usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), if (usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
0x01, 0xC0, (freq>>8)&0x00ff, freq&0xff, DSB100_TUNE,
radio->transfer_buffer, 8, 300)<0 || USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), (freq>>8)&0x00ff, freq&0xff,
0x00, 0xC0, 0x96, 0xB7, radio->transfer_buffer, 8, 300)<0 || radio->transfer_buffer, 8, 300)<0 ||
usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
0x00, 0xC0, 0x00, 0x24, radio->transfer_buffer, 8, 300)<0) { USB_REQ_GET_STATUS,
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x96, 0xB7, radio->transfer_buffer, 8, 300)<0 ||
usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
USB_REQ_GET_STATUS,
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x00, 0x24, radio->transfer_buffer, 8, 300)<0) {
radio->stereo = -1; radio->stereo = -1;
return -1; return -1;
} }
...@@ -184,61 +214,92 @@ static int dsbr100_setfreq(usb_dsbr100 *radio, int freq) ...@@ -184,61 +214,92 @@ static int dsbr100_setfreq(usb_dsbr100 *radio, int freq)
return (radio->transfer_buffer)[0]; return (radio->transfer_buffer)[0];
} }
static void dsbr100_getstat(usb_dsbr100 *radio) /* return the device status. This is, in effect, just whether it
sees a stereo signal or not. Pity. */
static void dsbr100_getstat(dsbr100_device *radio)
{ {
if (usb_control_msg(radio->dev, usb_rcvctrlpipe(radio->dev, 0), if (usb_control_msg(radio->usbdev, usb_rcvctrlpipe(radio->usbdev, 0),
0x00, 0xC0, 0x00 , 0x24, radio->transfer_buffer, 8, 300)<0) USB_REQ_GET_STATUS,
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
0x00 , 0x24, radio->transfer_buffer, 8, 300)<0)
radio->stereo = -1; radio->stereo = -1;
else else
radio->stereo = ! (radio->transfer_buffer[0]&0x01); radio->stereo = ! (radio->transfer_buffer[0]&0x01);
} }
/* USB subsystem interface begins here */
/* check if the device is present and register with v4l and
usb if it is */
static int usb_dsbr100_probe(struct usb_interface *intf, static int usb_dsbr100_probe(struct usb_interface *intf,
const struct usb_device_id *id) const struct usb_device_id *id)
{ {
usb_dsbr100 *radio; dsbr100_device *radio;
struct video_device *videodev;
if (!(radio = kmalloc(sizeof(usb_dsbr100),GFP_KERNEL))) if (!(radio = kmalloc(sizeof(dsbr100_device), GFP_KERNEL)))
return -ENOMEM;
if (!(radio->videodev = video_device_alloc())) {
kfree(radio);
return -ENOMEM; return -ENOMEM;
usb_dsbr100_radio.priv = radio; }
radio->dev = interface_to_usbdev (intf); memcpy(radio->videodev, &dsbr100_videodev_template,
sizeof(dsbr100_videodev_template));
radio->removed = 0;
radio->users = 0;
radio->usbdev = interface_to_usbdev(intf);
radio->curfreq = FREQ_MIN*FREQ_MUL; radio->curfreq = FREQ_MIN*FREQ_MUL;
usb_set_intfdata (intf, radio); video_set_drvdata(radio->videodev, radio);
if (video_register_device(radio->videodev, VFL_TYPE_RADIO,
radio_nr)) {
warn("Could not register video device");
video_device_release(radio->videodev);
kfree(radio);
return -EIO;
}
usb_set_intfdata(intf, radio);
return 0; return 0;
} }
/* handle unplugging of the device, release data structures
if nothing keeps us from doing it. If something is still
keeping us busy, the release callback of v4l will take care
of releasing it. stv680.c does not relase its private
data, so I don't do this here either. Checking out the
code I'd expect I better did that, but if there's a memory
leak here it's tiny (~50 bytes per disconnect) */
static void usb_dsbr100_disconnect(struct usb_interface *intf) static void usb_dsbr100_disconnect(struct usb_interface *intf)
{ {
usb_dsbr100 *radio = usb_get_intfdata (intf); dsbr100_device *radio = usb_get_intfdata(intf);
usb_set_intfdata (intf, NULL); usb_set_intfdata (intf, NULL);
if (radio) { if (radio) {
lock_kernel(); video_unregister_device(radio->videodev);
if (users) { radio->videodev = NULL;
unlock_kernel(); if (radio->users) {
return; kfree(radio);
} else {
radio->removed = 1;
} }
kfree(radio);
usb_dsbr100_radio.priv = NULL;
unlock_kernel();
} }
} }
/* Video for Linux interface */
static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file, static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file,
unsigned int cmd, void *arg) unsigned int cmd, void *arg)
{ {
struct video_device *dev = video_devdata(file); dsbr100_device *radio=video_get_drvdata(video_devdata(file));
usb_dsbr100 *radio=dev->priv;
if (!radio) if (!radio)
return -EINVAL; return -EIO;
switch(cmd) switch(cmd) {
{
case VIDIOCGCAP: { case VIDIOCGCAP: {
struct video_capability *v = arg; struct video_capability *v = arg;
memset(v, 0, sizeof(*v)); memset(v, 0, sizeof(*v));
v->type = VID_TYPE_TUNER; v->type = VID_TYPE_TUNER;
v->channels = 1; v->channels = 1;
...@@ -248,6 +309,7 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file, ...@@ -248,6 +309,7 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file,
} }
case VIDIOCGTUNER: { case VIDIOCGTUNER: {
struct video_tuner *v = arg; struct video_tuner *v = arg;
dsbr100_getstat(radio); dsbr100_getstat(radio);
if(v->tuner) /* Only 1 tuner */ if(v->tuner) /* Only 1 tuner */
return -EINVAL; return -EINVAL;
...@@ -263,21 +325,21 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file, ...@@ -263,21 +325,21 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file,
} }
case VIDIOCSTUNER: { case VIDIOCSTUNER: {
struct video_tuner *v = arg; struct video_tuner *v = arg;
if(v->tuner!=0) if(v->tuner!=0)
return -EINVAL; return -EINVAL;
/* Only 1 tuner so no setting needed ! */ /* Only 1 tuner so no setting needed ! */
return 0; return 0;
} }
case VIDIOCGFREQ: case VIDIOCGFREQ: {
{
int *freq = arg; int *freq = arg;
if (radio->curfreq==-1) if (radio->curfreq==-1)
return -EINVAL; return -EINVAL;
*freq = radio->curfreq; *freq = radio->curfreq;
return 0; return 0;
} }
case VIDIOCSFREQ: case VIDIOCSFREQ: {
{
int *freq = arg; int *freq = arg;
radio->curfreq = *freq; radio->curfreq = *freq;
...@@ -287,6 +349,7 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file, ...@@ -287,6 +349,7 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file,
} }
case VIDIOCGAUDIO: { case VIDIOCGAUDIO: {
struct video_audio *v = arg; struct video_audio *v = arg;
memset(v, 0, sizeof(*v)); memset(v, 0, sizeof(*v));
v->flags |= VIDEO_AUDIO_MUTABLE; v->flags |= VIDEO_AUDIO_MUTABLE;
v->mode = VIDEO_SOUND_STEREO; v->mode = VIDEO_SOUND_STEREO;
...@@ -297,9 +360,9 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file, ...@@ -297,9 +360,9 @@ static int usb_dsbr100_do_ioctl(struct inode *inode, struct file *file,
} }
case VIDIOCSAUDIO: { case VIDIOCSAUDIO: {
struct video_audio *v = arg; struct video_audio *v = arg;
if (v->audio) if (v->audio)
return -EINVAL; return -EINVAL;
if (v->flags&VIDEO_AUDIO_MUTE) { if (v->flags&VIDEO_AUDIO_MUTE) {
if (dsbr100_stop(radio)==-1) if (dsbr100_stop(radio)==-1)
warn("Radio did not respond properly"); warn("Radio did not respond properly");
...@@ -322,64 +385,40 @@ static int usb_dsbr100_ioctl(struct inode *inode, struct file *file, ...@@ -322,64 +385,40 @@ static int usb_dsbr100_ioctl(struct inode *inode, struct file *file,
static int usb_dsbr100_open(struct inode *inode, struct file *file) static int usb_dsbr100_open(struct inode *inode, struct file *file)
{ {
struct video_device *dev = video_devdata(file); dsbr100_device *radio=video_get_drvdata(video_devdata(file));
usb_dsbr100 *radio=dev->priv;
if (! radio) { radio->users = 1;
warn("Radio not initialised"); if (dsbr100_start(radio)<0) {
return -EAGAIN;
}
if(users)
{
warn("Radio in use");
return -EBUSY;
}
users++;
if (dsbr100_start(radio)<0)
warn("Radio did not start up properly"); warn("Radio did not start up properly");
radio->users = 0;
return -EIO;
}
dsbr100_setfreq(radio, radio->curfreq); dsbr100_setfreq(radio, radio->curfreq);
return 0; return 0;
} }
static int usb_dsbr100_close(struct inode *inode, struct file *file) static int usb_dsbr100_close(struct inode *inode, struct file *file)
{ {
struct video_device *dev = video_devdata(file); dsbr100_device *radio=video_get_drvdata(video_devdata(file));
usb_dsbr100 *radio=dev->priv;
if (!radio) if (!radio)
return -ENODEV; return -ENODEV;
users--; radio->users = 0;
if (radio->removed) {
kfree(radio);
}
return 0; return 0;
} }
static int __init dsbr100_init(void) static int __init dsbr100_init(void)
{ {
int retval; int retval = usb_register(&usb_dsbr100_driver);
usb_dsbr100_radio.priv = NULL;
retval = usb_register(&usb_dsbr100_driver);
if (retval)
goto failed_usb_register;
retval = video_register_device(&usb_dsbr100_radio, VFL_TYPE_RADIO,
radio_nr);
if (retval) {
warn("Couldn't register video device");
goto failed_video_register;
}
info(DRIVER_VERSION ":" DRIVER_DESC); info(DRIVER_VERSION ":" DRIVER_DESC);
return 0;
failed_video_register:
usb_deregister(&usb_dsbr100_driver);
failed_usb_register:
return retval; return retval;
} }
static void __exit dsbr100_exit(void) static void __exit dsbr100_exit(void)
{ {
usb_dsbr100 *radio=usb_dsbr100_radio.priv;
if (radio)
dsbr100_stop(radio);
video_unregister_device(&usb_dsbr100_radio);
usb_deregister(&usb_dsbr100_driver); usb_deregister(&usb_dsbr100_driver);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment