2005-04-17 02:20:36 +04: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 16:14:53 +04:00
* Author Rickard E . ( Rik ) Faith < faith @ valinux . com >
* Author Gareth Hughes < gareth @ valinux . com >
*
2005-04-17 02:20:36 +04: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 21:01:07 +04:00
# include <linux/export.h>
2018-12-20 03:00:15 +03:00
# include <linux/nospec.h>
2019-05-26 20:35:35 +03:00
# include <linux/pci.h>
# include <linux/uaccess.h>
# include <drm/drm_auth.h>
# include <drm/drm_crtc.h>
# include <drm/drm_drv.h>
# include <drm/drm_file.h>
# include <drm/drm_ioctl.h>
# include <drm/drm_print.h>
# include "drm_crtc_internal.h"
# include "drm_internal.h"
# include "drm_legacy.h"
2005-04-17 02:20:36 +04:00
2016-06-21 15:08:33 +03: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
2022-06-21 16:51:40 +03:00
* means the unique name for the master node just opening is _not_ filled
2016-06-21 15:08:33 +03:00
* 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 .
*/
2015-09-08 14:56:27 +03:00
/*
2005-04-17 02:20:36 +04:00
* Get the bus id .
2005-09-25 08:28:13 +04:00
*
2005-04-17 02:20:36 +04:00
* \ param inode device inode .
2007-08-25 14:23:09 +04:00
* \ param file_priv DRM file private .
2005-04-17 02:20:36 +04: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 .
*/
2017-05-24 21:15:20 +03:00
int drm_getunique ( struct drm_device * dev , void * data ,
2007-09-03 06:06:45 +04:00
struct drm_file * file_priv )
2005-04-17 02:20:36 +04:00
{
2007-09-03 06:06:45 +04:00
struct drm_unique * u = data ;
2021-06-08 14:04:36 +03:00
struct drm_master * master ;
2005-04-17 02:20:36 +04:00
2021-06-08 14:04:36 +03:00
mutex_lock ( & dev - > master_mutex ) ;
master = file_priv - > master ;
2008-11-28 07:22:24 +03:00
if ( u - > unique_len > = master - > unique_len ) {
2016-12-11 00:52:52 +03:00
if ( copy_to_user ( u - > unique , master - > unique , master - > unique_len ) ) {
2021-06-08 14:04:36 +03:00
mutex_unlock ( & dev - > master_mutex ) ;
2005-04-17 02:20:36 +04:00
return - EFAULT ;
2016-12-11 00:52:52 +03:00
}
2005-04-17 02:20:36 +04:00
}
2008-11-28 07:22:24 +03:00
u - > unique_len = master - > unique_len ;
2021-06-08 14:04:36 +03:00
mutex_unlock ( & dev - > master_mutex ) ;
2007-09-03 06:06:45 +04:00
2005-04-17 02:20:36 +04:00
return 0 ;
}
2010-08-04 14:09:42 +04: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 07:22:24 +03:00
static int drm_set_busid ( struct drm_device * dev , struct drm_file * file_priv )
2005-04-17 02:20:36 +04:00
{
2008-11-28 07:22:24 +03:00
struct drm_master * master = file_priv - > master ;
2010-12-14 20:16:38 +03:00
int ret ;
2010-08-04 14:09:42 +04:00
if ( master - > unique ! = NULL )
drm_unset_busid ( dev , master ) ;
2006-01-02 13:19:39 +03:00
2017-06-21 16:04:29 +03:00
if ( dev - > dev & & dev_is_pci ( dev - > dev ) ) {
2017-05-24 17:51:39 +03:00
ret = drm_pci_set_busid ( dev , master ) ;
2014-08-29 14:12:43 +04:00
if ( ret ) {
drm_unset_busid ( dev , master ) ;
return ret ;
}
2014-04-11 17:23:00 +04:00
} else {
2016-06-21 11:54:14 +03:00
WARN_ON ( ! dev - > unique ) ;
2014-04-11 17:23:00 +04:00
master - > unique = kstrdup ( dev - > unique , GFP_KERNEL ) ;
if ( master - > unique )
master - > unique_len = strlen ( dev - > unique ) ;
}
2005-04-17 02:20:36 +04:00
return 0 ;
}
2015-09-08 14:56:27 +03:00
/*
2005-04-17 02:20:36 +04:00
* Get client information .
*
* \ param inode device inode .
2007-08-25 14:23:09 +04:00
* \ param file_priv DRM file private .
2005-04-17 02:20:36 +04:00
* \ param cmd command .
* \ param arg user argument , pointing to a drm_client structure .
2005-09-25 08:28:13 +04:00
*
2005-04-17 02:20:36 +04: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
*/
2017-05-25 02:10:32 +03:00
int drm_getclient ( struct drm_device * dev , void * data ,
2007-09-03 06:06:45 +04:00
struct drm_file * file_priv )
2005-04-17 02:20:36 +04:00
{
2007-09-03 06:06:45 +04: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-29 01:43:28 +04:00
2013-08-08 17:41:31 +04: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 17:35:05 +03:00
client - > pid = task_pid_vnr ( current ) ;
client - > uid = overflowuid ;
2013-08-08 17:41:31 +04:00
client - > magic = 0 ;
client - > iocs = 0 ;
return 0 ;
} else {
return - EINVAL ;
2007-05-25 23:01:51 +04:00
}
2005-04-17 02:20:36 +04:00
}
2015-09-08 14:56:27 +03:00
/*
2005-09-25 08:28:13 +04:00
* Get statistics information .
*
2005-04-17 02:20:36 +04:00
* \ param inode device inode .
2007-08-25 14:23:09 +04:00
* \ param file_priv DRM file private .
2005-04-17 02:20:36 +04:00
* \ param cmd command .
* \ param arg user argument , pointing to a drm_stats structure .
2005-09-25 08:28:13 +04:00
*
2005-04-17 02:20:36 +04:00
* \ return zero on success or a negative number on failure .
*/
2014-09-10 14:43:55 +04:00
static int drm_getstats ( struct drm_device * dev , void * data ,
2007-09-03 06:06:45 +04:00
struct drm_file * file_priv )
2005-04-17 02:20:36 +04:00
{
2007-09-03 06:06:45 +04:00
struct drm_stats * stats = data ;
2005-04-17 02:20:36 +04:00
2013-08-08 17:41:32 +04:00
/* Clear stats to prevent userspace from eating its stack garbage. */
2007-11-05 05:53:09 +03:00
memset ( stats , 0 , sizeof ( * stats ) ) ;
2005-09-25 08:28:13 +04:00
2005-04-17 02:20:36 +04:00
return 0 ;
}
2015-09-08 14:56:27 +03:00
/*
2011-02-21 04:17:35 +03:00
* Get device / driver capabilities
*/
2014-09-10 14:43:55 +04:00
static int drm_getcap ( struct drm_device * dev , void * data , struct drm_file * file_priv )
2011-02-21 04:17:35 +03:00
{
struct drm_get_cap * req = data ;
2016-08-08 10:23:39 +03:00
struct drm_crtc * crtc ;
2011-02-21 04:17:35 +03:00
req - > value = 0 ;
2016-12-01 10:37:31 +03:00
2016-12-07 17:49:39 +03:00
/* Only some caps make sense with UMS/render-only drivers. */
switch ( req - > capability ) {
case DRM_CAP_TIMESTAMP_MONOTONIC :
2017-10-11 18:20:13 +03:00
req - > value = 1 ;
2016-12-01 10:37:31 +03:00
return 0 ;
2016-12-07 17:49:39 +03: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 ;
2017-04-04 06:26:24 +03:00
case DRM_CAP_SYNCOBJ :
req - > value = drm_core_check_feature ( dev , DRIVER_SYNCOBJ ) ;
return 0 ;
2019-04-16 15:57:50 +03:00
case DRM_CAP_SYNCOBJ_TIMELINE :
req - > value = drm_core_check_feature ( dev , DRIVER_SYNCOBJ_TIMELINE ) ;
return 0 ;
2016-12-01 10:37:31 +03:00
}
/* Other caps only work with KMS drivers */
if ( ! drm_core_check_feature ( dev , DRIVER_MODESET ) )
2018-09-13 22:20:50 +03:00
return - EOPNOTSUPP ;
2016-12-01 10:37:31 +03:00
2011-03-04 07:50:28 +03:00
switch ( req - > capability ) {
case DRM_CAP_DUMB_BUFFER :
if ( dev - > driver - > dumb_create )
req - > value = 1 ;
break ;
2011-03-24 13:54:35 +03:00
case DRM_CAP_VBLANK_HIGH_CRTC :
2011-03-19 00:58:04 +03:00
req - > value = 1 ;
break ;
2011-09-29 19:20:42 +04: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-23 05:50:00 +04:00
case DRM_CAP_ASYNC_PAGE_FLIP :
req - > value = dev - > mode_config . async_page_flip ;
break ;
2016-08-08 10:23:39 +03:00
case DRM_CAP_PAGE_FLIP_TARGET :
2016-12-01 10:37:31 +03:00
req - > value = 1 ;
drm_for_each_crtc ( crtc , dev ) {
if ( ! crtc - > funcs - > page_flip_target )
req - > value = 0 ;
2016-08-08 10:23:39 +03:00
}
break ;
2014-02-12 21:48:23 +04: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 17:41:52 +03:00
case DRM_CAP_ADDFB2_MODIFIERS :
2022-01-28 09:08:34 +03:00
req - > value = ! dev - > mode_config . fb_modifiers_not_supported ;
2015-02-05 17:41:52 +03:00
break ;
2017-04-04 19:52:21 +03:00
case DRM_CAP_CRTC_IN_VBLANK_EVENT :
req - > value = 1 ;
break ;
2011-03-04 07:50:28 +03:00
default :
return - EINVAL ;
}
2011-02-21 04:17:35 +03:00
return 0 ;
}
2015-09-08 14:56:27 +03:00
/*
2013-09-25 19:45:20 +04:00
* Set device / driver capabilities
*/
2014-09-10 14:43:55 +04:00
static int
2013-09-25 19:45:20 +04:00
drm_setclientcap ( struct drm_device * dev , void * data , struct drm_file * file_priv )
{
2013-09-25 19:45:22 +04:00
struct drm_set_client_cap * req = data ;
2018-09-18 20:48:09 +03:00
/* No render-only settable capabilities for now */
/* Below caps that only works with KMS drivers */
if ( ! drm_core_check_feature ( dev , DRIVER_MODESET ) )
return - EOPNOTSUPP ;
2013-09-25 19:45:22 +04:00
switch ( req - > capability ) {
case DRM_CLIENT_CAP_STEREO_3D :
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > stereo_allowed = req - > value ;
break ;
2014-04-02 02:22:42 +04:00
case DRM_CLIENT_CAP_UNIVERSAL_PLANES :
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > universal_planes = req - > value ;
break ;
2014-12-19 00:01:50 +03:00
case DRM_CLIENT_CAP_ATOMIC :
if ( ! drm_core_check_feature ( dev , DRIVER_ATOMIC ) )
2018-09-13 22:20:50 +03:00
return - EOPNOTSUPP ;
2019-09-05 21:53:18 +03:00
/* The modesetting DDX has a totally broken idea of atomic. */
if ( current - > comm [ 0 ] = = ' X ' & & req - > value = = 1 ) {
pr_info ( " broken atomic modeset userspace detected, disabling atomic \n " ) ;
return - EOPNOTSUPP ;
}
if ( req - > value > 2 )
2014-12-19 00:01:50 +03:00
return - EINVAL ;
file_priv - > atomic = req - > value ;
2014-12-19 12:33:52 +03:00
file_priv - > universal_planes = req - > value ;
2018-05-08 14:09:41 +03:00
/*
* No atomic user - space blows up on aspect ratio mode bits .
*/
file_priv - > aspect_ratio_allowed = req - > value ;
break ;
case DRM_CLIENT_CAP_ASPECT_RATIO :
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > aspect_ratio_allowed = req - > value ;
2014-12-19 00:01:50 +03:00
break ;
2018-02-28 17:11:23 +03:00
case DRM_CLIENT_CAP_WRITEBACK_CONNECTORS :
if ( ! file_priv - > atomic )
return - EINVAL ;
if ( req - > value > 1 )
return - EINVAL ;
file_priv - > writeback_connectors = req - > value ;
break ;
2013-09-25 19:45:22 +04:00
default :
return - EINVAL ;
}
return 0 ;
2013-09-25 19:45:20 +04:00
}
2015-09-08 14:56:27 +03:00
/*
2005-04-17 02:20:36 +04:00
* Setversion ioctl .
*
* \ param inode device inode .
2007-08-25 14:23:09 +04:00
* \ param file_priv DRM file private .
2005-04-17 02:20:36 +04: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 14:43:55 +04:00
static int drm_setversion ( struct drm_device * dev , void * data , struct drm_file * file_priv )
2005-04-17 02:20:36 +04:00
{
2007-09-03 06:06:45 +04:00
struct drm_set_version * sv = data ;
int if_version , retcode = 0 ;
2016-12-11 00:52:52 +03:00
mutex_lock ( & dev - > master_mutex ) ;
2007-09-03 06:06:45 +04: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 14:07:43 +04:00
dev - > if_version = max ( if_version , dev - > if_version ) ;
2007-09-03 06:06:45 +04:00
if ( sv - > drm_di_minor > = 1 ) {
2005-04-17 02:20:36 +04:00
/*
* Version 1.1 includes tying of DRM to specific device
2010-08-06 07:55:10 +04:00
* Version 1.4 has proper PCI domain support
2005-04-17 02:20:36 +04:00
*/
2010-08-04 14:09:42 +04:00
retcode = drm_set_busid ( dev , file_priv ) ;
if ( retcode )
goto done ;
2005-04-17 02:20:36 +04:00
}
}
2007-09-03 06:06:45 +04: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-17 02:20:36 +04:00
}
2007-09-03 06:06:45 +04: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-11 00:52:52 +03:00
mutex_unlock ( & dev - > master_mutex ) ;
2007-09-03 06:06:45 +04:00
return retcode ;
2005-04-17 02:20:36 +04:00
}
2015-09-08 14:56:27 +03:00
/**
2021-07-30 16:27:29 +03:00
* drm_noop - DRM no - op ioctl implementation
2015-09-08 14:56:27 +03:00
* @ 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 06:06:45 +04:00
int drm_noop ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2005-04-17 02:20:36 +04:00
{
2022-12-23 14:23:02 +03:00
drm_dbg_core ( dev , " \n " ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2012-01-17 15:50:12 +04:00
EXPORT_SYMBOL ( drm_noop ) ;
2014-07-23 16:14:53 +04:00
2015-09-08 14:56:27 +03:00
/**
2021-07-30 16:27:29 +03:00
* drm_invalid_op - DRM invalid ioctl implementation
2015-09-08 14:56:27 +03:00
* @ 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 14:56:26 +03: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 14:56:27 +03:00
/*
2014-07-23 16:14:53 +04:00
* Copy and IOCTL return string to user space
*/
static int drm_copy_field ( char __user * buf , size_t * buf_len , const char * value )
{
2022-07-05 13:02:13 +03:00
size_t len ;
2014-07-23 16:14:53 +04:00
2022-07-05 13:02:14 +03:00
/* don't attempt to copy a NULL pointer */
if ( WARN_ONCE ( ! value , " BUG: the value to copy was not set! " ) ) {
* buf_len = 0 ;
return 0 ;
}
2014-07-23 16:14:53 +04:00
/* 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 14:56:27 +03:00
/*
2014-07-23 16:14:53 +04: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 .
*/
2017-05-24 21:11:03 +03:00
int drm_version ( struct drm_device * dev , void * data ,
2014-07-23 16:14:53 +04:00
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 ;
}
2021-08-13 11:54:50 +03:00
static int drm_ioctl_permit ( u32 flags , struct drm_file * file_priv )
2014-07-23 16:14:53 +04:00
{
/* ROOT_ONLY is only for CAP_SYS_ADMIN */
if ( unlikely ( ( flags & DRM_ROOT_ONLY ) & & ! capable ( CAP_SYS_ADMIN ) ) )
return - EACCES ;
2019-04-17 23:46:33 +03:00
/* AUTH is only for authenticated or render client */
if ( unlikely ( ( flags & DRM_AUTH ) & & ! drm_is_render_client ( file_priv ) & &
! file_priv - > authenticated ) )
return - EACCES ;
2014-07-23 16:14:53 +04:00
/* MASTER is only for master or control clients */
2018-01-04 01:21:04 +03:00
if ( unlikely ( ( flags & DRM_MASTER ) & &
2018-04-20 09:51:59 +03:00
! drm_is_current_master ( file_priv ) ) )
2014-07-23 16:14:53 +04:00
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-03-30 20:10:36 +03:00
# define DRM_IOCTL_DEF(ioctl, _func, _flags) \
[ DRM_IOCTL_NR ( ioctl ) ] = { \
. cmd = ioctl , \
. func = _func , \
. flags = _flags , \
. name = # ioctl \
}
2014-09-10 14:43:55 +04:00
2019-04-18 10:10:40 +03:00
# if IS_ENABLED(CONFIG_DRM_LEGACY)
# define DRM_LEGACY_IOCTL_DEF(ioctl, _func, _flags) DRM_IOCTL_DEF(ioctl, _func, _flags)
# else
# define DRM_LEGACY_IOCTL_DEF(ioctl, _func, _flags) DRM_IOCTL_DEF(ioctl, drm_invalid_op, _flags)
# endif
2015-09-08 14:56:27 +03:00
/* Ioctl table */
2014-09-10 14:43:55 +04:00
static const struct drm_ioctl_desc drm_ioctls [ ] = {
2019-06-05 15:08:35 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_VERSION , drm_version , DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_UNIQUE , drm_getunique , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_MAGIC , drm_getmagic , 0 ) ,
2021-05-16 22:51:35 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_IRQ_BUSID , drm_legacy_irq_by_busid ,
DRM_MASTER | DRM_ROOT_ONLY ) ,
2019-04-18 10:10:40 +03:00
2019-06-05 15:08:35 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_GET_MAP , drm_legacy_getmap_ioctl , 0 ) ,
2019-04-18 10:10:40 +03:00
2019-06-05 15:08:35 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_GET_CLIENT , drm_getclient , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_STATS , drm_getstats , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GET_CAP , drm_getcap , DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_SET_CLIENT_CAP , drm_setclientcap , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_SET_VERSION , drm_setversion , DRM_MASTER ) ,
2014-09-10 14:43:55 +04:00
2016-06-21 11:54:17 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SET_UNIQUE , drm_invalid_op , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2014-09-10 14:43:55 +04: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 ) ,
2019-06-05 15:08:35 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_AUTH_MAGIC , drm_authmagic , DRM_MASTER ) ,
2014-09-10 14:43:55 +04:00
2019-04-18 10:10:40 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_ADD_MAP , drm_legacy_addmap_ioctl , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_RM_MAP , drm_legacy_rmmap_ioctl , DRM_AUTH ) ,
2014-09-10 14:43:55 +04:00
2019-04-18 10:10:40 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_SET_SAREA_CTX , drm_legacy_setsareactx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_GET_SAREA_CTX , drm_legacy_getsareactx , DRM_AUTH ) ,
2014-09-10 14:43:55 +04:00
2020-03-19 20:29:29 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SET_MASTER , drm_setmaster_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_DROP_MASTER , drm_dropmaster_ioctl , 0 ) ,
2014-09-10 14:43:55 +04:00
2019-04-18 10:10:40 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_ADD_CTX , drm_legacy_addctx , DRM_AUTH | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_RM_CTX , drm_legacy_rmctx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_MOD_CTX , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_GET_CTX , drm_legacy_getctx , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_SWITCH_CTX , drm_legacy_switchctx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_NEW_CTX , drm_legacy_newctx , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_RES_CTX , drm_legacy_resctx , DRM_AUTH ) ,
2014-09-10 14:43:55 +04:00
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 ) ,
2019-04-18 10:10:40 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_LOCK , drm_legacy_lock , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_UNLOCK , drm_legacy_unlock , DRM_AUTH ) ,
2014-09-10 14:43:55 +04:00
DRM_IOCTL_DEF ( DRM_IOCTL_FINISH , drm_noop , DRM_AUTH ) ,
2019-04-18 10:10:40 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_ADD_BUFS , drm_legacy_addbufs , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_MARK_BUFS , drm_legacy_markbufs , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_INFO_BUFS , drm_legacy_infobufs , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_MAP_BUFS , drm_legacy_mapbufs , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_FREE_BUFS , drm_legacy_freebufs , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_DMA , drm_legacy_dma_ioctl , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_CONTROL , drm_legacy_irq_control , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2014-09-10 14:43:55 +04:00
2015-09-09 17:45:52 +03:00
# if IS_ENABLED(CONFIG_AGP)
2021-05-07 21:57:09 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_ACQUIRE , drm_legacy_agp_acquire_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_RELEASE , drm_legacy_agp_release_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_ENABLE , drm_legacy_agp_enable_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_INFO , drm_legacy_agp_info_ioctl , DRM_AUTH ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_ALLOC , drm_legacy_agp_alloc_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_FREE , drm_legacy_agp_free_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_BIND , drm_legacy_agp_bind_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_AGP_UNBIND , drm_legacy_agp_unbind_ioctl ,
DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2014-09-10 14:43:55 +04:00
# endif
2019-04-18 10:10:40 +03:00
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_SG_ALLOC , drm_legacy_sg_alloc , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
DRM_LEGACY_IOCTL_DEF ( DRM_IOCTL_SG_FREE , drm_legacy_sg_free , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2014-09-10 14:43:55 +04:00
2017-05-24 17:51:46 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_WAIT_VBLANK , drm_wait_vblank_ioctl , DRM_UNLOCKED ) ,
2014-09-10 14:43:55 +04:00
2017-05-24 17:51:46 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_MODESET_CTL , drm_legacy_modeset_ctl_ioctl , 0 ) ,
2014-09-10 14:43:55 +04:00
DRM_IOCTL_DEF ( DRM_IOCTL_UPDATE_DRAW , drm_noop , DRM_AUTH | DRM_MASTER | DRM_ROOT_ONLY ) ,
2019-06-05 15:08:35 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_GEM_CLOSE , drm_gem_close_ioctl , DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GEM_FLINK , drm_gem_flink_ioctl , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_GEM_OPEN , drm_gem_open_ioctl , DRM_AUTH ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETRESOURCES , drm_mode_getresources , 0 ) ,
2019-11-01 16:03:13 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_PRIME_HANDLE_TO_FD , drm_prime_handle_to_fd_ioctl , DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_PRIME_FD_TO_HANDLE , drm_prime_fd_to_handle_ioctl , DRM_RENDER_ALLOW ) ,
2019-06-05 15:08:35 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPLANERESOURCES , drm_mode_getplane_res , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETCRTC , drm_mode_getcrtc , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETCRTC , drm_mode_setcrtc , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPLANE , drm_mode_getplane , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETPLANE , drm_mode_setplane , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CURSOR , drm_mode_cursor_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETGAMMA , drm_mode_gamma_get_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETGAMMA , drm_mode_gamma_set_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETENCODER , drm_mode_getencoder , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETCONNECTOR , drm_mode_getconnector , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ATTACHMODE , drm_noop , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DETACHMODE , drm_noop , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPROPERTY , drm_mode_getproperty_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_SETPROPERTY , drm_connector_property_set_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETPROPBLOB , drm_mode_getblob_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETFB , drm_mode_getfb , 0 ) ,
2019-12-17 06:46:43 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GETFB2 , drm_mode_getfb2_ioctl , 0 ) ,
2019-06-05 15:08:35 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ADDFB , drm_mode_addfb_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ADDFB2 , drm_mode_addfb2_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_RMFB , drm_mode_rmfb_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_PAGE_FLIP , drm_mode_page_flip_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DIRTYFB , drm_mode_dirtyfb_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CREATE_DUMB , drm_mode_create_dumb_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_MAP_DUMB , drm_mode_mmap_dumb_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DESTROY_DUMB , drm_mode_destroy_dumb_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_OBJ_GETPROPERTIES , drm_mode_obj_get_properties_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_OBJ_SETPROPERTY , drm_mode_obj_set_property_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CURSOR2 , drm_mode_cursor2_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_ATOMIC , drm_mode_atomic_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CREATEPROPBLOB , drm_mode_createblob_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_DESTROYPROPBLOB , drm_mode_destroyblob_ioctl , 0 ) ,
2017-04-04 06:26:24 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_CREATE , drm_syncobj_create_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2017-04-04 06:26:24 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_DESTROY , drm_syncobj_destroy_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2017-04-04 06:26:24 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_HANDLE_TO_FD , drm_syncobj_handle_to_fd_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2017-04-04 06:26:24 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_FD_TO_HANDLE , drm_syncobj_fd_to_handle_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2019-04-01 12:51:01 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_TRANSFER , drm_syncobj_transfer_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2017-08-25 20:52:22 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_WAIT , drm_syncobj_wait_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2019-04-01 12:50:57 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_TIMELINE_WAIT , drm_syncobj_timeline_wait_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2017-08-29 00:10:27 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_RESET , drm_syncobj_reset_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2017-08-29 00:10:28 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_SIGNAL , drm_syncobj_signal_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2019-04-01 12:51:02 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_TIMELINE_SIGNAL , drm_syncobj_timeline_signal_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
2019-04-01 12:50:58 +03:00
DRM_IOCTL_DEF ( DRM_IOCTL_SYNCOBJ_QUERY , drm_syncobj_query_ioctl ,
2019-06-05 15:08:35 +03:00
DRM_RENDER_ALLOW ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_CRTC_GET_SEQUENCE , drm_crtc_get_sequence_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_CRTC_QUEUE_SEQUENCE , drm_crtc_queue_sequence_ioctl , 0 ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_CREATE_LEASE , drm_mode_create_lease_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_LIST_LESSEES , drm_mode_list_lessees_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_GET_LEASE , drm_mode_get_lease_ioctl , DRM_MASTER ) ,
DRM_IOCTL_DEF ( DRM_IOCTL_MODE_REVOKE_LEASE , drm_mode_revoke_lease_ioctl , DRM_MASTER ) ,
2014-09-10 14:43:55 +04:00
} ;
2021-08-28 15:39:42 +03:00
# define DRM_CORE_IOCTL_COUNT ARRAY_SIZE(drm_ioctls)
2014-09-10 14:43:55 +04:00
2017-04-04 12:52:57 +03:00
/**
* DOC : driver specific ioctls
*
* First things first , driver private IOCTLs should only be needed for drivers
* supporting rendering . Kernel modesetting is all standardized , and extended
* through properties . There are a few exceptions in some existing drivers ,
* which define IOCTL for use by the display DRM master , but they all predate
* properties .
*
* Now if you do have a render driver you always have to support it through
* driver private properties . There ' s a few steps needed to wire all the things
* up .
*
* First you need to define the structure for your IOCTL in your driver private
* UAPI header in ` ` include / uapi / drm / my_driver_drm . h ` ` : :
*
* struct my_driver_operation {
* u32 some_thing ;
* u32 another_thing ;
* } ;
*
* Please make sure that you follow all the best practices from
2020-04-14 19:48:35 +03:00
* ` ` Documentation / process / botching - up - ioctls . rst ` ` . Note that drm_ioctl ( )
2017-04-04 12:52:57 +03:00
* automatically zero - extends structures , hence make sure you can add more stuff
* at the end , i . e . don ' t put a variable sized array there .
*
* Then you need to define your IOCTL number , using one of DRM_IO ( ) , DRM_IOR ( ) ,
* DRM_IOW ( ) or DRM_IOWR ( ) . It must start with the DRM_IOCTL \ _ prefix : :
*
* # # define DRM_IOCTL_MY_DRIVER_OPERATION \
* DRM_IOW ( DRM_COMMAND_BASE , struct my_driver_operation )
2018-01-04 01:21:04 +03:00
*
2017-04-04 12:52:57 +03:00
* DRM driver private IOCTL must be in the range from DRM_COMMAND_BASE to
* DRM_COMMAND_END . Finally you need an array of & struct drm_ioctl_desc to wire
2017-05-31 12:20:45 +03:00
* up the handlers and set the access rights : :
2017-04-04 12:52:57 +03:00
*
* static const struct drm_ioctl_desc my_driver_ioctls [ ] = {
* DRM_IOCTL_DEF_DRV ( MY_DRIVER_OPERATION , my_driver_operation ,
* DRM_AUTH | DRM_RENDER_ALLOW ) ,
* } ;
*
* And then assign this to the & drm_driver . ioctls field in your driver
* structure .
2017-05-31 12:20:45 +03:00
*
* See the separate chapter on : ref : ` file operations < drm_driver_fops > ` for how
* the driver - specific IOCTLs are wired up .
2017-04-04 12:52:57 +03:00
*/
2017-05-24 19:31:12 +03:00
long drm_ioctl_kernel ( struct file * file , drm_ioctl_t * func , void * kdata ,
u32 flags )
{
struct drm_file * file_priv = file - > private_data ;
struct drm_device * dev = file_priv - > minor - > dev ;
int retcode ;
2017-08-02 14:56:02 +03:00
if ( drm_dev_is_unplugged ( dev ) )
2017-05-24 19:31:12 +03:00
return - ENODEV ;
retcode = drm_ioctl_permit ( flags , file_priv ) ;
if ( unlikely ( retcode ) )
return retcode ;
/* Enforce sane locking for modern driver ioctls. */
2019-06-05 15:08:35 +03:00
if ( likely ( ! drm_core_check_feature ( dev , DRIVER_LEGACY ) ) | |
2017-05-24 19:31:12 +03:00
( flags & DRM_UNLOCKED ) )
retcode = func ( dev , kdata , file_priv ) ;
else {
mutex_lock ( & drm_global_mutex ) ;
retcode = func ( dev , kdata , file_priv ) ;
mutex_unlock ( & drm_global_mutex ) ;
}
return retcode ;
}
2017-06-04 04:33:26 +03:00
EXPORT_SYMBOL ( drm_ioctl_kernel ) ;
2017-05-24 19:31:12 +03:00
2014-07-23 16:14:53 +04:00
/**
2015-09-08 14:56:27 +03: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 16:14:53 +04:00
*
2017-04-04 12:52:57 +03:00
* Looks up the ioctl function in the DRM core and the driver dispatch table ,
* stored in & drm_driver . ioctls . It checks for necessary permission by calling
* drm_ioctl_permit ( ) , and dispatches to the respective function .
2015-09-08 14:56:27 +03:00
*
* Returns :
* Zero on success , negative error code on failure .
2014-07-23 16:14:53 +04: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 17:59:50 +03:00
unsigned int in_size , out_size , drv_size , ksize ;
2015-09-28 22:42:40 +03:00
bool is_driver_ioctl ;
2014-07-23 16:14:53 +04:00
dev = file_priv - > minor - > dev ;
2017-08-02 14:56:02 +03:00
if ( drm_dev_is_unplugged ( dev ) )
2014-07-23 16:14:53 +04:00
return - ENODEV ;
2021-08-28 15:39:42 +03:00
if ( DRM_IOCTL_TYPE ( cmd ) ! = DRM_IOCTL_BASE )
return - ENOTTY ;
2021-07-16 19:43:12 +03:00
2015-09-28 22:42:40 +03:00
is_driver_ioctl = nr > = DRM_COMMAND_BASE & & nr < DRM_COMMAND_END ;
if ( is_driver_ioctl ) {
2015-03-27 16:51:58 +03:00
/* driver ioctl */
2018-12-20 03:00:15 +03:00
unsigned int index = nr - DRM_COMMAND_BASE ;
if ( index > = dev - > driver - > num_ioctls )
2015-03-27 16:51:58 +03:00
goto err_i1 ;
2018-12-20 03:00:15 +03:00
index = array_index_nospec ( index , dev - > driver - > num_ioctls ) ;
ioctl = & dev - > driver - > ioctls [ index ] ;
2015-03-27 16:51:58 +03:00
} else {
/* core ioctl */
if ( nr > = DRM_CORE_IOCTL_COUNT )
goto err_i1 ;
2018-12-20 03:00:15 +03:00
nr = array_index_nospec ( nr , DRM_CORE_IOCTL_COUNT ) ;
2014-07-23 16:14:53 +04:00
ioctl = & drm_ioctls [ nr ] ;
2015-03-27 16:51:58 +03:00
}
2014-07-23 16:14:53 +04:00
2015-03-27 16:51:58 +03:00
drv_size = _IOC_SIZE ( ioctl - > cmd ) ;
2016-07-12 17:59:50 +03: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 16:14:53 +04:00
2022-12-23 14:23:02 +03:00
drm_dbg_core ( dev , " comm= \" %s \" pid=%d, dev=0x%lx, auth=%d, %s \n " ,
current - > comm , task_pid_nr ( current ) ,
( long ) old_encode_dev ( file_priv - > minor - > kdev - > devt ) ,
file_priv - > authenticated , ioctl - > name ) ;
2014-07-23 16:14:53 +04:00
/* Do not trust userspace, use our own definition */
func = ioctl - > func ;
if ( unlikely ( ! func ) ) {
2022-12-23 14:23:02 +03:00
drm_dbg_core ( dev , " no function \n " ) ;
2014-07-23 16:14:53 +04:00
retcode = - EINVAL ;
goto err_i1 ;
}
2016-07-12 17:59:50 +03: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 16:14:53 +04:00
}
}
2016-07-12 17:59:50 +03:00
if ( copy_from_user ( kdata , ( void __user * ) arg , in_size ) ! = 0 ) {
retcode = - EFAULT ;
goto err_i1 ;
2014-07-23 16:14:53 +04:00
}
2016-07-12 17:59:50 +03:00
if ( ksize > in_size )
memset ( kdata + in_size , 0 , ksize - in_size ) ;
2017-05-24 19:31:12 +03:00
retcode = drm_ioctl_kernel ( filp , func , kdata , ioctl - > flags ) ;
2016-07-12 17:59:50 +03:00
if ( copy_to_user ( ( void __user * ) arg , kdata , out_size ) ! = 0 )
retcode = - EFAULT ;
2014-07-23 16:14:53 +04:00
err_i1 :
if ( ! ioctl )
2022-12-23 14:23:02 +03:00
drm_dbg_core ( dev ,
" invalid ioctl: comm= \" %s \" , pid=%d, dev=0x%lx, auth=%d, cmd=0x%02x, nr=0x%02x \n " ,
current - > comm , task_pid_nr ( current ) ,
( long ) old_encode_dev ( file_priv - > minor - > kdev - > devt ) ,
file_priv - > authenticated , cmd , nr ) ;
2014-07-23 16:14:53 +04:00
if ( kdata ! = stack_kdata )
kfree ( kdata ) ;
if ( retcode )
2022-12-23 14:23:02 +03:00
drm_dbg_core ( dev , " comm= \" %s \" , pid=%d, ret=%d \n " ,
current - > comm , task_pid_nr ( current ) , retcode ) ;
2014-07-23 16:14:53 +04:00
return retcode ;
}
EXPORT_SYMBOL ( drm_ioctl ) ;
/**
* drm_ioctl_flags - Check for core ioctl and return ioctl permission flags
2015-09-08 14:56:27 +03: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 16:14:53 +04:00
*
2015-09-08 14:56:27 +03:00
* Returns :
2016-06-30 02:47:06 +03:00
* True if the @ nr corresponds to a DRM core ioctl number , false otherwise .
2014-07-23 16:14:53 +04:00
*/
bool drm_ioctl_flags ( unsigned int nr , unsigned int * flags )
{
2015-03-27 16:52:00 +03:00
if ( nr > = DRM_COMMAND_BASE & & nr < DRM_COMMAND_END )
return false ;
if ( nr > = DRM_CORE_IOCTL_COUNT )
return false ;
2018-12-20 03:00:15 +03:00
nr = array_index_nospec ( nr , DRM_CORE_IOCTL_COUNT ) ;
2014-07-23 16:14:53 +04:00
2015-03-27 16:52:00 +03:00
* flags = drm_ioctls [ nr ] . flags ;
return true ;
2014-07-23 16:14:53 +04:00
}
EXPORT_SYMBOL ( drm_ioctl_flags ) ;