[PATCH v3 10/10] tegra: add GPIO controller driver
antonynpavlov at gmail.com
antonynpavlov at gmail.com
Sun Apr 7 23:51:46 EDT 2013
On Tue, 2 Apr 2013 08:19:12 +0200
Lucas Stach <dev at lynxeye.de> wrote:
> Taken from the Linux kernel, simplified and reworked to match barebox.
>
> Signed-off-by: Lucas Stach <dev at lynxeye.de>
> ---
> arch/arm/Kconfig | 2 +
> arch/arm/dts/tegra20.dtsi | 16 +++
> arch/arm/mach-tegra/include/mach/gpio.h | 1 +
> drivers/gpio/Kconfig | 7 +
> drivers/gpio/Makefile | 1 +
> drivers/gpio/gpio-tegra.c | 226 ++++++++++++++++++++++++++++++++
> 6 files changed, 253 insertions(+)
> create mode 100644 arch/arm/mach-tegra/include/mach/gpio.h
> create mode 100644 drivers/gpio/gpio-tegra.c
>
> diff --git a/arch/arm/Kconfig b/arch/arm/Kconfig
> index e163e18..9151e66 100644
> --- a/arch/arm/Kconfig
> +++ b/arch/arm/Kconfig
> @@ -142,6 +142,8 @@ config ARCH_TEGRA
> select BUILTIN_DTB
> select COMMON_CLK
> select CLKDEV_LOOKUP
> + select GPIOLIB
> + select GPIO_TEGRA
> select OFTREE
>
> config ARCH_ZYNQ
> diff --git a/arch/arm/dts/tegra20.dtsi b/arch/arm/dts/tegra20.dtsi
> index 91858ec..b7d1e27 100644
> --- a/arch/arm/dts/tegra20.dtsi
> +++ b/arch/arm/dts/tegra20.dtsi
> @@ -18,6 +18,22 @@
> #clock-cells = <1>;
> };
>
> + gpio: gpio {
> + compatible = "nvidia,tegra20-gpio";
> + reg = <0x6000d000 0x1000>;
> + interrupts = <0 32 0x04
> + 0 33 0x04
> + 0 34 0x04
> + 0 35 0x04
> + 0 55 0x04
> + 0 87 0x04
> + 0 89 0x04>;
> + #gpio-cells = <2>;
> + gpio-controller;
> + #interrupt-cells = <2>;
> + interrupt-controller;
> + };
> +
> pmc {
> compatible = "nvidia,tegra20-pmc";
> reg = <0x7000e400 0x400>;
> diff --git a/arch/arm/mach-tegra/include/mach/gpio.h b/arch/arm/mach-tegra/include/mach/gpio.h
> new file mode 100644
> index 0000000..306ab4c
> --- /dev/null
> +++ b/arch/arm/mach-tegra/include/mach/gpio.h
> @@ -0,0 +1 @@
> +#include <asm-generic/gpio.h>
> diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig
> index ea07028..d5e0ed1 100644
> --- a/drivers/gpio/Kconfig
> +++ b/drivers/gpio/Kconfig
> @@ -36,6 +36,13 @@ config GPIO_PL061
> config GPIO_STMPE
> depends on MFD_STMPE
> bool "STMPE GPIO Expander"
> +
> +config GPIO_TEGRA
> + bool "GPIO support for the Tegra SoCs"
> + depends on ARCH_TEGRA
> + help
> + Say yes here to include the driver for the GPIO controller found on the
> + Tegra line of SoCs.
> endmenu
>
> endif
> diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile
> index 00acb68..5dcb6c8 100644
> --- a/drivers/gpio/Makefile
> +++ b/drivers/gpio/Makefile
> @@ -4,3 +4,4 @@ obj-$(CONFIG_GPIO_CLPS711X) += gpio-clps711x.o
> obj-$(CONFIG_GPIO_GENERIC) += gpio-generic.o
> obj-$(CONFIG_GPIO_PL061) += gpio-pl061.o
> obj-$(CONFIG_GPIO_STMPE) += gpio-stmpe.o
> +obj-$(CONFIG_GPIO_TEGRA) += gpio-tegra.o
> diff --git a/drivers/gpio/gpio-tegra.c b/drivers/gpio/gpio-tegra.c
> new file mode 100644
> index 0000000..55276f6
> --- /dev/null
> +++ b/drivers/gpio/gpio-tegra.c
> @@ -0,0 +1,226 @@
> +/* *
> + * Copyright (C) 2010 Erik Gilling <konkers at google.com>, Google, Inc
> + * Copyright (C) 2013 Lucas Stach <l.stach at pengutronix.de>
> + *
> + * This program is free software; you can redistribute it and/or modify it
> + * under the terms and conditions of the GNU General Public License,
> + * version 2, as published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope 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.
> + *
> + * You should have received a copy of the GNU General Public License
> + * along with this program. If not, see <http://www.gnu.org/licenses/>.
> + */
> +
> +#include <common.h>
> +#include <gpio.h>
> +#include <init.h>
> +#include <io.h>
> +#include <linux/err.h>
> +
> +#define GPIO_BANK(x) ((x) >> 5)
> +#define GPIO_PORT(x) (((x) >> 3) & 0x3)
> +#define GPIO_BIT(x) ((x) & 0x7)
> +
> +#define GPIO_REG(x) (GPIO_BANK(x) * config->bank_stride + \
> + GPIO_PORT(x) * 4)
> +
> +#define GPIO_CNF(x) (GPIO_REG(x) + 0x00)
> +#define GPIO_OE(x) (GPIO_REG(x) + 0x10)
> +#define GPIO_OUT(x) (GPIO_REG(x) + 0x20)
> +#define GPIO_IN(x) (GPIO_REG(x) + 0x30)
> +#define GPIO_INT_ENB(x) (GPIO_REG(x) + 0x50)
> +
> +#define GPIO_MSK_CNF(x) (GPIO_REG(x) + config->upper_offset + 0x00)
> +#define GPIO_MSK_OE(x) (GPIO_REG(x) + config->upper_offset + 0x10)
> +#define GPIO_MSK_OUT(x) (GPIO_REG(x) + config->upper_offset + 0X20)
> +
> +struct tegra_gpio_bank {
> + int bank;
> + int irq;
> +};
> +
> +struct tegra_gpio_soc_config {
> + u32 bank_stride;
> + u32 upper_offset;
> + u32 bank_count;
> +};
> +
> +static void __iomem *gpio_base;
> +static struct tegra_gpio_soc_config *config;
> +
> +static inline void tegra_gpio_writel(u32 val, u32 reg)
> +{
> + writel(val, gpio_base + reg);
> +}
> +
> +static inline u32 tegra_gpio_readl(u32 reg)
> +{
> + return readl(gpio_base + reg);
> +}
> +
> +static int tegra_gpio_compose(int bank, int port, int bit)
> +{
> + return (bank << 5) | ((port & 0x3) << 3) | (bit & 0x7);
> +}
> +
> +static void tegra_gpio_mask_write(u32 reg, int gpio, int value)
> +{
> + u32 val;
> +
> + val = 0x100 << GPIO_BIT(gpio);
> + if (value)
> + val |= 1 << GPIO_BIT(gpio);
> + tegra_gpio_writel(val, reg);
> +}
> +
> +static void tegra_gpio_enable(int gpio)
> +{
> + tegra_gpio_mask_write(GPIO_MSK_CNF(gpio), gpio, 1);
> +}
> +
> +static void tegra_gpio_disable(int gpio)
> +{
> + tegra_gpio_mask_write(GPIO_MSK_CNF(gpio), gpio, 0);
> +}
> +
> +static int tegra_gpio_request(struct gpio_chip *chip, unsigned offset)
> +{
> + return 0;
> +}
> +
> +static void tegra_gpio_free(struct gpio_chip *chip, unsigned offset)
> +{
> + tegra_gpio_disable(offset);
> +}
> +
> +static void tegra_gpio_set(struct gpio_chip *chip, unsigned offset, int value)
> +{
> + tegra_gpio_mask_write(GPIO_MSK_OUT(offset), offset, value);
> +}
> +
> +static int tegra_gpio_get(struct gpio_chip *chip, unsigned offset)
> +{
> + /* If gpio is in output mode then read from the out value */
> + if ((tegra_gpio_readl(GPIO_OE(offset)) >> GPIO_BIT(offset)) & 1) {
> + printf("GPIO output mode\n");
> + return (tegra_gpio_readl(GPIO_OUT(offset)) >>
> + GPIO_BIT(offset)) & 0x1;
> + }
> +
> + printf("GPIO input mode\n");
> + return (tegra_gpio_readl(GPIO_IN(offset)) >> GPIO_BIT(offset)) & 0x1;
> +}
> +
> +static int tegra_gpio_direction_input(struct gpio_chip *chip, unsigned offset)
> +{
> + tegra_gpio_mask_write(GPIO_MSK_OE(offset), offset, 0);
> + tegra_gpio_enable(offset);
> + return 0;
> +}
> +
> +static int tegra_gpio_direction_output(struct gpio_chip *chip, unsigned offset,
> + int value)
> +{
> + tegra_gpio_set(chip, offset, value);
> + tegra_gpio_mask_write(GPIO_MSK_OE(offset), offset, 1);
> + tegra_gpio_enable(offset);
> + return 0;
> +}
> +
> +static struct gpio_ops tegra_gpio_ops = {
> + .request = tegra_gpio_request,
> + .free = tegra_gpio_free,
> + .direction_input = tegra_gpio_direction_input,
> + .direction_output = tegra_gpio_direction_output,
> + .get = tegra_gpio_get,
> + .set = tegra_gpio_set,
> +};
> +
> +static struct gpio_chip tegra_gpio_chip = {
> + .ops = &tegra_gpio_ops,
> + .base = 0,
> +};
> +
> +static int tegra_gpio_probe(struct device_d *dev)
> +{
> + int i, j, ret;
> +
> + ret = dev_get_drvdata(dev, (unsigned long *)&config);
> + if (ret) {
> + dev_err(dev, "dev_get_drvdata failed: %d\n", ret);
> + return ret;
> + }
> +
> + gpio_base = dev_request_mem_region(dev, 0);
> + if (!gpio_base) {
> + dev_err(dev, "could not get memory region\n");
> + return -ENODEV;
> + }
> +
> + for (i = 0; i < config->bank_count; i++) {
> + for (j = 0; j < 4; j++) {
> + int gpio = tegra_gpio_compose(i, j, 0);
> + tegra_gpio_writel(0x00, GPIO_INT_ENB(gpio));
> + }
> + }
> +
> + tegra_gpio_chip.ngpio = config->bank_count * 32;
> + tegra_gpio_chip.dev = dev;
> +
> + gpiochip_add(&tegra_gpio_chip);
> +
> + return 0;
> +}
> +
> +static struct tegra_gpio_soc_config tegra20_gpio_config = {
> + .bank_stride = 0x80,
> + .upper_offset = 0x800,
> + .bank_count = 7,
> +};
> +
> +static struct tegra_gpio_soc_config tegra30_gpio_config = {
> + .bank_stride = 0x100,
> + .upper_offset = 0x80,
> + .bank_count = 8,
> +};
Hmmm, tegra30 support here... But we don't have tegra30 boards yet.
> +
> +static struct platform_device_id tegra_gpio_ids[] = {
> + {
> + .name = "tegra20-gpio",
> + .driver_data = (unsigned long)&tegra20_gpio_config,
> + }, {
> + .name = "tegra30-gpio",
> + .driver_data = (unsigned long)&tegra30_gpio_config,
and here ...
> + }, {
> + /* sentinel */
> + },
> +};
> +
> +static __maybe_unused struct of_device_id tegra_gpio_dt_ids[] = {
> + {
> + .compatible = "nvidia,tegra30-gpio",
> + .data = (unsigned long)&tegra30_gpio_config
> + }, {
and here.
> + .compatible = "nvidia,tegra20-gpio",
> + .data = (unsigned long)&tegra20_gpio_config
> + }, {
> + /* sentinel */
> + },
> +};
> +
> +static struct driver_d tegra_gpio_driver = {
> + .name = "tegra-gpio",
> + .id_table = tegra_gpio_ids,
> + .of_compatible = DRV_OF_COMPAT(tegra_gpio_dt_ids),
> + .probe = tegra_gpio_probe,
> +};
> +
> +static int __init tegra_gpio_init(void)
> +{
> + return platform_driver_register(&tegra_gpio_driver);
> +}
> +coredevice_initcall(tegra_gpio_init);
> --
> 1.8.1.4
>
--
--
Best regards,
Antony Pavlov
More information about the barebox
mailing list