mirror of
https://github.com/torvalds/linux.git
synced 2024-11-10 14:11:52 +00:00
gpio fixes for v6.0-rc4
- MAINTAINERS update - fix resource leaks in gpio-mockup and gpio-pxa - add missing locking in gpio-pca953x - use 32-bit I/O in gpio-realtek-otto - make irq_chip structures immutable in four more drivers -----BEGIN PGP SIGNATURE----- iQIzBAABCgAdFiEEFp3rbAvDxGAT0sefEacuoBRx13IFAmMTwMsACgkQEacuoBRx 13IAkQ//SmLx/kNTjpuWtqa6MQKR3hbbXmLYbpMI3hTqpQ+Q/SXR6pxEJXmj1+K6 GlC9oSN3oV0AsayLaJ2YgmqH59rtjS+dJILXpmIx1jchnnBD7xXNeA3uX4GoJid0 NEKbBRb9dUz/CwXNiLmGeOPaDoi81K0ha+9RP4/lnjEzmexc6jfWd4cdics2XCyr Zr8qY8bk0kNzWWdEuhXIN2WFL2XMnWSF/Y9L0+LtpMZf5BtT1vVu/6qwL4Da2KTQ pnmMmDjQG1npe1AJ/CvObhzxYyaJRFPA9zCjhCJskm2SiwxwXTCArjMsl2vFcboi f8JdqYfbH2D4LpJsWh03K7OEhPRzThdLBKy1HOJkrbqqVTKmUl750121O1p2AV9H bMUpsmXw+ali/C4e4wBvNcS0XZK7vP1T3yAzjukZK2KZvhEDBo9YQECWnuUhFh/J kB6qt9PlhdwHdLhe7+LnPWTuUdF5yQL4DR+53Tigjnqt9tQ5Mh1d2V/XVjRATppa +cmCKp5NvVTmoenjO1c9QnMErzrCWoQ8mZSEalIoC5vQ6wzXM9OaiRla0V9DS8aV dpankWmKyZ74i3f0QWMKHoqx+AKP+6AC7iRtPcSzZe2QjVOsWeStwHl6+GzlazDk /Ml+iK9eZsOCJvJFq+5/r3n3sJWGcl5a4WpkZ2QYQ95fJprZhNQ= =VkA4 -----END PGP SIGNATURE----- Merge tag 'gpio-fixes-for-v6.0-rc4' of git://git.kernel.org/pub/scm/linux/kernel/git/brgl/linux Pull gpio fixes from Bartosz Golaszewski: "A a set of fixes from the GPIO subsystem. Most are small driver fixes except the realtek-otto driver patch which is pretty big but addresses a significant flaw that can cause the CPU to stay infinitely busy on uncleared ISR on some platforms. Summary: - MAINTAINERS update - fix resource leaks in gpio-mockup and gpio-pxa - add missing locking in gpio-pca953x - use 32-bit I/O in gpio-realtek-otto - make irq_chip structures immutable in four more drivers" * tag 'gpio-fixes-for-v6.0-rc4' of git://git.kernel.org/pub/scm/linux/kernel/git/brgl/linux: gpio: ws16c48: Make irq_chip immutable gpio: 104-idio-16: Make irq_chip immutable gpio: 104-idi-48: Make irq_chip immutable gpio: 104-dio-48e: Make irq_chip immutable gpio: realtek-otto: switch to 32-bit I/O gpio: pca953x: Add mutex_lock for regcache sync in PM gpio: mockup: remove gpio debugfs when remove device gpio: pxa: use devres for the clock struct MAINTAINERS: rectify entry for XILINX GPIO DRIVER
This commit is contained in:
commit
7726d4c3e6
@ -22307,7 +22307,7 @@ M: Shubhrajyoti Datta <shubhrajyoti.datta@xilinx.com>
|
||||
R: Srinivas Neeli <srinivas.neeli@xilinx.com>
|
||||
R: Michal Simek <michal.simek@xilinx.com>
|
||||
S: Maintained
|
||||
F: Documentation/devicetree/bindings/gpio/gpio-xilinx.txt
|
||||
F: Documentation/devicetree/bindings/gpio/xlnx,gpio-xilinx.yaml
|
||||
F: Documentation/devicetree/bindings/gpio/gpio-zynq.yaml
|
||||
F: drivers/gpio/gpio-xilinx.c
|
||||
F: drivers/gpio/gpio-zynq.c
|
||||
|
@ -164,6 +164,7 @@ static void dio48e_irq_mask(struct irq_data *data)
|
||||
dio48egpio->irq_mask &= ~BIT(0);
|
||||
else
|
||||
dio48egpio->irq_mask &= ~BIT(1);
|
||||
gpiochip_disable_irq(chip, offset);
|
||||
|
||||
if (!dio48egpio->irq_mask)
|
||||
/* disable interrupts */
|
||||
@ -191,6 +192,7 @@ static void dio48e_irq_unmask(struct irq_data *data)
|
||||
iowrite8(0x00, &dio48egpio->reg->enable_interrupt);
|
||||
}
|
||||
|
||||
gpiochip_enable_irq(chip, offset);
|
||||
if (offset == 19)
|
||||
dio48egpio->irq_mask |= BIT(0);
|
||||
else
|
||||
@ -213,12 +215,14 @@ static int dio48e_irq_set_type(struct irq_data *data, unsigned int flow_type)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct irq_chip dio48e_irqchip = {
|
||||
static const struct irq_chip dio48e_irqchip = {
|
||||
.name = "104-dio-48e",
|
||||
.irq_ack = dio48e_irq_ack,
|
||||
.irq_mask = dio48e_irq_mask,
|
||||
.irq_unmask = dio48e_irq_unmask,
|
||||
.irq_set_type = dio48e_irq_set_type
|
||||
.irq_set_type = dio48e_irq_set_type,
|
||||
.flags = IRQCHIP_IMMUTABLE,
|
||||
GPIOCHIP_IRQ_RESOURCE_HELPERS,
|
||||
};
|
||||
|
||||
static irqreturn_t dio48e_irq_handler(int irq, void *dev_id)
|
||||
@ -322,7 +326,7 @@ static int dio48e_probe(struct device *dev, unsigned int id)
|
||||
dio48egpio->chip.set_multiple = dio48e_gpio_set_multiple;
|
||||
|
||||
girq = &dio48egpio->chip.irq;
|
||||
girq->chip = &dio48e_irqchip;
|
||||
gpio_irq_chip_set_chip(girq, &dio48e_irqchip);
|
||||
/* This will let us handle the parent IRQ in the driver */
|
||||
girq->parent_handler = NULL;
|
||||
girq->num_parents = 0;
|
||||
|
@ -113,6 +113,7 @@ static void idi_48_irq_mask(struct irq_data *data)
|
||||
spin_lock_irqsave(&idi48gpio->lock, flags);
|
||||
|
||||
idi48gpio->irq_mask[boundary] &= ~mask;
|
||||
gpiochip_disable_irq(chip, offset);
|
||||
|
||||
/* Exit early if there are still input lines with IRQ unmasked */
|
||||
if (idi48gpio->irq_mask[boundary])
|
||||
@ -140,6 +141,7 @@ static void idi_48_irq_unmask(struct irq_data *data)
|
||||
|
||||
prev_irq_mask = idi48gpio->irq_mask[boundary];
|
||||
|
||||
gpiochip_enable_irq(chip, offset);
|
||||
idi48gpio->irq_mask[boundary] |= mask;
|
||||
|
||||
/* Exit early if IRQ was already unmasked for this boundary */
|
||||
@ -164,12 +166,14 @@ static int idi_48_irq_set_type(struct irq_data *data, unsigned int flow_type)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct irq_chip idi_48_irqchip = {
|
||||
static const struct irq_chip idi_48_irqchip = {
|
||||
.name = "104-idi-48",
|
||||
.irq_ack = idi_48_irq_ack,
|
||||
.irq_mask = idi_48_irq_mask,
|
||||
.irq_unmask = idi_48_irq_unmask,
|
||||
.irq_set_type = idi_48_irq_set_type
|
||||
.irq_set_type = idi_48_irq_set_type,
|
||||
.flags = IRQCHIP_IMMUTABLE,
|
||||
GPIOCHIP_IRQ_RESOURCE_HELPERS,
|
||||
};
|
||||
|
||||
static irqreturn_t idi_48_irq_handler(int irq, void *dev_id)
|
||||
@ -267,7 +271,7 @@ static int idi_48_probe(struct device *dev, unsigned int id)
|
||||
idi48gpio->chip.get_multiple = idi_48_gpio_get_multiple;
|
||||
|
||||
girq = &idi48gpio->chip.irq;
|
||||
girq->chip = &idi_48_irqchip;
|
||||
gpio_irq_chip_set_chip(girq, &idi_48_irqchip);
|
||||
/* This will let us handle the parent IRQ in the driver */
|
||||
girq->parent_handler = NULL;
|
||||
girq->num_parents = 0;
|
||||
|
@ -174,10 +174,11 @@ static void idio_16_irq_mask(struct irq_data *data)
|
||||
{
|
||||
struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
|
||||
struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
|
||||
const unsigned long mask = BIT(irqd_to_hwirq(data));
|
||||
const unsigned long offset = irqd_to_hwirq(data);
|
||||
unsigned long flags;
|
||||
|
||||
idio16gpio->irq_mask &= ~mask;
|
||||
idio16gpio->irq_mask &= ~BIT(offset);
|
||||
gpiochip_disable_irq(chip, offset);
|
||||
|
||||
if (!idio16gpio->irq_mask) {
|
||||
raw_spin_lock_irqsave(&idio16gpio->lock, flags);
|
||||
@ -192,11 +193,12 @@ static void idio_16_irq_unmask(struct irq_data *data)
|
||||
{
|
||||
struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
|
||||
struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
|
||||
const unsigned long mask = BIT(irqd_to_hwirq(data));
|
||||
const unsigned long offset = irqd_to_hwirq(data);
|
||||
const unsigned long prev_irq_mask = idio16gpio->irq_mask;
|
||||
unsigned long flags;
|
||||
|
||||
idio16gpio->irq_mask |= mask;
|
||||
gpiochip_enable_irq(chip, offset);
|
||||
idio16gpio->irq_mask |= BIT(offset);
|
||||
|
||||
if (!prev_irq_mask) {
|
||||
raw_spin_lock_irqsave(&idio16gpio->lock, flags);
|
||||
@ -217,12 +219,14 @@ static int idio_16_irq_set_type(struct irq_data *data, unsigned int flow_type)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct irq_chip idio_16_irqchip = {
|
||||
static const struct irq_chip idio_16_irqchip = {
|
||||
.name = "104-idio-16",
|
||||
.irq_ack = idio_16_irq_ack,
|
||||
.irq_mask = idio_16_irq_mask,
|
||||
.irq_unmask = idio_16_irq_unmask,
|
||||
.irq_set_type = idio_16_irq_set_type
|
||||
.irq_set_type = idio_16_irq_set_type,
|
||||
.flags = IRQCHIP_IMMUTABLE,
|
||||
GPIOCHIP_IRQ_RESOURCE_HELPERS,
|
||||
};
|
||||
|
||||
static irqreturn_t idio_16_irq_handler(int irq, void *dev_id)
|
||||
@ -299,7 +303,7 @@ static int idio_16_probe(struct device *dev, unsigned int id)
|
||||
idio16gpio->out_state = 0xFFFF;
|
||||
|
||||
girq = &idio16gpio->chip.irq;
|
||||
girq->chip = &idio_16_irqchip;
|
||||
gpio_irq_chip_set_chip(girq, &idio_16_irqchip);
|
||||
/* This will let us handle the parent IRQ in the driver */
|
||||
girq->parent_handler = NULL;
|
||||
girq->num_parents = 0;
|
||||
|
@ -373,6 +373,13 @@ static void gpio_mockup_debugfs_setup(struct device *dev,
|
||||
}
|
||||
}
|
||||
|
||||
static void gpio_mockup_debugfs_cleanup(void *data)
|
||||
{
|
||||
struct gpio_mockup_chip *chip = data;
|
||||
|
||||
debugfs_remove_recursive(chip->dbg_dir);
|
||||
}
|
||||
|
||||
static void gpio_mockup_dispose_mappings(void *data)
|
||||
{
|
||||
struct gpio_mockup_chip *chip = data;
|
||||
@ -455,7 +462,7 @@ static int gpio_mockup_probe(struct platform_device *pdev)
|
||||
|
||||
gpio_mockup_debugfs_setup(dev, chip);
|
||||
|
||||
return 0;
|
||||
return devm_add_action_or_reset(dev, gpio_mockup_debugfs_cleanup, chip);
|
||||
}
|
||||
|
||||
static const struct of_device_id gpio_mockup_of_match[] = {
|
||||
|
@ -1175,7 +1175,9 @@ static int pca953x_suspend(struct device *dev)
|
||||
{
|
||||
struct pca953x_chip *chip = dev_get_drvdata(dev);
|
||||
|
||||
mutex_lock(&chip->i2c_lock);
|
||||
regcache_cache_only(chip->regmap, true);
|
||||
mutex_unlock(&chip->i2c_lock);
|
||||
|
||||
if (atomic_read(&chip->wakeup_path))
|
||||
device_set_wakeup_path(dev);
|
||||
@ -1198,13 +1200,17 @@ static int pca953x_resume(struct device *dev)
|
||||
}
|
||||
}
|
||||
|
||||
mutex_lock(&chip->i2c_lock);
|
||||
regcache_cache_only(chip->regmap, false);
|
||||
regcache_mark_dirty(chip->regmap);
|
||||
ret = pca953x_regcache_sync(dev);
|
||||
if (ret)
|
||||
if (ret) {
|
||||
mutex_unlock(&chip->i2c_lock);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = regcache_sync(chip->regmap);
|
||||
mutex_unlock(&chip->i2c_lock);
|
||||
if (ret) {
|
||||
dev_err(dev, "Failed to restore register map: %d\n", ret);
|
||||
return ret;
|
||||
|
@ -661,24 +661,17 @@ static int pxa_gpio_probe(struct platform_device *pdev)
|
||||
if (IS_ERR(gpio_reg_base))
|
||||
return PTR_ERR(gpio_reg_base);
|
||||
|
||||
clk = clk_get(&pdev->dev, NULL);
|
||||
clk = devm_clk_get_enabled(&pdev->dev, NULL);
|
||||
if (IS_ERR(clk)) {
|
||||
dev_err(&pdev->dev, "Error %ld to get gpio clock\n",
|
||||
PTR_ERR(clk));
|
||||
return PTR_ERR(clk);
|
||||
}
|
||||
ret = clk_prepare_enable(clk);
|
||||
if (ret) {
|
||||
clk_put(clk);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Initialize GPIO chips */
|
||||
ret = pxa_init_gpio_chip(pchip, pxa_last_gpio + 1, gpio_reg_base);
|
||||
if (ret) {
|
||||
clk_put(clk);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* clear all GPIO edge detects */
|
||||
for_each_gpio_bank(gpio, c, pchip) {
|
||||
|
@ -46,10 +46,20 @@
|
||||
* @lock: Lock for accessing the IRQ registers and values
|
||||
* @intr_mask: Mask for interrupts lines
|
||||
* @intr_type: Interrupt type selection
|
||||
* @bank_read: Read a bank setting as a single 32-bit value
|
||||
* @bank_write: Write a bank setting as a single 32-bit value
|
||||
* @imr_line_pos: Bit shift of an IRQ line's IMR value.
|
||||
*
|
||||
* The DIR, DATA, and ISR registers consist of four 8-bit port values, packed
|
||||
* into a single 32-bit register. Use @bank_read (@bank_write) to get (assign)
|
||||
* a value from (to) these registers. The IMR register consists of four 16-bit
|
||||
* port values, packed into two 32-bit registers. Use @imr_line_pos to get the
|
||||
* bit shift of the 2-bit field for a line's IMR settings. Shifts larger than
|
||||
* 32 overflow into the second register.
|
||||
*
|
||||
* Because the interrupt mask register (IMR) combines the function of IRQ type
|
||||
* selection and masking, two extra values are stored. @intr_mask is used to
|
||||
* mask/unmask the interrupts for a GPIO port, and @intr_type is used to store
|
||||
* mask/unmask the interrupts for a GPIO line, and @intr_type is used to store
|
||||
* the selected interrupt types. The logical AND of these values is written to
|
||||
* IMR on changes.
|
||||
*/
|
||||
@ -59,10 +69,11 @@ struct realtek_gpio_ctrl {
|
||||
void __iomem *cpumask_base;
|
||||
struct cpumask cpu_irq_maskable;
|
||||
raw_spinlock_t lock;
|
||||
u16 intr_mask[REALTEK_GPIO_PORTS_PER_BANK];
|
||||
u16 intr_type[REALTEK_GPIO_PORTS_PER_BANK];
|
||||
unsigned int (*port_offset_u8)(unsigned int port);
|
||||
unsigned int (*port_offset_u16)(unsigned int port);
|
||||
u8 intr_mask[REALTEK_GPIO_MAX];
|
||||
u8 intr_type[REALTEK_GPIO_MAX];
|
||||
u32 (*bank_read)(void __iomem *reg);
|
||||
void (*bank_write)(void __iomem *reg, u32 value);
|
||||
unsigned int (*line_imr_pos)(unsigned int line);
|
||||
};
|
||||
|
||||
/* Expand with more flags as devices with other quirks are added */
|
||||
@ -101,14 +112,22 @@ static struct realtek_gpio_ctrl *irq_data_to_ctrl(struct irq_data *data)
|
||||
* port. The two interrupt mask registers store two bits per GPIO, so use u16
|
||||
* values.
|
||||
*/
|
||||
static unsigned int realtek_gpio_port_offset_u8(unsigned int port)
|
||||
static u32 realtek_gpio_bank_read_swapped(void __iomem *reg)
|
||||
{
|
||||
return port;
|
||||
return ioread32be(reg);
|
||||
}
|
||||
|
||||
static unsigned int realtek_gpio_port_offset_u16(unsigned int port)
|
||||
static void realtek_gpio_bank_write_swapped(void __iomem *reg, u32 value)
|
||||
{
|
||||
return 2 * port;
|
||||
iowrite32be(value, reg);
|
||||
}
|
||||
|
||||
static unsigned int realtek_gpio_line_imr_pos_swapped(unsigned int line)
|
||||
{
|
||||
unsigned int port_pin = line % 8;
|
||||
unsigned int port = line / 8;
|
||||
|
||||
return 2 * (8 * (port ^ 1) + port_pin);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -119,66 +138,67 @@ static unsigned int realtek_gpio_port_offset_u16(unsigned int port)
|
||||
* per GPIO, so use u16 values. The first register contains ports 1 and 0, the
|
||||
* second ports 3 and 2.
|
||||
*/
|
||||
static unsigned int realtek_gpio_port_offset_u8_rev(unsigned int port)
|
||||
static u32 realtek_gpio_bank_read(void __iomem *reg)
|
||||
{
|
||||
return 3 - port;
|
||||
return ioread32(reg);
|
||||
}
|
||||
|
||||
static unsigned int realtek_gpio_port_offset_u16_rev(unsigned int port)
|
||||
static void realtek_gpio_bank_write(void __iomem *reg, u32 value)
|
||||
{
|
||||
return 2 * (port ^ 1);
|
||||
iowrite32(value, reg);
|
||||
}
|
||||
|
||||
static void realtek_gpio_write_imr(struct realtek_gpio_ctrl *ctrl,
|
||||
unsigned int port, u16 irq_type, u16 irq_mask)
|
||||
static unsigned int realtek_gpio_line_imr_pos(unsigned int line)
|
||||
{
|
||||
iowrite16(irq_type & irq_mask,
|
||||
ctrl->base + REALTEK_GPIO_REG_IMR + ctrl->port_offset_u16(port));
|
||||
return 2 * line;
|
||||
}
|
||||
|
||||
static void realtek_gpio_clear_isr(struct realtek_gpio_ctrl *ctrl,
|
||||
unsigned int port, u8 mask)
|
||||
static void realtek_gpio_clear_isr(struct realtek_gpio_ctrl *ctrl, u32 mask)
|
||||
{
|
||||
iowrite8(mask, ctrl->base + REALTEK_GPIO_REG_ISR + ctrl->port_offset_u8(port));
|
||||
ctrl->bank_write(ctrl->base + REALTEK_GPIO_REG_ISR, mask);
|
||||
}
|
||||
|
||||
static u8 realtek_gpio_read_isr(struct realtek_gpio_ctrl *ctrl, unsigned int port)
|
||||
static u32 realtek_gpio_read_isr(struct realtek_gpio_ctrl *ctrl)
|
||||
{
|
||||
return ioread8(ctrl->base + REALTEK_GPIO_REG_ISR + ctrl->port_offset_u8(port));
|
||||
return ctrl->bank_read(ctrl->base + REALTEK_GPIO_REG_ISR);
|
||||
}
|
||||
|
||||
/* Set the rising and falling edge mask bits for a GPIO port pin */
|
||||
static u16 realtek_gpio_imr_bits(unsigned int pin, u16 value)
|
||||
/* Set the rising and falling edge mask bits for a GPIO pin */
|
||||
static void realtek_gpio_update_line_imr(struct realtek_gpio_ctrl *ctrl, unsigned int line)
|
||||
{
|
||||
return (value & REALTEK_GPIO_IMR_LINE_MASK) << 2 * pin;
|
||||
void __iomem *reg = ctrl->base + REALTEK_GPIO_REG_IMR;
|
||||
unsigned int line_shift = ctrl->line_imr_pos(line);
|
||||
unsigned int shift = line_shift % 32;
|
||||
u32 irq_type = ctrl->intr_type[line];
|
||||
u32 irq_mask = ctrl->intr_mask[line];
|
||||
u32 reg_val;
|
||||
|
||||
reg += 4 * (line_shift / 32);
|
||||
reg_val = ioread32(reg);
|
||||
reg_val &= ~(REALTEK_GPIO_IMR_LINE_MASK << shift);
|
||||
reg_val |= (irq_type & irq_mask & REALTEK_GPIO_IMR_LINE_MASK) << shift;
|
||||
iowrite32(reg_val, reg);
|
||||
}
|
||||
|
||||
static void realtek_gpio_irq_ack(struct irq_data *data)
|
||||
{
|
||||
struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data);
|
||||
irq_hw_number_t line = irqd_to_hwirq(data);
|
||||
unsigned int port = line / 8;
|
||||
unsigned int port_pin = line % 8;
|
||||
|
||||
realtek_gpio_clear_isr(ctrl, port, BIT(port_pin));
|
||||
realtek_gpio_clear_isr(ctrl, BIT(line));
|
||||
}
|
||||
|
||||
static void realtek_gpio_irq_unmask(struct irq_data *data)
|
||||
{
|
||||
struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data);
|
||||
unsigned int line = irqd_to_hwirq(data);
|
||||
unsigned int port = line / 8;
|
||||
unsigned int port_pin = line % 8;
|
||||
unsigned long flags;
|
||||
u16 m;
|
||||
|
||||
gpiochip_enable_irq(&ctrl->gc, line);
|
||||
|
||||
raw_spin_lock_irqsave(&ctrl->lock, flags);
|
||||
m = ctrl->intr_mask[port];
|
||||
m |= realtek_gpio_imr_bits(port_pin, REALTEK_GPIO_IMR_LINE_MASK);
|
||||
ctrl->intr_mask[port] = m;
|
||||
realtek_gpio_write_imr(ctrl, port, ctrl->intr_type[port], m);
|
||||
ctrl->intr_mask[line] = REALTEK_GPIO_IMR_LINE_MASK;
|
||||
realtek_gpio_update_line_imr(ctrl, line);
|
||||
raw_spin_unlock_irqrestore(&ctrl->lock, flags);
|
||||
}
|
||||
|
||||
@ -186,16 +206,11 @@ static void realtek_gpio_irq_mask(struct irq_data *data)
|
||||
{
|
||||
struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data);
|
||||
unsigned int line = irqd_to_hwirq(data);
|
||||
unsigned int port = line / 8;
|
||||
unsigned int port_pin = line % 8;
|
||||
unsigned long flags;
|
||||
u16 m;
|
||||
|
||||
raw_spin_lock_irqsave(&ctrl->lock, flags);
|
||||
m = ctrl->intr_mask[port];
|
||||
m &= ~realtek_gpio_imr_bits(port_pin, REALTEK_GPIO_IMR_LINE_MASK);
|
||||
ctrl->intr_mask[port] = m;
|
||||
realtek_gpio_write_imr(ctrl, port, ctrl->intr_type[port], m);
|
||||
ctrl->intr_mask[line] = 0;
|
||||
realtek_gpio_update_line_imr(ctrl, line);
|
||||
raw_spin_unlock_irqrestore(&ctrl->lock, flags);
|
||||
|
||||
gpiochip_disable_irq(&ctrl->gc, line);
|
||||
@ -205,10 +220,8 @@ static int realtek_gpio_irq_set_type(struct irq_data *data, unsigned int flow_ty
|
||||
{
|
||||
struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data);
|
||||
unsigned int line = irqd_to_hwirq(data);
|
||||
unsigned int port = line / 8;
|
||||
unsigned int port_pin = line % 8;
|
||||
unsigned long flags;
|
||||
u16 type, t;
|
||||
u8 type;
|
||||
|
||||
switch (flow_type & IRQ_TYPE_SENSE_MASK) {
|
||||
case IRQ_TYPE_EDGE_FALLING:
|
||||
@ -227,11 +240,8 @@ static int realtek_gpio_irq_set_type(struct irq_data *data, unsigned int flow_ty
|
||||
irq_set_handler_locked(data, handle_edge_irq);
|
||||
|
||||
raw_spin_lock_irqsave(&ctrl->lock, flags);
|
||||
t = ctrl->intr_type[port];
|
||||
t &= ~realtek_gpio_imr_bits(port_pin, REALTEK_GPIO_IMR_LINE_MASK);
|
||||
t |= realtek_gpio_imr_bits(port_pin, type);
|
||||
ctrl->intr_type[port] = t;
|
||||
realtek_gpio_write_imr(ctrl, port, t, ctrl->intr_mask[port]);
|
||||
ctrl->intr_type[line] = type;
|
||||
realtek_gpio_update_line_imr(ctrl, line);
|
||||
raw_spin_unlock_irqrestore(&ctrl->lock, flags);
|
||||
|
||||
return 0;
|
||||
@ -242,28 +252,21 @@ static void realtek_gpio_irq_handler(struct irq_desc *desc)
|
||||
struct gpio_chip *gc = irq_desc_get_handler_data(desc);
|
||||
struct realtek_gpio_ctrl *ctrl = gpiochip_get_data(gc);
|
||||
struct irq_chip *irq_chip = irq_desc_get_chip(desc);
|
||||
unsigned int lines_done;
|
||||
unsigned int port_pin_count;
|
||||
unsigned long status;
|
||||
int offset;
|
||||
|
||||
chained_irq_enter(irq_chip, desc);
|
||||
|
||||
for (lines_done = 0; lines_done < gc->ngpio; lines_done += 8) {
|
||||
status = realtek_gpio_read_isr(ctrl, lines_done / 8);
|
||||
port_pin_count = min(gc->ngpio - lines_done, 8U);
|
||||
for_each_set_bit(offset, &status, port_pin_count)
|
||||
generic_handle_domain_irq(gc->irq.domain, offset + lines_done);
|
||||
}
|
||||
status = realtek_gpio_read_isr(ctrl);
|
||||
for_each_set_bit(offset, &status, gc->ngpio)
|
||||
generic_handle_domain_irq(gc->irq.domain, offset);
|
||||
|
||||
chained_irq_exit(irq_chip, desc);
|
||||
}
|
||||
|
||||
static inline void __iomem *realtek_gpio_irq_cpu_mask(struct realtek_gpio_ctrl *ctrl,
|
||||
unsigned int port, int cpu)
|
||||
static inline void __iomem *realtek_gpio_irq_cpu_mask(struct realtek_gpio_ctrl *ctrl, int cpu)
|
||||
{
|
||||
return ctrl->cpumask_base + ctrl->port_offset_u8(port) +
|
||||
REALTEK_GPIO_PORTS_PER_BANK * cpu;
|
||||
return ctrl->cpumask_base + REALTEK_GPIO_PORTS_PER_BANK * cpu;
|
||||
}
|
||||
|
||||
static int realtek_gpio_irq_set_affinity(struct irq_data *data,
|
||||
@ -271,12 +274,10 @@ static int realtek_gpio_irq_set_affinity(struct irq_data *data,
|
||||
{
|
||||
struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data);
|
||||
unsigned int line = irqd_to_hwirq(data);
|
||||
unsigned int port = line / 8;
|
||||
unsigned int port_pin = line % 8;
|
||||
void __iomem *irq_cpu_mask;
|
||||
unsigned long flags;
|
||||
int cpu;
|
||||
u8 v;
|
||||
u32 v;
|
||||
|
||||
if (!ctrl->cpumask_base)
|
||||
return -ENXIO;
|
||||
@ -284,15 +285,15 @@ static int realtek_gpio_irq_set_affinity(struct irq_data *data,
|
||||
raw_spin_lock_irqsave(&ctrl->lock, flags);
|
||||
|
||||
for_each_cpu(cpu, &ctrl->cpu_irq_maskable) {
|
||||
irq_cpu_mask = realtek_gpio_irq_cpu_mask(ctrl, port, cpu);
|
||||
v = ioread8(irq_cpu_mask);
|
||||
irq_cpu_mask = realtek_gpio_irq_cpu_mask(ctrl, cpu);
|
||||
v = ctrl->bank_read(irq_cpu_mask);
|
||||
|
||||
if (cpumask_test_cpu(cpu, dest))
|
||||
v |= BIT(port_pin);
|
||||
v |= BIT(line);
|
||||
else
|
||||
v &= ~BIT(port_pin);
|
||||
v &= ~BIT(line);
|
||||
|
||||
iowrite8(v, irq_cpu_mask);
|
||||
ctrl->bank_write(irq_cpu_mask, v);
|
||||
}
|
||||
|
||||
raw_spin_unlock_irqrestore(&ctrl->lock, flags);
|
||||
@ -305,16 +306,17 @@ static int realtek_gpio_irq_set_affinity(struct irq_data *data,
|
||||
static int realtek_gpio_irq_init(struct gpio_chip *gc)
|
||||
{
|
||||
struct realtek_gpio_ctrl *ctrl = gpiochip_get_data(gc);
|
||||
unsigned int port;
|
||||
u32 mask_all = GENMASK(gc->ngpio - 1, 0);
|
||||
unsigned int line;
|
||||
int cpu;
|
||||
|
||||
for (port = 0; (port * 8) < gc->ngpio; port++) {
|
||||
realtek_gpio_write_imr(ctrl, port, 0, 0);
|
||||
realtek_gpio_clear_isr(ctrl, port, GENMASK(7, 0));
|
||||
for (line = 0; line < gc->ngpio; line++)
|
||||
realtek_gpio_update_line_imr(ctrl, line);
|
||||
|
||||
for_each_cpu(cpu, &ctrl->cpu_irq_maskable)
|
||||
iowrite8(GENMASK(7, 0), realtek_gpio_irq_cpu_mask(ctrl, port, cpu));
|
||||
}
|
||||
realtek_gpio_clear_isr(ctrl, mask_all);
|
||||
|
||||
for_each_cpu(cpu, &ctrl->cpu_irq_maskable)
|
||||
ctrl->bank_write(realtek_gpio_irq_cpu_mask(ctrl, cpu), mask_all);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -387,12 +389,14 @@ static int realtek_gpio_probe(struct platform_device *pdev)
|
||||
|
||||
if (dev_flags & GPIO_PORTS_REVERSED) {
|
||||
bgpio_flags = 0;
|
||||
ctrl->port_offset_u8 = realtek_gpio_port_offset_u8_rev;
|
||||
ctrl->port_offset_u16 = realtek_gpio_port_offset_u16_rev;
|
||||
ctrl->bank_read = realtek_gpio_bank_read;
|
||||
ctrl->bank_write = realtek_gpio_bank_write;
|
||||
ctrl->line_imr_pos = realtek_gpio_line_imr_pos;
|
||||
} else {
|
||||
bgpio_flags = BGPIOF_BIG_ENDIAN_BYTE_ORDER;
|
||||
ctrl->port_offset_u8 = realtek_gpio_port_offset_u8;
|
||||
ctrl->port_offset_u16 = realtek_gpio_port_offset_u16;
|
||||
ctrl->bank_read = realtek_gpio_bank_read_swapped;
|
||||
ctrl->bank_write = realtek_gpio_bank_write_swapped;
|
||||
ctrl->line_imr_pos = realtek_gpio_line_imr_pos_swapped;
|
||||
}
|
||||
|
||||
err = bgpio_init(&ctrl->gc, dev, 4,
|
||||
|
@ -265,6 +265,7 @@ static void ws16c48_irq_mask(struct irq_data *data)
|
||||
raw_spin_lock_irqsave(&ws16c48gpio->lock, flags);
|
||||
|
||||
ws16c48gpio->irq_mask &= ~mask;
|
||||
gpiochip_disable_irq(chip, offset);
|
||||
port_state = ws16c48gpio->irq_mask >> (8 * port);
|
||||
|
||||
/* Select Register Page 2; Unlock all I/O ports */
|
||||
@ -295,6 +296,7 @@ static void ws16c48_irq_unmask(struct irq_data *data)
|
||||
|
||||
raw_spin_lock_irqsave(&ws16c48gpio->lock, flags);
|
||||
|
||||
gpiochip_enable_irq(chip, offset);
|
||||
ws16c48gpio->irq_mask |= mask;
|
||||
port_state = ws16c48gpio->irq_mask >> (8 * port);
|
||||
|
||||
@ -356,12 +358,14 @@ static int ws16c48_irq_set_type(struct irq_data *data, unsigned flow_type)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct irq_chip ws16c48_irqchip = {
|
||||
static const struct irq_chip ws16c48_irqchip = {
|
||||
.name = "ws16c48",
|
||||
.irq_ack = ws16c48_irq_ack,
|
||||
.irq_mask = ws16c48_irq_mask,
|
||||
.irq_unmask = ws16c48_irq_unmask,
|
||||
.irq_set_type = ws16c48_irq_set_type
|
||||
.irq_set_type = ws16c48_irq_set_type,
|
||||
.flags = IRQCHIP_IMMUTABLE,
|
||||
GPIOCHIP_IRQ_RESOURCE_HELPERS,
|
||||
};
|
||||
|
||||
static irqreturn_t ws16c48_irq_handler(int irq, void *dev_id)
|
||||
@ -463,7 +467,7 @@ static int ws16c48_probe(struct device *dev, unsigned int id)
|
||||
ws16c48gpio->chip.set_multiple = ws16c48_gpio_set_multiple;
|
||||
|
||||
girq = &ws16c48gpio->chip.irq;
|
||||
girq->chip = &ws16c48_irqchip;
|
||||
gpio_irq_chip_set_chip(girq, &ws16c48_irqchip);
|
||||
/* This will let us handle the parent IRQ in the driver */
|
||||
girq->parent_handler = NULL;
|
||||
girq->num_parents = 0;
|
||||
|
Loading…
Reference in New Issue
Block a user