driver: PS/2: npcx: add driver support for Nuvoton npcx family
The PS/2 module in npcx provides a hardware accelerator mechanism including an 8-bit shift register, a state machine, and control logic that handle both the incoming and outgoing data. The hardware accelerator mechanism is shared by 4 PS/2 channels. To support it, this CL separates the PS/2 driver into channel and controller drivers. The controller driver is in charge of the PS/2 transaction. The channel driver is in charge of the connection between the Zehpyr PS/2 API interface and controller driver. Signed-off-by: Jun Lin <CHLin56@nuvoton.com>
This commit is contained in:
parent
8e4d9ae2e8
commit
ba39c47187
11 changed files with 755 additions and 0 deletions
148
drivers/ps2/ps2_npcx_channel.c
Normal file
148
drivers/ps2/ps2_npcx_channel.c
Normal file
|
@ -0,0 +1,148 @@
|
|||
/*
|
||||
* Copyright (c) 2021 Nuvoton Technology Corporation.
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#define DT_DRV_COMPAT nuvoton_npcx_ps2_channel
|
||||
|
||||
/**
|
||||
* @file
|
||||
* @brief Nuvoton NPCX PS/2 driver
|
||||
*
|
||||
* This file contains the driver of PS/2 buses (channels) which provides the
|
||||
* connection between Zephyr PS/2 API fucntions and NPCX PS/2 controller driver
|
||||
* to support PS/2 transactions.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <drivers/clock_control.h>
|
||||
#include <drivers/ps2.h>
|
||||
#include <soc.h>
|
||||
|
||||
#include <logging/log.h>
|
||||
LOG_MODULE_REGISTER(ps2_npcx_channel, CONFIG_PS2_LOG_LEVEL);
|
||||
|
||||
#include "ps2_npcx_controller.h"
|
||||
|
||||
/* Device config */
|
||||
struct ps2_npcx_ch_config {
|
||||
/* pinmux configuration */
|
||||
const uint8_t alts_size;
|
||||
const struct npcx_alt *alts_list;
|
||||
/* Indicate the channel's number of the PS/2 channel device */
|
||||
uint8_t channel_id;
|
||||
};
|
||||
|
||||
/* Driver data */
|
||||
struct ps2_npcx_ch_data {
|
||||
const struct device *ps2_ctrl;
|
||||
};
|
||||
|
||||
/* Driver convenience defines */
|
||||
#define DRV_CONFIG(dev) ((const struct ps2_npcx_ch_config *)(dev)->config)
|
||||
|
||||
#define DRV_DATA(dev) ((struct ps2_npcx_ch_data *)(dev)->data)
|
||||
|
||||
/* PS/2 api functions */
|
||||
static int ps2_npcx_ch_configure(const struct device *dev,
|
||||
ps2_callback_t callback_isr)
|
||||
{
|
||||
const struct ps2_npcx_ch_config *const config = DRV_CONFIG(dev);
|
||||
struct ps2_npcx_ch_data *const data = DRV_DATA(dev);
|
||||
int ret;
|
||||
|
||||
ret = ps2_npcx_ctrl_configure(data->ps2_ctrl, config->channel_id,
|
||||
callback_isr);
|
||||
if (ret != 0)
|
||||
return ret;
|
||||
|
||||
return ps2_npcx_ctrl_enable_interface(data->ps2_ctrl,
|
||||
config->channel_id, 1);
|
||||
}
|
||||
|
||||
static int ps2_npcx_ch_write(const struct device *dev, uint8_t value)
|
||||
{
|
||||
const struct ps2_npcx_ch_config *const config = DRV_CONFIG(dev);
|
||||
struct ps2_npcx_ch_data *const data = DRV_DATA(dev);
|
||||
|
||||
return ps2_npcx_ctrl_write(data->ps2_ctrl, config->channel_id, value);
|
||||
}
|
||||
|
||||
static int ps2_npcx_ch_enable_interface(const struct device *dev)
|
||||
{
|
||||
const struct ps2_npcx_ch_config *const config = DRV_CONFIG(dev);
|
||||
struct ps2_npcx_ch_data *const data = DRV_DATA(dev);
|
||||
|
||||
return ps2_npcx_ctrl_enable_interface(data->ps2_ctrl,
|
||||
config->channel_id, 1);
|
||||
}
|
||||
|
||||
static int ps2_npcx_ch_inhibit_interface(const struct device *dev)
|
||||
{
|
||||
const struct ps2_npcx_ch_config *const config = DRV_CONFIG(dev);
|
||||
struct ps2_npcx_ch_data *const data = DRV_DATA(dev);
|
||||
|
||||
return ps2_npcx_ctrl_enable_interface(data->ps2_ctrl,
|
||||
config->channel_id, 0);
|
||||
}
|
||||
|
||||
/* PS/2 driver registration */
|
||||
static int ps2_npcx_channel_init(const struct device *dev)
|
||||
{
|
||||
const struct ps2_npcx_ch_config *const config = DRV_CONFIG(dev);
|
||||
|
||||
/* Configure pin-mux for PS/2 device */
|
||||
npcx_pinctrl_mux_configure(config->alts_list, config->alts_size, 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct ps2_driver_api ps2_channel_npcx_driver_api = {
|
||||
.config = ps2_npcx_ch_configure,
|
||||
.read = NULL,
|
||||
.write = ps2_npcx_ch_write,
|
||||
.disable_callback = ps2_npcx_ch_inhibit_interface,
|
||||
.enable_callback = ps2_npcx_ch_enable_interface,
|
||||
};
|
||||
|
||||
/* PS/2 channel initialization macro functions */
|
||||
#define NPCX_PS2_CHANNEL_INIT_FUNC(inst) _CONCAT(ps2_npcx_channel_init_, inst)
|
||||
#define NPCX_PS2_CHANNEL_INIT_FUNC_DECL(inst) \
|
||||
static int ps2_npcx_channel_init_##inst(const struct device *dev)
|
||||
|
||||
#define NPCX_PS2_CHANNEL_INIT_FUNC_IMPL(inst) \
|
||||
static int ps2_npcx_channel_init_##inst(const struct device *dev) \
|
||||
{ \
|
||||
struct ps2_npcx_ch_data *const data = DRV_DATA(dev); \
|
||||
\
|
||||
data->ps2_ctrl = device_get_binding(DT_INST_BUS_LABEL(inst)); \
|
||||
return ps2_npcx_channel_init(dev); \
|
||||
}
|
||||
|
||||
#define NPCX_PS2_CHANNEL_INIT(inst) \
|
||||
NPCX_PS2_CHANNEL_INIT_FUNC_DECL(inst); \
|
||||
\
|
||||
static const struct npcx_alt ps2_channel_alts##inst[] = \
|
||||
NPCX_DT_ALT_ITEMS_LIST(inst); \
|
||||
\
|
||||
static const struct ps2_npcx_ch_config ps2_npcx_ch_cfg_##inst = { \
|
||||
.channel_id = DT_INST_PROP(inst, channel), \
|
||||
.alts_size = ARRAY_SIZE(ps2_channel_alts##inst), \
|
||||
.alts_list = ps2_channel_alts##inst, \
|
||||
}; \
|
||||
\
|
||||
static struct ps2_npcx_ch_data ps2_npcx_ch_data_##inst; \
|
||||
\
|
||||
DEVICE_DT_INST_DEFINE(inst, NPCX_PS2_CHANNEL_INIT_FUNC(inst), NULL, \
|
||||
&ps2_npcx_ch_data_##inst, \
|
||||
&ps2_npcx_ch_cfg_##inst, POST_KERNEL, \
|
||||
CONFIG_PS2_CHANNEL_INIT_PRIORITY, \
|
||||
&ps2_channel_npcx_driver_api); \
|
||||
\
|
||||
NPCX_PS2_CHANNEL_INIT_FUNC_IMPL(inst)
|
||||
|
||||
DT_INST_FOREACH_STATUS_OKAY(NPCX_PS2_CHANNEL_INIT)
|
||||
|
||||
/* PS/2 channel driver must be initialized after PS/2 controller driver */
|
||||
BUILD_ASSERT(CONFIG_PS2_CHANNEL_INIT_PRIORITY > CONFIG_PS2_INIT_PRIORITY);
|
Loading…
Add table
Add a link
Reference in a new issue