arm64 patch: gicv3 support
authorDavid Feng <fenghua@phytium.com.cn>
Fri, 14 Mar 2014 06:26:27 +0000 (14:26 +0800)
committerAlbert ARIBAUD <albert.u.boot@aribaud.net>
Mon, 7 Apr 2014 22:15:12 +0000 (00:15 +0200)
This patch add gicv3 support to uboot armv8 platform.

Changes for v2:
  - rename arm/cpu/armv8/gic.S with arm/lib/gic_64.S
  - move smp_kick_all_cpus() from gic.S to start.S, it would be
    implementation dependent.
  - Each core initialize it's own ReDistributor instead of master
    initializeing all ReDistributors. This is advised by arnab.basu
    <arnab.basu@freescale.com>.

Signed-off-by: David Feng <fenghua@phytium.com.cn>
arch/arm/cpu/armv8/Makefile
arch/arm/cpu/armv8/gic.S [deleted file]
arch/arm/cpu/armv8/start.S
arch/arm/include/asm/gic.h
arch/arm/lib/Makefile
arch/arm/lib/gic_64.S [new file with mode: 0644]
include/configs/vexpress_aemv8a.h

index b6eb6de5e3553a104635b08725576496d9cb5192..7d93f59428ea79e7c7f7ee825376e62b2b598061 100644 (file)
@@ -13,5 +13,4 @@ obj-y += cache_v8.o
 obj-y  += exceptions.o
 obj-y  += cache.o
 obj-y  += tlb.o
-obj-y  += gic.o
 obj-y  += transition.o
diff --git a/arch/arm/cpu/armv8/gic.S b/arch/arm/cpu/armv8/gic.S
deleted file mode 100644 (file)
index 599aa8f..0000000
+++ /dev/null
@@ -1,106 +0,0 @@
-/*
- * GIC Initialization Routines.
- *
- * (C) Copyright 2013
- * David Feng <fenghua@phytium.com.cn>
- *
- * SPDX-License-Identifier:    GPL-2.0+
- */
-
-#include <asm-offsets.h>
-#include <config.h>
-#include <linux/linkage.h>
-#include <asm/macro.h>
-#include <asm/gic.h>
-
-
-/*************************************************************************
- *
- * void gic_init(void) __attribute__((weak));
- *
- * Currently, this routine only initialize secure copy of GIC
- * with Security Extensions at EL3.
- *
- *************************************************************************/
-WEAK(gic_init)
-       branch_if_slave x0, 2f
-
-       /* Initialize Distributor and SPIs */
-       ldr     x1, =GICD_BASE
-       mov     w0, #0x3                /* EnableGrp0 | EnableGrp1 */
-       str     w0, [x1, GICD_CTLR]     /* Secure GICD_CTLR */
-       ldr     w0, [x1, GICD_TYPER]
-       and     w2, w0, #0x1f           /* ITLinesNumber */
-       cbz     w2, 2f                  /* No SPIs */
-       add     x1, x1, (GICD_IGROUPRn + 4)
-       mov     w0, #~0                 /* Config SPIs as Grp1 */
-1:     str     w0, [x1], #0x4
-       sub     w2, w2, #0x1
-       cbnz    w2, 1b
-
-       /* Initialize SGIs and PPIs */
-2:     ldr     x1, =GICD_BASE
-       mov     w0, #~0                 /* Config SGIs and PPIs as Grp1 */
-       str     w0, [x1, GICD_IGROUPRn] /* GICD_IGROUPR0 */
-       mov     w0, #0x1                /* Enable SGI 0 */
-       str     w0, [x1, GICD_ISENABLERn]
-
-       /* Initialize Cpu Interface */
-       ldr     x1, =GICC_BASE
-       mov     w0, #0x1e7              /* Disable IRQ/FIQ Bypass & */
-                                       /* Enable Ack Group1 Interrupt & */
-                                       /* EnableGrp0 & EnableGrp1 */
-       str     w0, [x1, GICC_CTLR]     /* Secure GICC_CTLR */
-
-       mov     w0, #0x1 << 7           /* Non-Secure access to GICC_PMR */
-       str     w0, [x1, GICC_PMR]
-
-       ret
-ENDPROC(gic_init)
-
-
-/*************************************************************************
- *
- * void gic_send_sgi(u64 sgi) __attribute__((weak));
- *
- *************************************************************************/
-WEAK(gic_send_sgi)
-       ldr     x1, =GICD_BASE
-       mov     w2, #0x8000
-       movk    w2, #0x100, lsl #16
-       orr     w2, w2, w0
-       str     w2, [x1, GICD_SGIR]
-       ret
-ENDPROC(gic_send_sgi)
-
-
-/*************************************************************************
- *
- * void wait_for_wakeup(void) __attribute__((weak));
- *
- * Wait for SGI 0 from master.
- *
- *************************************************************************/
-WEAK(wait_for_wakeup)
-       ldr     x1, =GICC_BASE
-0:     wfi
-       ldr     w0, [x1, GICC_AIAR]
-       str     w0, [x1, GICC_AEOIR]
-       cbnz    w0, 0b
-       ret
-ENDPROC(wait_for_wakeup)
-
-
-/*************************************************************************
- *
- * void smp_kick_all_cpus(void) __attribute__((weak));
- *
- *************************************************************************/
-WEAK(smp_kick_all_cpus)
-       /* Kick secondary cpus up by SGI 0 interrupt */
-       mov     x0, xzr                 /* SGI 0 */
-       mov     x29, lr                 /* Save LR */
-       bl      gic_send_sgi
-       mov     lr, x29                 /* Restore LR */
-       ret
-ENDPROC(smp_kick_all_cpus)
index 4f95289b5ebb9808f0b1f43a1669a91637afbcca..33d3f3688ae83bc45f13a79e12c84a0848d20d05 100644 (file)
@@ -50,7 +50,10 @@ reset:
         */
        adr     x0, vectors
        switch_el x1, 3f, 2f, 1f
