tlbex.c 33.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7
/*
 * This file is subject to the terms and conditions of the GNU General Public
 * License.  See the file "COPYING" in the main directory of this archive
 * for more details.
 *
 * Synthesize TLB refill handlers at runtime.
 *
8
 * Copyright (C) 2004, 2005, 2006, 2008  Thiemo Seufer
9
 * Copyright (C) 2005, 2007  Maciej W. Rozycki
10 11 12 13 14 15 16 17 18 19
 * Copyright (C) 2006  Ralf Baechle (ralf@linux-mips.org)
 *
 * ... and the days got worse and worse and now you see
 * I've gone completly out of my mind.
 *
 * They're coming to take me a away haha
 * they're coming to take me a away hoho hihi haha
 * to the funny farm where code is beautiful all the time ...
 *
 * (Condolences to Napoleon XIV)
L
Linus Torvalds 已提交
20 21 22 23 24 25 26 27 28 29
 */

#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/string.h>
#include <linux/init.h>

#include <asm/mmu_context.h>
#include <asm/war.h>

30 31
#include "uasm.h"

32
static inline int r45k_bvahwbug(void)
L
Linus Torvalds 已提交
33 34 35 36 37
{
	/* XXX: We should probe for the presence of this bug, but we don't. */
	return 0;
}

38
static inline int r4k_250MHZhwbug(void)
L
Linus Torvalds 已提交
39 40 41 42 43
{
	/* XXX: We should probe for the presence of this bug, but we don't. */
	return 0;
}

44
static inline int __maybe_unused bcm1250_m3_war(void)
L
Linus Torvalds 已提交
45 46 47 48
{
	return BCM1250_M3_WAR;
}

49
static inline int __maybe_unused r10000_llsc_war(void)
L
Linus Torvalds 已提交
50 51 52 53
{
	return R10000_LLSC_WAR;
}

54 55 56 57 58 59 60 61 62
/*
 * Found by experiment: At least some revisions of the 4kc throw under
 * some circumstances a machine check exception, triggered by invalid
 * values in the index register.  Delaying the tlbp instruction until
 * after the next branch,  plus adding an additional nop in front of
 * tlbwi/tlbwr avoids the invalid index register values. Nobody knows
 * why; it's not an issue caused by the core RTL.
 *
 */
63
static int __cpuinit m4kc_tlbp_war(void)
64 65 66 67 68
{
	return (current_cpu_data.processor_id & 0xffff00) ==
	       (PRID_COMP_MIPS | PRID_IMP_4KC);
}

69
/* Handle labels (which must be positive integers). */
L
Linus Torvalds 已提交
70
enum label_id {
71
	label_second_part = 1,
L
Linus Torvalds 已提交
72
	label_leave,
73 74 75
#ifdef MODULE_START
	label_module_alloc,
#endif
L
Linus Torvalds 已提交
76 77 78 79 80 81 82 83 84 85 86
	label_vmalloc,
	label_vmalloc_done,
	label_tlbw_hazard,
	label_split,
	label_nopage_tlbl,
	label_nopage_tlbs,
	label_nopage_tlbm,
	label_smp_pgtable_change,
	label_r3000_write_probe_fail,
};

87 88
UASM_L_LA(_second_part)
UASM_L_LA(_leave)
89
#ifdef MODULE_START
90
UASM_L_LA(_module_alloc)
91
#endif
92 93 94 95 96 97 98 99 100
UASM_L_LA(_vmalloc)
UASM_L_LA(_vmalloc_done)
UASM_L_LA(_tlbw_hazard)
UASM_L_LA(_split)
UASM_L_LA(_nopage_tlbl)
UASM_L_LA(_nopage_tlbs)
UASM_L_LA(_nopage_tlbm)
UASM_L_LA(_smp_pgtable_change)
UASM_L_LA(_r3000_write_probe_fail)
101

102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117
/*
 * For debug purposes.
 */
static inline void dump_handler(const u32 *handler, int count)
{
	int i;

	pr_debug("\t.set push\n");
	pr_debug("\t.set noreorder\n");

	for (i = 0; i < count; i++)
		pr_debug("\t%p\t.word 0x%08x\n", &handler[i], handler[i]);

	pr_debug("\t.set pop\n");
}

L
Linus Torvalds 已提交
118 119 120 121 122
/* The only general purpose registers allowed in TLB handlers. */
#define K0		26
#define K1		27

/* Some CP0 registers */
123 124 125 126 127 128 129 130 131
#define C0_INDEX	0, 0
#define C0_ENTRYLO0	2, 0
#define C0_TCBIND	2, 2
#define C0_ENTRYLO1	3, 0
#define C0_CONTEXT	4, 0
#define C0_BADVADDR	8, 0
#define C0_ENTRYHI	10, 0
#define C0_EPC		14, 0
#define C0_XCONTEXT	20, 0
L
Linus Torvalds 已提交
132

133
#ifdef CONFIG_64BIT
134
# define GET_CONTEXT(buf, reg) UASM_i_MFC0(buf, reg, C0_XCONTEXT)
L
Linus Torvalds 已提交
135
#else
136
# define GET_CONTEXT(buf, reg) UASM_i_MFC0(buf, reg, C0_CONTEXT)
L
Linus Torvalds 已提交
137 138 139 140 141 142 143 144 145 146
#endif

/* The worst case length of the handler is around 18 instructions for
 * R3000-style TLBs and up to 63 instructions for R4000-style TLBs.
 * Maximum space available is 32 instructions for R3000 and 64
 * instructions for R4000.
 *
 * We deliberately chose a buffer size of 128, so we won't scribble
 * over anything important on overflow before we panic.
 */
147
static u32 tlb_handler[128] __cpuinitdata;
L
Linus Torvalds 已提交
148 149

/* simply assume worst case size for labels and relocs */
150 151
static struct uasm_label labels[128] __cpuinitdata;
static struct uasm_reloc relocs[128] __cpuinitdata;
L
Linus Torvalds 已提交
152 153 154 155

/*
 * The R3000 TLB handler is simple.
 */
156
static void __cpuinit build_r3000_tlb_refill_handler(void)
L
Linus Torvalds 已提交
157 158 159 160 161 162 163
{
	long pgdc = (long)pgd_current;
	u32 *p;

	memset(tlb_handler, 0, sizeof(tlb_handler));
	p = tlb_handler;

164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
	uasm_i_mfc0(&p, K0, C0_BADVADDR);
	uasm_i_lui(&p, K1, uasm_rel_hi(pgdc)); /* cp0 delay */
	uasm_i_lw(&p, K1, uasm_rel_lo(pgdc), K1);
	uasm_i_srl(&p, K0, K0, 22); /* load delay */
	uasm_i_sll(&p, K0, K0, 2);
	uasm_i_addu(&p, K1, K1, K0);
	uasm_i_mfc0(&p, K0, C0_CONTEXT);
	uasm_i_lw(&p, K1, 0, K1); /* cp0 delay */
	uasm_i_andi(&p, K0, K0, 0xffc); /* load delay */
	uasm_i_addu(&p, K1, K1, K0);
	uasm_i_lw(&p, K0, 0, K1);
	uasm_i_nop(&p); /* load delay */
	uasm_i_mtc0(&p, K0, C0_ENTRYLO0);
	uasm_i_mfc0(&p, K1, C0_EPC); /* cp0 delay */
	uasm_i_tlbwr(&p); /* cp0 delay */
	uasm_i_jr(&p, K1);
	uasm_i_rfe(&p); /* branch delay */
L
Linus Torvalds 已提交
181 182 183 184

	if (p > tlb_handler + 32)
		panic("TLB refill handler space exceeded");

185 186
	pr_debug("Wrote TLB refill handler (%u instructions).\n",
		 (unsigned int)(p - tlb_handler));
L
Linus Torvalds 已提交
187

188
	memcpy((void *)ebase, tlb_handler, 0x80);
189 190

	dump_handler((u32 *)ebase, 32);
L
Linus Torvalds 已提交
191 192 193 194 195 196 197 198 199
}

