1 /*
2 * Copyright (c) 2020 Intel Corporation
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT microchip_xec_peci
8
9 #include <errno.h>
10 #include <zephyr/device.h>
11 #include <zephyr/kernel.h>
12 #ifdef CONFIG_SOC_SERIES_MEC172X
13 #include <zephyr/drivers/clock_control/mchp_xec_clock_control.h>
14 #include <zephyr/drivers/interrupt_controller/intc_mchp_xec_ecia.h>
15 #endif
16 #include <zephyr/drivers/peci.h>
17 #include <zephyr/drivers/pinctrl.h>
18 #include <zephyr/pm/device.h>
19 #include <zephyr/pm/policy.h>
20 #include <soc.h>
21 #include <zephyr/logging/log.h>
22 #include <zephyr/irq.h>
23 LOG_MODULE_REGISTER(peci_mchp_xec, CONFIG_PECI_LOG_LEVEL);
24
25 /* Maximum PECI core clock is the main clock 48Mhz */
26 #define MAX_PECI_CORE_CLOCK 48000u
27 /* 1 ms */
28 #define PECI_RESET_DELAY 1000u
29 #define PECI_RESET_DELAY_MS 1u
30 /* 100 us */
31 #define PECI_IDLE_DELAY 100u
32 /* 5 ms */
33 #define PECI_IDLE_TIMEOUT 50u
34 /* Maximum retries */
35 #define PECI_TIMEOUT_RETRIES 3u
36 /* Maximum read buffer fill wait retries */
37 #define PECI_RX_BUF_FILL_WAIT_RETRY 100u
38
39 /* 10 us */
40 #define PECI_IO_DELAY 10
41
42 #define OPT_BIT_TIME_MSB_OFS 8u
43
44 #define PECI_FCS_LEN 2
45
46 struct peci_xec_config {
47 struct peci_regs * const regs;
48 uint8_t irq_num;
49 uint8_t girq;
50 uint8_t girq_pos;
51 uint8_t pcr_idx;
52 uint8_t pcr_pos;
53 const struct pinctrl_dev_config *pcfg;
54 };
55
56 enum peci_pm_policy_state_flag {
57 PECI_PM_POLICY_FLAG,
58 PECI_PM_POLICY_FLAG_COUNT,
59 };
60
61 struct peci_xec_data {
62 struct k_sem tx_lock;
63 uint32_t bitrate;
64 int timeout_retries;
65 #ifdef CONFIG_PM_DEVICE
66 ATOMIC_DEFINE(pm_policy_state_flag, PECI_PM_POLICY_FLAG_COUNT);
67 #endif
68 };
69
70 #ifdef CONFIG_PM_DEVICE
peci_xec_pm_policy_state_lock_get(struct peci_xec_data * data,enum peci_pm_policy_state_flag flag)71 static void peci_xec_pm_policy_state_lock_get(struct peci_xec_data *data,
72 enum peci_pm_policy_state_flag flag)
73 {
74 if (atomic_test_and_set_bit(data->pm_policy_state_flag, flag) == 0) {
75 pm_policy_state_lock_get(PM_STATE_SUSPEND_TO_IDLE, PM_ALL_SUBSTATES);
76 }
77 }
78
peci_xec_pm_policy_state_lock_put(struct peci_xec_data * data,enum peci_pm_policy_state_flag flag)79 static void peci_xec_pm_policy_state_lock_put(struct peci_xec_data *data,
80 enum peci_pm_policy_state_flag flag)
81 {
82 if (atomic_test_and_clear_bit(data->pm_policy_state_flag, flag) == 1) {
83 pm_policy_state_lock_put(PM_STATE_SUSPEND_TO_IDLE, PM_ALL_SUBSTATES);
84 }
85 }
86 #endif
87
88 #ifdef CONFIG_SOC_SERIES_MEC172X
peci_girq_enable(const struct device * dev)89 static inline void peci_girq_enable(const struct device *dev)
90 {
91 const struct peci_xec_config * const cfg = dev->config;
92
93 mchp_xec_ecia_girq_src_en(cfg->girq, cfg->girq_pos);
94 }
95
peci_girq_status_clear(const struct device * dev)96 static inline void peci_girq_status_clear(const struct device *dev)
97 {
98 const struct peci_xec_config * const cfg = dev->config;
99
100 mchp_soc_ecia_girq_src_clr(cfg->girq, cfg->girq_pos);
101 }
102
peci_clr_slp_en(const struct device * dev)103 static inline void peci_clr_slp_en(const struct device *dev)
104 {
105 const struct peci_xec_config * const cfg = dev->config;
106
107 z_mchp_xec_pcr_periph_sleep(cfg->pcr_idx, cfg->pcr_pos, 0);
108 }
109 #else
peci_girq_enable(const struct device * dev)110 static inline void peci_girq_enable(const struct device *dev)
111 {
112 const struct peci_xec_config * const cfg = dev->config;
113
114 MCHP_GIRQ_ENSET(cfg->girq) = BIT(cfg->girq_pos);
115 }
116
peci_girq_status_clear(const struct device * dev)117 static inline void peci_girq_status_clear(const struct device *dev)
118 {
119 const struct peci_xec_config * const cfg = dev->config;
120
121 MCHP_GIRQ_SRC(cfg->girq) = BIT(cfg->girq_pos);
122 }
123
peci_clr_slp_en(const struct device * dev)124 static inline void peci_clr_slp_en(const struct device *dev)
125 {
126 ARG_UNUSED(dev);
127
128 mchp_pcr_periph_slp_ctrl(PCR_PECI, 0);
129 }
130 #endif
131
check_bus_idle(struct peci_regs * const regs)132 static int check_bus_idle(struct peci_regs * const regs)
133 {
134 uint8_t delay_cnt = PECI_IDLE_TIMEOUT;
135
136 /* Wait until PECI bus becomes idle.
137 * Note that when IDLE bit in the status register changes, HW do not
138 * generate an interrupt, so need to poll.
139 */
140 while (!(regs->STATUS2 & MCHP_PECI_STS2_IDLE)) {
141 k_busy_wait(PECI_IDLE_DELAY);
142 delay_cnt--;
143
144 if (!delay_cnt) {
145 LOG_WRN("Bus is busy");
146 return -EBUSY;
147 }
148 }
149 return 0;
150 }
151
peci_xec_configure(const struct device * dev,uint32_t bitrate)152 static int peci_xec_configure(const struct device *dev, uint32_t bitrate)
153 {
154 const struct peci_xec_config * const cfg = dev->config;
155 struct peci_xec_data * const data = dev->data;
156 struct peci_regs * const regs = cfg->regs;
157 uint16_t value;
158
159 data->bitrate = bitrate;
160
161 /* Power down PECI interface */
162 regs->CONTROL = MCHP_PECI_CTRL_PD;
163
164 /* Adjust bitrate */
165 value = MAX_PECI_CORE_CLOCK / bitrate;
166 regs->OPT_BIT_TIME_LSB = value & MCHP_PECI_OPT_BT_LSB_MASK;
167 regs->OPT_BIT_TIME_MSB = ((value >> OPT_BIT_TIME_MSB_OFS) &
168 MCHP_PECI_OPT_BT_MSB_MASK);
169
170 /* Power up PECI interface */
171 regs->CONTROL &= ~MCHP_PECI_CTRL_PD;
172
173 return 0;
174 }
175
peci_xec_disable(const struct device * dev)176 static int peci_xec_disable(const struct device *dev)
177 {
178 const struct peci_xec_config * const cfg = dev->config;
179 struct peci_regs * const regs = cfg->regs;
180 int ret;
181
182 /* Make sure no transaction is interrupted before disabling the HW */
183 ret = check_bus_idle(regs);
184 if (ret) {
185 return ret;
186 }
187
188 #ifdef CONFIG_PECI_INTERRUPT_DRIVEN
189 peci_girq_status_clear(dev);
190 NVIC_ClearPendingIRQ(cfg->irq_num);
191 irq_disable(cfg->irq_num);
192 #endif
193 regs->CONTROL |= MCHP_PECI_CTRL_PD;
194
195 return 0;
196 }
197
peci_xec_enable(const struct device * dev)198 static int peci_xec_enable(const struct device *dev)
199 {
200 const struct peci_xec_config * const cfg = dev->config;
201 struct peci_regs * const regs = cfg->regs;
202
203 regs->CONTROL &= ~MCHP_PECI_CTRL_PD;
204
205 #ifdef CONFIG_PECI_INTERRUPT_DRIVEN
206 peci_girq_status_clear(dev);
207 peci_girq_enable(dev);
208 irq_enable(cfg->irq_num);
209 #endif
210 return 0;
211 }
212
peci_xec_bus_recovery(const struct device * dev,bool full_reset)213 static void peci_xec_bus_recovery(const struct device *dev, bool full_reset)
214 {
215 const struct peci_xec_config * const cfg = dev->config;
216 struct peci_xec_data * const data = dev->data;
217 struct peci_regs * const regs = cfg->regs;
218
219 LOG_WRN("%s full_reset:%d", __func__, full_reset);
220 if (full_reset) {
221 regs->CONTROL = MCHP_PECI_CTRL_PD | MCHP_PECI_CTRL_RST;
222
223 if (k_is_in_isr()) {
224 k_busy_wait(PECI_RESET_DELAY_MS);
225 } else {
226 k_msleep(PECI_RESET_DELAY);
227 }
228
229 regs->CONTROL &= ~MCHP_PECI_CTRL_RST;
230
231 peci_xec_configure(dev, data->bitrate);
232 } else {
233 /* Only reset internal FIFOs */
234 regs->CONTROL |= MCHP_PECI_CTRL_FRST;
235 }
236 }
237
peci_xec_write(const struct device * dev,struct peci_msg * msg)238 static int peci_xec_write(const struct device *dev, struct peci_msg *msg)
239 {
240 const struct peci_xec_config * const cfg = dev->config;
241 struct peci_xec_data * const data = dev->data;
242 struct peci_regs * const regs = cfg->regs;
243 int i;
244 int ret;
245
246 struct peci_buf *tx_buf = &msg->tx_buffer;
247 struct peci_buf *rx_buf = &msg->rx_buffer;
248
249 /* Check if FIFO is full */
250 if (regs->STATUS2 & MCHP_PECI_STS2_WFF) {
251 LOG_WRN("%s FIFO is full", __func__);
252 return -EIO;
253 }
254
255 regs->CONTROL &= ~MCHP_PECI_CTRL_FRST;
256
257 /* Add PECI transaction header to TX FIFO */
258 regs->WR_DATA = msg->addr;
259 regs->WR_DATA = tx_buf->len;
260 regs->WR_DATA = rx_buf->len;
261
262 /* Add PECI payload to Tx FIFO only if write length is valid */
263 if (tx_buf->len) {
264 regs->WR_DATA = msg->cmd_code;
265 for (i = 0; i < tx_buf->len - 1; i++) {
266 if (!(regs->STATUS2 & MCHP_PECI_STS2_WFF)) {
267 regs->WR_DATA = tx_buf->buf[i];
268 }
269 }
270 }
271
272 /* Check bus is idle before starting a new transfer */
273 ret = check_bus_idle(regs);
274 if (ret) {
275 return ret;
276 }
277
278 regs->CONTROL |= MCHP_PECI_CTRL_TXEN;
279 k_busy_wait(PECI_IO_DELAY);
280
281 /* Wait for transmission to complete */
282 #ifdef CONFIG_PECI_INTERRUPT_DRIVEN
283 if (k_sem_take(&data->tx_lock, PECI_IO_DELAY * tx_buf->len)) {
284 return -ETIMEDOUT;
285 }
286 #else
287 /* In worst case, overall timeout will be 1msec (100 * 10usec) */
288 uint8_t wait_timeout_cnt = 100;
289
290 while (!(regs->STATUS1 & MCHP_PECI_STS1_EOF)) {
291 k_busy_wait(PECI_IO_DELAY);
292 wait_timeout_cnt--;
293 if (!wait_timeout_cnt) {
294 LOG_WRN("Tx timeout");
295 data->timeout_retries++;
296 /* Full reset only if multiple consecutive failures */
297 if (data->timeout_retries > PECI_TIMEOUT_RETRIES) {
298 peci_xec_bus_recovery(dev, true);
299 } else {
300 peci_xec_bus_recovery(dev, false);
301 }
302
303 return -ETIMEDOUT;
304 }
305 }
306 #endif
307 data->timeout_retries = 0;
308
309 return 0;
310 }
311
peci_xec_read(const struct device * dev,struct peci_msg * msg)312 static int peci_xec_read(const struct device *dev, struct peci_msg *msg)
313 {
314 const struct peci_xec_config * const cfg = dev->config;
315 struct peci_regs * const regs = cfg->regs;
316 int i;
317 int ret;
318 uint8_t tx_fcs;
319 uint8_t bytes_rcvd;
320 uint8_t wait_timeout_cnt;
321 struct peci_buf *rx_buf = &msg->rx_buffer;
322
323 /* Attempt to read data from RX FIFO */
324 bytes_rcvd = 0;
325 for (i = 0; i < (rx_buf->len + PECI_FCS_LEN); i++) {
326 /* Worst case timeout will be 1msec (100 * 10usec) */
327 wait_timeout_cnt = PECI_RX_BUF_FILL_WAIT_RETRY;
328 /* Wait for read buffer to fill up */
329 while (regs->STATUS2 & MCHP_PECI_STS2_RFE) {
330 k_usleep(PECI_IO_DELAY);
331 wait_timeout_cnt--;
332 if (!wait_timeout_cnt) {
333 LOG_WRN("Rx buffer empty");
334 return -ETIMEDOUT;
335 }
336 }
337
338 if (i == 0) {
339 /* Get write block FCS just for debug */
340 tx_fcs = regs->RD_DATA;
341 LOG_DBG("TX FCS %x", tx_fcs);
342
343 /* If a Ping is done, write Tx fcs to rx buffer*/
344 if (msg->cmd_code == PECI_CMD_PING) {
345 rx_buf->buf[0] = tx_fcs;
346 break;
347 }
348 } else if (i == (rx_buf->len + 1)) {
349 /* Get read block FCS, but don't count it */
350 rx_buf->buf[i-1] = regs->RD_DATA;
351 } else {
352 /* Get response */
353 rx_buf->buf[i-1] = regs->RD_DATA;
354 bytes_rcvd++;
355 }
356 }
357
358 /* Check if transaction is as expected */
359 if (rx_buf->len != bytes_rcvd) {
360 LOG_INF("Incomplete %x vs %x", bytes_rcvd, rx_buf->len);
361 }
362
363 /* Once write-read transaction is complete, ensure bus is idle
364 * before resetting the internal FIFOs
365 */
366 ret = check_bus_idle(regs);
367 if (ret) {
368 return ret;
369 }
370
371 return 0;
372 }
373
peci_xec_transfer(const struct device * dev,struct peci_msg * msg)374 static int peci_xec_transfer(const struct device *dev, struct peci_msg *msg)
375 {
376 const struct peci_xec_config * const cfg = dev->config;
377 struct peci_regs * const regs = cfg->regs;
378 int ret = 0;
379 uint8_t err_val = 0;
380 #ifdef CONFIG_PM_DEVICE
381 struct peci_xec_data *data = dev->data;
382
383 peci_xec_pm_policy_state_lock_get(data, PECI_PM_POLICY_FLAG);
384 #endif
385
386 do {
387 ret = peci_xec_write(dev, msg);
388 if (ret) {
389 break;
390 }
391
392 /* If a PECI transmission is successful, it may or not involve
393 * a read operation, check if transaction expects a response
394 * Also perform a read when PECI cmd is Ping to get Write FCS
395 */
396 if (msg->rx_buffer.len || (msg->cmd_code == PECI_CMD_PING)) {
397 ret = peci_xec_read(dev, msg);
398 if (ret) {
399 break;
400 }
401 }
402
403 /* Cleanup */
404 if (regs->STATUS1 & MCHP_PECI_STS1_EOF) {
405 regs->STATUS1 |= MCHP_PECI_STS1_EOF;
406 }
407
408 /* Check for error conditions and perform bus recovery if necessary */
409 err_val = regs->ERROR;
410 if (err_val) {
411 if (err_val & MCHP_PECI_ERR_RDOV) {
412 LOG_ERR("Read buffer is not empty");
413 }
414
415 if (err_val & MCHP_PECI_ERR_WRUN) {
416 LOG_ERR("Write buffer is not empty");
417 }
418
419 if (err_val & MCHP_PECI_ERR_BERR) {
420 LOG_ERR("PECI bus error");
421 }
422
423 LOG_DBG("PECI err %x", err_val);
424 LOG_DBG("PECI sts1 %x", regs->STATUS1);
425 LOG_DBG("PECI sts2 %x", regs->STATUS2);
426
427 /* ERROR is a clear-on-write register, need to clear errors
428 * occurring at the end of a transaction. A temp variable is
429 * used to overcome complaints by the static code analyzer
430 */
431 regs->ERROR = err_val;
432 peci_xec_bus_recovery(dev, false);
433 ret = -EIO;
434 break;
435 }
436 } while (0);
437
438 #ifdef CONFIG_PM_DEVICE
439 peci_xec_pm_policy_state_lock_put(data, PECI_PM_POLICY_FLAG);
440 #endif
441 return ret;
442 }
443
444 #ifdef CONFIG_PM_DEVICE
peci_xec_pm_action(const struct device * dev,enum pm_device_action action)445 static int peci_xec_pm_action(const struct device *dev, enum pm_device_action action)
446 {
447 const struct peci_xec_config *const devcfg = dev->config;
448 struct peci_regs * const regs = devcfg->regs;
449 struct ecs_regs * const ecs_regs = (struct ecs_regs *)(DT_REG_ADDR(DT_NODELABEL(ecs)));
450 int ret;
451
452 switch (action) {
453 case PM_DEVICE_ACTION_RESUME:
454 ret = pinctrl_apply_state(devcfg->pcfg, PINCTRL_STATE_DEFAULT);
455 /* VREF_VTT function is enabled*/
456 ecs_regs->PECI_DIS = 0x00u;
457
458 /* Power up PECI interface */
459 regs->CONTROL &= ~MCHP_PECI_CTRL_PD;
460 break;
461 case PM_DEVICE_ACTION_SUSPEND:
462 regs->CONTROL |= MCHP_PECI_CTRL_PD;
463 /* This bit reduces leakage current through the CPU voltage reference
464 * pin if PECI is not used. VREF_VTT function is disabled.
465 */
466 ecs_regs->PECI_DIS = 0x01u;
467
468 /* If application does not want to turn off PECI pins it will
469 * not define pinctrl-1 for this node.
470 */
471 ret = pinctrl_apply_state(devcfg->pcfg, PINCTRL_STATE_SLEEP);
472 if (ret == -ENOENT) { /* pinctrl-1 does not exist. */
473 ret = 0;
474 }
475 break;
476 default:
477 ret = -ENOTSUP;
478 }
479
480 return ret;
481 }
482 #endif /* CONFIG_PM_DEVICE */
483
484 #ifdef CONFIG_PECI_INTERRUPT_DRIVEN
peci_xec_isr(const void * arg)485 static void peci_xec_isr(const void *arg)
486 {
487 const struct device *dev = arg;
488 struct peci_xec_config * const cfg = dev->config;
489 struct peci_xec_data * const data = dev->data;
490 struct peci_regs * const regs = cfg->regs;
491 uint8_t peci_error = regs->ERROR;
492 uint8_t peci_status2 = regs->STATUS2;
493
494 peci_girq_status_clear(dev);
495
496 if (peci_error) {
497 regs->ERROR = peci_error;
498 }
499
500 if (peci_status2 & MCHP_PECI_STS2_WFE) {
501 LOG_WRN("TX FIFO empty ST2:%x", peci_status2);
502 k_sem_give(&data->tx_lock);
503 }
504
505 if (peci_status2 & MCHP_PECI_STS2_RFE) {
506 LOG_WRN("RX FIFO full ST2:%x", peci_status2);
507 }
508 }
509 #endif
510
511 static DEVICE_API(peci, peci_xec_driver_api) = {
512 .config = peci_xec_configure,
513 .enable = peci_xec_enable,
514 .disable = peci_xec_disable,
515 .transfer = peci_xec_transfer,
516 };
517
peci_xec_init(const struct device * dev)518 static int peci_xec_init(const struct device *dev)
519 {
520 const struct peci_xec_config * const cfg = dev->config;
521 struct peci_regs * const regs = cfg->regs;
522 struct ecs_regs * const ecs_regs = (struct ecs_regs *)(DT_REG_ADDR(DT_NODELABEL(ecs)));
523
524 int ret = pinctrl_apply_state(cfg->pcfg, PINCTRL_STATE_DEFAULT);
525
526 if (ret != 0) {
527 LOG_ERR("XEC PECI pinctrl init failed (%d)", ret);
528 return ret;
529 }
530
531 #ifdef CONFIG_PECI_INTERRUPT_DRIVEN
532 k_sem_init(&data->tx_lock, 0, 1);
533 #endif
534
535 peci_clr_slp_en(dev);
536
537 ecs_regs->PECI_DIS = 0x00u;
538
539 /* Reset PECI interface */
540 regs->CONTROL |= MCHP_PECI_CTRL_RST;
541 k_msleep(PECI_RESET_DELAY_MS);
542 regs->CONTROL &= ~MCHP_PECI_CTRL_RST;
543
544 #ifdef CONFIG_PECI_INTERRUPT_DRIVEN
545 /* Enable interrupt for errors */
546 regs->INT_EN1 = (MCHP_PECI_IEN1_EREN | MCHP_PECI_IEN1_EIEN);
547
548 /* Enable interrupt for Tx FIFO is empty */
549 regs->INT_EN2 |= MCHP_PECI_IEN2_ENWFE;
550 /* Enable interrupt for Rx FIFO is full */
551 regs->INT_EN2 |= MCHP_PECI_IEN2_ENRFF;
552
553 regs->CONTROL |= MCHP_PECI_CTRL_MIEN;
554
555 /* Direct NVIC */
556 IRQ_CONNECT(cfg->irq_num,
557 DT_INST_IRQ(0, priority),
558 peci_xec_isr, NULL, 0);
559 #endif
560 return 0;
561 }
562
563 static struct peci_xec_data peci_data;
564
565 PINCTRL_DT_INST_DEFINE(0);
566
567 static const struct peci_xec_config peci_xec_config = {
568 .regs = (struct peci_regs * const)(DT_INST_REG_ADDR(0)),
569 .irq_num = DT_INST_IRQN(0),
570 .girq = DT_INST_PROP_BY_IDX(0, girqs, 0),
571 .girq_pos = DT_INST_PROP_BY_IDX(0, girqs, 1),
572 .pcr_idx = DT_INST_PROP_BY_IDX(0, pcrs, 0),
573 .pcr_pos = DT_INST_PROP_BY_IDX(0, pcrs, 1),
574 .pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(0),
575 };
576
577 PM_DEVICE_DT_INST_DEFINE(0, peci_xec_pm_action);
578
579 DEVICE_DT_INST_DEFINE(0,
580 &peci_xec_init,
581 PM_DEVICE_DT_INST_GET(0),
582 &peci_data, &peci_xec_config,
583 POST_KERNEL, CONFIG_PECI_INIT_PRIORITY,
584 &peci_xec_driver_api);
585