1 /*
2 * SPDX-License-Identifier: Apache-2.0
3 *
4 * Copyright (c) 2021 ASPEED Technology Inc.
5 */
6
7 #include <zephyr/init.h>
8 #include <zephyr/kernel.h>
9 #include <stdint.h>
10 #include <string.h>
11 #include <zephyr/linker/linker-defs.h>
12 #include <zephyr/device.h>
13 #include <zephyr/cache.h>
14 #include <soc.h>
15
16 extern char __bss_nc_start__[];
17 extern char __bss_nc_end__[];
18
19 /* SCU registers */
20 #define JTAG_PINMUX_REG 0x41c
21
22 /* ASPEED System reset contrl/status register */
23 #define SYS_WDT4_SW_RESET BIT(31)
24 #define SYS_WDT4_ARM_RESET BIT(30)
25 #define SYS_WDT4_FULL_RESET BIT(29)
26 #define SYS_WDT4_SOC_RESET BIT(28)
27 #define SYS_WDT3_SW_RESET BIT(27)
28 #define SYS_WDT3_ARM_RESET BIT(26)
29 #define SYS_WDT3_FULL_RESET BIT(25)
30 #define SYS_WDT3_SOC_RESET BIT(24)
31 #define SYS_WDT2_SW_RESET BIT(23)
32 #define SYS_WDT2_ARM_RESET BIT(22)
33 #define SYS_WDT2_FULL_RESET BIT(21)
34 #define SYS_WDT2_SOC_RESET BIT(20)
35 #define SYS_WDT1_SW_RESET BIT(19)
36 #define SYS_WDT1_ARM_RESET BIT(18)
37 #define SYS_WDT1_FULL_RESET BIT(17)
38 #define SYS_WDT1_SOC_RESET BIT(16)
39
40 #define SYS_FLASH_ABR_RESET BIT(2)
41 #define SYS_EXT_RESET BIT(1)
42 #define SYS_PWR_RESET_FLAG BIT(0)
43
44 #define BIT_WDT_SOC(x) SYS_WDT ## x ## _SOC_RESET
45 #define BIT_WDT_FULL(x) SYS_WDT ## x ## _FULL_RESET
46 #define BIT_WDT_ARM(x) SYS_WDT ## x ## _ARM_RESET
47 #define BIT_WDT_SW(x) SYS_WDT ## x ## _SW_RESET
48
49 #define HANDLE_WDTx_RESET(x, event_log, event_log_reg) \
50 if (event_log & (BIT_WDT_SOC(x) | BIT_WDT_FULL(x) | BIT_WDT_ARM(x) | BIT_WDT_SW(x))) { \
51 printk("RST: WDT%d ", x); \
52 if (event_log & BIT_WDT_SOC(x)) { \
53 printk("SOC "); \
54 sys_write32(BIT_WDT_SOC(x), event_log_reg); \
55 } \
56 if (event_log & BIT_WDT_FULL(x)) { \
57 printk("FULL "); \
58 sys_write32(BIT_WDT_FULL(x), event_log_reg); \
59 } \
60 if (event_log & BIT_WDT_ARM(x)) { \
61 printk("ARM "); \
62 sys_write32(BIT_WDT_ARM(x), event_log_reg); \
63 } \
64 if (event_log & BIT_WDT_SW(x)) { \
65 printk("SW "); \
66 sys_write32(BIT_WDT_SW(x), event_log_reg); \
67 } \
68 printk("\n"); \
69 } \
70 (void)(x)
71
72 /* secure boot header : provide image size to bootROM for SPI boot */
73 struct sb_header {
74 uint32_t key_location;
75 uint32_t enc_img_addr;
76 uint32_t img_size;
77 uint32_t sign_location;
78 uint32_t header_rev[2];
79 uint32_t patch_location;
80 uint32_t checksum;
81 };
82
83 struct sb_header sbh __attribute((used, section(".sboot"))) = {
84 .img_size = (uint32_t)&__bss_start,
85 };
86
soc_reset_hook(void)87 void soc_reset_hook(void)
88 {
89 uint32_t jtag_pinmux;
90 uint32_t base = DT_REG_ADDR(DT_NODELABEL(syscon));
91
92 /* enable JTAG pins */
93 jtag_pinmux = sys_read32(base + JTAG_PINMUX_REG);
94 jtag_pinmux |= (0x1f << 25);
95 sys_write32(jtag_pinmux, base + JTAG_PINMUX_REG);
96
97 /* clear non-cached .bss */
98 if (CONFIG_SRAM_NC_SIZE > 0) {
99 (void)memset(__bss_nc_start__, 0, __bss_nc_end__ - __bss_nc_start__);
100 }
101
102 sys_cache_instr_enable();
103 }
104
aspeed_print_abr_wdt_mode(void)105 void aspeed_print_abr_wdt_mode(void)
106 {
107 /* ABR enable */
108 if (sys_read32(HW_STRAP2_SCU510) & BIT(11)) {
109 printk("FMC ABR: Enable");
110 if (sys_read32(HW_STRAP2_SCU510) & BIT(12)) {
111 printk(", Single flash");
112 } else {
113 printk(", Dual flashes");
114 }
115
116 printk(", Source: %s (%d)",
117 (sys_read32(ASPEED_FMC_WDT2_CTRL) & BIT(4)) ? "Alternate" : "Primary",
118 (sys_read32(HW_STRAP1_SCU500) & BIT(3)) ? 1 : 0);
119
120 if (sys_read32(HW_STRAP2_SCU510) & GENMASK(15, 13)) {
121 printk(", bspi sz: %ldMB",
122 BIT((sys_read32(HW_STRAP2_SCU510) >> 13) & 0x7) / 2);
123 }
124 printk("\n");
125 }
126 }
127
aspeed_print_sysrst_info(void)128 void aspeed_print_sysrst_info(void)
129 {
130 uint32_t rest1 = sys_read32(SYS_RESET_LOG_REG1);
131 uint32_t rest2 = sys_read32(SYS_RESET_LOG_REG2);
132
133 if (rest1 & SYS_PWR_RESET_FLAG) {
134 printk("RST: Power On\n");
135 sys_write32(rest1, SYS_RESET_LOG_REG1);
136 } else {
137 HANDLE_WDTx_RESET(4, rest1, SYS_RESET_LOG_REG1);
138 HANDLE_WDTx_RESET(3, rest1, SYS_RESET_LOG_REG1);
139 HANDLE_WDTx_RESET(2, rest1, SYS_RESET_LOG_REG1);
140 HANDLE_WDTx_RESET(1, rest1, SYS_RESET_LOG_REG1);
141
142 if (rest1 & SYS_FLASH_ABR_RESET) {
143 printk("RST: SYS_FLASH_ABR_RESET\n");
144 sys_write32(SYS_FLASH_ABR_RESET, SYS_RESET_LOG_REG1);
145 }
146
147 if (rest1 & SYS_EXT_RESET) {
148 printk("RST: External\n");
149 sys_write32(SYS_EXT_RESET, SYS_RESET_LOG_REG1);
150 }
151 }
152
153 ARG_UNUSED(rest2);
154
155 aspeed_print_abr_wdt_mode();
156 }
157