2022-06-02 22:23:48 +03:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Xen grant DMA - mapping layer - contains special DMA - mapping routines
* for providing grant references as DMA addresses to be used by frontends
* ( e . g . virtio ) in Xen guests
*
* Copyright ( c ) 2021 , Juergen Gross < jgross @ suse . com >
*/
# include <linux/module.h>
# include <linux/dma-map-ops.h>
# include <linux/of.h>
# include <linux/pfn.h>
# include <linux/xarray.h>
2022-06-22 09:38:38 +03:00
# include <linux/virtio_anchor.h>
# include <linux/virtio.h>
2022-06-02 22:23:48 +03:00
# include <xen/xen.h>
# include <xen/xen-ops.h>
# include <xen/grant_table.h>
struct xen_grant_dma_data {
/* The ID of backend domain */
domid_t backend_domid ;
/* Is device behaving sane? */
bool broken ;
} ;
2022-10-05 20:48:23 +03:00
static DEFINE_XARRAY_FLAGS ( xen_grant_dma_devices , XA_FLAGS_LOCK_IRQ ) ;
2022-06-02 22:23:48 +03:00
# define XEN_GRANT_DMA_ADDR_OFF (1ULL << 63)
static inline dma_addr_t grant_to_dma ( grant_ref_t grant )
{
2022-10-08 18:10:13 +03:00
return XEN_GRANT_DMA_ADDR_OFF | ( ( dma_addr_t ) grant < < XEN_PAGE_SHIFT ) ;
2022-06-02 22:23:48 +03:00
}
static inline grant_ref_t dma_to_grant ( dma_addr_t dma )
{
2022-10-08 18:10:13 +03:00
return ( grant_ref_t ) ( ( dma & ~ XEN_GRANT_DMA_ADDR_OFF ) > > XEN_PAGE_SHIFT ) ;
2022-06-02 22:23:48 +03:00
}
static struct xen_grant_dma_data * find_xen_grant_dma_data ( struct device * dev )
{
struct xen_grant_dma_data * data ;
2022-10-05 20:48:23 +03:00
unsigned long flags ;
2022-06-02 22:23:48 +03:00
2022-10-05 20:48:23 +03:00
xa_lock_irqsave ( & xen_grant_dma_devices , flags ) ;
2022-06-02 22:23:48 +03:00
data = xa_load ( & xen_grant_dma_devices , ( unsigned long ) dev ) ;
2022-10-05 20:48:23 +03:00
xa_unlock_irqrestore ( & xen_grant_dma_devices , flags ) ;
2022-06-02 22:23:48 +03:00
return data ;
}
2022-10-05 20:48:23 +03:00
static int store_xen_grant_dma_data ( struct device * dev ,
struct xen_grant_dma_data * data )
{
unsigned long flags ;
int ret ;
xa_lock_irqsave ( & xen_grant_dma_devices , flags ) ;
ret = xa_err ( __xa_store ( & xen_grant_dma_devices , ( unsigned long ) dev , data ,
GFP_ATOMIC ) ) ;
xa_unlock_irqrestore ( & xen_grant_dma_devices , flags ) ;
return ret ;
}
2022-06-02 22:23:48 +03:00
/*
* DMA ops for Xen frontends ( e . g . virtio ) .
*
* Used to act as a kind of software IOMMU for Xen guests by using grants as
* DMA addresses .
* Such a DMA address is formed by using the grant reference as a frame
* number and setting the highest address bit ( this bit is for the backend
* to be able to distinguish it from e . g . a mmio address ) .
*/
static void * xen_grant_dma_alloc ( struct device * dev , size_t size ,
dma_addr_t * dma_handle , gfp_t gfp ,
unsigned long attrs )
{
struct xen_grant_dma_data * data ;
2022-10-08 18:10:13 +03:00
unsigned int i , n_pages = XEN_PFN_UP ( size ) ;
2022-06-02 22:23:48 +03:00
unsigned long pfn ;
grant_ref_t grant ;
void * ret ;
data = find_xen_grant_dma_data ( dev ) ;
if ( ! data )
return NULL ;
if ( unlikely ( data - > broken ) )
return NULL ;
2022-10-08 18:10:13 +03:00
ret = alloc_pages_exact ( n_pages * XEN_PAGE_SIZE , gfp ) ;
2022-06-02 22:23:48 +03:00
if ( ! ret )
return NULL ;
pfn = virt_to_pfn ( ret ) ;
if ( gnttab_alloc_grant_reference_seq ( n_pages , & grant ) ) {
2022-10-08 18:10:13 +03:00
free_pages_exact ( ret , n_pages * XEN_PAGE_SIZE ) ;
2022-06-02 22:23:48 +03:00
return NULL ;
}
for ( i = 0 ; i < n_pages ; i + + ) {
gnttab_grant_foreign_access_ref ( grant + i , data - > backend_domid ,
pfn_to_gfn ( pfn + i ) , 0 ) ;
}
* dma_handle = grant_to_dma ( grant ) ;
return ret ;
}
static void xen_grant_dma_free ( struct device * dev , size_t size , void * vaddr ,
dma_addr_t dma_handle , unsigned long attrs )
{
struct xen_grant_dma_data * data ;
2022-10-08 18:10:13 +03:00
unsigned int i , n_pages = XEN_PFN_UP ( size ) ;
2022-06-02 22:23:48 +03:00
grant_ref_t grant ;
data = find_xen_grant_dma_data ( dev ) ;
if ( ! data )
return ;
if ( unlikely ( data - > broken ) )
return ;
grant = dma_to_grant ( dma_handle ) ;
for ( i = 0 ; i < n_pages ; i + + ) {
if ( unlikely ( ! gnttab_end_foreign_access_ref ( grant + i ) ) ) {
dev_alert ( dev , " Grant still in use by backend domain, disabled for further use \n " ) ;
data - > broken = true ;
return ;
}
}
gnttab_free_grant_reference_seq ( grant , n_pages ) ;
2022-10-08 18:10:13 +03:00
free_pages_exact ( vaddr , n_pages * XEN_PAGE_SIZE ) ;
2022-06-02 22:23:48 +03:00
}
static struct page * xen_grant_dma_alloc_pages ( struct device * dev , size_t size ,
dma_addr_t * dma_handle ,
enum dma_data_direction dir ,
gfp_t gfp )
{
void * vaddr ;
vaddr = xen_grant_dma_alloc ( dev , size , dma_handle , gfp , 0 ) ;
if ( ! vaddr )
return NULL ;
return virt_to_page ( vaddr ) ;
}
static void xen_grant_dma_free_pages ( struct device * dev , size_t size ,
struct page * vaddr , dma_addr_t dma_handle ,
enum dma_data_direction dir )
{
xen_grant_dma_free ( dev , size , page_to_virt ( vaddr ) , dma_handle , 0 ) ;
}
static dma_addr_t xen_grant_dma_map_page ( struct device * dev , struct page * page ,
unsigned long offset , size_t size ,
enum dma_data_direction dir ,
unsigned long attrs )
{
struct xen_grant_dma_data * data ;
2022-10-08 18:10:13 +03:00
unsigned long dma_offset = xen_offset_in_page ( offset ) ,
pfn_offset = XEN_PFN_DOWN ( offset ) ;
unsigned int i , n_pages = XEN_PFN_UP ( dma_offset + size ) ;
2022-06-02 22:23:48 +03:00
grant_ref_t grant ;
dma_addr_t dma_handle ;
if ( WARN_ON ( dir = = DMA_NONE ) )
return DMA_MAPPING_ERROR ;
data = find_xen_grant_dma_data ( dev ) ;
if ( ! data )
return DMA_MAPPING_ERROR ;
if ( unlikely ( data - > broken ) )
return DMA_MAPPING_ERROR ;
if ( gnttab_alloc_grant_reference_seq ( n_pages , & grant ) )
return DMA_MAPPING_ERROR ;
for ( i = 0 ; i < n_pages ; i + + ) {
gnttab_grant_foreign_access_ref ( grant + i , data - > backend_domid ,
2022-10-08 18:10:12 +03:00
pfn_to_gfn ( page_to_xen_pfn ( page ) + i + pfn_offset ) ,
dir = = DMA_TO_DEVICE ) ;
2022-06-02 22:23:48 +03:00
}
2022-10-08 18:10:12 +03:00
dma_handle = grant_to_dma ( grant ) + dma_offset ;
2022-06-02 22:23:48 +03:00
return dma_handle ;
}
static void xen_grant_dma_unmap_page ( struct device * dev , dma_addr_t dma_handle ,
size_t size , enum dma_data_direction dir ,
unsigned long attrs )
{
struct xen_grant_dma_data * data ;
2022-10-08 18:10:13 +03:00
unsigned long dma_offset = xen_offset_in_page ( dma_handle ) ;
unsigned int i , n_pages = XEN_PFN_UP ( dma_offset + size ) ;
2022-06-02 22:23:48 +03:00
grant_ref_t grant ;
if ( WARN_ON ( dir = = DMA_NONE ) )
return ;
data = find_xen_grant_dma_data ( dev ) ;
if ( ! data )
return ;
if ( unlikely ( data - > broken ) )
return ;
grant = dma_to_grant ( dma_handle ) ;
for ( i = 0 ; i < n_pages ; i + + ) {
if ( unlikely ( ! gnttab_end_foreign_access_ref ( grant + i ) ) ) {
dev_alert ( dev , " Grant still in use by backend domain, disabled for further use \n " ) ;
data - > broken = true ;
return ;
}
}
gnttab_free_grant_reference_seq ( grant , n_pages ) ;
}
static void xen_grant_dma_unmap_sg ( struct device * dev , struct scatterlist * sg ,
int nents , enum dma_data_direction dir ,
unsigned long attrs )
{
struct scatterlist * s ;
unsigned int i ;
if ( WARN_ON ( dir = = DMA_NONE ) )
return ;
for_each_sg ( sg , s , nents , i )
xen_grant_dma_unmap_page ( dev , s - > dma_address , sg_dma_len ( s ) , dir ,
attrs ) ;
}
static int xen_grant_dma_map_sg ( struct device * dev , struct scatterlist * sg ,
int nents , enum dma_data_direction dir ,
unsigned long attrs )
{
struct scatterlist * s ;
unsigned int i ;
if ( WARN_ON ( dir = = DMA_NONE ) )
return - EINVAL ;
for_each_sg ( sg , s , nents , i ) {
s - > dma_address = xen_grant_dma_map_page ( dev , sg_page ( s ) , s - > offset ,
s - > length , dir , attrs ) ;
if ( s - > dma_address = = DMA_MAPPING_ERROR )
goto out ;
sg_dma_len ( s ) = s - > length ;
}
return nents ;
out :
xen_grant_dma_unmap_sg ( dev , sg , i , dir , attrs | DMA_ATTR_SKIP_CPU_SYNC ) ;
sg_dma_len ( sg ) = 0 ;
return - EIO ;
}
static int xen_grant_dma_supported ( struct device * dev , u64 mask )
{
return mask = = DMA_BIT_MASK ( 64 ) ;
}
static const struct dma_map_ops xen_grant_dma_ops = {
. alloc = xen_grant_dma_alloc ,
. free = xen_grant_dma_free ,
. alloc_pages = xen_grant_dma_alloc_pages ,
. free_pages = xen_grant_dma_free_pages ,
. mmap = dma_common_mmap ,
. get_sgtable = dma_common_get_sgtable ,
. map_page = xen_grant_dma_map_page ,
. unmap_page = xen_grant_dma_unmap_page ,
. map_sg = xen_grant_dma_map_sg ,
. unmap_sg = xen_grant_dma_unmap_sg ,
. dma_supported = xen_grant_dma_supported ,
} ;
2022-08-29 14:26:07 +03:00
static int xen_dt_grant_init_backend_domid ( struct device * dev ,
2022-10-25 19:20:03 +03:00
domid_t * backend_domid )
2022-06-02 22:23:48 +03:00
{
2022-06-02 22:23:52 +03:00
struct of_phandle_args iommu_spec ;
2022-06-02 22:23:48 +03:00
2022-06-02 22:23:52 +03:00
if ( of_parse_phandle_with_args ( dev - > of_node , " iommus " , " #iommu-cells " ,
0 , & iommu_spec ) ) {
2022-10-25 19:20:03 +03:00
dev_dbg ( dev , " Cannot parse iommus property \n " ) ;
2022-08-29 14:26:07 +03:00
return - ESRCH ;
2022-06-02 22:23:52 +03:00
}
if ( ! of_device_is_compatible ( iommu_spec . np , " xen,grant-dma " ) | |
iommu_spec . args_count ! = 1 ) {
2022-10-25 19:20:03 +03:00
dev_dbg ( dev , " Incompatible IOMMU node \n " ) ;
2022-06-02 22:23:52 +03:00
of_node_put ( iommu_spec . np ) ;
2022-08-29 14:26:07 +03:00
return - ESRCH ;
2022-06-02 22:23:52 +03:00
}
of_node_put ( iommu_spec . np ) ;
2022-08-29 14:26:07 +03:00
/*
* The endpoint ID here means the ID of the domain where the
* corresponding backend is running
*/
2022-10-25 19:20:03 +03:00
* backend_domid = iommu_spec . args [ 0 ] ;
2022-08-29 14:26:07 +03:00
return 0 ;
}
2022-10-25 19:20:03 +03:00
static int xen_grant_init_backend_domid ( struct device * dev ,
domid_t * backend_domid )
{
int ret = - ENODEV ;
if ( dev - > of_node ) {
ret = xen_dt_grant_init_backend_domid ( dev , backend_domid ) ;
} else if ( IS_ENABLED ( CONFIG_XEN_VIRTIO_FORCE_GRANT ) | | xen_pv_domain ( ) ) {
dev_info ( dev , " Using dom0 as backend \n " ) ;
* backend_domid = 0 ;
ret = 0 ;
}
return ret ;
}
static void xen_grant_setup_dma_ops ( struct device * dev , domid_t backend_domid )
2022-08-29 14:26:07 +03:00
{
struct xen_grant_dma_data * data ;
data = find_xen_grant_dma_data ( dev ) ;
if ( data ) {
dev_err ( dev , " Xen grant DMA data is already created \n " ) ;
return ;
}
2022-06-02 22:23:48 +03:00
data = devm_kzalloc ( dev , sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data )
goto err ;
2022-10-25 19:20:03 +03:00
data - > backend_domid = backend_domid ;
2022-06-02 22:23:48 +03:00
2022-10-05 20:48:23 +03:00
if ( store_xen_grant_dma_data ( dev , data ) ) {
2022-06-02 22:23:48 +03:00
dev_err ( dev , " Cannot store Xen grant DMA data \n " ) ;
goto err ;
}
dev - > dma_ops = & xen_grant_dma_ops ;
return ;
err :
2022-08-29 14:26:07 +03:00
devm_kfree ( dev , data ) ;
2022-06-02 22:23:48 +03:00
dev_err ( dev , " Cannot set up Xen grant DMA ops, retain platform DMA ops \n " ) ;
}
2022-08-29 14:26:08 +03:00
bool xen_virtio_restricted_mem_acc ( struct virtio_device * dev )
{
2022-10-25 19:20:03 +03:00
domid_t backend_domid ;
2022-08-29 14:26:08 +03:00
2022-10-25 19:20:03 +03:00
if ( ! xen_grant_init_backend_domid ( dev - > dev . parent , & backend_domid ) ) {
xen_grant_setup_dma_ops ( dev - > dev . parent , backend_domid ) ;
return true ;
}
2022-08-29 14:26:08 +03:00
2022-10-25 19:20:03 +03:00
return false ;
2022-08-29 14:26:08 +03:00
}
2022-06-02 22:23:48 +03:00
MODULE_DESCRIPTION ( " Xen grant DMA-mapping layer " ) ;
MODULE_AUTHOR ( " Juergen Gross <jgross@suse.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;