2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2013-01-18 15:12:20 +05:30
/*
* Copyright ( C ) 2004 , 2007 - 2010 , 2011 - 2012 Synopsys , Inc . ( www . synopsys . com )
*/
2018-05-18 15:41:32 +02:00
# include <linux/dma-noncoherent.h>
2015-05-25 19:54:28 +03:00
# include <asm/cache.h>
2013-01-18 15:12:20 +05:30
# include <asm/cacheflush.h>
2018-07-30 19:26:34 +03:00
/*
2019-06-14 16:26:41 +02:00
* ARCH specific callbacks for generic noncoherent DMA ops
2018-07-30 19:26:34 +03:00
* - hardware IOC not available ( or " dma-coherent " not set for device in DT )
* - But still handle both coherent and non - coherent requests from caller
*
* For DMA coherent hardware ( IOC ) generic code suffices
*/
2013-01-18 15:12:20 +05:30
2019-06-14 16:26:41 +02:00
void arch_dma_prep_coherent ( struct page * page , size_t size )
{
2015-04-03 12:37:07 +03:00
/*
* Evict any existing L1 and / or L2 lines for the backing page
* in case it was used earlier as a normal " cached " page .
* Yeah this bit us - STAR 9000898266
*
* Although core does call flush_cache_vmap ( ) , it gets kvaddr hence
* can ' t be used to efficiently flush L1 and / or L2 which need paddr
* Currently flush_cache_vmap nukes the L1 cache completely which
* will be optimized as a separate commit
*/
2019-06-14 16:26:41 +02:00
dma_cache_wback_inv ( page_to_phys ( page ) , size ) ;
2016-11-03 18:06:13 +03:00
}
2018-07-24 17:13:02 +03:00
/*
* Cache operations depending on function and direction argument , inspired by
* https : //lkml.org/lkml/2018/5/18/979
* " dma_sync_*_for_cpu and direction=TO_DEVICE (was Re: [PATCH 02/20]
* dma - mapping : provide a generic dma - noncoherent implementation ) "
*
* | map = = for_device | unmap = = for_cpu
* | - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* TO_DEV | writeback writeback | none none
* FROM_DEV | invalidate invalidate | invalidate * invalidate *
* BIDIR | writeback + inv writeback + inv | invalidate invalidate
*
* [ * ] needed for CPU speculative prefetches
*
* NOTE : we don ' t check the validity of direction argument as it is done in
* upper layer functions ( in include / linux / dma - mapping . h )
*/
2018-05-18 15:41:32 +02:00
void arch_sync_dma_for_device ( struct device * dev , phys_addr_t paddr ,
size_t size , enum dma_data_direction dir )
2016-01-20 15:01:26 -08:00
{
2018-07-24 17:13:02 +03:00
switch ( dir ) {
case DMA_TO_DEVICE :
dma_cache_wback ( paddr , size ) ;
break ;
case DMA_FROM_DEVICE :
dma_cache_inv ( paddr , size ) ;
break ;
case DMA_BIDIRECTIONAL :
dma_cache_wback_inv ( paddr , size ) ;
break ;
default :
break ;
}
2016-01-20 15:01:26 -08:00
}
2018-05-18 15:41:32 +02:00
void arch_sync_dma_for_cpu ( struct device * dev , phys_addr_t paddr ,
size_t size , enum dma_data_direction dir )
2017-07-18 12:14:09 -07:00
{
2018-07-24 17:13:02 +03:00
switch ( dir ) {
case DMA_TO_DEVICE :
break ;
/* FROM_DEVICE invalidate needed if speculative CPU prefetch only */
case DMA_FROM_DEVICE :
case DMA_BIDIRECTIONAL :
dma_cache_inv ( paddr , size ) ;
break ;
default :
break ;
}
2017-07-18 12:14:09 -07:00
}
2018-07-30 19:26:34 +03:00
/*
2018-09-08 11:22:43 +02:00
* Plug in direct dma map ops .
2018-07-30 19:26:34 +03:00
*/
void arch_setup_dma_ops ( struct device * dev , u64 dma_base , u64 size ,
const struct iommu_ops * iommu , bool coherent )
{
/*
* IOC hardware snoops all DMA traffic keeping the caches consistent
* with memory - eliding need for any explicit cache maintenance of
2018-09-08 11:22:43 +02:00
* DMA buffers .
2018-07-30 19:26:34 +03:00
*/
2018-09-08 11:22:43 +02:00
if ( is_isa_arcv2 ( ) & & ioc_enable & & coherent )
dev - > dma_coherent = true ;
2019-07-19 21:46:00 +03:00
dev_info ( dev , " use %scoherent DMA ops \n " ,
2018-09-08 11:22:43 +02:00
dev - > dma_coherent ? " " : " non " ) ;
2018-07-30 19:26:34 +03:00
}