1 /*
2  * Copyright (c) 2019 Microchip Technology Inc.
3  * Copyright (c) 2016 Intel Corporation.
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #include <zephyr/kernel.h>
9 #include <zephyr/sys/sys_io.h>
10 #include <zephyr/sys/__assert.h>
11 #include <zephyr/pm/pm.h>
12 #include <soc.h>
13 
14 /*
15  * CPU will spin up to DEEP_SLEEP_WAIT_SPIN_CLK_REQ times
16  * waiting for PCR CLK_REQ bits to clear except for the
17  * CPU bit itself. This is not necessary as the sleep hardware
18  * will wait for all CLK_REQ to clear once WFI has executed.
19  * Once all CLK_REQ signals are clear the hardware will transition
20  * to the low power state.
21  */
22 /* #define DEEP_SLEEP_WAIT_ON_CLK_REQ_ENABLE */
23 #define DEEP_SLEEP_WAIT_SPIN_CLK_REQ		1000
24 
25 
26 /*
27  * Some peripherals if enabled always assert their CLK_REQ bits.
28  * For example, any peripheral with a clock generator such as
29  * timers, counters, UART, etc. We save the enables for these
30  * peripherals, disable them, and restore the enabled state upon
31  * wake.
32  */
33 #define DEEP_SLEEP_PERIPH_SAVE_RESTORE
34 
35 
36 /*
37  * Light sleep: PLL remains on. Fastest wake latency.
38  */
soc_lite_sleep_enable(void)39 void soc_lite_sleep_enable(void)
40 {
41 	SCB->SCR &= ~(1ul << 2);
42 	PCR_REGS->SYS_SLP_CTRL = MCHP_PCR_SYS_SLP_LIGHT;
43 }
44 
45 /*
46  * Deep sleep: PLL is turned off. Wake is fast. PLL requires
47  * a minimum of 3ms to lock. During this time the main clock
48  * will be ramping up from ~16 to 24 MHz.
49  */
soc_deep_sleep_enable(void)50 void soc_deep_sleep_enable(void)
51 {
52 	SCB->SCR = (1ul << 2); /* Cortex-M4 SLEEPDEEP */
53 	PCR_REGS->SYS_SLP_CTRL = MCHP_PCR_SYS_SLP_HEAVY;
54 }
55 
56 /*
57  * Clear PCR Sleep control sleep all causing HW to de-assert all peripheral
58  * SLP_EN signals. HW will does this automatically only if it vectors to an
59  * ISR after wake. We are masking ISR's from running until we restore
60  * peripheral state therefore we force HW to de-assert the SLP_EN signals.
61  */
soc_deep_sleep_disable(void)62 void soc_deep_sleep_disable(void)
63 {
64 	PCR_REGS->SYS_SLP_CTRL = 0U;
65 	SCB->SCR &= ~(1ul << 2); /* disable Cortex-M4 SLEEPDEEP */
66 }
67 
68 
soc_deep_sleep_wait_clk_idle(void)69 void soc_deep_sleep_wait_clk_idle(void)
70 {
71 #ifdef DEEP_SLEEP_WAIT_ON_CLK_REQ_ENABLE
72 	uint32_t clkreq, cnt;
73 
74 	cnt = DEEP_SLEEP_WAIT_CLK_REQ;
75 	do {
76 		clkreq = PCR_REGS->CLK_REQ0 | PCR_REGS->CLK_REQ1
77 			 | PCR_REGS->CLK_REQ2 | PCR_REGS->CLK_REQ3
78 			 | PCR_REGS->CLK_REQ4;
79 	} while ((clkreq != (1ul << MCHP_PCR1_CPU_POS)) && (cnt-- != 0));
80 #endif
81 }
82 
83 
84 /*
85  * Allow peripherals connected to external masters to wake the PLL but not
86  * the EC. Once the peripheral has serviced the external master the PLL
87  * will be turned back off. For example, if the eSPI master requests eSPI
88  * configuration information or state of virtual wires the EC doesn't need
89  * to be involved. The hardware can power on the PLL long enough to service
90  * the request and then turn the PLL back off.  The SMBus and I2C peripherals
91  * in slave mode can also make use of this feature.
92  */
soc_deep_sleep_non_wake_en(void)93 void soc_deep_sleep_non_wake_en(void)
94 {
95 #ifdef CONFIG_ESPI_XEC
96 	GIRQ22_REGS->SRC = 0xfffffffful;
97 	GIRQ22_REGS->EN_SET = (1ul << 9);
98 #endif
99 }
100 
soc_deep_sleep_non_wake_dis(void)101 void soc_deep_sleep_non_wake_dis(void)
102 {
103 #ifdef CONFIG_ESPI_XEC
104 	GIRQ22_REGS->EN_CLR = 0xfffffffful;
105 	GIRQ22_REGS->SRC = 0xfffffffful;
106 #endif
107 }
108 
109 /* Variables used to save various HW state */
110 #ifdef DEEP_SLEEP_PERIPH_SAVE_RESTORE
111 
112 static uint32_t ecs[1];
113 
deep_sleep_save_ecs(void)114 static void deep_sleep_save_ecs(void)
115 {
116 	ecs[0] = ECS_REGS->ETM_CTRL;
117 	ECS_REGS->ETM_CTRL = 0;
118 }
119 
120 struct ds_timer_info {
121 	uintptr_t addr;
122 	uint32_t restore_mask;
123 };
124 
125 const struct ds_timer_info ds_timer_tbl[] = {
126 	{
127 		(uintptr_t)&B16TMR0_REGS->CTRL, 0
128 	},
129 	{
130 		(uintptr_t)&B16TMR1_REGS->CTRL, 0
131 	},
132 	{
133 		(uintptr_t)&B32TMR0_REGS->CTRL, 0
134 	},
135 	{
136 		(uintptr_t)&B32TMR1_REGS->CTRL, 0
137 	},
138 	{
139 		(uintptr_t)&CCT_REGS->CTRL,
140 		(MCHP_CCT_CTRL_COMP1_SET | MCHP_CCT_CTRL_COMP0_SET),
141 	},
142 };
143 #define NUM_DS_TIMER_ENTRIES \
144 	(sizeof(ds_timer_tbl) / sizeof(struct ds_timer_info))
145 
146 
147 static uint32_t timers[NUM_DS_TIMER_ENTRIES];
148 static uint8_t uart_activate[3];
149 
deep_sleep_save_uarts(void)150 static void deep_sleep_save_uarts(void)
151 {
152 	uart_activate[0] = UART0_REGS->ACTV;
153 	if (uart_activate[0]) {
154 		while ((UART0_REGS->LSR & MCHP_UART_LSR_TEMT) == 0) {
155 		}
156 	}
157 	UART0_REGS->ACTV = 0;
158 	uart_activate[1] = UART1_REGS->ACTV;
159 	if (uart_activate[1]) {
160 		while ((UART1_REGS->LSR & MCHP_UART_LSR_TEMT) == 0) {
161 		}
162 	}
163 	UART1_REGS->ACTV = 0;
164 	uart_activate[2] = UART2_REGS->ACTV;
165 	if (uart_activate[2]) {
166 		while ((UART2_REGS->LSR & MCHP_UART_LSR_TEMT) == 0) {
167 		}
168 	}
169 	UART2_REGS->ACTV = 0;
170 }
171 
deep_sleep_save_timers(void)172 static void deep_sleep_save_timers(void)
173 {
174 	const struct ds_timer_info *p;
175 	uint32_t i;
176 
177 	p = &ds_timer_tbl[0];
178 	for (i = 0; i < NUM_DS_TIMER_ENTRIES; i++) {
179 		timers[i] = REG32(p->addr);
180 		REG32(p->addr) = 0;
181 		p++;
182 	}
183 }
184 
deep_sleep_restore_ecs(void)185 static void deep_sleep_restore_ecs(void)
186 {
187 	ECS_REGS->ETM_CTRL = ecs[0];
188 }
189 
deep_sleep_restore_uarts(void)190 static void deep_sleep_restore_uarts(void)
191 {
192 	UART0_REGS->ACTV = uart_activate[0];
193 	UART1_REGS->ACTV = uart_activate[1];
194 	UART2_REGS->ACTV = uart_activate[2];
195 }
196 
deep_sleep_restore_timers(void)197 static void deep_sleep_restore_timers(void)
198 {
199 	const struct ds_timer_info *p;
200 	uint32_t i;
201 
202 	p = &ds_timer_tbl[0];
203 	for (i = 0; i < NUM_DS_TIMER_ENTRIES; i++) {
204 		REG32(p->addr) = timers[i] & ~p->restore_mask;
205 		p++;
206 	}
207 }
208 
soc_deep_sleep_periph_save(void)209 void soc_deep_sleep_periph_save(void)
210 {
211 	deep_sleep_save_uarts();
212 	deep_sleep_save_ecs();
213 	deep_sleep_save_timers();
214 }
215 
soc_deep_sleep_periph_restore(void)216 void soc_deep_sleep_periph_restore(void)
217 {
218 	deep_sleep_restore_ecs();
219 	deep_sleep_restore_uarts();
220 	deep_sleep_restore_timers();
221 }
222 
223 #else
224 
soc_deep_sleep_periph_save(void)225 void soc_deep_sleep_periph_save(void)
226 {
227 }
228 
soc_deep_sleep_periph_restore(void)229 void soc_deep_sleep_periph_restore(void)
230 {
231 }
232 
233 #endif /* DEEP_SLEEP_PERIPH_SAVE_RESTORE */
234