/*
 * The R4000 TLB handler is much more complicated. We have two
 * consecutive handler areas with 32 instructions space each.
 * Since they aren't used at the same time, we can overflow in the
 * other one.To keep things simple, we first assume linear space,
 * then we relocate it to the final handler layout as needed.
 */
200
static u32 final_handler[64] __cpuinitdata;
L
Linus Torvalds 已提交
201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223

/*
 * Hazards
 *
 * From the IDT errata for the QED RM5230 (Nevada), processor revision 1.0:
 * 2. A timing hazard exists for the TLBP instruction.
 *
 *      stalling_instruction
 *      TLBP
 *
 * The JTLB is being read for the TLBP throughout the stall generated by the
 * previous instruction. This is not really correct as the stalling instruction
 * can modify the address used to access the JTLB.  The failure symptom is that
 * the TLBP instruction will use an address created for the stalling instruction
 * and not the address held in C0_ENHI and thus report the wrong results.
 *
 * The software work-around is to not allow the instruction preceding the TLBP
 * to stall - make it an NOP or some other instruction guaranteed not to stall.
 *
 * Errata 2 will not be fixed.  This errata is also on the R5000.
 *
 * As if we MIPS hackers wouldn't know how to nop pipelines happy ...
 */
224
static void __cpuinit __maybe_unused build_tlb_probe_entry(u32 **p)
L
Linus Torvalds 已提交
225
{
226
	switch (current_cpu_type()) {
227
	/* Found by experiment: R4600 v2.0/R4700 needs this, too.  */
228
	case CPU_R4600:
229
	case CPU_R4700:
L
Linus Torvalds 已提交
230 231 232
	case CPU_R5000:
	case CPU_R5000A:
	case CPU_NEVADA:
233 234
		uasm_i_nop(p);
		uasm_i_tlbp(p);
L
Linus Torvalds 已提交
235 236 237
		break;

	default:
238
		uasm_i_tlbp(p);
L
Linus Torvalds 已提交
239 240 241 242 243 244 245 246 247 248
		break;
	}
}

/*
 * Write random or indexed TLB entry, and care about the hazards from
 * the preceeding mtc0 and for the following eret.
 */
enum tlb_write_entry { tlb_random, tlb_indexed };

249
static void __cpuinit build_tlb_write_entry(u32 **p, struct uasm_label **l,
250
					 struct uasm_reloc **r,
L
Linus Torvalds 已提交
251 252 253 254 255
					 enum tlb_write_entry wmode)
{
	void(*tlbw)(u32 **) = NULL;

	switch (wmode) {
256 257
	case tlb_random: tlbw = uasm_i_tlbwr; break;
	case tlb_indexed: tlbw = uasm_i_tlbwi; break;
L
Linus Torvalds 已提交
258 259
	}

260
	if (cpu_has_mips_r2) {
261
		uasm_i_ehb(p);
262 263 264 265
		tlbw(p);
		return;
	}

266
	switch (current_cpu_type()) {
L
Linus Torvalds 已提交
267 268 269 270 271 272 273 274 275 276
	case CPU_R4000PC:
	case CPU_R4000SC:
	case CPU_R4000MC:
	case CPU_R4400PC:
	case CPU_R4400SC:
	case CPU_R4400MC:
		/*
		 * This branch uses up a mtc0 hazard nop slot and saves
		 * two nops after the tlbw instruction.
		 */
277
		uasm_il_bgezl(p, r, 0, label_tlbw_hazard);
L
Linus Torvalds 已提交
278
		tlbw(p);
279 280
		uasm_l_tlbw_hazard(l, *p);
		uasm_i_nop(p);
L
Linus Torvalds 已提交
281 282 283 284 285 286
		break;

	case CPU_R4600:
	case CPU_R4700:
	case CPU_R5000:
	case CPU_R5000A:
287
		uasm_i_nop(p);
288
		tlbw(p);
289
		uasm_i_nop(p);
290 291 292
		break;

	case CPU_R4300:
L
Linus Torvalds 已提交
293 294 295 296 297 298
	case CPU_5KC:
	case CPU_TX49XX:
	case CPU_AU1000:
	case CPU_AU1100:
	case CPU_AU1500:
	case CPU_AU1550:
P
Pete Popov 已提交
299
	case CPU_AU1200:
300 301
	case CPU_AU1210:
	case CPU_AU1250:
302
	case CPU_PR4450:
303
		uasm_i_nop(p);
L
Linus Torvalds 已提交
304 305 306 307 308
		tlbw(p);
		break;

	case CPU_R10000:
	case CPU_R12000:
K
Kumba 已提交
309
	case CPU_R14000:
L
Linus Torvalds 已提交
310
	case CPU_4KC:
311
	case CPU_4KEC:
L
Linus Torvalds 已提交
312
	case CPU_SB1:
A
Andrew Isaacson 已提交
313
	case CPU_SB1A:
L
Linus Torvalds 已提交
314 315 316
	case CPU_4KSC:
	case CPU_20KC:
	case CPU_25KF:
317 318
	case CPU_BCM3302:
	case CPU_BCM4710:
319
	case CPU_LOONGSON2:
320
	case CPU_CAVIUM_OCTEON:
321
	case CPU_R5500:
322
		if (m4kc_tlbp_war())
323
			uasm_i_nop(p);
L
Linus Torvalds 已提交
324 325 326 327
		tlbw(p);
		break;

	case CPU_NEVADA:
328
		uasm_i_nop(p); /* QED specifies 2 nops hazard */
L
Linus Torvalds 已提交
329 330 331 332
		/*
		 * This branch uses up a mtc0 hazard nop slot and saves
		 * a nop after the tlbw instruction.
		 */
333
		uasm_il_bgezl(p, r, 0, label_tlbw_hazard);
L
Linus Torvalds 已提交
334
		tlbw(p);
335
		uasm_l_tlbw_hazard(l, *p);
L
Linus Torvalds 已提交
336 337 338
		break;

	case CPU_RM7000:
339 340 341 342
		uasm_i_nop(p);
		uasm_i_nop(p);
		uasm_i_nop(p);
		uasm_i_nop(p);
L
Linus Torvalds 已提交
343 344 345 346 347 348 349 350 351 352
		tlbw(p);
		break;

	case CPU_RM9000:
		/*
		 * When the JTLB is updated by tlbwi or tlbwr, a subsequent
		 * use of the JTLB for instructions should not occur for 4
		 * cpu cycles and use for data translations should not occur
		 * for 3 cpu cycles.
		 */
353 354 355 356
		uasm_i_ssnop(p);
		uasm_i_ssnop(p);
		uasm_i_ssnop(p);
		uasm_i_ssnop(p);
L
Linus Torvalds 已提交
357
		tlbw(p);
358 359 360 361
		uasm_i_ssnop(p);
		uasm_i_ssnop(p);
		uasm_i_ssnop(p);
		uasm_i_ssnop(p);
L
Linus Torvalds 已提交
362 363 364 365 366 367 368
		break;

	case CPU_VR4111:
	case CPU_VR4121:
	case CPU_VR4122:
	case CPU_VR4181:
	case CPU_VR4181A:
369 370
		uasm_i_nop(p);
		uasm_i_nop(p);
L
Linus Torvalds 已提交
371
		tlbw(p);
372 373
		uasm_i_nop(p);
		uasm_i_nop(p);
L
Linus Torvalds 已提交
374 375 376 377
		break;

	case CPU_VR4131:
	case CPU_VR4133:
378
	case CPU_R5432:
379 380
		uasm_i_nop(p);
		uasm_i_nop(p);
L
Linus Torvalds 已提交
381 382 383 384 385 386 387 388 389 390
		tlbw(p);
		break;

	default:
		panic("No TLB refill handler yet (CPU type: %d)",
		      current_cpu_data.cputype);
		break;
	}
}

