2008-07-30 12:06:12 -07:00
/**************************************************************************
*
* Copyright ( c ) 2006 - 2007 Tungsten Graphics , Inc . , Cedar Park , TX . , USA
* All Rights Reserved .
*
* 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 , sub license , 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 ( including the
* next paragraph ) 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 NON - INFRINGEMENT . IN NO EVENT SHALL
* THE COPYRIGHT HOLDERS , AUTHORS AND / OR ITS SUPPLIERS 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 .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/*
* Authors : Thomas Hellström < thomas - at - tungstengraphics - dot - com >
*/
2011-08-30 18:16:33 -04:00
# include <linux/export.h>
2012-10-02 18:01:07 +01:00
# include <drm/drmP.h>
2008-07-30 12:06:12 -07:00
# if defined(CONFIG_X86)
2014-02-26 12:06:51 -07:00
/*
* clflushopt is an unordered instruction which needs fencing with mfence or
* sfence to avoid ordering issues . For drm_clflush_page this fencing happens
* in the caller .
*/
2008-07-30 12:06:12 -07:00
static void
drm_clflush_page ( struct page * page )
{
uint8_t * page_virtual ;
unsigned int i ;
2012-09-19 11:12:41 +10:00
const int size = boot_cpu_data . x86_clflush_size ;
2008-07-30 12:06:12 -07:00
if ( unlikely ( page = = NULL ) )
return ;
2011-11-25 23:14:20 +08:00
page_virtual = kmap_atomic ( page ) ;
2012-09-19 11:12:41 +10:00
for ( i = 0 ; i < PAGE_SIZE ; i + = size )
2014-02-26 12:06:51 -07:00
clflushopt ( page_virtual + i ) ;
2011-11-25 23:14:20 +08:00
kunmap_atomic ( page_virtual ) ;
2008-07-30 12:06:12 -07:00
}
2009-08-27 09:53:47 +10:00
static void drm_cache_flush_clflush ( struct page * pages [ ] ,
unsigned long num_pages )
{
unsigned long i ;
mb ( ) ;
for ( i = 0 ; i < num_pages ; i + + )
drm_clflush_page ( * pages + + ) ;
mb ( ) ;
}
static void
drm_clflush_ipi_handler ( void * null )
{
wbinvd ( ) ;
}
# endif
2009-09-02 09:41:13 +10:00
2008-07-30 12:06:12 -07:00
void
drm_clflush_pages ( struct page * pages [ ] , unsigned long num_pages )
{
# if defined(CONFIG_X86)
if ( cpu_has_clflush ) {
2009-08-27 09:53:47 +10:00
drm_cache_flush_clflush ( pages , num_pages ) ;
2008-07-30 12:06:12 -07:00
return ;
}
2009-08-27 09:53:47 +10:00
if ( on_each_cpu ( drm_clflush_ipi_handler , NULL , 1 ) ! = 0 )
printk ( KERN_ERR " Timed out waiting for cache flush. \n " ) ;
# elif defined(__powerpc__)
unsigned long i ;
for ( i = 0 ; i < num_pages ; i + + ) {
struct page * page = pages [ i ] ;
void * page_virtual ;
if ( unlikely ( page = = NULL ) )
continue ;
2011-11-25 23:14:20 +08:00
page_virtual = kmap_atomic ( page ) ;
2009-08-27 09:53:47 +10:00
flush_dcache_range ( ( unsigned long ) page_virtual ,
( unsigned long ) page_virtual + PAGE_SIZE ) ;
2011-11-25 23:14:20 +08:00
kunmap_atomic ( page_virtual ) ;
2009-08-27 09:53:47 +10:00
}
# else
2009-09-02 09:41:13 +10:00
printk ( KERN_ERR " Architecture has no drm_cache.c support \n " ) ;
WARN_ON_ONCE ( 1 ) ;
2008-10-07 13:41:49 +10:00
# endif
2008-07-30 12:06:12 -07:00
}
EXPORT_SYMBOL ( drm_clflush_pages ) ;
2012-03-25 19:47:30 +02:00
2012-06-01 15:20:22 +01:00
void
drm_clflush_sg ( struct sg_table * st )
{
# if defined(CONFIG_X86)
if ( cpu_has_clflush ) {
2013-02-18 19:28:01 +02:00
struct sg_page_iter sg_iter ;
2012-06-01 15:20:22 +01:00
mb ( ) ;
2013-02-18 19:28:01 +02:00
for_each_sg_page ( st - > sgl , & sg_iter , st - > nents , 0 )
2013-03-26 15:14:18 +02:00
drm_clflush_page ( sg_page_iter_page ( & sg_iter ) ) ;
2012-06-01 15:20:22 +01:00
mb ( ) ;
return ;
}
if ( on_each_cpu ( drm_clflush_ipi_handler , NULL , 1 ) ! = 0 )
printk ( KERN_ERR " Timed out waiting for cache flush. \n " ) ;
# else
printk ( KERN_ERR " Architecture has no drm_cache.c support \n " ) ;
WARN_ON_ONCE ( 1 ) ;
# endif
}
EXPORT_SYMBOL ( drm_clflush_sg ) ;
2012-03-25 19:47:30 +02:00
void
2014-04-01 12:59:08 +03:00
drm_clflush_virt_range ( void * addr , unsigned long length )
2012-03-25 19:47:30 +02:00
{
# if defined(CONFIG_X86)
if ( cpu_has_clflush ) {
2014-04-01 12:59:08 +03:00
void * end = addr + length ;
2012-03-25 19:47:30 +02:00
mb ( ) ;
for ( ; addr < end ; addr + = boot_cpu_data . x86_clflush_size )
2014-05-14 09:41:12 -06:00
clflushopt ( addr ) ;
2014-02-26 12:06:52 -07:00
clflushopt ( end - 1 ) ;
2012-03-25 19:47:30 +02:00
mb ( ) ;
return ;
}
if ( on_each_cpu ( drm_clflush_ipi_handler , NULL , 1 ) ! = 0 )
printk ( KERN_ERR " Timed out waiting for cache flush. \n " ) ;
# else
printk ( KERN_ERR " Architecture has no drm_cache.c support \n " ) ;
WARN_ON_ONCE ( 1 ) ;
# endif
}
EXPORT_SYMBOL ( drm_clflush_virt_range ) ;