diff options
author | David S. Miller <davem@sunset.davemloft.net> | 2006-02-09 20:21:53 -0500 |
---|---|---|
committer | David S. Miller <davem@sunset.davemloft.net> | 2006-03-20 04:12:06 -0500 |
commit | 618e9ed98aed924a1fc664eb6522db4a5e927043 (patch) | |
tree | 08ace6185b8f9709cb22a23d329def1dae622666 /arch | |
parent | aa9143b9719c07fb6f1f6207790c9c5086ae07e7 (diff) |
[SPARC64]: Hypervisor TSB context switching.
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'arch')
-rw-r--r-- | arch/sparc64/kernel/tsb.S | 42 | ||||
-rw-r--r-- | arch/sparc64/mm/tsb.c | 48 |
2 files changed, 74 insertions, 16 deletions
diff --git a/arch/sparc64/kernel/tsb.S b/arch/sparc64/kernel/tsb.S index c848c8847cdc..a53ec6fb7697 100644 --- a/arch/sparc64/kernel/tsb.S +++ b/arch/sparc64/kernel/tsb.S | |||
@@ -4,6 +4,7 @@ | |||
4 | */ | 4 | */ |
5 | 5 | ||
6 | #include <asm/tsb.h> | 6 | #include <asm/tsb.h> |
7 | #include <asm/hypervisor.h> | ||
7 | 8 | ||
8 | .text | 9 | .text |
9 | .align 32 | 10 | .align 32 |
@@ -233,6 +234,7 @@ tsb_flush: | |||
233 | * %o1: TSB register value | 234 | * %o1: TSB register value |
234 | * %o2: TSB virtual address | 235 | * %o2: TSB virtual address |
235 | * %o3: TSB mapping locked PTE | 236 | * %o3: TSB mapping locked PTE |
237 | * %o4: Hypervisor TSB descriptor physical address | ||
236 | * | 238 | * |
237 | * We have to run this whole thing with interrupts | 239 | * We have to run this whole thing with interrupts |
238 | * disabled so that the current cpu doesn't change | 240 | * disabled so that the current cpu doesn't change |
@@ -251,30 +253,40 @@ __tsb_context_switch: | |||
251 | add %g2, %g1, %g2 | 253 | add %g2, %g1, %g2 |
252 | stx %o0, [%g2 + TRAP_PER_CPU_PGD_PADDR] | 254 | stx %o0, [%g2 + TRAP_PER_CPU_PGD_PADDR] |
253 | 255 | ||
254 | 661: mov TSB_REG, %g1 | 256 | sethi %hi(tlb_type), %g1 |
255 | stxa %o1, [%g1] ASI_DMMU | 257 | lduw [%g1 + %lo(tlb_type)], %g1 |
256 | .section .sun4v_2insn_patch, "ax" | 258 | cmp %g1, 3 |
257 | .word 661b | 259 | bne,pt %icc, 1f |
260 | nop | ||
261 | |||
262 | /* Hypervisor TSB switch. */ | ||
258 | mov SCRATCHPAD_UTSBREG1, %g1 | 263 | mov SCRATCHPAD_UTSBREG1, %g1 |
259 | stxa %o1, [%g1] ASI_SCRATCHPAD | 264 | stxa %o1, [%g1] ASI_SCRATCHPAD |
260 | .previous | 265 | mov -1, %g2 |
266 | mov SCRATCHPAD_UTSBREG2, %g1 | ||
267 | stxa %g2, [%g1] ASI_SCRATCHPAD | ||
261 | 268 | ||
262 | membar #Sync | 269 | mov HV_FAST_MMU_TSB_CTXNON0, %o0 |
270 | mov 1, %o1 | ||
271 | mov %o4, %o2 | ||
272 | ta HV_FAST_TRAP | ||
273 | |||
274 | ba,pt %xcc, 9f | ||
275 | nop | ||
263 | 276 | ||
264 | 661: stxa %o1, [%g1] ASI_IMMU | 277 | /* SUN4U TSB switch. */ |
278 | 1: mov TSB_REG, %g1 | ||
279 | stxa %o1, [%g1] ASI_DMMU | ||
280 | membar #Sync | ||
281 | stxa %o1, [%g1] ASI_IMMU | ||
265 | membar #Sync | 282 | membar #Sync |
266 | .section .sun4v_2insn_patch, "ax" | ||
267 | .word 661b | ||
268 | nop | ||
269 | nop | ||
270 | .previous | ||
271 | 283 | ||
272 | brz %o2, 9f | 284 | 2: brz %o2, 9f |
273 | nop | 285 | nop |
274 | 286 | ||
275 | sethi %hi(sparc64_highest_unlocked_tlb_ent), %o4 | 287 | sethi %hi(sparc64_highest_unlocked_tlb_ent), %g2 |
276 | mov TLB_TAG_ACCESS, %g1 | 288 | mov TLB_TAG_ACCESS, %g1 |
277 | lduw [%o4 + %lo(sparc64_highest_unlocked_tlb_ent)], %g2 | 289 | lduw [%g2 + %lo(sparc64_highest_unlocked_tlb_ent)], %g2 |
278 | stxa %o2, [%g1] ASI_DMMU | 290 | stxa %o2, [%g1] ASI_DMMU |
279 | membar #Sync | 291 | membar #Sync |
280 | sllx %g2, 3, %g2 | 292 | sllx %g2, 3, %g2 |
diff --git a/arch/sparc64/mm/tsb.c b/arch/sparc64/mm/tsb.c index 2cc8e6528c63..6ae2a5a702cb 100644 --- a/arch/sparc64/mm/tsb.c +++ b/arch/sparc64/mm/tsb.c | |||
@@ -149,7 +149,7 @@ static void setup_tsb_params(struct mm_struct *mm, unsigned long tsb_bytes) | |||
149 | BUG(); | 149 | BUG(); |
150 | }; | 150 | }; |
151 | 151 | ||
152 | if (tlb_type == cheetah_plus) { | 152 | if (tlb_type == cheetah_plus || tlb_type == hypervisor) { |
153 | /* Physical mapping, no locked TLB entry for TSB. */ | 153 | /* Physical mapping, no locked TLB entry for TSB. */ |
154 | tsb_reg |= tsb_paddr; | 154 | tsb_reg |= tsb_paddr; |
155 | 155 | ||
@@ -166,6 +166,52 @@ static void setup_tsb_params(struct mm_struct *mm, unsigned long tsb_bytes) | |||
166 | mm->context.tsb_map_pte = tte; | 166 | mm->context.tsb_map_pte = tte; |
167 | } | 167 | } |
168 | 168 | ||
169 | /* Setup the Hypervisor TSB descriptor. */ | ||
170 | if (tlb_type == hypervisor) { | ||
171 | struct hv_tsb_descr *hp = &mm->context.tsb_descr; | ||
172 | |||
173 | switch (PAGE_SIZE) { | ||
174 | case 8192: | ||
175 | default: | ||
176 | hp->pgsz_idx = HV_PGSZ_IDX_8K; | ||
177 | break; | ||
178 | |||
179 | case 64 * 1024: | ||
180 | hp->pgsz_idx = HV_PGSZ_IDX_64K; | ||
181 | break; | ||
182 | |||
183 | case 512 * 1024: | ||
184 | hp->pgsz_idx = HV_PGSZ_IDX_512K; | ||
185 | break; | ||
186 | |||
187 | case 4 * 1024 * 1024: | ||
188 | hp->pgsz_idx = HV_PGSZ_IDX_4MB; | ||
189 | break; | ||
190 | }; | ||
191 | hp->assoc = 1; | ||
192 | hp->num_ttes = tsb_bytes / 16; | ||
193 | hp->ctx_idx = 0; | ||
194 | switch (PAGE_SIZE) { | ||
195 | case 8192: | ||
196 | default: | ||
197 | hp->pgsz_mask = HV_PGSZ_MASK_8K; | ||
198 | break; | ||
199 | |||
200 | case 64 * 1024: | ||
201 | hp->pgsz_mask = HV_PGSZ_MASK_64K; | ||
202 | break; | ||
203 | |||
204 | case 512 * 1024: | ||
205 | hp->pgsz_mask = HV_PGSZ_MASK_512K; | ||
206 | break; | ||
207 | |||
208 | case 4 * 1024 * 1024: | ||
209 | hp->pgsz_mask = HV_PGSZ_MASK_4MB; | ||
210 | break; | ||
211 | }; | ||
212 | hp->tsb_base = tsb_paddr; | ||
213 | hp->resv = 0; | ||
214 | } | ||
169 | } | 215 | } |
170 | 216 | ||
171 | /* The page tables are locked against modifications while this | 217 | /* The page tables are locked against modifications while this |