On Tue, Jul 23, 2013 at 3:07 AM, Linus Walleij <linus.walleij@xxxxxxxxxx> wrote: > On Mon, Jul 8, 2013 at 9:04 AM, <thloh.linux@xxxxxxxxx> wrote: > >> From: Tien Hock Loh <thloh@xxxxxxxxxx> >> >> Add driver support for Altera GPIO soft IP, including interrupts and I/O. >> Tested on Altera CV SoC board using dipsw and LED using LED framework. >> >> Signed-off-by: Tien Hock Loh <thloh@xxxxxxxxxx> > > Note that the new mailing list for devicetree patches is > devicetree@xxxxxxxxxxxxxxx nowadays. > >> +Altera GPIO specific properties: >> +- width: Width of the GPIO bank, range from 1-32 > > Can it really vary like that? 1, 2, 3, ... Yes, any number within the range of 32. > > Can there be "holes" in this range then, so you might prefer > a bitmask over a numeral? No, there is no "holes" in the Altera GPIO soft IP. > >> +- level_trigger: Specifies whether the GPIO interrupt is level trigger (high). >> + This field is required if the Altera GPIO controller used has IRQ enabled. >> +- edge_type: Specifies edge type if the GPIO interrupt is edge trigger: >> + 0 = Rising edge >> + 1 = Falling edge >> + 2 = Both edge > > No thanks. I don't understand the comment here. Could you elaborate further? The edge_type is required as the soft IP's interrupt type isn't software configurable, and thus is a hardware parameter passed in from device tree blob. > > #include <dt-bindings/interrupt-controller/irq.h> > and define these things per-irq by using more interrupt-cells. > >> +Example: >> + >> +gpio_altr: gpio_altr { >> + compatible = "altr,pio-1.0"; >> + reg = <0xff200000 0x10>; >> + interrupts = <0 45 4>; >> + width = <32>; >> + level_trigger = <0>; >> + #gpio-cells = <1>; >> + gpio-controller; >> + #interrupt-cells = <1>; > > This needs to be 2 or something so you can pass the edge specifics > for each line. OK noted. > >> +config GPIO_ALTERA >> + tristate "Altera GPIO" > > Do you *really* load this as a module sometimes? Yes, as this peripheral is a soft IP resides in FPGA, and the FPGA may not be up and running by the time the system boots, and thus in most cases they should be loaded as module instead of a built in driver. > >> +++ b/drivers/gpio/gpio-altera.c > >> +#include <linux/gpio.h> >> +#include <linux/kernel.h> >> +#include <linux/slab.h> >> +#include <linux/module.h> >> +#include <linux/io.h> >> +#include <linux/irq.h> >> +#include <linux/of_irq.h> >> +#include <linux/irqdomain.h> >> +#include <linux/interrupt.h> >> +#include <linux/of_gpio.h> >> +#include <linux/platform_device.h> > > #include <linux/bitops.h> > > BIT() comes from this file. Noted. > >> + >> +#define ALTERA_GPIO_DATA 0x0 >> +#define ALTERA_GPIO_DIR 0x4 >> +#define ALTERA_GPIO_IRQ_MASK 0x8 >> +#define ALTERA_GPIO_EDGE_CAP 0xc >> +#define ALTERA_GPIO_OUTSET 0x10 >> +#define ALTERA_GPIO_OUTCLEAR 0x14 >> +#define ALTERA_IRQ_RISING 0 >> +#define ALTERA_IRQ_FALLING 1 >> +#define ALTERA_IRQ_BOTH 2 > > Why do you need these local redefinitions of rising/falling/both? > Use the defines from the subsystem. The Altera tool that generates the device tree uses the number that is not the same as the subsystem ones (notice ALTERA_IRQ_RISING is 0, but IRQ_TYPE_EDGE_RISING is 1), thus these defines are required. Changes to the tool will break existing compatibility. > >> + >> +/** >> +* struct altera_gpio_chip >> +* @mmchip : memory mapped chip structure. >> +* @irq : irq domain that this driver is registered to. > > This is confusing. Rename this to "irqdomain" instead. Noted. > >> +* @gpio_lock : synchronization lock so that new irq/set/get requests >> + will be blocked until the current one completes. >> +* @level_trigger : specifies if the controller is level triggered >> + (high). >> +* @edge_type : specify the edge type of the controller (rising, >> + falling, both). Only used for !level_trigger. >> +* @mapped_irq : kernel mapped irq number. >> +*/ >> +struct altera_gpio_chip { >> + struct of_mm_gpio_chip mmchip; >> + struct irq_domain *irq; >> + spinlock_t gpio_lock; >> + int level_trigger; >> + int edge_type; >> + int mapped_irq; > > And name this "irq". OK noted. > >> +}; >> + >> +static void altera_gpio_irq_unmask(struct irq_data *d) >> +{ >> + struct altera_gpio_chip *altera_gc = irq_data_get_irq_chip_data(d); >> + struct of_mm_gpio_chip *mm_gc = &altera_gc->mmchip; >> + unsigned long flags; >> + unsigned int intmask; > > Use u32 for this variable. Noted. > >> + >> + spin_lock_irqsave(&altera_gc->gpio_lock, flags); >> + intmask = readl(mm_gc->regs + ALTERA_GPIO_IRQ_MASK); >> + /* Set ALTERA_GPIO_IRQ_MASK bit to unmask */ >> + intmask |= BIT(irqd_to_hwirq(d)); >> + writel(intmask, mm_gc->regs + ALTERA_GPIO_IRQ_MASK); >> + spin_unlock_irqrestore(&altera_gc->gpio_lock, flags); >> +} >> + >> +static void altera_gpio_irq_mask(struct irq_data *d) >> +{ >> + struct altera_gpio_chip *altera_gc = irq_data_get_irq_chip_data(d); >> + struct of_mm_gpio_chip *mm_gc = &altera_gc->mmchip; >> + unsigned long flags; >> + unsigned int intmask; > > u32 Noted. > >> + >> + spin_lock_irqsave(&altera_gc->gpio_lock, flags); >> + intmask = readl(mm_gc->regs + ALTERA_GPIO_IRQ_MASK); >> + /* Clear ALTERA_GPIO_IRQ_MASK bit to mask */ >> + intmask &= BIT(irqd_to_hwirq(d)); >> + writel(intmask, mm_gc->regs + ALTERA_GPIO_IRQ_MASK); >> + spin_unlock_irqrestore(&altera_gc->gpio_lock, flags); >> +} >> + >> +static int altera_gpio_irq_set_type(struct irq_data *d, >> + unsigned int type) >> +{ >> + struct altera_gpio_chip *altera_gc = irq_data_get_irq_chip_data(d); >> + >> + if (type == IRQ_TYPE_NONE) >> + return 0; >> + >> + if (altera_gc->level_trigger) { >> + if (type == IRQ_TYPE_LEVEL_HIGH) >> + return 0; >> + } else { >> + if (type == IRQ_TYPE_EDGE_RISING && >> + altera_gc->edge_type == ALTERA_IRQ_RISING) >> + return 0; >> + else if (type == IRQ_TYPE_EDGE_FALLING && >> + altera_gc->edge_type == ALTERA_IRQ_FALLING) >> + return 0; >> + else if (type == IRQ_TYPE_EDGE_BOTH && >> + altera_gc->edge_type == ALTERA_IRQ_BOTH) >> + return 0; > > I don't quite realize why you need the local version of > the IRQflag at all. Just store the standard edge indicator > in an unsigned long? I don't understand the comment. Can you elaborate further? > > Also: if you have a property in the device tree setting all IRQs > to one or the other type (as the weird binding is now) why > are you still supporting all these variants on each and every > IRQ? > This is actually to handle scenarios like when drivers (like gpio-led) wishes to "set" the software configurable parameters (usually exist in general GPIO), it needs to check if the IP is configured the exact same configuration requested and return the status correctly (ie. if the soft IP is set as edge falling interrupt, and the gpio-led requests edge rising, this will return non zero and thus the gpio-led will know this IP failed to be configured with the requested interrupt type). >> + } >> + >> + return -EINVAL; >> +} >> + >> +static struct irq_chip altera_irq_chip = { >> + .name = "altera-gpio", >> + .irq_mask = altera_gpio_irq_mask, >> + .irq_unmask = altera_gpio_irq_unmask, >> + .irq_set_type = altera_gpio_irq_set_type, >> +}; >> + >> +static int altera_gpio_get(struct gpio_chip *gc, unsigned offset) >> +{ >> + struct of_mm_gpio_chip *mm_gc = to_of_mm_gpio_chip(gc); >> + >> + return (readl(mm_gc->regs + ALTERA_GPIO_DATA) >> offset) & 1; > > Pls write it like this: > return !!(readl(mm_gc->regs + ALTERA_GPIO_DATA) & BIT(offset))); OK noted. > >> +} >> + >> +static void altera_gpio_set(struct gpio_chip *gc, unsigned offset, int value) >> +{ >> + struct of_mm_gpio_chip *mm_gc = to_of_mm_gpio_chip(gc); >> + struct altera_gpio_chip *chip = container_of(mm_gc, >> + struct altera_gpio_chip, mmchip); >> + unsigned long flags; >> + unsigned int data_reg; > > u32 Noted. > >> + >> + spin_lock_irqsave(&chip->gpio_lock, flags); >> + data_reg = readl(mm_gc->regs + ALTERA_GPIO_DATA); >> + data_reg = (data_reg & ~BIT(offset)) | (value << offset); > > Value is always 0 or 1. > > I would prefer this idom even though its more lines: > > if (val) > data_reg |= BIT(offset); > else > data_reg &= ~BIT(offset); > Noted. >> + writel(data_reg, mm_gc->regs + ALTERA_GPIO_DATA); >> + spin_unlock_irqrestore(&chip->gpio_lock, flags); >> +} >> + >> +static int altera_gpio_direction_input(struct gpio_chip *gc, unsigned offset) >> +{ >> + struct of_mm_gpio_chip *mm_gc = to_of_mm_gpio_chip(gc); >> + struct altera_gpio_chip *chip = container_of(mm_gc, >> + struct altera_gpio_chip, mmchip); >> + unsigned long flags; >> + unsigned int gpio_ddr; > > u32 Noted. > >> + >> + spin_lock_irqsave(&chip->gpio_lock, flags); >> + /* Set pin as input, assumes software controlled IP */ >> + gpio_ddr = readl(mm_gc->regs + ALTERA_GPIO_DIR); >> + gpio_ddr &= ~BIT(offset); >> + writel(gpio_ddr, mm_gc->regs + ALTERA_GPIO_DIR); >> + spin_unlock_irqrestore(&chip->gpio_lock, flags); >> + >> + return 0; >> +} >> + >> +static int altera_gpio_direction_output(struct gpio_chip *gc, >> + unsigned offset, int value) >> +{ >> + struct of_mm_gpio_chip *mm_gc = to_of_mm_gpio_chip(gc); >> + struct altera_gpio_chip *chip = container_of(mm_gc, >> + struct altera_gpio_chip, mmchip); >> + unsigned long flags; >> + unsigned int data_reg, gpio_ddr; > > u32 Noted > >> + >> + spin_lock_irqsave(&chip->gpio_lock, flags); >> + /* Sets the GPIO value */ >> + data_reg = readl(mm_gc->regs + ALTERA_GPIO_DATA); >> + data_reg = (data_reg & ~BIT(offset)) | (value << offset); > > Use same construct as indicated above. Noted. > >> + writel(data_reg, mm_gc->regs + ALTERA_GPIO_DATA); >> + >> + /* Set pin as output, assumes software controlled IP */ >> + gpio_ddr = readl(mm_gc->regs + ALTERA_GPIO_DIR); >> + gpio_ddr |= BIT(offset); >> + writel(gpio_ddr, mm_gc->regs + ALTERA_GPIO_DIR); >> + spin_unlock_irqrestore(&chip->gpio_lock, flags); >> + >> + return 0; >> +} >> + >> +static int altera_gpio_to_irq(struct gpio_chip *gc, unsigned offset) >> +{ >> + struct of_mm_gpio_chip *mm_gc = to_of_mm_gpio_chip(gc); >> + struct altera_gpio_chip *altera_gc = container_of(mm_gc, >> + struct altera_gpio_chip, mmchip); >> + >> + if (altera_gc->irq == 0) >> + return -ENXIO; >> + if ((altera_gc->irq && offset) < altera_gc->mmchip.gc.ngpio) > > I don't understand this. Why are you making a binary and && > with the offset and then checking if that is less that the number > of GPIO? I'll look into this. Thanks for catching this. > >> + return irq_create_mapping(altera_gc->irq, offset); >> + else >> + return -ENXIO; >> +} >> + >> +static void altera_gpio_irq_handler(unsigned int irq, struct irq_desc *desc) >> +{ >> + struct altera_gpio_chip *altera_gc = irq_desc_get_handler_data(desc); >> + struct irq_chip *chip = irq_desc_get_chip(desc); >> + struct of_mm_gpio_chip *mm_gc = &altera_gc->mmchip; >> + unsigned long status; > > u32 Noted. > >> + >> + int base; > > Is this an int really? It looks like u8 would suffice. > Why is this named "base"? Can you give it a more self-explanatory > name? Else just use "i". OK noted. > >> + chip->irq_mask(&desc->irq_data); >> + >> + if (altera_gc->level_trigger) { >> + status = readl_relaxed(mm_gc->regs + ALTERA_GPIO_DATA); >> + status &= readl_relaxed(mm_gc->regs + ALTERA_GPIO_IRQ_MASK); >> + >> + for (base = 0; base < mm_gc->gc.ngpio; base++) { >> + if (BIT(base) & status) { >> + generic_handle_irq(irq_linear_revmap( >> + altera_gc->irq, base)); >> + } >> + } >> + } else { >> + while ((status = >> + (readl_relaxed(mm_gc->regs + ALTERA_GPIO_EDGE_CAP) & >> + readl_relaxed(mm_gc->regs + ALTERA_GPIO_IRQ_MASK)))) { >> + writel_relaxed(status, >> + mm_gc->regs + ALTERA_GPIO_EDGE_CAP); > > Nice use if relaxed accessors in the irq handler! I don't understand the comment. Could you elaborate further? > >> + for (base = 0; base < mm_gc->gc.ngpio; base++) { >> + if (BIT(base) & status) { >> + generic_handle_irq(irq_linear_revmap( >> + altera_gc->irq, base)); >> + } >> + } >> + } >> + } >> + >> + chip->irq_eoi(irq_desc_get_irq_data(desc)); >> + chip->irq_unmask(&desc->irq_data); >> +} >> + >> +static int altera_gpio_irq_map(struct irq_domain *h, unsigned int virq, >> + irq_hw_number_t hw_irq_num) >> +{ >> + irq_set_chip_data(virq, h->host_data); >> + irq_set_chip_and_handler(virq, &altera_irq_chip, handle_level_irq); >> + irq_set_irq_type(virq, IRQ_TYPE_NONE); >> + >> + return 0; >> +} >> + >> +static struct irq_domain_ops altera_gpio_irq_ops = { >> + .map = altera_gpio_irq_map, >> + .xlate = irq_domain_xlate_onecell, >> +}; >> + >> +int altera_gpio_probe(struct platform_device *pdev) >> +{ >> + struct device_node *node = pdev->dev.of_node; >> + int id, reg, ret; >> + struct altera_gpio_chip *altera_gc = devm_kzalloc(&pdev->dev, >> + sizeof(*altera_gc), GFP_KERNEL); >> + if (altera_gc == NULL) { >> + pr_err("%s: out of memory\n", node->full_name); >> + return -ENOMEM; >> + } >> + altera_gc->irq = 0; >> + >> + spin_lock_init(&altera_gc->gpio_lock); >> + >> + id = pdev->id; >> + >> + if (of_property_read_u32(node, "width", ®)) >> + /*By default assume full GPIO controller*/ >> + altera_gc->mmchip.gc.ngpio = 32; >> + else >> + altera_gc->mmchip.gc.ngpio = reg; > > I question this property above... > >> + >> + if (altera_gc->mmchip.gc.ngpio > 32) { >> + pr_warn("%s: ngpio is greater than 32, defaulting to 32\n", >> + node->full_name); >> + altera_gc->mmchip.gc.ngpio = 32; >> + } >> + >> + altera_gc->mmchip.gc.direction_input = altera_gpio_direction_input; >> + altera_gc->mmchip.gc.direction_output = altera_gpio_direction_output; >> + altera_gc->mmchip.gc.get = altera_gpio_get; >> + altera_gc->mmchip.gc.set = altera_gpio_set; >> + altera_gc->mmchip.gc.to_irq = altera_gpio_to_irq; >> + altera_gc->mmchip.gc.owner = THIS_MODULE; >> + >> + ret = of_mm_gpiochip_add(node, &altera_gc->mmchip); >> + if (ret) { >> + pr_err("%s: Failed adding memory mapped gpiochip\n", >> + node->full_name); >> + return ret; >> + } >> + >> + platform_set_drvdata(pdev, altera_gc); >> + >> + if (of_get_property(node, "interrupts", ®) == NULL) >> + goto skip_irq; >> + altera_gc->mapped_irq = irq_of_parse_and_map(node, 0); >> + >> + if (altera_gc->mapped_irq == NO_IRQ) >> + goto skip_irq; > > Just return 0; here. > >> + >> + altera_gc->irq = irq_domain_add_linear(node, altera_gc->mmchip.gc.ngpio, >> + &altera_gpio_irq_ops, altera_gc); >> + >> + if (!altera_gc->irq) { >> + ret = -ENODEV; >> + goto dispose_irq; >> + } >> + >> + if (of_property_read_u32(node, "level_trigger", ®)) { >> + ret = -EINVAL; >> + pr_err("%s: level_trigger value not set in device tree\n", >> + node->full_name); >> + goto teardown; >> + } >> + altera_gc->level_trigger = reg; >> + >> + /* >> + * If it is not level triggered PIO >> + * Check what edge type it is >> + */ >> + if (!altera_gc->level_trigger) { >> + if (of_property_read_u32(node, "edge_type", ®)) { >> + ret = -EINVAL; >> + pr_err("%s: edge_type value not set in device tree\n" >> + , node->full_name); >> + goto teardown; >> + } >> + } >> + altera_gc->edge_type = reg; >> + >> + irq_set_handler_data(altera_gc->mapped_irq, altera_gc); >> + irq_set_chained_handler(altera_gc->mapped_irq, altera_gpio_irq_handler); >> + >> + return 0; >> + >> +teardown: >> + irq_domain_remove(altera_gc->irq); >> +dispose_irq: >> + irq_dispose_mapping(altera_gc->mapped_irq); >> + WARN_ON(gpiochip_remove(&altera_gc->mmchip.gc) < 0); >> + >> + pr_err("%s: registration failed with status %d\n", >> + node->full_name, ret); >> + >> + return ret; >> +skip_irq: >> + return 0; >> +} >> + >> +static int altera_gpio_remove(struct platform_device *pdev) >> +{ >> + unsigned int irq, i; >> + int status; >> + struct altera_gpio_chip *altera_gc = platform_get_drvdata(pdev); >> + >> + status = gpiochip_remove(&altera_gc->mmchip.gc); >> + >> + if (status < 0) >> + return status; >> + >> + if (altera_gc->irq) { >> + irq_dispose_mapping(altera_gc->mapped_irq); >> + >> + for (i = 0; i < altera_gc->mmchip.gc.ngpio; i++) { >> + irq = irq_find_mapping(altera_gc->irq, i); >> + if (irq > 0) >> + irq_dispose_mapping(irq); >> + } > > This loop, isn't thad done inside irq_domain_remove() already? Noted. I'll remove this. > > That's all review comments so far. As you see it needs some > rough edges fixed up... > > Yours, > Linus Walleij -- To unsubscribe from this list: send the line "unsubscribe linux-doc" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html