2005-04-17 02:20:36 +04:00
/**
2005-09-25 08:28:13 +04:00
* \ file drm_scatter . c
2005-04-17 02:20:36 +04:00
* IOCTLs to manage scatter / gather memory
*
* \ author Gareth Hughes < gareth @ valinux . com >
*/
/*
* Created : Mon Dec 18 23 : 20 : 54 2000 by gareth @ valinux . com
*
* 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
* PRECISION INSIGHT 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 .
*/
# include <linux/vmalloc.h>
# include "drmP.h"
# define DEBUG_SCATTER 0
2007-07-11 10:53:40 +04:00
void drm_sg_cleanup ( struct drm_sg_mem * entry )
2005-04-17 02:20:36 +04:00
{
struct page * page ;
int i ;
2005-09-25 08:28:13 +04:00
for ( i = 0 ; i < entry - > pages ; i + + ) {
2005-04-17 02:20:36 +04:00
page = entry - > pagelist [ i ] ;
2005-09-25 08:28:13 +04:00
if ( page )
ClearPageReserved ( page ) ;
2005-04-17 02:20:36 +04:00
}
2005-09-25 08:28:13 +04:00
vfree ( entry - > virtual ) ;
drm_free ( entry - > busaddr ,
entry - > pages * sizeof ( * entry - > busaddr ) , DRM_MEM_PAGES ) ;
drm_free ( entry - > pagelist ,
entry - > pages * sizeof ( * entry - > pagelist ) , DRM_MEM_PAGES ) ;
drm_free ( entry , sizeof ( * entry ) , DRM_MEM_SGLISTS ) ;
2005-04-17 02:20:36 +04:00
}
2005-08-05 16:11:22 +04:00
# ifdef _LP64
# define ScatterHandle(x) (unsigned int)((x >> 32) + (x & ((1L << 32) - 1)))
# else
# define ScatterHandle(x) (unsigned int)(x)
# endif
2005-09-25 08:28:13 +04:00
int drm_sg_alloc ( struct inode * inode , struct file * filp ,
unsigned int cmd , unsigned long arg )
2005-04-17 02:20:36 +04:00
{
2007-07-11 09:53:27 +04:00
struct drm_file * priv = filp - > private_data ;
struct drm_device * dev = priv - > head - > dev ;
2007-07-11 09:27:12 +04:00
struct drm_scatter_gather __user * argp = ( void __user * ) arg ;
struct drm_scatter_gather request ;
2007-07-11 10:53:40 +04:00
struct drm_sg_mem * entry ;
2005-04-17 02:20:36 +04:00
unsigned long pages , i , j ;
2005-09-25 08:28:13 +04:00
DRM_DEBUG ( " %s \n " , __FUNCTION__ ) ;
2005-04-17 02:20:36 +04:00
if ( ! drm_core_check_feature ( dev , DRIVER_SG ) )
return - EINVAL ;
2005-09-25 08:28:13 +04:00
if ( dev - > sg )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
2005-09-25 08:28:13 +04:00
if ( copy_from_user ( & request , argp , sizeof ( request ) ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
2005-09-25 08:28:13 +04:00
entry = drm_alloc ( sizeof ( * entry ) , DRM_MEM_SGLISTS ) ;
if ( ! entry )
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
2005-09-25 08:28:13 +04:00
memset ( entry , 0 , sizeof ( * entry ) ) ;
2005-04-17 02:20:36 +04:00
pages = ( request . size + PAGE_SIZE - 1 ) / PAGE_SIZE ;
2005-09-25 08:28:13 +04:00
DRM_DEBUG ( " sg size=%ld pages=%ld \n " , request . size , pages ) ;
2005-04-17 02:20:36 +04:00
entry - > pages = pages ;
2005-09-25 08:28:13 +04:00
entry - > pagelist = drm_alloc ( pages * sizeof ( * entry - > pagelist ) ,
DRM_MEM_PAGES ) ;
if ( ! entry - > pagelist ) {
drm_free ( entry , sizeof ( * entry ) , DRM_MEM_SGLISTS ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
memset ( entry - > pagelist , 0 , pages * sizeof ( * entry - > pagelist ) ) ;
2005-09-25 08:28:13 +04:00
entry - > busaddr = drm_alloc ( pages * sizeof ( * entry - > busaddr ) ,
DRM_MEM_PAGES ) ;
if ( ! entry - > busaddr ) {
drm_free ( entry - > pagelist ,
entry - > pages * sizeof ( * entry - > pagelist ) ,
DRM_MEM_PAGES ) ;
drm_free ( entry , sizeof ( * entry ) , DRM_MEM_SGLISTS ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
2005-09-25 08:28:13 +04:00
memset ( ( void * ) entry - > busaddr , 0 , pages * sizeof ( * entry - > busaddr ) ) ;
entry - > virtual = vmalloc_32 ( pages < < PAGE_SHIFT ) ;
if ( ! entry - > virtual ) {
drm_free ( entry - > busaddr ,
entry - > pages * sizeof ( * entry - > busaddr ) , DRM_MEM_PAGES ) ;
drm_free ( entry - > pagelist ,
entry - > pages * sizeof ( * entry - > pagelist ) ,
DRM_MEM_PAGES ) ;
drm_free ( entry , sizeof ( * entry ) , DRM_MEM_SGLISTS ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
/* This also forces the mapping of COW pages, so our page list
* will be valid . Please don ' t remove it . . .
*/
2005-09-25 08:28:13 +04:00
memset ( entry - > virtual , 0 , pages < < PAGE_SHIFT ) ;
2005-04-17 02:20:36 +04:00
2005-08-05 16:11:22 +04:00
entry - > handle = ScatterHandle ( ( unsigned long ) entry - > virtual ) ;
2005-04-17 02:20:36 +04:00
2005-09-25 08:28:13 +04:00
DRM_DEBUG ( " sg alloc handle = %08lx \n " , entry - > handle ) ;
DRM_DEBUG ( " sg alloc virtual = %p \n " , entry - > virtual ) ;
2005-04-17 02:20:36 +04:00
2005-09-25 08:28:13 +04:00
for ( i = ( unsigned long ) entry - > virtual , j = 0 ; j < pages ;
i + = PAGE_SIZE , j + + ) {
2005-04-17 02:20:36 +04:00
entry - > pagelist [ j ] = vmalloc_to_page ( ( void * ) i ) ;
if ( ! entry - > pagelist [ j ] )
goto failed ;
SetPageReserved ( entry - > pagelist [ j ] ) ;
}
request . handle = entry - > handle ;
2005-09-25 08:28:13 +04:00
if ( copy_to_user ( argp , & request , sizeof ( request ) ) ) {
drm_sg_cleanup ( entry ) ;
2005-04-17 02:20:36 +04:00
return - EFAULT ;
}
dev - > sg = entry ;
# if DEBUG_SCATTER
/* Verify that each page points to its virtual address, and vice
* versa .
*/
{
2005-09-25 08:28:13 +04:00
int error = 0 ;
2005-04-17 02:20:36 +04:00
2005-09-25 08:28:13 +04:00
for ( i = 0 ; i < pages ; i + + ) {
unsigned long * tmp ;
2005-04-17 02:20:36 +04:00
2005-09-25 08:28:13 +04:00
tmp = page_address ( entry - > pagelist [ i ] ) ;
for ( j = 0 ;
j < PAGE_SIZE / sizeof ( unsigned long ) ;
j + + , tmp + + ) {
* tmp = 0xcafebabe ;
}
tmp = ( unsigned long * ) ( ( u8 * ) entry - > virtual +
( PAGE_SIZE * i ) ) ;
for ( j = 0 ;
j < PAGE_SIZE / sizeof ( unsigned long ) ;
j + + , tmp + + ) {
if ( * tmp ! = 0xcafebabe & & error = = 0 ) {
error = 1 ;
DRM_ERROR ( " Scatter allocation error, "
" pagelist does not match "
" virtual mapping \n " ) ;
}
}
tmp = page_address ( entry - > pagelist [ i ] ) ;
for ( j = 0 ;
j < PAGE_SIZE / sizeof ( unsigned long ) ;
j + + , tmp + + ) {
* tmp = 0 ;
2005-04-17 02:20:36 +04:00
}
}
2005-09-25 08:28:13 +04:00
if ( error = = 0 )
DRM_ERROR ( " Scatter allocation matches pagelist \n " ) ;
2005-04-17 02:20:36 +04:00
}
# endif
return 0 ;
2005-09-25 08:28:13 +04:00
failed :
drm_sg_cleanup ( entry ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
2005-09-25 08:28:13 +04:00
int drm_sg_free ( struct inode * inode , struct file * filp ,
unsigned int cmd , unsigned long arg )
2005-04-17 02:20:36 +04:00
{
2007-07-11 09:53:27 +04:00
struct drm_file * priv = filp - > private_data ;
struct drm_device * dev = priv - > head - > dev ;
2007-07-11 09:27:12 +04:00
struct drm_scatter_gather request ;
2007-07-11 10:53:40 +04:00
struct drm_sg_mem * entry ;
2005-04-17 02:20:36 +04:00
if ( ! drm_core_check_feature ( dev , DRIVER_SG ) )
return - EINVAL ;
2005-09-25 08:28:13 +04:00
if ( copy_from_user ( & request ,
2007-07-11 09:27:12 +04:00
( struct drm_scatter_gather __user * ) arg ,
2005-09-25 08:28:13 +04:00
sizeof ( request ) ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
entry = dev - > sg ;
dev - > sg = NULL ;
2005-09-25 08:28:13 +04:00
if ( ! entry | | entry - > handle ! = request . handle )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
2005-09-25 08:28:13 +04:00
DRM_DEBUG ( " sg free virtual = %p \n " , entry - > virtual ) ;
2005-04-17 02:20:36 +04:00
2005-09-25 08:28:13 +04:00
drm_sg_cleanup ( entry ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}