-3:     msr     vbar_el3, x0
+3:     mrs     x0, scr_el3
+       orr     x0, x0, #0xf                    /* SCR_EL3.NS|IRQ|FIQ|EA */
+       msr     scr_el3, x0
+       msr     vbar_el3, x0
        msr     cptr_el3, xzr                   /* Enable FP/SIMD */
        ldr     x0, =COUNTER_FREQUENCY
        msr     cntfrq_el0, x0                  /* Initialize CNTFRQ */
@@ -95,32 +98,61 @@ master_cpu:
 /*-----------------------------------------------------------------------*/
 
 WEAK(lowlevel_init)
-       /* Initialize GIC Secure Bank Status */
        mov     x29, lr                 /* Save LR */
-       bl      gic_init
 
-       branch_if_master x0, x1, 1f
+#if defined(CONFIG_GICV2) || defined(CONFIG_GICV3)
+       branch_if_slave x0, 1f
+       ldr     x0, =GICD_BASE
+       bl      gic_init_secure
+1:
+#if defined(CONFIG_GICV3)
+       ldr     x0, =GICR_BASE
+       bl      gic_init_secure_percpu
+#elif defined(CONFIG_GICV2)
+       ldr     x0, =GICD_BASE
+       ldr     x1, =GICC_BASE
+       bl      gic_init_secure_percpu
+#endif
+#endif
+
+       branch_if_master x0, x1, 2f
 
        /*
         * Slave should wait for master clearing spin table.
         * This sync prevent salves observing incorrect
         * value of spin table and jumping to wrong place.
         */
-       bl      wait_for_wakeup
+#if defined(CONFIG_GICV2) || defined(CONFIG_GICV3)
+#ifdef CONFIG_GICV2
+       ldr     x0, =GICC_BASE
+#endif
+       bl      gic_wait_for_interrupt
+#endif
 
        /*
-        * All processors will enter EL2 and optionally EL1.
+        * All slaves will enter EL2 and optionally EL1.
         */
        bl      armv8_switch_to_el2
 #ifdef CONFIG_ARMV8_SWITCH_TO_EL1
        bl      armv8_switch_to_el1
 #endif
 
-1:
+2:
        mov     lr, x29                 /* Restore LR */
        ret
 ENDPROC(lowlevel_init)
 
+WEAK(smp_kick_all_cpus)
+       /* Kick secondary cpus up by SGI 0 interrupt */
+       mov     x29, lr                 /* Save LR */
+#if defined(CONFIG_GICV2) || defined(CONFIG_GICV3)
+       ldr     x0, =GICD_BASE
+       bl      gic_kick_secondary_cpus
+#endif
+       mov     lr, x29                 /* Restore LR */
+       ret
+ENDPROC(smp_kick_all_cpus)
+
 /*-----------------------------------------------------------------------*/
 
 ENTRY(c_runtime_cpu_setup)
