2005-04-17 02:20:36 +04:00
/*
* Created : Tue Feb 2 08 : 37 : 54 1999 by faith @ valinux . com
*
* Copyright 1999 Precision Insight , Inc . , Cedar Park , Texas .
* Copyright 2000 VA Linux Systems , Inc . , Sunnyvale , California .
* All Rights Reserved .
*
2015-05-04 22:01:30 +03: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 <drm/drmP.h>
2014-09-10 14:43:53 +04:00
# include "drm_internal.h"
2016-06-14 21:50:59 +03:00
# include "drm_legacy.h"
2005-04-17 02:20:36 +04:00
/**
2015-05-04 22:01:30 +03:00
* drm_getmagic - Get unique magic of a client
* @ dev : DRM device to operate on
* @ data : ioctl data containing the drm_auth object
* @ file_priv : DRM file that performs the operation
2005-04-17 02:20:36 +04:00
*
2015-05-04 22:01:30 +03:00
* This looks up the unique magic of the passed client and returns it . If the
* client did not have a magic assigned , yet , a new one is registered . The magic
* is stored in the passed drm_auth object .
2005-04-17 02:20:36 +04:00
*
2015-05-04 22:01:30 +03:00
* Returns : 0 on success , negative error code on failure .
2005-04-17 02:20:36 +04:00
*/
2007-09-03 06:06:45 +04:00
int drm_getmagic ( 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_auth * auth = data ;
2015-05-04 22:01:30 +03:00
int ret = 0 ;
2005-04-17 02:20:36 +04:00
2015-05-04 22:01:30 +03:00
mutex_lock ( & dev - > struct_mutex ) ;
if ( ! file_priv - > magic ) {
ret = idr_alloc ( & file_priv - > master - > magic_map , file_priv ,
1 , 0 , GFP_KERNEL ) ;
if ( ret > = 0 )
file_priv - > magic = ret ;
2005-04-17 02:20:36 +04:00
}
2015-05-04 22:01:30 +03:00
auth - > magic = file_priv - > magic ;
mutex_unlock ( & dev - > struct_mutex ) ;
2005-04-17 02:20:36 +04:00
2007-09-03 06:06:45 +04:00
DRM_DEBUG ( " %u \n " , auth - > magic ) ;
2015-05-04 22:01:30 +03:00
return ret < 0 ? ret : 0 ;
2005-04-17 02:20:36 +04:00
}
/**
2015-05-04 22:01:30 +03:00
* drm_authmagic - Authenticate client with a magic
* @ dev : DRM device to operate on
* @ data : ioctl data containing the drm_auth object
* @ file_priv : DRM file that performs the operation
2005-04-17 02:20:36 +04:00
*
2015-05-04 22:01:30 +03:00
* This looks up a DRM client by the passed magic and authenticates it .
2005-04-17 02:20:36 +04:00
*
2015-05-04 22:01:30 +03:00
* Returns : 0 on success , negative error code on failure .
2005-04-17 02:20:36 +04:00
*/
2007-09-03 06:06:45 +04:00
int drm_authmagic ( 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_auth * auth = data ;
2007-07-11 09:53:27 +04:00
struct drm_file * file ;
2005-04-17 02:20:36 +04:00
2007-09-03 06:06:45 +04:00
DRM_DEBUG ( " %u \n " , auth - > magic ) ;
2015-05-04 22:01:30 +03:00
mutex_lock ( & dev - > struct_mutex ) ;
file = idr_find ( & file_priv - > master - > magic_map , auth - > magic ) ;
if ( file ) {
2005-04-17 02:20:36 +04:00
file - > authenticated = 1 ;
2015-05-04 22:01:30 +03:00
idr_replace ( & file_priv - > master - > magic_map , NULL , auth - > magic ) ;
2005-04-17 02:20:36 +04:00
}
2015-05-04 22:01:30 +03:00
mutex_unlock ( & dev - > struct_mutex ) ;
return file ? 0 : - EINVAL ;
2005-04-17 02:20:36 +04:00
}
2016-06-14 21:50:59 +03:00
static struct drm_master * drm_master_create ( struct drm_device * dev )
{
struct drm_master * master ;
master = kzalloc ( sizeof ( * master ) , GFP_KERNEL ) ;
if ( ! master )
return NULL ;
kref_init ( & master - > refcount ) ;
spin_lock_init ( & master - > lock . spinlock ) ;
init_waitqueue_head ( & master - > lock . lock_queue ) ;
idr_init ( & master - > magic_map ) ;
master - > dev = dev ;
return master ;
}
/*
* drm_new_set_master - Allocate a new master object and become master for the
* associated master realm .
*
* @ dev : The associated device .
* @ fpriv : File private identifying the client .
*
* This function must be called with dev : : struct_mutex held .
* Returns negative error code on failure . Zero on success .
*/
int drm_new_set_master ( struct drm_device * dev , struct drm_file * fpriv )
{
struct drm_master * old_master ;
int ret ;
lockdep_assert_held_once ( & dev - > master_mutex ) ;
/* create a new master */
fpriv - > minor - > master = drm_master_create ( fpriv - > minor - > dev ) ;
if ( ! fpriv - > minor - > master )
return - ENOMEM ;
/* take another reference for the copy in the local file priv */
old_master = fpriv - > master ;
fpriv - > master = drm_master_get ( fpriv - > minor - > master ) ;
if ( dev - > driver - > master_create ) {
ret = dev - > driver - > master_create ( dev , fpriv - > master ) ;
if ( ret )
goto out_err ;
}
if ( dev - > driver - > master_set ) {
ret = dev - > driver - > master_set ( dev , fpriv , true ) ;
if ( ret )
goto out_err ;
}
fpriv - > is_master = 1 ;
fpriv - > allowed_master = 1 ;
fpriv - > authenticated = 1 ;
if ( old_master )
drm_master_put ( & old_master ) ;
return 0 ;
out_err :
/* drop both references and restore old master on failure */
drm_master_put ( & fpriv - > minor - > master ) ;
drm_master_put ( & fpriv - > master ) ;
fpriv - > master = old_master ;
return ret ;
}
int drm_setmaster_ioctl ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
{
int ret = 0 ;
mutex_lock ( & dev - > master_mutex ) ;
if ( file_priv - > is_master )
goto out_unlock ;
if ( file_priv - > minor - > master ) {
ret = - EINVAL ;
goto out_unlock ;
}
if ( ! file_priv - > master ) {
ret = - EINVAL ;
goto out_unlock ;
}
if ( ! file_priv - > allowed_master ) {
ret = drm_new_set_master ( dev , file_priv ) ;
goto out_unlock ;
}
file_priv - > minor - > master = drm_master_get ( file_priv - > master ) ;
file_priv - > is_master = 1 ;
if ( dev - > driver - > master_set ) {
ret = dev - > driver - > master_set ( dev , file_priv , false ) ;
if ( unlikely ( ret ! = 0 ) ) {
file_priv - > is_master = 0 ;
drm_master_put ( & file_priv - > minor - > master ) ;
}
}
out_unlock :
mutex_unlock ( & dev - > master_mutex ) ;
return ret ;
}
int drm_dropmaster_ioctl ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
{
int ret = - EINVAL ;
mutex_lock ( & dev - > master_mutex ) ;
if ( ! file_priv - > is_master )
goto out_unlock ;
if ( ! file_priv - > minor - > master )
goto out_unlock ;
ret = 0 ;
if ( dev - > driver - > master_drop )
dev - > driver - > master_drop ( dev , file_priv , false ) ;
drm_master_put ( & file_priv - > minor - > master ) ;
file_priv - > is_master = 0 ;
out_unlock :
mutex_unlock ( & dev - > master_mutex ) ;
return ret ;
}
struct drm_master * drm_master_get ( struct drm_master * master )
{
kref_get ( & master - > refcount ) ;
return master ;
}
EXPORT_SYMBOL ( drm_master_get ) ;
static void drm_master_destroy ( struct kref * kref )
{
struct drm_master * master = container_of ( kref , struct drm_master , refcount ) ;
struct drm_device * dev = master - > dev ;
if ( dev - > driver - > master_destroy )
dev - > driver - > master_destroy ( dev , master ) ;
drm_legacy_master_rmmaps ( dev , master ) ;
idr_destroy ( & master - > magic_map ) ;
kfree ( master - > unique ) ;
kfree ( master ) ;
}
void drm_master_put ( struct drm_master * * master )
{
kref_put ( & ( * master ) - > refcount , drm_master_destroy ) ;
* master = NULL ;
}
EXPORT_SYMBOL ( drm_master_put ) ;