2017-11-01 03:56:19 +10:00
/*
* Copyright 2017 Red Hat Inc .
*
* 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 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
* THE COPYRIGHT HOLDER ( S ) OR AUTHOR ( S ) 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 "nouveau_vmm.h"
# include "nouveau_drv.h"
# include "nouveau_bo.h"
2018-07-05 12:57:12 +10:00
# include "nouveau_svm.h"
2017-11-01 03:56:19 +10:00
# include "nouveau_mem.h"
void
nouveau_vma_unmap ( struct nouveau_vma * vma )
{
if ( vma - > mem ) {
2017-11-01 03:56:20 +10:00
nvif_vmm_unmap ( & vma - > vmm - > vmm , vma - > addr ) ;
2017-11-01 03:56:19 +10:00
vma - > mem = NULL ;
}
}
int
nouveau_vma_map ( struct nouveau_vma * vma , struct nouveau_mem * mem )
{
2017-11-01 03:56:20 +10:00
struct nvif_vma tmp = { . addr = vma - > addr } ;
int ret = nouveau_mem_map ( mem , & vma - > vmm - > vmm , & tmp ) ;
2017-11-01 03:56:19 +10:00
if ( ret )
return ret ;
vma - > mem = mem ;
return 0 ;
}
struct nouveau_vma *
nouveau_vma_find ( struct nouveau_bo * nvbo , struct nouveau_vmm * vmm )
{
struct nouveau_vma * vma ;
list_for_each_entry ( vma , & nvbo - > vma_list , head ) {
if ( vma - > vmm = = vmm )
return vma ;
}
return NULL ;
}
void
nouveau_vma_del ( struct nouveau_vma * * pvma )
{
struct nouveau_vma * vma = * pvma ;
if ( vma & & - - vma - > refs < = 0 ) {
2017-11-01 03:56:20 +10:00
if ( likely ( vma - > addr ! = ~ 0ULL ) ) {
struct nvif_vma tmp = { . addr = vma - > addr , . size = 1 } ;
nvif_vmm_put ( & vma - > vmm - > vmm , & tmp ) ;
}
2017-11-01 03:56:19 +10:00
list_del ( & vma - > head ) ;
kfree ( * pvma ) ;
}
2020-01-23 13:55:19 +10:00
* pvma = NULL ;
2017-11-01 03:56:19 +10:00
}
int
nouveau_vma_new ( struct nouveau_bo * nvbo , struct nouveau_vmm * vmm ,
struct nouveau_vma * * pvma )
{
struct nouveau_mem * mem = nouveau_mem ( & nvbo - > bo . mem ) ;
struct nouveau_vma * vma ;
2017-11-01 03:56:20 +10:00
struct nvif_vma tmp ;
2017-11-01 03:56:19 +10:00
int ret ;
if ( ( vma = * pvma = nouveau_vma_find ( nvbo , vmm ) ) ) {
vma - > refs + + ;
return 0 ;
}
if ( ! ( vma = * pvma = kmalloc ( sizeof ( * vma ) , GFP_KERNEL ) ) )
return - ENOMEM ;
vma - > vmm = vmm ;
vma - > refs = 1 ;
vma - > addr = ~ 0ULL ;
vma - > mem = NULL ;
2018-05-08 20:39:47 +10:00
vma - > fence = NULL ;
2017-11-01 03:56:19 +10:00
list_add_tail ( & vma - > head , & nvbo - > vma_list ) ;
if ( nvbo - > bo . mem . mem_type ! = TTM_PL_SYSTEM & &
mem - > mem . page = = nvbo - > page ) {
2017-11-01 03:56:20 +10:00
ret = nvif_vmm_get ( & vmm - > vmm , LAZY , false , mem - > mem . page , 0 ,
mem - > mem . size , & tmp ) ;
2017-11-01 03:56:19 +10:00
if ( ret )
goto done ;
2017-11-01 03:56:20 +10:00
vma - > addr = tmp . addr ;
2017-11-01 03:56:19 +10:00
ret = nouveau_vma_map ( vma , mem ) ;
} else {
2017-11-01 03:56:20 +10:00
ret = nvif_vmm_get ( & vmm - > vmm , PTES , false , mem - > mem . page , 0 ,
mem - > mem . size , & tmp ) ;
vma - > addr = tmp . addr ;
2017-11-01 03:56:19 +10:00
}
done :
if ( ret )
nouveau_vma_del ( pvma ) ;
return ret ;
}
void
nouveau_vmm_fini ( struct nouveau_vmm * vmm )
{
2018-07-05 12:57:12 +10:00
nouveau_svmm_fini ( & vmm - > svmm ) ;
2017-11-01 03:56:20 +10:00
nvif_vmm_fini ( & vmm - > vmm ) ;
vmm - > cli = NULL ;
2017-11-01 03:56:19 +10:00
}
int
nouveau_vmm_init ( struct nouveau_cli * cli , s32 oclass , struct nouveau_vmm * vmm )
{
2018-06-13 16:25:53 +10:00
int ret = nvif_vmm_init ( & cli - > mmu , oclass , false , PAGE_SIZE , 0 , NULL , 0 ,
2017-11-01 03:56:20 +10:00
& vmm - > vmm ) ;
if ( ret )
return ret ;
2017-11-01 03:56:19 +10:00
vmm - > cli = cli ;
2017-11-01 03:56:20 +10:00
return 0 ;
2017-11-01 03:56:19 +10:00
}