summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/gpu/drm/drm_file.c22
-rw-r--r--drivers/gpu/drm/drm_ioctl.c22
-rw-r--r--include/drm/drm_drv.h2
3 files changed, 34 insertions, 12 deletions
diff --git a/drivers/gpu/drm/drm_file.c b/drivers/gpu/drm/drm_file.c
index b1838a41ad43..83a5bbca6e7e 100644
--- a/drivers/gpu/drm/drm_file.c
+++ b/drivers/gpu/drm/drm_file.c
@@ -262,6 +262,18 @@ void drm_file_free(struct drm_file *file)
262 kfree(file); 262 kfree(file);
263} 263}
264 264
265static void drm_close_helper(struct file *filp)
266{
267 struct drm_file *file_priv = filp->private_data;
268 struct drm_device *dev = file_priv->minor->dev;
269
270 mutex_lock(&dev->filelist_mutex);
271 list_del(&file_priv->lhead);
272 mutex_unlock(&dev->filelist_mutex);
273
274 drm_file_free(file_priv);
275}
276
265static int drm_setup(struct drm_device * dev) 277static int drm_setup(struct drm_device * dev)
266{ 278{
267 int ret; 279 int ret;
@@ -318,8 +330,10 @@ int drm_open(struct inode *inode, struct file *filp)
318 goto err_undo; 330 goto err_undo;
319 if (need_setup) { 331 if (need_setup) {
320 retcode = drm_setup(dev); 332 retcode = drm_setup(dev);
321 if (retcode) 333 if (retcode) {
334 drm_close_helper(filp);
322 goto err_undo; 335 goto err_undo;
336 }
323 } 337 }
324 return 0; 338 return 0;
325 339
@@ -473,11 +487,7 @@ int drm_release(struct inode *inode, struct file *filp)
473 487
474 DRM_DEBUG("open_count = %d\n", dev->open_count); 488 DRM_DEBUG("open_count = %d\n", dev->open_count);
475 489
476 mutex_lock(&dev->filelist_mutex); 490 drm_close_helper(filp);
477 list_del(&file_priv->lhead);
478 mutex_unlock(&dev->filelist_mutex);
479
480 drm_file_free(file_priv);
481 491
482 if (!--dev->open_count) { 492 if (!--dev->open_count) {
483 drm_lastclose(dev); 493 drm_lastclose(dev);
diff --git a/drivers/gpu/drm/drm_ioctl.c b/drivers/gpu/drm/drm_ioctl.c
index 7e6746b2d704..687943df58e1 100644
--- a/drivers/gpu/drm/drm_ioctl.c
+++ b/drivers/gpu/drm/drm_ioctl.c
@@ -508,6 +508,13 @@ int drm_version(struct drm_device *dev, void *data,
508 return err; 508 return err;
509} 509}
510 510
511static inline bool
512drm_render_driver_and_ioctl(const struct drm_device *dev, u32 flags)
513{
514 return drm_core_check_feature(dev, DRIVER_RENDER) &&
515 (flags & DRM_RENDER_ALLOW);
516}
517
511/** 518/**
512 * drm_ioctl_permit - Check ioctl permissions against caller 519 * drm_ioctl_permit - Check ioctl permissions against caller
513 * 520 *
@@ -522,14 +529,19 @@ int drm_version(struct drm_device *dev, void *data,
522 */ 529 */
523int drm_ioctl_permit(u32 flags, struct drm_file *file_priv) 530int drm_ioctl_permit(u32 flags, struct drm_file *file_priv)
524{ 531{
532 const struct drm_device *dev = file_priv->minor->dev;
533
525 /* ROOT_ONLY is only for CAP_SYS_ADMIN */ 534 /* ROOT_ONLY is only for CAP_SYS_ADMIN */
526 if (unlikely((flags & DRM_ROOT_ONLY) && !capable(CAP_SYS_ADMIN))) 535 if (unlikely((flags & DRM_ROOT_ONLY) && !capable(CAP_SYS_ADMIN)))
527 return -EACCES; 536 return -EACCES;
528 537
529 /* AUTH is only for authenticated or render client */ 538 /* AUTH is only for master ... */
530 if (unlikely((flags & DRM_AUTH) && !drm_is_render_client(file_priv) && 539 if (unlikely((flags & DRM_AUTH) && drm_is_primary_client(file_priv))) {
531 !file_priv->authenticated)) 540 /* authenticated ones, or render capable on DRM_RENDER_ALLOW. */
532 return -EACCES; 541 if (!file_priv->authenticated &&
542 !drm_render_driver_and_ioctl(dev, flags))
543 return -EACCES;
544 }
533 545
534 /* MASTER is only for master or control clients */ 546 /* MASTER is only for master or control clients */
535 if (unlikely((flags & DRM_MASTER) && 547 if (unlikely((flags & DRM_MASTER) &&
@@ -570,7 +582,7 @@ static const struct drm_ioctl_desc drm_ioctls[] = {
570 DRM_IOCTL_DEF(DRM_IOCTL_SET_UNIQUE, drm_invalid_op, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY), 582 DRM_IOCTL_DEF(DRM_IOCTL_SET_UNIQUE, drm_invalid_op, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
571 DRM_IOCTL_DEF(DRM_IOCTL_BLOCK, drm_noop, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY), 583 DRM_IOCTL_DEF(DRM_IOCTL_BLOCK, drm_noop, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
572 DRM_IOCTL_DEF(DRM_IOCTL_UNBLOCK, drm_noop, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY), 584 DRM_IOCTL_DEF(DRM_IOCTL_UNBLOCK, drm_noop, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
573 DRM_IOCTL_DEF(DRM_IOCTL_AUTH_MAGIC, drm_authmagic, DRM_AUTH|DRM_UNLOCKED|DRM_MASTER), 585 DRM_IOCTL_DEF(DRM_IOCTL_AUTH_MAGIC, drm_authmagic, DRM_UNLOCKED|DRM_MASTER),
574 586
575 DRM_IOCTL_DEF(DRM_IOCTL_ADD_MAP, drm_legacy_addmap_ioctl, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY), 587 DRM_IOCTL_DEF(DRM_IOCTL_ADD_MAP, drm_legacy_addmap_ioctl, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
576 DRM_IOCTL_DEF(DRM_IOCTL_RM_MAP, drm_legacy_rmmap_ioctl, DRM_AUTH), 588 DRM_IOCTL_DEF(DRM_IOCTL_RM_MAP, drm_legacy_rmmap_ioctl, DRM_AUTH),
diff --git a/include/drm/drm_drv.h b/include/drm/drm_drv.h
index ca46a45a9cce..570f9d03b2eb 100644
--- a/include/drm/drm_drv.h
+++ b/include/drm/drm_drv.h
@@ -767,7 +767,7 @@ static inline bool drm_dev_is_unplugged(struct drm_device *dev)
767 * 767 *
768 * Returns true if the @feature is supported, false otherwise. 768 * Returns true if the @feature is supported, false otherwise.
769 */ 769 */
770static inline bool drm_core_check_feature(struct drm_device *dev, u32 feature) 770static inline bool drm_core_check_feature(const struct drm_device *dev, u32 feature)
771{ 771{
772 return dev->driver->driver_features & dev->driver_features & feature; 772 return dev->driver->driver_features & dev->driver_features & feature;
773} 773}