diff options
Diffstat (limited to 'drivers/gpio/gpio-sch.c')
| -rw-r--r-- | drivers/gpio/gpio-sch.c | 198 | 
1 files changed, 190 insertions, 8 deletions
| diff --git a/drivers/gpio/gpio-sch.c b/drivers/gpio/gpio-sch.c index 3a1b1adb08c6..a6f0421d6e50 100644 --- a/drivers/gpio/gpio-sch.c +++ b/drivers/gpio/gpio-sch.c @@ -7,33 +7,55 @@   */  #include <linux/acpi.h> +#include <linux/bitops.h>  #include <linux/errno.h>  #include <linux/gpio/driver.h>  #include <linux/io.h> +#include <linux/irq.h>  #include <linux/kernel.h>  #include <linux/module.h>  #include <linux/pci_ids.h>  #include <linux/platform_device.h> +#include <linux/types.h>  #define GEN	0x00  #define GIO	0x04  #define GLV	0x08 +#define GTPE	0x0c +#define GTNE	0x10 +#define GGPE	0x14 +#define GSMI	0x18 +#define GTS	0x1c + +#define CORE_BANK_OFFSET	0x00 +#define RESUME_BANK_OFFSET	0x20 + +/* + * iLB datasheet describes GPE0BLK registers, in particular GPE0E.GPIO bit. + * Document Number: 328195-001 + */ +#define GPE0E_GPIO	14  struct sch_gpio {  	struct gpio_chip chip; +	struct irq_chip irqchip;  	spinlock_t lock;  	unsigned short iobase;  	unsigned short resume_base; + +	/* GPE handling */ +	u32 gpe; +	acpi_gpe_handler gpe_handler;  };  static unsigned int sch_gpio_offset(struct sch_gpio *sch, unsigned int gpio,  				unsigned int reg)  { -	unsigned int base = 0; +	unsigned int base = CORE_BANK_OFFSET;  	if (gpio >= sch->resume_base) {  		gpio -= sch->resume_base; -		base += 0x20; +		base = RESUME_BANK_OFFSET;  	}  	return base + reg + gpio / 8; @@ -79,10 +101,11 @@ static void sch_gpio_reg_set(struct sch_gpio *sch, unsigned int gpio, unsigned i  static int sch_gpio_direction_in(struct gpio_chip *gc, unsigned int gpio_num)  {  	struct sch_gpio *sch = gpiochip_get_data(gc); +	unsigned long flags; -	spin_lock(&sch->lock); +	spin_lock_irqsave(&sch->lock, flags);  	sch_gpio_reg_set(sch, gpio_num, GIO, 1); -	spin_unlock(&sch->lock); +	spin_unlock_irqrestore(&sch->lock, flags);  	return 0;  } @@ -96,20 +119,22 @@ static int sch_gpio_get(struct gpio_chip *gc, unsigned int gpio_num)  static void sch_gpio_set(struct gpio_chip *gc, unsigned int gpio_num, int val)  {  	struct sch_gpio *sch = gpiochip_get_data(gc); +	unsigned long flags; -	spin_lock(&sch->lock); +	spin_lock_irqsave(&sch->lock, flags);  	sch_gpio_reg_set(sch, gpio_num, GLV, val); -	spin_unlock(&sch->lock); +	spin_unlock_irqrestore(&sch->lock, flags);  }  static int sch_gpio_direction_out(struct gpio_chip *gc, unsigned int gpio_num,  				  int val)  {  	struct sch_gpio *sch = gpiochip_get_data(gc); +	unsigned long flags; -	spin_lock(&sch->lock); +	spin_lock_irqsave(&sch->lock, flags);  	sch_gpio_reg_set(sch, gpio_num, GIO, 0); -	spin_unlock(&sch->lock); +	spin_unlock_irqrestore(&sch->lock, flags);  	/*  	 * according to the datasheet, writing to the level register has no @@ -144,10 +169,145 @@ static const struct gpio_chip sch_gpio_chip = {  	.get_direction		= sch_gpio_get_direction,  }; +static int sch_irq_type(struct irq_data *d, unsigned int type) +{ +	struct gpio_chip *gc = irq_data_get_irq_chip_data(d); +	struct sch_gpio *sch = gpiochip_get_data(gc); +	irq_hw_number_t gpio_num = irqd_to_hwirq(d); +	unsigned long flags; +	int rising, falling; + +	switch (type & IRQ_TYPE_SENSE_MASK) { +	case IRQ_TYPE_EDGE_RISING: +		rising = 1; +		falling = 0; +		break; +	case IRQ_TYPE_EDGE_FALLING: +		rising = 0; +		falling = 1; +		break; +	case IRQ_TYPE_EDGE_BOTH: +		rising = 1; +		falling = 1; +		break; +	default: +		return -EINVAL; +	} + +	spin_lock_irqsave(&sch->lock, flags); + +	sch_gpio_reg_set(sch, gpio_num, GTPE, rising); +	sch_gpio_reg_set(sch, gpio_num, GTNE, falling); + +	irq_set_handler_locked(d, handle_edge_irq); + +	spin_unlock_irqrestore(&sch->lock, flags); + +	return 0; +} + +static void sch_irq_ack(struct irq_data *d) +{ +	struct gpio_chip *gc = irq_data_get_irq_chip_data(d); +	struct sch_gpio *sch = gpiochip_get_data(gc); +	irq_hw_number_t gpio_num = irqd_to_hwirq(d); +	unsigned long flags; + +	spin_lock_irqsave(&sch->lock, flags); +	sch_gpio_reg_set(sch, gpio_num, GTS, 1); +	spin_unlock_irqrestore(&sch->lock, flags); +} + +static void sch_irq_mask_unmask(struct irq_data *d, int val) +{ +	struct gpio_chip *gc = irq_data_get_irq_chip_data(d); +	struct sch_gpio *sch = gpiochip_get_data(gc); +	irq_hw_number_t gpio_num = irqd_to_hwirq(d); +	unsigned long flags; + +	spin_lock_irqsave(&sch->lock, flags); +	sch_gpio_reg_set(sch, gpio_num, GGPE, val); +	spin_unlock_irqrestore(&sch->lock, flags); +} + +static void sch_irq_mask(struct irq_data *d) +{ +	sch_irq_mask_unmask(d, 0); +} + +static void sch_irq_unmask(struct irq_data *d) +{ +	sch_irq_mask_unmask(d, 1); +} + +static u32 sch_gpio_gpe_handler(acpi_handle gpe_device, u32 gpe, void *context) +{ +	struct sch_gpio *sch = context; +	struct gpio_chip *gc = &sch->chip; +	unsigned long core_status, resume_status; +	unsigned long pending; +	unsigned long flags; +	int offset; +	u32 ret; + +	spin_lock_irqsave(&sch->lock, flags); + +	core_status = inl(sch->iobase + CORE_BANK_OFFSET + GTS); +	resume_status = inl(sch->iobase + RESUME_BANK_OFFSET + GTS); + +	spin_unlock_irqrestore(&sch->lock, flags); + +	pending = (resume_status << sch->resume_base) | core_status; +	for_each_set_bit(offset, &pending, sch->chip.ngpio) +		generic_handle_irq(irq_find_mapping(gc->irq.domain, offset)); + +	/* Set returning value depending on whether we handled an interrupt */ +	ret = pending ? ACPI_INTERRUPT_HANDLED : ACPI_INTERRUPT_NOT_HANDLED; + +	/* Acknowledge GPE to ACPICA */ +	ret |= ACPI_REENABLE_GPE; + +	return ret; +} + +static void sch_gpio_remove_gpe_handler(void *data) +{ +	struct sch_gpio *sch = data; + +	acpi_disable_gpe(NULL, sch->gpe); +	acpi_remove_gpe_handler(NULL, sch->gpe, sch->gpe_handler); +} + +static int sch_gpio_install_gpe_handler(struct sch_gpio *sch) +{ +	struct device *dev = sch->chip.parent; +	acpi_status status; + +	status = acpi_install_gpe_handler(NULL, sch->gpe, ACPI_GPE_LEVEL_TRIGGERED, +					  sch->gpe_handler, sch); +	if (ACPI_FAILURE(status)) { +		dev_err(dev, "Failed to install GPE handler for %u: %s\n", +			sch->gpe, acpi_format_exception(status)); +		return -ENODEV; +	} + +	status = acpi_enable_gpe(NULL, sch->gpe); +	if (ACPI_FAILURE(status)) { +		dev_err(dev, "Failed to enable GPE handler for %u: %s\n", +			sch->gpe, acpi_format_exception(status)); +		acpi_remove_gpe_handler(NULL, sch->gpe, sch->gpe_handler); +		return -ENODEV; +	} + +	return devm_add_action_or_reset(dev, sch_gpio_remove_gpe_handler, sch); +} +  static int sch_gpio_probe(struct platform_device *pdev)  { +	struct gpio_irq_chip *girq;  	struct sch_gpio *sch;  	struct resource *res; +	int ret;  	sch = devm_kzalloc(&pdev->dev, sizeof(*sch), GFP_KERNEL);  	if (!sch) @@ -207,6 +367,28 @@ static int sch_gpio_probe(struct platform_device *pdev)  	platform_set_drvdata(pdev, sch); +	sch->irqchip.name = "sch_gpio"; +	sch->irqchip.irq_ack = sch_irq_ack; +	sch->irqchip.irq_mask = sch_irq_mask; +	sch->irqchip.irq_unmask = sch_irq_unmask; +	sch->irqchip.irq_set_type = sch_irq_type; + +	girq = &sch->chip.irq; +	girq->chip = &sch->irqchip; +	girq->num_parents = 0; +	girq->parents = NULL; +	girq->parent_handler = NULL; +	girq->default_type = IRQ_TYPE_NONE; +	girq->handler = handle_bad_irq; + +	/* GPE setup is optional */ +	sch->gpe = GPE0E_GPIO; +	sch->gpe_handler = sch_gpio_gpe_handler; + +	ret = sch_gpio_install_gpe_handler(sch); +	if (ret) +		dev_warn(&pdev->dev, "Can't setup GPE, no IRQ support\n"); +  	return devm_gpiochip_add_data(&pdev->dev, &sch->chip, sch);  } | 