391
#ifdef CONFIG_64BIT
L
Linus Torvalds 已提交
392 393 394 395
/*
 * TMP and PTR are scratch.
 * TMP will be clobbered, PTR will hold the pmd entry.
 */
396
static void __cpuinit
397
build_get_pmde64(u32 **p, struct uasm_label **l, struct uasm_reloc **r,
L
Linus Torvalds 已提交
398 399 400 401 402 403 404
		 unsigned int tmp, unsigned int ptr)
{
	long pgdc = (long)pgd_current;

	/*
	 * The vmalloc handling is not in the hotpath.
	 */
405
	uasm_i_dmfc0(p, tmp, C0_BADVADDR);
406
#ifdef MODULE_START
407
	uasm_il_bltz(p, r, tmp, label_module_alloc);
408
#else
409
	uasm_il_bltz(p, r, tmp, label_vmalloc);
410
#endif
411
	/* No uasm_i_nop needed here, since the next insn doesn't touch TMP. */
L
Linus Torvalds 已提交
412 413

#ifdef CONFIG_SMP
414 415 416 417
# ifdef  CONFIG_MIPS_MT_SMTC
	/*
	 * SMTC uses TCBind value as "CPU" index
	 */
418 419
	uasm_i_mfc0(p, ptr, C0_TCBIND);
	uasm_i_dsrl(p, ptr, ptr, 19);
420
# else
L
Linus Torvalds 已提交
421
	/*
422
	 * 64 bit SMP running in XKPHYS has smp_processor_id() << 3
L
Linus Torvalds 已提交
423 424
	 * stored in CONTEXT.
	 */
425 426
	uasm_i_dmfc0(p, ptr, C0_CONTEXT);
	uasm_i_dsrl(p, ptr, ptr, 23);
427
#endif
428 429 430 431
	UASM_i_LA_mostly(p, tmp, pgdc);
	uasm_i_daddu(p, ptr, ptr, tmp);
	uasm_i_dmfc0(p, tmp, C0_BADVADDR);
	uasm_i_ld(p, ptr, uasm_rel_lo(pgdc), ptr);
L
Linus Torvalds 已提交
432
#else
433 434
	UASM_i_LA_mostly(p, ptr, pgdc);
	uasm_i_ld(p, ptr, uasm_rel_lo(pgdc), ptr);
L
Linus Torvalds 已提交
435 436
#endif

437
	uasm_l_vmalloc_done(l, *p);
R
Ralf Baechle 已提交
438 439

	if (PGDIR_SHIFT - 3 < 32)		/* get pgd offset in bytes */
440
		uasm_i_dsrl(p, tmp, tmp, PGDIR_SHIFT-3);
R
Ralf Baechle 已提交
441
	else
442 443 444 445 446 447 448 449 450
		uasm_i_dsrl32(p, tmp, tmp, PGDIR_SHIFT - 3 - 32);

	uasm_i_andi(p, tmp, tmp, (PTRS_PER_PGD - 1)<<3);
	uasm_i_daddu(p, ptr, ptr, tmp); /* add in pgd offset */
	uasm_i_dmfc0(p, tmp, C0_BADVADDR); /* get faulting address */
	uasm_i_ld(p, ptr, 0, ptr); /* get pmd pointer */
	uasm_i_dsrl(p, tmp, tmp, PMD_SHIFT-3); /* get pmd offset in bytes */
	uasm_i_andi(p, tmp, tmp, (PTRS_PER_PMD - 1)<<3);
	uasm_i_daddu(p, ptr, ptr, tmp); /* add in pmd offset */
L
Linus Torvalds 已提交
451 452 453 454 455 456
}

/*
 * BVADDR is the faulting address, PTR is scratch.
 * PTR will hold the pgd for vmalloc.
 */
457
static void __cpuinit
458
build_get_pgd_vmalloc64(u32 **p, struct uasm_label **l, struct uasm_reloc **r,
L
Linus Torvalds 已提交
459 460 461 462
			unsigned int bvaddr, unsigned int ptr)
{
	long swpd = (long)swapper_pg_dir;

463 464 465
#ifdef MODULE_START
	long modd = (long)module_pg_dir;

466
	uasm_l_module_alloc(l, *p);
467 468 469 470 471
	/*
	 * Assumption:
	 * VMALLOC_START >= 0xc000000000000000UL
	 * MODULE_START >= 0xe000000000000000UL
	 */
472 473
	UASM_i_SLL(p, ptr, bvaddr, 2);
	uasm_il_bgez(p, r, ptr, label_vmalloc);
474

475 476 477
	if (uasm_in_compat_space_p(MODULE_START) &&
	    !uasm_rel_lo(MODULE_START)) {
		uasm_i_lui(p, ptr, uasm_rel_hi(MODULE_START)); /* delay slot */
478 479
	} else {
		/* unlikely configuration */
480 481
		uasm_i_nop(p); /* delay slot */
		UASM_i_LA(p, ptr, MODULE_START);
482
	}
483
	uasm_i_dsubu(p, bvaddr, bvaddr, ptr);
484

485 486 487
	if (uasm_in_compat_space_p(modd) && !uasm_rel_lo(modd)) {
		uasm_il_b(p, r, label_vmalloc_done);
		uasm_i_lui(p, ptr, uasm_rel_hi(modd));
488
	} else {
489 490 491 492
		UASM_i_LA_mostly(p, ptr, modd);
		uasm_il_b(p, r, label_vmalloc_done);
		if (uasm_in_compat_space_p(modd))
			uasm_i_addiu(p, ptr, ptr, uasm_rel_lo(modd));
493
		else
494
			uasm_i_daddiu(p, ptr, ptr, uasm_rel_lo(modd));
495 496
	}

497 498 499
	uasm_l_vmalloc(l, *p);
	if (uasm_in_compat_space_p(MODULE_START) &&
	    !uasm_rel_lo(MODULE_START) &&
500
	    MODULE_START << 32 == VMALLOC_START)
501
		uasm_i_dsll32(p, ptr, ptr, 0);	/* typical case */
502
	else
503
		UASM_i_LA(p, ptr, VMALLOC_START);
504
#else
505 506
	uasm_l_vmalloc(l, *p);
	UASM_i_LA(p, ptr, VMALLOC_START);
507
#endif
508
	uasm_i_dsubu(p, bvaddr, bvaddr, ptr);
L
Linus Torvalds 已提交
509

510 511 512
	if (uasm_in_compat_space_p(swpd) && !uasm_rel_lo(swpd)) {
		uasm_il_b(p, r, label_vmalloc_done);
		uasm_i_lui(p, ptr, uasm_rel_hi(swpd));
L
Linus Torvalds 已提交
513
	} else {
514 515 516 517
		UASM_i_LA_mostly(p, ptr, swpd);
		uasm_il_b(p, r, label_vmalloc_done);
		if (uasm_in_compat_space_p(swpd))
			uasm_i_addiu(p, ptr, ptr, uasm_rel_lo(swpd));
518
		else
519
			uasm_i_daddiu(p, ptr, ptr, uasm_rel_lo(swpd));
L
Linus Torvalds 已提交
520 521 522
	}
}

