|
@@ -465,7 +465,10 @@ static void drm_fs_inode_free(struct inode *inode)
|
|
|
* that do embed &struct drm_device it must be placed first in the overall
|
|
|
* structure, and the overall structure must be allocated using kmalloc(): The
|
|
|
* drm core's release function unconditionally calls kfree() on the @dev pointer
|
|
|
- * when the final reference is released.
|
|
|
+ * when the final reference is released. To override this behaviour, and so
|
|
|
+ * allow embedding of the drm_device inside the driver's device struct at an
|
|
|
+ * arbitrary offset, you must supply a &drm_driver.release callback and control
|
|
|
+ * the finalization explicitly.
|
|
|
*
|
|
|
* RETURNS:
|
|
|
* 0 on success, or error code on failure.
|
|
@@ -552,6 +555,41 @@ err_free:
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_dev_init);
|
|
|
|
|
|
+/**
|
|
|
+ * drm_dev_fini - Finalize a dead DRM device
|
|
|
+ * @dev: DRM device
|
|
|
+ *
|
|
|
+ * Finalize a dead DRM device. This is the converse to drm_dev_init() and
|
|
|
+ * frees up all data allocated by it. All driver private data should be
|
|
|
+ * finalized first. Note that this function does not free the @dev, that is
|
|
|
+ * left to the caller.
|
|
|
+ *
|
|
|
+ * The ref-count of @dev must be zero, and drm_dev_fini() should only be called
|
|
|
+ * from a &drm_driver.release callback.
|
|
|
+ */
|
|
|
+void drm_dev_fini(struct drm_device *dev)
|
|
|
+{
|
|
|
+ drm_vblank_cleanup(dev);
|
|
|
+
|
|
|
+ if (drm_core_check_feature(dev, DRIVER_GEM))
|
|
|
+ drm_gem_destroy(dev);
|
|
|
+
|
|
|
+ drm_legacy_ctxbitmap_cleanup(dev);
|
|
|
+ drm_ht_remove(&dev->map_hash);
|
|
|
+ drm_fs_inode_free(dev->anon_inode);
|
|
|
+
|
|
|
+ drm_minor_free(dev, DRM_MINOR_PRIMARY);
|
|
|
+ drm_minor_free(dev, DRM_MINOR_RENDER);
|
|
|
+ drm_minor_free(dev, DRM_MINOR_CONTROL);
|
|
|
+
|
|
|
+ mutex_destroy(&dev->master_mutex);
|
|
|
+ mutex_destroy(&dev->ctxlist_mutex);
|
|
|
+ mutex_destroy(&dev->filelist_mutex);
|
|
|
+ mutex_destroy(&dev->struct_mutex);
|
|
|
+ kfree(dev->unique);
|
|
|
+}
|
|
|
+EXPORT_SYMBOL(drm_dev_fini);
|
|
|
+
|
|
|
/**
|
|
|
* drm_dev_alloc - Allocate new DRM device
|
|
|
* @driver: DRM driver to allocate device for
|
|
@@ -598,25 +636,12 @@ static void drm_dev_release(struct kref *ref)
|
|
|
{
|
|
|
struct drm_device *dev = container_of(ref, struct drm_device, ref);
|
|
|
|
|
|
- drm_vblank_cleanup(dev);
|
|
|
-
|
|
|
- if (drm_core_check_feature(dev, DRIVER_GEM))
|
|
|
- drm_gem_destroy(dev);
|
|
|
-
|
|
|
- drm_legacy_ctxbitmap_cleanup(dev);
|
|
|
- drm_ht_remove(&dev->map_hash);
|
|
|
- drm_fs_inode_free(dev->anon_inode);
|
|
|
-
|
|
|
- drm_minor_free(dev, DRM_MINOR_PRIMARY);
|
|
|
- drm_minor_free(dev, DRM_MINOR_RENDER);
|
|
|
- drm_minor_free(dev, DRM_MINOR_CONTROL);
|
|
|
-
|
|
|
- mutex_destroy(&dev->master_mutex);
|
|
|
- mutex_destroy(&dev->ctxlist_mutex);
|
|
|
- mutex_destroy(&dev->filelist_mutex);
|
|
|
- mutex_destroy(&dev->struct_mutex);
|
|
|
- kfree(dev->unique);
|
|
|
- kfree(dev);
|
|
|
+ if (dev->driver->release) {
|
|
|
+ dev->driver->release(dev);
|
|
|
+ } else {
|
|
|
+ drm_dev_fini(dev);
|
|
|
+ kfree(dev);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
/**
|