staging: comedi: remove manually unconfigured dynamic devices
authorIan Abbott <abbotti@mev.co.uk>
Thu, 4 Apr 2013 13:58:54 +0000 (14:58 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 5 Apr 2013 21:33:18 +0000 (14:33 -0700)
If a dynamically allocated (non-legacy, and automatically configured)
comedi device has been successfully unconfigured by use of the
`COMEDI_DEVCONFIG` ioctl, then remove the device afterwards.
(Dynamically identified comedi devices are identified by their minor
device number being `comedi_num_legacy_minors` or greater.)  This is
done in `comedi_unlocked_ioctl()` on return from `do_devconfig_ioctl()`.

Note that there is an unlikely race condition with some other thread
that has just called `comedi_file_info_from_minor()` or
`comedi_dev_from_minor()` and is about to use the device, but that race
condition also exists for automatically removed devices and will be
dealt with properly once reference counting of comedi devices has been
implemented.  We do avoid a race condition between automatic removal and
removal by the `COMEDI_DEVCONFIG` ioctl though.

Also add an extra precaution in `do_devconfig_ioctl()` to avoid
configuring a dynamically allocated device since there is a tight
window avoiding the race condition where this could happen and the
device is about to be removed anyway.

Signed-off-by: Ian Abbott <abbotti@mev.co.uk>
Reviewed-by: H Hartley Sweeten <hsweeten@visionengravers.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/staging/comedi/comedi_fops.c

index a155bf2c659222f6402d0a5bc0a097977854c6bc..d80060a14977b84e685cec2b9ac35366620ce3bf 100644 (file)
@@ -87,6 +87,9 @@ struct comedi_file_info {
 static DEFINE_SPINLOCK(comedi_file_info_table_lock);
 static struct comedi_file_info *comedi_file_info_table[COMEDI_NUM_MINORS];
 
+static struct comedi_file_info *comedi_clear_minor(unsigned minor);
+static void comedi_free_board_file_info(struct comedi_file_info *info);
+
 static struct comedi_file_info *comedi_file_info_from_minor(unsigned minor)
 {
        struct comedi_file_info *info;
@@ -490,6 +493,10 @@ static int do_devconfig_ioctl(struct comedi_device *dev,
                return -EINVAL;
        }
 
+       if (dev->minor >= comedi_num_legacy_minors)
+               /* don't re-use dynamically allocated comedi devices */
+               return -EBUSY;
+
        ret = comedi_device_attach(dev, &it);
        if (ret == 0) {
                if (!try_module_get(dev->driver->module)) {
@@ -1635,6 +1642,19 @@ static long comedi_unlocked_ioctl(struct file *file, unsigned int cmd,
                }
                rc = do_devconfig_ioctl(dev,
                                        (struct comedi_devconfig __user *)arg);
+               if (rc == 0) {
+                       if (arg == 0 &&
+                           dev->minor >= comedi_num_legacy_minors) {
+                               /* Successfully unconfigured a dynamically
+                                * allocated device.  Try and remove it. */
+                               info = comedi_clear_minor(dev->minor);
+                               if (info) {
+                                       mutex_unlock(&dev->mutex);
+                                       comedi_free_board_file_info(info);
+                                       return rc;
+                               }
+                       }
+               }
                goto done;
        }