2006-01-11 22:44:42 +01:00
/* Glue code to lib/swiotlb.c */
# include <linux/pci.h>
# include <linux/cache.h>
2016-07-13 20:18:56 -04:00
# include <linux/init.h>
2008-12-16 12:17:35 -08:00
# include <linux/swiotlb.h>
# include <linux/bootmem.h>
2006-09-29 01:59:48 -07:00
# include <linux/dma-mapping.h>
2008-07-11 10:23:42 +09:00
# include <asm/iommu.h>
2006-01-11 22:44:42 +01:00
# include <asm/swiotlb.h>
# include <asm/dma.h>
2010-08-26 13:58:00 -04:00
# include <asm/xen/swiotlb-xen.h>
# include <asm/iommu_table.h>
2006-01-11 22:44:42 +01:00
int swiotlb __read_mostly ;
2014-06-04 16:06:50 -07:00
void * x86_swiotlb_alloc_coherent ( struct device * hwdev , size_t size ,
2012-03-27 14:28:18 +02:00
dma_addr_t * dma_handle , gfp_t flags ,
struct dma_attrs * attrs )
2008-10-23 23:14:29 +09:00
{
void * vaddr ;
2015-06-10 17:49:41 +02:00
/*
* Don ' t print a warning when the first allocation attempt fails .
* swiotlb_alloc_coherent ( ) will print a warning when the DMA
* memory allocation ultimately failed .
*/
flags | = __GFP_NOWARN ;
2012-03-27 14:28:18 +02:00
vaddr = dma_generic_alloc_coherent ( hwdev , size , dma_handle , flags ,
attrs ) ;
2008-10-23 23:14:29 +09:00
if ( vaddr )
return vaddr ;
return swiotlb_alloc_coherent ( hwdev , size , dma_handle , flags ) ;
}
2014-06-04 16:06:50 -07:00
void x86_swiotlb_free_coherent ( struct device * dev , size_t size ,
2012-03-27 14:28:18 +02:00
void * vaddr , dma_addr_t dma_addr ,
struct dma_attrs * attrs )
{
2014-06-04 16:06:50 -07:00
if ( is_swiotlb_buffer ( dma_to_phys ( dev , dma_addr ) ) )
swiotlb_free_coherent ( dev , size , vaddr , dma_addr ) ;
else
dma_generic_free_coherent ( dev , size , vaddr , dma_addr , attrs ) ;
2012-03-27 14:28:18 +02:00
}
2009-04-12 23:24:21 +05:30
static struct dma_map_ops swiotlb_dma_ops = {
2006-01-11 22:44:42 +01:00
. mapping_error = swiotlb_dma_mapping_error ,
2012-03-27 14:28:18 +02:00
. alloc = x86_swiotlb_alloc_coherent ,
. free = x86_swiotlb_free_coherent ,
2006-01-11 22:44:42 +01:00
. sync_single_for_cpu = swiotlb_sync_single_for_cpu ,
. sync_single_for_device = swiotlb_sync_single_for_device ,
. sync_sg_for_cpu = swiotlb_sync_sg_for_cpu ,
. sync_sg_for_device = swiotlb_sync_sg_for_device ,
2009-01-05 23:59:02 +09:00
. map_sg = swiotlb_map_sg_attrs ,
. unmap_sg = swiotlb_unmap_sg_attrs ,
2009-01-05 23:47:22 +09:00
. map_page = swiotlb_map_page ,
. unmap_page = swiotlb_unmap_page ,
2006-01-11 22:44:42 +01:00
. dma_supported = NULL ,
} ;
2009-11-12 00:03:28 +09:00
/*
2010-08-26 13:57:59 -04:00
* pci_swiotlb_detect_override - set swiotlb to 1 if necessary
2009-11-12 00:03:28 +09:00
*
* This returns non - zero if we are forced to use swiotlb ( by the boot
* option ) .
*/
2010-08-26 13:57:59 -04:00
int __init pci_swiotlb_detect_override ( void )
2006-01-11 22:44:42 +01:00
{
2009-11-25 08:46:28 +09:00
int use_swiotlb = swiotlb | swiotlb_force ;
2010-08-26 13:57:59 -04:00
if ( swiotlb_force )
swiotlb = 1 ;
return use_swiotlb ;
}
2010-08-26 13:58:00 -04:00
IOMMU_INIT_FINISH ( pci_swiotlb_detect_override ,
pci_xen_swiotlb_detect ,
pci_swiotlb_init ,
pci_swiotlb_late_init ) ;
2010-08-26 13:57:59 -04:00
/*
* if 4 GB or more detected ( and iommu = off not set ) return 1
* and set swiotlb to 1.
*/
int __init pci_swiotlb_detect_4gb ( void )
{
2006-01-11 22:44:42 +01:00
/* don't initialize swiotlb if iommu=off (no_iommu=1) */
2008-12-16 12:17:36 -08:00
# ifdef CONFIG_X86_64
2015-12-04 14:07:06 +01:00
if ( ! no_iommu & & max_possible_pfn > MAX_DMA32_PFN )
2009-08-04 16:19:20 +01:00
swiotlb = 1 ;
2008-12-16 12:17:36 -08:00
# endif
2010-08-26 13:57:59 -04:00
return swiotlb ;
2009-12-15 20:47:56 +09:00
}
2010-08-26 13:58:00 -04:00
IOMMU_INIT ( pci_swiotlb_detect_4gb ,
pci_swiotlb_detect_override ,
pci_swiotlb_init ,
pci_swiotlb_late_init ) ;
2009-12-15 20:47:56 +09:00
void __init pci_swiotlb_init ( void )
{
2006-01-11 22:44:42 +01:00
if ( swiotlb ) {
2009-11-10 19:46:19 +09:00
swiotlb_init ( 0 ) ;
2006-01-11 22:44:42 +01:00
dma_ops = & swiotlb_dma_ops ;
2009-11-14 20:46:36 +09:00
}
2006-01-11 22:44:42 +01:00
}
2010-08-26 13:57:59 -04:00
void __init pci_swiotlb_late_init ( void )
{
/* An IOMMU turned us off. */
if ( ! swiotlb )
swiotlb_free ( ) ;
else {
printk ( KERN_INFO " PCI-DMA: "
" Using software bounce buffering for IO (SWIOTLB) \n " ) ;
swiotlb_print_info ( ) ;
}
}