[PATCH 4/5] drm: add core support for unplugging a device
Daniel Vetter
daniel at ffwll.ch
Tue Feb 21 00:59:05 PST 2012
On Mon, Feb 20, 2012 at 04:13:48PM +0000, Dave Airlie wrote:
> From: Dave Airlie <airlied at redhat.com>
>
> Two parts to this, one is simple unplug from sysfs for the device node.
>
> The second adds an unplugged state, if we have device opens, we
> just set the unplugged state and return, if we have no device
> opens we drop the drm device.
>
> If after a lastclose we discover we are unplugged we then
> drop the drm device.
>
> Signed-off-by: Dave Airlie <airlied at redhat.com>
I like the unplugged state to handle whole-device removal, but I think
this needs a few extensions:
- I think we should unconditionally unregister the drm dev minors. Simply
checking dev->unplugged should do the trick because we can't just tear
down the minor structs (we can only unregister the dev node).
- writing to dev->unplugged needs to be protected by the same mutex (i.e.
drm_global_mutex) as the actual unregister step.
- I think we should add a simply drm_device_is_unplugged so that drivers
can check the unplugged state in a race-free manner. Adding an smp_wmb()
after writing to dev->unplugged and an smb_rmb() before reading it in
that helper (plus ensuring that dev->unplugged is a separate word in
drm_device) should do the trick without overhead.
- I think we wan't the drm core to return -ENODEV for any ioctls an open
calls and return a SIGBUS on the fault handler. Currently you roll that
yourself in udl, but I can't think of a use-case where we want to stick
the interfaces around after a hotremove of the entire device.
Cheers, Daniel
> ---
> drivers/gpu/drm/drm_fops.c | 2 ++
> drivers/gpu/drm/drm_stub.c | 22 ++++++++++++++++++++++
> include/drm/drmP.h | 2 ++
> 3 files changed, 26 insertions(+), 0 deletions(-)
>
> diff --git a/drivers/gpu/drm/drm_fops.c b/drivers/gpu/drm/drm_fops.c
> index 6263b01..154cfac 100644
> --- a/drivers/gpu/drm/drm_fops.c
> +++ b/drivers/gpu/drm/drm_fops.c
> @@ -579,6 +579,8 @@ int drm_release(struct inode *inode, struct file *filp)
> retcode = -EBUSY;
> } else
> retcode = drm_lastclose(dev);
> + if (dev->unplugged)
> + drm_put_dev(dev);
> }
> mutex_unlock(&drm_global_mutex);
>
> diff --git a/drivers/gpu/drm/drm_stub.c b/drivers/gpu/drm/drm_stub.c
> index 6d7b083..b04c92d 100644
> --- a/drivers/gpu/drm/drm_stub.c
> +++ b/drivers/gpu/drm/drm_stub.c
> @@ -429,6 +429,11 @@ int drm_put_minor(struct drm_minor **minor_p)
> return 0;
> }
>
> +static void drm_unplug_minor(struct drm_minor *minor)
> +{
> + drm_sysfs_device_remove(minor);
> +}
> +
> /**
> * Called via drm_exit() at module unload time or when pci device is
> * unplugged.
> @@ -492,3 +497,20 @@ void drm_put_dev(struct drm_device *dev)
> kfree(dev);
> }
> EXPORT_SYMBOL(drm_put_dev);
> +
> +void drm_unplug_dev(struct drm_device *dev)
> +{
> + /* for a USB device */
> + if (drm_core_check_feature(dev, DRIVER_MODESET))
> + drm_unplug_minor(dev->control);
> + drm_unplug_minor(dev->primary);
> +
> + dev->unplugged = true;
> +
> + mutex_lock(&drm_global_mutex);
> + if (dev->open_count == 0) {
> + drm_put_dev(dev);
> + }
> + mutex_unlock(&drm_global_mutex);
> +}
> +EXPORT_SYMBOL(drm_unplug_dev);
> diff --git a/include/drm/drmP.h b/include/drm/drmP.h
> index 92f0981..8c11797 100644
> --- a/include/drm/drmP.h
> +++ b/include/drm/drmP.h
> @@ -1170,6 +1170,7 @@ struct drm_device {
> struct idr object_name_idr;
> /*@} */
> int switch_power_state;
> + bool unplugged; /* device has been unplugged or gone away */
> };
>
> #define DRM_SWITCH_POWER_ON 0
> @@ -1464,6 +1465,7 @@ extern void drm_master_put(struct drm_master **master);
>
> extern void drm_put_dev(struct drm_device *dev);
> extern int drm_put_minor(struct drm_minor **minor);
> +extern void drm_unplug_dev(struct drm_device *dev);
> extern unsigned int drm_debug;
>
> extern unsigned int drm_vblank_offdelay;
> --
> 1.7.6
>
> _______________________________________________
> dri-devel mailing list
> dri-devel at lists.freedesktop.org
> http://lists.freedesktop.org/mailman/listinfo/dri-devel
--
Daniel Vetter
Mail: daniel at ffwll.ch
Mobile: +41 (0)79 365 57 48
More information about the dri-devel
mailing list