2019-07-15 09:42:53 -04:00
/*
2005-09-25 14:28:13 +10:00
* \ file drm_dma . c
2005-04-16 15:20:36 -07:00
* DMA IOCTL and function support
*
* \ author Rickard E . ( Rik ) Faith < faith @ valinux . com >
* \ author Gareth Hughes < gareth @ valinux . com >
*/
/*
* Created : Fri Mar 19 14 : 30 : 16 1999 by faith @ valinux . com
*
* Copyright 1999 , 2000 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-08-30 18:16:33 -04:00
# include <linux/export.h>
2020-04-03 13:06:10 +02:00
# include <linux/pci.h>
2019-05-26 19:35:35 +02:00
# include <drm/drm_drv.h>
# include <drm/drm_print.h>
2014-09-11 07:43:25 +02:00
# include "drm_legacy.h"
2005-04-16 15:20:36 -07:00
/**
2020-01-14 17:01:35 +01:00
* drm_legacy_dma_setup ( ) - Initialize the DMA data .
2005-09-25 14:28:13 +10:00
*
2020-01-14 17:01:35 +01:00
* @ dev : DRM device .
* Return : zero on success or a negative value on failure .
2005-04-16 15:20:36 -07:00
*
* Allocate and initialize a drm_device_dma structure .
*/
2013-08-08 15:41:19 +02:00
int drm_legacy_dma_setup ( struct drm_device * dev )
2005-04-16 15:20:36 -07:00
{
int i ;
2013-08-08 15:41:19 +02:00
if ( ! drm_core_check_feature ( dev , DRIVER_HAVE_DMA ) | |
2016-08-03 21:11:10 +02:00
! drm_core_check_feature ( dev , DRIVER_LEGACY ) )
2013-08-08 15:41:19 +02:00
return 0 ;
dev - > buf_use = 0 ;
atomic_set ( & dev - > buf_alloc , 0 ) ;
2010-05-13 21:58:56 +02:00
dev - > dma = kzalloc ( sizeof ( * dev - > dma ) , GFP_KERNEL ) ;
2005-09-25 14:28:13 +10:00
if ( ! dev - > dma )
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
2005-09-25 14:28:13 +10:00
for ( i = 0 ; i < = DRM_MAX_ORDER ; i + + )
2005-04-16 15:20:36 -07:00
memset ( & dev - > dma - > bufs [ i ] , 0 , sizeof ( dev - > dma - > bufs [ 0 ] ) ) ;
return 0 ;
}
/**
2020-01-14 17:01:35 +01:00
* drm_legacy_dma_takedown ( ) - Cleanup the DMA resources .
2005-04-16 15:20:36 -07:00
*
2020-01-14 17:01:35 +01:00
* @ dev : DRM device .
2005-04-16 15:20:36 -07:00
*
* Free all pages associated with DMA buffers , the buffers and pages lists , and
2007-05-09 08:57:56 +02:00
* finally the drm_device : : dma structure itself .
2005-04-16 15:20:36 -07:00
*/
2013-08-08 15:41:19 +02:00
void drm_legacy_dma_takedown ( struct drm_device * dev )
2005-04-16 15:20:36 -07:00
{
2007-07-11 16:32:08 +10:00
struct drm_device_dma * dma = dev - > dma ;
2021-04-22 19:02:43 -07:00
drm_dma_handle_t * dmah ;
2005-09-25 14:28:13 +10:00
int i , j ;
2005-04-16 15:20:36 -07:00
2013-08-08 15:41:19 +02:00
if ( ! drm_core_check_feature ( dev , DRIVER_HAVE_DMA ) | |
2016-08-03 21:11:10 +02:00
! drm_core_check_feature ( dev , DRIVER_LEGACY ) )
2013-08-08 15:41:19 +02:00
return ;
2005-09-25 14:28:13 +10:00
if ( ! dma )
return ;
2005-04-16 15:20:36 -07:00
2005-09-25 14:28:13 +10:00
/* Clear dma buffers */
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < = DRM_MAX_ORDER ; i + + ) {
if ( dma - > bufs [ i ] . seg_count ) {
DRM_DEBUG ( " order %d: buf_count = %d, "
" seg_count = %d \n " ,
i ,
dma - > bufs [ i ] . buf_count ,
dma - > bufs [ i ] . seg_count ) ;
for ( j = 0 ; j < dma - > bufs [ i ] . seg_count ; j + + ) {
if ( dma - > bufs [ i ] . seglist [ j ] ) {
2021-04-22 19:02:43 -07:00
dmah = dma - > bufs [ i ] . seglist [ j ] ;
dma_free_coherent ( dev - > dev ,
dmah - > size ,
dmah - > vaddr ,
dmah - > busaddr ) ;
kfree ( dmah ) ;
2005-04-16 15:20:36 -07:00
}
}
2009-03-24 12:23:04 -07:00
kfree ( dma - > bufs [ i ] . seglist ) ;
2005-04-16 15:20:36 -07:00
}
2005-09-25 14:28:13 +10:00
if ( dma - > bufs [ i ] . buf_count ) {
for ( j = 0 ; j < dma - > bufs [ i ] . buf_count ; j + + ) {
2009-03-24 12:23:04 -07:00
kfree ( dma - > bufs [ i ] . buflist [ j ] . dev_private ) ;
2005-04-16 15:20:36 -07:00
}
2009-03-24 12:23:04 -07:00
kfree ( dma - > bufs [ i ] . buflist ) ;
2005-04-16 15:20:36 -07:00
}
}
2009-03-24 12:23:04 -07:00
kfree ( dma - > buflist ) ;
kfree ( dma - > pagelist ) ;
kfree ( dev - > dma ) ;
2005-04-16 15:20:36 -07:00
dev - > dma = NULL ;
}
/**
2020-01-14 17:01:35 +01:00
* drm_legacy_free_buffer ( ) - Free a buffer .
2005-04-16 15:20:36 -07:00
*
2020-01-14 17:01:35 +01:00
* @ dev : DRM device .
* @ buf : buffer to free .
2005-09-25 14:28:13 +10:00
*
2005-04-16 15:20:36 -07:00
* Resets the fields of \ p buf .
*/
2014-09-11 07:41:51 +02:00
void drm_legacy_free_buffer ( struct drm_device * dev , struct drm_buf * buf )
2005-04-16 15:20:36 -07:00
{
2005-09-25 14:28:13 +10:00
if ( ! buf )
return ;
2005-04-16 15:20:36 -07:00
2005-09-25 14:28:13 +10:00
buf - > waiting = 0 ;
buf - > pending = 0 ;
2007-08-25 20:23:09 +10:00
buf - > file_priv = NULL ;
2005-09-25 14:28:13 +10:00
buf - > used = 0 ;
2005-04-16 15:20:36 -07:00
}
/**
2020-01-14 17:01:35 +01:00
* drm_legacy_reclaim_buffers ( ) - Reclaim the buffers .
2005-04-16 15:20:36 -07:00
*
2020-01-14 17:01:35 +01:00
* @ dev : DRM device .
* @ file_priv : DRM file private .
2005-04-16 15:20:36 -07:00
*
2007-08-25 20:23:09 +10:00
* Frees each buffer associated with \ p file_priv not already on the hardware .
2005-04-16 15:20:36 -07:00
*/
2014-09-11 07:41:51 +02:00
void drm_legacy_reclaim_buffers ( struct drm_device * dev ,
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
2007-07-11 16:32:08 +10:00
struct drm_device_dma * dma = dev - > dma ;
2005-09-25 14:28:13 +10:00
int i ;
2005-04-16 15:20:36 -07:00
2005-09-25 14:28:13 +10:00
if ( ! dma )
return ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < dma - > buf_count ; i + + ) {
2007-08-25 20:23:09 +10:00
if ( dma - > buflist [ i ] - > file_priv = = file_priv ) {
2005-04-16 15:20:36 -07:00
switch ( dma - > buflist [ i ] - > list ) {
case DRM_LIST_NONE :
2014-09-11 07:41:51 +02:00
drm_legacy_free_buffer ( dev , dma - > buflist [ i ] ) ;
2005-04-16 15:20:36 -07:00
break ;
case DRM_LIST_WAIT :
dma - > buflist [ i ] - > list = DRM_LIST_RECLAIM ;
break ;
default :
/* Buffer already on hardware. */
break ;
}
}
}
}