aboutsummaryrefslogtreecommitdiffstats
path: root/arch/sh/mm/cache-sh4.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/sh/mm/cache-sh4.c')
-rw-r--r--arch/sh/mm/cache-sh4.c61
1 files changed, 26 insertions, 35 deletions
diff --git a/arch/sh/mm/cache-sh4.c b/arch/sh/mm/cache-sh4.c
index 9201b37c7cca..e3b77f0fa470 100644
--- a/arch/sh/mm/cache-sh4.c
+++ b/arch/sh/mm/cache-sh4.c
@@ -48,48 +48,44 @@ static void sh4_flush_icache_range(void *args)
48 struct flusher_data *data = args; 48 struct flusher_data *data = args;
49 int icacheaddr; 49 int icacheaddr;
50 unsigned long start, end; 50 unsigned long start, end;
51 unsigned long flags, v; 51 unsigned long v;
52 int i; 52 int i;
53 53
54 start = data->addr1; 54 start = data->addr1;
55 end = data->addr2; 55 end = data->addr2;
56 56
57 /* If there are too many pages then just blow the caches */ 57 /* If there are too many pages then just blow the caches */
58 if (((end - start) >> PAGE_SHIFT) >= MAX_ICACHE_PAGES) { 58 if (((end - start) >> PAGE_SHIFT) >= MAX_ICACHE_PAGES) {
59 local_flush_cache_all(args); 59 local_flush_cache_all(args);
60 } else { 60 } else {
61 /* selectively flush d-cache then invalidate the i-cache */ 61 /* selectively flush d-cache then invalidate the i-cache */
62 /* this is inefficient, so only use for small ranges */ 62 /* this is inefficient, so only use for small ranges */
63 start &= ~(L1_CACHE_BYTES-1); 63 start &= ~(L1_CACHE_BYTES-1);
64 end += L1_CACHE_BYTES-1; 64 end += L1_CACHE_BYTES-1;
65 end &= ~(L1_CACHE_BYTES-1); 65 end &= ~(L1_CACHE_BYTES-1);
66 66
67 local_irq_save(flags); 67 jump_to_uncached();
68 jump_to_uncached(); 68
69 69 for (v = start; v < end; v+=L1_CACHE_BYTES) {
70 for (v = start; v < end; v+=L1_CACHE_BYTES) { 70 __ocbwb(v);
71 asm volatile("ocbwb %0" 71
72 : /* no output */ 72 icacheaddr = CACHE_IC_ADDRESS_ARRAY |
73 : "m" (__m(v))); 73 (v & cpu_data->icache.entry_mask);
74 74
75 icacheaddr = CACHE_IC_ADDRESS_ARRAY | ( 75 for (i = 0; i < cpu_data->icache.ways;
76 v & cpu_data->icache.entry_mask); 76 i++, icacheaddr += cpu_data->icache.way_incr)
77 77 /* Clear i-cache line valid-bit */
78 for (i = 0; i < cpu_data->icache.ways; 78 ctrl_outl(0, icacheaddr);
79 i++, icacheaddr += cpu_data->icache.way_incr) 79 }
80 /* Clear i-cache line valid-bit */
81 ctrl_outl(0, icacheaddr);
82 }
83 80
84 back_to_cached(); 81 back_to_cached();
85 local_irq_restore(flags);
86 } 82 }
87} 83}
88 84
89static inline void flush_cache_4096(unsigned long start, 85static inline void flush_cache_4096(unsigned long start,
90 unsigned long phys) 86 unsigned long phys)
91{ 87{
92 unsigned long flags, exec_offset = 0; 88 unsigned long exec_offset = 0;
93 89
94 /* 90 /*
95 * All types of SH-4 require PC to be in P2 to operate on the I-cache. 91 * All types of SH-4 require PC to be in P2 to operate on the I-cache.
@@ -99,10 +95,8 @@ static inline void flush_cache_4096(unsigned long start,
99 (start < CACHE_OC_ADDRESS_ARRAY)) 95 (start < CACHE_OC_ADDRESS_ARRAY))
100 exec_offset = 0x20000000; 96 exec_offset = 0x20000000;
101 97
102 local_irq_save(flags);
103 __flush_cache_4096(start | SH_CACHE_ASSOC, 98 __flush_cache_4096(start | SH_CACHE_ASSOC,
104 P1SEGADDR(phys), exec_offset); 99 P1SEGADDR(phys), exec_offset);
105 local_irq_restore(flags);
106} 100}
107 101
108/* 102/*
@@ -135,9 +129,8 @@ static void sh4_flush_dcache_page(void *page)
135/* TODO: Selective icache invalidation through IC address array.. */ 129/* TODO: Selective icache invalidation through IC address array.. */
136static void __uses_jump_to_uncached flush_icache_all(void) 130static void __uses_jump_to_uncached flush_icache_all(void)
137{ 131{
138 unsigned long flags, ccr; 132 unsigned long ccr;
139 133
140 local_irq_save(flags);
141 jump_to_uncached(); 134 jump_to_uncached();
142 135
143 /* Flush I-cache */ 136 /* Flush I-cache */
@@ -149,9 +142,7 @@ static void __uses_jump_to_uncached flush_icache_all(void)
149 * back_to_cached() will take care of the barrier for us, don't add 142 * back_to_cached() will take care of the barrier for us, don't add
150 * another one! 143 * another one!
151 */ 144 */
152
153 back_to_cached(); 145 back_to_cached();
154 local_irq_restore(flags);
155} 146}
156 147
157static inline void flush_dcache_all(void) 148static inline void flush_dcache_all(void)