523
#else /* !CONFIG_64BIT */
L
Linus Torvalds 已提交
524 525 526 527 528

/*
 * TMP and PTR are scratch.
 * TMP will be clobbered, PTR will hold the pgd entry.
 */
529
static void __cpuinit __maybe_unused
L
Linus Torvalds 已提交
530 531 532 533 534 535
build_get_pgde32(u32 **p, unsigned int tmp, unsigned int ptr)
{
	long pgdc = (long)pgd_current;

	/* 32 bit SMP has smp_processor_id() stored in CONTEXT. */
#ifdef CONFIG_SMP
536 537 538 539
#ifdef  CONFIG_MIPS_MT_SMTC
	/*
	 * SMTC uses TCBind value as "CPU" index
	 */
540 541 542
	uasm_i_mfc0(p, ptr, C0_TCBIND);
	UASM_i_LA_mostly(p, tmp, pgdc);
	uasm_i_srl(p, ptr, ptr, 19);
543 544 545 546
#else
	/*
	 * smp_processor_id() << 3 is stored in CONTEXT.
         */
547 548 549
	uasm_i_mfc0(p, ptr, C0_CONTEXT);
	UASM_i_LA_mostly(p, tmp, pgdc);
	uasm_i_srl(p, ptr, ptr, 23);
550
#endif
551
	uasm_i_addu(p, ptr, tmp, ptr);
L
Linus Torvalds 已提交
552
#else
553
	UASM_i_LA_mostly(p, ptr, pgdc);
L
Linus Torvalds 已提交
554
#endif
555 556 557 558 559
	uasm_i_mfc0(p, tmp, C0_BADVADDR); /* get faulting address */
	uasm_i_lw(p, ptr, uasm_rel_lo(pgdc), ptr);
	uasm_i_srl(p, tmp, tmp, PGDIR_SHIFT); /* get pgd only bits */
	uasm_i_sll(p, tmp, tmp, PGD_T_LOG2);
	uasm_i_addu(p, ptr, ptr, tmp); /* add in pgd offset */
L
Linus Torvalds 已提交
560 561
}

562
#endif /* !CONFIG_64BIT */
L
Linus Torvalds 已提交
563

564
static void __cpuinit build_adjust_context(u32 **p, unsigned int ctx)
L
Linus Torvalds 已提交
565
{
R
Ralf Baechle 已提交
566
	unsigned int shift = 4 - (PTE_T_LOG2 + 1) + PAGE_SHIFT - 12;
L
Linus Torvalds 已提交
567 568
	unsigned int mask = (PTRS_PER_PTE / 2 - 1) << (PTE_T_LOG2 + 1);

569
	switch (current_cpu_type()) {
L
Linus Torvalds 已提交
570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585
	case CPU_VR41XX:
	case CPU_VR4111:
	case CPU_VR4121:
	case CPU_VR4122:
	case CPU_VR4131:
	case CPU_VR4181:
	case CPU_VR4181A:
	case CPU_VR4133:
		shift += 2;
		break;

	default:
		break;
	}

	if (shift)
586 587
		UASM_i_SRL(p, ctx, ctx, shift);
	uasm_i_andi(p, ctx, ctx, mask);
L
Linus Torvalds 已提交
588 589
}

590
static void __cpuinit build_get_ptep(u32 **p, unsigned int tmp, unsigned int ptr)
L
Linus Torvalds 已提交
591 592 593 594 595 596 597 598
{
	/*
	 * Bug workaround for the Nevada. It seems as if under certain
	 * circumstances the move from cp0_context might produce a
	 * bogus result when the mfc0 instruction and its consumer are
	 * in a different cacheline or a load instruction, probably any
	 * memory reference, is between them.
	 */
599
	switch (current_cpu_type()) {
L
Linus Torvalds 已提交
600
	case CPU_NEVADA:
601
		UASM_i_LW(p, ptr, 0, ptr);
L
Linus Torvalds 已提交
602 603 604 605 606
		GET_CONTEXT(p, tmp); /* get context reg */
		break;

	default:
		GET_CONTEXT(p, tmp); /* get context reg */
607
		UASM_i_LW(p, ptr, 0, ptr);
L
Linus Torvalds 已提交
608 609 610 611
		break;
	}

	build_adjust_context(p, tmp);
612
	UASM_i_ADDU(p, ptr, ptr, tmp); /* add in offset */
L
Linus Torvalds 已提交
613 614
}

615
static void __cpuinit build_update_entries(u32 **p, unsigned int tmp,
L
Linus Torvalds 已提交
616 617 618 619 620 621 622 623
					unsigned int ptep)
{
	/*
	 * 64bit address support (36bit on a 32bit CPU) in a 32bit
	 * Kernel is a special case. Only a few CPUs use it.
	 */
#ifdef CONFIG_64BIT_PHYS_ADDR
	if (cpu_has_64bits) {
624 625 626 627 628 629
		uasm_i_ld(p, tmp, 0, ptep); /* get even pte */
		uasm_i_ld(p, ptep, sizeof(pte_t), ptep); /* get odd pte */
		uasm_i_dsrl(p, tmp, tmp, 6); /* convert to entrylo0 */
		uasm_i_mtc0(p, tmp, C0_ENTRYLO0); /* load it */
		uasm_i_dsrl(p, ptep, ptep, 6); /* convert to entrylo1 */
		uasm_i_mtc0(p, ptep, C0_ENTRYLO1); /* load it */
L
Linus Torvalds 已提交
630 631 632 633 634
	} else {
		int pte_off_even = sizeof(pte_t) / 2;
		int pte_off_odd = pte_off_even + sizeof(pte_t);

		/* The pte entries are pre-shifted */
635 636 637 638
		uasm_i_lw(p, tmp, pte_off_even, ptep); /* get even pte */
		uasm_i_mtc0(p, tmp, C0_ENTRYLO0); /* load it */
		uasm_i_lw(p, ptep, pte_off_odd, ptep); /* get odd pte */
		uasm_i_mtc0(p, ptep, C0_ENTRYLO1); /* load it */
L
Linus Torvalds 已提交
639 640
	}
#else
641 642
	UASM_i_LW(p, tmp, 0, ptep); /* get even pte */
	UASM_i_LW(p, ptep, sizeof(pte_t), ptep); /* get odd pte */
L
Linus Torvalds 已提交
643 644
	if (r45k_bvahwbug())
		build_tlb_probe_entry(p);
645
	UASM_i_SRL(p, tmp, tmp, 6); /* convert to entrylo0 */
L
Linus Torvalds 已提交
646
	if (r4k_250MHZhwbug())
647 648 649
		uasm_i_mtc0(p, 0, C0_ENTRYLO0);
	uasm_i_mtc0(p, tmp, C0_ENTRYLO0); /* load it */
	UASM_i_SRL(p, ptep, ptep, 6); /* convert to entrylo1 */
L
Linus Torvalds 已提交
650
	if (r45k_bvahwbug())
651
		uasm_i_mfc0(p, tmp, C0_INDEX);
L
Linus Torvalds 已提交
652
	if (r4k_250MHZhwbug())
653 654
		uasm_i_mtc0(p, 0, C0_ENTRYLO1);
	uasm_i_mtc0(p, ptep, C0_ENTRYLO1); /* load it */
L
Linus Torvalds 已提交
655 656 657
#endif
}

