forked from Minki/linux
832f5dacfa
Currently CONFIG_ARCH_HAVE_CUSTOM_GPIO_H is defined for all MIPS machines, and each machine type provides its own gpio.h. However only a handful really implement the GPIO API, most just forward everythings to gpiolib. The Alchemy machine is notable as it provides a system to allow implementing the GPIO API at the board level. But it is not used by any board currently supported, so it can also be removed. For most machine types we can just remove the custom gpio.h, as well as the custom wrappers if some exists. Some of the code found in the wrappers must be moved to the respective GPIO driver. A few more fixes are need in some drivers as they rely on linux/gpio.h to provides some machine specific definitions, or used asm/gpio.h instead of linux/gpio.h for the gpio API. Signed-off-by: Alban Bedel <albeu@free.fr> Reviewed-by: Linus Walleij <linus.walleij@linaro.org> Cc: linux-mips@linux-mips.org Cc: Hauke Mehrtens <hauke@hauke-m.de> Cc: Rafał Miłecki <zajec5@gmail.com> Cc: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com> Cc: Tejun Heo <tj@kernel.org> Cc: Alexandre Courbot <gnurou@gmail.com> Cc: Dmitry Torokhov <dmitry.torokhov@gmail.com> Cc: Florian Fainelli <florian@openwrt.org> Cc: Manuel Lauss <manuel.lauss@gmail.com> Cc: Joe Perches <joe@perches.com> Cc: Daniel Walter <dwalter@google.com> Cc: Sergey Ryazanov <ryazanov.s.a@gmail.com> Cc: Huacai Chen <chenhc@lemote.com> Cc: James Hartley <james.hartley@imgtec.com> Cc: Andrew Bresticker <abrestic@chromium.org> Cc: Paul Burton <paul.burton@imgtec.com> Cc: Jiri Kosina <jkosina@suse.cz> Cc: Bjorn Helgaas <bhelgaas@google.com> Cc: Wolfram Sang <wsa@the-dreams.de> Cc: Randy Dunlap <rdunlap@infradead.org> Cc: Varka Bhadram <varkabhadram@gmail.com> Cc: Masanari Iida <standby24x7@gmail.com> Cc: Tomi Valkeinen <tomi.valkeinen@ti.com> Cc: Michael Buesch <m@bues.ch> Cc: abdoulaye berthe <berthe.ab@gmail.com> Cc: linux-kernel@vger.kernel.org Cc: linux-ide@vger.kernel.org Cc: linux-gpio@vger.kernel.org Cc: linux-input@vger.kernel.org Cc: netdev@vger.kernel.org Patchwork: https://patchwork.linux-mips.org/patch/10828/ Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
205 lines
5.1 KiB
C
205 lines
5.1 KiB
C
/*
|
|
* Atheros AR71XX/AR724X/AR913X GPIO API support
|
|
*
|
|
* Copyright (C) 2010-2011 Jaiganesh Narayanan <jnarayanan@atheros.com>
|
|
* Copyright (C) 2008-2011 Gabor Juhos <juhosg@openwrt.org>
|
|
* Copyright (C) 2008 Imre Kaloz <kaloz@openwrt.org>
|
|
*
|
|
* Parts of this file are based on Atheros' 2.6.15/2.6.31 BSP
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 as published
|
|
* by the Free Software Foundation.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/init.h>
|
|
#include <linux/module.h>
|
|
#include <linux/types.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/io.h>
|
|
#include <linux/ioport.h>
|
|
#include <linux/gpio.h>
|
|
#include <linux/platform_data/gpio-ath79.h>
|
|
#include <linux/of_device.h>
|
|
|
|
#include <asm/mach-ath79/ar71xx_regs.h>
|
|
|
|
static void __iomem *ath79_gpio_base;
|
|
static u32 ath79_gpio_count;
|
|
static DEFINE_SPINLOCK(ath79_gpio_lock);
|
|
|
|
static void __ath79_gpio_set_value(unsigned gpio, int value)
|
|
{
|
|
void __iomem *base = ath79_gpio_base;
|
|
|
|
if (value)
|
|
__raw_writel(1 << gpio, base + AR71XX_GPIO_REG_SET);
|
|
else
|
|
__raw_writel(1 << gpio, base + AR71XX_GPIO_REG_CLEAR);
|
|
}
|
|
|
|
static int __ath79_gpio_get_value(unsigned gpio)
|
|
{
|
|
return (__raw_readl(ath79_gpio_base + AR71XX_GPIO_REG_IN) >> gpio) & 1;
|
|
}
|
|
|
|
static int ath79_gpio_get_value(struct gpio_chip *chip, unsigned offset)
|
|
{
|
|
return __ath79_gpio_get_value(offset);
|
|
}
|
|
|
|
static void ath79_gpio_set_value(struct gpio_chip *chip,
|
|
unsigned offset, int value)
|
|
{
|
|
__ath79_gpio_set_value(offset, value);
|
|
}
|
|
|
|
static int ath79_gpio_direction_input(struct gpio_chip *chip,
|
|
unsigned offset)
|
|
{
|
|
void __iomem *base = ath79_gpio_base;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&ath79_gpio_lock, flags);
|
|
|
|
__raw_writel(__raw_readl(base + AR71XX_GPIO_REG_OE) & ~(1 << offset),
|
|
base + AR71XX_GPIO_REG_OE);
|
|
|
|
spin_unlock_irqrestore(&ath79_gpio_lock, flags);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ath79_gpio_direction_output(struct gpio_chip *chip,
|
|
unsigned offset, int value)
|
|
{
|
|
void __iomem *base = ath79_gpio_base;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&ath79_gpio_lock, flags);
|
|
|
|
if (value)
|
|
__raw_writel(1 << offset, base + AR71XX_GPIO_REG_SET);
|
|
else
|
|
__raw_writel(1 << offset, base + AR71XX_GPIO_REG_CLEAR);
|
|
|
|
__raw_writel(__raw_readl(base + AR71XX_GPIO_REG_OE) | (1 << offset),
|
|
base + AR71XX_GPIO_REG_OE);
|
|
|
|
spin_unlock_irqrestore(&ath79_gpio_lock, flags);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ar934x_gpio_direction_input(struct gpio_chip *chip, unsigned offset)
|
|
{
|
|
void __iomem *base = ath79_gpio_base;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&ath79_gpio_lock, flags);
|
|
|
|
__raw_writel(__raw_readl(base + AR71XX_GPIO_REG_OE) | (1 << offset),
|
|
base + AR71XX_GPIO_REG_OE);
|
|
|
|
spin_unlock_irqrestore(&ath79_gpio_lock, flags);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ar934x_gpio_direction_output(struct gpio_chip *chip, unsigned offset,
|
|
int value)
|
|
{
|
|
void __iomem *base = ath79_gpio_base;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&ath79_gpio_lock, flags);
|
|
|
|
if (value)
|
|
__raw_writel(1 << offset, base + AR71XX_GPIO_REG_SET);
|
|
else
|
|
__raw_writel(1 << offset, base + AR71XX_GPIO_REG_CLEAR);
|
|
|
|
__raw_writel(__raw_readl(base + AR71XX_GPIO_REG_OE) & ~(1 << offset),
|
|
base + AR71XX_GPIO_REG_OE);
|
|
|
|
spin_unlock_irqrestore(&ath79_gpio_lock, flags);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct gpio_chip ath79_gpio_chip = {
|
|
.label = "ath79",
|
|
.get = ath79_gpio_get_value,
|
|
.set = ath79_gpio_set_value,
|
|
.direction_input = ath79_gpio_direction_input,
|
|
.direction_output = ath79_gpio_direction_output,
|
|
.base = 0,
|
|
};
|
|
|
|
static const struct of_device_id ath79_gpio_of_match[] = {
|
|
{ .compatible = "qca,ar7100-gpio" },
|
|
{ .compatible = "qca,ar9340-gpio" },
|
|
{},
|
|
};
|
|
|
|
static int ath79_gpio_probe(struct platform_device *pdev)
|
|
{
|
|
struct ath79_gpio_platform_data *pdata = pdev->dev.platform_data;
|
|
struct device_node *np = pdev->dev.of_node;
|
|
struct resource *res;
|
|
bool oe_inverted;
|
|
int err;
|
|
|
|
if (np) {
|
|
err = of_property_read_u32(np, "ngpios", &ath79_gpio_count);
|
|
if (err) {
|
|
dev_err(&pdev->dev, "ngpios property is not valid\n");
|
|
return err;
|
|
}
|
|
if (ath79_gpio_count >= 32) {
|
|
dev_err(&pdev->dev, "ngpios must be less than 32\n");
|
|
return -EINVAL;
|
|
}
|
|
oe_inverted = of_device_is_compatible(np, "qca,ar9340-gpio");
|
|
} else if (pdata) {
|
|
ath79_gpio_count = pdata->ngpios;
|
|
oe_inverted = pdata->oe_inverted;
|
|
} else {
|
|
dev_err(&pdev->dev, "No DT node or platform data found\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
ath79_gpio_base = devm_ioremap_nocache(
|
|
&pdev->dev, res->start, resource_size(res));
|
|
if (!ath79_gpio_base)
|
|
return -ENOMEM;
|
|
|
|
ath79_gpio_chip.dev = &pdev->dev;
|
|
ath79_gpio_chip.ngpio = ath79_gpio_count;
|
|
if (oe_inverted) {
|
|
ath79_gpio_chip.direction_input = ar934x_gpio_direction_input;
|
|
ath79_gpio_chip.direction_output = ar934x_gpio_direction_output;
|
|
}
|
|
|
|
err = gpiochip_add(&ath79_gpio_chip);
|
|
if (err) {
|
|
dev_err(&pdev->dev,
|
|
"cannot add AR71xx GPIO chip, error=%d", err);
|
|
return err;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct platform_driver ath79_gpio_driver = {
|
|
.driver = {
|
|
.name = "ath79-gpio",
|
|
.of_match_table = ath79_gpio_of_match,
|
|
},
|
|
.probe = ath79_gpio_probe,
|
|
};
|
|
|
|
module_platform_driver(ath79_gpio_driver);
|