2005-04-16 15:20:36 -07:00
/*
* Created : Fri Jan 8 09 : 01 : 26 1999 by faith @ valinux . com
*
* Copyright 1999 Precision Insight , Inc . , Cedar Park , Texas .
* Copyright 2000 VA Linux Systems , Inc . , Sunnyvale , California .
* All Rights Reserved .
*
2014-07-23 14:14:53 +02:00
* Author Rickard E . ( Rik ) Faith < faith @ valinux . com >
* Author Gareth Hughes < gareth @ valinux . com >
*
2005-04-16 15:20:36 -07:00
* Permission is hereby granted , free of charge , to any person obtaining a
* copy of this software and associated documentation files ( the " Software " ) ,
* to deal in the Software without restriction , including without limitation
* the rights to use , copy , modify , merge , publish , distribute , sublicense ,
* and / or sell copies of the Software , and to permit persons to whom the
* Software is furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice ( including the next
* paragraph ) shall be included in all copies or substantial portions of the
* Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
* VA LINUX SYSTEMS AND / OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM , DAMAGES OR
* OTHER LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE ,
* ARISING FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE .
*/
2012-10-02 18:01:07 +01:00
# include <drm/drmP.h>
2016-06-21 10:54:22 +02:00
# include <drm/drm_auth.h>
2014-07-24 12:10:04 +02:00
# include "drm_legacy.h"
2014-09-10 08:16:10 +02:00
# include "drm_internal.h"
2014-12-18 22:49:02 +01:00
# include "drm_crtc_internal.h"
2005-04-16 15:20:36 -07:00
2012-10-02 18:01:07 +01:00
# include <linux/pci.h>
# include <linux/export.h>
2005-04-16 15:20:36 -07:00
2016-06-21 14:08:33 +02:00
/**
* DOC : getunique and setversion story
*
* BEWARE THE DRAGONS ! MIND THE TRAPDOORS !
*
* In an attempt to warn anyone else who ' s trying to figure out what ' s going
* on here , I ' ll try to summarize the story . First things first , let ' s clear up
* the names , because the kernel internals , libdrm and the ioctls are all named
* differently :
*
* - GET_UNIQUE ioctl , implemented by drm_getunique is wrapped up in libdrm
* through the drmGetBusid function .
* - The libdrm drmSetBusid function is backed by the SET_UNIQUE ioctl . All
* that code is nerved in the kernel with drm_invalid_op ( ) .
* - The internal set_busid kernel functions and driver callbacks are
* exclusively use by the SET_VERSION ioctl , because only drm 1.0 ( which is
* nerved ) allowed userspace to set the busid through the above ioctl .
* - Other ioctls and functions involved are named consistently .
*
* For anyone wondering what ' s the difference between drm 1.1 and 1.4 : Correctly
* handling pci domains in the busid on ppc . Doing this correctly was only
* implemented in libdrm in 2010 , hence can ' t be nerved yet . No one knows what ' s
* special with drm 1.2 and 1.3 .
*
* Now the actual horror story of how device lookup in drm works . At large ,
* there ' s 2 different ways , either by busid , or by device driver name .
*
* Opening by busid is fairly simple :
*
* 1. First call SET_VERSION to make sure pci domains are handled properly . As a
* side - effect this fills out the unique name in the master structure .
* 2. Call GET_UNIQUE to read out the unique name from the master structure ,
* which matches the busid thanks to step 1. If it doesn ' t , proceed to try
* the next device node .
*
* Opening by name is slightly different :
*
* 1. Directly call VERSION to get the version and to match against the driver
* name returned by that ioctl . Note that SET_VERSION is not called , which
* means the the unique name for the master node just opening is _not_ filled
* out . This despite that with current drm device nodes are always bound to
* one device , and can ' t be runtime assigned like with drm 1.0 .
* 2. Match driver name . If it mismatches , proceed to the next device node .
* 3. Call GET_UNIQUE , and check whether the unique name has length zero ( by
* checking that the first byte in the string is 0 ) . If that ' s not the case
* libdrm skips and proceeds to the next device node . Probably this is just
* copypasta from drm 1.0 times where a set unique name meant that the driver
* was in use already , but that ' s just conjecture .
*
* Long story short : To keep the open by name logic working , GET_UNIQUE must
* _not_ return a unique string when SET_VERSION hasn ' t been called yet ,
* otherwise libdrm breaks . Even when that unique string can ' t ever change , and
* is totally irrelevant for actually opening the device because runtime
* assignable device instances were only support in drm 1.0 , which is long dead .
* But the libdrm code in drmOpenByName somehow survived , hence this can ' t be
* broken .
*/
2014-07-23 14:14:53 +02:00
static int drm_version ( struct drm_device * dev , void * data ,
struct drm_file * file_priv ) ;
2015-09-08 13:56:27 +02:00
/*
2005-04-16 15:20:36 -07:00
* Get the bus id .
2005-09-25 14:28:13 +10:00
*
2005-04-16 15:20:36 -07:00
* \ param inode device inode .
2007-08-25 20:23:09 +10:00
* \ param file_priv DRM file private .
2005-04-16 15:20:36 -07:00
* \ param cmd command .
* \ param arg user argument , pointing to a drm_unique structure .
* \ return zero on success or a negative number on failure .
*
* Copies the bus id from drm_device : : unique into user space .
*/
2014-09-10 12:43:55 +02:00
static int drm_getunique ( struct drm_device * dev , void * data ,
2007-09-03 12:06:45 +10:00
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2007-09-03 12:06:45 +10:00
struct drm_unique * u = data ;
2008-11-28 14:22:24 +10:00
struct drm_master * master = file_priv - > master ;
2005-04-16 15:20:36 -07:00
2016-12-10 22:52:52 +01:00
mutex_lock ( & master - > dev - > master_mutex ) ;
2008-11-28 14:22:24 +10:00
if ( u - > unique_len > = master - > unique_len ) {
2016-12-10 22:52:52 +01:00
if ( copy_to_user ( u - > unique , master - > unique , master - > unique_len ) ) {
mutex_unlock ( & master - > dev - > master_mutex ) ;
2005-04-16 15:20:36 -07:00
return - EFAULT ;
2016-12-10 22:52:52 +01:00
}
2005-04-16 15:20:36 -07:00
}
2008-11-28 14:22:24 +10:00
u - > unique_len = master - > unique_len ;
2016-12-10 22:52:52 +01:00
mutex_unlock ( & master - > dev - > master_mutex ) ;
2007-09-03 12:06:45 +10:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
2010-08-04 11:09:42 +01:00
static void
drm_unset_busid ( struct drm_device * dev ,
struct drm_master * master )
{
kfree ( master - > unique ) ;
master - > unique = NULL ;
master - > unique_len = 0 ;
}
2008-11-28 14:22:24 +10:00
static int drm_set_busid ( struct drm_device * dev , struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2008-11-28 14:22:24 +10:00
struct drm_master * master = file_priv - > master ;
2010-12-15 03:16:38 +10:00
int ret ;
2010-08-04 11:09:42 +01:00
if ( master - > unique ! = NULL )
drm_unset_busid ( dev , master ) ;
2006-01-02 21:19:39 +11:00
2014-08-29 12:12:43 +02:00
if ( dev - > driver - > set_busid ) {
ret = dev - > driver - > set_busid ( dev , master ) ;
if ( ret ) {
drm_unset_busid ( dev , master ) ;
return ret ;
}
2014-04-11 15:23:00 +02:00
} else {
2016-06-21 10:54:14 +02:00
WARN_ON ( ! dev - > unique ) ;
2014-04-11 15:23:00 +02:00
master - > unique = kstrdup ( dev - > unique , GFP_KERNEL ) ;
if ( master - > unique )
master - > unique_len = strlen ( dev - > unique ) ;
}
2005-04-16 15:20:36 -07:00
return 0 ;
}
2015-09-08 13:56:27 +02:00
/*
2005-04-16 15:20:36 -07:00
* Get client information .
*
* \ param inode device inode .
2007-08-25 20:23:09 +10:00
* \ param file_priv DRM file private .
2005-04-16 15:20:36 -07:00
* \ param cmd command .
* \ param arg user argument , pointing to a drm_client structure .
2005-09-25 14:28:13 +10:00
*
2005-04-16 15:20:36 -07:00
* \ return zero on success or a negative number on failure .
*
* Searches for the client with the specified index and copies its information
* into userspace
*/
2014-09-10 12:43:55 +02:00
static int drm_getclient ( struct drm_device * dev , void * data ,
2007-09-03 12:06:45 +10:00
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2007-09-03 12:06:45 +10:00
struct drm_client * client = data ;
drm: make DRM_UNLOCKED ioctls with their own mutex
drm_getclient, drm_getstats and drm_getmap (with a few minor
adjustments) do not need global mutex, so fix that and
make the said ioctls DRM_UNLOCKED. Details:
drm_getclient: the only thing that should be protected here
is dev->filelist and that is already protected everywhere with
dev->struct_mutex.
drm_getstats: there is no need for any mutex here because the
loop runs through quasi-static (set at load time only)
data, and the actual count access is done with atomic_read()
drm_getmap already uses dev->struct_mutex to protect
dev->maplist, which also used to protect the same structure
everywhere else except at three places:
* drm_getsarea, which doesn't grab *any* mutex before
touching dev->maplist (so no drm_global_mutex doesn't help
here either; different issue for a different patch).
However, drivers seem to call it only at
initialization time so it probably doesn't matter
* drm_master_destroy, which is called from drm_master_put,
which in turn is protected with dev->struct_mutex
everywhere else in drm module, so we are good here too.
* drm_getsareactx, which releases the dev->struct_mutex
too early, but this patch includes the fix for that.
v2: * incorporate comments received from Daniel Vetter
* include the (long) explanation above to make it clear what
we are doing (and why), also at Daniel Vetter's request
* tighten up mutex grab/release locations to only
encompass real critical sections, rather than some
random code around them
Signed-off-by: Ilija Hadzic <ihadzic@research.bell-labs.com>
Reviewed-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Signed-off-by: Dave Airlie <airlied@redhat.com>
2011-10-28 17:43:28 -04:00
2013-08-08 15:41:31 +02:00
/*
* Hollowed - out getclient ioctl to keep some dead old drm tests / tools
* not breaking completely . Userspace tools stop enumerating one they
* get - EINVAL , hence this is the return value we need to hand back for
* no clients tracked .
*
* Unfortunately some clients ( * cough * libva * cough * ) use this in a fun
* attempt to figure out whether they ' re authenticated or not . Since
* that ' s the only thing they care about , give it to the directly
* instead of walking one giant list .
*/
if ( client - > idx = = 0 ) {
client - > auth = file_priv - > authenticated ;
2016-08-25 16:35:05 +02:00
client - > pid = task_pid_vnr ( current ) ;
client - > uid = overflowuid ;
2013-08-08 15:41:31 +02:00
client - > magic = 0 ;
client - > iocs = 0 ;
return 0 ;
} else {
return - EINVAL ;
2007-05-26 05:01:51 +10:00
}
2005-04-16 15:20:36 -07:00
}
2015-09-08 13:56:27 +02:00
/*
2005-09-25 14:28:13 +10:00
* Get statistics information .
*
2005-04-16 15:20:36 -07:00
* \ param inode device inode .
2007-08-25 20:23:09 +10:00
* \ param file_priv DRM file private .
2005-04-16 15:20:36 -07:00
* \ param cmd command .
* \ param arg user argument , pointing to a drm_stats structure .
2005-09-25 14:28:13 +10:00
*
2005-04-16 15:20:36 -07:00
* \ return zero on success or a negative number on failure .
*/
2014-09-10 12:43:55 +02:00
static int drm_getstats ( struct drm_device * dev , void * data ,
2007-09-03 12:06:45 +10:00
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2007-09-03 12:06:45 +10:00
struct drm_stats * stats = data ;
2005-04-16 15:20:36 -07:00
2013-08-08 15:41:32 +02:00
/* Clear stats to prevent userspace from eating its stack garbage. */
2007-11-05 12:53:09 +10:00
memset ( stats , 0 , sizeof ( * stats ) ) ;
2005-09-25 14:28:13 +10:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
2015-09-08 13:56:27 +02:00
/*
2011-02-21 11:17:35 +10:00
* Get device / driver capabilities
*/
2014-09-10 12:43:55 +02:00
static int drm_getcap ( struct drm_device * dev , void * data , struct drm_file * file_priv )
2011-02-21 11:17:35 +10:00
{
struct drm_get_cap * req = data ;
2016-08-08 16:23:39 +09:00
struct drm_crtc * crtc ;
2011-02-21 11:17:35 +10:00
req - > value = 0 ;
2016-12-01 16:37:31 +09:00
2016-12-07 15:49:39 +01:00
/* Only some caps make sense with UMS/render-only drivers. */
switch ( req - > capability ) {
case DRM_CAP_TIMESTAMP_MONOTONIC :
2016-12-01 16:37:31 +09:00
req - > value = drm_timestamp_monotonic ;
return 0 ;
2016-12-07 15:49:39 +01:00
case DRM_CAP_PRIME :
req - > value | = dev - > driver - > prime_fd_to_handle ? DRM_PRIME_CAP_IMPORT : 0 ;
req - > value | = dev - > driver - > prime_handle_to_fd ? DRM_PRIME_CAP_EXPORT : 0 ;
return 0 ;
2016-12-01 16:37:31 +09:00
}
/* Other caps only work with KMS drivers */
if ( ! drm_core_check_feature ( dev , DRIVER_MODESET ) )
return - ENOTSUPP ;
2011-03-04 14:50:28 +10:00
switch ( req - > capability ) {
case DRM_CAP_DUMB_BUFFER :
if ( dev - > driver - > dumb_create )
req - > value = 1 ;
break ;
2011-03-24 20:54:35 +10:00
case DRM_CAP_VBLANK_HIGH_CRTC :
2011-03-18 16:58:04 -05:00
req - > value = 1 ;
break ;
2011-09-29 16:20:42 +01:00
case DRM_CAP_DUMB_PREFERRED_DEPTH :
req - > value = dev - > mode_config . preferred_depth ;
break ;
case DRM_CAP_DUMB_PREFER_SHADOW :
req - > value = dev - > mode_config . prefer_shadow ;
break ;
2013-07-22 18:50:00 -07:00
case DRM_CAP_ASYNC_PAGE_FLIP :
req - > value = dev - > mode_config . async_page_flip ;
break ;
2016-08-08 16:23:39 +09:00
case DRM_CAP_PAGE_FLIP_TARGET :
2016-12-01 16:37:31 +09:00
req - > value = 1 ;
drm_for_each_crtc ( crtc , dev ) {
if ( ! crtc - > funcs - > page_flip_target )
req - > value = 0 ;
2016-08-08 16:23:39 +09:00
}
break ;
2014-02-12 12:48:23 -05:00
case DRM_CAP_CURSOR_WIDTH :
if ( dev - > mode_config . cursor_width )
req - > value = dev - > mode_config . cursor_width ;
else
req - > value = 64 ;
break ;
case DRM_CAP_CURSOR_HEIGHT :
if ( dev - > mode_config . cursor_height )
req - > value = dev - > mode_config . cursor_height ;
else
req - > value = 64 ;
break ;
2015-02-05 14:41:52 +00:00
case DRM_CAP_ADDFB2_MODIFIERS :
req - > value = dev - > mode_config . allow_fb_modifiers ;
break ;
2011-03-04 14:50:28 +10:00
default :
return - EINVAL ;
}
2011-02-21 11:17:35 +10:00
return 0 ;
}
2015-09-08 13:56:27 +02:00
/*
2013-09-25 16:45:20 +01:00
* Set device / driver capabilities
*/
2014-09-10 12:43:55 +02:00
static int
2013-09-25 16:45:20 +01:00
drm_setclientcap ( struct drm_device * dev , void * data , struct drm_file * file_priv )
{
2013-09-25 16:45:22 +01:00
struct drm_set_client_cap * req = data ;
switch ( req - > capability ) {
case DRM_CLIENT_CAP_STEREO_3D :
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > stereo_allowed = req - > value ;
break ;
2014-04-01 15:22:42 -07:00
case DRM_CLIENT_CAP_UNIVERSAL_PLANES :
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > universal_planes = req - > value ;
break ;
2014-12-18 16:01:50 -05:00
case DRM_CLIENT_CAP_ATOMIC :
if ( ! drm_core_check_feature ( dev , DRIVER_ATOMIC ) )
return - EINVAL ;
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > atomic = req - > value ;
2014-12-19 10:33:52 +01:00
file_priv - > universal_planes = req - > value ;
2014-12-18 16:01:50 -05:00
break ;
2013-09-25 16:45:22 +01:00
default :
return - EINVAL ;
}
return 0 ;
2013-09-25 16:45:20 +01:00
}
2015-09-08 13:56:27 +02:00
/*
2005-04-16 15:20:36 -07:00
* Setversion ioctl .
*
* \ param inode device inode .
2007-08-25 20:23:09 +10:00
* \ param file_priv DRM file private .
2005-04-16 15:20:36 -07:00
* \ param cmd command .
* \ param arg user argument , pointing to a drm_lock structure .
* \ return zero on success or negative number on failure .
*
* Sets the requested interface version
*/
2014-09-10 12:43:55 +02:00
static int drm_setversion ( struct drm_device * dev , void * data , struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2007-09-03 12:06:45 +10:00
struct drm_set_version * sv = data ;
int if_version , retcode = 0 ;
2016-12-10 22:52:52 +01:00
mutex_lock ( & dev - > master_mutex ) ;
2007-09-03 12:06:45 +10:00
if ( sv - > drm_di_major ! = - 1 ) {
if ( sv - > drm_di_major ! = DRM_IF_MAJOR | |
sv - > drm_di_minor < 0 | | sv - > drm_di_minor > DRM_IF_MINOR ) {
retcode = - EINVAL ;
goto done ;
}
if_version = DRM_IF_VERSION ( sv - > drm_di_major ,
sv - > drm_di_minor ) ;
2006-08-07 20:07:43 +10:00
dev - > if_version = max ( if_version , dev - > if_version ) ;
2007-09-03 12:06:45 +10:00
if ( sv - > drm_di_minor > = 1 ) {
2005-04-16 15:20:36 -07:00
/*
* Version 1.1 includes tying of DRM to specific device
2010-08-06 13:55:10 +10:00
* Version 1.4 has proper PCI domain support
2005-04-16 15:20:36 -07:00
*/
2010-08-04 11:09:42 +01:00
retcode = drm_set_busid ( dev , file_priv ) ;
if ( retcode )
goto done ;
2005-04-16 15:20:36 -07:00
}
}
2007-09-03 12:06:45 +10:00
if ( sv - > drm_dd_major ! = - 1 ) {
if ( sv - > drm_dd_major ! = dev - > driver - > major | |
sv - > drm_dd_minor < 0 | | sv - > drm_dd_minor >
dev - > driver - > minor ) {
retcode = - EINVAL ;
goto done ;
}
2005-04-16 15:20:36 -07:00
}
2007-09-03 12:06:45 +10:00
done :
sv - > drm_di_major = DRM_IF_MAJOR ;
sv - > drm_di_minor = DRM_IF_MINOR ;
sv - > drm_dd_major = dev - > driver - > major ;
sv - > drm_dd_minor = dev - > driver - > minor ;
2016-12-10 22:52:52 +01:00
mutex_unlock ( & dev - > master_mutex ) ;
2007-09-03 12:06:45 +10:00
return retcode ;
2005-04-16 15:20:36 -07:00
}
2015-09-08 13:56:27 +02:00
/**
* drm_noop - DRM no - op ioctl implemntation
* @ dev : DRM device for the ioctl
* @ data : data pointer for the ioctl
* @ file_priv : DRM file for the ioctl call
*
* This no - op implementation for drm ioctls is useful for deprecated
* functionality where we can ' t return a failure code because existing userspace
* checks the result of the ioctl , but doesn ' t care about the action .
*
* Always returns successfully with 0.
*/
2007-09-03 12:06:45 +10:00
int drm_noop ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
DRM_DEBUG ( " \n " ) ;
return 0 ;
}
2012-01-17 12:50:12 +01:00
EXPORT_SYMBOL ( drm_noop ) ;
2014-07-23 14:14:53 +02:00
2015-09-08 13:56:27 +02:00
/**
* drm_invalid_op - DRM invalid ioctl implemntation
* @ dev : DRM device for the ioctl
* @ data : data pointer for the ioctl
* @ file_priv : DRM file for the ioctl call
*
* This no - op implementation for drm ioctls is useful for deprecated
* functionality where we really don ' t want to allow userspace to call the ioctl
* any more . This is the case for old ums interfaces for drivers that
* transitioned to kms gradually and so kept the old legacy tables around . This
* only applies to radeon and i915 kms drivers , other drivers shouldn ' t need to
* use this function .
*
* Always fails with a return value of - EINVAL .
*/
2015-09-08 13:56:26 +02:00
int drm_invalid_op ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
{
return - EINVAL ;
}
EXPORT_SYMBOL ( drm_invalid_op ) ;
2015-09-08 13:56:27 +02:00
/*
2014-07-23 14:14:53 +02:00
* Copy and IOCTL return string to user space
*/
static int drm_copy_field ( char __user * buf , size_t * buf_len , const char * value )
{
int len ;
/* don't overflow userbuf */
len = strlen ( value ) ;
if ( len > * buf_len )
len = * buf_len ;
/* let userspace know exact length of driver value (which could be
* larger than the userspace - supplied buffer ) */
* buf_len = strlen ( value ) ;
/* finally, try filling in the userbuf */
if ( len & & buf )
if ( copy_to_user ( buf , value , len ) )
return - EFAULT ;
return 0 ;
}
2015-09-08 13:56:27 +02:00
/*
2014-07-23 14:14:53 +02:00
* Get version information
*
* \ param inode device inode .
* \ param filp file pointer .
* \ param cmd command .
* \ param arg user argument , pointing to a drm_version structure .
* \ return zero on success or negative number on failure .
*
* Fills in the version information in \ p arg .
*/
static int drm_version ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
{
struct drm_version * version = data ;
int err ;
version - > version_major = dev - > driver - > major ;
version - > version_minor = dev - > driver - > minor ;
version - > version_patchlevel = dev - > driver - > patchlevel ;
err = drm_copy_field ( version - > name , & version - > name_len ,
dev - > driver - > name ) ;
if ( ! err )
err = drm_copy_field ( version - > date , & version - > date_len ,
dev - > driver - > date ) ;
if ( ! err )
err = drm_copy_field ( version - > desc , & version - > desc_len ,
dev - > driver - > desc ) ;
return err ;
}
2015-09-08 13:56:27 +02:00
/*
2014-07-23 14:14:53 +02:00
* drm_ioctl_permit - Check ioctl permissions against caller
*
* @ flags : ioctl permission flags .
* @ file_priv : Pointer to struct drm_file identifying the caller .
*
* Checks whether the caller is allowed to run an ioctl with the
* indicated permissions . If so , returns zero . Otherwise returns an
* error code suitable for ioctl return .
*/
2015-07-10 22:31:08 -07:00
int drm_ioctl_permit ( u32 flags , struct drm_file * file_priv )
2014-07-23 14:14:53 +02:00
{
/* ROOT_ONLY is only for CAP_SYS_ADMIN */
if ( unlikely ( ( flags & DRM_ROOT_ONLY ) & & ! capable ( CAP_SYS_ADMIN ) ) )
return - EACCES ;
/* AUTH is only for authenticated or render client */
if ( unlikely ( ( flags & DRM_AUTH ) & & ! drm_is_render_client ( file_priv ) & &
! file_priv - > authenticated ) )
return - EACCES ;
/* MASTER is only for master or control clients */
2016-06-21 10:54:20 +02:00
if ( unlikely ( ( flags & DRM_MASTER ) & &
! drm_is_current_master ( file_priv ) & &
2014-07-23 14:14:53 +02:00
! drm_is_control_client ( file_priv ) ) )
return - EACCES ;
/* Control clients must be explicitly allowed */
if ( unlikely ( ! ( flags & DRM_CONTROL_ALLOW ) & &
drm_is_control_client ( file_priv ) ) )
return - EACCES ;
/* Render clients must be explicitly allowed */
if ( unlikely ( ! ( flags & DRM_RENDER_ALLOW ) & &
drm_is_render_client ( file_priv ) ) )
return - EACCES ;
return 0 ;
}
2015-07-10 22:31:08 -07:00
EXPORT_SYMBOL ( drm_ioctl_permit ) ;
2014-07-23 14:14:53 +02:00
2015-03-30 17:10:36 +00:00
# define DRM_IOCTL_DEF(ioctl, _func, _flags) \
[ DRM_IOCTL_NR ( ioctl ) ] = { \
. cmd = ioctl , \
. func = _func , \
. flags = _flags , \
. name = # ioctl \
}
2014-09-10 12:43:55 +02:00
2015-09-08 13:56:27 +02:00
/* Ioctl table */
2014-09-10 12:43:55 +02:00
static const struct drm_ioctl_desc drm_ioctls [ ] = {
2015-09-15 01:11:42 -07:00
DRM_IOCTL_DEF ( DRM_IOCTL_VERSION , drm_version ,
DRM_UNLOCKED | DRM_RENDER_ALLOW | DRM_CONTROL_ALLOW ) ,
2016-12-10 22:52:52 +01:00
DRM_IOCTL_DEF ( DRM_IOCTL_GET_UNIQUE , drm_getunique , DRM_UNLOCKED ) ,
2016-06-17 09:33:22 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_GET_MAGIC , drm_getmagic , DRM_UNLOCKED ) ,
2014-09-10 12:43:55 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_IRQ_BUSID , drm_irq_by_busid , DRM_MASTER | DRM_ROOT_ONLY ) ,
2016-04-26 19:29:36 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_GET_MAP , drm_legacy_getmap_ioctl , DRM_UNLOCKED ) ,
2014-09-10 12:43:55 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_GET_CLIENT , drm_getclient , DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_STATS , drm_getstats , DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_CAP , drm_getcap , DRM_UNLOCKED | DRM_RENDER_ALLOW ) ,
2016-12-10 22:52:53 +01:00
DRM_IOCTL_DEF ( DRM_IOCTL_SET_CLIENT_CAP , drm_setclientcap , DRM_UNLOCKED ) ,
2016-12-10 22:52:52 +01:00
DRM_IOCTL_DEF ( DRM_IOCTL_SET_VERSION , drm_setversion , DRM_UNLOCKED | DRM_MASTER ) ,
2014-09-10 12:43:55 +02:00
2016-06-21 10:54:17 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_SET_UNIQUE , drm_invalid_op , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2014-09-10 12:43:55 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_BLOCK , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_UNBLOCK , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2016-06-17 09:33:22 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_AUTH_MAGIC , drm_authmagic , DRM_AUTH | DRM_UNLOCKED | DRM_MASTER ) ,
2014-09-10 12:43:55 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_ADD_MAP , drm_legacy_addmap_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_RM_MAP , drm_legacy_rmmap_ioctl , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_SET_SAREA_CTX , drm_legacy_setsareactx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_SAREA_CTX , drm_legacy_getsareactx , DRM_AUTH ) ,
2016-06-17 09:33:23 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_SET_MASTER , drm_setmaster_ioctl , DRM_UNLOCKED | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_DROP_MASTER , drm_dropmaster_ioctl , DRM_UNLOCKED | DRM_ROOT_ONLY ) ,
2014-09-10 12:43:55 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_ADD_CTX , drm_legacy_addctx , DRM_AUTH | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_RM_CTX , drm_legacy_rmctx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MOD_CTX , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_CTX , drm_legacy_getctx , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_SWITCH_CTX , drm_legacy_switchctx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_NEW_CTX , drm_legacy_newctx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_RES_CTX , drm_legacy_resctx , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_ADD_DRAW , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_RM_DRAW , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_LOCK , drm_legacy_lock , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_UNLOCK , drm_legacy_unlock , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_FINISH , drm_noop , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_ADD_BUFS , drm_legacy_addbufs , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MARK_BUFS , drm_legacy_markbufs , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_INFO_BUFS , drm_legacy_infobufs , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MAP_BUFS , drm_legacy_mapbufs , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_FREE_BUFS , drm_legacy_freebufs , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_DMA , drm_legacy_dma_ioctl , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_CONTROL , drm_control , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2015-09-09 16:45:52 +02:00
# if IS_ENABLED(CONFIG_AGP)
2014-09-10 12:43:55 +02:00
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_ACQUIRE , drm_agp_acquire_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_RELEASE , drm_agp_release_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_ENABLE , drm_agp_enable_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_INFO , drm_agp_info_ioctl , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_ALLOC , drm_agp_alloc_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_FREE , drm_agp_free_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_BIND , drm_agp_bind_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_AGP_UNBIND , drm_agp_unbind_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
# endif
DRM_IOCTL_DEF ( DRM_IOCTL_SG_ALLOC , drm_legacy_sg_alloc , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_SG_FREE , drm_legacy_sg_free , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_WAIT_VBLANK , drm_wait_vblank , DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODESET_CTL , drm_modeset_ctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_UPDATE_DRAW , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GEM_CLOSE , drm_gem_close_ioctl , DRM_UNLOCKED | DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GEM_FLINK , drm_gem_flink_ioctl , DRM_AUTH | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GEM_OPEN , drm_gem_open_ioctl , DRM_AUTH | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETRESOURCES , drm_mode_getresources , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_PRIME_HANDLE_TO_FD , drm_prime_handle_to_fd_ioctl , DRM_AUTH | DRM_UNLOCKED | DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_PRIME_FD_TO_HANDLE , drm_prime_fd_to_handle_ioctl , DRM_AUTH | DRM_UNLOCKED | DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPLANERESOURCES , drm_mode_getplane_res , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETCRTC , drm_mode_getcrtc , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETCRTC , drm_mode_setcrtc , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPLANE , drm_mode_getplane , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETPLANE , drm_mode_setplane , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CURSOR , drm_mode_cursor_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETGAMMA , drm_mode_gamma_get_ioctl , DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETGAMMA , drm_mode_gamma_set_ioctl , DRM_MASTER | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETENCODER , drm_mode_getencoder , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETCONNECTOR , drm_mode_getconnector , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ATTACHMODE , drm_noop , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DETACHMODE , drm_noop , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPROPERTY , drm_mode_getproperty_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETPROPERTY , drm_mode_connector_property_set_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPROPBLOB , drm_mode_getblob_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETFB , drm_mode_getfb , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ADDFB , drm_mode_addfb , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ADDFB2 , drm_mode_addfb2 , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_RMFB , drm_mode_rmfb , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_PAGE_FLIP , drm_mode_page_flip_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DIRTYFB , drm_mode_dirtyfb_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CREATE_DUMB , drm_mode_create_dumb_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_MAP_DUMB , drm_mode_mmap_dumb_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DESTROY_DUMB , drm_mode_destroy_dumb_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_OBJ_GETPROPERTIES , drm_mode_obj_get_properties_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_OBJ_SETPROPERTY , drm_mode_obj_set_property_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CURSOR2 , drm_mode_cursor2_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
2014-12-18 16:01:56 -05:00
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ATOMIC , drm_mode_atomic_ioctl , DRM_MASTER | DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
2015-05-22 13:34:51 +01:00
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CREATEPROPBLOB , drm_mode_createblob_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DESTROYPROPBLOB , drm_mode_destroyblob_ioctl , DRM_CONTROL_ALLOW | DRM_UNLOCKED ) ,
2014-09-10 12:43:55 +02:00
} ;
# define DRM_CORE_IOCTL_COUNT ARRAY_SIZE( drm_ioctls )
2014-07-23 14:14:53 +02:00
/**
2015-09-08 13:56:27 +02:00
* drm_ioctl - ioctl callback implementation for DRM drivers
* @ filp : file this ioctl is called on
* @ cmd : ioctl cmd number
* @ arg : user argument
2014-07-23 14:14:53 +02:00
*
* Looks up the ioctl function in the : : ioctls table , checking for root
* previleges if so required , and dispatches to the respective function .
2015-09-08 13:56:27 +02:00
*
* Returns :
* Zero on success , negative error code on failure .
2014-07-23 14:14:53 +02:00
*/
long drm_ioctl ( struct file * filp ,
unsigned int cmd , unsigned long arg )
{
struct drm_file * file_priv = filp - > private_data ;
struct drm_device * dev ;
const struct drm_ioctl_desc * ioctl = NULL ;
drm_ioctl_t * func ;
unsigned int nr = DRM_IOCTL_NR ( cmd ) ;
int retcode = - EINVAL ;
char stack_kdata [ 128 ] ;
char * kdata = NULL ;
2016-07-12 15:59:50 +01:00
unsigned int in_size , out_size , drv_size , ksize ;
2015-09-28 21:42:40 +02:00
bool is_driver_ioctl ;
2014-07-23 14:14:53 +02:00
dev = file_priv - > minor - > dev ;
if ( drm_device_is_unplugged ( dev ) )
return - ENODEV ;
2015-09-28 21:42:40 +02:00
is_driver_ioctl = nr > = DRM_COMMAND_BASE & & nr < DRM_COMMAND_END ;
if ( is_driver_ioctl ) {
2015-03-27 15:51:58 +02:00
/* driver ioctl */
if ( nr - DRM_COMMAND_BASE > = dev - > driver - > num_ioctls )
goto err_i1 ;
2014-07-23 14:14:53 +02:00
ioctl = & dev - > driver - > ioctls [ nr - DRM_COMMAND_BASE ] ;
2015-03-27 15:51:58 +02:00
} else {
/* core ioctl */
if ( nr > = DRM_CORE_IOCTL_COUNT )
goto err_i1 ;
2014-07-23 14:14:53 +02:00
ioctl = & drm_ioctls [ nr ] ;
2015-03-27 15:51:58 +02:00
}
2014-07-23 14:14:53 +02:00
2015-03-27 15:51:58 +02:00
drv_size = _IOC_SIZE ( ioctl - > cmd ) ;
2016-07-12 15:59:50 +01:00
out_size = in_size = _IOC_SIZE ( cmd ) ;
if ( ( cmd & ioctl - > cmd & IOC_IN ) = = 0 )
in_size = 0 ;
if ( ( cmd & ioctl - > cmd & IOC_OUT ) = = 0 )
out_size = 0 ;
ksize = max ( max ( in_size , out_size ) , drv_size ) ;
2014-07-23 14:14:53 +02:00
DRM_DEBUG ( " pid=%d, dev=0x%lx, auth=%d, %s \n " ,
task_pid_nr ( current ) ,
( long ) old_encode_dev ( file_priv - > minor - > kdev - > devt ) ,
file_priv - > authenticated , ioctl - > name ) ;
/* Do not trust userspace, use our own definition */
func = ioctl - > func ;
if ( unlikely ( ! func ) ) {
DRM_DEBUG ( " no function \n " ) ;
retcode = - EINVAL ;
goto err_i1 ;
}
retcode = drm_ioctl_permit ( ioctl - > flags , file_priv ) ;
if ( unlikely ( retcode ) )
goto err_i1 ;
2016-07-12 15:59:50 +01:00
if ( ksize < = sizeof ( stack_kdata ) ) {
kdata = stack_kdata ;
} else {
kdata = kmalloc ( ksize , GFP_KERNEL ) ;
if ( ! kdata ) {
retcode = - ENOMEM ;
goto err_i1 ;
2014-07-23 14:14:53 +02:00
}
}
2016-07-12 15:59:50 +01:00
if ( copy_from_user ( kdata , ( void __user * ) arg , in_size ) ! = 0 ) {
retcode = - EFAULT ;
goto err_i1 ;
2014-07-23 14:14:53 +02:00
}
2016-07-12 15:59:50 +01:00
if ( ksize > in_size )
memset ( kdata + in_size , 0 , ksize - in_size ) ;
2016-12-10 22:52:54 +01:00
/* Enforce sane locking for modern driver ioctls. */
if ( ! drm_core_check_feature ( dev , DRIVER_LEGACY ) | |
2015-09-28 21:42:40 +02:00
( ioctl - > flags & DRM_UNLOCKED ) )
2014-07-23 14:14:53 +02:00
retcode = func ( dev , kdata , file_priv ) ;
else {
mutex_lock ( & drm_global_mutex ) ;
retcode = func ( dev , kdata , file_priv ) ;
mutex_unlock ( & drm_global_mutex ) ;
}
2016-07-12 15:59:50 +01:00
if ( copy_to_user ( ( void __user * ) arg , kdata , out_size ) ! = 0 )
retcode = - EFAULT ;
2014-07-23 14:14:53 +02:00
err_i1 :
if ( ! ioctl )
DRM_DEBUG ( " invalid ioctl: pid=%d, dev=0x%lx, auth=%d, cmd=0x%02x, nr=0x%02x \n " ,
task_pid_nr ( current ) ,
( long ) old_encode_dev ( file_priv - > minor - > kdev - > devt ) ,
file_priv - > authenticated , cmd , nr ) ;
if ( kdata ! = stack_kdata )
kfree ( kdata ) ;
if ( retcode )
DRM_DEBUG ( " ret = %d \n " , retcode ) ;
return retcode ;
}
EXPORT_SYMBOL ( drm_ioctl ) ;
/**
* drm_ioctl_flags - Check for core ioctl and return ioctl permission flags
2015-09-08 13:56:27 +02:00
* @ nr : ioctl number
* @ flags : where to return the ioctl permission flags
*
* This ioctl is only used by the vmwgfx driver to augment the access checks
* done by the drm core and insofar a pretty decent layering violation . This
* shouldn ' t be used by any drivers .
2014-07-23 14:14:53 +02:00
*
2015-09-08 13:56:27 +02:00
* Returns :
2016-06-30 08:47:06 +09:00
* True if the @ nr corresponds to a DRM core ioctl number , false otherwise .
2014-07-23 14:14:53 +02:00
*/
bool drm_ioctl_flags ( unsigned int nr , unsigned int * flags )
{
2015-03-27 15:52:00 +02:00
if ( nr > = DRM_COMMAND_BASE & & nr < DRM_COMMAND_END )
return false ;
if ( nr > = DRM_CORE_IOCTL_COUNT )
return false ;
2014-07-23 14:14:53 +02:00
2015-03-27 15:52:00 +02:00
* flags = drm_ioctls [ nr ] . flags ;
return true ;
2014-07-23 14:14:53 +02:00
}
EXPORT_SYMBOL ( drm_ioctl_flags ) ;