* [PATCH] au1000: convert to using gpiolib @ 2009-01-15 15:46 Florian Fainelli 2009-01-15 20:58 ` Manuel Lauss 2009-02-26 14:35 ` Ralf Baechle 0 siblings, 2 replies; 12+ messages in thread From: Florian Fainelli @ 2009-01-15 15:46 UTC (permalink / raw) To: ralf, Manuel Lauss, Linux-MIPS This patch converts the GPIO board code to use gpiolib. Signed-off-by: Florian Fainelli <florian@openwrt.org> --- diff --git a/arch/mips/alchemy/Kconfig b/arch/mips/alchemy/Kconfig index 7f8ef13..2fc5c13 100644 --- a/arch/mips/alchemy/Kconfig +++ b/arch/mips/alchemy/Kconfig @@ -135,3 +135,4 @@ config SOC_AU1X00 select SYS_SUPPORTS_32BIT_KERNEL select SYS_SUPPORTS_APM_EMULATION select GENERIC_HARDIRQS_NO__DO_IRQ + select ARCH_REQUIRE_GPIOLIB diff --git a/arch/mips/alchemy/common/gpio.c b/arch/mips/alchemy/common/gpio.c index e660ddd..46be37b 100644 --- a/arch/mips/alchemy/common/gpio.c +++ b/arch/mips/alchemy/common/gpio.c @@ -1,5 +1,5 @@ /* - * Copyright (C) 2007, OpenWrt.org, Florian Fainelli <florian@openwrt.org> + * Copyright (C) 2007-2009, OpenWrt.org, Florian Fainelli <florian@openwrt.org> * Architecture specific GPIO support * * This program is free software; you can redistribute it and/or modify it @@ -27,122 +27,175 @@ * others have a second one : GPIO2 */ +#include <linux/kernel.h> #include <linux/module.h> +#include <linux/types.h> +#include <linux/platform_device.h> +#include <linux/gpio.h> #include <asm/mach-au1x00/au1000.h> #include <asm/gpio.h> -#define gpio1 sys -#if !defined(CONFIG_SOC_AU1000) - -static struct au1x00_gpio2 *const gpio2 = (struct au1x00_gpio2 *) GPIO2_BASE; -#define GPIO2_OUTPUT_ENABLE_MASK 0x00010000 +struct au1000_gpio_chip { + struct gpio_chip chip; + void __iomem *regbase; +}; -static int au1xxx_gpio2_read(unsigned gpio) +#if !defined(CONFIG_SOC_AU1000) +static int au1000_gpio2_get(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - return ((gpio2->pinstate >> gpio) & 0x01); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO2_ST) & mask; } -static void au1xxx_gpio2_write(unsigned gpio, int value) +static void au1000_gpio2_set(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; + u32 mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + local_irq_save(flags); + writel(mask, gpch->regbase + AU1000_GPIO2_OUT); + local_irq_restore(flags); } -static int au1xxx_gpio2_direction_input(unsigned gpio) +static int au1000_gpio2_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir &= ~(0x01 << gpio); + u32 mask = 1 << offset; + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp &= ~mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + local_irq_restore(flags); + return 0; } -static int au1xxx_gpio2_direction_output(unsigned gpio, int value) +static int au1000_gpio2_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir |= 0x01 << gpio; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + u32 mask = 1 << offset; + u32 out_mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp |= mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + writel(out_mask, gpch->regbase + AU1000_GPIO2_OUT); + local_irq_restore(flags); + return 0; } - #endif /* !defined(CONFIG_SOC_AU1000) */ -static int au1xxx_gpio1_read(unsigned gpio) +static int au1000_gpio1_get(struct gpio_chip *chip, unsigned offset) { - return (gpio1->pinstaterd >> gpio) & 0x01; + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO1_ST) & mask; } -static void au1xxx_gpio1_write(unsigned gpio, int value) +static void au1000_gpio1_set(struct gpio_chip *chip, + unsigned offset, int value) { + u32 mask = 1 << offset; + u32 reg_offset; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + if (value) - gpio1->outputset = (0x01 << gpio); + reg_offset = AU1000_GPIO1_OUT; else - /* Output a zero */ - gpio1->outputclr = (0x01 << gpio); + reg_offset = AU1000_GPIO1_CLR; + + local_irq_save(flags); + writel(mask, gpch->regbase + reg_offset); + local_irq_restore(flags); } -static int au1xxx_gpio1_direction_input(unsigned gpio) +static int au1000_gpio1_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio1->pininputen = (0x01 << gpio); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO1_ST); + return 0; } -static int au1xxx_gpio1_direction_output(unsigned gpio, int value) +static int au1000_gpio1_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - gpio1->trioutclr = (0x01 & gpio); - au1xxx_gpio1_write(gpio, value); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + writel(mask, gpch->regbase + AU1000_GPIO1_TRI_OUT); + au1000_gpio1_set(chip, offset, value); + return 0; } -int au1xxx_gpio_get_value(unsigned gpio) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return 0; -#else - return au1xxx_gpio2_read(gpio); +struct au1000_gpio_chip au1000_gpio_chip[] = { + [0] = { + .regbase = (void __iomem *)SYS_BASE, + .chip = { + .label = "au1000-gpio1", + .direction_input = au1000_gpio1_direction_input, + .direction_output = au1000_gpio1_direction_output, + .get = au1000_gpio1_get, + .set = au1000_gpio1_set, + .base = 0, + .ngpio = 32, + }, + }, +#if !defined(CONFIG_SOC_AU1000) + [1] = { + .regbase = (void __iomem *)GPIO2_BASE, + .chip = { + .label = "au1000-gpio2", + .direction_input = au1000_gpio2_direction_input, + .direction_output = au1000_gpio2_direction_output, + .get = au1000_gpio2_get, + .set = au1000_gpio2_set, + .base = AU1XXX_GPIO_BASE, + .ngpio = 32, + }, + }, #endif - else - return au1xxx_gpio1_read(gpio); -} -EXPORT_SYMBOL(au1xxx_gpio_get_value); +}; -void au1xxx_gpio_set_value(unsigned gpio, int value) +static int __init au1000_gpio_init(void) { - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - ; -#else - au1xxx_gpio2_write(gpio, value); -#endif - else - au1xxx_gpio1_write(gpio, value); -} -EXPORT_SYMBOL(au1xxx_gpio_set_value); + gpiochip_add(&au1000_gpio_chip[0].chip); +#if !defined(CONFIG_SOC_AU1000) + gpiochip_add(&au1000_gpio_chip[1].chip); -int au1xxx_gpio_direction_input(unsigned gpio) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_input(gpio); + return 0; #endif - - return au1xxx_gpio1_direction_input(gpio); } -EXPORT_SYMBOL(au1xxx_gpio_direction_input); +arch_initcall(au1000_gpio_init); -int au1xxx_gpio_direction_output(unsigned gpio, int value) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_output(gpio, value); -#endif - - return au1xxx_gpio1_direction_output(gpio, value); -} -EXPORT_SYMBOL(au1xxx_gpio_direction_output); diff --git a/arch/mips/include/asm/mach-au1x00/gpio.h b/arch/mips/include/asm/mach-au1x00/gpio.h index 2dc61e0..34d9b72 100644 --- a/arch/mips/include/asm/mach-au1x00/gpio.h +++ b/arch/mips/include/asm/mach-au1x00/gpio.h @@ -5,65 +5,29 @@ #define AU1XXX_GPIO_BASE 200 -struct au1x00_gpio2 { - u32 dir; - u32 reserved; - u32 output; - u32 pinstate; - u32 inten; - u32 enable; -}; +/* GPIO bank 1 offsets */ +#define AU1000_GPIO1_TRI_OUT 0x0100 +#define AU1000_GPIO1_OUT 0x0108 +#define AU1000_GPIO1_ST 0x0110 +#define AU1000_GPIO1_CLR 0x010C -extern int au1xxx_gpio_get_value(unsigned gpio); -extern void au1xxx_gpio_set_value(unsigned gpio, int value); -extern int au1xxx_gpio_direction_input(unsigned gpio); -extern int au1xxx_gpio_direction_output(unsigned gpio, int value); +/* GPIO bank 2 offsets */ +#define AU1000_GPIO2_DIR 0x00 +#define AU1000_GPIO2_RSVD 0x04 +#define AU1000_GPIO2_OUT 0x08 +#define AU1000_GPIO2_ST 0x0C +#define AU1000_GPIO2_INT 0x10 +#define AU1000_GPIO2_EN 0x14 +#define GPIO2_OUT_EN_MASK 0x00010000 -/* Wrappers for the arch-neutral GPIO API */ +#define gpio_to_irq(gpio) NULL -static inline int gpio_request(unsigned gpio, const char *label) -{ - /* Not yet implemented */ - return 0; -} +#define gpio_get_value __gpio_get_value +#define gpio_set_value __gpio_set_value -static inline void gpio_free(unsigned gpio) -{ - /* Not yet implemented */ -} +#define gpio_cansleep __gpio_cansleep -static inline int gpio_direction_input(unsigned gpio) -{ - return au1xxx_gpio_direction_input(gpio); -} - -static inline int gpio_direction_output(unsigned gpio, int value) -{ - return au1xxx_gpio_direction_output(gpio, value); -} - -static inline int gpio_get_value(unsigned gpio) -{ - return au1xxx_gpio_get_value(gpio); -} - -static inline void gpio_set_value(unsigned gpio, int value) -{ - au1xxx_gpio_set_value(gpio, value); -} - -static inline int gpio_to_irq(unsigned gpio) -{ - return gpio; -} - -static inline int irq_to_gpio(unsigned irq) -{ - return irq; -} - -/* For cansleep */ #include <asm-generic/gpio.h> #endif /* _AU1XXX_GPIO_H_ */ ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-15 15:46 [PATCH] au1000: convert to using gpiolib Florian Fainelli @ 2009-01-15 20:58 ` Manuel Lauss 2009-01-16 10:10 ` Florian Fainelli 2009-02-26 14:35 ` Ralf Baechle 1 sibling, 1 reply; 12+ messages in thread From: Manuel Lauss @ 2009-01-15 20:58 UTC (permalink / raw) To: Florian Fainelli; +Cc: ralf, Linux-MIPS Hi Florian, On Thu, Jan 15, 2009 at 04:46:48PM +0100, Florian Fainelli wrote: > This patch converts the GPIO board code to use gpiolib Very nice; a few minor things though: > --- a/arch/mips/alchemy/common/gpio.c > +++ b/arch/mips/alchemy/common/gpio.c > -static void au1xxx_gpio2_write(unsigned gpio, int value) > +static void au1000_gpio2_set(struct gpio_chip *chip, > + unsigned offset, int value) > { > - gpio -= AU1XXX_GPIO_BASE; > + u32 mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); > + struct au1000_gpio_chip *gpch; > + unsigned long flags; > + > + gpch = container_of(chip, struct au1000_gpio_chip, chip); > > - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); > + local_irq_save(flags); > + writel(mask, gpch->regbase + AU1000_GPIO2_OUT); > + local_irq_restore(flags); Because of the 'enable | value' scheme I believe you don't require any locking here. > +struct au1000_gpio_chip au1000_gpio_chip[] = { > + [0] = { > + .regbase = (void __iomem *)SYS_BASE, > + .chip = { > + .label = "au1000-gpio1", > + .direction_input = au1000_gpio1_direction_input, > + .direction_output = au1000_gpio1_direction_output, > + .get = au1000_gpio1_get, > + .set = au1000_gpio1_set, > + .base = 0, > + .ngpio = 32, > + }, > + }, > +#if !defined(CONFIG_SOC_AU1000) > + [1] = { > + .regbase = (void __iomem *)GPIO2_BASE, > + .chip = { > + .label = "au1000-gpio2", > + .direction_input = au1000_gpio2_direction_input, > + .direction_output = au1000_gpio2_direction_output, > + .get = au1000_gpio2_get, > + .set = au1000_gpio2_set, > + .base = AU1XXX_GPIO_BASE, > + .ngpio = 32, > + }, > + }, > #endif > - else > - return au1xxx_gpio1_read(gpio); > -} > -EXPORT_SYMBOL(au1xxx_gpio_get_value); > +}; [...] > +static int __init au1000_gpio_init(void) > { > - if (gpio >= AU1XXX_GPIO_BASE) > -#if defined(CONFIG_SOC_AU1000) > - ; > -#else > - au1xxx_gpio2_write(gpio, value); > -#endif > - else > - au1xxx_gpio1_write(gpio, value); > -} > -EXPORT_SYMBOL(au1xxx_gpio_set_value); > + gpiochip_add(&au1000_gpio_chip[0].chip); > +#if !defined(CONFIG_SOC_AU1000) > + gpiochip_add(&au1000_gpio_chip[1].chip); > [...] > +arch_initcall(au1000_gpio_init); Can you please make the gpiolib registration dependent on a CONFIG symbol? I.e. make the au1000_gpio{,2}_direction() and friends calls globally visible but let the individual boards decide whether they want to use the gpio numbering imposed by this patch. Long explanation: I maintain a number of modules with a common connector interface, based on different architectures (sh, mips and arm so far). I also maintain a few baseboards which can carry theese modules. Modules provide 16 gpios numbered 0-15, which are used by the baseboards. Since I need to keep the baseboard code free of arch-specific hacks, every module provides its own gpio-chip which distributes the gpio-lib calls to various on- and off-chip peripherals. On my alchemy board in particular, those 16 gpios are provided by a mixture of gpio1, gpio2 and FPGA based pins (yes I repeatedly moanoed to the hw guys about this but pin multiplexing and required features make a sane implementation difficult; but at least I was allowed to write the VHDL for the fpga-based gpios). If this explanation doesn't make sense I'll gladly whip up an addon patch. > diff --git a/arch/mips/include/asm/mach-au1x00/gpio.h b/arch/mips/include/asm/mach-au1x00/gpio.h > index 2dc61e0..34d9b72 100644 > --- a/arch/mips/include/asm/mach-au1x00/gpio.h > +++ b/arch/mips/include/asm/mach-au1x00/gpio.h > @@ -5,65 +5,29 @@ > > #define AU1XXX_GPIO_BASE 200 please change this to AU1XXX_GPIO2_BASE (it's the base number of the GPIO2 block pins after all) Thanks! Manuel Lauss ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-15 20:58 ` Manuel Lauss @ 2009-01-16 10:10 ` Florian Fainelli 2009-01-16 17:47 ` Manuel Lauss 0 siblings, 1 reply; 12+ messages in thread From: Florian Fainelli @ 2009-01-16 10:10 UTC (permalink / raw) To: Manuel Lauss; +Cc: ralf, Linux-MIPS [-- Attachment #1: Type: text/plain, Size: 4236 bytes --] Hi Manuel, 2009/1/15 Manuel Lauss <mano@roarinelk.homelinux.net> > Hi Florian, > > On Thu, Jan 15, 2009 at 04:46:48PM +0100, Florian Fainelli wrote: > > This patch converts the GPIO board code to use gpiolib > > Because of the 'enable | value' scheme I believe you don't require any > locking here. Right, thanks. > > > > > +struct au1000_gpio_chip au1000_gpio_chip[] = { > > + [0] = { > > + .regbase = (void __iomem *)SYS_BASE, > > + .chip = { > > + .label = "au1000-gpio1", > > + .direction_input = > au1000_gpio1_direction_input, > > + .direction_output = > au1000_gpio1_direction_output, > > + .get = au1000_gpio1_get, > > + .set = au1000_gpio1_set, > > + .base = 0, > > + .ngpio = 32, > > + }, > > + }, > > +#if !defined(CONFIG_SOC_AU1000) > > + [1] = { > > + .regbase = (void __iomem > *)GPIO2_BASE, > > + .chip = { > > + .label = "au1000-gpio2", > > + .direction_input = > au1000_gpio2_direction_input, > > + .direction_output = > au1000_gpio2_direction_output, > > + .get = au1000_gpio2_get, > > + .set = au1000_gpio2_set, > > + .base = AU1XXX_GPIO_BASE, > > + .ngpio = 32, > > + }, > > + }, > > #endif > > - else > > - return au1xxx_gpio1_read(gpio); > > -} > > -EXPORT_SYMBOL(au1xxx_gpio_get_value); > > +}; > [...] > > +static int __init au1000_gpio_init(void) > > { > > - if (gpio >= AU1XXX_GPIO_BASE) > > -#if defined(CONFIG_SOC_AU1000) > > - ; > > -#else > > - au1xxx_gpio2_write(gpio, value); > > -#endif > > - else > > - au1xxx_gpio1_write(gpio, value); > > -} > > -EXPORT_SYMBOL(au1xxx_gpio_set_value); > > + gpiochip_add(&au1000_gpio_chip[0].chip); > > +#if !defined(CONFIG_SOC_AU1000) > > + gpiochip_add(&au1000_gpio_chip[1].chip); > > > [...] > > +arch_initcall(au1000_gpio_init); > > Can you please make the gpiolib registration dependent on a > CONFIG symbol? I.e. make the au1000_gpio{,2}_direction() and > friends calls globally visible but let the individual boards > decide whether they want to use the gpio numbering imposed by > this patch. Would something like #ifdef CONFIG_AU1000_NON_STD_GPIOS be ok with you ? Or maybe we could get the base information from board-specific code ? > > > Long explanation: I maintain a number of modules with a common connector > interface, based on different architectures (sh, mips and arm so far). > I also maintain a few baseboards which can carry theese modules. Modules > provide 16 gpios numbered 0-15, which are used by the baseboards. Since > I need to keep the baseboard code free of arch-specific hacks, every module > provides its own gpio-chip which distributes the gpio-lib calls to various > on- and off-chip peripherals. On my alchemy board in particular, those 16 > gpios are provided by a mixture of gpio1, gpio2 and FPGA based pins (yes > I repeatedly moanoed to the hw guys about this but pin multiplexing and > required features make a sane implementation difficult; but at least I was > allowed to write the VHDL for the fpga-based gpios). Your explanation makes perfect sense to me. > > > If this explanation doesn't make sense I'll gladly whip up an addon patch. > > > > diff --git a/arch/mips/include/asm/mach-au1x00/gpio.h > b/arch/mips/include/asm/mach-au1x00/gpio.h > > index 2dc61e0..34d9b72 100644 > > --- a/arch/mips/include/asm/mach-au1x00/gpio.h > > +++ b/arch/mips/include/asm/mach-au1x00/gpio.h > > @@ -5,65 +5,29 @@ > > > > #define AU1XXX_GPIO_BASE 200 > > please change this to AU1XXX_GPIO2_BASE (it's the base number > of the GPIO2 block pins after all) Ok. Thanks for your comments, I will respin with your comments. [-- Attachment #2: Type: text/html, Size: 7423 bytes --] ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-16 10:10 ` Florian Fainelli @ 2009-01-16 17:47 ` Manuel Lauss 2009-01-19 17:12 ` Florian Fainelli 0 siblings, 1 reply; 12+ messages in thread From: Manuel Lauss @ 2009-01-16 17:47 UTC (permalink / raw) To: Florian Fainelli; +Cc: ralf, Linux-MIPS Hi Florian, On Fri, Jan 16, 2009 at 11:10:25AM +0100, Florian Fainelli wrote: > > Can you please make the gpiolib registration dependent on a > > CONFIG symbol? I.e. make the au1000_gpio{,2}_direction() and > > friends calls globally visible but let the individual boards > > decide whether they want to use the gpio numbering imposed by > > this patch. > > > Would something like #ifdef CONFIG_AU1000_NON_STD_GPIOS be ok with you ? > Or maybe we could get the base information from board-specific code ? Well, we could move the core_initcall() to all boards' setup function, and stick a comment on top of it explaining why it's there. Config symbol or the above, either is fine with me. Thanks! Manuel Lauss ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-16 17:47 ` Manuel Lauss @ 2009-01-19 17:12 ` Florian Fainelli 2009-01-26 22:40 ` Florian Fainelli 0 siblings, 1 reply; 12+ messages in thread From: Florian Fainelli @ 2009-01-19 17:12 UTC (permalink / raw) To: Manuel Lauss; +Cc: ralf, Linux-MIPS Hi Manuel, Le Friday 16 January 2009 18:47:53 Manuel Lauss, vous avez écrit : > Hi Florian, > > On Fri, Jan 16, 2009 at 11:10:25AM +0100, Florian Fainelli wrote: > > > Can you please make the gpiolib registration dependent on a > > > CONFIG symbol? I.e. make the au1000_gpio{,2}_direction() and > > > friends calls globally visible but let the individual boards > > > decide whether they want to use the gpio numbering imposed by > > > this patch. > > > > Would something like #ifdef CONFIG_AU1000_NON_STD_GPIOS be ok with you ? > > Or maybe we could get the base information from board-specific code ? > > Well, we could move the core_initcall() to all boards' setup function, > and stick a comment on top of it explaining why it's there. > > Config symbol or the above, either is fine with me. I would prefer the config symbol, there is only one file to change ;). Patch attached below: -- From: Florian Fainelli <florian@openwrt.org> Subject: [PATCH v2] au1000: convert to using gpiolib This patch converts the GPIO board code to use gpiolib. Changes from v1: - allow users not to use the default gpio accessors - do not lock au1000_gpio2_set Signed-off-by: Florian Fainelli <florian@openwrt.org> -- diff --git a/arch/mips/alchemy/Kconfig b/arch/mips/alchemy/Kconfig index 7f8ef13..2fc5c13 100644 --- a/arch/mips/alchemy/Kconfig +++ b/arch/mips/alchemy/Kconfig @@ -135,3 +135,4 @@ config SOC_AU1X00 select SYS_SUPPORTS_32BIT_KERNEL select SYS_SUPPORTS_APM_EMULATION select GENERIC_HARDIRQS_NO__DO_IRQ + select ARCH_REQUIRE_GPIOLIB diff --git a/arch/mips/alchemy/common/gpio.c b/arch/mips/alchemy/common/gpio.c index e660ddd..9ac1e4c 100644 --- a/arch/mips/alchemy/common/gpio.c +++ b/arch/mips/alchemy/common/gpio.c @@ -1,5 +1,5 @@ /* - * Copyright (C) 2007, OpenWrt.org, Florian Fainelli <florian@openwrt.org> + * Copyright (C) 2007-2009, OpenWrt.org, Florian Fainelli <florian@openwrt.org> * Architecture specific GPIO support * * This program is free software; you can redistribute it and/or modify it @@ -27,122 +27,172 @@ * others have a second one : GPIO2 */ +#include <linux/kernel.h> #include <linux/module.h> +#include <linux/types.h> +#include <linux/platform_device.h> +#include <linux/gpio.h> #include <asm/mach-au1x00/au1000.h> #include <asm/gpio.h> -#define gpio1 sys -#if !defined(CONFIG_SOC_AU1000) - -static struct au1x00_gpio2 *const gpio2 = (struct au1x00_gpio2 *) GPIO2_BASE; -#define GPIO2_OUTPUT_ENABLE_MASK 0x00010000 +struct au1000_gpio_chip { + struct gpio_chip chip; + void __iomem *regbase; +}; -static int au1xxx_gpio2_read(unsigned gpio) +#if !defined(CONFIG_SOC_AU1000) +static int au1000_gpio2_get(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - return ((gpio2->pinstate >> gpio) & 0x01); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO2_ST) & mask; } -static void au1xxx_gpio2_write(unsigned gpio, int value) +static void au1000_gpio2_set(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; + u32 mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + struct au1000_gpio_chip *gpch; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO2_OUT); } -static int au1xxx_gpio2_direction_input(unsigned gpio) +static int au1000_gpio2_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir &= ~(0x01 << gpio); + u32 mask = 1 << offset; + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp &= ~mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + local_irq_restore(flags); + return 0; } -static int au1xxx_gpio2_direction_output(unsigned gpio, int value) +static int au1000_gpio2_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir |= 0x01 << gpio; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + u32 mask = 1 << offset; + u32 out_mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp |= mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + writel(out_mask, gpch->regbase + AU1000_GPIO2_OUT); + local_irq_restore(flags); + return 0; } - #endif /* !defined(CONFIG_SOC_AU1000) */ -static int au1xxx_gpio1_read(unsigned gpio) +static int au1000_gpio1_get(struct gpio_chip *chip, unsigned offset) { - return (gpio1->pinstaterd >> gpio) & 0x01; + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO1_ST) & mask; } -static void au1xxx_gpio1_write(unsigned gpio, int value) +static void au1000_gpio1_set(struct gpio_chip *chip, + unsigned offset, int value) { + u32 mask = 1 << offset; + u32 reg_offset; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + if (value) - gpio1->outputset = (0x01 << gpio); + reg_offset = AU1000_GPIO1_OUT; else - /* Output a zero */ - gpio1->outputclr = (0x01 << gpio); + reg_offset = AU1000_GPIO1_CLR; + + local_irq_save(flags); + writel(mask, gpch->regbase + reg_offset); + local_irq_restore(flags); } -static int au1xxx_gpio1_direction_input(unsigned gpio) +static int au1000_gpio1_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio1->pininputen = (0x01 << gpio); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO1_ST); + return 0; } -static int au1xxx_gpio1_direction_output(unsigned gpio, int value) +static int au1000_gpio1_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - gpio1->trioutclr = (0x01 & gpio); - au1xxx_gpio1_write(gpio, value); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + writel(mask, gpch->regbase + AU1000_GPIO1_TRI_OUT); + au1000_gpio1_set(chip, offset, value); + return 0; } -int au1xxx_gpio_get_value(unsigned gpio) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return 0; -#else - return au1xxx_gpio2_read(gpio); +struct au1000_gpio_chip au1000_gpio_chip[] = { + [0] = { + .regbase = (void __iomem *)SYS_BASE, + .chip = { + .label = "au1000-gpio1", + .direction_input = au1000_gpio1_direction_input, + .direction_output = au1000_gpio1_direction_output, + .get = au1000_gpio1_get, + .set = au1000_gpio1_set, + .base = 0, + .ngpio = 32, + }, + }, +#if !defined(CONFIG_SOC_AU1000) + [1] = { + .regbase = (void __iomem *)GPIO2_BASE, + .chip = { + .label = "au1000-gpio2", + .direction_input = au1000_gpio2_direction_input, + .direction_output = au1000_gpio2_direction_output, + .get = au1000_gpio2_get, + .set = au1000_gpio2_set, + .base = AU1XXX_GPIO2_BASE, + .ngpio = 32, + }, + }, #endif - else - return au1xxx_gpio1_read(gpio); -} -EXPORT_SYMBOL(au1xxx_gpio_get_value); +}; -void au1xxx_gpio_set_value(unsigned gpio, int value) +static int __init au1000_gpio_init(void) { - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - ; -#else - au1xxx_gpio2_write(gpio, value); -#endif - else - au1xxx_gpio1_write(gpio, value); -} -EXPORT_SYMBOL(au1xxx_gpio_set_value); + gpiochip_add(&au1000_gpio_chip[0].chip); +#if !defined(CONFIG_SOC_AU1000) + gpiochip_add(&au1000_gpio_chip[1].chip); -int au1xxx_gpio_direction_input(unsigned gpio) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_input(gpio); + return 0; #endif - - return au1xxx_gpio1_direction_input(gpio); } -EXPORT_SYMBOL(au1xxx_gpio_direction_input); - -int au1xxx_gpio_direction_output(unsigned gpio, int value) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_output(gpio, value); +#ifndef CONFIG_AU1X00_NON_STD_GPIOS +arch_initcall(au1000_gpio_init); #endif - - return au1xxx_gpio1_direction_output(gpio, value); -} -EXPORT_SYMBOL(au1xxx_gpio_direction_output); diff --git a/arch/mips/include/asm/mach-au1x00/gpio.h b/arch/mips/include/asm/mach-au1x00/gpio.h index 2dc61e0..f1a5d48 100644 --- a/arch/mips/include/asm/mach-au1x00/gpio.h +++ b/arch/mips/include/asm/mach-au1x00/gpio.h @@ -3,67 +3,31 @@ #include <linux/types.h> -#define AU1XXX_GPIO_BASE 200 +#define AU1XXX_GPIO2_BASE 200 -struct au1x00_gpio2 { - u32 dir; - u32 reserved; - u32 output; - u32 pinstate; - u32 inten; - u32 enable; -}; +/* GPIO bank 1 offsets */ +#define AU1000_GPIO1_TRI_OUT 0x0100 +#define AU1000_GPIO1_OUT 0x0108 +#define AU1000_GPIO1_ST 0x0110 +#define AU1000_GPIO1_CLR 0x010C -extern int au1xxx_gpio_get_value(unsigned gpio); -extern void au1xxx_gpio_set_value(unsigned gpio, int value); -extern int au1xxx_gpio_direction_input(unsigned gpio); -extern int au1xxx_gpio_direction_output(unsigned gpio, int value); +/* GPIO bank 2 offsets */ +#define AU1000_GPIO2_DIR 0x00 +#define AU1000_GPIO2_RSVD 0x04 +#define AU1000_GPIO2_OUT 0x08 +#define AU1000_GPIO2_ST 0x0C +#define AU1000_GPIO2_INT 0x10 +#define AU1000_GPIO2_EN 0x14 +#define GPIO2_OUT_EN_MASK 0x00010000 -/* Wrappers for the arch-neutral GPIO API */ +#define gpio_to_irq(gpio) NULL -static inline int gpio_request(unsigned gpio, const char *label) -{ - /* Not yet implemented */ - return 0; -} +#define gpio_get_value __gpio_get_value +#define gpio_set_value __gpio_set_value -static inline void gpio_free(unsigned gpio) -{ - /* Not yet implemented */ -} +#define gpio_cansleep __gpio_cansleep -static inline int gpio_direction_input(unsigned gpio) -{ - return au1xxx_gpio_direction_input(gpio); -} - -static inline int gpio_direction_output(unsigned gpio, int value) -{ - return au1xxx_gpio_direction_output(gpio, value); -} - -static inline int gpio_get_value(unsigned gpio) -{ - return au1xxx_gpio_get_value(gpio); -} - -static inline void gpio_set_value(unsigned gpio, int value) -{ - au1xxx_gpio_set_value(gpio, value); -} - -static inline int gpio_to_irq(unsigned gpio) -{ - return gpio; -} - -static inline int irq_to_gpio(unsigned irq) -{ - return irq; -} - -/* For cansleep */ #include <asm-generic/gpio.h> #endif /* _AU1XXX_GPIO_H_ */ -- Best regards, Florian Fainelli Email : florian@openwrt.org http://openwrt.org ------------------------------- ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-19 17:12 ` Florian Fainelli @ 2009-01-26 22:40 ` Florian Fainelli 2009-01-27 6:58 ` Manuel Lauss 0 siblings, 1 reply; 12+ messages in thread From: Florian Fainelli @ 2009-01-26 22:40 UTC (permalink / raw) To: Manuel Lauss; +Cc: ralf, Linux-MIPS Hi Manuel, Le Monday 19 January 2009 18:12:23 Florian Fainelli, vous avez écrit : > This patch converts the GPIO board code to use gpiolib. > Changes from v1: > - allow users not to use the default gpio accessors > - do not lock au1000_gpio2_set I did not receive comments from you on this patch, can I consider it being ok ? -- Best regards, Florian Fainelli Email : florian@openwrt.org http://openwrt.org ------------------------------- ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-26 22:40 ` Florian Fainelli @ 2009-01-27 6:58 ` Manuel Lauss 2009-01-27 9:39 ` Florian Fainelli 0 siblings, 1 reply; 12+ messages in thread From: Manuel Lauss @ 2009-01-27 6:58 UTC (permalink / raw) To: Florian Fainelli; +Cc: ralf, Linux-MIPS Florian, > Le Monday 19 January 2009 18:12:23 Florian Fainelli, vous avez ?crit?: > > This patch converts the GPIO board code to use gpiolib. > > Changes from v1: > > - allow users not to use the default gpio accessors > > - do not lock au1000_gpio2_set > > I did not receive comments from you on this patch, can I consider it being > ok ? Oh sorry. Please export all the au1000_gpioX_* functions and I'm happy. Thanks! Manuel Lauss ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-27 6:58 ` Manuel Lauss @ 2009-01-27 9:39 ` Florian Fainelli 2009-01-27 9:47 ` Manuel Lauss 0 siblings, 1 reply; 12+ messages in thread From: Florian Fainelli @ 2009-01-27 9:39 UTC (permalink / raw) To: Manuel Lauss; +Cc: ralf, Linux-MIPS Le Tuesday 27 January 2009 07:58:27 Manuel Lauss, vous avez écrit : > Florian, > > > Le Monday 19 January 2009 18:12:23 Florian Fainelli, vous avez ?crit?: > > > This patch converts the GPIO board code to use gpiolib. > > > Changes from v1: > > > - allow users not to use the default gpio accessors > > > - do not lock au1000_gpio2_set > > > > I did not receive comments from you on this patch, can I consider it > > being ok ? > > Oh sorry. Please export all the au1000_gpioX_* functions and I'm happy. > > Thanks! > Manuel Lauss Here we go. Thanks ! -- From: Florian Fainelli <florian@openwrt.org> Subject: [PATCH v3] Alchemy: convert to gpiolib This patch converts Alchemy SoC GPIO code to use gpiolib instead of the old GENERIC_GPIO implementation. Changes from v2: - export all gpio accessors so that custom boards can use them Changes from v1: - allow users not to use the default gpio accessors - do not lock au1000_gpio2_set Signed-off-by: Florian Fainelli <florian@openwrt.org> --- diff --git a/arch/mips/alchemy/Kconfig b/arch/mips/alchemy/Kconfig index 7f8ef13..2fc5c13 100644 --- a/arch/mips/alchemy/Kconfig +++ b/arch/mips/alchemy/Kconfig @@ -135,3 +135,4 @@ config SOC_AU1X00 select SYS_SUPPORTS_32BIT_KERNEL select SYS_SUPPORTS_APM_EMULATION select GENERIC_HARDIRQS_NO__DO_IRQ + select ARCH_REQUIRE_GPIOLIB diff --git a/arch/mips/alchemy/common/gpio.c b/arch/mips/alchemy/common/gpio.c index e660ddd..535c565 100644 --- a/arch/mips/alchemy/common/gpio.c +++ b/arch/mips/alchemy/common/gpio.c @@ -1,5 +1,5 @@ /* - * Copyright (C) 2007, OpenWrt.org, Florian Fainelli <florian@openwrt.org> + * Copyright (C) 2007-2009, OpenWrt.org, Florian Fainelli <florian@openwrt.org> * Architecture specific GPIO support * * This program is free software; you can redistribute it and/or modify it @@ -27,122 +27,180 @@ * others have a second one : GPIO2 */ +#include <linux/kernel.h> #include <linux/module.h> +#include <linux/types.h> +#include <linux/platform_device.h> +#include <linux/gpio.h> #include <asm/mach-au1x00/au1000.h> #include <asm/gpio.h> -#define gpio1 sys -#if !defined(CONFIG_SOC_AU1000) - -static struct au1x00_gpio2 *const gpio2 = (struct au1x00_gpio2 *) GPIO2_BASE; -#define GPIO2_OUTPUT_ENABLE_MASK 0x00010000 +struct au1000_gpio_chip { + struct gpio_chip chip; + void __iomem *regbase; +}; -static int au1xxx_gpio2_read(unsigned gpio) +#if !defined(CONFIG_SOC_AU1000) +int au1000_gpio2_get(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - return ((gpio2->pinstate >> gpio) & 0x01); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO2_ST) & mask; } +EXPORT_SYMBOL(au1000_gpio2_get); -static void au1xxx_gpio2_write(unsigned gpio, int value) +void au1000_gpio2_set(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; + u32 mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + struct au1000_gpio_chip *gpch; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO2_OUT); } +EXPORT_SYMBOL(au1000_gpio2_set); -static int au1xxx_gpio2_direction_input(unsigned gpio) +int au1000_gpio2_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir &= ~(0x01 << gpio); + u32 mask = 1 << offset; + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp &= ~mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + local_irq_restore(flags); + return 0; } +EXPORT_SYMBOL(au1000_gpio2_direction_input); -static int au1xxx_gpio2_direction_output(unsigned gpio, int value) +int au1000_gpio2_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir |= 0x01 << gpio; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + u32 mask = 1 << offset; + u32 out_mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp |= mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + writel(out_mask, gpch->regbase + AU1000_GPIO2_OUT); + local_irq_restore(flags); + return 0; } - +EXPORT_SYMBOL(au1000_gpio2_direction_output); #endif /* !defined(CONFIG_SOC_AU1000) */ -static int au1xxx_gpio1_read(unsigned gpio) +int au1000_gpio1_get(struct gpio_chip *chip, unsigned offset) { - return (gpio1->pinstaterd >> gpio) & 0x01; + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO1_ST) & mask; } +EXPORT_SYMBOL(au1000_gpio1_get); -static void au1xxx_gpio1_write(unsigned gpio, int value) +void au1000_gpio1_set(struct gpio_chip *chip, + unsigned offset, int value) { + u32 mask = 1 << offset; + u32 reg_offset; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + if (value) - gpio1->outputset = (0x01 << gpio); + reg_offset = AU1000_GPIO1_OUT; else - /* Output a zero */ - gpio1->outputclr = (0x01 << gpio); -} + reg_offset = AU1000_GPIO1_CLR; -static int au1xxx_gpio1_direction_input(unsigned gpio) -{ - gpio1->pininputen = (0x01 << gpio); - return 0; + local_irq_save(flags); + writel(mask, gpch->regbase + reg_offset); + local_irq_restore(flags); } +EXPORT_SYMBOL(au1000_gpio1_set); -static int au1xxx_gpio1_direction_output(unsigned gpio, int value) +int au1000_gpio1_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio1->trioutclr = (0x01 & gpio); - au1xxx_gpio1_write(gpio, value); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO1_ST); + return 0; } +EXPORT_SYMBOL(au1000_gpio1_direction_input); -int au1xxx_gpio_get_value(unsigned gpio) +int au1000_gpio1_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return 0; -#else - return au1xxx_gpio2_read(gpio); -#endif - else - return au1xxx_gpio1_read(gpio); -} -EXPORT_SYMBOL(au1xxx_gpio_get_value); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; -void au1xxx_gpio_set_value(unsigned gpio, int value) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - ; -#else - au1xxx_gpio2_write(gpio, value); -#endif - else - au1xxx_gpio1_write(gpio, value); -} -EXPORT_SYMBOL(au1xxx_gpio_set_value); + gpch = container_of(chip, struct au1000_gpio_chip, chip); -int au1xxx_gpio_direction_input(unsigned gpio) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_input(gpio); -#endif + writel(mask, gpch->regbase + AU1000_GPIO1_TRI_OUT); + au1000_gpio1_set(chip, offset, value); - return au1xxx_gpio1_direction_input(gpio); + return 0; } -EXPORT_SYMBOL(au1xxx_gpio_direction_input); +EXPORT_SYMBOL(au1000_gpio1_direction_output); + +struct au1000_gpio_chip au1000_gpio_chip[] = { + [0] = { + .regbase = (void __iomem *)SYS_BASE, + .chip = { + .label = "au1000-gpio1", + .direction_input = au1000_gpio1_direction_input, + .direction_output = au1000_gpio1_direction_output, + .get = au1000_gpio1_get, + .set = au1000_gpio1_set, + .base = 0, + .ngpio = 32, + }, + }, +#if !defined(CONFIG_SOC_AU1000) + [1] = { + .regbase = (void __iomem *)GPIO2_BASE, + .chip = { + .label = "au1000-gpio2", + .direction_input = au1000_gpio2_direction_input, + .direction_output = au1000_gpio2_direction_output, + .get = au1000_gpio2_get, + .set = au1000_gpio2_set, + .base = AU1XXX_GPIO2_BASE, + .ngpio = 32, + }, + }, +#endif +}; -int au1xxx_gpio_direction_output(unsigned gpio, int value) +static int __init au1000_gpio_init(void) { - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_output(gpio, value); -#endif + gpiochip_add(&au1000_gpio_chip[0].chip); +#if !defined(CONFIG_SOC_AU1000) + gpiochip_add(&au1000_gpio_chip[1].chip); - return au1xxx_gpio1_direction_output(gpio, value); + return 0; +#endif } -EXPORT_SYMBOL(au1xxx_gpio_direction_output); +#ifndef CONFIG_AU1X00_NON_STD_GPIOS +arch_initcall(au1000_gpio_init); +#endif diff --git a/arch/mips/include/asm/mach-au1x00/gpio.h b/arch/mips/include/asm/mach-au1x00/gpio.h index 2dc61e0..f1a5d48 100644 --- a/arch/mips/include/asm/mach-au1x00/gpio.h +++ b/arch/mips/include/asm/mach-au1x00/gpio.h @@ -3,67 +3,31 @@ #include <linux/types.h> -#define AU1XXX_GPIO_BASE 200 +#define AU1XXX_GPIO2_BASE 200 -struct au1x00_gpio2 { - u32 dir; - u32 reserved; - u32 output; - u32 pinstate; - u32 inten; - u32 enable; -}; +/* GPIO bank 1 offsets */ +#define AU1000_GPIO1_TRI_OUT 0x0100 +#define AU1000_GPIO1_OUT 0x0108 +#define AU1000_GPIO1_ST 0x0110 +#define AU1000_GPIO1_CLR 0x010C -extern int au1xxx_gpio_get_value(unsigned gpio); -extern void au1xxx_gpio_set_value(unsigned gpio, int value); -extern int au1xxx_gpio_direction_input(unsigned gpio); -extern int au1xxx_gpio_direction_output(unsigned gpio, int value); +/* GPIO bank 2 offsets */ +#define AU1000_GPIO2_DIR 0x00 +#define AU1000_GPIO2_RSVD 0x04 +#define AU1000_GPIO2_OUT 0x08 +#define AU1000_GPIO2_ST 0x0C +#define AU1000_GPIO2_INT 0x10 +#define AU1000_GPIO2_EN 0x14 +#define GPIO2_OUT_EN_MASK 0x00010000 -/* Wrappers for the arch-neutral GPIO API */ +#define gpio_to_irq(gpio) NULL -static inline int gpio_request(unsigned gpio, const char *label) -{ - /* Not yet implemented */ - return 0; -} +#define gpio_get_value __gpio_get_value +#define gpio_set_value __gpio_set_value -static inline void gpio_free(unsigned gpio) -{ - /* Not yet implemented */ -} +#define gpio_cansleep __gpio_cansleep -static inline int gpio_direction_input(unsigned gpio) -{ - return au1xxx_gpio_direction_input(gpio); -} - -static inline int gpio_direction_output(unsigned gpio, int value) -{ - return au1xxx_gpio_direction_output(gpio, value); -} - -static inline int gpio_get_value(unsigned gpio) -{ - return au1xxx_gpio_get_value(gpio); -} - -static inline void gpio_set_value(unsigned gpio, int value) -{ - au1xxx_gpio_set_value(gpio, value); -} - -static inline int gpio_to_irq(unsigned gpio) -{ - return gpio; -} - -static inline int irq_to_gpio(unsigned irq) -{ - return irq; -} - -/* For cansleep */ #include <asm-generic/gpio.h> #endif /* _AU1XXX_GPIO_H_ */ ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-27 9:39 ` Florian Fainelli @ 2009-01-27 9:47 ` Manuel Lauss 2009-01-27 10:02 ` Florian Fainelli 0 siblings, 1 reply; 12+ messages in thread From: Manuel Lauss @ 2009-01-27 9:47 UTC (permalink / raw) To: Florian Fainelli; +Cc: ralf, Linux-MIPS On Tue, Jan 27, 2009 at 10:39:30AM +0100, Florian Fainelli wrote: > Le Tuesday 27 January 2009 07:58:27 Manuel Lauss, vous avez ?crit?: > > Florian, > > > > > Le Monday 19 January 2009 18:12:23 Florian Fainelli, vous avez ?crit?: > > > > This patch converts the GPIO board code to use gpiolib. > > > > Changes from v1: > > > > - allow users not to use the default gpio accessors > > > > - do not lock au1000_gpio2_set > > > > > > I did not receive comments from you on this patch, can I consider it > > > being ok ? > > > > Oh sorry. Please export all the au1000_gpioX_* functions and I'm happy. > > > > Thanks! > > Manuel Lauss > > Here we go. Thanks ! One last nit: please also add function prototypes to the gpio header. -- ml. ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-27 9:47 ` Manuel Lauss @ 2009-01-27 10:02 ` Florian Fainelli 2009-01-27 10:11 ` Manuel Lauss 0 siblings, 1 reply; 12+ messages in thread From: Florian Fainelli @ 2009-01-27 10:02 UTC (permalink / raw) To: Manuel Lauss; +Cc: ralf, Linux-MIPS Le Tuesday 27 January 2009 10:47:16 Manuel Lauss, vous avez écrit : > On Tue, Jan 27, 2009 at 10:39:30AM +0100, Florian Fainelli wrote: > > Le Tuesday 27 January 2009 07:58:27 Manuel Lauss, vous avez ?crit?: > > > Florian, > > > > > > > Le Monday 19 January 2009 18:12:23 Florian Fainelli, vous avez ?crit?: > > > > > This patch converts the GPIO board code to use gpiolib. > > > > > Changes from v1: > > > > > - allow users not to use the default gpio accessors > > > > > - do not lock au1000_gpio2_set > > > > > > > > I did not receive comments from you on this patch, can I consider it > > > > being ok ? > > > > > > Oh sorry. Please export all the au1000_gpioX_* functions and I'm > > > happy. > > > > > > Thanks! > > > Manuel Lauss > > > > Here we go. Thanks ! > > One last nit: please also add function prototypes to the gpio header. Right, sorry about that. -- From: Florian Fainelli <florian@openwrt.org> Subject: [PATCH v4] Alchemy: convert to gpiolib This patch converts Alchemy SoC GPIO code to use gpiolib instead of the old GENERIC_GPIO implementation. Changes from v3: - add function prototypes in gpio header Changes from v2: - export all gpio accessors so that custom boards can use them Changes from v1: - allow users not to use the default gpio accessors - do not lock au1000_gpio2_set Signed-off-by: Florian Fainelli <florian@openwrt.org> --- diff --git a/arch/mips/alchemy/Kconfig b/arch/mips/alchemy/Kconfig index 7f8ef13..2fc5c13 100644 --- a/arch/mips/alchemy/Kconfig +++ b/arch/mips/alchemy/Kconfig @@ -135,3 +135,4 @@ config SOC_AU1X00 select SYS_SUPPORTS_32BIT_KERNEL select SYS_SUPPORTS_APM_EMULATION select GENERIC_HARDIRQS_NO__DO_IRQ + select ARCH_REQUIRE_GPIOLIB diff --git a/arch/mips/alchemy/common/gpio.c b/arch/mips/alchemy/common/gpio.c index e660ddd..535c565 100644 --- a/arch/mips/alchemy/common/gpio.c +++ b/arch/mips/alchemy/common/gpio.c @@ -1,5 +1,5 @@ /* - * Copyright (C) 2007, OpenWrt.org, Florian Fainelli <florian@openwrt.org> + * Copyright (C) 2007-2009, OpenWrt.org, Florian Fainelli <florian@openwrt.org> * Architecture specific GPIO support * * This program is free software; you can redistribute it and/or modify it @@ -27,122 +27,180 @@ * others have a second one : GPIO2 */ +#include <linux/kernel.h> #include <linux/module.h> +#include <linux/types.h> +#include <linux/platform_device.h> +#include <linux/gpio.h> #include <asm/mach-au1x00/au1000.h> #include <asm/gpio.h> -#define gpio1 sys -#if !defined(CONFIG_SOC_AU1000) - -static struct au1x00_gpio2 *const gpio2 = (struct au1x00_gpio2 *) GPIO2_BASE; -#define GPIO2_OUTPUT_ENABLE_MASK 0x00010000 +struct au1000_gpio_chip { + struct gpio_chip chip; + void __iomem *regbase; +}; -static int au1xxx_gpio2_read(unsigned gpio) +#if !defined(CONFIG_SOC_AU1000) +int au1000_gpio2_get(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - return ((gpio2->pinstate >> gpio) & 0x01); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO2_ST) & mask; } +EXPORT_SYMBOL(au1000_gpio2_get); -static void au1xxx_gpio2_write(unsigned gpio, int value) +void au1000_gpio2_set(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; + u32 mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + struct au1000_gpio_chip *gpch; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO2_OUT); } +EXPORT_SYMBOL(au1000_gpio2_set); -static int au1xxx_gpio2_direction_input(unsigned gpio) +int au1000_gpio2_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir &= ~(0x01 << gpio); + u32 mask = 1 << offset; + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp &= ~mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + local_irq_restore(flags); + return 0; } +EXPORT_SYMBOL(au1000_gpio2_direction_input); -static int au1xxx_gpio2_direction_output(unsigned gpio, int value) +int au1000_gpio2_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - gpio -= AU1XXX_GPIO_BASE; - gpio2->dir |= 0x01 << gpio; - gpio2->output = (GPIO2_OUTPUT_ENABLE_MASK << gpio) | ((!!value) << gpio); + u32 mask = 1 << offset; + u32 out_mask = ((GPIO2_OUT_EN_MASK << offset) | (!!value << offset)); + u32 tmp; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + + local_irq_save(flags); + tmp = readl(gpch->regbase + AU1000_GPIO2_DIR); + tmp |= mask; + writel(tmp, gpch->regbase + AU1000_GPIO2_DIR); + writel(out_mask, gpch->regbase + AU1000_GPIO2_OUT); + local_irq_restore(flags); + return 0; } - +EXPORT_SYMBOL(au1000_gpio2_direction_output); #endif /* !defined(CONFIG_SOC_AU1000) */ -static int au1xxx_gpio1_read(unsigned gpio) +int au1000_gpio1_get(struct gpio_chip *chip, unsigned offset) { - return (gpio1->pinstaterd >> gpio) & 0x01; + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + return readl(gpch->regbase + AU1000_GPIO1_ST) & mask; } +EXPORT_SYMBOL(au1000_gpio1_get); -static void au1xxx_gpio1_write(unsigned gpio, int value) +void au1000_gpio1_set(struct gpio_chip *chip, + unsigned offset, int value) { + u32 mask = 1 << offset; + u32 reg_offset; + struct au1000_gpio_chip *gpch; + unsigned long flags; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + if (value) - gpio1->outputset = (0x01 << gpio); + reg_offset = AU1000_GPIO1_OUT; else - /* Output a zero */ - gpio1->outputclr = (0x01 << gpio); -} + reg_offset = AU1000_GPIO1_CLR; -static int au1xxx_gpio1_direction_input(unsigned gpio) -{ - gpio1->pininputen = (0x01 << gpio); - return 0; + local_irq_save(flags); + writel(mask, gpch->regbase + reg_offset); + local_irq_restore(flags); } +EXPORT_SYMBOL(au1000_gpio1_set); -static int au1xxx_gpio1_direction_output(unsigned gpio, int value) +int au1000_gpio1_direction_input(struct gpio_chip *chip, unsigned offset) { - gpio1->trioutclr = (0x01 & gpio); - au1xxx_gpio1_write(gpio, value); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; + + gpch = container_of(chip, struct au1000_gpio_chip, chip); + writel(mask, gpch->regbase + AU1000_GPIO1_ST); + return 0; } +EXPORT_SYMBOL(au1000_gpio1_direction_input); -int au1xxx_gpio_get_value(unsigned gpio) +int au1000_gpio1_direction_output(struct gpio_chip *chip, + unsigned offset, int value) { - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return 0; -#else - return au1xxx_gpio2_read(gpio); -#endif - else - return au1xxx_gpio1_read(gpio); -} -EXPORT_SYMBOL(au1xxx_gpio_get_value); + u32 mask = 1 << offset; + struct au1000_gpio_chip *gpch; -void au1xxx_gpio_set_value(unsigned gpio, int value) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - ; -#else - au1xxx_gpio2_write(gpio, value); -#endif - else - au1xxx_gpio1_write(gpio, value); -} -EXPORT_SYMBOL(au1xxx_gpio_set_value); + gpch = container_of(chip, struct au1000_gpio_chip, chip); -int au1xxx_gpio_direction_input(unsigned gpio) -{ - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_input(gpio); -#endif + writel(mask, gpch->regbase + AU1000_GPIO1_TRI_OUT); + au1000_gpio1_set(chip, offset, value); - return au1xxx_gpio1_direction_input(gpio); + return 0; } -EXPORT_SYMBOL(au1xxx_gpio_direction_input); +EXPORT_SYMBOL(au1000_gpio1_direction_output); + +struct au1000_gpio_chip au1000_gpio_chip[] = { + [0] = { + .regbase = (void __iomem *)SYS_BASE, + .chip = { + .label = "au1000-gpio1", + .direction_input = au1000_gpio1_direction_input, + .direction_output = au1000_gpio1_direction_output, + .get = au1000_gpio1_get, + .set = au1000_gpio1_set, + .base = 0, + .ngpio = 32, + }, + }, +#if !defined(CONFIG_SOC_AU1000) + [1] = { + .regbase = (void __iomem *)GPIO2_BASE, + .chip = { + .label = "au1000-gpio2", + .direction_input = au1000_gpio2_direction_input, + .direction_output = au1000_gpio2_direction_output, + .get = au1000_gpio2_get, + .set = au1000_gpio2_set, + .base = AU1XXX_GPIO2_BASE, + .ngpio = 32, + }, + }, +#endif +}; -int au1xxx_gpio_direction_output(unsigned gpio, int value) +static int __init au1000_gpio_init(void) { - if (gpio >= AU1XXX_GPIO_BASE) -#if defined(CONFIG_SOC_AU1000) - return -ENODEV; -#else - return au1xxx_gpio2_direction_output(gpio, value); -#endif + gpiochip_add(&au1000_gpio_chip[0].chip); +#if !defined(CONFIG_SOC_AU1000) + gpiochip_add(&au1000_gpio_chip[1].chip); - return au1xxx_gpio1_direction_output(gpio, value); + return 0; +#endif } -EXPORT_SYMBOL(au1xxx_gpio_direction_output); +#ifndef CONFIG_AU1X00_NON_STD_GPIOS +arch_initcall(au1000_gpio_init); +#endif diff --git a/arch/mips/include/asm/mach-au1x00/gpio.h b/arch/mips/include/asm/mach-au1x00/gpio.h index 2dc61e0..099f70f 100644 --- a/arch/mips/include/asm/mach-au1x00/gpio.h +++ b/arch/mips/include/asm/mach-au1x00/gpio.h @@ -2,68 +2,42 @@ #define _AU1XXX_GPIO_H_ #include <linux/types.h> +#include <linux/gpio.h> -#define AU1XXX_GPIO_BASE 200 +#define AU1XXX_GPIO2_BASE 200 -struct au1x00_gpio2 { - u32 dir; - u32 reserved; - u32 output; - u32 pinstate; - u32 inten; - u32 enable; -}; +/* GPIO bank 1 offsets */ +#define AU1000_GPIO1_TRI_OUT 0x0100 +#define AU1000_GPIO1_OUT 0x0108 +#define AU1000_GPIO1_ST 0x0110 +#define AU1000_GPIO1_CLR 0x010C -extern int au1xxx_gpio_get_value(unsigned gpio); -extern void au1xxx_gpio_set_value(unsigned gpio, int value); -extern int au1xxx_gpio_direction_input(unsigned gpio); -extern int au1xxx_gpio_direction_output(unsigned gpio, int value); +/* GPIO bank 2 offsets */ +#define AU1000_GPIO2_DIR 0x00 +#define AU1000_GPIO2_RSVD 0x04 +#define AU1000_GPIO2_OUT 0x08 +#define AU1000_GPIO2_ST 0x0C +#define AU1000_GPIO2_INT 0x10 +#define AU1000_GPIO2_EN 0x14 +#define GPIO2_OUT_EN_MASK 0x00010000 -/* Wrappers for the arch-neutral GPIO API */ +#define gpio_to_irq(gpio) NULL -static inline int gpio_request(unsigned gpio, const char *label) -{ - /* Not yet implemented */ - return 0; -} +#define gpio_get_value __gpio_get_value +#define gpio_set_value __gpio_set_value -static inline void gpio_free(unsigned gpio) -{ - /* Not yet implemented */ -} +#define gpio_cansleep __gpio_cansleep -static inline int gpio_direction_input(unsigned gpio) -{ - return au1xxx_gpio_direction_input(gpio); -} - -static inline int gpio_direction_output(unsigned gpio, int value) -{ - return au1xxx_gpio_direction_output(gpio, value); -} - -static inline int gpio_get_value(unsigned gpio) -{ - return au1xxx_gpio_get_value(gpio); -} - -static inline void gpio_set_value(unsigned gpio, int value) -{ - au1xxx_gpio_set_value(gpio, value); -} - -static inline int gpio_to_irq(unsigned gpio) -{ - return gpio; -} - -static inline int irq_to_gpio(unsigned irq) -{ - return irq; -} - -/* For cansleep */ #include <asm-generic/gpio.h> +int au1000_gpio2_get(struct gpio_chip *chip, unsigned offset); +void au1000_gpio2_set(struct gpio_chip *chip, unsigned offset, int value); +int au1000_gpio2_direction_input(struct gpio_chip *chip, unsigned offset); +int au1000_gpio2_direction_output(struct gpio_chip *chip, unsigned offset, int value); +int au1000_gpio1_get(struct gpio_chip *chip, unsigned offset); +void au1000_gpio1_set(struct gpio_chip *chip, unsigned offset, int value); +int au1000_gpio1_direction_input(struct gpio_chip *chip, unsigned offset); +int au1000_gpio1_direction_output(struct gpio_chip *chip, unsigned offset, int value); + #endif /* _AU1XXX_GPIO_H_ */ ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-27 10:02 ` Florian Fainelli @ 2009-01-27 10:11 ` Manuel Lauss 0 siblings, 0 replies; 12+ messages in thread From: Manuel Lauss @ 2009-01-27 10:11 UTC (permalink / raw) To: Florian Fainelli; +Cc: ralf, Linux-MIPS On Tue, Jan 27, 2009 at 11:02:36AM +0100, Florian Fainelli wrote: > Le Tuesday 27 January 2009 10:47:16 Manuel Lauss, vous avez ?crit?: > > On Tue, Jan 27, 2009 at 10:39:30AM +0100, Florian Fainelli wrote: > > > Le Tuesday 27 January 2009 07:58:27 Manuel Lauss, vous avez ?crit?: > > > > Florian, > > > > > > > > > Le Monday 19 January 2009 18:12:23 Florian Fainelli, vous avez ?crit?: > > > > > > This patch converts the GPIO board code to use gpiolib. > > > > > > Changes from v1: > > > > > > - allow users not to use the default gpio accessors > > > > > > - do not lock au1000_gpio2_set > > > > > > > > > > I did not receive comments from you on this patch, can I consider it > > > > > being ok ? > > > > > > > > Oh sorry. Please export all the au1000_gpioX_* functions and I'm > > > > happy. > > > > > > > > Thanks! > > > > Manuel Lauss > > > > > > Here we go. Thanks ! > > > > One last nit: please also add function prototypes to the gpio header. > > Right, sorry about that. > -- > From: Florian Fainelli <florian@openwrt.org> > Subject: [PATCH v4] Alchemy: convert to gpiolib > > This patch converts Alchemy SoC GPIO code to use > gpiolib instead of the old GENERIC_GPIO implementation. > > Changes from v3: > - add function prototypes in gpio header > > Changes from v2: > - export all gpio accessors so that custom boards can use them > > Changes from v1: > - allow users not to use the default gpio accessors > - do not lock au1000_gpio2_set > > Signed-off-by: Florian Fainelli <florian@openwrt.org> Acked-by: Manuel Lauss <mano@roarinelk.homelinux.net> Thank you! Manuel Lauss ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH] au1000: convert to using gpiolib 2009-01-15 15:46 [PATCH] au1000: convert to using gpiolib Florian Fainelli 2009-01-15 20:58 ` Manuel Lauss @ 2009-02-26 14:35 ` Ralf Baechle 1 sibling, 0 replies; 12+ messages in thread From: Ralf Baechle @ 2009-02-26 14:35 UTC (permalink / raw) To: Florian Fainelli; +Cc: Manuel Lauss, Linux-MIPS On Thu, Jan 15, 2009 at 04:46:48PM +0100, Florian Fainelli wrote: > This patch converts the GPIO board code to use gpiolib. Queued for 2.6.30. Thanks! Ralf ^ permalink raw reply [flat|nested] 12+ messages in thread
end of thread, other threads:[~2009-02-26 14:35 UTC | newest] Thread overview: 12+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2009-01-15 15:46 [PATCH] au1000: convert to using gpiolib Florian Fainelli 2009-01-15 20:58 ` Manuel Lauss 2009-01-16 10:10 ` Florian Fainelli 2009-01-16 17:47 ` Manuel Lauss 2009-01-19 17:12 ` Florian Fainelli 2009-01-26 22:40 ` Florian Fainelli 2009-01-27 6:58 ` Manuel Lauss 2009-01-27 9:39 ` Florian Fainelli 2009-01-27 9:47 ` Manuel Lauss 2009-01-27 10:02 ` Florian Fainelli 2009-01-27 10:11 ` Manuel Lauss 2009-02-26 14:35 ` Ralf Baechle
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox