Skip to content

Commit

Permalink
Input: Add a new serio driver for Phytium PCI PS/2 controller
Browse files Browse the repository at this point in the history
This driver supports Phytium PCI PS/2 controller.

Signed-off-by: Cheng Quan <[email protected]>
Signed-off-by: Chen Baozi <[email protected]>
Signed-off-by: Chen Zhenhua <[email protected]>
Signed-off-by: Wang Yinfeng <[email protected]>
Signed-off-by: Hu Yeqing <[email protected]>
  • Loading branch information
Hu Yeqing authored and Avenger-285714 committed May 24, 2024
1 parent d61dbba commit 8fa583c
Show file tree
Hide file tree
Showing 3 changed files with 199 additions and 0 deletions.
12 changes: 12 additions & 0 deletions drivers/input/serio/Kconfig
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
1 change: 1 addition & 0 deletions drivers/input/serio/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
186 changes: 186 additions & 0 deletions drivers/input/serio/phytium-ps2.c
Original file line number Diff line number Diff line change
@@ -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;
strscpy(serio->name, pci_name(pdev), sizeof(serio->name));
strscpy(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 <[email protected]>");
MODULE_DESCRIPTION("Phytium PCI PS/2 controller driver");
MODULE_LICENSE("GPL");

0 comments on commit 8fa583c

Please sign in to comment.