make vfs & filesystems use failable copying
[minix3.git] / kernel / arch / earm / include / cpufunc.h
blobc5606dbe4d06ada8578dd868948c8d5e9e37a8e1
1 #ifndef _ARM_CPUFUNC_H
2 #define _ARM_CPUFUNC_H
4 #if 0
5 /* check interrupt state */
6 static inline void check_int(unsigned int state, int line)
8 unsigned int cpsr = 0;
10 asm volatile("mrs %0, cpsr" : "=r" (cpsr));
12 if ((cpsr & PSR_F) != (state & PSR_F))
13 printf("%d: FIQs are unexpectedly %s\n", line, (cpsr & PSR_F) ? "MASKED" : "UNMASKED");
15 if ((cpsr & PSR_I) != (state & PSR_I))
16 printf("%d: IRQs are unexpectedly %s\n", line, (cpsr & PSR_I) ? "MASKED" : "UNMASKED");
19 #endif
21 /* Data memory barrier */
22 static inline void dmb(void)
24 asm volatile("dmb" : : : "memory");
27 /* Data synchronization barrier */
28 static inline void dsb(void)
30 asm volatile("dsb" : : : "memory");
33 /* Instruction synchronization barrier */
34 static inline void isb(void)
36 asm volatile("isb" : : : "memory");
39 static inline void barrier(void)
41 dsb();
42 isb();
46 /* Read CLIDR, Cache Level ID Register */
47 static inline u32_t read_clidr(){
48 u32_t clidr;
49 asm volatile("mrc p15, 1, %[clidr], c0, c0 , 1 @ READ CLIDR\n\t"
50 : [clidr] "=r" (clidr));
51 return clidr;
55 /* Read CSSELR, Cache Size Selection Register */
56 static inline u32_t read_csselr(){
57 u32_t csselr;
58 asm volatile("mrc p15, 2, %[csselr], c0, c0 , 0 @ READ CSSELR\n\t"
59 : [csselr] "=r" (csselr));
60 return csselr;
63 /* Write CSSELR, Cache Size Selection Register */
64 static inline void write_csselr(u32_t csselr){
65 asm volatile("mcr p15, 2, %[csselr], c0, c0 , 0 @ WRITE CSSELR\n\t"
66 : : [csselr] "r" (csselr));
69 /* Read Cache Size ID Register */
70 static inline u32_t read_ccsidr()
72 u32_t ccsidr;
73 asm volatile("mrc p15, 1, %[ccsidr], c0, c0, 0 @ Read CCSIDR\n\t"
74 : [ccsidr] "=r" (ccsidr));
75 return ccsidr;
78 /* Read TLBTR, TLB Type Register */
79 static inline u32_t read_tlbtr()
81 u32_t tlbtr;
82 asm volatile("mrc p15, 0, %[tlbtr], c0, c0, 3 @ Read TLBTR\n\t"
83 : [tlbtr] "=r" (tlbtr));
84 return tlbtr;
87 /* keesj:move these out */
88 static inline u32_t ilog2(u32_t t)
90 u32_t counter =0;
91 while( (t = t >> 1) ) counter ++;
92 return counter;
95 /* keesj:move these out */
96 static inline u32_t ipow2(u32_t t)
98 return 1 << t;
102 * type = 1 == CLEAN
103 * type = 2 == INVALIDATE
105 static inline void dcache_maint(int type){
106 u32_t cache_level ;
107 u32_t clidr;
108 u32_t ctype;
109 u32_t ccsidr;
110 u32_t line_size,line_length;
111 u32_t number_of_sets,number_of_ways;
112 u32_t set,way;
114 clidr = read_clidr();
115 u32_t loc = ( clidr >> 24) & 0x7;
116 u32_t louu = ( clidr >> 27) & 0x7;
117 u32_t louis = ( clidr >> 21) & 0x7;
118 for (cache_level =0 ; cache_level < loc; cache_level++){
119 /* get current cache type */
120 ctype = ( clidr >> cache_level*3) & 0x7;
121 /* select data or unified or cache level */
122 write_csselr(cache_level << 1);
123 isb();
124 ccsidr = read_ccsidr();
125 line_size = ccsidr & 0x7;
126 line_length = 2 << (line_size + 1) ; /* 2**(line_size + 2) */
127 number_of_sets = ((ccsidr >> 13) & 0x7fff) + 1;
128 number_of_ways = ((ccsidr >> 3) & 0x3ff) + 1;
130 u32_t way_bits = ilog2(number_of_ways);
131 if(ipow2(ilog2(number_of_ways) < number_of_ways) ) {
132 way_bits++;
135 u32_t l = ilog2(line_length);
136 for (way =0 ; way < number_of_ways; way++) {
137 for (set =0 ; set < number_of_sets; set++) {
138 u32_t val = ( way << (32 - way_bits) ) | (set << l) | (cache_level << 1 );
139 if (type == 1) {
140 /* DCCISW, Data Cache Clean and Invalidate by Set/Way */
141 asm volatile("mcr p15, 0, %[set], c7, c14, 2 @ DCCISW"
142 : : [set] "r" (val));
143 } else if (type ==2 ){
144 /* DCISW, Data Cache Invalidate by Set/Way */
145 asm volatile("mcr p15, 0, %[set], c7, c6, 2"
146 : : [set] "r" (val));
151 dsb();
152 isb();
155 static inline void dcache_clean(){
156 dcache_maint(1);
158 static inline void dcache_invalidate (){
159 dcache_maint(2);
162 static inline void refresh_tlb(void)
164 dsb();
166 /* Invalidate entire unified TLB */
167 asm volatile("mcr p15, 0, %[zero], c8, c7, 0 @ TLBIALL\n\t" : : [zero] "r" (0));
169 #if 0
170 /* Invalidate entire data TLB */
171 asm volatile("mcr p15, 0, %[zero], c8, c6, 0" : : [zero] "r" (0));
173 /* Invalidate entire instruction TLB */
174 asm volatile("mcr p15, 0, %[zero], c8, c5, 0" : : [zero] "r" (0));
175 #endif
178 * Invalidate all instruction caches to PoU.
179 * Also flushes branch target cache.
181 asm volatile("mcr p15, 0, %[zero], c7, c5, 0" : : [zero] "r" (0));
183 /* Invalidate entire branch predictor array */
184 asm volatile("mcr p15, 0, %[zero], c7, c5, 6" : : [zero] "r" (0)); /* flush BTB */
186 dsb();
187 isb();
191 /* Read System Control Register */
192 static inline u32_t read_sctlr()
194 u32_t ctl;
196 asm volatile("mrc p15, 0, %[ctl], c1, c0, 0 @ Read SCTLR\n\t"
197 : [ctl] "=r" (ctl));
199 return ctl;
202 /* Write System Control Register */
203 static inline void write_sctlr(u32_t ctl)
205 asm volatile("mcr p15, 0, %[ctl], c1, c0, 0 @ Write SCTLR\n\t"
206 : : [ctl] "r" (ctl));
207 isb();
210 /* Read Translation Table Base Register 0 */
211 static inline u32_t read_ttbr0()
213 u32_t bar;
215 asm volatile("mrc p15, 0, %[bar], c2, c0, 0 @ Read TTBR0\n\t"
216 : [bar] "=r" (bar));
218 return bar & ARM_TTBR_ADDR_MASK;
221 /* Write Translation Table Base Register 0 */
222 static inline void write_ttbr0(u32_t bar)
224 barrier();
225 /* In our setup TTBR contains the base address *and* the flags
226 but other pieces of the kernel code expect ttbr to be the
227 base address of the l1 page table. We therefore add the
228 flags here and remove them in the read_ttbr0 */
229 u32_t v = (bar & ARM_TTBR_ADDR_MASK ) | ARM_TTBR_FLAGS_CACHED;
230 asm volatile("mcr p15, 0, %[bar], c2, c0, 0 @ Write TTBR0\n\t"
231 : : [bar] "r" (v));
233 refresh_tlb();
236 /* Reload Translation Table Base Register 0 */
237 static inline void reload_ttbr0(void)
239 reg_t ttbr = read_ttbr0();
240 write_ttbr0(ttbr);
243 /* Read Translation Table Base Register 1 */
244 static inline u32_t read_ttbr1()
246 u32_t bar;
248 asm volatile("mrc p15, 0, %[bar], c2, c0, 1 @ Read TTBR1\n\t"
249 : [bar] "=r" (bar));
251 return bar;
254 /* Write Translation Table Base Register 1 */
255 static inline void write_ttbr1(u32_t bar)
257 barrier();
259 asm volatile("mcr p15, 0, %[bar], c2, c0, 1 @ Write TTBR1\n\t"
260 : : [bar] "r" (bar));
262 refresh_tlb();
265 /* Reload Translation Table Base Register 1 */
266 static inline void reload_ttbr1(void)
268 reg_t ttbr = read_ttbr1();
270 write_ttbr1(ttbr);
273 /* Read Translation Table Base Control Register */
274 static inline u32_t read_ttbcr()
276 u32_t bcr;
278 asm volatile("mrc p15, 0, %[bcr], c2, c0, 2 @ Read TTBCR\n\t"
279 : [bcr] "=r" (bcr));
281 return bcr;
284 /* Write Translation Table Base Control Register */
285 static inline void write_ttbcr(u32_t bcr)
287 asm volatile("mcr p15, 0, %[bcr], c2, c0, 2 @ Write TTBCR\n\t"
288 : : [bcr] "r" (bcr));
290 isb();
293 /* Read Domain Access Control Register */
294 static inline u32_t read_dacr()
296 u32_t dacr;
298 asm volatile("mrc p15, 0, %[dacr], c3, c0, 0 @ Read DACR\n\t"
299 : [dacr] "=r" (dacr));
301 return dacr;
304 /* Write Domain Access Control Register */
305 static inline void write_dacr(u32_t dacr)
307 asm volatile("mcr p15, 0, %[dacr], c3, c0, 0 @ Write DACR\n\t"
308 : : [dacr] "r" (dacr));
310 isb();
313 /* Read Data Fault Status Register */
314 static inline u32_t read_dfsr()
316 u32_t fsr;
318 asm volatile("mrc p15, 0, %[fsr], c5, c0, 0 @ Read DFSR\n\t"
319 : [fsr] "=r" (fsr));
321 return fsr;
324 /* Write Data Fault Status Register */
325 static inline void write_dfsr(u32_t fsr)
327 asm volatile("mcr p15, 0, %[fsr], c5, c0, 0 @ Write DFSR\n\t"
328 : : [fsr] "r" (fsr));
330 isb();
333 /* Read Instruction Fault Status Register */
334 static inline u32_t read_ifsr()
336 u32_t fsr;
338 asm volatile("mrc p15, 0, %[fsr], c5, c0, 1 @ Read IFSR\n\t"
339 : [fsr] "=r" (fsr));
341 return fsr;
344 /* Write Instruction Fault Status Register */
345 static inline void write_ifsr(u32_t fsr)
347 asm volatile("mcr p15, 0, %[fsr], c5, c0, 1 @ Write IFSR\n\t"
348 : : [fsr] "r" (fsr));
350 isb();
353 /* Read Data Fault Address Register */
354 static inline u32_t read_dfar()
356 u32_t far;
358 asm volatile("mrc p15, 0, %[far], c6, c0, 0 @ Read DFAR\n\t"
359 : [far] "=r" (far));
361 return far;
364 /* Write Data Fault Address Register */
365 static inline void write_dfar(u32_t far)
367 asm volatile("mcr p15, 0, %[far], c6, c0, 0 @ Write DFAR\n\t"
368 : : [far] "r" (far));
370 isb();
373 /* Read Instruction Fault Address Register */
374 static inline u32_t read_ifar()
376 u32_t far;
378 asm volatile("mrc p15, 0, %[far], c6, c0, 2 @ Read IFAR\n\t"
379 : [far] "=r" (far));
381 return far;
384 /* Write Instruction Fault Address Register */
385 static inline void write_ifar(u32_t far)
387 asm volatile("mcr p15, 0, %[far], c6, c0, 2 @ Write IFAR\n\t"
388 : : [far] "r" (far));
390 isb();
393 /* Read Vector Base Address Register */
394 static inline u32_t read_vbar()
396 u32_t vbar;
398 asm volatile("mrc p15, 0, %[vbar], c12, c0, 0 @ Read VBAR\n\t"
399 : [vbar] "=r" (vbar));
401 return vbar;
404 /* Write Vector Base Address Register */
405 static inline void write_vbar(u32_t vbar)
407 asm volatile("mcr p15, 0, %[vbar], c12, c0, 0 @ Write VBAR\n\t"
408 : : [vbar] "r" (vbar));
410 isb();
413 /* Read the Main ID Register */
414 static inline u32_t read_midr()
416 u32_t id;
418 asm volatile("mrc p15, 0, %[id], c0, c0, 0 @ read MIDR\n\t"
419 : [id] "=r" (id));
421 return id;
424 /* Read Auxiliary Control Register */
425 static inline u32_t read_actlr()
427 u32_t ctl;
429 asm volatile("mrc p15, 0, %[ctl], c1, c0, 1 @ Read ACTLR\n\t"
430 : [ctl] "=r" (ctl));
432 return ctl;
435 /* Write Auxiliary Control Register */
436 static inline void write_actlr(u32_t ctl)
438 //http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ddi0344k/Babjbjbb.html
439 asm volatile("mcr p15, 0, %[ctl], c1, c0, 1 @ Write ACTLR\n\t"
440 : : [ctl] "r" (ctl));
442 isb();
445 /* Read Current Program Status Register */
446 static inline u32_t read_cpsr()
448 u32_t status;
450 asm volatile("mrs %[status], cpsr @ read CPSR"
451 : [status] "=r" (status));
453 return status;
456 /* Write Current Program Status Register */
457 static inline void write_cpsr(u32_t status)
459 asm volatile("msr cpsr_c, %[status] @ write CPSR"
460 : : [status] "r" (status));
463 #endif /* _ARM_CPUFUNC_H */