diff options
Diffstat (limited to 'arch/avr32')
-rw-r--r-- | arch/avr32/kernel/entry-avr32b.S | 54 |
1 files changed, 24 insertions, 30 deletions
diff --git a/arch/avr32/kernel/entry-avr32b.S b/arch/avr32/kernel/entry-avr32b.S index 484e08310f15..42657f1703b2 100644 --- a/arch/avr32/kernel/entry-avr32b.S +++ b/arch/avr32/kernel/entry-avr32b.S | |||
@@ -100,55 +100,49 @@ dtlb_miss_write: | |||
100 | 100 | ||
101 | .global tlb_miss_common | 101 | .global tlb_miss_common |
102 | tlb_miss_common: | 102 | tlb_miss_common: |
103 | mfsr r0, SYSREG_PTBR | 103 | mfsr r0, SYSREG_TLBEAR |
104 | mfsr r1, SYSREG_TLBEAR | 104 | mfsr r1, SYSREG_PTBR |
105 | 105 | ||
106 | /* Is it the vmalloc space? */ | 106 | /* Is it the vmalloc space? */ |
107 | bld r1, 31 | 107 | bld r0, 31 |
108 | brcs handle_vmalloc_miss | 108 | brcs handle_vmalloc_miss |
109 | 109 | ||
110 | /* First level lookup */ | 110 | /* First level lookup */ |
111 | pgtbl_lookup: | 111 | pgtbl_lookup: |
112 | lsr r2, r1, PGDIR_SHIFT | 112 | lsr r2, r0, PGDIR_SHIFT |
113 | ld.w r0, r0[r2 << 2] | 113 | ld.w r3, r1[r2 << 2] |
114 | bld r0, _PAGE_BIT_PRESENT | 114 | bfextu r1, r0, PAGE_SHIFT, PGDIR_SHIFT - PAGE_SHIFT |
115 | bld r3, _PAGE_BIT_PRESENT | ||
115 | brcc page_table_not_present | 116 | brcc page_table_not_present |
116 | 117 | ||
117 | /* TODO: Check access rights on page table if necessary */ | ||
118 | |||
119 | /* Translate to virtual address in P1. */ | 118 | /* Translate to virtual address in P1. */ |
120 | andl r0, 0xf000 | 119 | andl r3, 0xf000 |
121 | sbr r0, 31 | 120 | sbr r3, 31 |
122 | 121 | ||
123 | /* Second level lookup */ | 122 | /* Second level lookup */ |
124 | lsl r1, (32 - PGDIR_SHIFT) | 123 | ld.w r2, r3[r1 << 2] |
125 | lsr r1, (32 - PGDIR_SHIFT) + PAGE_SHIFT | 124 | mfsr r0, SYSREG_TLBARLO |
126 | add r2, r0, r1 << 2 | 125 | bld r2, _PAGE_BIT_PRESENT |
127 | ld.w r1, r2[0] | ||
128 | bld r1, _PAGE_BIT_PRESENT | ||
129 | brcc page_not_present | 126 | brcc page_not_present |
130 | 127 | ||
131 | /* Mark the page as accessed */ | 128 | /* Mark the page as accessed */ |
132 | sbr r1, _PAGE_BIT_ACCESSED | 129 | sbr r2, _PAGE_BIT_ACCESSED |
133 | st.w r2[0], r1 | 130 | st.w r3[r1 << 2], r2 |
134 | 131 | ||
135 | /* Drop software flags */ | 132 | /* Drop software flags */ |
136 | andl r1, _PAGE_FLAGS_HARDWARE_MASK & 0xffff | 133 | andl r2, _PAGE_FLAGS_HARDWARE_MASK & 0xffff |
137 | mtsr SYSREG_TLBELO, r1 | 134 | mtsr SYSREG_TLBELO, r2 |
138 | 135 | ||
139 | /* Figure out which entry we want to replace */ | 136 | /* Figure out which entry we want to replace */ |
140 | mfsr r0, SYSREG_TLBARLO | 137 | mfsr r1, SYSREG_MMUCR |
141 | clz r2, r0 | 138 | clz r2, r0 |
142 | brcc 1f | 139 | brcc 1f |
143 | mov r1, -1 /* All entries have been accessed, */ | 140 | mov r3, -1 /* All entries have been accessed, */ |
144 | mtsr SYSREG_TLBARLO, r1 /* so reset TLBAR */ | 141 | mov r2, 0 /* so start at 0 */ |
145 | mov r2, 0 /* and start at 0 */ | 142 | mtsr SYSREG_TLBARLO, r3 /* and reset TLBAR */ |
146 | 1: mfsr r1, SYSREG_MMUCR | ||
147 | lsl r2, 14 | ||
148 | andl r1, 0x3fff, COH | ||
149 | or r1, r2 | ||
150 | mtsr SYSREG_MMUCR, r1 | ||
151 | 143 | ||
144 | 1: bfins r1, r2, SYSREG_DRP_OFFSET, SYSREG_DRP_SIZE | ||
145 | mtsr SYSREG_MMUCR, r1 | ||
152 | tlbw | 146 | tlbw |
153 | 147 | ||
154 | tlbmiss_restore | 148 | tlbmiss_restore |
@@ -156,8 +150,8 @@ pgtbl_lookup: | |||
156 | 150 | ||
157 | handle_vmalloc_miss: | 151 | handle_vmalloc_miss: |
158 | /* Simply do the lookup in init's page table */ | 152 | /* Simply do the lookup in init's page table */ |
159 | mov r0, lo(swapper_pg_dir) | 153 | mov r1, lo(swapper_pg_dir) |
160 | orh r0, hi(swapper_pg_dir) | 154 | orh r1, hi(swapper_pg_dir) |
161 | rjmp pgtbl_lookup | 155 | rjmp pgtbl_lookup |
162 | 156 | ||
163 | 157 | ||