diff options
| -rw-r--r-- | drivers/sh/intc/chip.c | 53 | ||||
| -rw-r--r-- | drivers/sh/intc/core.c | 39 | ||||
| -rw-r--r-- | drivers/sh/intc/internals.h | 2 | ||||
| -rw-r--r-- | drivers/sh/intc/virq.c | 14 |
4 files changed, 68 insertions, 40 deletions
diff --git a/drivers/sh/intc/chip.c b/drivers/sh/intc/chip.c index 35c03706cc21..de885a0f917a 100644 --- a/drivers/sh/intc/chip.c +++ b/drivers/sh/intc/chip.c | |||
| @@ -12,15 +12,16 @@ | |||
| 12 | #include <linux/io.h> | 12 | #include <linux/io.h> |
| 13 | #include "internals.h" | 13 | #include "internals.h" |
| 14 | 14 | ||
| 15 | void _intc_enable(unsigned int irq, unsigned long handle) | 15 | void _intc_enable(struct irq_data *data, unsigned long handle) |
| 16 | { | 16 | { |
| 17 | unsigned int irq = data->irq; | ||
| 17 | struct intc_desc_int *d = get_intc_desc(irq); | 18 | struct intc_desc_int *d = get_intc_desc(irq); |
| 18 | unsigned long addr; | 19 | unsigned long addr; |
| 19 | unsigned int cpu; | 20 | unsigned int cpu; |
| 20 | 21 | ||
| 21 | for (cpu = 0; cpu < SMP_NR(d, _INTC_ADDR_E(handle)); cpu++) { | 22 | for (cpu = 0; cpu < SMP_NR(d, _INTC_ADDR_E(handle)); cpu++) { |
| 22 | #ifdef CONFIG_SMP | 23 | #ifdef CONFIG_SMP |
| 23 | if (!cpumask_test_cpu(cpu, irq_to_desc(irq)->affinity)) | 24 | if (!cpumask_test_cpu(cpu, data->affinity)) |
| 24 | continue; | 25 | continue; |
| 25 | #endif | 26 | #endif |
| 26 | addr = INTC_REG(d, _INTC_ADDR_E(handle), cpu); | 27 | addr = INTC_REG(d, _INTC_ADDR_E(handle), cpu); |
| @@ -31,15 +32,16 @@ void _intc_enable(unsigned int irq, unsigned long handle) | |||
| 31 | intc_balancing_enable(irq); | 32 | intc_balancing_enable(irq); |
| 32 | } | 33 | } |
| 33 | 34 | ||
| 34 | static void intc_enable(unsigned int irq) | 35 | static void intc_enable(struct irq_data *data) |
| 35 | { | 36 | { |
| 36 | _intc_enable(irq, (unsigned long)get_irq_chip_data(irq)); | 37 | _intc_enable(data, (unsigned long)irq_data_get_irq_chip_data(data)); |
| 37 | } | 38 | } |
| 38 | 39 | ||
| 39 | static void intc_disable(unsigned int irq) | 40 | static void intc_disable(struct irq_data *data) |
| 40 | { | 41 | { |
| 42 | unsigned int irq = data->irq; | ||
| 41 | struct intc_desc_int *d = get_intc_desc(irq); | 43 | struct intc_desc_int *d = get_intc_desc(irq); |
| 42 | unsigned long handle = (unsigned long)get_irq_chip_data(irq); | 44 | unsigned long handle = (unsigned long)irq_data_get_irq_chip_data(data); |
| 43 | unsigned long addr; | 45 | unsigned long addr; |
| 44 | unsigned int cpu; | 46 | unsigned int cpu; |
| 45 | 47 | ||
| @@ -47,7 +49,7 @@ static void intc_disable(unsigned int irq) | |||
| 47 | 49 | ||
| 48 | for (cpu = 0; cpu < SMP_NR(d, _INTC_ADDR_D(handle)); cpu++) { | 50 | for (cpu = 0; cpu < SMP_NR(d, _INTC_ADDR_D(handle)); cpu++) { |
| 49 | #ifdef CONFIG_SMP | 51 | #ifdef CONFIG_SMP |
| 50 | if (!cpumask_test_cpu(cpu, irq_to_desc(irq)->affinity)) | 52 | if (!cpumask_test_cpu(cpu, data->affinity)) |
| 51 | continue; | 53 | continue; |
| 52 | #endif | 54 | #endif |
| 53 | addr = INTC_REG(d, _INTC_ADDR_D(handle), cpu); | 55 | addr = INTC_REG(d, _INTC_ADDR_D(handle), cpu); |
| @@ -56,7 +58,7 @@ static void intc_disable(unsigned int irq) | |||
| 56 | } | 58 | } |
| 57 | } | 59 | } |
| 58 | 60 | ||
| 59 | static int intc_set_wake(unsigned int irq, unsigned int on) | 61 | static int intc_set_wake(struct irq_data *data, unsigned int on) |
| 60 | { | 62 | { |
| 61 | return 0; /* allow wakeup, but setup hardware in intc_suspend() */ | 63 | return 0; /* allow wakeup, but setup hardware in intc_suspend() */ |
| 62 | } | 64 | } |
| @@ -67,24 +69,27 @@ static int intc_set_wake(unsigned int irq, unsigned int on) | |||
| 67 | * additional locking here at the intc desc level. The affinity mask is | 69 | * additional locking here at the intc desc level. The affinity mask is |
| 68 | * later tested in the enable/disable paths. | 70 | * later tested in the enable/disable paths. |
| 69 | */ | 71 | */ |
| 70 | static int intc_set_affinity(unsigned int irq, const struct cpumask *cpumask) | 72 | static int intc_set_affinity(struct irq_data *data, |
| 73 | const struct cpumask *cpumask, | ||
| 74 | bool force) | ||
| 71 | { | 75 | { |
| 72 | if (!cpumask_intersects(cpumask, cpu_online_mask)) | 76 | if (!cpumask_intersects(cpumask, cpu_online_mask)) |
| 73 | return -1; | 77 | return -1; |
| 74 | 78 | ||
| 75 | cpumask_copy(irq_to_desc(irq)->affinity, cpumask); | 79 | cpumask_copy(data->affinity, cpumask); |
| 76 | 80 | ||
| 77 | return 0; | 81 | return 0; |
| 78 | } | 82 | } |
| 79 | #endif | 83 | #endif |
| 80 | 84 | ||
| 81 | static void intc_mask_ack(unsigned int irq) | 85 | static void intc_mask_ack(struct irq_data *data) |
| 82 | { | 86 | { |
| 87 | unsigned int irq = data->irq; | ||
| 83 | struct intc_desc_int *d = get_intc_desc(irq); | 88 | struct intc_desc_int *d = get_intc_desc(irq); |
| 84 | unsigned long handle = intc_get_ack_handle(irq); | 89 | unsigned long handle = intc_get_ack_handle(irq); |
| 85 | unsigned long addr; | 90 | unsigned long addr; |
| 86 | 91 | ||
| 87 | intc_disable(irq); | 92 | intc_disable(data); |
| 88 | 93 | ||
| 89 | /* read register and write zero only to the associated bit */ | 94 | /* read register and write zero only to the associated bit */ |
| 90 | if (handle) { | 95 | if (handle) { |
| @@ -144,6 +149,7 @@ static struct intc_handle_int *intc_find_irq(struct intc_handle_int *hp, | |||
| 144 | int intc_set_priority(unsigned int irq, unsigned int prio) | 149 | int intc_set_priority(unsigned int irq, unsigned int prio) |
| 145 | { | 150 | { |
| 146 | struct intc_desc_int *d = get_intc_desc(irq); | 151 | struct intc_desc_int *d = get_intc_desc(irq); |
| 152 | struct irq_data *data = irq_get_irq_data(irq); | ||
| 147 | struct intc_handle_int *ihp; | 153 | struct intc_handle_int *ihp; |
| 148 | 154 | ||
| 149 | if (!intc_get_prio_level(irq) || prio <= 1) | 155 | if (!intc_get_prio_level(irq) || prio <= 1) |
| @@ -162,7 +168,7 @@ int intc_set_priority(unsigned int irq, unsigned int prio) | |||
| 162 | * priority level will be set during next enable() | 168 | * priority level will be set during next enable() |
| 163 | */ | 169 | */ |
| 164 | if (_INTC_FN(ihp->handle) != REG_FN_ERR) | 170 | if (_INTC_FN(ihp->handle) != REG_FN_ERR) |
| 165 | _intc_enable(irq, ihp->handle); | 171 | _intc_enable(data, ihp->handle); |
| 166 | } | 172 | } |
| 167 | return 0; | 173 | return 0; |
| 168 | } | 174 | } |
| @@ -181,8 +187,9 @@ static unsigned char intc_irq_sense_table[IRQ_TYPE_SENSE_MASK + 1] = { | |||
| 181 | #endif | 187 | #endif |
| 182 | }; | 188 | }; |
| 183 | 189 | ||
| 184 | static int intc_set_type(unsigned int irq, unsigned int type) | 190 | static int intc_set_type(struct irq_data *data, unsigned int type) |
| 185 | { | 191 | { |
| 192 | unsigned int irq = data->irq; | ||
| 186 | struct intc_desc_int *d = get_intc_desc(irq); | 193 | struct intc_desc_int *d = get_intc_desc(irq); |
| 187 | unsigned char value = intc_irq_sense_table[type & IRQ_TYPE_SENSE_MASK]; | 194 | unsigned char value = intc_irq_sense_table[type & IRQ_TYPE_SENSE_MASK]; |
| 188 | struct intc_handle_int *ihp; | 195 | struct intc_handle_int *ihp; |
| @@ -201,15 +208,15 @@ static int intc_set_type(unsigned int irq, unsigned int type) | |||
| 201 | } | 208 | } |
| 202 | 209 | ||
| 203 | struct irq_chip intc_irq_chip = { | 210 | struct irq_chip intc_irq_chip = { |
| 204 | .mask = intc_disable, | 211 | .irq_mask = intc_disable, |
| 205 | .unmask = intc_enable, | 212 | .irq_unmask = intc_enable, |
| 206 | .mask_ack = intc_mask_ack, | 213 | .irq_mask_ack = intc_mask_ack, |
| 207 | .enable = intc_enable, | 214 | .irq_enable = intc_enable, |
| 208 | .disable = intc_disable, | 215 | .irq_disable = intc_disable, |
| 209 | .shutdown = intc_disable, | 216 | .irq_shutdown = intc_disable, |
| 210 | .set_type = intc_set_type, | 217 | .irq_set_type = intc_set_type, |
| 211 | .set_wake = intc_set_wake, | 218 | .irq_set_wake = intc_set_wake, |
| 212 | #ifdef CONFIG_SMP | 219 | #ifdef CONFIG_SMP |
| 213 | .set_affinity = intc_set_affinity, | 220 | .irq_set_affinity = intc_set_affinity, |
| 214 | #endif | 221 | #endif |
| 215 | }; | 222 | }; |
diff --git a/drivers/sh/intc/core.c b/drivers/sh/intc/core.c index 0801089828e7..338fad2a3fa4 100644 --- a/drivers/sh/intc/core.c +++ b/drivers/sh/intc/core.c | |||
| @@ -71,6 +71,7 @@ static void __init intc_register_irq(struct intc_desc *desc, | |||
| 71 | unsigned int irq) | 71 | unsigned int irq) |
| 72 | { | 72 | { |
| 73 | struct intc_handle_int *hp; | 73 | struct intc_handle_int *hp; |
| 74 | struct irq_data *irq_data; | ||
| 74 | unsigned int data[2], primary; | 75 | unsigned int data[2], primary; |
| 75 | unsigned long flags; | 76 | unsigned long flags; |
| 76 | 77 | ||
| @@ -111,6 +112,8 @@ static void __init intc_register_irq(struct intc_desc *desc, | |||
| 111 | 112 | ||
| 112 | BUG_ON(!data[primary]); /* must have primary masking method */ | 113 | BUG_ON(!data[primary]); /* must have primary masking method */ |
| 113 | 114 | ||
| 115 | irq_data = irq_get_irq_data(irq); | ||
| 116 | |||
| 114 | disable_irq_nosync(irq); | 117 | disable_irq_nosync(irq); |
| 115 | set_irq_chip_and_handler_name(irq, &d->chip, | 118 | set_irq_chip_and_handler_name(irq, &d->chip, |
| 116 | handle_level_irq, "level"); | 119 | handle_level_irq, "level"); |
| @@ -123,7 +126,7 @@ static void __init intc_register_irq(struct intc_desc *desc, | |||
| 123 | 126 | ||
| 124 | /* enable secondary masking method if present */ | 127 | /* enable secondary masking method if present */ |
| 125 | if (data[!primary]) | 128 | if (data[!primary]) |
| 126 | _intc_enable(irq, data[!primary]); | 129 | _intc_enable(irq_data, data[!primary]); |
| 127 | 130 | ||
| 128 | /* add irq to d->prio list if priority is available */ | 131 | /* add irq to d->prio list if priority is available */ |
| 129 | if (data[1]) { | 132 | if (data[1]) { |
| @@ -151,7 +154,7 @@ static void __init intc_register_irq(struct intc_desc *desc, | |||
| 151 | } | 154 | } |
| 152 | 155 | ||
| 153 | /* irq should be disabled by default */ | 156 | /* irq should be disabled by default */ |
| 154 | d->chip.mask(irq); | 157 | d->chip.irq_mask(irq_data); |
| 155 | 158 | ||
| 156 | intc_set_ack_handle(irq, desc, d, enum_id); | 159 | intc_set_ack_handle(irq, desc, d, enum_id); |
| 157 | intc_set_dist_handle(irq, desc, d, enum_id); | 160 | intc_set_dist_handle(irq, desc, d, enum_id); |
| @@ -284,7 +287,7 @@ int __init register_intc_controller(struct intc_desc *desc) | |||
| 284 | for (i = 0; i < hw->nr_ack_regs; i++) | 287 | for (i = 0; i < hw->nr_ack_regs; i++) |
| 285 | k += save_reg(d, k, hw->ack_regs[i].set_reg, 0); | 288 | k += save_reg(d, k, hw->ack_regs[i].set_reg, 0); |
| 286 | else | 289 | else |
| 287 | d->chip.mask_ack = d->chip.disable; | 290 | d->chip.irq_mask_ack = d->chip.irq_disable; |
| 288 | 291 | ||
| 289 | /* disable bits matching force_disable before registering irqs */ | 292 | /* disable bits matching force_disable before registering irqs */ |
| 290 | if (desc->force_disable) | 293 | if (desc->force_disable) |
| @@ -387,7 +390,9 @@ static SYSDEV_ATTR(name, S_IRUGO, show_intc_name, NULL); | |||
| 387 | static int intc_suspend(struct sys_device *dev, pm_message_t state) | 390 | static int intc_suspend(struct sys_device *dev, pm_message_t state) |
| 388 | { | 391 | { |
| 389 | struct intc_desc_int *d; | 392 | struct intc_desc_int *d; |
| 393 | struct irq_data *data; | ||
| 390 | struct irq_desc *desc; | 394 | struct irq_desc *desc; |
| 395 | struct irq_chip *chip; | ||
| 391 | int irq; | 396 | int irq; |
| 392 | 397 | ||
| 393 | /* get intc controller associated with this sysdev */ | 398 | /* get intc controller associated with this sysdev */ |
| @@ -398,17 +403,24 @@ static int intc_suspend(struct sys_device *dev, pm_message_t state) | |||
| 398 | if (d->state.event != PM_EVENT_FREEZE) | 403 | if (d->state.event != PM_EVENT_FREEZE) |
| 399 | break; | 404 | break; |
| 400 | 405 | ||
| 401 | for_each_irq_desc(irq, desc) { | 406 | for_each_irq_nr(irq) { |
| 407 | desc = irq_to_desc(irq); | ||
| 408 | if (!desc) | ||
| 409 | continue; | ||
| 410 | |||
| 411 | data = irq_get_irq_data(irq); | ||
| 412 | chip = irq_data_get_irq_chip(data); | ||
| 413 | |||
| 402 | /* | 414 | /* |
| 403 | * This will catch the redirect and VIRQ cases | 415 | * This will catch the redirect and VIRQ cases |
| 404 | * due to the dummy_irq_chip being inserted. | 416 | * due to the dummy_irq_chip being inserted. |
| 405 | */ | 417 | */ |
| 406 | if (desc->chip != &d->chip) | 418 | if (chip != &d->chip) |
| 407 | continue; | 419 | continue; |
| 408 | if (desc->status & IRQ_DISABLED) | 420 | if (desc->status & IRQ_DISABLED) |
| 409 | desc->chip->disable(irq); | 421 | chip->irq_disable(data); |
| 410 | else | 422 | else |
| 411 | desc->chip->enable(irq); | 423 | chip->irq_enable(data); |
| 412 | } | 424 | } |
| 413 | break; | 425 | break; |
| 414 | case PM_EVENT_FREEZE: | 426 | case PM_EVENT_FREEZE: |
| @@ -416,11 +428,18 @@ static int intc_suspend(struct sys_device *dev, pm_message_t state) | |||
| 416 | break; | 428 | break; |
| 417 | case PM_EVENT_SUSPEND: | 429 | case PM_EVENT_SUSPEND: |
| 418 | /* enable wakeup irqs belonging to this intc controller */ | 430 | /* enable wakeup irqs belonging to this intc controller */ |
| 419 | for_each_irq_desc(irq, desc) { | 431 | for_each_irq_nr(irq) { |
| 420 | if (desc->chip != &d->chip) | 432 | desc = irq_to_desc(irq); |
| 433 | if (!desc) | ||
| 434 | continue; | ||
| 435 | |||
| 436 | data = irq_get_irq_data(irq); | ||
| 437 | chip = irq_data_get_irq_chip(data); | ||
| 438 | |||
| 439 | if (chip != &d->chip) | ||
| 421 | continue; | 440 | continue; |
| 422 | if ((desc->status & IRQ_WAKEUP)) | 441 | if ((desc->status & IRQ_WAKEUP)) |
| 423 | desc->chip->enable(irq); | 442 | chip->irq_enable(data); |
| 424 | } | 443 | } |
| 425 | break; | 444 | break; |
| 426 | } | 445 | } |
diff --git a/drivers/sh/intc/internals.h b/drivers/sh/intc/internals.h index d49482c623fa..0cf8260971d4 100644 --- a/drivers/sh/intc/internals.h +++ b/drivers/sh/intc/internals.h | |||
| @@ -152,7 +152,7 @@ intc_set_dist_handle(unsigned int irq, struct intc_desc *desc, | |||
| 152 | 152 | ||
| 153 | /* chip.c */ | 153 | /* chip.c */ |
| 154 | extern struct irq_chip intc_irq_chip; | 154 | extern struct irq_chip intc_irq_chip; |
| 155 | void _intc_enable(unsigned int irq, unsigned long handle); | 155 | void _intc_enable(struct irq_data *data, unsigned long handle); |
| 156 | 156 | ||
| 157 | /* core.c */ | 157 | /* core.c */ |
| 158 | extern struct list_head intc_list; | 158 | extern struct list_head intc_list; |
diff --git a/drivers/sh/intc/virq.c b/drivers/sh/intc/virq.c index 643dfd4d2057..e5bf5d3c698e 100644 --- a/drivers/sh/intc/virq.c +++ b/drivers/sh/intc/virq.c | |||
| @@ -83,11 +83,11 @@ EXPORT_SYMBOL_GPL(intc_irq_lookup); | |||
| 83 | static int add_virq_to_pirq(unsigned int irq, unsigned int virq) | 83 | static int add_virq_to_pirq(unsigned int irq, unsigned int virq) |
| 84 | { | 84 | { |
| 85 | struct intc_virq_list **last, *entry; | 85 | struct intc_virq_list **last, *entry; |
| 86 | struct irq_desc *desc = irq_to_desc(irq); | 86 | struct irq_data *data = irq_get_irq_data(irq); |
| 87 | 87 | ||
| 88 | /* scan for duplicates */ | 88 | /* scan for duplicates */ |
| 89 | last = (struct intc_virq_list **)&desc->handler_data; | 89 | last = (struct intc_virq_list **)&data->handler_data; |
| 90 | for_each_virq(entry, desc->handler_data) { | 90 | for_each_virq(entry, data->handler_data) { |
| 91 | if (entry->irq == virq) | 91 | if (entry->irq == virq) |
| 92 | return 0; | 92 | return 0; |
| 93 | last = &entry->next; | 93 | last = &entry->next; |
| @@ -108,10 +108,12 @@ static int add_virq_to_pirq(unsigned int irq, unsigned int virq) | |||
| 108 | 108 | ||
| 109 | static void intc_virq_handler(unsigned int irq, struct irq_desc *desc) | 109 | static void intc_virq_handler(unsigned int irq, struct irq_desc *desc) |
| 110 | { | 110 | { |
| 111 | struct intc_virq_list *entry, *vlist = get_irq_data(irq); | 111 | struct irq_data *data = irq_get_irq_data(irq); |
| 112 | struct irq_chip *chip = irq_data_get_irq_chip(data); | ||
| 113 | struct intc_virq_list *entry, *vlist = irq_data_get_irq_data(data); | ||
| 112 | struct intc_desc_int *d = get_intc_desc(irq); | 114 | struct intc_desc_int *d = get_intc_desc(irq); |
| 113 | 115 | ||
| 114 | desc->chip->mask_ack(irq); | 116 | chip->irq_mask_ack(data); |
| 115 | 117 | ||
| 116 | for_each_virq(entry, vlist) { | 118 | for_each_virq(entry, vlist) { |
| 117 | unsigned long addr, handle; | 119 | unsigned long addr, handle; |
| @@ -123,7 +125,7 @@ static void intc_virq_handler(unsigned int irq, struct irq_desc *desc) | |||
| 123 | generic_handle_irq(entry->irq); | 125 | generic_handle_irq(entry->irq); |
| 124 | } | 126 | } |
| 125 | 127 | ||
| 126 | desc->chip->unmask(irq); | 128 | chip->irq_unmask(data); |
| 127 | } | 129 | } |
| 128 | 130 | ||
| 129 | static unsigned long __init intc_subgroup_data(struct intc_subgroup *subgroup, | 131 | static unsigned long __init intc_subgroup_data(struct intc_subgroup *subgroup, |
