2 * Meta cache partition manipulation.
4 * Copyright 2010 Imagination Technologies Ltd.
7 #include <linux/kernel.h>
9 #include <linux/errno.h>
10 #include <asm/processor.h>
11 #include <asm/cachepart.h>
12 #include <asm/metag_isa.h>
13 #include <asm/metag_mem.h>
15 #define SYSC_DCPART(n) (SYSC_DCPART0 + SYSC_xCPARTn_STRIDE * (n))
16 #define SYSC_ICPART(n) (SYSC_ICPART0 + SYSC_xCPARTn_STRIDE * (n))
18 #define CACHE_ASSOCIATIVITY 4 /* 4 way set-assosiative */
22 /* The CORE_CONFIG2 register is not available on Meta 1 */
23 #ifdef CONFIG_METAG_META21
24 unsigned int get_dcache_size(void)
26 unsigned int config2
= metag_in32(METAC_CORE_CONFIG2
);
27 unsigned int sz
= 0x1000 << ((config2
& METAC_CORECFG2_DCSZ_BITS
)
28 >> METAC_CORECFG2_DCSZ_S
);
29 if (config2
& METAC_CORECFG2_DCSMALL_BIT
)
34 unsigned int get_icache_size(void)
36 unsigned int config2
= metag_in32(METAC_CORE_CONFIG2
);
37 unsigned int sz
= 0x1000 << ((config2
& METAC_CORE_C2ICSZ_BITS
)
38 >> METAC_CORE_C2ICSZ_S
);
39 if (config2
& METAC_CORECFG2_ICSMALL_BIT
)
44 unsigned int get_global_dcache_size(void)
46 unsigned int cpart
= metag_in32(SYSC_DCPART(hard_processor_id()));
47 unsigned int temp
= cpart
& SYSC_xCPARTG_AND_BITS
;
48 return (get_dcache_size() * ((temp
>> SYSC_xCPARTG_AND_S
) + 1)) >> 4;
51 unsigned int get_global_icache_size(void)
53 unsigned int cpart
= metag_in32(SYSC_ICPART(hard_processor_id()));
54 unsigned int temp
= cpart
& SYSC_xCPARTG_AND_BITS
;
55 return (get_icache_size() * ((temp
>> SYSC_xCPARTG_AND_S
) + 1)) >> 4;
58 static unsigned int get_thread_cache_size(unsigned int cache
, int thread_id
)
60 unsigned int cache_size
;
61 unsigned int t_cache_part
;
62 unsigned int isEnabled
;
63 unsigned int offset
= 0;
64 isEnabled
= (cache
== DCACHE
? metag_in32(MMCU_DCACHE_CTRL_ADDR
) & 0x1 :
65 metag_in32(MMCU_ICACHE_CTRL_ADDR
) & 0x1);
68 #if PAGE_OFFSET >= LINGLOBAL_BASE
69 /* Checking for global cache */
70 cache_size
= (cache
== DCACHE
? get_global_dcache_size() :
71 get_global_icache_size());
74 cache_size
= (cache
== DCACHE
? get_dcache_size() :
77 t_cache_part
= (cache
== DCACHE
?
78 (metag_in32(SYSC_DCPART(thread_id
)) >> offset
) & 0xF :
79 (metag_in32(SYSC_ICPART(thread_id
)) >> offset
) & 0xF);
80 switch (t_cache_part
) {
84 return cache_size
/ 2;
86 return cache_size
/ 4;
88 return cache_size
/ 8;
90 return cache_size
/ 16;
95 void check_for_cache_aliasing(int thread_id
)
97 unsigned int thread_cache_size
;
98 unsigned int cache_type
;
99 for (cache_type
= ICACHE
; cache_type
<= DCACHE
; cache_type
++) {
101 get_thread_cache_size(cache_type
, thread_id
);
102 if (thread_cache_size
< 0)
103 pr_emerg("Can't read %s cache size\n",
104 cache_type
? "DCACHE" : "ICACHE");
105 else if (thread_cache_size
== 0)
106 /* Cache is off. No need to check for aliasing */
108 if (thread_cache_size
/ CACHE_ASSOCIATIVITY
> PAGE_SIZE
) {
109 pr_emerg("Potential cache aliasing detected in %s on Thread %d\n",
110 cache_type
? "DCACHE" : "ICACHE", thread_id
);
111 pr_warn("Total %s size: %u bytes\n",
112 cache_type
? "DCACHE" : "ICACHE",
113 cache_type
? get_dcache_size()
114 : get_icache_size());
115 pr_warn("Thread %s size: %d bytes\n",
116 cache_type
? "CACHE" : "ICACHE",
118 pr_warn("Page Size: %lu bytes\n", PAGE_SIZE
);
119 panic("Potential cache aliasing detected");
126 void check_for_cache_aliasing(int thread_id
)