index ac2b2bfbed22f1f2dc810db290c6fcf00b845b4b..bd3a80cdf70add02b4a43ceb9db05833292fe539 100644 (file)
 #define GICC_IIDR              0x00fc
 #define GICC_DIR               0x1000
 
+/* ReDistributor Registers for Control and Physical LPIs */
+#define GICR_CTLR              0x0000
+#define GICR_IIDR              0x0004
+#define GICR_TYPER             0x0008
+#define GICR_STATUSR           0x0010
+#define GICR_WAKER             0x0014
+#define GICR_SETLPIR           0x0040
+#define GICR_CLRLPIR           0x0048
+#define GICR_SEIR              0x0068
+#define GICR_PROPBASER         0x0070
+#define GICR_PENDBASER         0x0078
+#define GICR_INVLPIR           0x00a0
+#define GICR_INVALLR           0x00b0
+#define GICR_SYNCR             0x00c0
+#define GICR_MOVLPIR           0x0100
+#define GICR_MOVALLR           0x0110
+
+/* ReDistributor Registers for SGIs and PPIs */
+#define GICR_IGROUPRn          0x0080
+#define GICR_ISENABLERn                0x0100
+#define GICR_ICENABLERn                0x0180
+#define GICR_ISPENDRn          0x0200
+#define GICR_ICPENDRn          0x0280
+#define GICR_ISACTIVERn                0x0300
+#define GICR_ICACTIVERn                0x0380
+#define GICR_IPRIORITYRn       0x0400
+#define GICR_ICFGR0            0x0c00
+#define GICR_ICFGR1            0x0c04
+#define GICR_IGROUPMODRn       0x0d00
+#define GICR_NSACRn            0x0e00
+
+/* Cpu Interface System Registers */
+#define ICC_IAR0_EL1           S3_0_C12_C8_0
+#define ICC_IAR1_EL1           S3_0_C12_C12_0
+#define ICC_EOIR0_EL1          S3_0_C12_C8_1
+#define ICC_EOIR1_EL1          S3_0_C12_C12_1
+#define ICC_HPPIR0_EL1         S3_0_C12_C8_2
+#define ICC_HPPIR1_EL1         S3_0_C12_C12_2
+#define ICC_BPR0_EL1           S3_0_C12_C8_3
+#define ICC_BPR1_EL1           S3_0_C12_C12_3
+#define ICC_DIR_EL1            S3_0_C12_C11_1
+#define ICC_PMR_EL1            S3_0_C4_C6_0
+#define ICC_RPR_EL1            S3_0_C12_C11_3
+#define ICC_CTLR_EL1           S3_0_C12_C12_4
+#define ICC_CTLR_EL3           S3_6_C12_C12_4
+#define ICC_SRE_EL1            S3_0_C12_C12_5
+#define ICC_SRE_EL2            S3_4_C12_C9_5
+#define ICC_SRE_EL3            S3_6_C12_C12_5
+#define ICC_IGRPEN0_EL1                S3_0_C12_C12_6
+#define ICC_IGRPEN1_EL1                S3_0_C12_C12_7
+#define ICC_IGRPEN1_EL3                S3_6_C12_C12_7
+#define ICC_SEIEN_EL1          S3_0_C12_C13_0
+#define ICC_SGI0R_EL1          S3_0_C12_C11_7
+#define ICC_SGI1R_EL1          S3_0_C12_C11_5
+#define ICC_ASGI1R_EL1         S3_0_C12_C11_6
+
 #endif /* __GIC_H__ */
index 9fc81cd012996ba6970ae1b541973ca9b05e39db..e035d6acc00b4b0c7397980a2d520b232fc45a80 100644 (file)
@@ -35,6 +35,7 @@ endif
 
 obj-y  += sections.o
 ifdef CONFIG_ARM64
+obj-y  += gic_64.o
 obj-y  += interrupts_64.o
 else
 obj-y  += interrupts.o
