1 /* Copyright 2005 Oleg I. Vdovikin (oleg@cs.msu.su) */
2 /* cache manipulation adapted from Broadcom code */
3 /* idea taken from original bunzip2 decompressor code */
4 /* Copyright 2004 Manuel Novoa III (mjn3@codepoet.org) */
5 /* Licensed under the linux kernel's version of the GPL.*/
8 #include <asm/regdef.h>
10 #define KSEG0 0x80000000
16 #define CONF1_DA_SHIFT 7 /* D$ associativity */
17 #define CONF1_DA_MASK 0x00000380
18 #define CONF1_DA_BASE 1
19 #define CONF1_DL_SHIFT 10 /* D$ line size */
20 #define CONF1_DL_MASK 0x00001c00
21 #define CONF1_DL_BASE 2
22 #define CONF1_DS_SHIFT 13 /* D$ sets/way */
23 #define CONF1_DS_MASK 0x0000e000
24 #define CONF1_DS_BASE 64
25 #define CONF1_IA_SHIFT 16 /* I$ associativity */
26 #define CONF1_IA_MASK 0x00070000
27 #define CONF1_IA_BASE 1
28 #define CONF1_IL_SHIFT 19 /* I$ line size */
29 #define CONF1_IL_MASK 0x00380000
30 #define CONF1_IL_BASE 2
31 #define CONF1_IS_SHIFT 22 /* Instruction cache sets/way */
32 #define CONF1_IS_MASK 0x01c00000
33 #define CONF1_IS_BASE 64
35 #define CONF_AR (7 << 10)
37 #define Index_Invalidate_I 0x00
38 #define Index_Writeback_Inv_D 0x01
49 /* Copy decompressor code to the right place */
62 /* At this point we need to invalidate dcache and */
63 /* icache before jumping to new code */
65 1: /* Get cache sizes */
77 sll s1,t0,s1 /* s1 has D$ cache line size */
82 addiu s2,CONF1_DA_BASE /* s2 now has D$ associativity */
88 sll s3,s3,t0 /* s3 has D$ sets per way */
90 multu s2,s3 /* sets/way * associativity */
91 mflo t0 /* total cache lines */
93 multu s1,t0 /* D$ linesize * lines */
94 mflo s2 /* s2 is now D$ size in bytes */
96 /* Figure if it is a mips32r2 CPU which we take as an indication that
97 * there is no BRCM CP0 register and the D$ tags are in select 2
100 andi s6,CONF_AR # s6 != 0 if mips32r2
101 /* Initilize the D$: */
105 mtc0 zero,C0_TAGLO,2 # For mips32r2 the D$ Tags are in select 2
116 li t0,KSEG0 /* Just an address for the first $ line */
117 addu t1,t0,s2 /* + size of cache == end */
120 3: cache Index_Writeback_Inv_D,0(t0)
126 /* Now we get to do it all again for the I$ */
128 move s3,zero /* just in case there is no icache */
136 srl t0,CONF1_IL_SHIFT
138 sll s3,t0 /* s3 has I$ cache line size */
142 srl t0,CONF1_IA_SHIFT
143 addiu s4,t0,CONF1_IA_BASE /* s4 now has I$ associativity */
147 srl t0,CONF1_IS_SHIFT
149 sll s5,t0 /* s5 has I$ sets per way */
151 multu s4,s5 /* sets/way * associativity */
152 mflo t0 /* s4 is now total cache lines */
154 multu s3,t0 /* I$ linesize * lines */
155 mflo s4 /* s4 is cache size in bytes */
157 /* Initilize the I$: */
161 li t0,KSEG0 /* Just an address for the first $ line */
162 addu t1,t0,s4 /* + size of cache == end */
165 1: cache Index_Invalidate_I,0(t0)
171 move a0,s3 /* icache line size */
172 move a1,s4 /* icache size */
173 move a2,s1 /* dcache line size */
175 move a3,s2 /* dcache size */