diff options
-rw-r--r-- | drivers/gpu/drm/drm_file.c | 22 | ||||
-rw-r--r-- | drivers/gpu/drm/drm_ioctl.c | 22 | ||||
-rw-r--r-- | include/drm/drm_drv.h | 2 |
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 | ||
265 | static 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 | |||
265 | static int drm_setup(struct drm_device * dev) | 277 | static 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 | ||
511 | static inline bool | ||
512 | drm_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 | */ |
523 | int drm_ioctl_permit(u32 flags, struct drm_file *file_priv) | 530 | int 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 | */ |
770 | static inline bool drm_core_check_feature(struct drm_device *dev, u32 feature) | 770 | static 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 | } |