blob: e9f054fc7e870cf060623d79c077cbfb5ee423be [file] [log] [blame]
Lennert Buytenhek573a6522009-11-24 19:33:52 +02001/*
2 * arch/arm/mm/cache-tauros2.c - Tauros2 L2 cache controller support
3 *
4 * Copyright (C) 2008 Marvell Semiconductor
5 *
6 * This file is licensed under the terms of the GNU General Public
7 * License version 2. This program is licensed "as is" without any
8 * warranty of any kind, whether express or implied.
9 *
10 * References:
11 * - PJ1 CPU Core Datasheet,
12 * Document ID MV-S104837-01, Rev 0.7, January 24 2008.
13 * - PJ4 CPU Core Datasheet,
14 * Document ID MV-S105190-00, Rev 0.7, March 14 2008.
15 */
16
17#include <linux/init.h>
18#include <asm/cacheflush.h>
Russell King15d07dc2012-03-28 18:30:01 +010019#include <asm/cp15.h>
Chao Xiefa79b8d2012-07-31 14:13:11 +080020#include <asm/cputype.h>
Lennert Buytenhek573a6522009-11-24 19:33:52 +020021#include <asm/hardware/cache-tauros2.h>
22
23
24/*
25 * When Tauros2 is used on a CPU that supports the v7 hierarchical
26 * cache operations, the cache handling code in proc-v7.S takes care
27 * of everything, including handling DMA coherency.
28 *
29 * So, we only need to register outer cache operations here if we're
30 * being used on a pre-v7 CPU, and we only need to build support for
31 * outer cache operations into the kernel image if the kernel has been
32 * configured to support a pre-v7 CPU.
33 */
34#if __LINUX_ARM_ARCH__ < 7
35/*
36 * Low-level cache maintenance operations.
37 */
38static inline void tauros2_clean_pa(unsigned long addr)
39{
40 __asm__("mcr p15, 1, %0, c7, c11, 3" : : "r" (addr));
41}
42
43static inline void tauros2_clean_inv_pa(unsigned long addr)
44{
45 __asm__("mcr p15, 1, %0, c7, c15, 3" : : "r" (addr));
46}
47
48static inline void tauros2_inv_pa(unsigned long addr)
49{
50 __asm__("mcr p15, 1, %0, c7, c7, 3" : : "r" (addr));
51}
52
53
54/*
55 * Linux primitives.
56 *
57 * Note that the end addresses passed to Linux primitives are
58 * noninclusive.
59 */
60#define CACHE_LINE_SIZE 32
61
62static void tauros2_inv_range(unsigned long start, unsigned long end)
63{
64 /*
65 * Clean and invalidate partial first cache line.
66 */
67 if (start & (CACHE_LINE_SIZE - 1)) {
68 tauros2_clean_inv_pa(start & ~(CACHE_LINE_SIZE - 1));
69 start = (start | (CACHE_LINE_SIZE - 1)) + 1;
70 }
71
72 /*
73 * Clean and invalidate partial last cache line.
74 */
75 if (end & (CACHE_LINE_SIZE - 1)) {
76 tauros2_clean_inv_pa(end & ~(CACHE_LINE_SIZE - 1));
77 end &= ~(CACHE_LINE_SIZE - 1);
78 }
79
80 /*
81 * Invalidate all full cache lines between 'start' and 'end'.
82 */
83 while (start < end) {
84 tauros2_inv_pa(start);
85 start += CACHE_LINE_SIZE;
86 }
87
88 dsb();
89}
90
91static void tauros2_clean_range(unsigned long start, unsigned long end)
92{
93 start &= ~(CACHE_LINE_SIZE - 1);
94 while (start < end) {
95 tauros2_clean_pa(start);
96 start += CACHE_LINE_SIZE;
97 }
98
99 dsb();
100}
101
102static void tauros2_flush_range(unsigned long start, unsigned long end)
103{
104 start &= ~(CACHE_LINE_SIZE - 1);
105 while (start < end) {
106 tauros2_clean_inv_pa(start);
107 start += CACHE_LINE_SIZE;
108 }
109
110 dsb();
111}
Chao Xie89326f762012-05-07 11:23:59 +0800112
113static void tauros2_disable(void)
114{
115 __asm__ __volatile__ (
116 "mcr p15, 1, %0, c7, c11, 0 @L2 Cache Clean All\n\t"
117 "mrc p15, 0, %0, c1, c0, 0\n\t"
118 "bic %0, %0, #(1 << 26)\n\t"
119 "mcr p15, 0, %0, c1, c0, 0 @Disable L2 Cache\n\t"
120 : : "r" (0x0));
121}
122
123static void tauros2_resume(void)
124{
125 __asm__ __volatile__ (
126 "mcr p15, 1, %0, c7, c7, 0 @L2 Cache Invalidate All\n\t"
127 "mrc p15, 0, %0, c1, c0, 0\n\t"
128 "orr %0, %0, #(1 << 26)\n\t"
129 "mcr p15, 0, %0, c1, c0, 0 @Enable L2 Cache\n\t"
130 : : "r" (0x0));
131}
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200132#endif
133
134static inline u32 __init read_extra_features(void)
135{
136 u32 u;
137
138 __asm__("mrc p15, 1, %0, c15, c1, 0" : "=r" (u));
139
140 return u;
141}
142
143static inline void __init write_extra_features(u32 u)
144{
145 __asm__("mcr p15, 1, %0, c15, c1, 0" : : "r" (u));
146}
147
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200148static inline int __init cpuid_scheme(void)
149{
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200150 return !!((processor_id & 0x000f0000) == 0x000f0000);
151}
152
153static inline u32 __init read_mmfr3(void)
154{
155 u32 mmfr3;
156
157 __asm__("mrc p15, 0, %0, c0, c1, 7\n" : "=r" (mmfr3));
158
159 return mmfr3;
160}
161
162static inline u32 __init read_actlr(void)
163{
164 u32 actlr;
165
166 __asm__("mrc p15, 0, %0, c1, c0, 1\n" : "=r" (actlr));
167
168 return actlr;
169}
170
171static inline void __init write_actlr(u32 actlr)
172{
173 __asm__("mcr p15, 0, %0, c1, c0, 1\n" : : "r" (actlr));
174}
175
Chao Xie38f2e372012-07-31 14:13:12 +0800176static void enable_extra_feature(unsigned int features)
177{
178 u32 u;
179
180 u = read_extra_features();
181
182 if (features & CACHE_TAUROS2_PREFETCH_ON)
183 u &= ~0x01000000;
184 else
185 u |= 0x01000000;
186 printk(KERN_INFO "Tauros2: %s L2 prefetch.\n",
187 (features & CACHE_TAUROS2_PREFETCH_ON)
188 ? "Enabling" : "Disabling");
189
190 if (features & CACHE_TAUROS2_LINEFILL_BURST8)
191 u |= 0x00100000;
192 else
193 u &= ~0x00100000;
194 printk(KERN_INFO "Tauros2: %s line fill burt8.\n",
195 (features & CACHE_TAUROS2_LINEFILL_BURST8)
196 ? "Enabling" : "Disabling");
197
198 write_extra_features(u);
199}
200
201void __init tauros2_init(unsigned int features)
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200202{
Chao Xie5967b542012-07-31 14:13:10 +0800203 char *mode = NULL;
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200204
Chao Xie38f2e372012-07-31 14:13:12 +0800205 enable_extra_feature(features);
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200206
207#ifdef CONFIG_CPU_32v5
208 if ((processor_id & 0xff0f0000) == 0x56050000) {
209 u32 feat;
210
211 /*
212 * v5 CPUs with Tauros2 have the L2 cache enable bit
213 * located in the CPU Extra Features register.
214 */
215 feat = read_extra_features();
216 if (!(feat & 0x00400000)) {
217 printk(KERN_INFO "Tauros2: Enabling L2 cache.\n");
218 write_extra_features(feat | 0x00400000);
219 }
220
221 mode = "ARMv5";
222 outer_cache.inv_range = tauros2_inv_range;
223 outer_cache.clean_range = tauros2_clean_range;
224 outer_cache.flush_range = tauros2_flush_range;
Chao Xie89326f762012-05-07 11:23:59 +0800225 outer_cache.disable = tauros2_disable;
226 outer_cache.resume = tauros2_resume;
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200227 }
228#endif
229
230#ifdef CONFIG_CPU_32v6
231 /*
232 * Check whether this CPU lacks support for the v7 hierarchical
233 * cache ops. (PJ4 is in its v6 personality mode if the MMFR3
234 * register indicates no support for the v7 hierarchical cache
235 * ops.)
236 */
237 if (cpuid_scheme() && (read_mmfr3() & 0xf) == 0) {
238 /*
239 * When Tauros2 is used in an ARMv6 system, the L2
240 * enable bit is in the ARMv6 ARM-mandated position
241 * (bit [26] of the System Control Register).
242 */
243 if (!(get_cr() & 0x04000000)) {
244 printk(KERN_INFO "Tauros2: Enabling L2 cache.\n");
245 adjust_cr(0x04000000, 0x04000000);
246 }
247
248 mode = "ARMv6";
249 outer_cache.inv_range = tauros2_inv_range;
250 outer_cache.clean_range = tauros2_clean_range;
251 outer_cache.flush_range = tauros2_flush_range;
Chao Xie89326f762012-05-07 11:23:59 +0800252 outer_cache.disable = tauros2_disable;
253 outer_cache.resume = tauros2_resume;
Lennert Buytenhek573a6522009-11-24 19:33:52 +0200254 }
255#endif
256
257#ifdef CONFIG_CPU_32v7
258 /*
259 * Check whether this CPU has support for the v7 hierarchical
260 * cache ops. (PJ4 is in its v7 personality mode if the MMFR3
261 * register indicates support for the v7 hierarchical cache
262 * ops.)
263 *
264 * (Although strictly speaking there may exist CPUs that
265 * implement the v7 cache ops but are only ARMv6 CPUs (due to
266 * not complying with all of the other ARMv7 requirements),
267 * there are no real-life examples of Tauros2 being used on
268 * such CPUs as of yet.)
269 */
270 if (cpuid_scheme() && (read_mmfr3() & 0xf) == 1) {
271 u32 actlr;
272
273 /*
274 * When Tauros2 is used in an ARMv7 system, the L2
275 * enable bit is located in the Auxiliary System Control
276 * Register (which is the only register allowed by the
277 * ARMv7 spec to contain fine-grained cache control bits).
278 */
279 actlr = read_actlr();
280 if (!(actlr & 0x00000002)) {
281 printk(KERN_INFO "Tauros2: Enabling L2 cache.\n");
282 write_actlr(actlr | 0x00000002);
283 }
284
285 mode = "ARMv7";
286 }
287#endif
288
289 if (mode == NULL) {
290 printk(KERN_CRIT "Tauros2: Unable to detect CPU mode.\n");
291 return;
292 }
293
294 printk(KERN_INFO "Tauros2: L2 cache support initialised "
295 "in %s mode.\n", mode);
296}