658
static void __cpuinit build_r4000_tlb_refill_handler(void)
L
Linus Torvalds 已提交
659 660
{
	u32 *p = tlb_handler;
661 662
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
663 664 665 666 667 668 669 670 671 672 673 674
	u32 *f;
	unsigned int final_len;

	memset(tlb_handler, 0, sizeof(tlb_handler));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));
	memset(final_handler, 0, sizeof(final_handler));

	/*
	 * create the plain linear handler
	 */
	if (bcm1250_m3_war()) {
675 676 677 678 679 680
		UASM_i_MFC0(&p, K0, C0_BADVADDR);
		UASM_i_MFC0(&p, K1, C0_ENTRYHI);
		uasm_i_xor(&p, K0, K0, K1);
		UASM_i_SRL(&p, K0, K0, PAGE_SHIFT + 1);
		uasm_il_bnez(&p, &r, K0, label_leave);
		/* No need for uasm_i_nop */
L
Linus Torvalds 已提交
681 682
	}

683
#ifdef CONFIG_64BIT
L
Linus Torvalds 已提交
684 685 686 687 688 689 690 691
	build_get_pmde64(&p, &l, &r, K0, K1); /* get pmd in K1 */
#else
	build_get_pgde32(&p, K0, K1); /* get pgd in K1 */
#endif

	build_get_ptep(&p, K0, K1);
	build_update_entries(&p, K0, K1);
	build_tlb_write_entry(&p, &l, &r, tlb_random);
692 693
	uasm_l_leave(&l, p);
	uasm_i_eret(&p); /* return from trap */
L
Linus Torvalds 已提交
694

695
#ifdef CONFIG_64BIT
L
Linus Torvalds 已提交
696 697 698 699 700 701 702
	build_get_pgd_vmalloc64(&p, &l, &r, K0, K1);
#endif

	/*
	 * Overflow check: For the 64bit handler, we need at least one
	 * free instruction slot for the wrap-around branch. In worst
	 * case, if the intended insertion point is a delay slot, we
M
Matt LaPlante 已提交
703
	 * need three, with the second nop'ed and the third being
L
Linus Torvalds 已提交
704 705
	 * unused.
	 */
706 707
	/* Loongson2 ebase is different than r4k, we have more space */
#if defined(CONFIG_32BIT) || defined(CONFIG_CPU_LOONGSON2)
L
Linus Torvalds 已提交
708 709 710 711 712
	if ((p - tlb_handler) > 64)
		panic("TLB refill handler space exceeded");
#else
	if (((p - tlb_handler) > 63)
	    || (((p - tlb_handler) > 61)
713
		&& uasm_insn_has_bdelay(relocs, tlb_handler + 29)))
L
Linus Torvalds 已提交
714 715 716 717 718 719
		panic("TLB refill handler space exceeded");
#endif

	/*
	 * Now fold the handler in the TLB refill handler space.
	 */
720
#if defined(CONFIG_32BIT) || defined(CONFIG_CPU_LOONGSON2)
L
Linus Torvalds 已提交
721 722
	f = final_handler;
	/* Simplest case, just copy the handler. */
723
	uasm_copy_handler(relocs, labels, tlb_handler, p, f);
L
Linus Torvalds 已提交
724
	final_len = p - tlb_handler;
725
#else /* CONFIG_64BIT */
L
Linus Torvalds 已提交
726 727 728
	f = final_handler + 32;
	if ((p - tlb_handler) <= 32) {
		/* Just copy the handler. */
729
		uasm_copy_handler(relocs, labels, tlb_handler, p, f);
L
Linus Torvalds 已提交
730 731 732 733 734 735 736
		final_len = p - tlb_handler;
	} else {
		u32 *split = tlb_handler + 30;

		/*
		 * Find the split point.
		 */
737
		if (uasm_insn_has_bdelay(relocs, split - 1))
L
Linus Torvalds 已提交
738 739 740
			split--;

		/* Copy first part of the handler. */
741
		uasm_copy_handler(relocs, labels, tlb_handler, split, f);
L
Linus Torvalds 已提交
742 743 744
		f += split - tlb_handler;

		/* Insert branch. */
745 746 747 748
		uasm_l_split(&l, final_handler);
		uasm_il_b(&f, &r, label_split);
		if (uasm_insn_has_bdelay(relocs, split))
			uasm_i_nop(&f);
L
Linus Torvalds 已提交
749
		else {
750 751
			uasm_copy_handler(relocs, labels, split, split + 1, f);
			uasm_move_labels(labels, f, f + 1, -1);
L
Linus Torvalds 已提交
752 753 754 755 756
			f++;
			split++;
		}

		/* Copy the rest of the handler. */
757
		uasm_copy_handler(relocs, labels, split, p, final_handler);
L
Linus Torvalds 已提交
758 759
		final_len = (f - (final_handler + 32)) + (p - split);
	}
760
#endif /* CONFIG_64BIT */
L
Linus Torvalds 已提交
761

762 763 764
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB refill handler (%u instructions).\n",
		 final_len);
L
Linus Torvalds 已提交
765

766
	memcpy((void *)ebase, final_handler, 0x100);
767 768

	dump_handler((u32 *)ebase, 64);
L
Linus Torvalds 已提交
769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785
}

/*
 * TLB load/store/modify handlers.
 *
 * Only the fastpath gets synthesized at runtime, the slowpath for
 * do_page_fault remains normal asm.
 */
extern void tlb_do_page_fault_0(void);
extern void tlb_do_page_fault_1(void);

/*
 * 128 instructions for the fastpath handler is generous and should
 * never be exceeded.
 */
#define FASTPATH_SIZE 128

786 787 788
u32 handle_tlbl[FASTPATH_SIZE] __cacheline_aligned;
u32 handle_tlbs[FASTPATH_SIZE] __cacheline_aligned;
u32 handle_tlbm[FASTPATH_SIZE] __cacheline_aligned;
L
Linus Torvalds 已提交
789

790
static void __cpuinit
791
iPTE_LW(u32 **p, struct uasm_label **l, unsigned int pte, unsigned int ptr)
L
Linus Torvalds 已提交
792 793 794 795
{
#ifdef CONFIG_SMP
# ifdef CONFIG_64BIT_PHYS_ADDR
	if (cpu_has_64bits)
796
		uasm_i_lld(p, pte, 0, ptr);
L
Linus Torvalds 已提交
797 798
	else
# endif
799
		UASM_i_LL(p, pte, 0, ptr);
L
Linus Torvalds 已提交
800 801 802
#else
# ifdef CONFIG_64BIT_PHYS_ADDR
	if (cpu_has_64bits)
803
		uasm_i_ld(p, pte, 0, ptr);
L
Linus Torvalds 已提交
804 805
	else
# endif
806
		UASM_i_LW(p, pte, 0, ptr);
L
Linus Torvalds 已提交
807 808 809
#endif
}

810
static void __cpuinit
811
iPTE_SW(u32 **p, struct uasm_reloc **r, unsigned int pte, unsigned int ptr,
812
	unsigned int mode)
