2005-04-16 15:20:36 -07:00
/**
2005-09-25 14:28:13 +10:00
* \ file drm_lock . c
2005-04-16 15:20:36 -07:00
* IOCTLs for locking
2005-09-25 14:28:13 +10:00
*
2005-04-16 15:20:36 -07:00
* \ author Rickard E . ( Rik ) Faith < faith @ valinux . com >
* \ author Gareth Hughes < gareth @ valinux . com >
*/
/*
* 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 .
*
* 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 .
*/
2011-12-22 19:09:01 +00:00
# include <linux/export.h>
2012-10-02 18:01:07 +01:00
# include <drm/drmP.h>
2014-07-24 12:10:04 +02:00
# include "drm_legacy.h"
2014-09-10 12:43:54 +02:00
# include "drm_internal.h"
2005-04-16 15:20:36 -07:00
2010-08-23 22:53:34 +02:00
static int drm_lock_take ( struct drm_lock_data * lock_data , unsigned int context ) ;
2016-06-14 20:50:57 +02:00
/**
* Take the heavyweight lock .
*
* \ param lock lock pointer .
* \ param context locking context .
* \ return one if the lock is held , or zero otherwise .
*
* Attempt to mark the lock as held by the given context , via the \ p cmpxchg instruction .
*/
static
int drm_lock_take ( struct drm_lock_data * lock_data ,
unsigned int context )
{
unsigned int old , new , prev ;
volatile unsigned int * lock = & lock_data - > hw_lock - > lock ;
spin_lock_bh ( & lock_data - > spinlock ) ;
do {
old = * lock ;
if ( old & _DRM_LOCK_HELD )
new = old | _DRM_LOCK_CONT ;
else {
new = context | _DRM_LOCK_HELD |
( ( lock_data - > user_waiters + lock_data - > kernel_waiters > 1 ) ?
_DRM_LOCK_CONT : 0 ) ;
}
prev = cmpxchg ( lock , old , new ) ;
} while ( prev ! = old ) ;
spin_unlock_bh ( & lock_data - > spinlock ) ;
if ( _DRM_LOCKING_CONTEXT ( old ) = = context ) {
if ( old & _DRM_LOCK_HELD ) {
if ( context ! = DRM_KERNEL_CONTEXT ) {
DRM_ERROR ( " %d holds heavyweight lock \n " ,
context ) ;
}
return 0 ;
}
}
if ( ( _DRM_LOCKING_CONTEXT ( new ) ) = = context & & ( new & _DRM_LOCK_HELD ) ) {
/* Have lock */
return 1 ;
}
return 0 ;
}
/**
* This takes a lock forcibly and hands it to context . Should ONLY be used
* inside * _unlock to give lock to kernel before calling * _dma_schedule .
*
* \ param dev DRM device .
* \ param lock lock pointer .
* \ param context locking context .
* \ return always one .
*
* Resets the lock file pointer .
* Marks the lock as held by the given context , via the \ p cmpxchg instruction .
*/
static int drm_lock_transfer ( struct drm_lock_data * lock_data ,
unsigned int context )
{
unsigned int old , new , prev ;
volatile unsigned int * lock = & lock_data - > hw_lock - > lock ;
lock_data - > file_priv = NULL ;
do {
old = * lock ;
new = context | _DRM_LOCK_HELD ;
prev = cmpxchg ( lock , old , new ) ;
} while ( prev ! = old ) ;
return 1 ;
}
static int drm_legacy_lock_free ( struct drm_lock_data * lock_data ,
unsigned int context )
{
unsigned int old , new , prev ;
volatile unsigned int * lock = & lock_data - > hw_lock - > lock ;
spin_lock_bh ( & lock_data - > spinlock ) ;
if ( lock_data - > kernel_waiters ! = 0 ) {
drm_lock_transfer ( lock_data , 0 ) ;
lock_data - > idle_has_lock = 1 ;
spin_unlock_bh ( & lock_data - > spinlock ) ;
return 1 ;
}
spin_unlock_bh ( & lock_data - > spinlock ) ;
do {
old = * lock ;
new = _DRM_LOCKING_CONTEXT ( old ) ;
prev = cmpxchg ( lock , old , new ) ;
} while ( prev ! = old ) ;
if ( _DRM_LOCK_IS_HELD ( old ) & & _DRM_LOCKING_CONTEXT ( old ) ! = context ) {
DRM_ERROR ( " %d freed heavyweight lock held by %d \n " ,
context , _DRM_LOCKING_CONTEXT ( old ) ) ;
return 1 ;
}
wake_up_interruptible ( & lock_data - > lock_queue ) ;
return 0 ;
}
2005-09-25 14:28:13 +10:00
/**
2005-04-16 15:20:36 -07:00
* Lock 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 .
*
* Add the current task to the lock wait queue , and attempt to take to lock .
*/
2014-08-29 12:12:46 +02:00
int drm_legacy_lock ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2005-09-25 14:28:13 +10:00
DECLARE_WAITQUEUE ( entry , current ) ;
2007-09-03 12:06:45 +10:00
struct drm_lock * lock = data ;
2008-11-28 14:22:24 +10:00
struct drm_master * master = file_priv - > master ;
2005-09-25 14:28:13 +10:00
int ret = 0 ;
2005-04-16 15:20:36 -07:00
2016-08-03 21:11:10 +02:00
if ( ! drm_core_check_feature ( dev , DRIVER_LEGACY ) )
2015-06-23 11:34:21 +02:00
return - EINVAL ;
2007-08-25 20:23:09 +10:00
+ + file_priv - > lock_count ;
2005-04-16 15:20:36 -07:00
2007-09-03 12:06:45 +10:00
if ( lock - > context = = DRM_KERNEL_CONTEXT ) {
2005-09-25 14:28:13 +10:00
DRM_ERROR ( " Process %d using kernel context %d \n " ,
2007-10-18 23:40:40 -07:00
task_pid_nr ( current ) , lock - > context ) ;
2005-09-25 14:28:13 +10:00
return - EINVAL ;
}
2005-04-16 15:20:36 -07:00
2005-09-25 14:28:13 +10:00
DRM_DEBUG ( " %d (pid %d) requests lock (0x%08x), flags = 0x%08x \n " ,
2007-10-18 23:40:40 -07:00
lock - > context , task_pid_nr ( current ) ,
2016-11-27 17:09:09 +00:00
master - > lock . hw_lock ? master - > lock . hw_lock - > lock : - 1 ,
lock - > flags ) ;
2005-04-16 15:20:36 -07:00
2008-11-28 14:22:24 +10:00
add_wait_queue ( & master - > lock . lock_queue , & entry ) ;
spin_lock_bh ( & master - > lock . spinlock ) ;
master - > lock . user_waiters + + ;
spin_unlock_bh ( & master - > lock . spinlock ) ;
2005-04-16 15:20:36 -07:00
for ( ; ; ) {
__set_current_state ( TASK_INTERRUPTIBLE ) ;
2008-11-28 14:22:24 +10:00
if ( ! master - > lock . hw_lock ) {
2005-04-16 15:20:36 -07:00
/* Device has been unregistered */
2009-03-02 11:10:56 +01:00
send_sig ( SIGTERM , current , 0 ) ;
2005-04-16 15:20:36 -07:00
ret = - EINTR ;
break ;
}
2008-11-28 14:22:24 +10:00
if ( drm_lock_take ( & master - > lock , lock - > context ) ) {
master - > lock . file_priv = file_priv ;
master - > lock . lock_time = jiffies ;
2005-09-25 14:28:13 +10:00
break ; /* Got lock */
2005-04-16 15:20:36 -07:00
}
2005-09-25 14:28:13 +10:00
2005-04-16 15:20:36 -07:00
/* Contention */
2010-08-27 08:55:28 +10:00
mutex_unlock ( & drm_global_mutex ) ;
2005-04-16 15:20:36 -07:00
schedule ( ) ;
2010-08-27 08:55:28 +10:00
mutex_lock ( & drm_global_mutex ) ;
2005-09-25 14:28:13 +10:00
if ( signal_pending ( current ) ) {
2009-03-02 11:10:54 +01:00
ret = - EINTR ;
2005-04-16 15:20:36 -07:00
break ;
}
}
2008-11-28 14:22:24 +10:00
spin_lock_bh ( & master - > lock . spinlock ) ;
master - > lock . user_waiters - - ;
spin_unlock_bh ( & master - > lock . spinlock ) ;
2005-04-16 15:20:36 -07:00
__set_current_state ( TASK_RUNNING ) ;
2008-11-28 14:22:24 +10:00
remove_wait_queue ( & master - > lock . lock_queue , & entry ) ;
2005-04-16 15:20:36 -07:00
2007-09-03 12:06:45 +10:00
DRM_DEBUG ( " %d %s \n " , lock - > context ,
ret ? " interrupted " : " has lock " ) ;
2007-03-23 13:28:33 +11:00
if ( ret ) return ret ;
2006-01-02 21:32:48 +11:00
2008-08-24 17:02:26 +10:00
/* don't set the block all signals on the master process for now
* really probably not the correct answer but lets us debug xkb
* xserver for now */
2016-06-21 10:54:20 +02:00
if ( ! drm_is_current_master ( file_priv ) ) {
2008-08-24 17:02:26 +10:00
dev - > sigdata . context = lock - > context ;
2008-11-28 14:22:24 +10:00
dev - > sigdata . lock = master - > lock . hw_lock ;
2008-08-24 17:02:26 +10:00
}
2005-09-25 14:28:13 +10:00
2007-09-03 12:06:45 +10:00
if ( dev - > driver - > dma_quiescent & & ( lock - > flags & _DRM_LOCK_QUIESCENT ) )
{
2006-01-02 21:32:48 +11:00
if ( dev - > driver - > dma_quiescent ( dev ) ) {
2007-09-03 12:06:45 +10:00
DRM_DEBUG ( " %d waiting for DMA quiescent \n " ,
lock - > context ) ;
2007-08-25 19:22:43 +10:00
return - EBUSY ;
2006-01-02 21:32:48 +11:00
}
}
2005-09-25 14:28:13 +10:00
2006-01-02 21:32:48 +11:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2005-09-25 14:28:13 +10:00
/**
2005-04-16 15:20:36 -07:00
* Unlock 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 .
*
* Transfer and free the lock .
*/
2014-08-29 12:12:46 +02:00
int drm_legacy_unlock ( 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_lock * lock = data ;
2010-09-26 00:24:48 +02:00
struct drm_master * master = file_priv - > master ;
2005-04-16 15:20:36 -07:00
2016-08-03 21:11:10 +02:00
if ( ! drm_core_check_feature ( dev , DRIVER_LEGACY ) )
2015-06-23 11:34:21 +02:00
return - EINVAL ;
2007-09-03 12:06:45 +10:00
if ( lock - > context = = DRM_KERNEL_CONTEXT ) {
2005-09-25 14:28:13 +10:00
DRM_ERROR ( " Process %d using kernel context %d \n " ,
2007-10-18 23:40:40 -07:00
task_pid_nr ( current ) , lock - > context ) ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
}
2014-08-29 12:12:46 +02:00
if ( drm_legacy_lock_free ( & master - > lock , lock - > context ) ) {
2010-09-26 00:24:48 +02:00
/* FIXME: Should really bail out here. */
}
2005-04-16 15:20:36 -07:00
return 0 ;
}
2007-03-23 13:28:33 +11:00
/**
* This function returns immediately and takes the hw lock
* with the kernel context if it is free , otherwise it gets the highest priority when and if
* it is eventually released .
*
* This guarantees that the kernel will _eventually_ have the lock _unless_ it is held
* by a blocked process . ( In the latter case an explicit wait for the hardware lock would cause
* a deadlock , which is why the " idlelock " was invented ) .
*
* This should be sufficient to wait for GPU idle without
* having to worry about starvation .
*/
2014-08-29 12:12:46 +02:00
void drm_legacy_idlelock_take ( struct drm_lock_data * lock_data )
2007-03-23 13:28:33 +11:00
{
2012-05-17 13:27:21 +02:00
int ret ;
2007-03-23 13:28:33 +11:00
2008-05-07 12:22:39 +10:00
spin_lock_bh ( & lock_data - > spinlock ) ;
2007-03-23 13:28:33 +11:00
lock_data - > kernel_waiters + + ;
if ( ! lock_data - > idle_has_lock ) {
2008-05-07 12:22:39 +10:00
spin_unlock_bh ( & lock_data - > spinlock ) ;
2007-03-23 13:28:33 +11:00
ret = drm_lock_take ( lock_data , DRM_KERNEL_CONTEXT ) ;
2008-05-07 12:22:39 +10:00
spin_lock_bh ( & lock_data - > spinlock ) ;
2007-03-23 13:28:33 +11:00
if ( ret = = 1 )
lock_data - > idle_has_lock = 1 ;
}
2008-05-07 12:22:39 +10:00
spin_unlock_bh ( & lock_data - > spinlock ) ;
2007-03-23 13:28:33 +11:00
}
2014-08-29 12:12:46 +02:00
EXPORT_SYMBOL ( drm_legacy_idlelock_take ) ;
2007-03-23 13:28:33 +11:00
2014-08-29 12:12:46 +02:00
void drm_legacy_idlelock_release ( struct drm_lock_data * lock_data )
2007-03-23 13:28:33 +11:00
{
unsigned int old , prev ;
volatile unsigned int * lock = & lock_data - > hw_lock - > lock ;
2008-05-07 12:22:39 +10:00
spin_lock_bh ( & lock_data - > spinlock ) ;
2007-03-23 13:28:33 +11:00
if ( - - lock_data - > kernel_waiters = = 0 ) {
if ( lock_data - > idle_has_lock ) {
do {
old = * lock ;
prev = cmpxchg ( lock , old , DRM_KERNEL_CONTEXT ) ;
} while ( prev ! = old ) ;
wake_up_interruptible ( & lock_data - > lock_queue ) ;
lock_data - > idle_has_lock = 0 ;
}
}
2008-05-07 12:22:39 +10:00
spin_unlock_bh ( & lock_data - > spinlock ) ;
2007-03-23 13:28:33 +11:00
}
2014-08-29 12:12:46 +02:00
EXPORT_SYMBOL ( drm_legacy_idlelock_release ) ;
2007-03-23 13:28:33 +11:00
2016-06-14 20:50:57 +02:00
static int drm_legacy_i_have_hw_lock ( struct drm_device * dev ,
struct drm_file * file_priv )
2007-03-23 13:28:33 +11:00
{
2008-11-28 14:22:24 +10:00
struct drm_master * master = file_priv - > master ;
return ( file_priv - > lock_count & & master - > lock . hw_lock & &
_DRM_LOCK_IS_HELD ( master - > lock . hw_lock - > lock ) & &
master - > lock . file_priv = = file_priv ) ;
2007-03-23 13:28:33 +11:00
}
2016-06-14 20:50:57 +02:00
void drm_legacy_lock_release ( struct drm_device * dev , struct file * filp )
{
struct drm_file * file_priv = filp - > private_data ;
/* if the master has gone away we can't do anything with the lock */
2016-06-21 10:54:12 +02:00
if ( ! dev - > master )
2016-06-14 20:50:57 +02:00
return ;
if ( drm_legacy_i_have_hw_lock ( dev , file_priv ) ) {
DRM_DEBUG ( " File %p released, freeing lock for context %d \n " ,
filp , _DRM_LOCKING_CONTEXT ( file_priv - > master - > lock . hw_lock - > lock ) ) ;
drm_legacy_lock_free ( & file_priv - > master - > lock ,
_DRM_LOCKING_CONTEXT ( file_priv - > master - > lock . hw_lock - > lock ) ) ;
}
}