2005-06-28 22:48:56 +10:00
/*
2006-08-07 22:03:22 +10:00
* Copyright 2006 Tungsten Graphics Inc . , Bismarck , ND . , USA .
* All rights reserved .
2005-06-28 22:48:56 +10: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 , sub license ,
* 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 NON - INFRINGEMENT . IN NO EVENT SHALL
2006-08-07 22:03:22 +10:00
* THE AUTHORS OR COPYRIGHT HOLDERS AND / OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM ,
* DAMAGES OR OTHER LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE ,
2005-06-28 22:48:56 +10:00
* ARISING FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE .
*/
2006-08-07 22:03:22 +10:00
/*
* Authors : Thomas Hellstr <EFBFBD> m < thomas - at - tungstengraphics - dot - com >
*/
2005-06-28 22:48:56 +10:00
# include "drmP.h"
# include "via_drm.h"
# include "via_drv.h"
2006-08-07 22:03:22 +10:00
# include "drm_sman.h"
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
# define VIA_MM_ALIGN_SHIFT 4
# define VIA_MM_ALIGN_MASK ( (1 << VIA_MM_ALIGN_SHIFT) - 1)
2005-06-28 22:48:56 +10:00
int via_agp_init ( DRM_IOCTL_ARGS )
{
2006-08-07 22:03:22 +10:00
DRM_DEVICE ;
2005-06-28 22:48:56 +10:00
drm_via_agp_t agp ;
2006-08-07 22:03:22 +10:00
drm_via_private_t * dev_priv = ( drm_via_private_t * ) dev - > dev_private ;
int ret ;
2005-06-28 22:48:56 +10:00
2005-07-27 11:43:56 -07:00
DRM_COPY_FROM_USER_IOCTL ( agp , ( drm_via_agp_t __user * ) data ,
sizeof ( agp ) ) ;
2006-08-07 22:03:22 +10:00
mutex_lock ( & dev - > struct_mutex ) ;
ret = drm_sman_set_range ( & dev_priv - > sman , VIA_MEM_AGP , 0 ,
agp . size > > VIA_MM_ALIGN_SHIFT ) ;
if ( ret ) {
DRM_ERROR ( " AGP memory manager initialisation error \n " ) ;
mutex_unlock ( & dev - > struct_mutex ) ;
return ret ;
}
2005-06-28 22:48:56 +10:00
2006-08-19 17:40:50 +10:00
dev_priv - > agp_initialized = 1 ;
2006-08-07 22:03:22 +10:00
dev_priv - > agp_offset = agp . offset ;
mutex_unlock ( & dev - > struct_mutex ) ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
DRM_DEBUG ( " offset = %u, size = %u " , agp . offset , agp . size ) ;
2005-06-28 22:48:56 +10:00
return 0 ;
}
int via_fb_init ( DRM_IOCTL_ARGS )
{
2006-08-07 22:03:22 +10:00
DRM_DEVICE ;
2005-06-28 22:48:56 +10:00
drm_via_fb_t fb ;
2006-08-07 22:03:22 +10:00
drm_via_private_t * dev_priv = ( drm_via_private_t * ) dev - > dev_private ;
int ret ;
2005-06-28 22:48:56 +10:00
2005-07-27 11:43:56 -07:00
DRM_COPY_FROM_USER_IOCTL ( fb , ( drm_via_fb_t __user * ) data , sizeof ( fb ) ) ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
mutex_lock ( & dev - > struct_mutex ) ;
ret = drm_sman_set_range ( & dev_priv - > sman , VIA_MEM_VIDEO , 0 ,
fb . size > > VIA_MM_ALIGN_SHIFT ) ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
if ( ret ) {
DRM_ERROR ( " VRAM memory manager initialisation error \n " ) ;
mutex_unlock ( & dev - > struct_mutex ) ;
return ret ;
}
2005-06-28 22:48:56 +10:00
2006-08-19 17:40:50 +10:00
dev_priv - > vram_initialized = 1 ;
2006-08-07 22:03:22 +10:00
dev_priv - > vram_offset = fb . offset ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
mutex_unlock ( & dev - > struct_mutex ) ;
DRM_DEBUG ( " offset = %u, size = %u " , fb . offset , fb . size ) ;
return 0 ;
2005-06-28 22:48:56 +10:00
}
int via_final_context ( struct drm_device * dev , int context )
2005-09-25 14:28:13 +10:00
{
2005-06-28 22:48:56 +10:00
drm_via_private_t * dev_priv = ( drm_via_private_t * ) dev - > dev_private ;
2005-09-25 14:28:13 +10:00
via_release_futex ( dev_priv , context ) ;
2005-06-28 22:48:56 +10:00
/* Linux specific until context tracking code gets ported to BSD */
/* Last context, perform cleanup */
if ( dev - > ctx_count = = 1 & & dev - > dev_private ) {
2005-09-25 14:28:13 +10:00
DRM_DEBUG ( " Last Context \n " ) ;
2005-06-28 22:48:56 +10:00
if ( dev - > irq )
drm_irq_uninstall ( dev ) ;
via_cleanup_futex ( dev_priv ) ;
via_do_cleanup_map ( dev ) ;
}
return 1 ;
}
2006-08-07 22:03:22 +10:00
void via_lastclose ( struct drm_device * dev )
{
drm_via_private_t * dev_priv = ( drm_via_private_t * ) dev - > dev_private ;
if ( ! dev_priv )
return ;
mutex_lock ( & dev - > struct_mutex ) ;
drm_sman_cleanup ( & dev_priv - > sman ) ;
2006-08-14 11:35:15 +10:00
dev_priv - > vram_initialized = 0 ;
dev_priv - > agp_initialized = 0 ;
2006-08-07 22:03:22 +10:00
mutex_unlock ( & dev - > struct_mutex ) ;
}
2005-06-28 22:48:56 +10:00
int via_mem_alloc ( DRM_IOCTL_ARGS )
{
2006-08-07 22:03:22 +10:00
DRM_DEVICE ;
2005-06-28 22:48:56 +10:00
drm_via_mem_t mem ;
2006-08-07 22:03:22 +10:00
int retval = 0 ;
drm_memblock_item_t * item ;
drm_via_private_t * dev_priv = ( drm_via_private_t * ) dev - > dev_private ;
unsigned long tmpSize ;
2005-06-28 22:48:56 +10:00
2005-07-27 11:43:56 -07:00
DRM_COPY_FROM_USER_IOCTL ( mem , ( drm_via_mem_t __user * ) data ,
sizeof ( mem ) ) ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
if ( mem . type > VIA_MEM_AGP ) {
DRM_ERROR ( " Unknown memory type allocation \n " ) ;
return DRM_ERR ( EINVAL ) ;
2005-06-28 22:48:56 +10:00
}
2006-08-07 22:03:22 +10:00
mutex_lock ( & dev - > struct_mutex ) ;
2006-08-14 11:35:15 +10:00
if ( 0 = = ( ( mem . type = = VIA_MEM_VIDEO ) ? dev_priv - > vram_initialized :
2006-08-07 22:03:22 +10:00
dev_priv - > agp_initialized ) ) {
DRM_ERROR
( " Attempt to allocate from uninitialized memory manager. \n " ) ;
mutex_unlock ( & dev - > struct_mutex ) ;
return DRM_ERR ( EINVAL ) ;
2005-06-28 22:48:56 +10:00
}
2006-08-07 22:03:22 +10:00
tmpSize = ( mem . size + VIA_MM_ALIGN_MASK ) > > VIA_MM_ALIGN_SHIFT ;
item = drm_sman_alloc ( & dev_priv - > sman , mem . type , tmpSize , 0 ,
( unsigned long ) priv ) ;
mutex_unlock ( & dev - > struct_mutex ) ;
if ( item ) {
mem . offset = ( ( mem . type = = VIA_MEM_VIDEO ) ?
dev_priv - > vram_offset : dev_priv - > agp_offset ) +
( item - > mm - >
offset ( item - > mm , item - > mm_info ) < < VIA_MM_ALIGN_SHIFT ) ;
mem . index = item - > user_hash . key ;
2005-06-28 22:48:56 +10:00
} else {
2006-08-07 22:03:22 +10:00
mem . offset = 0 ;
mem . size = 0 ;
mem . index = 0 ;
DRM_DEBUG ( " Video memory allocation failed \n " ) ;
retval = DRM_ERR ( ENOMEM ) ;
2005-06-28 22:48:56 +10:00
}
2006-08-07 22:03:22 +10:00
DRM_COPY_TO_USER_IOCTL ( ( drm_via_mem_t __user * ) data , mem , sizeof ( mem ) ) ;
2005-06-28 22:48:56 +10:00
return retval ;
}
int via_mem_free ( DRM_IOCTL_ARGS )
{
2006-08-07 22:03:22 +10:00
DRM_DEVICE ;
drm_via_private_t * dev_priv = dev - > dev_private ;
2005-06-28 22:48:56 +10:00
drm_via_mem_t mem ;
2006-08-07 22:03:22 +10:00
int ret ;
2005-06-28 22:48:56 +10:00
2005-07-27 11:43:56 -07:00
DRM_COPY_FROM_USER_IOCTL ( mem , ( drm_via_mem_t __user * ) data ,
sizeof ( mem ) ) ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
mutex_lock ( & dev - > struct_mutex ) ;
ret = drm_sman_free_key ( & dev_priv - > sman , mem . index ) ;
mutex_unlock ( & dev - > struct_mutex ) ;
DRM_DEBUG ( " free = 0x%lx \n " , mem . index ) ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
return ret ;
2005-06-28 22:48:56 +10:00
}
2006-08-07 22:03:22 +10:00
void via_reclaim_buffers_locked ( drm_device_t * dev , struct file * filp )
2005-06-28 22:48:56 +10:00
{
2006-08-07 22:03:22 +10:00
drm_via_private_t * dev_priv = dev - > dev_private ;
drm_file_t * priv = filp - > private_data ;
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
mutex_lock ( & dev - > struct_mutex ) ;
if ( drm_sman_owner_clean ( & dev_priv - > sman , ( unsigned long ) priv ) ) {
mutex_unlock ( & dev - > struct_mutex ) ;
return ;
2005-06-28 22:48:56 +10:00
}
2006-08-07 22:03:22 +10:00
if ( dev - > driver - > dma_quiescent ) {
dev - > driver - > dma_quiescent ( dev ) ;
}
2005-06-28 22:48:56 +10:00
2006-08-07 22:03:22 +10:00
drm_sman_owner_cleanup ( & dev_priv - > sman , ( unsigned long ) priv ) ;
mutex_unlock ( & dev - > struct_mutex ) ;
return ;
2005-06-28 22:48:56 +10:00
}