L
Linus Torvalds 已提交
813
{
814 815 816 817
#ifdef CONFIG_64BIT_PHYS_ADDR
	unsigned int hwmode = mode & (_PAGE_VALID | _PAGE_DIRTY);
#endif

818
	uasm_i_ori(p, pte, pte, mode);
L
Linus Torvalds 已提交
819 820 821
#ifdef CONFIG_SMP
# ifdef CONFIG_64BIT_PHYS_ADDR
	if (cpu_has_64bits)
822
		uasm_i_scd(p, pte, 0, ptr);
L
Linus Torvalds 已提交
823 824
	else
# endif
825
		UASM_i_SC(p, pte, 0, ptr);
L
Linus Torvalds 已提交
826 827

	if (r10000_llsc_war())
828
		uasm_il_beqzl(p, r, pte, label_smp_pgtable_change);
L
Linus Torvalds 已提交
829
	else
830
		uasm_il_beqz(p, r, pte, label_smp_pgtable_change);
L
Linus Torvalds 已提交
831 832 833

# ifdef CONFIG_64BIT_PHYS_ADDR
	if (!cpu_has_64bits) {
834 835 836 837 838 839 840
		/* no uasm_i_nop needed */
		uasm_i_ll(p, pte, sizeof(pte_t) / 2, ptr);
		uasm_i_ori(p, pte, pte, hwmode);
		uasm_i_sc(p, pte, sizeof(pte_t) / 2, ptr);
		uasm_il_beqz(p, r, pte, label_smp_pgtable_change);
		/* no uasm_i_nop needed */
		uasm_i_lw(p, pte, 0, ptr);
L
Linus Torvalds 已提交
841
	} else
842
		uasm_i_nop(p);
L
Linus Torvalds 已提交
843
# else
844
	uasm_i_nop(p);
L
Linus Torvalds 已提交
845 846 847 848
# endif
#else
# ifdef CONFIG_64BIT_PHYS_ADDR
	if (cpu_has_64bits)
849
		uasm_i_sd(p, pte, 0, ptr);
L
Linus Torvalds 已提交
850 851
	else
# endif
852
		UASM_i_SW(p, pte, 0, ptr);
L
Linus Torvalds 已提交
853 854 855

# ifdef CONFIG_64BIT_PHYS_ADDR
	if (!cpu_has_64bits) {
856 857 858 859
		uasm_i_lw(p, pte, sizeof(pte_t) / 2, ptr);
		uasm_i_ori(p, pte, pte, hwmode);
		uasm_i_sw(p, pte, sizeof(pte_t) / 2, ptr);
		uasm_i_lw(p, pte, 0, ptr);
L
Linus Torvalds 已提交
860 861 862 863 864 865 866 867 868 869
	}
# endif
#endif
}

/*
 * Check if PTE is present, if not then jump to LABEL. PTR points to
 * the page table where this PTE is located, PTE will be re-loaded
 * with it's original value.
 */
870
static void __cpuinit
871
build_pte_present(u32 **p, struct uasm_label **l, struct uasm_reloc **r,
L
Linus Torvalds 已提交
872 873
		  unsigned int pte, unsigned int ptr, enum label_id lid)
{
874 875 876
	uasm_i_andi(p, pte, pte, _PAGE_PRESENT | _PAGE_READ);
	uasm_i_xori(p, pte, pte, _PAGE_PRESENT | _PAGE_READ);
	uasm_il_bnez(p, r, pte, lid);
877
	iPTE_LW(p, l, pte, ptr);
L
Linus Torvalds 已提交
878 879 880
}

/* Make PTE valid, store result in PTR. */
881
static void __cpuinit
882
build_make_valid(u32 **p, struct uasm_reloc **r, unsigned int pte,
L
Linus Torvalds 已提交
883 884
		 unsigned int ptr)
{
885 886 887
	unsigned int mode = _PAGE_VALID | _PAGE_ACCESSED;

	iPTE_SW(p, r, pte, ptr, mode);
L
Linus Torvalds 已提交
888 889 890 891 892 893
}

/*
 * Check if PTE can be written to, if not branch to LABEL. Regardless
 * restore PTE with value from PTR when done.
 */
894
static void __cpuinit
895
build_pte_writable(u32 **p, struct uasm_label **l, struct uasm_reloc **r,
L
Linus Torvalds 已提交
896 897
		   unsigned int pte, unsigned int ptr, enum label_id lid)
{
898 899 900
	uasm_i_andi(p, pte, pte, _PAGE_PRESENT | _PAGE_WRITE);
	uasm_i_xori(p, pte, pte, _PAGE_PRESENT | _PAGE_WRITE);
	uasm_il_bnez(p, r, pte, lid);
901
	iPTE_LW(p, l, pte, ptr);
L
Linus Torvalds 已提交
902 903 904 905 906
}

/* Make PTE writable, update software status bits as well, then store
 * at PTR.
 */
907
static void __cpuinit
908
build_make_write(u32 **p, struct uasm_reloc **r, unsigned int pte,
L
Linus Torvalds 已提交
909 910
		 unsigned int ptr)
{
911 912 913 914
	unsigned int mode = (_PAGE_ACCESSED | _PAGE_MODIFIED | _PAGE_VALID
			     | _PAGE_DIRTY);

	iPTE_SW(p, r, pte, ptr, mode);
L
Linus Torvalds 已提交
915 916 917 918 919 920
}

/*
 * Check if PTE can be modified, if not branch to LABEL. Regardless
 * restore PTE with value from PTR when done.
 */
921
static void __cpuinit
922
build_pte_modifiable(u32 **p, struct uasm_label **l, struct uasm_reloc **r,
L
Linus Torvalds 已提交
923 924
		     unsigned int pte, unsigned int ptr, enum label_id lid)
{
925 926
	uasm_i_andi(p, pte, pte, _PAGE_WRITE);
	uasm_il_beqz(p, r, pte, lid);
927
	iPTE_LW(p, l, pte, ptr);
L
Linus Torvalds 已提交
928 929 930 931 932 933
}

/*
 * R3000 style TLB load/store/modify handlers.
 */

934 935 936 937
/*
 * This places the pte into ENTRYLO0 and writes it with tlbwi.
 * Then it returns.
 */
938
static void __cpuinit
939
build_r3000_pte_reload_tlbwi(u32 **p, unsigned int pte, unsigned int tmp)
L
Linus Torvalds 已提交
940
{
941 942 943 944 945
	uasm_i_mtc0(p, pte, C0_ENTRYLO0); /* cp0 delay */
	uasm_i_mfc0(p, tmp, C0_EPC); /* cp0 delay */
	uasm_i_tlbwi(p);
	uasm_i_jr(p, tmp);
	uasm_i_rfe(p); /* branch delay */
L
Linus Torvalds 已提交
946 947 948
}

/*
949 950 951 952
 * This places the pte into ENTRYLO0 and writes it with tlbwi
 * or tlbwr as appropriate.  This is because the index register
 * may have the probe fail bit set as a result of a trap on a
 * kseg2 access, i.e. without refill.  Then it returns.
L
Linus Torvalds 已提交
953
 */
954
static void __cpuinit
955 956 957 958 959 960 961 962 963 964 965 966 967 968 969
build_r3000_tlb_reload_write(u32 **p, struct uasm_label **l,
			     struct uasm_reloc **r, unsigned int pte,
			     unsigned int tmp)
{
	uasm_i_mfc0(p, tmp, C0_INDEX);
	uasm_i_mtc0(p, pte, C0_ENTRYLO0); /* cp0 delay */
	uasm_il_bltz(p, r, tmp, label_r3000_write_probe_fail); /* cp0 delay */
	uasm_i_mfc0(p, tmp, C0_EPC); /* branch delay */
	uasm_i_tlbwi(p); /* cp0 delay */
	uasm_i_jr(p, tmp);
	uasm_i_rfe(p); /* branch delay */
	uasm_l_r3000_write_probe_fail(l, *p);
	uasm_i_tlbwr(p); /* cp0 delay */
	uasm_i_jr(p, tmp);
	uasm_i_rfe(p); /* branch delay */
L
Linus Torvalds 已提交
970 971
}

