Merge git://git.kernel.org/pub/scm/linux/kernel/git/davem/net
[sfrench/cifs-2.6.git] / arch / sparc / kernel / ktlb.S
1 /* arch/sparc64/kernel/ktlb.S: Kernel mapping TLB miss handling.
2  *
3  * Copyright (C) 1995, 1997, 2005, 2008 David S. Miller <davem@davemloft.net>
4  * Copyright (C) 1996 Eddie C. Dost        (ecd@brainaid.de)
5  * Copyright (C) 1996 Miguel de Icaza      (miguel@nuclecu.unam.mx)
6  * Copyright (C) 1996,98,99 Jakub Jelinek  (jj@sunsite.mff.cuni.cz)
7  */
8
9 #include <asm/head.h>
10 #include <asm/asi.h>
11 #include <asm/page.h>
12 #include <asm/pgtable.h>
13 #include <asm/tsb.h>
14
15         .text
16         .align          32
17
18 kvmap_itlb:
19         /* g6: TAG TARGET */
20         mov             TLB_TAG_ACCESS, %g4
21         ldxa            [%g4] ASI_IMMU, %g4
22
23         /* sun4v_itlb_miss branches here with the missing virtual
24          * address already loaded into %g4
25          */
26 kvmap_itlb_4v:
27
28         /* Catch kernel NULL pointer calls.  */
29         sethi           %hi(PAGE_SIZE), %g5
30         cmp             %g4, %g5
31         blu,pn          %xcc, kvmap_itlb_longpath
32          nop
33
34         KERN_TSB_LOOKUP_TL1(%g4, %g6, %g5, %g1, %g2, %g3, kvmap_itlb_load)
35
36 kvmap_itlb_tsb_miss:
37         sethi           %hi(LOW_OBP_ADDRESS), %g5
38         cmp             %g4, %g5
39         blu,pn          %xcc, kvmap_itlb_vmalloc_addr
40          mov            0x1, %g5
41         sllx            %g5, 32, %g5
42         cmp             %g4, %g5
43         blu,pn          %xcc, kvmap_itlb_obp
44          nop
45
46 kvmap_itlb_vmalloc_addr:
47         KERN_PGTABLE_WALK(%g4, %g5, %g2, kvmap_itlb_longpath)
48
49         TSB_LOCK_TAG(%g1, %g2, %g7)
50         TSB_WRITE(%g1, %g5, %g6)
51
52         /* fallthrough to TLB load */
53
54 kvmap_itlb_load:
55
56 661:    stxa            %g5, [%g0] ASI_ITLB_DATA_IN
57         retry
58         .section        .sun4v_2insn_patch, "ax"
59         .word           661b
60         nop
61         nop
62         .previous
63
64         /* For sun4v the ASI_ITLB_DATA_IN store and the retry
65          * instruction get nop'd out and we get here to branch
66          * to the sun4v tlb load code.  The registers are setup
67          * as follows:
68          *
69          * %g4: vaddr
70          * %g5: PTE
71          * %g6: TAG
72          *
73          * The sun4v TLB load wants the PTE in %g3 so we fix that
74          * up here.
75          */
76         ba,pt           %xcc, sun4v_itlb_load
77          mov            %g5, %g3
78
79 kvmap_itlb_longpath:
80
81 661:    rdpr    %pstate, %g5
82         wrpr    %g5, PSTATE_AG | PSTATE_MG, %pstate
83         .section .sun4v_2insn_patch, "ax"
84         .word   661b
85         SET_GL(1)
86         nop
87         .previous
88
89         rdpr    %tpc, %g5
90         ba,pt   %xcc, sparc64_realfault_common
91          mov    FAULT_CODE_ITLB, %g4
92
93 kvmap_itlb_obp:
94         OBP_TRANS_LOOKUP(%g4, %g5, %g2, %g3, kvmap_itlb_longpath)
95
96         TSB_LOCK_TAG(%g1, %g2, %g7)
97
98         TSB_WRITE(%g1, %g5, %g6)
99
100         ba,pt           %xcc, kvmap_itlb_load
101          nop
102
103 kvmap_dtlb_obp:
104         OBP_TRANS_LOOKUP(%g4, %g5, %g2, %g3, kvmap_dtlb_longpath)
105
106         TSB_LOCK_TAG(%g1, %g2, %g7)
107
108         TSB_WRITE(%g1, %g5, %g6)
109
110         ba,pt           %xcc, kvmap_dtlb_load
111          nop
112
113 kvmap_linear_early:
114         sethi           %hi(kern_linear_pte_xor), %g7
115         ldx             [%g7 + %lo(kern_linear_pte_xor)], %g2
116         ba,pt           %xcc, kvmap_dtlb_tsb4m_load
117          xor            %g2, %g4, %g5
118
119         .align          32
120 kvmap_dtlb_tsb4m_load:
121         TSB_LOCK_TAG(%g1, %g2, %g7)
122         TSB_WRITE(%g1, %g5, %g6)
123         ba,pt           %xcc, kvmap_dtlb_load
124          nop
125
126 kvmap_dtlb:
127         /* %g6: TAG TARGET */
128         mov             TLB_TAG_ACCESS, %g4
129         ldxa            [%g4] ASI_DMMU, %g4
130
131         /* sun4v_dtlb_miss branches here with the missing virtual
132          * address already loaded into %g4
133          */
134 kvmap_dtlb_4v:
135         brgez,pn        %g4, kvmap_dtlb_nonlinear
136          nop
137
138 #ifdef CONFIG_DEBUG_PAGEALLOC
139         /* Index through the base page size TSB even for linear
140          * mappings when using page allocation debugging.
141          */
142         KERN_TSB_LOOKUP_TL1(%g4, %g6, %g5, %g1, %g2, %g3, kvmap_dtlb_load)
143 #else
144         /* Correct TAG_TARGET is already in %g6, check 4mb TSB.  */
145         KERN_TSB4M_LOOKUP_TL1(%g6, %g5, %g1, %g2, %g3, kvmap_dtlb_load)
146 #endif
147         /* Linear mapping TSB lookup failed.  Fallthrough to kernel
148          * page table based lookup.
149          */
150         .globl          kvmap_linear_patch
151 kvmap_linear_patch:
152         ba,a,pt         %xcc, kvmap_linear_early
153
154 kvmap_dtlb_vmalloc_addr:
155         KERN_PGTABLE_WALK(%g4, %g5, %g2, kvmap_dtlb_longpath)
156
157         TSB_LOCK_TAG(%g1, %g2, %g7)
158         TSB_WRITE(%g1, %g5, %g6)
159
160         /* fallthrough to TLB load */
161
162 kvmap_dtlb_load:
163
164 661:    stxa            %g5, [%g0] ASI_DTLB_DATA_IN     ! Reload TLB
165         retry
166         .section        .sun4v_2insn_patch, "ax"
167         .word           661b
168         nop
169         nop
170         .previous
171
172         /* For sun4v the ASI_DTLB_DATA_IN store and the retry
173          * instruction get nop'd out and we get here to branch
174          * to the sun4v tlb load code.  The registers are setup
175          * as follows:
176          *
177          * %g4: vaddr
178          * %g5: PTE
179          * %g6: TAG
180          *
181          * The sun4v TLB load wants the PTE in %g3 so we fix that
182          * up here.
183          */
184         ba,pt           %xcc, sun4v_dtlb_load
185          mov            %g5, %g3
186
187 #ifdef CONFIG_SPARSEMEM_VMEMMAP
188 kvmap_vmemmap:
189         KERN_PGTABLE_WALK(%g4, %g5, %g2, kvmap_dtlb_longpath)
190         ba,a,pt         %xcc, kvmap_dtlb_load
191 #endif
192
193 kvmap_dtlb_nonlinear:
194         /* Catch kernel NULL pointer derefs.  */
195         sethi           %hi(PAGE_SIZE), %g5
196         cmp             %g4, %g5
197         bleu,pn         %xcc, kvmap_dtlb_longpath
198          nop
199
200 #ifdef CONFIG_SPARSEMEM_VMEMMAP
201         /* Do not use the TSB for vmemmap.  */
202         sethi           %hi(VMEMMAP_BASE), %g5
203         ldx             [%g5 + %lo(VMEMMAP_BASE)], %g5
204         cmp             %g4,%g5
205         bgeu,pn         %xcc, kvmap_vmemmap
206          nop
207 #endif
208
209         KERN_TSB_LOOKUP_TL1(%g4, %g6, %g5, %g1, %g2, %g3, kvmap_dtlb_load)
210
211 kvmap_dtlb_tsbmiss:
212         sethi           %hi(MODULES_VADDR), %g5
213         cmp             %g4, %g5
214         blu,pn          %xcc, kvmap_dtlb_longpath
215          sethi          %hi(VMALLOC_END), %g5
216         ldx             [%g5 + %lo(VMALLOC_END)], %g5
217         cmp             %g4, %g5
218         bgeu,pn         %xcc, kvmap_dtlb_longpath
219          nop
220
221 kvmap_check_obp:
222         sethi           %hi(LOW_OBP_ADDRESS), %g5
223         cmp             %g4, %g5
224         blu,pn          %xcc, kvmap_dtlb_vmalloc_addr
225          mov            0x1, %g5
226         sllx            %g5, 32, %g5
227         cmp             %g4, %g5
228         blu,pn          %xcc, kvmap_dtlb_obp
229          nop
230         ba,pt           %xcc, kvmap_dtlb_vmalloc_addr
231          nop
232
233 kvmap_dtlb_longpath:
234
235 661:    rdpr    %pstate, %g5
236         wrpr    %g5, PSTATE_AG | PSTATE_MG, %pstate
237         .section .sun4v_2insn_patch, "ax"
238         .word   661b
239         SET_GL(1)
240         ldxa            [%g0] ASI_SCRATCHPAD, %g5
241         .previous
242
243         rdpr    %tl, %g3
244         cmp     %g3, 1
245
246 661:    mov     TLB_TAG_ACCESS, %g4
247         ldxa    [%g4] ASI_DMMU, %g5
248         .section .sun4v_2insn_patch, "ax"
249         .word   661b
250         ldx     [%g5 + HV_FAULT_D_ADDR_OFFSET], %g5
251         nop
252         .previous
253
254         be,pt   %xcc, sparc64_realfault_common
255          mov    FAULT_CODE_DTLB, %g4
256         ba,pt   %xcc, winfix_trampoline
257          nop