aboutsummaryrefslogtreecommitdiffstats
path: root/arch/parisc
diff options
context:
space:
mode:
authorJohn David Anglin <dave.anglin@bell.net>2017-03-11 18:03:34 -0500
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2017-03-26 07:05:57 -0400
commita690a42ae7b90ceb761816fb33ee39058962d58e (patch)
treeaf7bd0ed1d8d198798cd94628b455e0e5c347bd7 /arch/parisc
parent66e70bdca599abd2dc42c5849ae21895de141bfe (diff)
parisc: Optimize flush_kernel_vmap_range and invalidate_kernel_vmap_range
commit 316ec0624f951166daedbe446988ef92ae72b59f upstream. The previously submitted patch did not resolve the random segmentation faults observed on the phantom buildd system. There are still unresolved problems with the Debian 4.8 and 4.9 kernels on C8000. The attached patch removes the flush of the offset map pages and does a whole data cache flush for large ranges. No other arch flushes the offset map in these routines as far as I can tell. I have not observed any random segmentation faults on rp3440 in two weeks of testing with 4.10.0 and 4.10.1. Signed-off-by: John David Anglin <dave.anglin@bell.net> Signed-off-by: Helge Deller <deller@gmx.de> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'arch/parisc')
-rw-r--r--arch/parisc/include/asm/cacheflush.h23
-rw-r--r--arch/parisc/kernel/cache.c22
2 files changed, 24 insertions, 21 deletions
diff --git a/arch/parisc/include/asm/cacheflush.h b/arch/parisc/include/asm/cacheflush.h
index 7bd69bd43a01..1d8c24dc04d4 100644
--- a/arch/parisc/include/asm/cacheflush.h
+++ b/arch/parisc/include/asm/cacheflush.h
@@ -45,28 +45,9 @@ static inline void flush_kernel_dcache_page(struct page *page)
45 45
46#define flush_kernel_dcache_range(start,size) \ 46#define flush_kernel_dcache_range(start,size) \
47 flush_kernel_dcache_range_asm((start), (start)+(size)); 47 flush_kernel_dcache_range_asm((start), (start)+(size));
48/* vmap range flushes and invalidates. Architecturally, we don't need
49 * the invalidate, because the CPU should refuse to speculate once an
50 * area has been flushed, so invalidate is left empty */
51static inline void flush_kernel_vmap_range(void *vaddr, int size)
52{
53 unsigned long start = (unsigned long)vaddr;
54
55 flush_kernel_dcache_range_asm(start, start + size);
56}
57static inline void invalidate_kernel_vmap_range(void *vaddr, int size)
58{
59 unsigned long start = (unsigned long)vaddr;
60 void *cursor = vaddr;
61 48
62 for ( ; cursor < vaddr + size; cursor += PAGE_SIZE) { 49void flush_kernel_vmap_range(void *vaddr, int size);
63 struct page *page = vmalloc_to_page(cursor); 50void invalidate_kernel_vmap_range(void *vaddr, int size);
64
65 if (test_and_clear_bit(PG_dcache_dirty, &page->flags))
66 flush_kernel_dcache_page(page);
67 }
68 flush_kernel_dcache_range_asm(start, start + size);
69}
70 51
71#define flush_cache_vmap(start, end) flush_cache_all() 52#define flush_cache_vmap(start, end) flush_cache_all()
72#define flush_cache_vunmap(start, end) flush_cache_all() 53#define flush_cache_vunmap(start, end) flush_cache_all()
diff --git a/arch/parisc/kernel/cache.c b/arch/parisc/kernel/cache.c
index 977f0a4f5ecf..53ec75f8e237 100644
--- a/arch/parisc/kernel/cache.c
+++ b/arch/parisc/kernel/cache.c
@@ -633,3 +633,25 @@ flush_cache_page(struct vm_area_struct *vma, unsigned long vmaddr, unsigned long
633 __flush_cache_page(vma, vmaddr, PFN_PHYS(pfn)); 633 __flush_cache_page(vma, vmaddr, PFN_PHYS(pfn));
634 } 634 }
635} 635}
636
637void flush_kernel_vmap_range(void *vaddr, int size)
638{
639 unsigned long start = (unsigned long)vaddr;
640
641 if ((unsigned long)size > parisc_cache_flush_threshold)
642 flush_data_cache();
643 else
644 flush_kernel_dcache_range_asm(start, start + size);
645}
646EXPORT_SYMBOL(flush_kernel_vmap_range);
647
648void invalidate_kernel_vmap_range(void *vaddr, int size)
649{
650 unsigned long start = (unsigned long)vaddr;
651
652 if ((unsigned long)size > parisc_cache_flush_threshold)
653 flush_data_cache();
654 else
655 flush_kernel_dcache_range_asm(start, start + size);
656}
657EXPORT_SYMBOL(invalidate_kernel_vmap_range);