diff options
Diffstat (limited to 'arch/arm')
-rw-r--r-- | arch/arm/include/asm/cachetype.h | 118 | ||||
-rw-r--r-- | arch/arm/kernel/setup.c | 22 |
2 files changed, 59 insertions, 81 deletions
diff --git a/arch/arm/include/asm/cachetype.h b/arch/arm/include/asm/cachetype.h index b52386bfd505..d3a4c2cb9f2f 100644 --- a/arch/arm/include/asm/cachetype.h +++ b/arch/arm/include/asm/cachetype.h | |||
@@ -1,96 +1,52 @@ | |||
1 | #ifndef __ASM_ARM_CACHETYPE_H | 1 | #ifndef __ASM_ARM_CACHETYPE_H |
2 | #define __ASM_ARM_CACHETYPE_H | 2 | #define __ASM_ARM_CACHETYPE_H |
3 | 3 | ||
4 | #include <asm/cputype.h> | 4 | #define CACHEID_VIVT (1 << 0) |
5 | #define CACHEID_VIPT_NONALIASING (1 << 1) | ||
6 | #define CACHEID_VIPT_ALIASING (1 << 2) | ||
7 | #define CACHEID_VIPT (CACHEID_VIPT_ALIASING|CACHEID_VIPT_NONALIASING) | ||
8 | #define CACHEID_ASID_TAGGED (1 << 3) | ||
5 | 9 | ||
6 | #define __cacheid_present(val) (val != read_cpuid_id()) | 10 | extern unsigned int cacheid; |
7 | #define __cacheid_type_v7(val) ((val & (7 << 29)) == (4 << 29)) | ||
8 | 11 | ||
9 | #define __cacheid_vivt_prev7(val) ((val & (15 << 25)) != (14 << 25)) | 12 | #define cache_is_vivt() cacheid_is(CACHEID_VIVT) |
10 | #define __cacheid_vipt_prev7(val) ((val & (15 << 25)) == (14 << 25)) | 13 | #define cache_is_vipt() cacheid_is(CACHEID_VIPT) |
11 | #define __cacheid_vipt_nonaliasing_prev7(val) ((val & (15 << 25 | 1 << 23)) == (14 << 25)) | 14 | #define cache_is_vipt_nonaliasing() cacheid_is(CACHEID_VIPT_NONALIASING) |
12 | #define __cacheid_vipt_aliasing_prev7(val) ((val & (15 << 25 | 1 << 23)) == (14 << 25 | 1 << 23)) | 15 | #define cache_is_vipt_aliasing() cacheid_is(CACHEID_VIPT_ALIASING) |
16 | #define icache_is_vivt_asid_tagged() cacheid_is(CACHEID_ASID_TAGGED) | ||
13 | 17 | ||
14 | #define __cacheid_vivt(val) (__cacheid_type_v7(val) ? 0 : __cacheid_vivt_prev7(val)) | ||
15 | #define __cacheid_vipt(val) (__cacheid_type_v7(val) ? 1 : __cacheid_vipt_prev7(val)) | ||
16 | #define __cacheid_vipt_nonaliasing(val) (__cacheid_type_v7(val) ? 1 : __cacheid_vipt_nonaliasing_prev7(val)) | ||
17 | #define __cacheid_vipt_aliasing(val) (__cacheid_type_v7(val) ? 0 : __cacheid_vipt_aliasing_prev7(val)) | ||
18 | #define __cacheid_vivt_asid_tagged_instr(val) (__cacheid_type_v7(val) ? ((val & (3 << 14)) == (1 << 14)) : 0) | ||
19 | |||
20 | #if defined(CONFIG_CPU_CACHE_VIVT) && !defined(CONFIG_CPU_CACHE_VIPT) | ||
21 | /* | 18 | /* |
22 | * VIVT caches only | 19 | * __LINUX_ARM_ARCH__ is the minimum supported CPU architecture |
20 | * Mask out support which will never be present on newer CPUs. | ||
21 | * - v6+ is never VIVT | ||
22 | * - v7+ VIPT never aliases | ||
23 | */ | 23 | */ |
24 | #define cache_is_vivt() 1 | 24 | #if __LINUX_ARM_ARCH__ >= 7 |
25 | #define cache_is_vipt() 0 | 25 | #define __CACHEID_ARCH_MIN (CACHEID_VIPT_NONALIASING | CACHEID_ASID_TAGGED) |
26 | #define cache_is_vipt_nonaliasing() 0 | 26 | #elif __LINUX_ARM_ARCH__ >= 6 |
27 | #define cache_is_vipt_aliasing() 0 | 27 | #define __CACHEID_ARCH_MIN (~CACHEID_VIVT) |
28 | #define icache_is_vivt_asid_tagged() 0 | 28 | #else |
29 | #define __CACHEID_ARCH_MIN (~0) | ||
30 | #endif | ||
29 | 31 | ||
30 | #elif !defined(CONFIG_CPU_CACHE_VIVT) && defined(CONFIG_CPU_CACHE_VIPT) | ||
31 | /* | 32 | /* |
32 | * VIPT caches only | 33 | * Mask out support which isn't configured |
33 | */ | 34 | */ |
34 | #define cache_is_vivt() 0 | 35 | #if defined(CONFIG_CPU_CACHE_VIVT) && !defined(CONFIG_CPU_CACHE_VIPT) |
35 | #define cache_is_vipt() 1 | 36 | #define __CACHEID_ALWAYS (CACHEID_VIVT) |
36 | #define cache_is_vipt_nonaliasing() \ | 37 | #define __CACHEID_NEVER (~CACHEID_VIVT) |
37 | ({ \ | 38 | #elif !defined(CONFIG_CPU_CACHE_VIVT) && defined(CONFIG_CPU_CACHE_VIPT) |
38 | unsigned int __val = read_cpuid_cachetype(); \ | 39 | #define __CACHEID_ALWAYS (0) |
39 | __cacheid_vipt_nonaliasing(__val); \ | 40 | #define __CACHEID_NEVER (CACHEID_VIVT) |
40 | }) | ||
41 | |||
42 | #define cache_is_vipt_aliasing() \ | ||
43 | ({ \ | ||
44 | unsigned int __val = read_cpuid_cachetype(); \ | ||
45 | __cacheid_vipt_aliasing(__val); \ | ||
46 | }) | ||
47 | |||
48 | #define icache_is_vivt_asid_tagged() \ | ||
49 | ({ \ | ||
50 | unsigned int __val = read_cpuid_cachetype(); \ | ||
51 | __cacheid_vivt_asid_tagged_instr(__val); \ | ||
52 | }) | ||
53 | |||
54 | #else | 41 | #else |
55 | /* | 42 | #define __CACHEID_ALWAYS (0) |
56 | * VIVT or VIPT caches. Note that this is unreliable since ARM926 | 43 | #define __CACHEID_NEVER (0) |
57 | * and V6 CPUs satisfy the "(val & (15 << 25)) == (14 << 25)" test. | ||
58 | * There's no way to tell from the CacheType register what type (!) | ||
59 | * the cache is. | ||
60 | */ | ||
61 | #define cache_is_vivt() \ | ||
62 | ({ \ | ||
63 | unsigned int __val = read_cpuid_cachetype(); \ | ||
64 | (!__cacheid_present(__val)) || __cacheid_vivt(__val); \ | ||
65 | }) | ||
66 | |||
67 | #define cache_is_vipt() \ | ||
68 | ({ \ | ||
69 | unsigned int __val = read_cpuid_cachetype(); \ | ||
70 | __cacheid_present(__val) && __cacheid_vipt(__val); \ | ||
71 | }) | ||
72 | |||
73 | #define cache_is_vipt_nonaliasing() \ | ||
74 | ({ \ | ||
75 | unsigned int __val = read_cpuid_cachetype(); \ | ||
76 | __cacheid_present(__val) && \ | ||
77 | __cacheid_vipt_nonaliasing(__val); \ | ||
78 | }) | ||
79 | |||
80 | #define cache_is_vipt_aliasing() \ | ||
81 | ({ \ | ||
82 | unsigned int __val = read_cpuid_cachetype(); \ | ||
83 | __cacheid_present(__val) && \ | ||
84 | __cacheid_vipt_aliasing(__val); \ | ||
85 | }) | ||
86 | |||
87 | #define icache_is_vivt_asid_tagged() \ | ||
88 | ({ \ | ||
89 | unsigned int __val = read_cpuid_cachetype(); \ | ||
90 | __cacheid_present(__val) && \ | ||
91 | __cacheid_vivt_asid_tagged_instr(__val); \ | ||
92 | }) | ||
93 | |||
94 | #endif | 44 | #endif |
95 | 45 | ||
46 | static inline unsigned int __attribute__((pure)) cacheid_is(unsigned int mask) | ||
47 | { | ||
48 | return (__CACHEID_ALWAYS & mask) | | ||
49 | (~__CACHEID_NEVER & __CACHEID_ARCH_MIN & mask & cacheid); | ||
50 | } | ||
51 | |||
96 | #endif | 52 | #endif |
diff --git a/arch/arm/kernel/setup.c b/arch/arm/kernel/setup.c index 1939c904dae3..5b121d81f9eb 100644 --- a/arch/arm/kernel/setup.c +++ b/arch/arm/kernel/setup.c | |||
@@ -67,6 +67,8 @@ unsigned int processor_id; | |||
67 | EXPORT_SYMBOL(processor_id); | 67 | EXPORT_SYMBOL(processor_id); |
68 | unsigned int __machine_arch_type; | 68 | unsigned int __machine_arch_type; |
69 | EXPORT_SYMBOL(__machine_arch_type); | 69 | EXPORT_SYMBOL(__machine_arch_type); |
70 | unsigned int cacheid; | ||
71 | EXPORT_SYMBOL(cacheid); | ||
70 | 72 | ||
71 | unsigned int __atags_pointer __initdata; | 73 | unsigned int __atags_pointer __initdata; |
72 | 74 | ||
@@ -229,6 +231,25 @@ int cpu_architecture(void) | |||
229 | return cpu_arch; | 231 | return cpu_arch; |
230 | } | 232 | } |
231 | 233 | ||
234 | static void __init cacheid_init(void) | ||
235 | { | ||
236 | unsigned int cachetype = read_cpuid_cachetype(); | ||
237 | unsigned int arch = cpu_architecture(); | ||
238 | |||
239 | if (arch >= CPU_ARCH_ARMv7) { | ||
240 | cacheid = CACHEID_VIPT_NONALIASING; | ||
241 | if ((cachetype & (3 << 14)) == 1 << 14) | ||
242 | cacheid |= CACHEID_ASID_TAGGED; | ||
243 | } else if (arch >= CPU_ARCH_ARMv6) { | ||
244 | if (cachetype & (1 << 23)) | ||
245 | cacheid = CACHEID_VIPT_ALIASING; | ||
246 | else | ||
247 | cacheid = CACHEID_VIPT_NONALIASING; | ||
248 | } else { | ||
249 | cacheid = CACHEID_VIVT; | ||
250 | } | ||
251 | } | ||
252 | |||
232 | /* | 253 | /* |
233 | * These functions re-use the assembly code in head.S, which | 254 | * These functions re-use the assembly code in head.S, which |
234 | * already provide the required functionality. | 255 | * already provide the required functionality. |
@@ -278,6 +299,7 @@ static void __init setup_processor(void) | |||
278 | elf_hwcap &= ~HWCAP_THUMB; | 299 | elf_hwcap &= ~HWCAP_THUMB; |
279 | #endif | 300 | #endif |
280 | 301 | ||
302 | cacheid_init(); | ||
281 | cpu_proc_init(); | 303 | cpu_proc_init(); |
282 | } | 304 | } |
283 | 305 | ||