To be able to return to the BootROM FEL USB debug code, we must restore
the core's state as accurately as possible after the SPL has been run.
Since the BootROM runs in AArch32, but the SPL uses AArch64, this requires
a core reset, which clears the core's state.
So far we were saving and restoring the required registers like SCTLR
and VBAR, but could ignore the interrupt controller's state (GICC), since
that lives in MMIO registers, unaffected by a core reset.
Newer Allwinner SoCs now feature a GICv3 interrupt controller, which keeps
some GIC state in architected system registers, and those are cleared
when we switch back to AArch32.

To enable FEL operation on the Allwinner A523 SoC,
Add AArch32 assembly code to save and restore the ICC_PMR and ICC_IGRPEN1
system registers. The other GICv3 sysregs are either not relevant for the
BROM operation, or haven't been changed from their reset defaults by the
BROM anyway.

This enables FEL operation on the Allwinner A523 family of SoCs.

Signed-off-by: Andre Przywara <andre.przyw...@arm.com>
---
 arch/arm/cpu/armv8/fel_utils.S          |  7 +++++++
 arch/arm/include/asm/arch-sunxi/boot0.h | 10 +++++++++-
 arch/arm/mach-sunxi/board.c             |  2 ++
 arch/arm/mach-sunxi/rmr_switch.S        | 10 ++++++++++
 4 files changed, 28 insertions(+), 1 deletion(-)

diff --git a/arch/arm/cpu/armv8/fel_utils.S b/arch/arm/cpu/armv8/fel_utils.S
index f7707acdf1a..f9d0c9e1d0a 100644
--- a/arch/arm/cpu/armv8/fel_utils.S
+++ b/arch/arm/cpu/armv8/fel_utils.S
@@ -79,5 +79,12 @@ back_in_32:
        .word   0xe590100c      // ldr  r1, [r0, #12]
        .word   0xee011f10      // mcr  15, 0, r1, cr1, cr0, {0}  ; SCTLR
        .word   0xf57ff06f      // isb
+#ifdef CONFIG_MACH_SUN55I_A523
+       .word   0xe5901014      // ldr  r1, [r0, #20]
+       .word   0xee041f16      // mcr  15, 0, r1, cr4, cr6, {0}; ICC_PMR
+       .word   0xe5901018      // ldr  r1, [r0, #24]
+       .word   0xee0c1ffc      // mcr  15, 0, r1, cr12, cr12, {7}; ICC_IGRPEN1
+#endif
+
        .word   0xe12fff1e      // bx   lr              ; return to FEL
 ENDPROC(return_to_fel)
diff --git a/arch/arm/include/asm/arch-sunxi/boot0.h 
b/arch/arm/include/asm/arch-sunxi/boot0.h
index 24c81391d58..9baedc2e9af 100644
--- a/arch/arm/include/asm/arch-sunxi/boot0.h
+++ b/arch/arm/include/asm/arch-sunxi/boot0.h
@@ -30,7 +30,15 @@
        .word   0xe580e00c      // str     lr, [r0, #12]
        .word   0xee1cef10      // mrc     15, 0, lr, cr12, cr0, {0}
        .word   0xe580e010      // str     lr, [r0, #16]
-
+#ifdef CONFIG_MACH_SUN55I_A523
+       .word   0xee1cefbc      // mrc     15, 0, lr, cr12, cr12, {5}
+       .word   0xe31e0001      // tst     lr, #1
+       .word   0x0a000003      // beq     cc <start32+0x48>
+       .word   0xee14ef16      // mrc     15, 0, lr, cr4, cr6, {0}
+       .word   0xe580e014      // str     lr, [r0, #20]
+       .word   0xee1ceffc      // mrc     15, 0, lr, cr12, cr12, {7}
+       .word   0xe580e018      // str     lr, [r0, #24]
+#endif
        .word   0xe59f1034      // ldr     r1, [pc, #52] ; RVBAR_ADDRESS
        .word   0xe59f0034      // ldr     r0, [pc, #52] ; SUNXI_SRAMC_BASE
        .word   0xe5900024      // ldr     r0, [r0, #36] ; SRAM_VER_REG
diff --git a/arch/arm/mach-sunxi/board.c b/arch/arm/mach-sunxi/board.c
index 89aea61e8e8..195c40d00c6 100644
--- a/arch/arm/mach-sunxi/board.c
+++ b/arch/arm/mach-sunxi/board.c
@@ -35,6 +35,8 @@ struct fel_stash {
        uint32_t cpsr;
        uint32_t sctlr;
        uint32_t vbar;
+       uint32_t icc_pmr;
+       uint32_t icc_igrpen1;
 };
 
 struct fel_stash fel_stash __section(".data");
diff --git a/arch/arm/mach-sunxi/rmr_switch.S b/arch/arm/mach-sunxi/rmr_switch.S
index 422007c985b..de284c16b0b 100644
--- a/arch/arm/mach-sunxi/rmr_switch.S
+++ b/arch/arm/mach-sunxi/rmr_switch.S
@@ -53,6 +53,16 @@ start32:
        str     lr, [r0, #12]
        mrc     p15, 0, lr, cr12, cr0, 0        // VBAR
        str     lr, [r0, #16]
+//#ifdef CONFIG_MACH_SUN55I_A523
+       mrc     p15, 0, lr, cr12, cr12, 5       // ICC_SRE
+       tst     lr, #1
+       beq     1f
+       mrc     p15, 0, lr, c4, c6, 0           // ICC_PMR
+       str     lr, [r0, #20]
+       mrc     p15, 0, lr, c12, c12, 7         // ICC_IGRPEN1
+       str     lr, [r0, #24]
+1:
+//#endif
 
        ldr     r1, =CONFIG_SUNXI_RVBAR_ADDRESS
        ldr     r0, =SUNXI_SRAMC_BASE
-- 
2.46.3

Reply via email to