summaryrefslogtreecommitdiffstats
path: root/arch/sparc64/kernel/dtlb_miss.S
blob: 4d71d967c83b73e1b194284cac6c27250b6ab19d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
/* $Id: dtlb_miss.S,v 1.14 1997/10/14 01:48:28 davem Exp $
 * dtlb_miss.S:	Data TLB miss code, this is included directly
 *              into the trap table.
 *
 * Copyright (C) 1996,1997 David S. Miller (davem@caip.rutgers.edu)
 * Copyright (C) 1997 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
 */

/* The basic algorithm is:
 *
 * if(faulting_context != 0) {
 *		pgd = pgd_offset(current->mm.pgd, fault_address);
 * page_table_walk_continue:
 *		pmd = pmd_offset(pgd, fault_address);
 *		pte = pte_offset(pmd, fault_address);
 *		if(pte & _PAGE_V) {
 *			tlb_load(pte, fault_address);
 *			return_from_trap();
 *		}
 *		goto longer_processing;
 * } else {
 *		if(fault_address >= PAGE_OFFSET) {
 *			pte_val = PAGE_KERNEL;
 *			if (fault_address & 0x10000000000)
 *				pte_val = PAGE_KERNEL_IO;
 *			tlb_load(__pa(fault_address) | pte_val);
 *			return_from_trap();
 *		} else {
 *			pgd = pgd_offset(swapper_pg_dir, fault_address);
 *			goto page_table_walk_continue;
 *		}
 * }
 *
 * This is optimized for user TLB misses on purpose.
 */

#define KERN_HIGHBITS	((_PAGE_VALID | _PAGE_SZ4MB) ^ 0xfffff80000000000)
#define KERN_LOWBITS	(_PAGE_CP | _PAGE_CV | _PAGE_P | _PAGE_W)
#define KERN_LOWBITS_IO	((_PAGE_E | _PAGE_P | _PAGE_W) ^ KERN_LOWBITS)

				/* ICACHE line 1 */
  /*0x00*/	ldxa		[%g0] ASI_DMMU, %g1		! Get TAG_TARGET
  /*0x04*/	srlx		%g1, 8, %g3			! Position PGD offset
  /*0x08*/	srlx		%g1, 48, %g5			! Shift down CONTEXT bits
  /*0x0c*/	and		%g3, %g2, %g3			! Mask PGD offset
  /*0x10*/	sllx		%g1, 2, %g4			! Position PMD offset
  /*0x14*/	brz,pn		%g5, 3f				! Context 0 == kernel
  /*0x18*/	 and		%g4, %g2, %g4			! Mask PMD offset
  /*0x1c*/	ldxa		[%g0] ASI_DMMU_TSB_8KB_PTR, %g1	! For PTE offset

				/* ICACHE line 2 */
  /*0x20*/	ldxa		[%g7 + %g3] ASI_PHYS_USE_EC, %g5	! Load PGD
  /*0x24*/	srlx		%g1, 1, %g1				! PTE offset
2:/*0x28*/	ldxa		[%g5 + %g4] ASI_PHYS_USE_EC, %g3	! Load PMD
  /*0x2c*/	ldxa		[%g3 + %g1] ASI_PHYS_USE_EC, %g5	! Load PTE
  /*0x30*/	brgez,pn	%g5, sparc64_dtlb_refbit_catch		! Valid set?
  /*0x34*/	 nop							! delay
  /*0x38*/	stxa		%g5, [%g0] ASI_DTLB_DATA_IN		! TLB load
  /*0x3c*/	retry							! Trap return

3:				/* ICACHE line 3 */
  /*0x40*/	sllx		%g1, 22, %g5			! This is now physical page + PAGE_OFFSET
  /*0x44*/	brgez,pn	%g5, 4f				! If >= 0, then walk down page tables
  /*0x48*/	 sethi		%uhi(KERN_HIGHBITS), %g1	! Construct PTE ^ PAGE_OFFSET
  /*0x4c*/	andcc		%g3, 0x400, %g0			! Slick trick...
  /*0x50*/	sllx		%g1, 32, %g1			! Move high bits up
  /*0x54*/	or		%g1, (KERN_LOWBITS), %g1	! Assume not IO
  /*0x58*/	bne,a,pn	%icc, 5f			! Is it an IO page?
  /*0x5c*/	 xor		%g1, (KERN_LOWBITS_IO), %g1	! Aha, it is IO...

				/* ICACHE line 4 */
5:/*0x60*/	xor		%g1, %g5, %g1			! Slick trick II...
  /*0x64*/	stxa		%g1, [%g0] ASI_DTLB_DATA_IN	! TLB load
  /*0x68*/	retry						! Trap return
4:/*0x6c*/	ldxa		[%g0] ASI_DMMU_TSB_8KB_PTR, %g1	! For PTE offset
  /*0x70*/	ldxa		[%g6 + %g3] ASI_PHYS_USE_EC, %g5	! Load kern PGD
  /*0x74*/	ba,pt		%xcc, 2b			! Go back up top
  /*0x78*/	 srlx		%g1, 1, %g1
  /*0x7c*/	nop

#undef KERN_HIGHBITS
#undef KERN_LOWBITS
#undef KERN_LOWBITS_IO