1 /*
2  * Copyright (c) 2020 Nuvoton Technology Corporation.
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT nuvoton_npcx_host_sub
8 
9 /**
10  * @file
11  * @brief Nuvoton NPCX host sub modules driver
12  *
13  * This file contains the drivers of NPCX Host Sub-Modules that serve as an
14  * interface between the Host and Core domains. Please refer the block diagram.
15  *
16  *                                        +------------+
17  *                                        |   Serial   |---> TXD
18  *                                  +<--->|    Port    |<--- RXD
19  *                                  |     |            |<--> ...
20  *                                  |     +------------+
21  *                                  |     +------------+     |
22  *                +------------+    |<--->|  KBC & PM  |<--->|
23  *   eSPI_CLK --->|  eSPI Bus  |    |     |  Channels  |     |
24  *   eSPI_RST --->| Controller |    |     +------------+     |
25  * eSPI_IO3-0 <-->|            |<-->|     +------------+     |
26  *    eSPI_CS --->| (eSPI mode)|    |     |   Shared   |     |
27  * eSPI_ALERT <-->|            |    |<--->|   Memory   |<--->|
28  *                +------------+    |     +------------+     |
29  *                                  |     +------------+     |
30  *                                  |<--->|    MSWC    |<--->|
31  *                                  |     +------------+     |
32  *                                  |     +------------+     |
33  *                                  |     |    Core    |     |
34  *                                  |<--->|   to Host  |<--->|
35  *                                  |     |   Access   |     |
36  *                                  |     +------------+     |
37  *                                HMIB                       | Core Bus
38  *                     (Host Modules Internal Bus)           +------------
39  *
40  *
41  * For most of them, the Host can configure these modules via eSPI(Peripheral
42  * Channel)/LPC by accessing 'Configuration and Control register Set' which IO
43  * base address is 0x4E as default. (The table below illustrates structure of
44  * 'Configuration and Control Register Set') And the interrupts in core domain
45  * help handling any events from host side.
46  *
47  *   Index |     Configuration and Control Register Set
48  * --------|--------------------------------------------------+   Bank Select
49  *    07h  |      Logical Device Number Register (LDN)        |---------+
50  * --------|---------------------------------------------------         |
51  *  20-2Fh |        SuperI/O Configuration Registers          |         |
52  * ------------------------------------------------------------         |
53  * --------|---------------------------------------------------_        |
54  *    30h  |      Logical Device Control Register             | |_      |
55  * --------|--------------------------------------------------- | |_    |
56  *  60-63h |   I/O Space Configuration Registers              | | | |   |
57  * --------|--------------------------------------------------- | | |   |
58  *  70-71h |     Interrupt Configuration Registers            | | | |   |
59  * --------|--------------------------------------------------- | | |   |
60  *  73-74h | DMA Configuration Registers (No support in NPCX) | | | |   |
61  * --------|--------------------------------------------------- | | |<--+
62  *  F0-FFh | Special Logical Device Configuration Registers   | | | |
63  * --------|--------------------------------------------------- | | |
64  *           |--------------------------------------------------- | |
65  *             |--------------------------------------------------- |
66  *               |---------------------------------------------------
67  *
68  *
69  * This driver introduces six host sub-modules. It includes:
70  *
71  * 1. Keyboard and Mouse Controller (KBC) interface.
72  *   ● Intel 8051SL-compatible Host interface
73  *     — 8042 KBD standard interface (ports 60h, 64h)
74  *     — Legacy IRQ: IRQ1 (KBD) and IRQ12 (mouse) support
75  *   ● Configured by two logical devices: Keyboard and Mouse (LDN 0x06/0x05)
76  *
77  * 2. Power Management (PM) channels.
78  *   ● PM channel registers
79  *     — Command/Status register
80  *     — Data register
81  *       channel 1: legacy 62h, 66h; channel 2: legacy 68h, 6Ch;
82  *       channel 3: legacy 6Ah, 6Eh; channel 4: legacy 6Bh, 6Fh;
83  *   ● PM interrupt using:
84  *     — Serial IRQ
85  *     — SMI
86  *     — EC_SCI
87  *   ● Configured by four logical devices: PM1/2/3/4 (LDN 0x11/0x12/0x17/0x1E)
88  *
89  * 3. Shared Memory mechanism (SHM).
90  *   This module allows sharing of the on-chip RAM by both Core and the Host.
91  *   It also supports the following features:
92  *   ● Four Core/Host communication windows for direct RAM access
93  *   ● Eight Protection regions for each access window
94  *   ● Host IRQ and SMI generation
95  *   ● Port 80 debug support
96  *   ● Configured by one logical device: SHM (LDN 0x0F)
97  *
98  * 4. Core Access to Host Modules (C2H).
99  *   ● A interface to access module registers in host domain.
100  *     It enables the Core to access the registers in host domain (i.e., Host
101  *     Configuration, Serial Port, SHM, and MSWC), through HMIB.
102  *
103  * 5. Mobile System Wake-Up functions (MSWC).
104  *   It detects and handles wake-up events from various sources in the Host
105  *   modules and alerts the Core for better power consumption.
106  *
107  * 6. Serial Port (Legacy UART)
108  *   It provides UART functionality by supporting serial data communication
109  *   with a remote peripheral device or a modem.
110  *
111  * INCLUDE FILES: soc_host.h
112  *
113  */
114 
115 #include <assert.h>
116 #include <zephyr/drivers/espi.h>
117 #include <zephyr/drivers/gpio.h>
118 #include <zephyr/drivers/clock_control.h>
119 #include <zephyr/drivers/pinctrl.h>
120 #include <zephyr/kernel.h>
121 #include <zephyr/sys/ring_buffer.h>
122 #include <soc.h>
123 #include "espi_utils.h"
124 #include "soc_host.h"
125 #include "soc_espi.h"
126 #include "soc_miwu.h"
127 
128 #include <zephyr/logging/log.h>
129 #include <zephyr/irq.h>
130 LOG_MODULE_REGISTER(host_sub_npcx, CONFIG_ESPI_LOG_LEVEL);
131 
132 struct host_sub_npcx_config {
133 	/* host module instances */
134 	struct mswc_reg *const inst_mswc;
135 	struct shm_reg *const inst_shm;
136 	struct c2h_reg *const inst_c2h;
137 	struct kbc_reg *const inst_kbc;
138 	struct pmch_reg *const inst_pm_acpi;
139 	struct pmch_reg *const inst_pm_hcmd;
140 	/* clock configuration */
141 	const uint8_t clks_size;
142 	const struct npcx_clk_cfg *clks_list;
143 	/* mapping table between host access signals and wake-up input */
144 	struct npcx_wui host_acc_wui;
145 };
146 
147 struct host_sub_npcx_data {
148 	sys_slist_t *callbacks; /* pointer on the espi callback list */
149 	uint8_t plt_rst_asserted; /* current PLT_RST# status */
150 	uint8_t espi_rst_level; /* current ESPI_RST# status */
151 	const struct device *host_bus_dev; /* device for eSPI/LPC bus */
152 #ifdef CONFIG_ESPI_NPCX_PERIPHERAL_DEBUG_PORT_80_MULTI_BYTE
153 	struct ring_buf port80_ring_buf;
154 	uint8_t port80_data[CONFIG_ESPI_NPCX_PERIPHERAL_DEBUG_PORT_80_RING_BUF_SIZE];
155 	struct k_work work;
156 #endif
157 };
158 
159 struct npcx_dp80_buf {
160 	union {
161 		uint16_t offset_code_16;
162 		uint8_t offset_code[2];
163 	};
164 };
165 
166 static const struct npcx_clk_cfg host_dev_clk_cfg[] =
167 					NPCX_DT_CLK_CFG_ITEMS_LIST(0);
168 
169 struct host_sub_npcx_config host_sub_cfg = {
170 	.inst_mswc = (struct mswc_reg *)DT_INST_REG_ADDR_BY_NAME(0, mswc),
171 	.inst_shm = (struct shm_reg *)DT_INST_REG_ADDR_BY_NAME(0, shm),
172 	.inst_c2h = (struct c2h_reg *)DT_INST_REG_ADDR_BY_NAME(0, c2h),
173 	.inst_kbc = (struct kbc_reg *)DT_INST_REG_ADDR_BY_NAME(0, kbc),
174 	.inst_pm_acpi = (struct pmch_reg *)DT_INST_REG_ADDR_BY_NAME(0, pm_acpi),
175 	.inst_pm_hcmd = (struct pmch_reg *)DT_INST_REG_ADDR_BY_NAME(0, pm_hcmd),
176 	.host_acc_wui = NPCX_DT_WUI_ITEM_BY_NAME(0, host_acc_wui),
177 	.clks_size = ARRAY_SIZE(host_dev_clk_cfg),
178 	.clks_list = host_dev_clk_cfg,
179 };
180 
181 struct host_sub_npcx_data host_sub_data;
182 
183 /* Application shouldn't touch these flags in KBC status register directly */
184 #define NPCX_KBC_STS_MASK (BIT(NPCX_HIKMST_IBF) | BIT(NPCX_HIKMST_OBF) \
185 						| BIT(NPCX_HIKMST_A2))
186 
187 /* IO base address of EC Logical Device Configuration */
188 #define NPCX_EC_CFG_IO_ADDR 0x4E
189 
190 /* Timeout to wait for Core-to-Host transaction to be completed. */
191 #define NPCX_C2H_TRANSACTION_TIMEOUT_US 200
192 
193 /* Logical Device Number Assignments */
194 #define EC_CFG_LDN_SP    0x03
195 #define EC_CFG_LDN_MOUSE 0x05
196 #define EC_CFG_LDN_KBC   0x06
197 #define EC_CFG_LDN_SHM   0x0F
198 #define EC_CFG_LDN_ACPI  0x11 /* PM Channel 1 */
199 #define EC_CFG_LDN_HCMD  0x12 /* PM Channel 2 */
200 
201 /* Index of EC (4E/4F) Configuration Register */
202 #define EC_CFG_IDX_LDN            0x07
203 #define EC_CFG_IDX_CTRL           0x30
204 #define EC_CFG_IDX_DATA_IO_ADDR_H 0x60
205 #define EC_CFG_IDX_DATA_IO_ADDR_L 0x61
206 #define EC_CFG_IDX_CMD_IO_ADDR_H  0x62
207 #define EC_CFG_IDX_CMD_IO_ADDR_L  0x63
208 
209 /* LDN Activation Enable */
210 #define EC_CFG_IDX_CTRL_LDN_ENABLE 0x01
211 
212 /* Index of SuperI/O Control and Configuration Registers */
213 #define EC_CFG_IDX_SUPERIO_SIOCF9      0x29
214 #define EC_CFG_IDX_SUPERIO_SIOCF9_CKEN 2
215 
216 /* Index of Special Logical Device Configuration (Shared Memory Module) */
217 #define EC_CFG_IDX_SHM_CFG             0xF1
218 #define EC_CFG_IDX_SHM_WND1_ADDR_0     0xF4
219 #define EC_CFG_IDX_SHM_WND1_ADDR_1     0xF5
220 #define EC_CFG_IDX_SHM_WND1_ADDR_2     0xF6
221 #define EC_CFG_IDX_SHM_WND1_ADDR_3     0xF7
222 #define EC_CFG_IDX_SHM_WND2_ADDR_0     0xF8
223 #define EC_CFG_IDX_SHM_WND2_ADDR_1     0xF9
224 #define EC_CFG_IDX_SHM_WND2_ADDR_2     0xFA
225 #define EC_CFG_IDX_SHM_WND2_ADDR_3     0xFB
226 #define EC_CFG_IDX_SHM_DP80_ADDR_RANGE 0xFD
227 
228 /* Index of Special Logical Device Configuration (Serial Port/Host UART) */
229 #define EC_CFG_IDX_SP_CFG              0xF0
230 /* Enable selection of bank 2 and 3 for the Serial Port */
231 #define EC_CFG_IDX_SP_CFG_BK_SL_ENABLE 7
232 
233 /* Host sub-device local inline functions */
host_shd_mem_wnd_size_sl(uint32_t size)234 static inline uint8_t host_shd_mem_wnd_size_sl(uint32_t size)
235 {
236 	/* The minimum supported shared memory region size is 8 bytes */
237 	if (size <= 8U) {
238 		size = 8U;
239 	}
240 
241 	/* The maximum supported shared memory region size is 4K bytes */
242 	if (size >= 4096U) {
243 		size = 4096U;
244 	}
245 
246 	/*
247 	 * If window size is not a power-of-two, it is rounded-up to the next
248 	 * power-of-two value, and return value corresponds to RWINx_SIZE field.
249 	 */
250 	return (32 - __builtin_clz(size - 1U)) & 0xff;
251 }
252 
253 /* Host KBC sub-device local functions */
254 #if defined(CONFIG_ESPI_PERIPHERAL_8042_KBC)
host_kbc_ibf_isr(const void * arg)255 static void host_kbc_ibf_isr(const void *arg)
256 {
257 	ARG_UNUSED(arg);
258 	struct kbc_reg *const inst_kbc = host_sub_cfg.inst_kbc;
259 	struct espi_event evt = { ESPI_BUS_PERIPHERAL_NOTIFICATION,
260 		ESPI_PERIPHERAL_8042_KBC, ESPI_PERIPHERAL_NODATA
261 	};
262 	struct espi_evt_data_kbc *kbc_evt =
263 				(struct espi_evt_data_kbc *)&evt.evt_data;
264 
265 	/* KBC Input Buffer Full event */
266 	kbc_evt->evt = HOST_KBC_EVT_IBF;
267 	/* The data in KBC Input Buffer */
268 	kbc_evt->data = inst_kbc->HIKMDI;
269 	/*
270 	 * Indicates if the host sent a command or data.
271 	 * 0 = data
272 	 * 1 = Command.
273 	 */
274 	kbc_evt->type = IS_BIT_SET(inst_kbc->HIKMST, NPCX_HIKMST_A2);
275 
276 	LOG_DBG("%s: kbc data 0x%02x", __func__, evt.evt_data);
277 	espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev,
278 							evt);
279 }
280 
host_kbc_obe_isr(const void * arg)281 static void host_kbc_obe_isr(const void *arg)
282 {
283 	ARG_UNUSED(arg);
284 	struct kbc_reg *const inst_kbc = host_sub_cfg.inst_kbc;
285 	struct espi_event evt = { ESPI_BUS_PERIPHERAL_NOTIFICATION,
286 		ESPI_PERIPHERAL_8042_KBC, ESPI_PERIPHERAL_NODATA
287 	};
288 	struct espi_evt_data_kbc *kbc_evt =
289 				(struct espi_evt_data_kbc *)&evt.evt_data;
290 
291 	/* Disable KBC OBE interrupt first */
292 	inst_kbc->HICTRL &= ~BIT(NPCX_HICTRL_OBECIE);
293 
294 	LOG_DBG("%s: kbc status 0x%02x", __func__, inst_kbc->HIKMST);
295 
296 	/*
297 	 * Notify application that host already read out data. The application
298 	 * might need to clear status register via espi_api_lpc_write_request()
299 	 * with E8042_CLEAR_FLAG opcode in callback.
300 	 */
301 	kbc_evt->evt = HOST_KBC_EVT_OBE;
302 	kbc_evt->data = 0;
303 	kbc_evt->type = 0;
304 
305 	espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev,
306 							evt);
307 }
308 
host_kbc_init(void)309 static void host_kbc_init(void)
310 {
311 	struct kbc_reg *const inst_kbc = host_sub_cfg.inst_kbc;
312 
313 	/* Make sure the previous OBF and IRQ has been sent out. */
314 	k_busy_wait(4);
315 	/* Set FW_OBF to clear OBF flag in both STATUS and HIKMST to 0 */
316 	inst_kbc->HICTRL |= BIT(NPCX_HICTRL_FW_OBF);
317 	/* Ensure there is no OBF set in this period. */
318 	k_busy_wait(4);
319 
320 	/*
321 	 * Init KBC with:
322 	 * 1. Enable Input Buffer Full (IBF) core interrupt for Keyboard/mouse.
323 	 * 2. Enable Output Buffer Full Mouse(OBFM) SIRQ 12.
324 	 * 3. Enable Output Buffer Full Keyboard (OBFK) SIRQ 1.
325 	 */
326 	inst_kbc->HICTRL = BIT(NPCX_HICTRL_IBFCIE) | BIT(NPCX_HICTRL_OBFMIE)
327 						| BIT(NPCX_HICTRL_OBFKIE);
328 
329 	/* Configure SIRQ 1/12 type (level + high) */
330 	inst_kbc->HIIRQC = 0x00;
331 }
332 #endif
333 
334 /* Host ACPI sub-device local functions */
335 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_IO)
host_acpi_process_input_data(uint8_t data)336 static void host_acpi_process_input_data(uint8_t data)
337 {
338 	struct pmch_reg *const inst_acpi = host_sub_cfg.inst_pm_acpi;
339 	struct espi_event evt = {
340 		.evt_type = ESPI_BUS_PERIPHERAL_NOTIFICATION,
341 		.evt_details = ESPI_PERIPHERAL_HOST_IO,
342 		.evt_data = ESPI_PERIPHERAL_NODATA
343 	};
344 	struct espi_evt_data_acpi *acpi_evt =
345 				(struct espi_evt_data_acpi *)&evt.evt_data;
346 
347 	LOG_DBG("%s: acpi data 0x%02x", __func__, data);
348 
349 	/*
350 	 * Indicates if the host sent a command or data.
351 	 * 0 = data
352 	 * 1 = Command.
353 	 */
354 	acpi_evt->type = IS_BIT_SET(inst_acpi->HIPMST, NPCX_HIPMST_CMD);
355 	acpi_evt->data = data;
356 
357 	espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev,
358 							evt);
359 }
360 
host_acpi_init(void)361 static void host_acpi_init(void)
362 {
363 	struct pmch_reg *const inst_acpi = host_sub_cfg.inst_pm_acpi;
364 
365 	/* Use SMI/SCI postive polarity by default */
366 	inst_acpi->HIPMCTL &= ~BIT(NPCX_HIPMCTL_SCIPOL);
367 	inst_acpi->HIPMIC &= ~BIT(NPCX_HIPMIC_SMIPOL);
368 
369 	/* Set SMIB/SCIB bits to make sure SMI#/SCI# are driven to high */
370 	inst_acpi->HIPMIC |= BIT(NPCX_HIPMIC_SMIB) | BIT(NPCX_HIPMIC_SCIB);
371 
372 	/*
373 	 * Allow SMI#/SCI# generated from PM module.
374 	 * On eSPI bus, we suggest set VW value of SCI#/SMI# directly.
375 	 */
376 	inst_acpi->HIPMIE |= BIT(NPCX_HIPMIE_SCIE);
377 	inst_acpi->HIPMIE |= BIT(NPCX_HIPMIE_SMIE);
378 
379 	/*
380 	 * Init ACPI PM channel (Host IO) with:
381 	 * 1. Enable Input-Buffer Full (IBF) core interrupt.
382 	 * 2. BIT 7 must be 1.
383 	 */
384 	inst_acpi->HIPMCTL |= BIT(7) | BIT(NPCX_HIPMCTL_IBFIE);
385 }
386 #endif
387 
388 #if defined(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)
389 /* Host command argument and memory map buffers */
390 static uint8_t	shm_host_cmd[CONFIG_ESPI_NPCX_PERIPHERAL_HOST_CMD_PARAM_SIZE]
391 	__aligned(8);
392 /* Host command sub-device local functions */
host_hcmd_process_input_data(uint8_t data)393 static void host_hcmd_process_input_data(uint8_t data)
394 {
395 	struct espi_event evt = { ESPI_BUS_PERIPHERAL_NOTIFICATION,
396 		ESPI_PERIPHERAL_EC_HOST_CMD, ESPI_PERIPHERAL_NODATA
397 	};
398 
399 	evt.evt_data = data;
400 	LOG_DBG("%s: host cmd data 0x%02x", __func__, evt.evt_data);
401 	espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev,
402 							evt);
403 }
404 
host_hcmd_init(void)405 static void host_hcmd_init(void)
406 {
407 	struct pmch_reg *const inst_hcmd = host_sub_cfg.inst_pm_hcmd;
408 	struct shm_reg *const inst_shm = host_sub_cfg.inst_shm;
409 	uint32_t win_size = CONFIG_ESPI_NPCX_PERIPHERAL_HOST_CMD_PARAM_SIZE;
410 
411 	/* Don't stall SHM transactions */
412 	inst_shm->SHM_CTL &= ~0x40;
413 	/* Disable Window 1 protection */
414 	inst_shm->WIN1_WR_PROT = 0;
415 	inst_shm->WIN1_RD_PROT = 0;
416 
417 	/* Configure Win1 size for ec host command. */
418 	SET_FIELD(inst_shm->WIN_SIZE, NPCX_WIN_SIZE_RWIN1_SIZE_FIELD,
419 		host_shd_mem_wnd_size_sl(win_size));
420 	inst_shm->WIN_BASE1 = (uint32_t)shm_host_cmd;
421 
422 	/*
423 	 * Clear processing flag before enabling host's interrupts in case
424 	 * it's set by the other command during sysjump.
425 	 */
426 	inst_hcmd->HIPMST &= ~BIT(NPCX_HIPMST_F0);
427 
428 	/*
429 	 * Init Host Command PM channel (Host IO) with:
430 	 * 1. Enable Input-Buffer Full (IBF) core interrupt.
431 	 * 2. BIT 7 must be 1.
432 	 */
433 	inst_hcmd->HIPMCTL |= BIT(7) | BIT(NPCX_HIPMCTL_IBFIE);
434 }
435 #endif
436 
437 #if defined(CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION)
438 /* Host command argument and memory map buffers */
439 static uint8_t shm_acpi_mmap[CONFIG_ESPI_NPCX_PERIPHERAL_ACPI_SHD_MEM_SIZE]
440 	__aligned(8);
host_shared_mem_region_init(void)441 static void host_shared_mem_region_init(void)
442 {
443 	struct shm_reg *const inst_shm = host_sub_cfg.inst_shm;
444 	uint32_t win_size = CONFIG_ESPI_NPCX_PERIPHERAL_ACPI_SHD_MEM_SIZE;
445 
446 	/* Don't stall SHM transactions */
447 	inst_shm->SHM_CTL &= ~0x40;
448 	/* Disable Window 2 protection */
449 	inst_shm->WIN2_WR_PROT = 0;
450 	inst_shm->WIN2_RD_PROT = 0;
451 
452 	/* Configure Win2 size for ACPI shared mem region. */
453 	SET_FIELD(inst_shm->WIN_SIZE, NPCX_WIN_SIZE_RWIN2_SIZE_FIELD,
454 		host_shd_mem_wnd_size_sl(win_size));
455 	inst_shm->WIN_BASE2 = (uint32_t)shm_acpi_mmap;
456 	/* Enable write protect of Share memory window 2 */
457 	inst_shm->WIN2_WR_PROT = 0xFF;
458 
459 	/*
460 	 * TODO: Initialize shm_acpi_mmap buffer for host command flags. We
461 	 * might use EACPI_GET_SHARED_MEMORY in espi_api_lpc_read_request()
462 	 * instead of setting host command flags here directly.
463 	 */
464 }
465 #endif
466 
467 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_IO) || \
468 				defined(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)
469 /* Host pm (host io) sub-module isr function for all channels such as ACPI. */
host_pmch_ibf_isr(const void * arg)470 static void host_pmch_ibf_isr(const void *arg)
471 {
472 	ARG_UNUSED(arg);
473 	struct pmch_reg *const inst_acpi = host_sub_cfg.inst_pm_acpi;
474 	struct pmch_reg *const inst_hcmd = host_sub_cfg.inst_pm_hcmd;
475 	uint8_t in_data;
476 
477 	/* Host put data on input buffer of ACPI channel */
478 	if (IS_BIT_SET(inst_acpi->HIPMST, NPCX_HIPMST_IBF)) {
479 		/* Set processing flag before reading command byte */
480 		inst_acpi->HIPMST |= BIT(NPCX_HIPMST_F0);
481 		/* Read out input data and clear IBF pending bit */
482 		in_data = inst_acpi->HIPMDI;
483 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_IO)
484 		host_acpi_process_input_data(in_data);
485 #endif
486 	}
487 
488 	/* Host put data on input buffer of HOSTCMD channel */
489 	if (IS_BIT_SET(inst_hcmd->HIPMST, NPCX_HIPMST_IBF)) {
490 		/* Set processing flag before reading command byte */
491 		inst_hcmd->HIPMST |= BIT(NPCX_HIPMST_F0);
492 		/* Read out input data and clear IBF pending bit */
493 		in_data = inst_hcmd->HIPMDI;
494 #if defined(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)
495 		host_hcmd_process_input_data(in_data);
496 #endif
497 	}
498 }
499 #endif
500 
501 /* Host port80 sub-device local functions */
502 #if defined(CONFIG_ESPI_PERIPHERAL_DEBUG_PORT_80)
503 #if defined(CONFIG_ESPI_NPCX_PERIPHERAL_DEBUG_PORT_80_MULTI_BYTE)
host_port80_work_handler(struct k_work * item)504 static void host_port80_work_handler(struct k_work *item)
505 {
506 	uint32_t code = 0;
507 	struct host_sub_npcx_data *data = CONTAINER_OF(item, struct host_sub_npcx_data, work);
508 	struct ring_buf *rbuf = &data->port80_ring_buf;
509 	struct espi_event evt = {ESPI_BUS_PERIPHERAL_NOTIFICATION,
510 				 (ESPI_PERIPHERAL_INDEX_0 << 16) | ESPI_PERIPHERAL_DEBUG_PORT80,
511 				 ESPI_PERIPHERAL_NODATA};
512 
513 	while (!ring_buf_is_empty(rbuf)) {
514 		struct npcx_dp80_buf dp80_buf;
515 		uint8_t offset;
516 
517 		ring_buf_get(rbuf, &dp80_buf.offset_code[0], sizeof(dp80_buf.offset_code));
518 		offset = dp80_buf.offset_code[1];
519 		code |= dp80_buf.offset_code[0] << (8 * offset);
520 		if (ring_buf_is_empty(rbuf)) {
521 			evt.evt_data = code;
522 			espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev,
523 					    evt);
524 			break;
525 		}
526 		/* peek the offset of the next byte */
527 		ring_buf_peek(rbuf, &dp80_buf.offset_code[0], sizeof(dp80_buf.offset_code));
528 		offset = dp80_buf.offset_code[1];
529 		/*
530 		 * If the peeked next byte's offset is 0, it is the start of the new code.
531 		 * Pass the current code to the application layer to handle the Port80 code.
532 		 */
533 		if (offset == 0) {
534 			evt.evt_data = code;
535 			espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev,
536 					    evt);
537 			code = 0;
538 		}
539 	}
540 }
541 #endif
542 
host_port80_isr(const void * arg)543 static void host_port80_isr(const void *arg)
544 {
545 	ARG_UNUSED(arg);
546 	struct shm_reg *const inst_shm = host_sub_cfg.inst_shm;
547 	uint8_t status = inst_shm->DP80STS;
548 
549 #ifdef CONFIG_ESPI_NPCX_PERIPHERAL_DEBUG_PORT_80_MULTI_BYTE
550 	struct ring_buf *rbuf = &host_sub_data.port80_ring_buf;
551 
552 	while (IS_BIT_SET(inst_shm->DP80STS, NPCX_DP80STS_FNE)) {
553 		struct npcx_dp80_buf dp80_buf;
554 
555 		dp80_buf.offset_code_16 = inst_shm->DP80BUF;
556 		ring_buf_put(rbuf, &dp80_buf.offset_code[0], sizeof(dp80_buf.offset_code));
557 	}
558 	k_work_submit(&host_sub_data.work);
559 #else
560 	struct espi_event evt = {ESPI_BUS_PERIPHERAL_NOTIFICATION,
561 				 (ESPI_PERIPHERAL_INDEX_0 << 16) | ESPI_PERIPHERAL_DEBUG_PORT80,
562 				 ESPI_PERIPHERAL_NODATA};
563 
564 	/* Read out port80 data continuously if FIFO is not empty */
565 	while (IS_BIT_SET(inst_shm->DP80STS, NPCX_DP80STS_FNE)) {
566 		LOG_DBG("p80: %04x", inst_shm->DP80BUF);
567 		evt.evt_data = inst_shm->DP80BUF;
568 		espi_send_callbacks(host_sub_data.callbacks, host_sub_data.host_bus_dev, evt);
569 	}
570 #endif
571 	LOG_DBG("%s: p80 status 0x%02X", __func__, status);
572 
573 	/* If FIFO is overflow, show error msg */
574 	if (IS_BIT_SET(status, NPCX_DP80STS_FOR)) {
575 		inst_shm->DP80STS |= BIT(NPCX_DP80STS_FOR);
576 		LOG_DBG("Port80 FIFO Overflow!");
577 	}
578 
579 	/* If there are pending post codes remains in FIFO after processing and sending previous
580 	 * post codes, do not clear the FNE bit. This allows this handler to be called again
581 	 * immediately after it exists.
582 	 */
583 	if (!IS_BIT_SET(inst_shm->DP80STS, NPCX_DP80STS_FNE)) {
584 		/* Clear all pending bit indicates that FIFO was written by host */
585 		inst_shm->DP80STS |= BIT(NPCX_DP80STS_FWR);
586 	}
587 }
588 
host_port80_init(void)589 static void host_port80_init(void)
590 {
591 	struct shm_reg *const inst_shm = host_sub_cfg.inst_shm;
592 
593 	/*
594 	 * Init PORT80 which includes:
595 	 * Enables a Core interrupt on every Host write to the FIFO,
596 	 * SYNC mode (It must be 1 in eSPI mode), Read Auto Advance mode, and
597 	 * Port80 module itself.
598 	 */
599 	inst_shm->DP80CTL = BIT(NPCX_DP80CTL_CIEN) | BIT(NPCX_DP80CTL_RAA)
600 			| BIT(NPCX_DP80CTL_DP80EN) | BIT(NPCX_DP80CTL_SYNCEN);
601 }
602 #endif
603 
604 #if defined(CONFIG_ESPI_PERIPHERAL_CUSTOM_OPCODE)
host_cus_opcode_enable_interrupts(void)605 static void host_cus_opcode_enable_interrupts(void)
606 {
607 	/* Enable host KBC sub-device interrupt */
608 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_8042_KBC)) {
609 		irq_enable(DT_INST_IRQ_BY_NAME(0, kbc_ibf, irq));
610 		irq_enable(DT_INST_IRQ_BY_NAME(0, kbc_obe, irq));
611 	}
612 
613 	/* Enable host PM channel (Host IO) sub-device interrupt */
614 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_HOST_IO) ||
615 	    IS_ENABLED(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)) {
616 		irq_enable(DT_INST_IRQ_BY_NAME(0, pmch_ibf, irq));
617 	}
618 
619 	/* Enable host Port80 sub-device interrupt installation */
620 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_DEBUG_PORT_80)) {
621 		irq_enable(DT_INST_IRQ_BY_NAME(0, p80_fifo, irq));
622 	}
623 
624 	/* Enable host interface interrupts if its interface is eSPI */
625 	if (IS_ENABLED(CONFIG_ESPI)) {
626 		npcx_espi_enable_interrupts(host_sub_data.host_bus_dev);
627 	}
628 }
629 
host_cus_opcode_disable_interrupts(void)630 static void host_cus_opcode_disable_interrupts(void)
631 {
632 	/* Disable host KBC sub-device interrupt */
633 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_8042_KBC)) {
634 		irq_disable(DT_INST_IRQ_BY_NAME(0, kbc_ibf, irq));
635 		irq_disable(DT_INST_IRQ_BY_NAME(0, kbc_obe, irq));
636 	}
637 
638 	/* Disable host PM channel (Host IO) sub-device interrupt */
639 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_HOST_IO) ||
640 		IS_ENABLED(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)) {
641 		irq_disable(DT_INST_IRQ_BY_NAME(0, pmch_ibf, irq));
642 	}
643 
644 	/* Disable host Port80 sub-device interrupt installation */
645 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_DEBUG_PORT_80)) {
646 		irq_disable(DT_INST_IRQ_BY_NAME(0, p80_fifo, irq));
647 	}
648 
649 	/* Disable host interface interrupts if its interface is eSPI */
650 	if (IS_ENABLED(CONFIG_ESPI)) {
651 		npcx_espi_disable_interrupts(host_sub_data.host_bus_dev);
652 	}
653 }
654 #endif /* CONFIG_ESPI_PERIPHERAL_CUSTOM_OPCODE */
655 
656 #if defined(CONFIG_ESPI_PERIPHERAL_UART)
657 /* host uart pinmux configuration */
658 PINCTRL_DT_DEFINE(DT_INST(0, nuvoton_npcx_host_uart));
659 BUILD_ASSERT(DT_NUM_INST_STATUS_OKAY(nuvoton_npcx_host_uart) == 1,
660 	"only one 'nuvoton_npcx_host_uart' compatible node may be present");
661 const struct pinctrl_dev_config *huart_cfg =
662 			PINCTRL_DT_DEV_CONFIG_GET(DT_INST(0, nuvoton_npcx_host_uart));
663 /* Host UART sub-device local functions */
host_uart_init(void)664 void host_uart_init(void)
665 {
666 	struct c2h_reg *const inst_c2h = host_sub_cfg.inst_c2h;
667 
668 	/* Configure pin-mux for serial port device */
669 	pinctrl_apply_state(huart_cfg, PINCTRL_STATE_DEFAULT);
670 
671 	/* Make sure unlock host access of serial port */
672 	inst_c2h->LKSIOHA &= ~BIT(NPCX_LKSIOHA_LKSPHA);
673 	/* Clear 'Host lock violation occurred' bit of serial port initially */
674 	inst_c2h->SIOLV |= BIT(NPCX_SIOLV_SPLV);
675 }
676 #endif
677 
678 /* host core-to-host interface local functions */
host_c2h_wait_write_done(void)679 static void host_c2h_wait_write_done(void)
680 {
681 	struct c2h_reg *const inst_c2h = host_sub_cfg.inst_c2h;
682 	uint32_t elapsed_cycles;
683 	uint32_t start_cycles = k_cycle_get_32();
684 	uint32_t max_wait_cycles =
685 			k_us_to_cyc_ceil32(NPCX_C2H_TRANSACTION_TIMEOUT_US);
686 
687 	while (IS_BIT_SET(inst_c2h->SIBCTRL, NPCX_SIBCTRL_CSWR)) {
688 		elapsed_cycles = k_cycle_get_32() - start_cycles;
689 		if (elapsed_cycles > max_wait_cycles) {
690 			LOG_ERR("c2h write transaction expired!");
691 			break;
692 		}
693 	}
694 }
695 
host_c2h_wait_read_done(void)696 static void host_c2h_wait_read_done(void)
697 {
698 	struct c2h_reg *const inst_c2h = host_sub_cfg.inst_c2h;
699 	uint32_t elapsed_cycles;
700 	uint32_t start_cycles = k_cycle_get_32();
701 	uint32_t max_wait_cycles =
702 			k_us_to_cyc_ceil32(NPCX_C2H_TRANSACTION_TIMEOUT_US);
703 
704 	while (IS_BIT_SET(inst_c2h->SIBCTRL, NPCX_SIBCTRL_CSRD)) {
705 		elapsed_cycles = k_cycle_get_32() - start_cycles;
706 		if (elapsed_cycles > max_wait_cycles) {
707 			LOG_ERR("c2h read transaction expired!");
708 			break;
709 		}
710 	}
711 }
712 
host_c2h_write_io_cfg_reg(uint8_t reg_index,uint8_t reg_data)713 void host_c2h_write_io_cfg_reg(uint8_t reg_index, uint8_t reg_data)
714 {
715 	struct c2h_reg *const inst_c2h = host_sub_cfg.inst_c2h;
716 
717 	/* Disable interrupts */
718 	unsigned int key = irq_lock();
719 
720 	/* Lock host access EC configuration registers (0x4E/0x4F) */
721 	inst_c2h->LKSIOHA |= BIT(NPCX_LKSIOHA_LKCFG);
722 	/* Enable Core-to-Host access CFG module */
723 	inst_c2h->CRSMAE |= BIT(NPCX_CRSMAE_CFGAE);
724 
725 	/* Verify core-to-host modules is not in progress */
726 	host_c2h_wait_read_done();
727 	host_c2h_wait_write_done();
728 
729 	/*
730 	 * Specifying the in-direct IO address which A0 = 0 indicates the index
731 	 * register is accessed. Then write index address directly and it starts
732 	 * a write transaction to host sub-module on LPC/eSPI bus.
733 	 */
734 	inst_c2h->IHIOA = NPCX_EC_CFG_IO_ADDR;
735 	inst_c2h->IHD = reg_index;
736 	host_c2h_wait_write_done();
737 
738 	/*
739 	 * Specifying the in-direct IO address which A0 = 1 indicates the data
740 	 * register is accessed. Then write data directly and it starts a write
741 	 * transaction to host sub-module on LPC/eSPI bus.
742 	 */
743 	inst_c2h->IHIOA = NPCX_EC_CFG_IO_ADDR + 1;
744 	inst_c2h->IHD = reg_data;
745 	host_c2h_wait_write_done();
746 
747 	/* Disable Core-to-Host access CFG module */
748 	inst_c2h->CRSMAE &= ~BIT(NPCX_CRSMAE_CFGAE);
749 	/* Unlock host access EC configuration registers (0x4E/0x4F) */
750 	inst_c2h->LKSIOHA &= ~BIT(NPCX_LKSIOHA_LKCFG);
751 
752 	/* Enable interrupts */
753 	irq_unlock(key);
754 }
755 
host_c2h_read_io_cfg_reg(uint8_t reg_index)756 uint8_t host_c2h_read_io_cfg_reg(uint8_t reg_index)
757 {
758 	struct c2h_reg *const inst_c2h = host_sub_cfg.inst_c2h;
759 	uint8_t data_val;
760 
761 	/* Disable interrupts */
762 	unsigned int key = irq_lock();
763 
764 	/* Lock host access EC configuration registers (0x4E/0x4F) */
765 	inst_c2h->LKSIOHA |= BIT(NPCX_LKSIOHA_LKCFG);
766 	/* Enable Core-to-Host access CFG module */
767 	inst_c2h->CRSMAE |= BIT(NPCX_CRSMAE_CFGAE);
768 
769 	/* Verify core-to-host modules is not in progress */
770 	host_c2h_wait_read_done();
771 	host_c2h_wait_write_done();
772 
773 	/*
774 	 * Specifying the in-direct IO address which A0 = 0 indicates the index
775 	 * register is accessed. Then write index address directly and it starts
776 	 * a write transaction to host sub-module on LPC/eSPI bus.
777 	 */
778 	inst_c2h->IHIOA = NPCX_EC_CFG_IO_ADDR;
779 	inst_c2h->IHD = reg_index;
780 	host_c2h_wait_write_done();
781 
782 	/*
783 	 * Specifying the in-direct IO address which A0 = 1 indicates the data
784 	 * register is accessed. Then write CSRD bit in SIBCTRL to issue a read
785 	 * transaction to host sub-module on LPC/eSPI bus. Once it was done,
786 	 * read data out from IHD.
787 	 */
788 	inst_c2h->IHIOA = NPCX_EC_CFG_IO_ADDR + 1;
789 	inst_c2h->SIBCTRL |= BIT(NPCX_SIBCTRL_CSRD);
790 	host_c2h_wait_read_done();
791 	data_val = inst_c2h->IHD;
792 
793 	/* Disable Core-to-Host access CFG module */
794 	inst_c2h->CRSMAE &= ~BIT(NPCX_CRSMAE_CFGAE);
795 	/* Unlock host access EC configuration registers (0x4E/0x4F) */
796 	inst_c2h->LKSIOHA &= ~BIT(NPCX_LKSIOHA_LKCFG);
797 
798 	/* Enable interrupts */
799 	irq_unlock(key);
800 
801 	return data_val;
802 }
803 
804 /* Platform specific host sub modules functions */
npcx_host_periph_read_request(enum lpc_peripheral_opcode op,uint32_t * data)805 int npcx_host_periph_read_request(enum lpc_peripheral_opcode op,
806 								uint32_t *data)
807 {
808 	if (op >= E8042_START_OPCODE && op <= E8042_MAX_OPCODE) {
809 		struct kbc_reg *const inst_kbc = host_sub_cfg.inst_kbc;
810 
811 		/* Make sure kbc 8042 is on */
812 		if (!IS_BIT_SET(inst_kbc->HICTRL, NPCX_HICTRL_OBFKIE) ||
813 			!IS_BIT_SET(inst_kbc->HICTRL, NPCX_HICTRL_OBFMIE)) {
814 			return -ENOTSUP;
815 		}
816 
817 		switch (op) {
818 		case E8042_OBF_HAS_CHAR:
819 			/* EC has written data back to host. OBF is
820 			 * automatically cleared after host reads
821 			 * the data
822 			 */
823 			*data = IS_BIT_SET(inst_kbc->HIKMST, NPCX_HIKMST_OBF);
824 			break;
825 		case E8042_IBF_HAS_CHAR:
826 			*data = IS_BIT_SET(inst_kbc->HIKMST, NPCX_HIKMST_IBF);
827 			break;
828 		case E8042_READ_KB_STS:
829 			*data = inst_kbc->HIKMST;
830 			break;
831 		default:
832 			return -EINVAL;
833 		}
834 	} else if (op >= EACPI_START_OPCODE && op <= EACPI_MAX_OPCODE) {
835 		struct pmch_reg *const inst_acpi = host_sub_cfg.inst_pm_acpi;
836 
837 		/* Make sure pm channel for apci is on */
838 		if (!IS_BIT_SET(inst_acpi->HIPMCTL, NPCX_HIPMCTL_IBFIE)) {
839 			return -ENOTSUP;
840 		}
841 
842 		switch (op) {
843 		case EACPI_OBF_HAS_CHAR:
844 			/* EC has written data back to host. OBF is
845 			 * automatically cleared after host reads
846 			 * the data
847 			 */
848 			*data = IS_BIT_SET(inst_acpi->HIPMST, NPCX_HIPMST_OBF);
849 			break;
850 		case EACPI_IBF_HAS_CHAR:
851 			*data = IS_BIT_SET(inst_acpi->HIPMST, NPCX_HIPMST_IBF);
852 			break;
853 		case EACPI_READ_STS:
854 			*data = inst_acpi->HIPMST;
855 			break;
856 #if defined(CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION)
857 		case EACPI_GET_SHARED_MEMORY:
858 			*data = (uint32_t)shm_acpi_mmap;
859 			break;
860 #endif /* CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION */
861 		default:
862 			return -EINVAL;
863 		}
864 	}
865 #if defined(CONFIG_ESPI_PERIPHERAL_CUSTOM_OPCODE)
866 	else if (op >= ECUSTOM_START_OPCODE && op <= ECUSTOM_MAX_OPCODE) {
867 		/* Other customized op codes */
868 		switch (op) {
869 		case ECUSTOM_HOST_CMD_GET_PARAM_MEMORY:
870 			*data = (uint32_t)shm_host_cmd;
871 			break;
872 		case ECUSTOM_HOST_CMD_GET_PARAM_MEMORY_SIZE:
873 			*data = CONFIG_ESPI_NPCX_PERIPHERAL_HOST_CMD_PARAM_SIZE;
874 			break;
875 		default:
876 			return -EINVAL;
877 		}
878 	}
879 #endif /* CONFIG_ESPI_PERIPHERAL_CUSTOM_OPCODE */
880 	else {
881 		return -ENOTSUP;
882 	}
883 
884 	return 0;
885 }
886 
npcx_host_periph_write_request(enum lpc_peripheral_opcode op,const uint32_t * data)887 int npcx_host_periph_write_request(enum lpc_peripheral_opcode op,
888 							const uint32_t *data)
889 {
890 	volatile uint32_t __attribute__((unused)) dummy;
891 	struct kbc_reg *const inst_kbc = host_sub_cfg.inst_kbc;
892 
893 	if (op >= E8042_START_OPCODE && op <= E8042_MAX_OPCODE) {
894 		/* Make sure kbc 8042 is on */
895 		if (!IS_BIT_SET(inst_kbc->HICTRL, NPCX_HICTRL_OBFKIE) ||
896 			!IS_BIT_SET(inst_kbc->HICTRL, NPCX_HICTRL_OBFMIE)) {
897 			return -ENOTSUP;
898 		}
899 		if (data) {
900 			LOG_DBG("op 0x%x data %x", op, *data);
901 		} else {
902 			LOG_DBG("op 0x%x only", op);
903 		}
904 
905 		switch (op) {
906 		case E8042_WRITE_KB_CHAR:
907 			inst_kbc->HIKDO = *data & 0xff;
908 			/*
909 			 * Enable KBC OBE interrupt after putting data in
910 			 * keyboard data register.
911 			 */
912 			inst_kbc->HICTRL |= BIT(NPCX_HICTRL_OBECIE);
913 			break;
914 		case E8042_WRITE_MB_CHAR:
915 			inst_kbc->HIMDO = *data & 0xff;
916 			/*
917 			 * Enable KBC OBE interrupt after putting data in
918 			 * mouse data register.
919 			 */
920 			inst_kbc->HICTRL |= BIT(NPCX_HICTRL_OBECIE);
921 			break;
922 		case E8042_RESUME_IRQ:
923 			/* Enable KBC IBF interrupt */
924 			inst_kbc->HICTRL |= BIT(NPCX_HICTRL_IBFCIE);
925 			break;
926 		case E8042_PAUSE_IRQ:
927 			/* Disable KBC IBF interrupt */
928 			inst_kbc->HICTRL &= ~BIT(NPCX_HICTRL_IBFCIE);
929 			break;
930 		case E8042_CLEAR_OBF:
931 			/* Clear OBF flag in both STATUS and HIKMST to 0 */
932 			inst_kbc->HICTRL |= BIT(NPCX_HICTRL_FW_OBF);
933 			break;
934 		case E8042_SET_FLAG:
935 			/* FW shouldn't modify these flags directly */
936 			inst_kbc->HIKMST |= *data & ~NPCX_KBC_STS_MASK;
937 			break;
938 		case E8042_CLEAR_FLAG:
939 			/* FW shouldn't modify these flags directly */
940 			inst_kbc->HIKMST &= ~(*data | NPCX_KBC_STS_MASK);
941 			break;
942 		default:
943 			return -EINVAL;
944 		}
945 	} else if (op >= EACPI_START_OPCODE && op <= EACPI_MAX_OPCODE) {
946 		struct pmch_reg *const inst_acpi = host_sub_cfg.inst_pm_acpi;
947 
948 		/* Make sure pm channel for apci is on */
949 		if (!IS_BIT_SET(inst_acpi->HIPMCTL, NPCX_HIPMCTL_IBFIE)) {
950 			return -ENOTSUP;
951 		}
952 
953 		switch (op) {
954 		case EACPI_WRITE_CHAR:
955 			inst_acpi->HIPMDO = (*data & 0xff);
956 			break;
957 		case EACPI_WRITE_STS:
958 			inst_acpi->HIPMST = (*data & 0xff);
959 			break;
960 		default:
961 			return -EINVAL;
962 		}
963 	}
964 #if defined(CONFIG_ESPI_PERIPHERAL_CUSTOM_OPCODE)
965 	else if (op >= ECUSTOM_START_OPCODE && op <= ECUSTOM_MAX_OPCODE) {
966 		/* Other customized op codes */
967 		struct pmch_reg *const inst_hcmd = host_sub_cfg.inst_pm_hcmd;
968 
969 		switch (op) {
970 		case ECUSTOM_HOST_SUBS_INTERRUPT_EN:
971 			if (*data != 0) {
972 				host_cus_opcode_enable_interrupts();
973 			} else {
974 				host_cus_opcode_disable_interrupts();
975 			}
976 			break;
977 		case ECUSTOM_HOST_CMD_SEND_RESULT:
978 			/*
979 			 * Write result to the data byte.  This sets the TOH
980 			 * status bit.
981 			 */
982 			inst_hcmd->HIPMDO = (*data & 0xff);
983 			/* Clear processing flag */
984 			inst_hcmd->HIPMST &= ~BIT(NPCX_HIPMST_F0);
985 			break;
986 		default:
987 			return -EINVAL;
988 		}
989 	}
990 #endif /* CONFIG_ESPI_PERIPHERAL_CUSTOM_OPCODE */
991 	else {
992 		return -ENOTSUP;
993 	}
994 
995 	return 0;
996 }
997 
npcx_host_init_subs_host_domain(void)998 void npcx_host_init_subs_host_domain(void)
999 {
1000 	struct c2h_reg *const inst_c2h = host_sub_cfg.inst_c2h;
1001 
1002 	/* Enable Core-to-Host access module */
1003 	inst_c2h->SIBCTRL |= BIT(NPCX_SIBCTRL_CSAE);
1004 
1005 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_8042_KBC)) {
1006 		/*
1007 		 * Select Keyboard/Mouse banks which LDN are 0x06/05 and enable
1008 		 * modules by setting bit 0 in its Control (index is 0x30) reg.
1009 		 */
1010 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_LDN, EC_CFG_LDN_KBC);
1011 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CTRL, EC_CFG_IDX_CTRL_LDN_ENABLE);
1012 
1013 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_LDN, EC_CFG_LDN_MOUSE);
1014 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CTRL, EC_CFG_IDX_CTRL_LDN_ENABLE);
1015 	}
1016 
1017 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_HOST_IO)) {
1018 		/*
1019 		 * Select ACPI bank which LDN are 0x11 (PM Channel 1) and enable
1020 		 * module by setting bit 0 in its Control (index is 0x30) reg.
1021 		 */
1022 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_LDN, EC_CFG_LDN_ACPI);
1023 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CTRL, EC_CFG_IDX_CTRL_LDN_ENABLE);
1024 	}
1025 
1026 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD) ||
1027 	    IS_ENABLED(CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION)) {
1028 		/* Select 'Host Command' bank which LDN are 0x12 (PM chan 2) */
1029 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_LDN, EC_CFG_LDN_HCMD);
1030 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_CMD_DATA_PORT_NUM)
1031 		/* Configure IO address of CMD portt (default: 0x200) */
1032 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_DATA_IO_ADDR_H,
1033 		 (CONFIG_ESPI_PERIPHERAL_HOST_CMD_DATA_PORT_NUM >> 8) & 0xff);
1034 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_DATA_IO_ADDR_L,
1035 		 CONFIG_ESPI_PERIPHERAL_HOST_CMD_DATA_PORT_NUM & 0xff);
1036 		/* Configure IO address of Data portt (default: 0x204) */
1037 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CMD_IO_ADDR_H,
1038 		 ((CONFIG_ESPI_PERIPHERAL_HOST_CMD_DATA_PORT_NUM + 4) >> 8)
1039 		 & 0xff);
1040 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CMD_IO_ADDR_L,
1041 		 (CONFIG_ESPI_PERIPHERAL_HOST_CMD_DATA_PORT_NUM + 4) & 0xff);
1042 #endif
1043 		/* Enable 'Host Command' io port (PM Channel 2) */
1044 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CTRL, EC_CFG_IDX_CTRL_LDN_ENABLE);
1045 
1046 		/* Select 'Shared Memory' bank which LDN are 0x0F */
1047 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_LDN, EC_CFG_LDN_SHM);
1048 		/* WIN 1 & 2 mapping to IO space */
1049 		host_c2h_write_io_cfg_reg(0xF1,
1050 				host_c2h_read_io_cfg_reg(0xF1) | 0x30);
1051 		/* WIN1 as Host Command on the IO address (default: 0x0800) */
1052 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_CMD_PARAM_PORT_NUM)
1053 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_SHM_WND1_ADDR_1,
1054 		(CONFIG_ESPI_PERIPHERAL_HOST_CMD_PARAM_PORT_NUM >> 8) & 0xff);
1055 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_SHM_WND1_ADDR_0,
1056 		CONFIG_ESPI_PERIPHERAL_HOST_CMD_PARAM_PORT_NUM & 0xff);
1057 #endif
1058 		/* Set WIN2 as MEMMAP on the configured IO address */
1059 #if defined(CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION_PORT_NUM)
1060 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_SHM_WND2_ADDR_1,
1061 		(CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION_PORT_NUM >> 8) & 0xff);
1062 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_SHM_WND2_ADDR_0,
1063 		CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION_PORT_NUM & 0xff);
1064 #endif
1065 		if (IS_ENABLED(CONFIG_ESPI_NPCX_PERIPHERAL_DEBUG_PORT_80_MULTI_BYTE)) {
1066 			host_c2h_write_io_cfg_reg(EC_CFG_IDX_SHM_DP80_ADDR_RANGE, 0x0f);
1067 		}
1068 	/* Enable SHM direct memory access */
1069 	host_c2h_write_io_cfg_reg(EC_CFG_IDX_CTRL, EC_CFG_IDX_CTRL_LDN_ENABLE);
1070 	}
1071 
1072 	if (IS_ENABLED(CONFIG_ESPI_PERIPHERAL_UART)) {
1073 		/* Select Serial Port banks which LDN are 0x03. */
1074 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_LDN, EC_CFG_LDN_SP);
1075 		/* Enable SIO_CLK */
1076 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_SUPERIO_SIOCF9,
1077 					  host_c2h_read_io_cfg_reg(EC_CFG_IDX_SUPERIO_SIOCF9) |
1078 						  BIT(EC_CFG_IDX_SUPERIO_SIOCF9_CKEN));
1079 		/* Enable Bank Select */
1080 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_SP_CFG,
1081 					  host_c2h_read_io_cfg_reg(EC_CFG_IDX_SP_CFG) |
1082 						  BIT(EC_CFG_IDX_SP_CFG_BK_SL_ENABLE));
1083 		host_c2h_write_io_cfg_reg(EC_CFG_IDX_CTRL, EC_CFG_IDX_CTRL_LDN_ENABLE);
1084 	}
1085 
1086 	LOG_DBG("Hos sub-modules configurations are done!");
1087 }
1088 
npcx_host_enable_access_interrupt(void)1089 void npcx_host_enable_access_interrupt(void)
1090 {
1091 	npcx_miwu_irq_get_and_clear_pending(&host_sub_cfg.host_acc_wui);
1092 
1093 	npcx_miwu_irq_enable(&host_sub_cfg.host_acc_wui);
1094 }
1095 
npcx_host_disable_access_interrupt(void)1096 void npcx_host_disable_access_interrupt(void)
1097 {
1098 	npcx_miwu_irq_disable(&host_sub_cfg.host_acc_wui);
1099 }
1100 
npcx_host_init_subs_core_domain(const struct device * host_bus_dev,sys_slist_t * callbacks)1101 int npcx_host_init_subs_core_domain(const struct device *host_bus_dev,
1102 							sys_slist_t *callbacks)
1103 {
1104 	struct mswc_reg *const inst_mswc = host_sub_cfg.inst_mswc;
1105 	struct shm_reg *const inst_shm = host_sub_cfg.inst_shm;
1106 	const struct device *const clk_dev = DEVICE_DT_GET(NPCX_CLK_CTRL_NODE);
1107 	int i;
1108 	uint8_t shm_sts;
1109 
1110 	host_sub_data.callbacks = callbacks;
1111 	host_sub_data.host_bus_dev = host_bus_dev;
1112 
1113 	/* Turn on all host necessary sub-module clocks first */
1114 	for (i = 0; i < host_sub_cfg.clks_size; i++) {
1115 		int ret;
1116 
1117 		if (!device_is_ready(clk_dev)) {
1118 			return -ENODEV;
1119 		}
1120 
1121 		ret = clock_control_on(clk_dev, (clock_control_subsys_t)
1122 				&host_sub_cfg.clks_list[i]);
1123 		if (ret < 0) {
1124 			return ret;
1125 		}
1126 	}
1127 
1128 	/* Configure EC legacy configuration IO base address to 0x4E. */
1129 	if (!IS_BIT_SET(inst_mswc->MSWCTL1, NPCX_MSWCTL1_VHCFGA)) {
1130 		inst_mswc->HCBAL = NPCX_EC_CFG_IO_ADDR;
1131 		inst_mswc->HCBAH = 0x0;
1132 	}
1133 
1134 	/*
1135 	 * Set HOSTWAIT bit and avoid the other settings, then host can freely
1136 	 * communicate with slave (EC).
1137 	 */
1138 	inst_shm->SMC_CTL &= BIT(NPCX_SMC_CTL_HOSTWAIT);
1139 	/* Clear shared memory status */
1140 	shm_sts = inst_shm->SMC_STS;
1141 	inst_shm->SMC_STS = shm_sts;
1142 
1143 	/* host sub-module initialization in core domain */
1144 #if defined(CONFIG_ESPI_PERIPHERAL_8042_KBC)
1145 	host_kbc_init();
1146 #endif
1147 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_IO)
1148 	host_acpi_init();
1149 #endif
1150 #if defined(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)
1151 	host_hcmd_init();
1152 #endif
1153 #if defined(CONFIG_ESPI_PERIPHERAL_ACPI_SHM_REGION)
1154 	host_shared_mem_region_init();
1155 #endif
1156 #if defined(CONFIG_ESPI_PERIPHERAL_DEBUG_PORT_80)
1157 	host_port80_init();
1158 #if defined(CONFIG_ESPI_NPCX_PERIPHERAL_DEBUG_PORT_80_MULTI_BYTE)
1159 	ring_buf_init(&host_sub_data.port80_ring_buf, sizeof(host_sub_data.port80_data),
1160 		      host_sub_data.port80_data);
1161 	k_work_init(&host_sub_data.work, host_port80_work_handler);
1162 #endif
1163 #endif
1164 #if defined(CONFIG_ESPI_PERIPHERAL_UART)
1165 	host_uart_init();
1166 #endif
1167 
1168 	/* Host KBC sub-device interrupt installation */
1169 #if defined(CONFIG_ESPI_PERIPHERAL_8042_KBC)
1170 	IRQ_CONNECT(DT_INST_IRQ_BY_NAME(0, kbc_ibf, irq),
1171 		    DT_INST_IRQ_BY_NAME(0, kbc_ibf, priority),
1172 		    host_kbc_ibf_isr,
1173 		    NULL, 0);
1174 
1175 	IRQ_CONNECT(DT_INST_IRQ_BY_NAME(0, kbc_obe, irq),
1176 		    DT_INST_IRQ_BY_NAME(0, kbc_obe, priority),
1177 		    host_kbc_obe_isr,
1178 		    NULL, 0);
1179 #endif
1180 
1181 	/* Host PM channel (Host IO) sub-device interrupt installation */
1182 #if defined(CONFIG_ESPI_PERIPHERAL_HOST_IO) || \
1183 				defined(CONFIG_ESPI_PERIPHERAL_EC_HOST_CMD)
1184 	IRQ_CONNECT(DT_INST_IRQ_BY_NAME(0, pmch_ibf, irq),
1185 		    DT_INST_IRQ_BY_NAME(0, pmch_ibf, priority),
1186 		    host_pmch_ibf_isr,
1187 		    NULL, 0);
1188 #endif
1189 
1190 	/* Host Port80 sub-device interrupt installation */
1191 #if defined(CONFIG_ESPI_PERIPHERAL_DEBUG_PORT_80)
1192 	IRQ_CONNECT(DT_INST_IRQ_BY_NAME(0, p80_fifo, irq),
1193 		    DT_INST_IRQ_BY_NAME(0, p80_fifo, priority),
1194 		    host_port80_isr,
1195 		    NULL, 0);
1196 #endif
1197 
1198 	if (IS_ENABLED(CONFIG_PM)) {
1199 		/*
1200 		 * Configure the host access wake-up event triggered from a host
1201 		 * transaction on eSPI/LPC bus. Do not enable it here. Or plenty
1202 		 * of interrupts will jam the system in S0.
1203 		 */
1204 		npcx_miwu_interrupt_configure(&host_sub_cfg.host_acc_wui,
1205 				NPCX_MIWU_MODE_EDGE, NPCX_MIWU_TRIG_HIGH);
1206 	}
1207 
1208 	return 0;
1209 }
1210