4 * Copyright (C) 2001 Deep Blue Solutions Ltd.
5 * Copyright (C) 2012 ARM Ltd.
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 as
9 * published by the Free Software Foundation.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 #include <linux/linkage.h>
21 #include <linux/init.h>
22 #include <asm/assembler.h>
24 #include "proc-macros.S"
27 * __flush_dcache_all()
29 * Flush the whole D-cache.
31 * Corrupted registers: x0-x7, x9-x11
33 ENTRY(__flush_dcache_all)
34 dsb sy // ensure ordering with previous memory accesses
35 mrs x0, clidr_el1 // read clidr
36 and x3, x0, #0x7000000 // extract loc from clidr
37 lsr x3, x3, #23 // left align loc bit field
38 cbz x3, finished // if loc is 0, then no need to clean
39 mov x10, #0 // start clean at cache level 0
41 add x2, x10, x10, lsr #1 // work out 3x current cache level
42 lsr x1, x0, x2 // extract cache type bits from clidr
43 and x1, x1, #7 // mask of the bits for current cache only
44 cmp x1, #2 // see what cache we have at this level
45 b.lt skip // skip if no cache, or just i-cache
46 save_and_disable_irqs x9 // make CSSELR and CCSIDR access atomic
47 msr csselr_el1, x10 // select current cache level in csselr
48 isb // isb to sych the new cssr&csidr
49 mrs x1, ccsidr_el1 // read the new ccsidr
51 and x2, x1, #7 // extract the length of the cache lines
52 add x2, x2, #4 // add 4 (line length offset)
54 and x4, x4, x1, lsr #3 // find maximum number on the way size
55 clz w5, w4 // find bit position of way size increment
57 and x7, x7, x1, lsr #13 // extract max number of the index size
59 mov x9, x4 // create working copy of max way size
62 orr x11, x10, x6 // factor way and cache number into x11
64 orr x11, x11, x6 // factor index number into x11
65 dc cisw, x11 // clean & invalidate by set/way
66 subs x9, x9, #1 // decrement the way
68 subs x7, x7, #1 // decrement the index
71 add x10, x10, #2 // increment cache number
75 mov x10, #0 // swith back to cache level 0
76 msr csselr_el1, x10 // select current cache level in csselr
80 ENDPROC(__flush_dcache_all)
85 * Flush the entire cache system. The data cache flush is now achieved
86 * using atomic clean / invalidates working outwards from L1 cache. This
87 * is done using Set/Way based cache maintainance instructions. The
88 * instruction cache can still be invalidated back to the point of
89 * unification in a single instruction.
91 ENTRY(flush_cache_all)
95 ic ialluis // I+BTB cache invalidate
97 ENDPROC(flush_cache_all)
100 * flush_icache_range(start,end)
102 * Ensure that the I and D caches are coherent within specified region.
103 * This is typically used when code has been written to a memory region,
104 * and will be executed.
106 * - start - virtual start address of region
107 * - end - virtual end address of region
109 ENTRY(flush_icache_range)
113 * __flush_cache_user_range(start,end)
115 * Ensure that the I and D caches are coherent within specified region.
116 * This is typically used when code has been written to a memory region,
117 * and will be executed.
119 * - start - virtual start address of region
120 * - end - virtual end address of region
122 ENTRY(__flush_cache_user_range)
123 dcache_line_size x2, x3
127 USER(9f, dc cvau, x4 ) // clean D line to PoU
133 icache_line_size x2, x3
137 USER(9f, ic ivau, x4 ) // invalidate I line PoU
141 9: // ignore any faulting cache operation
145 ENDPROC(flush_icache_range)
146 ENDPROC(__flush_cache_user_range)
149 * __flush_kern_dcache_page(kaddr)
151 * Ensure that the data held in the page kaddr is written back to the
154 * - kaddr - kernel address
155 * - size - size in question
157 ENTRY(__flush_dcache_area)
158 dcache_line_size x2, x3
162 1: dc civac, x0 // clean & invalidate D line / unified line
168 ENDPROC(__flush_dcache_area)