Linux 4.2.2
[linux/fpc-iii.git] / arch / mips / kernel / cps-vec.S
blob9f71c06aebf6313c4a0b2bac9b1bbdc6228cee31
1 /*
2  * Copyright (C) 2013 Imagination Technologies
3  * Author: Paul Burton <paul.burton@imgtec.com>
4  *
5  * This program is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License as published by the
7  * Free Software Foundation;  either version 2 of the  License, or (at your
8  * option) any later version.
9  */
11 #include <asm/addrspace.h>
12 #include <asm/asm.h>
13 #include <asm/asm-offsets.h>
14 #include <asm/asmmacro.h>
15 #include <asm/cacheops.h>
16 #include <asm/eva.h>
17 #include <asm/mipsregs.h>
18 #include <asm/mipsmtregs.h>
19 #include <asm/pm.h>
21 #define GCR_CL_COHERENCE_OFS    0x2008
22 #define GCR_CL_ID_OFS           0x2028
24 .extern mips_cm_base
26 .set noreorder
28         /*
29          * Set dest to non-zero if the core supports the MT ASE, else zero. If
30          * MT is not supported then branch to nomt.
31          */
32         .macro  has_mt  dest, nomt
33         mfc0    \dest, CP0_CONFIG
34         bgez    \dest, \nomt
35          mfc0   \dest, CP0_CONFIG, 1
36         bgez    \dest, \nomt
37          mfc0   \dest, CP0_CONFIG, 2
38         bgez    \dest, \nomt
39          mfc0   \dest, CP0_CONFIG, 3
40         andi    \dest, \dest, MIPS_CONF3_MT
41         beqz    \dest, \nomt
42         .endm
44 .section .text.cps-vec
45 .balign 0x1000
47 LEAF(mips_cps_core_entry)
48         /*
49          * These first 12 bytes will be patched by cps_smp_setup to load the
50          * base address of the CM GCRs into register v1 and the CCA to use into
51          * register s0.
52          */
53         .quad   0
54         .word   0
56         /* Check whether we're here due to an NMI */
57         mfc0    k0, CP0_STATUS
58         and     k0, k0, ST0_NMI
59         beqz    k0, not_nmi
60          nop
62         /* This is an NMI */
63         PTR_LA  k0, nmi_handler
64         jr      k0
65          nop
67 not_nmi:
68         /* Setup Cause */
69         li      t0, CAUSEF_IV
70         mtc0    t0, CP0_CAUSE
72         /* Setup Status */
73         li      t0, ST0_CU1 | ST0_CU0
74         mtc0    t0, CP0_STATUS
76         /*
77          * Clear the bits used to index the caches. Note that the architecture
78          * dictates that writing to any of TagLo or TagHi selects 0 or 2 should
79          * be valid for all MIPS32 CPUs, even those for which said writes are
80          * unnecessary.
81          */
82         mtc0    zero, CP0_TAGLO, 0
83         mtc0    zero, CP0_TAGHI, 0
84         mtc0    zero, CP0_TAGLO, 2
85         mtc0    zero, CP0_TAGHI, 2
86         ehb
88         /* Primary cache configuration is indicated by Config1 */
89         mfc0    v0, CP0_CONFIG, 1
91         /* Detect I-cache line size */
92         _EXT    t0, v0, MIPS_CONF1_IL_SHF, MIPS_CONF1_IL_SZ
93         beqz    t0, icache_done
94          li     t1, 2
95         sllv    t0, t1, t0
97         /* Detect I-cache size */
98         _EXT    t1, v0, MIPS_CONF1_IS_SHF, MIPS_CONF1_IS_SZ
99         xori    t2, t1, 0x7
100         beqz    t2, 1f
101          li     t3, 32
102         addiu   t1, t1, 1
103         sllv    t1, t3, t1
104 1:      /* At this point t1 == I-cache sets per way */
105         _EXT    t2, v0, MIPS_CONF1_IA_SHF, MIPS_CONF1_IA_SZ
106         addiu   t2, t2, 1
107         mul     t1, t1, t0
108         mul     t1, t1, t2
110         li      a0, CKSEG0
111         PTR_ADD a1, a0, t1
112 1:      cache   Index_Store_Tag_I, 0(a0)
113         PTR_ADD a0, a0, t0
114         bne     a0, a1, 1b
115          nop
116 icache_done:
118         /* Detect D-cache line size */
119         _EXT    t0, v0, MIPS_CONF1_DL_SHF, MIPS_CONF1_DL_SZ
120         beqz    t0, dcache_done
121          li     t1, 2
122         sllv    t0, t1, t0
124         /* Detect D-cache size */
125         _EXT    t1, v0, MIPS_CONF1_DS_SHF, MIPS_CONF1_DS_SZ
126         xori    t2, t1, 0x7
127         beqz    t2, 1f
128          li     t3, 32
129         addiu   t1, t1, 1
130         sllv    t1, t3, t1
131 1:      /* At this point t1 == D-cache sets per way */
132         _EXT    t2, v0, MIPS_CONF1_DA_SHF, MIPS_CONF1_DA_SZ
133         addiu   t2, t2, 1
134         mul     t1, t1, t0
135         mul     t1, t1, t2
137         li      a0, CKSEG0
138         PTR_ADDU a1, a0, t1
139         PTR_SUBU a1, a1, t0
140 1:      cache   Index_Store_Tag_D, 0(a0)
141         bne     a0, a1, 1b
142          PTR_ADD a0, a0, t0
143 dcache_done:
145         /* Set Kseg0 CCA to that in s0 */
146         mfc0    t0, CP0_CONFIG
147         ori     t0, 0x7
148         xori    t0, 0x7
149         or      t0, t0, s0
150         mtc0    t0, CP0_CONFIG
151         ehb
153         /* Enter the coherent domain */
154         li      t0, 0xff
155         sw      t0, GCR_CL_COHERENCE_OFS(v1)
156         ehb
158         /* Jump to kseg0 */
159         PTR_LA  t0, 1f
160         jr      t0
161          nop
163         /*
164          * We're up, cached & coherent. Perform any further required core-level
165          * initialisation.
166          */
167 1:      jal     mips_cps_core_init
168          nop
170         /* Do any EVA initialization if necessary */
171         eva_init
173         /*
174          * Boot any other VPEs within this core that should be online, and
175          * deactivate this VPE if it should be offline.
176          */
177         jal     mips_cps_boot_vpes
178          nop
180         /* Off we go! */
181         PTR_L   t1, VPEBOOTCFG_PC(v0)
182         PTR_L   gp, VPEBOOTCFG_GP(v0)
183         PTR_L   sp, VPEBOOTCFG_SP(v0)
184         jr      t1
185          nop
186         END(mips_cps_core_entry)
188 .org 0x200
189 LEAF(excep_tlbfill)
190         b       .
191          nop
192         END(excep_tlbfill)
194 .org 0x280
195 LEAF(excep_xtlbfill)
196         b       .
197          nop
198         END(excep_xtlbfill)
200 .org 0x300
201 LEAF(excep_cache)
202         b       .
203          nop
204         END(excep_cache)
206 .org 0x380
207 LEAF(excep_genex)
208         b       .
209          nop
210         END(excep_genex)
212 .org 0x400
213 LEAF(excep_intex)
214         b       .
215          nop
216         END(excep_intex)
218 .org 0x480
219 LEAF(excep_ejtag)
220         PTR_LA  k0, ejtag_debug_handler
221         jr      k0
222          nop
223         END(excep_ejtag)
225 LEAF(mips_cps_core_init)
226 #ifdef CONFIG_MIPS_MT
227         /* Check that the core implements the MT ASE */
228         has_mt  t0, 3f
229          nop
231         .set    push
232         .set    mips64r2
233         .set    mt
235         /* Only allow 1 TC per VPE to execute... */
236         dmt
238         /* ...and for the moment only 1 VPE */
239         dvpe
240         PTR_LA  t1, 1f
241         jr.hb   t1
242          nop
244         /* Enter VPE configuration state */
245 1:      mfc0    t0, CP0_MVPCONTROL
246         ori     t0, t0, MVPCONTROL_VPC
247         mtc0    t0, CP0_MVPCONTROL
249         /* Retrieve the number of VPEs within the core */
250         mfc0    t0, CP0_MVPCONF0
251         srl     t0, t0, MVPCONF0_PVPE_SHIFT
252         andi    t0, t0, (MVPCONF0_PVPE >> MVPCONF0_PVPE_SHIFT)
253         addiu   ta3, t0, 1
255         /* If there's only 1, we're done */
256         beqz    t0, 2f
257          nop
259         /* Loop through each VPE within this core */
260         li      ta1, 1
262 1:      /* Operate on the appropriate TC */
263         mtc0    ta1, CP0_VPECONTROL
264         ehb
266         /* Bind TC to VPE (1:1 TC:VPE mapping) */
267         mttc0   ta1, CP0_TCBIND
269         /* Set exclusive TC, non-active, master */
270         li      t0, VPECONF0_MVP
271         sll     t1, ta1, VPECONF0_XTC_SHIFT
272         or      t0, t0, t1
273         mttc0   t0, CP0_VPECONF0
275         /* Set TC non-active, non-allocatable */
276         mttc0   zero, CP0_TCSTATUS
278         /* Set TC halted */
279         li      t0, TCHALT_H
280         mttc0   t0, CP0_TCHALT
282         /* Next VPE */
283         addiu   ta1, ta1, 1
284         slt     t0, ta1, ta3
285         bnez    t0, 1b
286          nop
288         /* Leave VPE configuration state */
289 2:      mfc0    t0, CP0_MVPCONTROL
290         xori    t0, t0, MVPCONTROL_VPC
291         mtc0    t0, CP0_MVPCONTROL
293 3:      .set    pop
294 #endif
295         jr      ra
296          nop
297         END(mips_cps_core_init)
299 LEAF(mips_cps_boot_vpes)
300         /* Retrieve CM base address */
301         PTR_LA  t0, mips_cm_base
302         PTR_L   t0, 0(t0)
304         /* Calculate a pointer to this cores struct core_boot_config */
305         lw      t0, GCR_CL_ID_OFS(t0)
306         li      t1, COREBOOTCFG_SIZE
307         mul     t0, t0, t1
308         PTR_LA  t1, mips_cps_core_bootcfg
309         PTR_L   t1, 0(t1)
310         PTR_ADDU t0, t0, t1
312         /* Calculate this VPEs ID. If the core doesn't support MT use 0 */
313         has_mt  ta2, 1f
314          li     t9, 0
316         /* Find the number of VPEs present in the core */
317         mfc0    t1, CP0_MVPCONF0
318         srl     t1, t1, MVPCONF0_PVPE_SHIFT
319         andi    t1, t1, MVPCONF0_PVPE >> MVPCONF0_PVPE_SHIFT
320         addiu   t1, t1, 1
322         /* Calculate a mask for the VPE ID from EBase.CPUNum */
323         clz     t1, t1
324         li      t2, 31
325         subu    t1, t2, t1
326         li      t2, 1
327         sll     t1, t2, t1
328         addiu   t1, t1, -1
330         /* Retrieve the VPE ID from EBase.CPUNum */
331         mfc0    t9, $15, 1
332         and     t9, t9, t1
334 1:      /* Calculate a pointer to this VPEs struct vpe_boot_config */
335         li      t1, VPEBOOTCFG_SIZE
336         mul     v0, t9, t1
337         PTR_L   ta3, COREBOOTCFG_VPECONFIG(t0)
338         PTR_ADDU v0, v0, ta3
340 #ifdef CONFIG_MIPS_MT
342         /* If the core doesn't support MT then return */
343         bnez    ta2, 1f
344          nop
345         jr      ra
346          nop
348         .set    push
349         .set    mips64r2
350         .set    mt
352 1:      /* Enter VPE configuration state */
353         dvpe
354         PTR_LA  t1, 1f
355         jr.hb   t1
356          nop
357 1:      mfc0    t1, CP0_MVPCONTROL
358         ori     t1, t1, MVPCONTROL_VPC
359         mtc0    t1, CP0_MVPCONTROL
360         ehb
362         /* Loop through each VPE */
363         PTR_L   ta2, COREBOOTCFG_VPEMASK(t0)
364         move    t8, ta2
365         li      ta1, 0
367         /* Check whether the VPE should be running. If not, skip it */
368 1:      andi    t0, ta2, 1
369         beqz    t0, 2f
370          nop
372         /* Operate on the appropriate TC */
373         mfc0    t0, CP0_VPECONTROL
374         ori     t0, t0, VPECONTROL_TARGTC
375         xori    t0, t0, VPECONTROL_TARGTC
376         or      t0, t0, ta1
377         mtc0    t0, CP0_VPECONTROL
378         ehb
380         /* Skip the VPE if its TC is not halted */
381         mftc0   t0, CP0_TCHALT
382         beqz    t0, 2f
383          nop
385         /* Calculate a pointer to the VPEs struct vpe_boot_config */
386         li      t0, VPEBOOTCFG_SIZE
387         mul     t0, t0, ta1
388         addu    t0, t0, ta3
390         /* Set the TC restart PC */
391         lw      t1, VPEBOOTCFG_PC(t0)
392         mttc0   t1, CP0_TCRESTART
394         /* Set the TC stack pointer */
395         lw      t1, VPEBOOTCFG_SP(t0)
396         mttgpr  t1, sp
398         /* Set the TC global pointer */
399         lw      t1, VPEBOOTCFG_GP(t0)
400         mttgpr  t1, gp
402         /* Copy config from this VPE */
403         mfc0    t0, CP0_CONFIG
404         mttc0   t0, CP0_CONFIG
406         /* Ensure no software interrupts are pending */
407         mttc0   zero, CP0_CAUSE
408         mttc0   zero, CP0_STATUS
410         /* Set TC active, not interrupt exempt */
411         mftc0   t0, CP0_TCSTATUS
412         li      t1, ~TCSTATUS_IXMT
413         and     t0, t0, t1
414         ori     t0, t0, TCSTATUS_A
415         mttc0   t0, CP0_TCSTATUS
417         /* Clear the TC halt bit */
418         mttc0   zero, CP0_TCHALT
420         /* Set VPE active */
421         mftc0   t0, CP0_VPECONF0
422         ori     t0, t0, VPECONF0_VPA
423         mttc0   t0, CP0_VPECONF0
425         /* Next VPE */
426 2:      srl     ta2, ta2, 1
427         addiu   ta1, ta1, 1
428         bnez    ta2, 1b
429          nop
431         /* Leave VPE configuration state */
432         mfc0    t1, CP0_MVPCONTROL
433         xori    t1, t1, MVPCONTROL_VPC
434         mtc0    t1, CP0_MVPCONTROL
435         ehb
436         evpe
438         /* Check whether this VPE is meant to be running */
439         li      t0, 1
440         sll     t0, t0, t9
441         and     t0, t0, t8
442         bnez    t0, 2f
443          nop
445         /* This VPE should be offline, halt the TC */
446         li      t0, TCHALT_H
447         mtc0    t0, CP0_TCHALT
448         PTR_LA  t0, 1f
449 1:      jr.hb   t0
450          nop
452 2:      .set    pop
454 #endif /* CONFIG_MIPS_MT */
456         /* Return */
457         jr      ra
458          nop
459         END(mips_cps_boot_vpes)
461 #if defined(CONFIG_MIPS_CPS_PM) && defined(CONFIG_CPU_PM)
463         /* Calculate a pointer to this CPUs struct mips_static_suspend_state */
464         .macro  psstate dest
465         .set    push
466         .set    noat
467         lw      $1, TI_CPU(gp)
468         sll     $1, $1, LONGLOG
469         PTR_LA  \dest, __per_cpu_offset
470         addu    $1, $1, \dest
471         lw      $1, 0($1)
472         PTR_LA  \dest, cps_cpu_state
473         addu    \dest, \dest, $1
474         .set    pop
475         .endm
477 LEAF(mips_cps_pm_save)
478         /* Save CPU state */
479         SUSPEND_SAVE_REGS
480         psstate t1
481         SUSPEND_SAVE_STATIC
482         jr      v0
483          nop
484         END(mips_cps_pm_save)
486 LEAF(mips_cps_pm_restore)
487         /* Restore CPU state */
488         psstate t1
489         RESUME_RESTORE_STATIC
490         RESUME_RESTORE_REGS_RETURN
491         END(mips_cps_pm_restore)
493 #endif /* CONFIG_MIPS_CPS_PM && CONFIG_CPU_PM */