2008-09-08 09:09:53 +00:00
/*
* Copyright ( C ) 2006 Benjamin Herrenschmidt , IBM Corporation
*
* Provide default implementations of the DMA mapping callbacks for
* busses using the iommu infrastructure
*/
2011-05-27 10:46:24 -04:00
# include <linux/export.h>
2008-09-08 09:09:53 +00:00
# include <asm/iommu.h>
/*
* Generic iommu implementation
*/
/* Allocates a contiguous real buffer and creates mappings over it.
* Returns the virtual address of the buffer and sets dma_handle
* to the dma address ( mapping ) of the first page .
*/
static void * dma_iommu_alloc_coherent ( struct device * dev , size_t size ,
2011-12-06 14:14:46 +01:00
dma_addr_t * dma_handle , gfp_t flag ,
2016-08-03 13:46:00 -07:00
unsigned long attrs )
2008-09-08 09:09:53 +00:00
{
2009-09-21 08:26:35 +00:00
return iommu_alloc_coherent ( dev , get_iommu_table_base ( dev ) , size ,
2010-10-18 07:27:04 +00:00
dma_handle , dev - > coherent_dma_mask , flag ,
2008-09-08 09:09:54 +00:00
dev_to_node ( dev ) ) ;
2008-09-08 09:09:53 +00:00
}
static void dma_iommu_free_coherent ( struct device * dev , size_t size ,
2011-12-06 14:14:46 +01:00
void * vaddr , dma_addr_t dma_handle ,
2016-08-03 13:46:00 -07:00
unsigned long attrs )
2008-09-08 09:09:53 +00:00
{
2009-09-21 08:26:35 +00:00
iommu_free_coherent ( get_iommu_table_base ( dev ) , size , vaddr , dma_handle ) ;
2008-09-08 09:09:53 +00:00
}
/* Creates TCEs for a user provided buffer. The user buffer must be
2008-10-27 20:38:08 +00:00
* contiguous real kernel storage ( not vmalloc ) . The address passed here
* comprises a page address and offset into that page . The dma_addr_t
* returned will point to the same byte within the page as was passed in .
2008-09-08 09:09:53 +00:00
*/
2008-10-27 20:38:08 +00:00
static dma_addr_t dma_iommu_map_page ( struct device * dev , struct page * page ,
unsigned long offset , size_t size ,
enum dma_data_direction direction ,
2016-08-03 13:46:00 -07:00
unsigned long attrs )
2008-09-08 09:09:53 +00:00
{
2009-09-21 08:26:35 +00:00
return iommu_map_page ( dev , get_iommu_table_base ( dev ) , page , offset ,
size , device_to_mask ( dev ) , direction , attrs ) ;
2008-09-08 09:09:53 +00:00
}
2008-10-27 20:38:08 +00:00
static void dma_iommu_unmap_page ( struct device * dev , dma_addr_t dma_handle ,
size_t size , enum dma_data_direction direction ,
2016-08-03 13:46:00 -07:00
unsigned long attrs )
2008-09-08 09:09:53 +00:00
{
2009-09-21 08:26:35 +00:00
iommu_unmap_page ( get_iommu_table_base ( dev ) , dma_handle , size , direction ,
2008-10-27 20:38:08 +00:00
attrs ) ;
2008-09-08 09:09:53 +00:00
}
static int dma_iommu_map_sg ( struct device * dev , struct scatterlist * sglist ,
int nelems , enum dma_data_direction direction ,
2016-08-03 13:46:00 -07:00
unsigned long attrs )
2008-09-08 09:09:53 +00:00
{
2014-11-05 15:28:30 +01:00
return ppc_iommu_map_sg ( dev , get_iommu_table_base ( dev ) , sglist , nelems ,
device_to_mask ( dev ) , direction , attrs ) ;
2008-09-08 09:09:53 +00:00
}
static void dma_iommu_unmap_sg ( struct device * dev , struct scatterlist * sglist ,
int nelems , enum dma_data_direction direction ,
2016-08-03 13:46:00 -07:00
unsigned long attrs )
2008-09-08 09:09:53 +00:00
{
2014-11-05 15:28:30 +01:00
ppc_iommu_unmap_sg ( get_iommu_table_base ( dev ) , sglist , nelems ,
direction , attrs ) ;
2008-09-08 09:09:53 +00:00
}
/* We support DMA to/from any memory page via the iommu */
2015-06-24 15:25:31 +10:00
int dma_iommu_dma_supported ( struct device * dev , u64 mask )
2008-09-08 09:09:53 +00:00
{
2009-09-21 08:26:35 +00:00
struct iommu_table * tbl = get_iommu_table_base ( dev ) ;
2008-09-08 09:09:53 +00:00
2010-09-15 08:05:45 +00:00
if ( ! tbl ) {
dev_info ( dev , " Warning: IOMMU dma not supported: mask 0x%08llx "
" , table unavailable \n " , mask ) ;
return 0 ;
}
2013-12-09 18:17:03 +11:00
if ( tbl - > it_offset > ( mask > > tbl - > it_page_shift ) ) {
2012-08-18 07:34:15 +00:00
dev_info ( dev , " Warning: IOMMU offset too big for device mask \n " ) ;
dev_info ( dev , " mask: 0x%08llx, table offset: 0x%08lx \n " ,
2013-12-09 18:17:03 +11:00
mask , tbl - > it_offset < < tbl - > it_page_shift ) ;
2008-09-08 09:09:53 +00:00
return 0 ;
} else
return 1 ;
}
2011-06-24 09:05:24 +00:00
static u64 dma_iommu_get_required_mask ( struct device * dev )
2011-06-24 09:05:22 +00:00
{
struct iommu_table * tbl = get_iommu_table_base ( dev ) ;
u64 mask ;
if ( ! tbl )
return 0 ;
mask = 1ULL < ( fls_long ( tbl - > it_offset + tbl - > it_size ) - 1 ) ;
mask + = mask - 1 ;
return mask ;
}
2009-08-04 19:08:25 +00:00
struct dma_map_ops dma_iommu_ops = {
2011-12-06 14:14:46 +01:00
. alloc = dma_iommu_alloc_coherent ,
. free = dma_iommu_free_coherent ,
2012-06-14 13:03:04 +02:00
. mmap = dma_direct_mmap_coherent ,
2011-06-24 09:05:25 +00:00
. map_sg = dma_iommu_map_sg ,
. unmap_sg = dma_iommu_unmap_sg ,
. dma_supported = dma_iommu_dma_supported ,
. map_page = dma_iommu_map_page ,
. unmap_page = dma_iommu_unmap_page ,
2011-06-24 09:05:24 +00:00
. get_required_mask = dma_iommu_get_required_mask ,
2008-09-08 09:09:53 +00:00
} ;
EXPORT_SYMBOL ( dma_iommu_ops ) ;