diff --git a/arch/arm/lib/gic_64.S b/arch/arm/lib/gic_64.S
new file mode 100644 (file)
index 0000000..d56396e
--- /dev/null
@@ -0,0 +1,194 @@
+/*
+ * GIC Initialization Routines.
+ *
+ * (C) Copyright 2013
+ * David Feng <fenghua@phytium.com.cn>
+ *
+ * SPDX-License-Identifier:    GPL-2.0+
+ */
+
+#include <asm-offsets.h>
+#include <config.h>
+#include <linux/linkage.h>
+#include <asm/macro.h>
+#include <asm/gic.h>
+
+
+/*************************************************************************
+ *
+ * void gic_init_secure(DistributorBase);
+ *
+ * Initialize secure copy of GIC at EL3.
+ *
+ *************************************************************************/
+ENTRY(gic_init_secure)
+       /*
+        * Initialize Distributor
+        * x0: Distributor Base
+        */
+#if defined(CONFIG_GICV3)
+       mov     w9, #0x37               /* EnableGrp0 | EnableGrp1NS */
+                                       /* EnableGrp1S | ARE_S | ARE_NS */
+       str     w9, [x0, GICD_CTLR]     /* Secure GICD_CTLR */
+       ldr     w9, [x0, GICD_TYPER]
+       and     w10, w9, #0x1f          /* ITLinesNumber */
+       cbz     w10, 1f                 /* No SPIs */
+       add     x11, x0, (GICD_IGROUPRn + 4)
+       add     x12, x0, (GICD_IGROUPMODRn + 4)
+       mov     w9, #~0
+0:     str     w9, [x11], #0x4
+       str     wzr, [x12], #0x4        /* Config SPIs as Group1NS */
+       sub     w10, w10, #0x1
+       cbnz    w10, 0b
+#elif defined(CONFIG_GICV2)
+       mov     w9, #0x3                /* EnableGrp0 | EnableGrp1 */
+       str     w9, [x0, GICD_CTLR]     /* Secure GICD_CTLR */
+       ldr     w9, [x0, GICD_TYPER]
+       and     w10, w9, #0x1f          /* ITLinesNumber */
+       cbz     w10, 1f                 /* No SPIs */
+       add     x11, x0, (GICD_IGROUPRn + 4)
+       mov     w9, #~0                 /* Config SPIs as Grp1 */
+0:     str     w9, [x11], #0x4
+       sub     w10, w10, #0x1
+       cbnz    w10, 0b
+#endif
+1:
+       ret
+ENDPROC(gic_init_secure)
+
+
+/*************************************************************************
+ * For Gicv2:
+ * void gic_init_secure_percpu(DistributorBase, CpuInterfaceBase);
+ * For Gicv3:
+ * void gic_init_secure_percpu(ReDistributorBase);
+ *
+ * Initialize secure copy of GIC at EL3.
+ *
+ *************************************************************************/
+ENTRY(gic_init_secure_percpu)
+#if defined(CONFIG_GICV3)
+       /*
+        * Initialize ReDistributor
+        * x0: ReDistributor Base
+        */
+       mrs     x10, mpidr_el1
+       lsr     x9, x10, #32
+       bfi     x10, x9, #24, #8        /* w10 is aff3:aff2:aff1:aff0 */
+       mov     x9, x0
+1:     ldr     x11, [x9, GICR_TYPER]
+       lsr     x11, x11, #32           /* w11 is aff3:aff2:aff1:aff0 */
+       cmp     w10, w11
+       b.eq    2f
+       add     x9, x9, #(2 << 16)
+       b       1b
+
+       /* x9: ReDistributor Base Address of Current CPU */
+2:     mov     w10, #~0x2
+       ldr     w11, [x9, GICR_WAKER]
+       and     w11, w11, w10           /* Clear ProcessorSleep */
+       str     w11, [x9, GICR_WAKER]
+       dsb     st
+       isb
+3:     ldr     w10, [x9, GICR_WAKER]
+       tbnz    w10, #2, 3b             /* Wait Children be Alive */
+
+       add     x10, x9, #(1 << 16)     /* SGI_Base */
+       mov     w11, #~0
+       str     w11, [x10, GICR_IGROUPRn]
+       str     wzr, [x10, GICR_IGROUPMODRn]    /* SGIs|PPIs Group1NS */
+       mov     w11, #0x1               /* Enable SGI 0 */
+       str     w11, [x10, GICR_ISENABLERn]
+
+       /* Initialize Cpu Interface */
+       mrs     x10, ICC_SRE_EL3
+       orr     x10, x10, #0xf          /* SRE & Disable IRQ/FIQ Bypass & */
+                                       /* Allow EL2 access to ICC_SRE_EL2 */
+       msr     ICC_SRE_EL3, x10
+       isb
+
+       mrs     x10, ICC_SRE_EL2
+       orr     x10, x10, #0xf          /* SRE & Disable IRQ/FIQ Bypass & */
+                                       /* Allow EL1 access to ICC_SRE_EL1 */
+       msr     ICC_SRE_EL2, x10
+       isb
+
+       mov     x10, #0x3               /* EnableGrp1NS | EnableGrp1S */
+       msr     ICC_IGRPEN1_EL3, x10
+       isb
+
+       msr     ICC_CTLR_EL3, xzr
+       isb
+
+       msr     ICC_CTLR_EL1, xzr       /* NonSecure ICC_CTLR_EL1 */
+       isb
+
+       mov     x10, #0x1 << 7          /* Non-Secure access to ICC_PMR_EL1 */
+       msr     ICC_PMR_EL1, x10
+       isb
+#elif defined(CONFIG_GICV2)
+       /*
+        * Initialize SGIs and PPIs
+        * x0: Distributor Base
+        * x1: Cpu Interface Base
+        */
+       mov     w9, #~0                 /* Config SGIs and PPIs as Grp1 */
+       str     w9, [x0, GICD_IGROUPRn] /* GICD_IGROUPR0 */
+       mov     w9, #0x1                /* Enable SGI 0 */
+       str     w9, [x0, GICD_ISENABLERn]
+
+       /* Initialize Cpu Interface */
+       mov     w9, #0x1e7              /* Disable IRQ/FIQ Bypass & */
+                                       /* Enable Ack Group1 Interrupt & */
+                                       /* EnableGrp0 & EnableGrp1 */
+       str     w9, [x1, GICC_CTLR]     /* Secure GICC_CTLR */
+
+       mov     w9, #0x1 << 7           /* Non-Secure access to GICC_PMR */
+       str     w9, [x1, GICC_PMR]
+#endif
+       ret
+ENDPROC(gic_init_secure_percpu)
+
+
+/*************************************************************************
+ * For Gicv2:
+ * void gic_kick_secondary_cpus(DistributorBase);
+ * For Gicv3:
+ * void gic_kick_secondary_cpus(void);
+ *
+ *************************************************************************/
+ENTRY(gic_kick_secondary_cpus)
+#if defined(CONFIG_GICV3)
+       mov     x9, #(1 << 40)
+       msr     ICC_ASGI1R_EL1, x9
+       isb
+#elif defined(CONFIG_GICV2)
+       mov     w9, #0x8000
+       movk    w9, #0x100, lsl #16
+       str     w9, [x0, GICD_SGIR]
+#endif
+       ret
+ENDPROC(gic_kick_secondary_cpus)
+
+
+/*************************************************************************
+ * For Gicv2:
+ * void gic_wait_for_interrupt(CpuInterfaceBase);
+ * For Gicv3:
+ * void gic_wait_for_interrupt(void);
+ *
+ * Wait for SGI 0 from master.
+ *
+ *************************************************************************/
+ENTRY(gic_wait_for_interrupt)
+0:     wfi
+#if defined(CONFIG_GICV3)
+       mrs     x9, ICC_IAR1_EL1
+       msr     ICC_EOIR1_EL1, x9
+#elif defined(CONFIG_GICV2)
+       ldr     w9, [x0, GICC_AIAR]
+       str     w9, [x0, GICC_AEOIR]
+#endif
+       cbnz    w9, 0b
+       ret
+ENDPROC(gic_wait_for_interrupt)
index e8517027e7df770a67a300becc7109756d5fc296..dff6adcc7cdff806fec915cd06d820b8f520c1c8 100644 (file)
@@ -12,6 +12,8 @@
 
 #define CONFIG_REMAKE_ELF
 
+#define CONFIG_GICV3
+
 /*#define CONFIG_ARMV8_SWITCH_TO_EL1*/
 
 /*#define CONFIG_SYS_GENERIC_BOARD*/
 #define COUNTER_FREQUENCY              (0x1800000)     /* 24MHz */
 
 /* Generic Interrupt Controller Definitions */
+#ifdef CONFIG_GICV3
+#define GICD_BASE                      (0x2f000000)
+#define GICR_BASE                      (0x2f100000)
+#else
 #define GICD_BASE                      (0x2C001000)
 #define GICC_BASE                      (0x2C002000)
+#endif
 
 #define CONFIG_SYS_MEMTEST_START       V2M_BASE
 #define CONFIG_SYS_MEMTEST_END         (V2M_BASE + 0x80000000)