972
static void __cpuinit
L
Linus Torvalds 已提交
973 974 975 976 977
build_r3000_tlbchange_handler_head(u32 **p, unsigned int pte,
				   unsigned int ptr)
{
	long pgdc = (long)pgd_current;

978 979 980 981 982 983 984 985 986 987 988 989
	uasm_i_mfc0(p, pte, C0_BADVADDR);
	uasm_i_lui(p, ptr, uasm_rel_hi(pgdc)); /* cp0 delay */
	uasm_i_lw(p, ptr, uasm_rel_lo(pgdc), ptr);
	uasm_i_srl(p, pte, pte, 22); /* load delay */
	uasm_i_sll(p, pte, pte, 2);
	uasm_i_addu(p, ptr, ptr, pte);
	uasm_i_mfc0(p, pte, C0_CONTEXT);
	uasm_i_lw(p, ptr, 0, ptr); /* cp0 delay */
	uasm_i_andi(p, pte, pte, 0xffc); /* load delay */
	uasm_i_addu(p, ptr, ptr, pte);
	uasm_i_lw(p, pte, 0, ptr);
	uasm_i_tlbp(p); /* load delay */
L
Linus Torvalds 已提交
990 991
}

992
static void __cpuinit build_r3000_tlb_load_handler(void)
L
Linus Torvalds 已提交
993 994
{
	u32 *p = handle_tlbl;
995 996
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
997 998 999 1000 1001 1002 1003

	memset(handle_tlbl, 0, sizeof(handle_tlbl));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));

	build_r3000_tlbchange_handler_head(&p, K0, K1);
	build_pte_present(&p, &l, &r, K0, K1, label_nopage_tlbl);
1004
	uasm_i_nop(&p); /* load delay */
L
Linus Torvalds 已提交
1005
	build_make_valid(&p, &r, K0, K1);
1006
	build_r3000_tlb_reload_write(&p, &l, &r, K0, K1);
L
Linus Torvalds 已提交
1007

1008 1009 1010
	uasm_l_nopage_tlbl(&l, p);
	uasm_i_j(&p, (unsigned long)tlb_do_page_fault_0 & 0x0fffffff);
	uasm_i_nop(&p);
L
Linus Torvalds 已提交
1011 1012 1013 1014

	if ((p - handle_tlbl) > FASTPATH_SIZE)
		panic("TLB load handler fastpath space exceeded");

1015 1016 1017
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB load handler fastpath (%u instructions).\n",
		 (unsigned int)(p - handle_tlbl));
L
Linus Torvalds 已提交
1018

1019
	dump_handler(handle_tlbl, ARRAY_SIZE(handle_tlbl));
L
Linus Torvalds 已提交
1020 1021
}

1022
static void __cpuinit build_r3000_tlb_store_handler(void)
L
Linus Torvalds 已提交
1023 1024
{
	u32 *p = handle_tlbs;
1025 1026
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
1027 1028 1029 1030 1031 1032 1033

	memset(handle_tlbs, 0, sizeof(handle_tlbs));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));

	build_r3000_tlbchange_handler_head(&p, K0, K1);
	build_pte_writable(&p, &l, &r, K0, K1, label_nopage_tlbs);
1034
	uasm_i_nop(&p); /* load delay */
L
Linus Torvalds 已提交
1035
	build_make_write(&p, &r, K0, K1);
1036
	build_r3000_tlb_reload_write(&p, &l, &r, K0, K1);
L
Linus Torvalds 已提交
1037

1038 1039 1040
	uasm_l_nopage_tlbs(&l, p);
	uasm_i_j(&p, (unsigned long)tlb_do_page_fault_1 & 0x0fffffff);
	uasm_i_nop(&p);
L
Linus Torvalds 已提交
1041 1042 1043 1044

	if ((p - handle_tlbs) > FASTPATH_SIZE)
		panic("TLB store handler fastpath space exceeded");

1045 1046 1047
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB store handler fastpath (%u instructions).\n",
		 (unsigned int)(p - handle_tlbs));
L
Linus Torvalds 已提交
1048

1049
	dump_handler(handle_tlbs, ARRAY_SIZE(handle_tlbs));
L
Linus Torvalds 已提交
1050 1051
}

1052
static void __cpuinit build_r3000_tlb_modify_handler(void)
L
Linus Torvalds 已提交
1053 1054
{
	u32 *p = handle_tlbm;
1055 1056
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
1057 1058 1059 1060 1061 1062 1063

	memset(handle_tlbm, 0, sizeof(handle_tlbm));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));

	build_r3000_tlbchange_handler_head(&p, K0, K1);
	build_pte_modifiable(&p, &l, &r, K0, K1, label_nopage_tlbm);
1064
	uasm_i_nop(&p); /* load delay */
L
Linus Torvalds 已提交
1065
	build_make_write(&p, &r, K0, K1);
1066
	build_r3000_pte_reload_tlbwi(&p, K0, K1);
L
Linus Torvalds 已提交
1067

1068 1069 1070
	uasm_l_nopage_tlbm(&l, p);
	uasm_i_j(&p, (unsigned long)tlb_do_page_fault_1 & 0x0fffffff);
	uasm_i_nop(&p);
L
Linus Torvalds 已提交
1071 1072 1073 1074

	if ((p - handle_tlbm) > FASTPATH_SIZE)
		panic("TLB modify handler fastpath space exceeded");

1075 1076 1077
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB modify handler fastpath (%u instructions).\n",
		 (unsigned int)(p - handle_tlbm));
L
Linus Torvalds 已提交
1078

1079
	dump_handler(handle_tlbm, ARRAY_SIZE(handle_tlbm));
L
Linus Torvalds 已提交
1080 1081 1082 1083 1084
}

/*
 * R4000 style TLB load/store/modify handlers.
 */
1085
static void __cpuinit
1086 1087
build_r4000_tlbchange_handler_head(u32 **p, struct uasm_label **l,
				   struct uasm_reloc **r, unsigned int pte,
L
Linus Torvalds 已提交
1088 1089
				   unsigned int ptr)
{
1090
#ifdef CONFIG_64BIT
L
Linus Torvalds 已提交
1091 1092 1093 1094 1095
	build_get_pmde64(p, l, r, pte, ptr); /* get pmd in ptr */
#else
	build_get_pgde32(p, pte, ptr); /* get pgd in ptr */
#endif

1096 1097 1098 1099 1100
	UASM_i_MFC0(p, pte, C0_BADVADDR);
	UASM_i_LW(p, ptr, 0, ptr);
	UASM_i_SRL(p, pte, pte, PAGE_SHIFT + PTE_ORDER - PTE_T_LOG2);
	uasm_i_andi(p, pte, pte, (PTRS_PER_PTE - 1) << PTE_T_LOG2);
	UASM_i_ADDU(p, ptr, ptr, pte);
L
Linus Torvalds 已提交
1101 1102

#ifdef CONFIG_SMP
1103 1104
	uasm_l_smp_pgtable_change(l, *p);
#endif
1105
	iPTE_LW(p, l, pte, ptr); /* get even pte */
1106 1107
	if (!m4kc_tlbp_war())
		build_tlb_probe_entry(p);
L
Linus Torvalds 已提交
1108 1109
}

1110
static void __cpuinit
1111 1112
build_r4000_tlbchange_handler_tail(u32 **p, struct uasm_label **l,
				   struct uasm_reloc **r, unsigned int tmp,
L
Linus Torvalds 已提交
1113 1114
				   unsigned int ptr)
{
1115 1116
	uasm_i_ori(p, ptr, ptr, sizeof(pte_t));
	uasm_i_xori(p, ptr, ptr, sizeof(pte_t));
L
Linus Torvalds 已提交
1117 1118
	build_update_entries(p, tmp, ptr);
	build_tlb_write_entry(p, l, r, tlb_indexed);
1119 1120
	uasm_l_leave(l, *p);
	uasm_i_eret(p); /* return from trap */
L
Linus Torvalds 已提交
1121

1122
#ifdef CONFIG_64BIT
L
Linus Torvalds 已提交
1123 1124 1125 1126
	build_get_pgd_vmalloc64(p, l, r, tmp, ptr);
#endif
}

