1 /*
2 * Copyright (c) 2022 Intel Corporation
3 *
4 * Intel I/O Controller Hub (ICH) later renamed to Intel Platform Controller
5 * Hub (PCH) SMBus driver.
6 *
7 * PCH provides SMBus 2.0 - compliant Host Controller.
8 *
9 * SPDX-License-Identifier: Apache-2.0
10 */
11
12 #include <zephyr/types.h>
13 #include <zephyr/kernel.h>
14 #include <zephyr/device.h>
15
16 #include <zephyr/drivers/smbus.h>
17 #include <zephyr/drivers/pcie/pcie.h>
18
19 #define DT_DRV_COMPAT intel_pch_smbus
20
21 #include <zephyr/logging/log.h>
22 LOG_MODULE_REGISTER(intel_pch, CONFIG_SMBUS_LOG_LEVEL);
23
24 #include "smbus_utils.h"
25 #include "intel_pch_smbus.h"
26
27 /**
28 * @note Following notions are used:
29 * * periph_addr - Peripheral address (Slave address mentioned in the Specs)
30 * * command - First byte to send in the SMBus protocol operations except for
31 * Quick and Byte Read. Also known as register.
32 */
33
34 /**
35 * Intel PCH configuration acquired from DTS during device initialization
36 */
37 struct pch_config {
38 /* IRQ configuration function */
39 void (*config_func)(const struct device *dev);
40 struct pcie_dev *pcie;
41 };
42
43 /**
44 * Intel PCH internal driver data
45 */
46 struct pch_data {
47 DEVICE_MMIO_RAM;
48 io_port_t sba;
49 uint32_t config;
50 uint8_t status;
51
52 struct k_mutex mutex;
53 struct k_sem completion_sync;
54 const struct device *dev;
55
56 #if defined(CONFIG_SMBUS_INTEL_PCH_SMBALERT)
57 /* smbalert callback list */
58 sys_slist_t smbalert_cbs;
59 /* smbalert work */
60 struct k_work smb_alert_work;
61 #endif /* CONFIG_SMBUS_INTEL_PCH_SMBALERT */
62
63 #if defined(CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY)
64 /* Host Notify callback list */
65 sys_slist_t host_notify_cbs;
66 /* Host Notify work */
67 struct k_work host_notify_work;
68 /* Host Notify peripheral device address */
69 uint8_t notify_addr;
70 /* Host Notify data received */
71 uint16_t notify_data;
72 #endif /* CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY */
73 };
74
75 /**
76 * Helpers for accessing Intel PCH SMBus registers. Depending on
77 * configuration option MMIO or IO method will be used.
78 */
79 #if defined(CONFIG_SMBUS_INTEL_PCH_ACCESS_MMIO)
pch_reg_read(const struct device * dev,uint8_t reg)80 static uint8_t pch_reg_read(const struct device *dev, uint8_t reg)
81 {
82 return sys_read8(DEVICE_MMIO_GET(dev) + reg);
83 }
84
pch_reg_write(const struct device * dev,uint8_t reg,uint8_t val)85 static void pch_reg_write(const struct device *dev, uint8_t reg, uint8_t val)
86 {
87 sys_write8(val, DEVICE_MMIO_GET(dev) + reg);
88 }
89 #elif defined(CONFIG_SMBUS_INTEL_PCH_ACCESS_IO)
pch_reg_read(const struct device * dev,uint8_t reg)90 static uint8_t pch_reg_read(const struct device *dev, uint8_t reg)
91 {
92 struct pch_data *data = dev->data;
93
94 return sys_in8(data->sba + reg);
95 }
96
pch_reg_write(const struct device * dev,uint8_t reg,uint8_t val)97 static void pch_reg_write(const struct device *dev, uint8_t reg, uint8_t val)
98 {
99 struct pch_data *data = dev->data;
100
101 sys_out8(val, data->sba + reg);
102 }
103 #else
104 #error Wrong PCH Register Access Mode
105 #endif
106
107 #if defined(CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY)
host_notify_work(struct k_work * work)108 static void host_notify_work(struct k_work *work)
109 {
110 struct pch_data *data = CONTAINER_OF(work, struct pch_data,
111 host_notify_work);
112 const struct device *dev = data->dev;
113 uint8_t addr = data->notify_addr;
114
115 smbus_fire_callbacks(&data->host_notify_cbs, dev, addr);
116 }
117
pch_smbus_host_notify_set_cb(const struct device * dev,struct smbus_callback * cb)118 static int pch_smbus_host_notify_set_cb(const struct device *dev,
119 struct smbus_callback *cb)
120 {
121 struct pch_data *data = dev->data;
122
123 LOG_DBG("dev %p cb %p", dev, cb);
124
125 return smbus_callback_set(&data->host_notify_cbs, cb);
126 }
127
pch_smbus_host_notify_remove_cb(const struct device * dev,struct smbus_callback * cb)128 static int pch_smbus_host_notify_remove_cb(const struct device *dev,
129 struct smbus_callback *cb)
130 {
131 struct pch_data *data = dev->data;
132
133 LOG_DBG("dev %p cb %p", dev, cb);
134
135 return smbus_callback_remove(&data->host_notify_cbs, cb);
136 }
137 #endif /* CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY */
138
139 #if defined(CONFIG_SMBUS_INTEL_PCH_SMBALERT)
smbalert_work(struct k_work * work)140 static void smbalert_work(struct k_work *work)
141 {
142 struct pch_data *data = CONTAINER_OF(work, struct pch_data,
143 smb_alert_work);
144 const struct device *dev = data->dev;
145
146 smbus_loop_alert_devices(dev, &data->smbalert_cbs);
147 }
148
pch_smbus_smbalert_set_sb(const struct device * dev,struct smbus_callback * cb)149 static int pch_smbus_smbalert_set_sb(const struct device *dev,
150 struct smbus_callback *cb)
151 {
152 struct pch_data *data = dev->data;
153
154 LOG_DBG("dev %p cb %p", dev, cb);
155
156 return smbus_callback_set(&data->smbalert_cbs, cb);
157 }
158
pch_smbus_smbalert_remove_sb(const struct device * dev,struct smbus_callback * cb)159 static int pch_smbus_smbalert_remove_sb(const struct device *dev,
160 struct smbus_callback *cb)
161 {
162 struct pch_data *data = dev->data;
163
164 LOG_DBG("dev %p cb %p", dev, cb);
165
166 return smbus_callback_remove(&data->smbalert_cbs, cb);
167 }
168 #endif /* CONFIG_SMBUS_INTEL_PCH_SMBALERT */
169
pch_configure(const struct device * dev,uint32_t config)170 static int pch_configure(const struct device *dev, uint32_t config)
171 {
172 struct pch_data *data = dev->data;
173
174 LOG_DBG("dev %p config 0x%x", dev, config);
175
176 if (config & SMBUS_MODE_HOST_NOTIFY) {
177 uint8_t status;
178
179 if (!IS_ENABLED(CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY)) {
180 LOG_ERR("Error configuring Host Notify");
181 return -EINVAL;
182 }
183
184 /* Enable Host Notify interrupts */
185 status = pch_reg_read(dev, PCH_SMBUS_SCMD);
186 status |= PCH_SMBUS_SCMD_HNI_EN;
187 pch_reg_write(dev, PCH_SMBUS_SCMD, status);
188 }
189
190 if (config & SMBUS_MODE_SMBALERT) {
191 uint8_t status;
192
193 if (!IS_ENABLED(CONFIG_SMBUS_INTEL_PCH_SMBALERT)) {
194 LOG_ERR("Error configuring SMBALERT");
195 return -EINVAL;
196 }
197
198 /* Disable SMBALERT_DIS */
199 status = pch_reg_read(dev, PCH_SMBUS_SCMD);
200 status &= ~PCH_SMBUS_SCMD_SMBALERT_DIS;
201 pch_reg_write(dev, PCH_SMBUS_SCMD, status);
202 }
203
204 /* Keep config for a moment */
205 data->config = config;
206
207 return 0;
208 }
209
pch_get_config(const struct device * dev,uint32_t * config)210 static int pch_get_config(const struct device *dev, uint32_t *config)
211 {
212 struct pch_data *data = dev->data;
213
214 *config = data->config;
215
216 return 0;
217 }
218
219 /* Device initialization function */
pch_smbus_init(const struct device * dev)220 static int pch_smbus_init(const struct device *dev)
221 {
222 const struct pch_config * const config = dev->config;
223 struct pch_data *data = dev->data;
224 struct pcie_bar mbar;
225 uint32_t val;
226
227 if (config->pcie->bdf == PCIE_BDF_NONE) {
228 LOG_ERR("Cannot probe PCI device");
229 return -ENODEV;
230 }
231
232 val = pcie_conf_read(config->pcie->bdf, PCIE_CONF_CMDSTAT);
233 if (val & PCIE_CONF_CMDSTAT_INTERRUPT) {
234 LOG_WRN("Pending interrupt, continuing");
235 }
236
237 if (IS_ENABLED(CONFIG_SMBUS_INTEL_PCH_ACCESS_MMIO)) {
238 pcie_probe_mbar(config->pcie->bdf, 0, &mbar);
239 pcie_set_cmd(config->pcie->bdf, PCIE_CONF_CMDSTAT_MEM, true);
240
241 device_map(DEVICE_MMIO_RAM_PTR(dev), mbar.phys_addr, mbar.size,
242 K_MEM_CACHE_NONE);
243
244 LOG_DBG("Mapped 0x%lx size 0x%lx to 0x%lx",
245 mbar.phys_addr, mbar.size, DEVICE_MMIO_GET(dev));
246 } else {
247 pcie_set_cmd(config->pcie->bdf, PCIE_CONF_CMDSTAT_IO, true);
248 val = pcie_conf_read(config->pcie->bdf, PCIE_CONF_BAR4);
249 if (!PCIE_CONF_BAR_IO(val)) {
250 LOG_ERR("Cannot read IO BAR");
251 return -EINVAL;
252 }
253
254 data->sba = PCIE_CONF_BAR_ADDR(val);
255
256 LOG_DBG("Using I/O address 0x%x", data->sba);
257 }
258
259 val = pcie_conf_read(config->pcie->bdf, PCH_SMBUS_HCFG);
260 if ((val & PCH_SMBUS_HCFG_HST_EN) == 0) {
261 LOG_ERR("SMBus Host Controller is disabled");
262 return -EINVAL;
263 }
264
265 /* Initialize mutex and semaphore */
266 k_mutex_init(&data->mutex);
267 k_sem_init(&data->completion_sync, 0, 1);
268
269 data->dev = dev;
270
271 /* Initialize work structures */
272
273 #if defined(CONFIG_SMBUS_INTEL_PCH_SMBALERT)
274 k_work_init(&data->smb_alert_work, smbalert_work);
275 #endif /* CONFIG_SMBUS_INTEL_PCH_SMBALERT */
276
277 #if defined(CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY)
278 k_work_init(&data->host_notify_work, host_notify_work);
279 #endif /* CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY */
280
281 config->config_func(dev);
282
283 if (pch_configure(dev, SMBUS_MODE_CONTROLLER)) {
284 LOG_ERR("SMBus: Cannot set default configuration");
285 return -EIO;
286 }
287
288 return 0;
289 }
290
pch_prepare_transfer(const struct device * dev)291 static int pch_prepare_transfer(const struct device *dev)
292 {
293 uint8_t hsts;
294
295 hsts = pch_reg_read(dev, PCH_SMBUS_HSTS);
296
297 pch_dump_register_hsts(hsts);
298
299 if (hsts & PCH_SMBUS_HSTS_HOST_BUSY) {
300 LOG_ERR("Return BUSY status");
301 return -EBUSY;
302 }
303
304 /* Check and clear HSTS status bits */
305 hsts &= PCH_SMBUS_HSTS_ERROR | PCH_SMBUS_HSTS_BYTE_DONE |
306 PCH_SMBUS_HSTS_INTERRUPT;
307 if (hsts) {
308 pch_reg_write(dev, PCH_SMBUS_HSTS, hsts);
309 }
310
311 /* TODO: Clear also CRC check bits */
312
313 return 0;
314 }
315
pch_check_status(const struct device * dev)316 static int pch_check_status(const struct device *dev)
317 {
318 struct pch_data *data = dev->data;
319 uint8_t status = data->status;
320
321 /**
322 * Device Error means following:
323 * - unsupported Command Field Unclaimed Cycle
324 * - Host Device timeout
325 * - CRC Error
326 */
327 if (status & PCH_SMBUS_HSTS_DEV_ERROR) {
328 uint8_t auxs = pch_reg_read(dev, PCH_SMBUS_AUXS);
329
330 LOG_WRN("Device Error (DERR) received");
331
332 if (auxs & PCH_SMBUS_AUXS_CRC_ERROR) {
333 LOG_DBG("AUXS register 0x%02x", auxs);
334 /* Clear CRC error status bit */
335 pch_reg_write(dev, PCH_SMBUS_AUXS,
336 PCH_SMBUS_AUXS_CRC_ERROR);
337 }
338
339 return -EIO;
340 }
341
342 /**
343 * Transaction collision, several masters are trying to access
344 * the bus and PCH detects arbitration lost.
345 */
346 if (status & PCH_SMBUS_HSTS_BUS_ERROR) {
347 LOG_WRN("Bus Error (BERR) received");
348
349 return -EAGAIN;
350 }
351
352 /**
353 * Source of interrupt is failed bus transaction. This is set in
354 * response to KILL set to terminate the host transaction
355 */
356 if (status & PCH_SMBUS_HSTS_FAILED) {
357 LOG_WRN("Failed (FAIL) received");
358
359 return -EIO;
360 }
361
362 return 0;
363 }
364
pch_smbus_block_start(const struct device * dev,uint16_t periph_addr,uint8_t rw,uint8_t command,uint8_t count,uint8_t * buf,uint8_t protocol)365 static int pch_smbus_block_start(const struct device *dev, uint16_t periph_addr,
366 uint8_t rw, uint8_t command, uint8_t count,
367 uint8_t *buf, uint8_t protocol)
368 {
369 uint8_t reg;
370 int ret;
371
372 LOG_DBG("addr %x rw %d command %x", periph_addr, rw, command);
373
374 /* Set TSA register */
375 reg = PCH_SMBUS_TSA_ADDR_SET(periph_addr);
376 reg |= rw & SMBUS_MSG_RW_MASK;
377 pch_reg_write(dev, PCH_SMBUS_TSA, reg);
378
379 /* Set HCMD register */
380 pch_reg_write(dev, PCH_SMBUS_HCMD, command);
381
382 /* Enable 32-byte buffer mode (E32b) to send block of data */
383 reg = pch_reg_read(dev, PCH_SMBUS_AUXC);
384 reg |= PCH_SMBUS_AUXC_EN_32BUF;
385 pch_reg_write(dev, PCH_SMBUS_AUXC, reg);
386
387 /* In E32B mode read and write to PCH_SMBUS_HBD translates to
388 * read and write to 32 byte storage array, index needs to be
389 * cleared by reading HCTL
390 */
391 reg = pch_reg_read(dev, PCH_SMBUS_HCTL);
392 ARG_UNUSED(reg); /* Avoid 'Dead assignment' warning */
393
394 if (rw == SMBUS_MSG_WRITE) {
395 /* Write count */
396 pch_reg_write(dev, PCH_SMBUS_HD0, count);
397
398 /* Write data to send */
399 for (int i = 0; i < count; i++) {
400 pch_reg_write(dev, PCH_SMBUS_HBD, buf[i]);
401 }
402 }
403
404 ret = pch_prepare_transfer(dev);
405 if (ret < 0) {
406 return ret;
407 }
408
409 /* Set HCTL register */
410 reg = PCH_SMBUS_HCTL_CMD_SET(protocol);
411 reg |= PCH_SMBUS_HCTL_START;
412 reg |= PCH_SMBUS_HCTL_INTR_EN;
413 pch_reg_write(dev, PCH_SMBUS_HCTL, reg);
414
415 return 0;
416 }
417
418 /* Start PCH SMBus operation */
pch_smbus_start(const struct device * dev,uint16_t periph_addr,enum smbus_direction rw,uint8_t command,uint8_t * buf,uint8_t protocol)419 static int pch_smbus_start(const struct device *dev, uint16_t periph_addr,
420 enum smbus_direction rw, uint8_t command,
421 uint8_t *buf, uint8_t protocol)
422 {
423 uint8_t reg;
424 int ret;
425
426 LOG_DBG("addr 0x%02x rw %d command %x", periph_addr, rw, command);
427
428 /* Set TSA register */
429 reg = PCH_SMBUS_TSA_ADDR_SET(periph_addr);
430 reg |= rw & SMBUS_MSG_RW_MASK;
431 pch_reg_write(dev, PCH_SMBUS_TSA, reg);
432
433 /* Write command for every but QUICK op */
434 if (protocol != SMBUS_CMD_QUICK) {
435 /* Set HCMD register */
436 pch_reg_write(dev, PCH_SMBUS_HCMD, command);
437
438 /* Set Host Data 0 (HD0) register */
439 if (rw == SMBUS_MSG_WRITE && protocol != SMBUS_CMD_BYTE) {
440 pch_reg_write(dev, PCH_SMBUS_HD0, buf[0]);
441
442 /* If we need to write second byte */
443 if (protocol == SMBUS_CMD_WORD_DATA ||
444 protocol == SMBUS_CMD_PROC_CALL) {
445 pch_reg_write(dev, PCH_SMBUS_HD1, buf[1]);
446 }
447 }
448 }
449
450 ret = pch_prepare_transfer(dev);
451 if (ret < 0) {
452 return ret;
453 }
454
455 /* Set HCTL register */
456 reg = PCH_SMBUS_HCTL_CMD_SET(protocol);
457 reg |= PCH_SMBUS_HCTL_START;
458 reg |= PCH_SMBUS_HCTL_INTR_EN;
459 pch_reg_write(dev, PCH_SMBUS_HCTL, reg);
460
461 return 0;
462 }
463
464 /* Implementation of PCH SMBus API */
465
466 /* Implementation of SMBus Quick */
pch_smbus_quick(const struct device * dev,uint16_t periph_addr,enum smbus_direction rw)467 static int pch_smbus_quick(const struct device *dev, uint16_t periph_addr,
468 enum smbus_direction rw)
469 {
470 struct pch_data *data = dev->data;
471 int ret;
472
473 LOG_DBG("dev %p addr 0x%02x direction %x", dev, periph_addr, rw);
474
475 k_mutex_lock(&data->mutex, K_FOREVER);
476
477 ret = pch_smbus_start(dev, periph_addr, rw, 0, NULL, SMBUS_CMD_QUICK);
478 if (ret < 0) {
479 goto unlock;
480 }
481
482 /* Wait for completion from ISR */
483 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
484 if (ret != 0) {
485 LOG_ERR("SMBus Quick timed out");
486 ret = -ETIMEDOUT;
487 goto unlock;
488 }
489
490 ret = pch_check_status(dev);
491
492 unlock:
493 k_mutex_unlock(&data->mutex);
494
495 return ret;
496 }
497
498 /* Implementation of SMBus Byte Write */
pch_smbus_byte_write(const struct device * dev,uint16_t periph_addr,uint8_t command)499 static int pch_smbus_byte_write(const struct device *dev, uint16_t periph_addr,
500 uint8_t command)
501 {
502 struct pch_data *data = dev->data;
503 int ret;
504
505 LOG_DBG("dev %p addr 0x%02x command 0x%02x", dev, periph_addr, command);
506
507 k_mutex_lock(&data->mutex, K_FOREVER);
508
509 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_WRITE, command, NULL,
510 SMBUS_CMD_BYTE);
511 if (ret < 0) {
512 goto unlock;
513 }
514
515 /* Wait for completion from ISR */
516 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
517 if (ret != 0) {
518 LOG_ERR("SMBus Byte Write timed out");
519 ret = -ETIMEDOUT;
520 goto unlock;
521 }
522
523 ret = pch_check_status(dev);
524
525 unlock:
526 k_mutex_unlock(&data->mutex);
527
528 return ret;
529 }
530
531 /* Implementation of SMBus Byte Read */
pch_smbus_byte_read(const struct device * dev,uint16_t periph_addr,uint8_t * byte)532 static int pch_smbus_byte_read(const struct device *dev, uint16_t periph_addr,
533 uint8_t *byte)
534 {
535 struct pch_data *data = dev->data;
536 int ret;
537
538 LOG_DBG("dev %p addr 0x%02x", dev, periph_addr);
539
540 k_mutex_lock(&data->mutex, K_FOREVER);
541
542 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_READ, 0, NULL,
543 SMBUS_CMD_BYTE);
544 if (ret < 0) {
545 goto unlock;
546 }
547
548 /* Wait for completion from ISR */
549 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
550 if (ret != 0) {
551 LOG_ERR("SMBus Byte Read timed out");
552 ret = -ETIMEDOUT;
553 goto unlock;
554 }
555
556 ret = pch_check_status(dev);
557 if (ret < 0) {
558 goto unlock;
559 }
560
561 *byte = pch_reg_read(dev, PCH_SMBUS_HD0);
562
563 unlock:
564 k_mutex_unlock(&data->mutex);
565
566 return ret;
567 }
568
569 /* Implementation of SMBus Byte Data Write */
pch_smbus_byte_data_write(const struct device * dev,uint16_t periph_addr,uint8_t command,uint8_t byte)570 static int pch_smbus_byte_data_write(const struct device *dev,
571 uint16_t periph_addr,
572 uint8_t command, uint8_t byte)
573 {
574 struct pch_data *data = dev->data;
575 int ret;
576
577 LOG_DBG("dev %p addr 0x%02x command 0x%02x", dev, periph_addr, command);
578
579 k_mutex_lock(&data->mutex, K_FOREVER);
580
581 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_WRITE, command,
582 &byte, SMBUS_CMD_BYTE_DATA);
583 if (ret < 0) {
584 goto unlock;
585 }
586
587 /* Wait for completion from ISR */
588 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
589 if (ret != 0) {
590 LOG_ERR("SMBus Byte Data Write timed out");
591 ret = -ETIMEDOUT;
592 goto unlock;
593 }
594
595 ret = pch_check_status(dev);
596
597 unlock:
598 k_mutex_unlock(&data->mutex);
599
600 return ret;
601 }
602
603 /* Implementation of SMBus Byte Data Read */
pch_smbus_byte_data_read(const struct device * dev,uint16_t periph_addr,uint8_t command,uint8_t * byte)604 static int pch_smbus_byte_data_read(const struct device *dev,
605 uint16_t periph_addr,
606 uint8_t command, uint8_t *byte)
607 {
608 struct pch_data *data = dev->data;
609 int ret;
610
611 LOG_DBG("dev %p addr 0x%02x command 0x%02x", dev, periph_addr, command);
612
613 k_mutex_lock(&data->mutex, K_FOREVER);
614
615 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_READ, command,
616 NULL, SMBUS_CMD_BYTE_DATA);
617 if (ret < 0) {
618 goto unlock;
619 }
620
621 /* Wait for completion from ISR */
622 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
623 if (ret != 0) {
624 LOG_ERR("SMBus Byte Data Read timed out");
625 ret = -ETIMEDOUT;
626 goto unlock;
627 }
628
629 ret = pch_check_status(dev);
630 if (ret < 0) {
631 goto unlock;
632 }
633
634 *byte = pch_reg_read(dev, PCH_SMBUS_HD0);
635
636 unlock:
637 k_mutex_unlock(&data->mutex);
638
639 return ret;
640 }
641
642 /* Implementation of SMBus Word Data Write */
pch_smbus_word_data_write(const struct device * dev,uint16_t periph_addr,uint8_t command,uint16_t word)643 static int pch_smbus_word_data_write(const struct device *dev,
644 uint16_t periph_addr,
645 uint8_t command, uint16_t word)
646 {
647 struct pch_data *data = dev->data;
648 int ret;
649
650 LOG_DBG("dev %p addr 0x%02x command 0x%02x", dev, periph_addr, command);
651
652 k_mutex_lock(&data->mutex, K_FOREVER);
653
654 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_WRITE, command,
655 (uint8_t *)&word, SMBUS_CMD_WORD_DATA);
656 if (ret < 0) {
657 goto unlock;
658 }
659
660 /* Wait for completion from ISR */
661 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
662 if (ret != 0) {
663 LOG_ERR("SMBus Word Data Write timed out");
664 ret = -ETIMEDOUT;
665 goto unlock;
666 }
667
668 ret = pch_check_status(dev);
669
670 unlock:
671 k_mutex_unlock(&data->mutex);
672
673 return ret;
674 }
675
676 /* Implementation of SMBus Word Data Read */
pch_smbus_word_data_read(const struct device * dev,uint16_t periph_addr,uint8_t command,uint16_t * word)677 static int pch_smbus_word_data_read(const struct device *dev,
678 uint16_t periph_addr,
679 uint8_t command, uint16_t *word)
680 {
681 struct pch_data *data = dev->data;
682 uint8_t *p = (uint8_t *)word;
683 int ret;
684
685 LOG_DBG("dev %p addr 0x%02x command 0x%02x", dev, periph_addr, command);
686
687 k_mutex_lock(&data->mutex, K_FOREVER);
688
689 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_READ, command,
690 NULL, SMBUS_CMD_WORD_DATA);
691 if (ret < 0) {
692 goto unlock;
693 }
694
695 /* Wait for completion from ISR */
696 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
697 if (ret != 0) {
698 LOG_ERR("SMBus Word Data Read timed out");
699 ret = -ETIMEDOUT;
700 goto unlock;
701 }
702
703 ret = pch_check_status(dev);
704 if (ret < 0) {
705 goto unlock;
706 }
707
708 p[0] = pch_reg_read(dev, PCH_SMBUS_HD0);
709 p[1] = pch_reg_read(dev, PCH_SMBUS_HD1);
710
711 unlock:
712 k_mutex_unlock(&data->mutex);
713
714 return ret;
715 }
716
717 /* Implementation of SMBus Process Call */
pch_smbus_pcall(const struct device * dev,uint16_t periph_addr,uint8_t command,uint16_t send_word,uint16_t * recv_word)718 static int pch_smbus_pcall(const struct device *dev,
719 uint16_t periph_addr, uint8_t command,
720 uint16_t send_word, uint16_t *recv_word)
721 {
722 struct pch_data *data = dev->data;
723 uint8_t *p = (uint8_t *)recv_word;
724 int ret;
725
726 LOG_DBG("dev %p addr 0x%02x command 0x%02x", dev, periph_addr, command);
727
728 k_mutex_lock(&data->mutex, K_FOREVER);
729
730 ret = pch_smbus_start(dev, periph_addr, SMBUS_MSG_WRITE, command,
731 (uint8_t *)&send_word, SMBUS_CMD_PROC_CALL);
732 if (ret < 0) {
733 goto unlock;
734 }
735
736 /* Wait for completion from ISR */
737 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
738 if (ret != 0) {
739 LOG_ERR("SMBus Proc Call timed out");
740 ret = -ETIMEDOUT;
741 goto unlock;
742 }
743
744 ret = pch_check_status(dev);
745 if (ret < 0) {
746 goto unlock;
747 }
748
749 p[0] = pch_reg_read(dev, PCH_SMBUS_HD0);
750 p[1] = pch_reg_read(dev, PCH_SMBUS_HD1);
751
752 unlock:
753 k_mutex_unlock(&data->mutex);
754
755 return ret;
756 }
757
758 /* Implementation of SMBus Block Write */
pch_smbus_block_write(const struct device * dev,uint16_t periph_addr,uint8_t command,uint8_t count,uint8_t * buf)759 static int pch_smbus_block_write(const struct device *dev, uint16_t periph_addr,
760 uint8_t command, uint8_t count, uint8_t *buf)
761 {
762 struct pch_data *data = dev->data;
763 int ret;
764
765 LOG_DBG("dev %p addr 0x%02x command 0x%02x count %u",
766 dev, periph_addr, command, count);
767
768 k_mutex_lock(&data->mutex, K_FOREVER);
769
770 ret = pch_smbus_block_start(dev, periph_addr, SMBUS_MSG_WRITE, command,
771 count, buf, SMBUS_CMD_BLOCK);
772 if (ret < 0) {
773 goto unlock;
774 }
775
776 /* Wait for completion from ISR */
777 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
778 if (ret != 0) {
779 LOG_ERR("SMBus Block Write timed out");
780 ret = -ETIMEDOUT;
781 goto unlock;
782 }
783
784 ret = pch_check_status(dev);
785
786 unlock:
787 k_mutex_unlock(&data->mutex);
788
789 return ret;
790 }
791
792 /* Implementation of SMBus Block Read */
pch_smbus_block_read(const struct device * dev,uint16_t periph_addr,uint8_t command,uint8_t * count,uint8_t * buf)793 static int pch_smbus_block_read(const struct device *dev, uint16_t periph_addr,
794 uint8_t command, uint8_t *count, uint8_t *buf)
795 {
796 struct pch_data *data = dev->data;
797 int ret;
798
799 LOG_DBG("dev %p addr 0x%02x command 0x%02x",
800 dev, periph_addr, command);
801
802 k_mutex_lock(&data->mutex, K_FOREVER);
803
804 ret = pch_smbus_block_start(dev, periph_addr, SMBUS_MSG_READ, command,
805 0, buf, SMBUS_CMD_BLOCK);
806 if (ret < 0) {
807 goto unlock;
808 }
809
810 /* Wait for completion from ISR */
811 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
812 if (ret != 0) {
813 LOG_ERR("SMBus Block Read timed out");
814 ret = -ETIMEDOUT;
815 goto unlock;
816 }
817
818 ret = pch_check_status(dev);
819 if (ret < 0) {
820 goto unlock;
821 }
822
823 *count = pch_reg_read(dev, PCH_SMBUS_HD0);
824 if (*count == 0 || *count > SMBUS_BLOCK_BYTES_MAX) {
825 ret = -ENODATA;
826 goto unlock;
827 }
828
829 for (int i = 0; i < *count; i++) {
830 buf[i] = pch_reg_read(dev, PCH_SMBUS_HBD);
831 }
832
833 unlock:
834 k_mutex_unlock(&data->mutex);
835
836 return ret;
837 }
838
839 /* Implementation of SMBus Block Process Call */
pch_smbus_block_pcall(const struct device * dev,uint16_t periph_addr,uint8_t command,uint8_t send_count,uint8_t * send_buf,uint8_t * recv_count,uint8_t * recv_buf)840 static int pch_smbus_block_pcall(const struct device *dev,
841 uint16_t periph_addr, uint8_t command,
842 uint8_t send_count, uint8_t *send_buf,
843 uint8_t *recv_count, uint8_t *recv_buf)
844 {
845 struct pch_data *data = dev->data;
846 int ret;
847
848 LOG_DBG("dev %p addr 0x%02x command 0x%02x",
849 dev, periph_addr, command);
850
851 k_mutex_lock(&data->mutex, K_FOREVER);
852
853 ret = pch_smbus_block_start(dev, periph_addr, SMBUS_MSG_WRITE, command,
854 send_count, send_buf, SMBUS_CMD_BLOCK_PROC);
855 if (ret < 0) {
856 goto unlock;
857 }
858
859 /* Wait for completion from ISR */
860 ret = k_sem_take(&data->completion_sync, K_MSEC(30));
861 if (ret != 0) {
862 LOG_ERR("SMBus Block Process Call timed out");
863 ret = -ETIMEDOUT;
864 goto unlock;
865 }
866
867 ret = pch_check_status(dev);
868 if (ret < 0) {
869 goto unlock;
870 }
871
872 *recv_count = pch_reg_read(dev, PCH_SMBUS_HD0);
873 if (*recv_count == 0 ||
874 *recv_count + send_count > SMBUS_BLOCK_BYTES_MAX) {
875 ret = -ENODATA;
876 goto unlock;
877 }
878
879 for (int i = 0; i < *recv_count; i++) {
880 recv_buf[i] = pch_reg_read(dev, PCH_SMBUS_HBD);
881 }
882
883 unlock:
884 k_mutex_unlock(&data->mutex);
885
886 return ret;
887 }
888
889 static DEVICE_API(smbus, funcs) = {
890 .configure = pch_configure,
891 .get_config = pch_get_config,
892 .smbus_quick = pch_smbus_quick,
893 .smbus_byte_write = pch_smbus_byte_write,
894 .smbus_byte_read = pch_smbus_byte_read,
895 .smbus_byte_data_write = pch_smbus_byte_data_write,
896 .smbus_byte_data_read = pch_smbus_byte_data_read,
897 .smbus_word_data_write = pch_smbus_word_data_write,
898 .smbus_word_data_read = pch_smbus_word_data_read,
899 .smbus_pcall = pch_smbus_pcall,
900 .smbus_block_write = pch_smbus_block_write,
901 .smbus_block_read = pch_smbus_block_read,
902 .smbus_block_pcall = pch_smbus_block_pcall,
903 #if defined(CONFIG_SMBUS_INTEL_PCH_SMBALERT)
904 .smbus_smbalert_set_cb = pch_smbus_smbalert_set_sb,
905 .smbus_smbalert_remove_cb = pch_smbus_smbalert_remove_sb,
906 #endif /* CONFIG_SMBUS_INTEL_PCH_SMBALERT */
907 #if defined(CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY)
908 .smbus_host_notify_set_cb = pch_smbus_host_notify_set_cb,
909 .smbus_host_notify_remove_cb = pch_smbus_host_notify_remove_cb,
910 #endif /* CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY */
911 };
912
smbus_isr(const struct device * dev)913 static void smbus_isr(const struct device *dev)
914 {
915 const struct pch_config * const config = dev->config;
916 struct pch_data *data = dev->data;
917 uint32_t sts;
918 uint8_t status;
919
920 sts = pcie_conf_read(config->pcie->bdf, PCIE_CONF_CMDSTAT);
921 if (!(sts & PCIE_CONF_CMDSTAT_INTERRUPT)) {
922 LOG_ERR("Not our interrupt");
923 return;
924 }
925
926 /**
927 * Handle first Host Notify since for that we need to read SSTS
928 * register and for all other sources HSTS.
929 *
930 * Intel PCH implements Host Notify protocol in hardware.
931 */
932 #if defined(CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY)
933 if (data->config & SMBUS_MODE_HOST_NOTIFY) {
934 status = pch_reg_read(dev, PCH_SMBUS_SSTS);
935 if (status & PCH_SMBUS_SSTS_HNS) {
936 /* Notify address */
937 data->notify_addr =
938 pch_reg_read(dev, PCH_SMBUS_NDA) >> 1;
939
940 /* Notify data */
941 data->notify_data = pch_reg_read(dev, PCH_SMBUS_NDLB);
942 data->notify_data |=
943 pch_reg_read(dev, PCH_SMBUS_NDHB) << 8;
944
945 k_work_submit(&data->host_notify_work);
946
947 /* Clear Host Notify */
948 pch_reg_write(dev, PCH_SMBUS_SSTS, PCH_SMBUS_SSTS_HNS);
949
950 return;
951 }
952 }
953 #endif /* CONFIG_SMBUS_INTEL_PCH_HOST_NOTIFY */
954
955 status = pch_reg_read(dev, PCH_SMBUS_HSTS);
956
957 /* HSTS dump if logging is enabled */
958 pch_dump_register_hsts(status);
959
960 if (status & PCH_SMBUS_HSTS_BYTE_DONE) {
961 LOG_WRN("BYTE_DONE interrupt is not used");
962 }
963
964 /* Handle SMBALERT# signal */
965 #if defined(CONFIG_SMBUS_INTEL_PCH_SMBALERT)
966 if (data->config & SMBUS_MODE_SMBALERT &&
967 status & PCH_SMBUS_HSTS_SMB_ALERT) {
968 k_work_submit(&data->smb_alert_work);
969 }
970 #endif /* CONFIG_SMBUS_INTEL_PCH_SMBALERT */
971
972 /* Clear IRQ sources */
973 pch_reg_write(dev, PCH_SMBUS_HSTS, status);
974
975 data->status = status;
976
977 k_sem_give(&data->completion_sync);
978 }
979
980 /* Device macro initialization / DTS hackery */
981
982 #define SMBUS_PCH_IRQ_FLAGS(n) \
983 COND_CODE_1(DT_INST_IRQ_HAS_CELL(n, sense), \
984 (DT_INST_IRQ(n, sense)), \
985 (0))
986
987 #define SMBUS_IRQ_CONFIG(n) \
988 BUILD_ASSERT(IS_ENABLED(CONFIG_DYNAMIC_INTERRUPTS), \
989 "SMBus PCIe requires dynamic interrupts"); \
990 static void pch_config_##n(const struct device *dev) \
991 { \
992 const struct pch_config * const config = dev->config; \
993 unsigned int irq; \
994 if (DT_INST_IRQN(n) == PCIE_IRQ_DETECT) { \
995 irq = pcie_alloc_irq(config->pcie->bdf); \
996 if (irq == PCIE_CONF_INTR_IRQ_NONE) { \
997 return; \
998 } \
999 } else { \
1000 irq = DT_INST_IRQN(n); \
1001 pcie_conf_write(config->pcie->bdf, \
1002 PCIE_CONF_INTR, irq); \
1003 } \
1004 pcie_connect_dynamic_irq(config->pcie->bdf, irq, \
1005 DT_INST_IRQ(n, priority), \
1006 (void (*)(const void *))smbus_isr, \
1007 DEVICE_DT_INST_GET(n), \
1008 SMBUS_PCH_IRQ_FLAGS(n)); \
1009 pcie_irq_enable(config->pcie->bdf, irq); \
1010 LOG_DBG("Configure irq %d", irq); \
1011 }
1012
1013 #define SMBUS_DEVICE_INIT(n) \
1014 DEVICE_PCIE_INST_DECLARE(n); \
1015 static void pch_config_##n(const struct device *dev); \
1016 static const struct pch_config pch_config_data_##n = { \
1017 DEVICE_PCIE_INST_INIT(n, pcie), \
1018 .config_func = pch_config_##n, \
1019 }; \
1020 static struct pch_data smbus_##n##_data; \
1021 SMBUS_DEVICE_DT_INST_DEFINE(n, pch_smbus_init, NULL, \
1022 &smbus_##n##_data, &pch_config_data_##n, \
1023 POST_KERNEL, CONFIG_SMBUS_INIT_PRIORITY, \
1024 &funcs); \
1025 SMBUS_IRQ_CONFIG(n);
1026
1027 DT_INST_FOREACH_STATUS_OKAY(SMBUS_DEVICE_INIT)
1028