Input: Add a new serio driver for Phytium PCI PS/2 controller
This driver supports Phytium PCI PS/2 controller.
Signed-off-by: Cheng Quan <chengquan@phytium.com.cn>
Signed-off-by: Chen Baozi <chenbaozi@phytium.com.cn>
Signed-off-by: Jiakun Shuai <shuaijiakun1288@phytium.com.cn>
(cherry picked from commit 399ce371c0
)
Signed-off-by: Alex Shi <alexsshi@tencent.com>
This commit is contained in:
parent
e3d704f429
commit
6043d7860a
|
@ -40,6 +40,18 @@ config SERIO_I8042
|
|||
To compile this driver as a module, choose M here: the
|
||||
module will be called i8042.
|
||||
|
||||
config SERIO_PHYTIUM_PS2
|
||||
depends on SERIO
|
||||
tristate "PHYTIUM PS/2 (keyboard and mouse)"
|
||||
default y if ARCH_PHYTIUM
|
||||
depends on PCI
|
||||
help
|
||||
This selects support for the PS/2 Host Controller on
|
||||
Phytium SoCs.
|
||||
|
||||
To compile this driver as a module, choose M here: the
|
||||
module will be called phytium-ps2.
|
||||
|
||||
config SERIO_SERPORT
|
||||
tristate "Serial port line discipline"
|
||||
default y
|
||||
|
|
|
@ -7,6 +7,7 @@
|
|||
|
||||
obj-$(CONFIG_SERIO) += serio.o
|
||||
obj-$(CONFIG_SERIO_I8042) += i8042.o
|
||||
obj-$(CONFIG_SERIO_PHYTIUM_PS2) += phytium-ps2.o
|
||||
obj-$(CONFIG_SERIO_PARKBD) += parkbd.o
|
||||
obj-$(CONFIG_SERIO_SERPORT) += serport.o
|
||||
obj-$(CONFIG_SERIO_CT82C710) += ct82c710.o
|
||||
|
|
|
@ -0,0 +1,186 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
/*
|
||||
* Phytium PS/2 keyboard controller driver.
|
||||
*
|
||||
* Copyright (C) 2021-2023, Phytium Technology Co., Ltd.
|
||||
*/
|
||||
|
||||
#include <linux/module.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/ioport.h>
|
||||
#include <linux/input.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/serio.h>
|
||||
#include <linux/delay.h>
|
||||
|
||||
#define DRV_NAME "phytium_ps2_pci"
|
||||
|
||||
#define REG_STAT 0x0
|
||||
#define REG_STAT_TX_TIMEOUT 0x1
|
||||
#define REG_STAT_RX_TIMEOUT 0x2
|
||||
#define REG_STAT_TX_FULL 0x4
|
||||
#define REG_CTRL 0x4
|
||||
#define REG_CTRL_RESET 0x1
|
||||
#define REG_CTRL_TX_TIMEOUT 0x2
|
||||
#define REG_CTRL_RX_TIMEOUT 0x4
|
||||
#define REG_CTRL_RX_INTR 0x8
|
||||
#define REG_INTR 0x8
|
||||
#define REG_INTR_TIMEOUT 0x1
|
||||
#define REG_INTR_RX 0x2
|
||||
#define REG_TX 0xc
|
||||
#define REG_RX 0x10
|
||||
#define REG_TIMER_VAL 0x14
|
||||
|
||||
#define REG_CTRL_ENABLE (REG_CTRL_TX_TIMEOUT|REG_CTRL_RX_TIMEOUT|REG_CTRL_RX_INTR)
|
||||
#define REG_DATA_PARITY 0x100
|
||||
|
||||
#define STAT_RX_COUNTER(stat) ((stat >> 8) & 0x1f)
|
||||
|
||||
struct phytium_ps2_data {
|
||||
void __iomem *base;
|
||||
struct serio *io;
|
||||
struct pci_dev *dev;
|
||||
};
|
||||
|
||||
static irqreturn_t phytium_ps2_irq(int irq, void *devid)
|
||||
{
|
||||
struct phytium_ps2_data *ps2if = devid;
|
||||
u32 status, scancode, val = 0;
|
||||
unsigned int flag;
|
||||
int i, rxcount;
|
||||
|
||||
status = readl(ps2if->base + REG_STAT);
|
||||
if (!status)
|
||||
return IRQ_NONE;
|
||||
|
||||
/* Check if there is timeout interrupt */
|
||||
if (status & (REG_STAT_RX_TIMEOUT|REG_STAT_TX_TIMEOUT))
|
||||
val |= REG_INTR_TIMEOUT;
|
||||
|
||||
rxcount = STAT_RX_COUNTER(status);
|
||||
for (i = 0; i < rxcount; i++) {
|
||||
scancode = readl(ps2if->base + REG_RX) & 0x1ff;
|
||||
|
||||
if (rxcount <= 16 && scancode != 0x1ff) {
|
||||
flag = ((scancode & REG_DATA_PARITY) ? SERIO_PARITY : 0);
|
||||
serio_interrupt(ps2if->io, scancode & 0xff, flag);
|
||||
}
|
||||
}
|
||||
|
||||
val |= REG_INTR_RX;
|
||||
writel(val, ps2if->base + REG_INTR);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
int phytium_ps2_write(struct serio *serio, unsigned char val)
|
||||
{
|
||||
struct phytium_ps2_data *ps2if = serio->port_data;
|
||||
unsigned int stat;
|
||||
|
||||
do {
|
||||
stat = readl(ps2if->base + REG_STAT);
|
||||
cpu_relax();
|
||||
} while (stat & REG_STAT_TX_FULL);
|
||||
|
||||
writel(val, ps2if->base + REG_TX);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int phytium_ps2_open(struct serio *io)
|
||||
{
|
||||
struct phytium_ps2_data *ps2if = io->port_data;
|
||||
|
||||
writel(REG_CTRL_RESET, ps2if->base + REG_CTRL);
|
||||
/* Wait 4ms for the controller to be reset */
|
||||
usleep_range(4000, 6000);
|
||||
writel(REG_CTRL_ENABLE, ps2if->base + REG_CTRL);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void phytium_ps2_close(struct serio *io)
|
||||
{
|
||||
struct phytium_ps2_data *ps2if = io->port_data;
|
||||
|
||||
writel(0, ps2if->base + REG_CTRL);
|
||||
}
|
||||
|
||||
static int phytium_pci_ps2_probe(struct pci_dev *pdev, const struct pci_device_id *id)
|
||||
{
|
||||
struct phytium_ps2_data *ps2if;
|
||||
struct serio *serio;
|
||||
int ret;
|
||||
|
||||
ret = pcim_enable_device(pdev);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
ret = pcim_iomap_regions(pdev, 0x1, DRV_NAME);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
ps2if = devm_kzalloc(&pdev->dev, sizeof(struct phytium_ps2_data), GFP_KERNEL);
|
||||
serio = kzalloc(sizeof(struct serio), GFP_KERNEL);
|
||||
if (!ps2if || !serio) {
|
||||
ret = -ENOMEM;
|
||||
goto free;
|
||||
}
|
||||
|
||||
serio->id.type = SERIO_8042;
|
||||
serio->write = phytium_ps2_write;
|
||||
serio->open = phytium_ps2_open;
|
||||
serio->close = phytium_ps2_close;
|
||||
strlcpy(serio->name, pci_name(pdev), sizeof(serio->name));
|
||||
strlcpy(serio->phys, dev_name(&pdev->dev), sizeof(serio->phys));
|
||||
serio->port_data = ps2if;
|
||||
serio->dev.parent = &pdev->dev;
|
||||
ps2if->io = serio;
|
||||
ps2if->dev = pdev;
|
||||
ps2if->base = pcim_iomap_table(pdev)[0];
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, pdev->irq, phytium_ps2_irq,
|
||||
IRQF_SHARED, DRV_NAME, ps2if);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "could not request IRQ %d\n", pdev->irq);
|
||||
goto free;
|
||||
}
|
||||
|
||||
pci_set_drvdata(pdev, ps2if);
|
||||
serio_register_port(ps2if->io);
|
||||
|
||||
return 0;
|
||||
|
||||
free:
|
||||
kfree(serio);
|
||||
out:
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void phytium_pci_ps2_remove(struct pci_dev *pdev)
|
||||
{
|
||||
struct phytium_ps2_data *ps2if = pci_get_drvdata(pdev);
|
||||
|
||||
serio_unregister_port(ps2if->io);
|
||||
pcim_iounmap_regions(pdev, 0x1);
|
||||
}
|
||||
|
||||
static const struct pci_device_id phytium_pci_ps2_ids[] = {
|
||||
{ PCI_VDEVICE(PHYTIUM, 0xdc34) },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(pci, phytium_pci_ps2_ids);
|
||||
|
||||
static struct pci_driver phytium_pci_ps2_driver = {
|
||||
.name = DRV_NAME,
|
||||
.id_table = phytium_pci_ps2_ids,
|
||||
.probe = phytium_pci_ps2_probe,
|
||||
.remove = phytium_pci_ps2_remove,
|
||||
};
|
||||
module_pci_driver(phytium_pci_ps2_driver);
|
||||
|
||||
MODULE_AUTHOR("Cheng Quan <chengquan@phytium.com.cn>");
|
||||
MODULE_DESCRIPTION("Phytium PCI PS/2 controller driver");
|
||||
MODULE_LICENSE("GPL");
|
Loading…
Reference in New Issue