395 lines
8.5 KiB
C
395 lines
8.5 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* linux/drivers/hil/hilkbd.c
|
|
*
|
|
* Copyright (C) 1998 Philip Blundell <philb@gnu.org>
|
|
* Copyright (C) 1999 Matthew Wilcox <willy@infradead.org>
|
|
* Copyright (C) 1999-2007 Helge Deller <deller@gmx.de>
|
|
*
|
|
* Very basic HP Human Interface Loop (HIL) driver.
|
|
* This driver handles the keyboard on HP300 (m68k) and on some
|
|
* HP700 (parisc) series machines.
|
|
*/
|
|
|
|
#include <linux/pci_ids.h>
|
|
#include <linux/ioport.h>
|
|
#include <linux/module.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/input.h>
|
|
#include <linux/init.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/hil.h>
|
|
#include <linux/io.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/spinlock.h>
|
|
#include <asm/irq.h>
|
|
#ifdef CONFIG_HP300
|
|
#include <asm/hwtest.h>
|
|
#endif
|
|
|
|
|
|
MODULE_AUTHOR("Philip Blundell, Matthew Wilcox, Helge Deller");
|
|
MODULE_DESCRIPTION("HIL keyboard driver (basic functionality)");
|
|
MODULE_LICENSE("GPL v2");
|
|
|
|
|
|
#if defined(CONFIG_PARISC)
|
|
|
|
#include <asm/io.h>
|
|
#include <asm/hardware.h>
|
|
#include <asm/parisc-device.h>
|
|
static unsigned long hil_base; /* HPA for the HIL device */
|
|
static unsigned int hil_irq;
|
|
#define HILBASE hil_base /* HPPA (parisc) port address */
|
|
#define HIL_DATA 0x800
|
|
#define HIL_CMD 0x801
|
|
#define HIL_IRQ hil_irq
|
|
#define hil_readb(p) gsc_readb(p)
|
|
#define hil_writeb(v,p) gsc_writeb((v),(p))
|
|
|
|
#elif defined(CONFIG_HP300)
|
|
|
|
#define HILBASE 0xf0428000UL /* HP300 (m68k) port address */
|
|
#define HIL_DATA 0x1
|
|
#define HIL_CMD 0x3
|
|
#define HIL_IRQ 2
|
|
#define hil_readb(p) readb((const volatile void __iomem *)(p))
|
|
#define hil_writeb(v, p) writeb((v), (volatile void __iomem *)(p))
|
|
|
|
#else
|
|
#error "HIL is not supported on this platform"
|
|
#endif
|
|
|
|
|
|
|
|
/* HIL helper functions */
|
|
|
|
#define hil_busy() (hil_readb(HILBASE + HIL_CMD) & HIL_BUSY)
|
|
#define hil_data_available() (hil_readb(HILBASE + HIL_CMD) & HIL_DATA_RDY)
|
|
#define hil_status() (hil_readb(HILBASE + HIL_CMD))
|
|
#define hil_command(x) do { hil_writeb((x), HILBASE + HIL_CMD); } while (0)
|
|
#define hil_read_data() (hil_readb(HILBASE + HIL_DATA))
|
|
#define hil_write_data(x) do { hil_writeb((x), HILBASE + HIL_DATA); } while (0)
|
|
|
|
/* HIL constants */
|
|
|
|
#define HIL_BUSY 0x02
|
|
#define HIL_DATA_RDY 0x01
|
|
|
|
#define HIL_SETARD 0xA0 /* set auto-repeat delay */
|
|
#define HIL_SETARR 0xA2 /* set auto-repeat rate */
|
|
#define HIL_SETTONE 0xA3 /* set tone generator */
|
|
#define HIL_CNMT 0xB2 /* clear nmi */
|
|
#define HIL_INTON 0x5C /* Turn on interrupts. */
|
|
#define HIL_INTOFF 0x5D /* Turn off interrupts. */
|
|
|
|
#define HIL_READKBDSADR 0xF9
|
|
#define HIL_WRITEKBDSADR 0xE9
|
|
|
|
static unsigned int hphilkeyb_keycode[HIL_KEYCODES_SET1_TBLSIZE] __read_mostly =
|
|
{ HIL_KEYCODES_SET1 };
|
|
|
|
/* HIL structure */
|
|
static struct {
|
|
struct input_dev *dev;
|
|
|
|
unsigned int curdev;
|
|
|
|
unsigned char s;
|
|
unsigned char c;
|
|
int valid;
|
|
|
|
unsigned char data[16];
|
|
unsigned int ptr;
|
|
spinlock_t lock;
|
|
|
|
void *dev_id; /* native bus device */
|
|
} hil_dev;
|
|
|
|
|
|
static void poll_finished(void)
|
|
{
|
|
int down;
|
|
int key;
|
|
unsigned char scode;
|
|
|
|
switch (hil_dev.data[0]) {
|
|
case 0x40:
|
|
down = (hil_dev.data[1] & 1) == 0;
|
|
scode = hil_dev.data[1] >> 1;
|
|
key = hphilkeyb_keycode[scode];
|
|
input_report_key(hil_dev.dev, key, down);
|
|
break;
|
|
}
|
|
hil_dev.curdev = 0;
|
|
}
|
|
|
|
|
|
static inline void handle_status(unsigned char s, unsigned char c)
|
|
{
|
|
if (c & 0x8) {
|
|
/* End of block */
|
|
if (c & 0x10)
|
|
poll_finished();
|
|
} else {
|
|
if (c & 0x10) {
|
|
if (hil_dev.curdev)
|
|
poll_finished(); /* just in case */
|
|
hil_dev.curdev = c & 7;
|
|
hil_dev.ptr = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static inline void handle_data(unsigned char s, unsigned char c)
|
|
{
|
|
if (hil_dev.curdev) {
|
|
hil_dev.data[hil_dev.ptr++] = c;
|
|
hil_dev.ptr &= 15;
|
|
}
|
|
}
|
|
|
|
|
|
/* handle HIL interrupts */
|
|
static irqreturn_t hil_interrupt(int irq, void *handle)
|
|
{
|
|
unsigned char s, c;
|
|
|
|
s = hil_status();
|
|
c = hil_read_data();
|
|
|
|
switch (s >> 4) {
|
|
case 0x5:
|
|
handle_status(s, c);
|
|
break;
|
|
case 0x6:
|
|
handle_data(s, c);
|
|
break;
|
|
case 0x4:
|
|
hil_dev.s = s;
|
|
hil_dev.c = c;
|
|
mb();
|
|
hil_dev.valid = 1;
|
|
break;
|
|
}
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
|
|
/* send a command to the HIL */
|
|
static void hil_do(unsigned char cmd, unsigned char *data, unsigned int len)
|
|
{
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&hil_dev.lock, flags);
|
|
while (hil_busy())
|
|
/* wait */;
|
|
hil_command(cmd);
|
|
while (len--) {
|
|
while (hil_busy())
|
|
/* wait */;
|
|
hil_write_data(*(data++));
|
|
}
|
|
spin_unlock_irqrestore(&hil_dev.lock, flags);
|
|
}
|
|
|
|
|
|
/* initialize HIL */
|
|
static int hil_keyb_init(void)
|
|
{
|
|
unsigned char c;
|
|
unsigned int i, kbid;
|
|
wait_queue_head_t hil_wait;
|
|
int err;
|
|
|
|
if (hil_dev.dev)
|
|
return -ENODEV; /* already initialized */
|
|
|
|
init_waitqueue_head(&hil_wait);
|
|
spin_lock_init(&hil_dev.lock);
|
|
|
|
hil_dev.dev = input_allocate_device();
|
|
if (!hil_dev.dev)
|
|
return -ENOMEM;
|
|
|
|
err = request_irq(HIL_IRQ, hil_interrupt, 0, "hil", hil_dev.dev_id);
|
|
if (err) {
|
|
printk(KERN_ERR "HIL: Can't get IRQ\n");
|
|
goto err1;
|
|
}
|
|
|
|
/* Turn on interrupts */
|
|
hil_do(HIL_INTON, NULL, 0);
|
|
|
|
/* Look for keyboards */
|
|
hil_dev.valid = 0; /* clear any pending data */
|
|
hil_do(HIL_READKBDSADR, NULL, 0);
|
|
|
|
wait_event_interruptible_timeout(hil_wait, hil_dev.valid, 3 * HZ);
|
|
if (!hil_dev.valid)
|
|
printk(KERN_WARNING "HIL: timed out, assuming no keyboard present\n");
|
|
|
|
c = hil_dev.c;
|
|
hil_dev.valid = 0;
|
|
if (c == 0) {
|
|
kbid = -1;
|
|
printk(KERN_WARNING "HIL: no keyboard present\n");
|
|
} else {
|
|
kbid = ffz(~c);
|
|
printk(KERN_INFO "HIL: keyboard found at id %d\n", kbid);
|
|
}
|
|
|
|
/* set it to raw mode */
|
|
c = 0;
|
|
hil_do(HIL_WRITEKBDSADR, &c, 1);
|
|
|
|
for (i = 0; i < HIL_KEYCODES_SET1_TBLSIZE; i++)
|
|
if (hphilkeyb_keycode[i] != KEY_RESERVED)
|
|
__set_bit(hphilkeyb_keycode[i], hil_dev.dev->keybit);
|
|
|
|
hil_dev.dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REP);
|
|
hil_dev.dev->ledbit[0] = BIT_MASK(LED_NUML) | BIT_MASK(LED_CAPSL) |
|
|
BIT_MASK(LED_SCROLLL);
|
|
hil_dev.dev->keycodemax = HIL_KEYCODES_SET1_TBLSIZE;
|
|
hil_dev.dev->keycodesize= sizeof(hphilkeyb_keycode[0]);
|
|
hil_dev.dev->keycode = hphilkeyb_keycode;
|
|
hil_dev.dev->name = "HIL keyboard";
|
|
hil_dev.dev->phys = "hpkbd/input0";
|
|
|
|
hil_dev.dev->id.bustype = BUS_HIL;
|
|
hil_dev.dev->id.vendor = PCI_VENDOR_ID_HP;
|
|
hil_dev.dev->id.product = 0x0001;
|
|
hil_dev.dev->id.version = 0x0010;
|
|
|
|
err = input_register_device(hil_dev.dev);
|
|
if (err) {
|
|
printk(KERN_ERR "HIL: Can't register device\n");
|
|
goto err2;
|
|
}
|
|
|
|
printk(KERN_INFO "input: %s, ID %d at 0x%08lx (irq %d) found and attached\n",
|
|
hil_dev.dev->name, kbid, HILBASE, HIL_IRQ);
|
|
|
|
return 0;
|
|
|
|
err2:
|
|
hil_do(HIL_INTOFF, NULL, 0);
|
|
free_irq(HIL_IRQ, hil_dev.dev_id);
|
|
err1:
|
|
input_free_device(hil_dev.dev);
|
|
hil_dev.dev = NULL;
|
|
return err;
|
|
}
|
|
|
|
static void hil_keyb_exit(void)
|
|
{
|
|
if (HIL_IRQ)
|
|
free_irq(HIL_IRQ, hil_dev.dev_id);
|
|
|
|
/* Turn off interrupts */
|
|
hil_do(HIL_INTOFF, NULL, 0);
|
|
|
|
input_unregister_device(hil_dev.dev);
|
|
hil_dev.dev = NULL;
|
|
}
|
|
|
|
#if defined(CONFIG_PARISC)
|
|
static int __init hil_probe_chip(struct parisc_device *dev)
|
|
{
|
|
/* Only allow one HIL keyboard */
|
|
if (hil_dev.dev)
|
|
return -ENODEV;
|
|
|
|
if (!dev->irq) {
|
|
printk(KERN_WARNING "HIL: IRQ not found for HIL bus at 0x%p\n",
|
|
(void *)dev->hpa.start);
|
|
return -ENODEV;
|
|
}
|
|
|
|
hil_base = dev->hpa.start;
|
|
hil_irq = dev->irq;
|
|
hil_dev.dev_id = dev;
|
|
|
|
printk(KERN_INFO "Found HIL bus at 0x%08lx, IRQ %d\n", hil_base, hil_irq);
|
|
|
|
return hil_keyb_init();
|
|
}
|
|
|
|
static int __exit hil_remove_chip(struct parisc_device *dev)
|
|
{
|
|
hil_keyb_exit();
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct parisc_device_id hil_tbl[] __initconst = {
|
|
{ HPHW_FIO, HVERSION_REV_ANY_ID, HVERSION_ANY_ID, 0x00073 },
|
|
{ 0, }
|
|
};
|
|
|
|
#if 0
|
|
/* Disabled to avoid conflicts with the HP SDC HIL drivers */
|
|
MODULE_DEVICE_TABLE(parisc, hil_tbl);
|
|
#endif
|
|
|
|
static struct parisc_driver hil_driver __refdata = {
|
|
.name = "hil",
|
|
.id_table = hil_tbl,
|
|
.probe = hil_probe_chip,
|
|
.remove = __exit_p(hil_remove_chip),
|
|
};
|
|
|
|
static int __init hil_init(void)
|
|
{
|
|
return register_parisc_driver(&hil_driver);
|
|
}
|
|
|
|
static void __exit hil_exit(void)
|
|
{
|
|
unregister_parisc_driver(&hil_driver);
|
|
}
|
|
|
|
#else /* !CONFIG_PARISC */
|
|
|
|
static int __init hil_init(void)
|
|
{
|
|
int error;
|
|
|
|
/* Only allow one HIL keyboard */
|
|
if (hil_dev.dev)
|
|
return -EBUSY;
|
|
|
|
if (!MACH_IS_HP300)
|
|
return -ENODEV;
|
|
|
|
if (!hwreg_present((void *)(HILBASE + HIL_DATA))) {
|
|
printk(KERN_ERR "HIL: hardware register was not found\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
if (!request_region(HILBASE + HIL_DATA, 2, "hil")) {
|
|
printk(KERN_ERR "HIL: IOPORT region already used\n");
|
|
return -EIO;
|
|
}
|
|
|
|
error = hil_keyb_init();
|
|
if (error) {
|
|
release_region(HILBASE + HIL_DATA, 2);
|
|
return error;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void __exit hil_exit(void)
|
|
{
|
|
hil_keyb_exit();
|
|
release_region(HILBASE + HIL_DATA, 2);
|
|
}
|
|
|
|
#endif /* CONFIG_PARISC */
|
|
|
|
module_init(hil_init);
|
|
module_exit(hil_exit);
|