summaryrefslogtreecommitdiff
path: root/arch/arm/mach-wmt/gpio.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/arm/mach-wmt/gpio.c')
-rwxr-xr-xarch/arm/mach-wmt/gpio.c614
1 files changed, 614 insertions, 0 deletions
diff --git a/arch/arm/mach-wmt/gpio.c b/arch/arm/mach-wmt/gpio.c
new file mode 100755
index 00000000..7558c137
--- /dev/null
+++ b/arch/arm/mach-wmt/gpio.c
@@ -0,0 +1,614 @@
+/* linux/arch/arm/mach-wmt/gpio.c
+ *
+ * Copyright (c) 2013 WonderMedia Technologies, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ */
+#include <linux/module.h>
+#include <linux/device.h>
+#include <linux/gpio.h>
+#include <linux/init.h>
+#include <linux/irq.h>
+#include <linux/io.h>
+#include <linux/seq_file.h>
+#include <linux/kobject.h>
+#include <linux/debugfs.h>
+
+#include <mach/hardware.h>
+#include <mach/wmt_iomux.h>
+
+#undef WMT_PIN
+#define WMT_PIN(__gp, __bit, __irq, __name) \
+ { .label = #__name, .regoff = __gp, .shift = __bit, .irqnum = __irq },
+
+const static struct wmt_gpio {
+ const char *label;
+ unsigned char regoff;
+ unsigned char shift;
+ int irqnum;
+} wmt_gpios[] = {
+ #include <mach/iomux.h>
+};
+
+#define to_wmt(__chip) container_of(__chip, struct wmt_gpio_port, chip)
+
+struct wmt_gpio_port {
+ uint32_t base;
+ uint32_t gpio_irq_no_base;
+ int gpio_irq_nr;
+ struct gpio_chip chip;
+ spinlock_t lock;
+};
+
+static struct wmt_gpio_port wmt_gpio_port;
+
+#define INVALUE_REGS 0x00
+#define ENABLE_REGS 0x40
+#define DIRECTION_REGS 0x80
+#define OUTVALUE_REGS 0xc0
+
+#define INTMASK_REGS 0x300
+#define INTSTAT_REGS 0x360
+
+#define PULLENABLE_REGS 0x480
+#define PULLCONTROL_REGS 0x4c0
+
+#define GPIO_INT_LOW_LEV 0x0
+#define GPIO_INT_HIGH_LEV 0x1
+#define GPIO_INT_FALL_EDGE 0x2
+#define GPIO_INT_RISE_EDGE 0x3
+#define GPIO_INT_BOTH_EDGE 0x4
+
+#define CHIP_GPIO_BASE 0
+
+static void _clear_gpio_irqstatus(u32 irqindex)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ u32 addr = port->base + INTSTAT_REGS + (irqindex >> 3);
+ u8 index = irqindex & 0x7;
+
+ __raw_writeb(1 << index, addr);
+}
+
+static int _gpio_irqstatus(u32 irqindex)
+{
+ u8 l;
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ u32 addr = port->base + INTSTAT_REGS + (irqindex >> 3);
+ u8 index = irqindex & 0x7;
+ l = __raw_readb(addr);
+ return l & (1<<index);
+}
+
+int gpio_irqstatus(unsigned int gpio)
+{
+ int offset = gpio - CHIP_GPIO_BASE;
+ int irqindex = wmt_gpios[offset].irqnum;
+ return _gpio_irqstatus(irqindex);
+}
+EXPORT_SYMBOL(gpio_irqstatus);
+
+static void _set_gpio_irqenable(u32 irqindex, int enable)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ u32 addr = port->base + INTMASK_REGS + irqindex;
+ u8 l;
+
+ l = __raw_readb(addr);
+ l = (l & 0x7f) | (!!enable << 7);
+ __raw_writeb(l, addr);
+}
+
+int is_gpio_irqenable(unsigned int gpio)
+{
+ u8 l;
+ int offset = gpio - CHIP_GPIO_BASE;
+ int irqindex = wmt_gpios[offset].irqnum;
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ u32 addr = port->base + INTMASK_REGS + irqindex;
+
+ l = __raw_readb(addr);
+ return (l & 0x80);
+}
+EXPORT_SYMBOL(is_gpio_irqenable);
+
+void wmt_gpio_ack_irq(unsigned int gpio)
+{
+ int offset = gpio - CHIP_GPIO_BASE;
+ int irqindex = wmt_gpios[offset].irqnum;
+
+ _clear_gpio_irqstatus(irqindex);
+}
+EXPORT_SYMBOL(wmt_gpio_ack_irq);
+
+void wmt_gpio_mask_irq(unsigned int gpio)
+{
+ int offset = gpio - CHIP_GPIO_BASE;
+ int irqindex = wmt_gpios[offset].irqnum;
+
+ _set_gpio_irqenable(irqindex, 0);
+}
+EXPORT_SYMBOL(wmt_gpio_mask_irq);
+
+void wmt_gpio_unmask_irq(unsigned int gpio)
+{
+ int offset = gpio - CHIP_GPIO_BASE;
+ int irqindex = wmt_gpios[offset].irqnum;
+
+ _set_gpio_irqenable(irqindex, 1);
+}
+EXPORT_SYMBOL(wmt_gpio_unmask_irq);
+
+int wmt_gpio_set_irq_type(unsigned int gpio, u32 type)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ int offset = gpio - CHIP_GPIO_BASE;
+ int irqindex = wmt_gpios[offset].irqnum;
+ int edge;
+ u32 reg, val;
+
+ switch (type) {
+ case IRQ_TYPE_EDGE_RISING:
+ edge = GPIO_INT_RISE_EDGE;
+ break;
+ case IRQ_TYPE_EDGE_FALLING:
+ edge = GPIO_INT_FALL_EDGE;
+ break;
+ case IRQ_TYPE_EDGE_BOTH:
+ edge = GPIO_INT_BOTH_EDGE;
+ break;
+ case IRQ_TYPE_LEVEL_LOW:
+ edge = GPIO_INT_LOW_LEV;
+ break;
+ case IRQ_TYPE_LEVEL_HIGH:
+ edge = GPIO_INT_HIGH_LEV;
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ reg = port->base + INTMASK_REGS + irqindex;
+ val = __raw_readb(reg) & 0xf8;
+ __raw_writeb(val | edge, reg);
+
+ _clear_gpio_irqstatus(irqindex);
+
+ return 0;
+}
+EXPORT_SYMBOL(wmt_gpio_set_irq_type);
+
+#ifdef WMT_GPIO_IRQ
+static void gpio_ack_irq(struct irq_data *d)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ int irqindex = d->irq - port->gpio_irq_no_base;
+
+ _clear_gpio_irqstatus(irqindex);
+}
+
+static void gpio_mask_irq(struct irq_data *d)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ int irqindex = d->irq - port->gpio_irq_no_base;
+
+ _set_gpio_irqenable(irqindex, 0);
+}
+
+static void gpio_unmask_irq(struct irq_data *d)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ int irqindex = d->irq - port->gpio_irq_no_base;
+
+ _set_gpio_irqenable(irqindex, 1);
+}
+
+static int gpio_set_irq_type(struct irq_data *d, u32 type)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ int edge, irqindex;
+ u32 reg, val;
+
+ switch (type) {
+ case IRQ_TYPE_EDGE_RISING:
+ edge = GPIO_INT_RISE_EDGE;
+ break;
+ case IRQ_TYPE_EDGE_FALLING:
+ edge = GPIO_INT_FALL_EDGE;
+ break;
+ case IRQ_TYPE_EDGE_BOTH:
+ edge = GPIO_INT_BOTH_EDGE;
+ break;
+ case IRQ_TYPE_LEVEL_LOW:
+ edge = GPIO_INT_LOW_LEV;
+ break;
+ case IRQ_TYPE_LEVEL_HIGH:
+ edge = GPIO_INT_HIGH_LEV;
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ irqindex = d->irq - port->gpio_irq_no_base;
+ reg = port->base + INTMASK_REGS + irqindex;
+ val = __raw_readb(reg) & 0xf8;
+ __raw_writeb(val | edge, reg);
+
+ _clear_gpio_irqstatus(irqindex);
+
+ return 0;
+}
+
+/* WMT has one interrupt *for all* gpio ports */
+static void wmt_gpio_irq_handler(u32 irq, struct irq_desc *desc)
+{
+ int i;
+ u32 irq_msk, irq_stat;
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+
+ /* walk through all interrupt status registers */
+ for (i = 0; i < 4; i++) {
+ irq_msk = __raw_readl(port->base + INTMASK_REGS + i*4);
+ if (!irq_msk)
+ continue;
+
+ irq_stat = __raw_readl(port->base + INTSTAT_REGS + i*4) & irq_msk;
+ while (irq_stat != 0) {
+ int irqoffset = fls(irq_stat) - 1;
+
+ generic_handle_irq(port->gpio_irq_no_base + i*32 + irqoffset);
+
+ irq_stat &= ~(1 << irqoffset);
+ }
+ }
+}
+
+static struct irq_chip gpio_irq_chip = {
+ .name = "GPIO",
+ .irq_ack = gpio_ack_irq,
+ .irq_mask = gpio_mask_irq,
+ .irq_unmask = gpio_unmask_irq,
+ .irq_set_type = gpio_set_irq_type,
+};
+#endif
+
+static int wmt_gpio_request(struct gpio_chip *chip, unsigned offset)
+{
+ struct wmt_gpio_port *port = to_wmt(chip);
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+
+ unsigned long flags;
+
+ spin_lock_irqsave(&port->lock, flags);
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + ENABLE_REGS + regoff);
+ val |= (1 << shift);
+ writeb(val, base + ENABLE_REGS + regoff);
+
+ spin_unlock_irqrestore(&port->lock, flags);
+ return 0;
+}
+
+static void wmt_gpio_free(struct gpio_chip *chip, unsigned offset)
+{
+ struct wmt_gpio_port *port = to_wmt(chip);
+// uint32_t base = port->base;
+// uint8_t regoff = wmt_gpios[offset].regoff;
+// uint8_t shift = wmt_gpios[offset].shift;
+// uint8_t val;
+
+ unsigned long flags;
+
+ spin_lock_irqsave(&port->lock, flags);
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+// val = readb(base + ENABLE_REGS + regoff);
+// val &= ~(1 << shift);
+// writeb(val, base + ENABLE_REGS + regoff);
+
+ spin_unlock_irqrestore(&port->lock, flags);
+}
+
+static void _set_gpio_direction(struct gpio_chip *chip, unsigned offset, int dir)
+{
+ struct wmt_gpio_port *port = to_wmt(chip);
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+ unsigned long flags;
+
+ spin_lock_irqsave(&port->lock, flags);
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + DIRECTION_REGS + regoff);
+ if (dir)
+ val |= (1 << shift);
+ else
+ val &= ~(1 << shift);
+ writeb(val, base + DIRECTION_REGS + regoff);
+
+ spin_unlock_irqrestore(&port->lock, flags);
+}
+
+static int wmt_gpio_get_value(struct gpio_chip *chip, unsigned offset)
+{
+ struct wmt_gpio_port *port = to_wmt(chip);
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+ int dir;
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + DIRECTION_REGS + regoff);
+ dir = (val >> shift) & 1;
+
+ if (dir)
+ return (readb(base + OUTVALUE_REGS + regoff) >> shift) & 1;
+ else
+ return (readb(base + INVALUE_REGS + regoff) >> shift) & 1;
+}
+
+static void wmt_gpio_set_value(struct gpio_chip *chip,
+ unsigned offset, int value)
+{
+ struct wmt_gpio_port *port = to_wmt(chip);
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+ unsigned long flags;
+
+ spin_lock_irqsave(&port->lock, flags);
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + OUTVALUE_REGS + regoff);
+
+ if (value)
+ val |= (1 << shift);
+ else
+ val &= ~(1 << shift);
+
+ writeb(val, base + OUTVALUE_REGS + regoff);
+
+ spin_unlock_irqrestore(&port->lock, flags);
+}
+
+static int wmt_gpio_direction_input(struct gpio_chip *chip, unsigned offset)
+{
+ _set_gpio_direction(chip, offset, 0);
+ return 0;
+}
+
+static int wmt_gpio_direction_output(struct gpio_chip *chip,
+ unsigned offset, int value)
+{
+ wmt_gpio_set_value(chip, offset, value);
+ _set_gpio_direction(chip, offset, 1);
+ return 0;
+}
+
+static void _set_gpio_pullenable(unsigned offset, int enable)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+ unsigned long flags;
+
+ spin_lock_irqsave(&port->lock, flags);
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + PULLENABLE_REGS + regoff);
+ if (enable)
+ val |= (1 << shift);
+ else
+ val &= ~(1 << shift);
+ writeb(val, base + PULLENABLE_REGS + regoff);
+
+ spin_unlock_irqrestore(&port->lock, flags);
+}
+
+static void _set_gpio_pullup(unsigned offset, int up)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+ unsigned long flags;
+
+ spin_lock_irqsave(&port->lock, flags);
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + PULLCONTROL_REGS + regoff);
+ if (up)
+ val |= (1 << shift);
+ else
+ val &= ~(1 << shift);
+ writeb(val, base + PULLCONTROL_REGS + regoff);
+
+ spin_unlock_irqrestore(&port->lock, flags);
+}
+
+int wmt_gpio_setpull(unsigned int gpio, enum wmt_gpio_pulltype pull)
+{
+ int offset = gpio - CHIP_GPIO_BASE;
+
+ switch (pull) {
+ case WMT_GPIO_PULL_NONE:
+ _set_gpio_pullenable(offset, 0);
+ break;
+ case WMT_GPIO_PULL_UP:
+ _set_gpio_pullenable(offset, 1);
+ _set_gpio_pullup(offset, 1);
+ break;
+ case WMT_GPIO_PULL_DOWN:
+ _set_gpio_pullenable(offset, 1);
+ _set_gpio_pullup(offset, 0);
+ break;
+ default:
+ return -EINVAL;
+ }
+ return 0;
+}
+
+static int _get_gpio_pull(unsigned offset)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+ uint32_t base = port->base;
+ uint8_t regoff = wmt_gpios[offset].regoff;
+ uint8_t shift = wmt_gpios[offset].shift;
+ uint8_t val;
+ int dir;
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ val = readb(base + DIRECTION_REGS + regoff);
+ dir = (val >> shift) & 1;
+
+ if (dir)
+ return (readb(base + OUTVALUE_REGS + regoff) >> shift) & 1;
+ else
+ return (readb(base + INVALUE_REGS + regoff) >> shift) & 1;
+}
+
+
+int wmt_gpio_getpull(unsigned int gpio)
+{
+ int offset = gpio - CHIP_GPIO_BASE;
+
+ return _get_gpio_pull(offset);
+}
+
+EXPORT_SYMBOL(wmt_gpio_setpull);
+EXPORT_SYMBOL(wmt_gpio_getpull);
+
+#ifdef WMT_GPIO_IRQ
+static int wmt_gpio_to_irq(struct gpio_chip *chip, unsigned offset)
+{
+ struct wmt_gpio_port *port = &wmt_gpio_port;
+
+ WARN_ON(offset >= ARRAY_SIZE(wmt_gpios));
+
+ if (wmt_gpios[offset].irqnum < 0)
+ return -EINVAL;
+
+ return port->gpio_irq_no_base + wmt_gpios[offset].irqnum;
+}
+#endif
+
+static struct wmt_gpio_port wmt_gpio_port = {
+ .base = GPIO_BASE_ADDR,
+#ifdef WMT_GPIO_IRQ
+ .gpio_irq_no_base = WMT_GPIO_IRQ_START,
+ .gpio_irq_nr = WMT_GPIO_IRQS,
+#endif
+ .chip = {
+ .label = "wmt-gpio",
+ .direction_input = wmt_gpio_direction_input,
+ .direction_output = wmt_gpio_direction_output,
+ .request = wmt_gpio_request,
+ .free = wmt_gpio_free,
+ .get = wmt_gpio_get_value,
+ .set = wmt_gpio_set_value,
+#ifdef WMT_GPIO_IRQ
+ .to_irq = wmt_gpio_to_irq,
+#endif
+ .can_sleep = 0,
+ .base = CHIP_GPIO_BASE,
+ .ngpio = ARRAY_SIZE(wmt_gpios),
+ },
+};
+
+void __init wmt_gpio_init(void)
+{
+#ifdef WMT_GPIO_IRQ
+ int irq;
+#endif
+
+ spin_lock_init(&wmt_gpio_port.lock);
+ BUG_ON(gpiochip_add(&wmt_gpio_port.chip) < 0);
+
+ /* XXX conflict with touchscreen irq */
+#ifdef WMT_GPIO_IRQ
+ for (irq = wmt_gpio_port.gpio_irq_no_base;
+ irq < wmt_gpio_port.gpio_irq_no_base + wmt_gpio_port.gpio_irq_nr;
+ irq++) {
+ irq_set_chip_and_handler(irq, &gpio_irq_chip, handle_level_irq);
+ set_irq_flags(irq, IRQF_VALID);
+ }
+
+ irq_set_chained_handler(IRQ_GPIO, wmt_gpio_irq_handler);
+#endif
+}
+
+const char *wmt_gpio_name(int gpio)
+{
+ if (gpio >= ARRAY_SIZE(wmt_gpios)) {
+ pr_err("wmt: can not find gpio-%d\n", gpio);
+ return NULL;
+ }
+
+ return wmt_gpios[gpio].label;
+}
+
+#ifdef CONFIG_DEBUG_FS
+
+static int wmt_gpio_show(struct seq_file *s, void *unused)
+{
+ unsigned i;
+
+ seq_printf(s, "GPIONUM , REGOFF , SHIFT , MACRO NAME\n");
+ seq_printf(s, "---------+--------+-------+-----------\n");
+
+ for (i = 0; i < wmt_gpio_port.chip.ngpio; i++) {
+ seq_printf(s, "gpio-%-3d , 0x%02x , %d , %s",
+ i, wmt_gpios[i].regoff, wmt_gpios[i].shift, wmt_gpios[i].label);
+ seq_printf(s, "\n");
+ }
+ return 0;
+}
+
+static int wmt_gpio_open(struct inode *inode, struct file *file)
+{
+ return single_open(file, wmt_gpio_show, NULL);
+}
+
+static const struct file_operations wmt_gpio_operations = {
+ .open = wmt_gpio_open,
+ .read = seq_read,
+ .llseek = seq_lseek,
+ .release = single_release,
+};
+
+static int __init wmt_gpio_debugfs_init(void)
+{
+ /* /sys/kernel/debug/wmt-gpio */
+ (void) debugfs_create_file("wmt-gpio", S_IFREG | S_IRUGO,
+ NULL, NULL, &wmt_gpio_operations);
+ return 0;
+}
+subsys_initcall(wmt_gpio_debugfs_init);
+
+#endif /* DEBUG_FS */
+