APBPS2 is a PS/2 core part of GRLIB found in SPARC32/LEON products. Signed-off-by: Daniel Hellstrom <daniel@xxxxxxxxxxx> --- .../bindings/input/ps2keyb-mouse-apbps2.txt | 20 ++ drivers/input/serio/Kconfig | 10 + drivers/input/serio/Makefile | 1 + drivers/input/serio/apbps2.c | 266 ++++++++++++++++++++ 4 files changed, 297 insertions(+), 0 deletions(-) create mode 100644 Documentation/devicetree/bindings/input/ps2keyb-mouse-apbps2.txt create mode 100644 drivers/input/serio/apbps2.c diff --git a/Documentation/devicetree/bindings/input/ps2keyb-mouse-apbps2.txt b/Documentation/devicetree/bindings/input/ps2keyb-mouse-apbps2.txt new file mode 100644 index 0000000..1553d28 --- /dev/null +++ b/Documentation/devicetree/bindings/input/ps2keyb-mouse-apbps2.txt @@ -0,0 +1,20 @@ +Aeroflex Gaisler APBPS2 PS/2 Core, supporting Keyboard or Mouse. + +The APBPS2 PS/2 core is available in the GRLIB VHDL IP core library. + +Note: In the ordinary environment for the APBPS2 core, a LEON SPARC system, +these properties are built from information in the AMBA plug&play and from +bootloader settings. + +Required properties: + +- name : Should be "GAISLER_APBPS2" or "01_060" +- reg : Address and length of the register set for the device +- interrupts : Interrupt numbers for this device + +Optional properties: +- keyboard : if present it indicates that a keyboard is connected, if not + present the driver will assume that a mouse is connected instead + +For further information look in the documentation for the GLIB IP core library: +http://www.gaisler.com/products/grlib/grip.pdf diff --git a/drivers/input/serio/Kconfig b/drivers/input/serio/Kconfig index 4a4e182..d0304c3 100644 --- a/drivers/input/serio/Kconfig +++ b/drivers/input/serio/Kconfig @@ -236,6 +236,7 @@ config SERIO_PS2MULT config SERIO_ARC_PS2 tristate "ARC PS/2 support" + depends on OF help Say Y here if you have an ARC FPGA platform with a PS/2 controller in it. @@ -243,4 +244,13 @@ config SERIO_ARC_PS2 To compile this driver as a module, choose M here; the module will be called arc_ps2. +config SERIO_APBPS2 + tristate "GRLIB APBPS2 PS/2 keyboard/mouse controller" + help + Say Y here if you want support for GRLIB APBPS2 peripherals used + to connect to PS/2 keyboard and/or mouse. + + To compile this driver as a module, choose M here: the module will + be called apbps2. + endif diff --git a/drivers/input/serio/Makefile b/drivers/input/serio/Makefile index 4b0c8f8..8edb36c 100644 --- a/drivers/input/serio/Makefile +++ b/drivers/input/serio/Makefile @@ -26,3 +26,4 @@ obj-$(CONFIG_SERIO_AMS_DELTA) += ams_delta_serio.o obj-$(CONFIG_SERIO_XILINX_XPS_PS2) += xilinx_ps2.o obj-$(CONFIG_SERIO_ALTERA_PS2) += altera_ps2.o obj-$(CONFIG_SERIO_ARC_PS2) += arc_ps2.o +obj-$(CONFIG_SERIO_APBPS2) += apbps2.o diff --git a/drivers/input/serio/apbps2.c b/drivers/input/serio/apbps2.c new file mode 100644 index 0000000..78b28e1 --- /dev/null +++ b/drivers/input/serio/apbps2.c @@ -0,0 +1,266 @@ +/* + * linux/drivers/input/serio/apbps2.c + * + * Copyright (C) 2013 Aeroflex Gaisler + * + * This driver supports the APBPS2 PS/2 core available in the GRLIB + * VHDL IP core library. + * + * Full documentation of the APBPS2 core can be found here: + * http://www.gaisler.com/products/grlib/grip.pdf + * + * See "Documentation/devicetree/bindings/input/ps2keyb-mouse-apbps2.txt" for + * information on open firmware properties. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * Contributors: Daniel Hellstrom <daniel@xxxxxxxxxxx> + */ +#include <linux/platform_device.h> +#include <linux/of_device.h> +#include <linux/module.h> +#include <linux/init.h> +#include <linux/serio.h> +#include <linux/errno.h> +#include <linux/interrupt.h> +#include <linux/device.h> +#include <linux/delay.h> +#include <linux/slab.h> +#include <linux/err.h> +#include <linux/string.h> +#include <linux/kernel.h> +#include <linux/io.h> + +struct apbps2_regs { + u32 data; /* 0x00 */ + u32 status; /* 0x04 */ + u32 ctrl; /* 0x08 */ + u32 reload; /* 0x0c */ +}; + +#define APBPS2_STATUS_DR (1<<0) +#define APBPS2_STATUS_PE (1<<1) +#define APBPS2_STATUS_FE (1<<2) +#define APBPS2_STATUS_KI (1<<3) +#define APBPS2_STATUS_RF (1<<4) +#define APBPS2_STATUS_TF (1<<5) +#define APBPS2_STATUS_TCNT (0x1f<<22) +#define APBPS2_STATUS_RCNT (0x1f<<27) + +#define APBPS2_CTRL_RE (1<<0) +#define APBPS2_CTRL_TE (1<<1) +#define APBPS2_CTRL_RI (1<<2) +#define APBPS2_CTRL_TI (1<<3) + +/* Register read/write functions */ +#define APBPS2_READ(reg) apbps2_read_reg(reg) +#define APBPS2_WRITE(reg, data) apbps2_write_reg(reg, data) + +struct apbps2_priv { + struct serio io; + struct apbps2_regs *regs; + int irq; +}; + +static inline unsigned long apbps2_read_reg(void __iomem *reg) +{ + return ioread32be(reg); +} + +static inline void apbps2_write_reg(void __iomem *reg, unsigned long data) +{ + iowrite32be(data, reg); +} + +static irqreturn_t apbps2_isr(int irq, void *dev_id) +{ + struct apbps2_priv *priv = dev_id; + unsigned long status, data, rxflags; + irqreturn_t ret = IRQ_NONE; + + while ((status = APBPS2_READ(&priv->regs->status)) & APBPS2_STATUS_DR) { + data = APBPS2_READ(&priv->regs->data); + rxflags = (status & APBPS2_STATUS_PE) ? SERIO_PARITY : 0; + rxflags |= (status & APBPS2_STATUS_PE) ? SERIO_FRAME : 0; + + /* clear error bits? */ + if (rxflags) + APBPS2_WRITE(&priv->regs->status, status); + + serio_interrupt(&priv->io, data, rxflags); + + ret = IRQ_HANDLED; + } + + return ret; +} + +static int apbps2_write(struct serio *io, unsigned char val) +{ + struct apbps2_priv *priv = io->port_data; + unsigned int tleft = 10000; /* timeout in 100ms */ + + /* delay until PS/2 controller has room for more chars */ + while ((APBPS2_READ(&priv->regs->status) & APBPS2_STATUS_TF) && tleft--) + udelay(10); + + if ((APBPS2_READ(&priv->regs->status) & APBPS2_STATUS_TF) == 0) { + APBPS2_WRITE(&priv->regs->data, val); + + APBPS2_WRITE(&priv->regs->ctrl, APBPS2_CTRL_RE | + APBPS2_CTRL_RI | + APBPS2_CTRL_TE); + return 0; + } + + return SERIO_TIMEOUT; +} + +static int apbps2_open(struct serio *io) +{ + struct apbps2_priv *priv = io->port_data; + int err, limit; + unsigned long tmp; + + err = devm_request_irq(&io->dev, priv->irq, apbps2_isr, IRQF_SHARED, + "apbps2", priv); + if (err) { + dev_err(&io->dev, "request IRQ%d failed\n", priv->irq); + return err; + } + + /* clear error flags */ + APBPS2_WRITE(&priv->regs->status, 0); + + /* Clear old data if available (unlikely) */ + limit = 1024; + while ((APBPS2_READ(&priv->regs->status) & APBPS2_STATUS_DR) && --limit) + tmp = APBPS2_READ(&priv->regs->data); + + /* Enable reciever and it's interrupt */ + APBPS2_WRITE(&priv->regs->ctrl, APBPS2_CTRL_RE | APBPS2_CTRL_RI); + + return 0; +} + +static void apbps2_close(struct serio *io) +{ + struct apbps2_priv *priv = io->port_data; + + /* stop interrupts at PS/2 HW level */ + APBPS2_WRITE(&priv->regs->ctrl, 0); + + /* unregister PS/2 interrupt handler */ + devm_free_irq(&io->dev, priv->irq, priv); +} + +/* Initialize one APBPS2 PS/2 core */ +static int apbps2_of_probe(struct platform_device *ofdev) +{ + struct apbps2_priv *priv; + int len; + u32 *addr, *freq_hz; + char *typestr; + struct resource *res; + + priv = devm_kzalloc(&ofdev->dev, sizeof(*priv), GFP_KERNEL); + if (!priv) { + dev_err(&ofdev->dev, "memory allocation failed\n"); + return -ENOMEM; + } + platform_set_drvdata(ofdev, priv); + + /* Find Device Address */ + res = platform_get_resource(ofdev, IORESOURCE_MEM, 0); + priv->regs = devm_request_and_ioremap(&ofdev->dev, res); + if (!priv->regs) { + dev_err(&ofdev->dev, "io-regs mapping failed\n"); + return -EADDRNOTAVAIL; + } + + /* IRQ */ + priv->irq = ofdev->archdata.irqs[0]; + + /* Get core frequency */ + freq_hz = (u32 *)of_get_property(ofdev->dev.of_node, "freq", &len); + if (!freq_hz) { + dev_err(&ofdev->dev, "unable to get core frequency\n"); + return -EINVAL; + } + + priv->io.open = apbps2_open; + priv->io.close = apbps2_close; + priv->io.port_data = priv; + + /* Get keyboard property. If no such property we know it is a mouse */ + addr = (u32 *)of_get_property(ofdev->dev.of_node, "keyboard", &len); + if (!addr) { + priv->io.id.type = SERIO_PS_PSTHRU; + priv->io.write = apbps2_write; + strlcpy(priv->io.name, "APBPS2 PS/2 Mouse", + sizeof(priv->io.name)); + strlcpy(priv->io.phys, "APBPS2 PS/2 Mouse", + sizeof(priv->io.phys)); + typestr = "Mouse"; + } else { + priv->io.id.type = SERIO_8042; + priv->io.write = apbps2_write; + strlcpy(priv->io.name, "APBPS2 PS/2 Keyboard", + sizeof(priv->io.name)); + strlcpy(priv->io.phys, "APBPS2 PS/2 Keyboard", + sizeof(priv->io.phys)); + typestr = "Keyboard"; + } + + dev_info(&ofdev->dev, "%s, irq = %d, base = 0x%p\n", + typestr, priv->irq, priv->regs); + + /* Set reload register to system freq in kHz/10 */ + APBPS2_WRITE(&priv->regs->reload, *freq_hz / 10000); + + serio_register_port(&priv->io); + + return 0; +} + +static int apbps2_of_remove(struct platform_device *of_dev) +{ + struct apbps2_priv *priv = platform_get_drvdata(of_dev); + + of_iounmap(&of_dev->resource[0], priv->regs, + resource_size(&of_dev->resource[0])); + kfree(priv); + + return 0; +} + +static struct of_device_id apbps2_of_match[] = { + { + .name = "GAISLER_APBPS2", + }, + { + .name = "01_060", + }, + {}, +}; + +MODULE_DEVICE_TABLE(of, apbps2_of_match); + +static struct platform_driver apbps2_of_driver = { + .driver = { + .name = "grlib-apbps2", + .owner = THIS_MODULE, + .of_match_table = apbps2_of_match, + }, + .probe = apbps2_of_probe, + .remove = apbps2_of_remove, +}; + +module_platform_driver(apbps2_of_driver); + +MODULE_AUTHOR("Aeroflex Gaisler AB."); +MODULE_DESCRIPTION("GRLIB APBPS2 PS/2 serial I/O"); +MODULE_LICENSE("GPL"); -- 1.7.0.4 -- To unsubscribe from this list: send the line "unsubscribe linux-input" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html