1127
static void __cpuinit build_r4000_tlb_load_handler(void)
L
Linus Torvalds 已提交
1128 1129
{
	u32 *p = handle_tlbl;
1130 1131
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
1132 1133 1134 1135 1136 1137

	memset(handle_tlbl, 0, sizeof(handle_tlbl));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));

	if (bcm1250_m3_war()) {
1138 1139 1140 1141 1142 1143
		UASM_i_MFC0(&p, K0, C0_BADVADDR);
		UASM_i_MFC0(&p, K1, C0_ENTRYHI);
		uasm_i_xor(&p, K0, K0, K1);
		UASM_i_SRL(&p, K0, K0, PAGE_SHIFT + 1);
		uasm_il_bnez(&p, &r, K0, label_leave);
		/* No need for uasm_i_nop */
L
Linus Torvalds 已提交
1144 1145 1146 1147
	}

	build_r4000_tlbchange_handler_head(&p, &l, &r, K0, K1);
	build_pte_present(&p, &l, &r, K0, K1, label_nopage_tlbl);
1148 1149
	if (m4kc_tlbp_war())
		build_tlb_probe_entry(&p);
L
Linus Torvalds 已提交
1150 1151 1152
	build_make_valid(&p, &r, K0, K1);
	build_r4000_tlbchange_handler_tail(&p, &l, &r, K0, K1);

1153 1154 1155
	uasm_l_nopage_tlbl(&l, p);
	uasm_i_j(&p, (unsigned long)tlb_do_page_fault_0 & 0x0fffffff);
	uasm_i_nop(&p);
L
Linus Torvalds 已提交
1156 1157 1158 1159

	if ((p - handle_tlbl) > FASTPATH_SIZE)
		panic("TLB load handler fastpath space exceeded");

1160 1161 1162
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB load handler fastpath (%u instructions).\n",
		 (unsigned int)(p - handle_tlbl));
L
Linus Torvalds 已提交
1163

1164
	dump_handler(handle_tlbl, ARRAY_SIZE(handle_tlbl));
L
Linus Torvalds 已提交
1165 1166
}

1167
static void __cpuinit build_r4000_tlb_store_handler(void)
L
Linus Torvalds 已提交
1168 1169
{
	u32 *p = handle_tlbs;
1170 1171
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
1172 1173 1174 1175 1176 1177 1178

	memset(handle_tlbs, 0, sizeof(handle_tlbs));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));

	build_r4000_tlbchange_handler_head(&p, &l, &r, K0, K1);
	build_pte_writable(&p, &l, &r, K0, K1, label_nopage_tlbs);
1179 1180
	if (m4kc_tlbp_war())
		build_tlb_probe_entry(&p);
L
Linus Torvalds 已提交
1181 1182 1183
	build_make_write(&p, &r, K0, K1);
	build_r4000_tlbchange_handler_tail(&p, &l, &r, K0, K1);

1184 1185 1186
	uasm_l_nopage_tlbs(&l, p);
	uasm_i_j(&p, (unsigned long)tlb_do_page_fault_1 & 0x0fffffff);
	uasm_i_nop(&p);
L
Linus Torvalds 已提交
1187 1188 1189 1190

	if ((p - handle_tlbs) > FASTPATH_SIZE)
		panic("TLB store handler fastpath space exceeded");

1191 1192 1193
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB store handler fastpath (%u instructions).\n",
		 (unsigned int)(p - handle_tlbs));
L
Linus Torvalds 已提交
1194

1195
	dump_handler(handle_tlbs, ARRAY_SIZE(handle_tlbs));
L
Linus Torvalds 已提交
1196 1197
}

1198
static void __cpuinit build_r4000_tlb_modify_handler(void)
L
Linus Torvalds 已提交
1199 1200
{
	u32 *p = handle_tlbm;
1201 1202
	struct uasm_label *l = labels;
	struct uasm_reloc *r = relocs;
L
Linus Torvalds 已提交
1203 1204 1205 1206 1207 1208 1209

	memset(handle_tlbm, 0, sizeof(handle_tlbm));
	memset(labels, 0, sizeof(labels));
	memset(relocs, 0, sizeof(relocs));

	build_r4000_tlbchange_handler_head(&p, &l, &r, K0, K1);
	build_pte_modifiable(&p, &l, &r, K0, K1, label_nopage_tlbm);
1210 1211
	if (m4kc_tlbp_war())
		build_tlb_probe_entry(&p);
L
Linus Torvalds 已提交
1212 1213 1214 1215
	/* Present and writable bits set, set accessed and dirty bits. */
	build_make_write(&p, &r, K0, K1);
	build_r4000_tlbchange_handler_tail(&p, &l, &r, K0, K1);

1216 1217 1218
	uasm_l_nopage_tlbm(&l, p);
	uasm_i_j(&p, (unsigned long)tlb_do_page_fault_1 & 0x0fffffff);
	uasm_i_nop(&p);
L
Linus Torvalds 已提交
1219 1220 1221 1222

	if ((p - handle_tlbm) > FASTPATH_SIZE)
		panic("TLB modify handler fastpath space exceeded");

1223 1224 1225
	uasm_resolve_relocs(relocs, labels);
	pr_debug("Wrote TLB modify handler fastpath (%u instructions).\n",
		 (unsigned int)(p - handle_tlbm));
1226

1227
	dump_handler(handle_tlbm, ARRAY_SIZE(handle_tlbm));
L
Linus Torvalds 已提交
1228 1229
}

1230
void __cpuinit build_tlb_refill_handler(void)
L
Linus Torvalds 已提交
1231 1232 1233 1234 1235 1236 1237 1238
{
	/*
	 * The refill handler is generated per-CPU, multi-node systems
	 * may have local storage for it. The other handlers are only
	 * needed once.
	 */
	static int run_once = 0;

1239
	switch (current_cpu_type()) {
L
Linus Torvalds 已提交
1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274
	case CPU_R2000:
	case CPU_R3000:
	case CPU_R3000A:
	case CPU_R3081E:
	case CPU_TX3912:
	case CPU_TX3922:
	case CPU_TX3927:
		build_r3000_tlb_refill_handler();
		if (!run_once) {
			build_r3000_tlb_load_handler();
			build_r3000_tlb_store_handler();
			build_r3000_tlb_modify_handler();
			run_once++;
		}
		break;

	case CPU_R6000:
	case CPU_R6000A:
		panic("No R6000 TLB refill handler yet");
		break;

	case CPU_R8000:
		panic("No R8000 TLB refill handler yet");
		break;

	default:
		build_r4000_tlb_refill_handler();
		if (!run_once) {
			build_r4000_tlb_load_handler();
			build_r4000_tlb_store_handler();
			build_r4000_tlb_modify_handler();
			run_once++;
		}
	}
}
1275

1276
void __cpuinit flush_tlb_handlers(void)
1277
{
1278
	local_flush_icache_range((unsigned long)handle_tlbl,
1279
			   (unsigned long)handle_tlbl + sizeof(handle_tlbl));
1280
	local_flush_icache_range((unsigned long)handle_tlbs,
1281
			   (unsigned long)handle_tlbs + sizeof(handle_tlbs));
1282
	local_flush_icache_range((unsigned long)handle_tlbm,
1283 1284
			   (unsigned long)handle_tlbm + sizeof(handle_tlbm));
}