diff options
author | Thomas Gleixner <tglx@linutronix.de> | 2017-11-14 05:23:05 -0500 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2017-11-14 05:23:05 -0500 |
commit | 41cc30412d6692c25506bf2d4e65c4364c70c10a (patch) | |
tree | d9adeb9c94a54998002d46aeebb99f86f1232016 | |
parent | b29c6ef7bb1257853c1e31616d84f55e561cf631 (diff) | |
parent | 29f411399aaa2e53882858e01d21981f3c301e2a (diff) |
Merge tag 'irqchip-4.15-4' of git://git.kernel.org/pub/scm/linux/kernel/git/maz/arm-platforms into irq/urgent
Pull irqchip updates for 4.15, take #4 from Marc Zyngier
- A core irq fix for legacy cases where the irq trigger is not reported
by firmware
- A couple of GICv3/4 fixes (Kconfig, of-node refcount, error handling)
- Trivial pr_err fixes
-rw-r--r-- | drivers/irqchip/Kconfig | 7 | ||||
-rw-r--r-- | drivers/irqchip/Makefile | 3 | ||||
-rw-r--r-- | drivers/irqchip/irq-gic-v3.c | 11 | ||||
-rw-r--r-- | drivers/irqchip/irq-gic-v4.c | 7 | ||||
-rw-r--r-- | drivers/irqchip/irq-s3c24xx.c | 4 | ||||
-rw-r--r-- | include/linux/irq.h | 11 | ||||
-rw-r--r-- | include/linux/irqchip/arm-gic-v4.h | 1 | ||||
-rw-r--r-- | kernel/irq/manage.c | 13 |
8 files changed, 47 insertions, 10 deletions
diff --git a/drivers/irqchip/Kconfig b/drivers/irqchip/Kconfig index 53380bd72ea4..c70476b34a53 100644 --- a/drivers/irqchip/Kconfig +++ b/drivers/irqchip/Kconfig | |||
@@ -41,8 +41,15 @@ config ARM_GIC_V3 | |||
41 | 41 | ||
42 | config ARM_GIC_V3_ITS | 42 | config ARM_GIC_V3_ITS |
43 | bool | 43 | bool |
44 | select GENERIC_MSI_IRQ_DOMAIN | ||
45 | default ARM_GIC_V3 | ||
46 | |||
47 | config ARM_GIC_V3_ITS_PCI | ||
48 | bool | ||
49 | depends on ARM_GIC_V3_ITS | ||
44 | depends on PCI | 50 | depends on PCI |
45 | depends on PCI_MSI | 51 | depends on PCI_MSI |
52 | default ARM_GIC_V3_ITS | ||
46 | 53 | ||
47 | config ARM_NVIC | 54 | config ARM_NVIC |
48 | bool | 55 | bool |
diff --git a/drivers/irqchip/Makefile b/drivers/irqchip/Makefile index dae7282bfdef..d2df34a54d38 100644 --- a/drivers/irqchip/Makefile +++ b/drivers/irqchip/Makefile | |||
@@ -30,7 +30,8 @@ obj-$(CONFIG_ARM_GIC_PM) += irq-gic-pm.o | |||
30 | obj-$(CONFIG_ARCH_REALVIEW) += irq-gic-realview.o | 30 | obj-$(CONFIG_ARCH_REALVIEW) += irq-gic-realview.o |
31 | obj-$(CONFIG_ARM_GIC_V2M) += irq-gic-v2m.o | 31 | obj-$(CONFIG_ARM_GIC_V2M) += irq-gic-v2m.o |
32 | obj-$(CONFIG_ARM_GIC_V3) += irq-gic-v3.o irq-gic-common.o | 32 | obj-$(CONFIG_ARM_GIC_V3) += irq-gic-v3.o irq-gic-common.o |
33 | obj-$(CONFIG_ARM_GIC_V3_ITS) += irq-gic-v3-its.o irq-gic-v3-its-pci-msi.o irq-gic-v3-its-platform-msi.o irq-gic-v4.o | 33 | obj-$(CONFIG_ARM_GIC_V3_ITS) += irq-gic-v3-its.o irq-gic-v3-its-platform-msi.o irq-gic-v4.o |
34 | obj-$(CONFIG_ARM_GIC_V3_ITS_PCI) += irq-gic-v3-its-pci-msi.o | ||
34 | obj-$(CONFIG_PARTITION_PERCPU) += irq-partition-percpu.o | 35 | obj-$(CONFIG_PARTITION_PERCPU) += irq-partition-percpu.o |
35 | obj-$(CONFIG_HISILICON_IRQ_MBIGEN) += irq-mbigen.o | 36 | obj-$(CONFIG_HISILICON_IRQ_MBIGEN) += irq-mbigen.o |
36 | obj-$(CONFIG_ARM_NVIC) += irq-nvic.o | 37 | obj-$(CONFIG_ARM_NVIC) += irq-nvic.o |
diff --git a/drivers/irqchip/irq-gic-v3.c b/drivers/irqchip/irq-gic-v3.c index b54b55597ffb..16fddff38f22 100644 --- a/drivers/irqchip/irq-gic-v3.c +++ b/drivers/irqchip/irq-gic-v3.c | |||
@@ -1103,18 +1103,18 @@ static void __init gic_populate_ppi_partitions(struct device_node *gic_node) | |||
1103 | int nr_parts; | 1103 | int nr_parts; |
1104 | struct partition_affinity *parts; | 1104 | struct partition_affinity *parts; |
1105 | 1105 | ||
1106 | parts_node = of_find_node_by_name(gic_node, "ppi-partitions"); | 1106 | parts_node = of_get_child_by_name(gic_node, "ppi-partitions"); |
1107 | if (!parts_node) | 1107 | if (!parts_node) |
1108 | return; | 1108 | return; |
1109 | 1109 | ||
1110 | nr_parts = of_get_child_count(parts_node); | 1110 | nr_parts = of_get_child_count(parts_node); |
1111 | 1111 | ||
1112 | if (!nr_parts) | 1112 | if (!nr_parts) |
1113 | return; | 1113 | goto out_put_node; |
1114 | 1114 | ||
1115 | parts = kzalloc(sizeof(*parts) * nr_parts, GFP_KERNEL); | 1115 | parts = kzalloc(sizeof(*parts) * nr_parts, GFP_KERNEL); |
1116 | if (WARN_ON(!parts)) | 1116 | if (WARN_ON(!parts)) |
1117 | return; | 1117 | goto out_put_node; |
1118 | 1118 | ||
1119 | for_each_child_of_node(parts_node, child_part) { | 1119 | for_each_child_of_node(parts_node, child_part) { |
1120 | struct partition_affinity *part; | 1120 | struct partition_affinity *part; |
@@ -1181,6 +1181,9 @@ static void __init gic_populate_ppi_partitions(struct device_node *gic_node) | |||
1181 | 1181 | ||
1182 | gic_data.ppi_descs[i] = desc; | 1182 | gic_data.ppi_descs[i] = desc; |
1183 | } | 1183 | } |
1184 | |||
1185 | out_put_node: | ||
1186 | of_node_put(parts_node); | ||
1184 | } | 1187 | } |
1185 | 1188 | ||
1186 | static void __init gic_of_setup_kvm_info(struct device_node *node) | 1189 | static void __init gic_of_setup_kvm_info(struct device_node *node) |
@@ -1521,7 +1524,7 @@ gic_acpi_init(struct acpi_subtable_header *header, const unsigned long end) | |||
1521 | 1524 | ||
1522 | err = gic_validate_dist_version(acpi_data.dist_base); | 1525 | err = gic_validate_dist_version(acpi_data.dist_base); |
1523 | if (err) { | 1526 | if (err) { |
1524 | pr_err("No distributor detected at @%p, giving up", | 1527 | pr_err("No distributor detected at @%p, giving up\n", |
1525 | acpi_data.dist_base); | 1528 | acpi_data.dist_base); |
1526 | goto out_dist_unmap; | 1529 | goto out_dist_unmap; |
1527 | } | 1530 | } |
diff --git a/drivers/irqchip/irq-gic-v4.c b/drivers/irqchip/irq-gic-v4.c index cd0bcc3b7e33..dba9d67cb9c1 100644 --- a/drivers/irqchip/irq-gic-v4.c +++ b/drivers/irqchip/irq-gic-v4.c | |||
@@ -177,6 +177,7 @@ int its_map_vlpi(int irq, struct its_vlpi_map *map) | |||
177 | .map = map, | 177 | .map = map, |
178 | }, | 178 | }, |
179 | }; | 179 | }; |
180 | int ret; | ||
180 | 181 | ||
181 | /* | 182 | /* |
182 | * The host will never see that interrupt firing again, so it | 183 | * The host will never see that interrupt firing again, so it |
@@ -184,7 +185,11 @@ int its_map_vlpi(int irq, struct its_vlpi_map *map) | |||
184 | */ | 185 | */ |
185 | irq_set_status_flags(irq, IRQ_DISABLE_UNLAZY); | 186 | irq_set_status_flags(irq, IRQ_DISABLE_UNLAZY); |
186 | 187 | ||
187 | return irq_set_vcpu_affinity(irq, &info); | 188 | ret = irq_set_vcpu_affinity(irq, &info); |
189 | if (ret) | ||
190 | irq_clear_status_flags(irq, IRQ_DISABLE_UNLAZY); | ||
191 | |||
192 | return ret; | ||
188 | } | 193 | } |
189 | 194 | ||
190 | int its_get_vlpi(int irq, struct its_vlpi_map *map) | 195 | int its_get_vlpi(int irq, struct its_vlpi_map *map) |
diff --git a/drivers/irqchip/irq-s3c24xx.c b/drivers/irqchip/irq-s3c24xx.c index c25ce5af091a..ec0e6a8cdb75 100644 --- a/drivers/irqchip/irq-s3c24xx.c +++ b/drivers/irqchip/irq-s3c24xx.c | |||
@@ -156,7 +156,7 @@ static int s3c_irq_type(struct irq_data *data, unsigned int type) | |||
156 | irq_set_handler(data->irq, handle_level_irq); | 156 | irq_set_handler(data->irq, handle_level_irq); |
157 | break; | 157 | break; |
158 | default: | 158 | default: |
159 | pr_err("No such irq type %d", type); | 159 | pr_err("No such irq type %d\n", type); |
160 | return -EINVAL; | 160 | return -EINVAL; |
161 | } | 161 | } |
162 | 162 | ||
@@ -204,7 +204,7 @@ static int s3c_irqext_type_set(void __iomem *gpcon_reg, | |||
204 | break; | 204 | break; |
205 | 205 | ||
206 | default: | 206 | default: |
207 | pr_err("No such irq type %d", type); | 207 | pr_err("No such irq type %d\n", type); |
208 | return -EINVAL; | 208 | return -EINVAL; |
209 | } | 209 | } |
210 | 210 | ||
diff --git a/include/linux/irq.h b/include/linux/irq.h index b01d06db9101..e140f69163b6 100644 --- a/include/linux/irq.h +++ b/include/linux/irq.h | |||
@@ -211,6 +211,7 @@ struct irq_data { | |||
211 | * IRQD_MANAGED_SHUTDOWN - Interrupt was shutdown due to empty affinity | 211 | * IRQD_MANAGED_SHUTDOWN - Interrupt was shutdown due to empty affinity |
212 | * mask. Applies only to affinity managed irqs. | 212 | * mask. Applies only to affinity managed irqs. |
213 | * IRQD_SINGLE_TARGET - IRQ allows only a single affinity target | 213 | * IRQD_SINGLE_TARGET - IRQ allows only a single affinity target |
214 | * IRQD_DEFAULT_TRIGGER_SET - Expected trigger already been set | ||
214 | */ | 215 | */ |
215 | enum { | 216 | enum { |
216 | IRQD_TRIGGER_MASK = 0xf, | 217 | IRQD_TRIGGER_MASK = 0xf, |
@@ -231,6 +232,7 @@ enum { | |||
231 | IRQD_IRQ_STARTED = (1 << 22), | 232 | IRQD_IRQ_STARTED = (1 << 22), |
232 | IRQD_MANAGED_SHUTDOWN = (1 << 23), | 233 | IRQD_MANAGED_SHUTDOWN = (1 << 23), |
233 | IRQD_SINGLE_TARGET = (1 << 24), | 234 | IRQD_SINGLE_TARGET = (1 << 24), |
235 | IRQD_DEFAULT_TRIGGER_SET = (1 << 25), | ||
234 | }; | 236 | }; |
235 | 237 | ||
236 | #define __irqd_to_state(d) ACCESS_PRIVATE((d)->common, state_use_accessors) | 238 | #define __irqd_to_state(d) ACCESS_PRIVATE((d)->common, state_use_accessors) |
@@ -260,18 +262,25 @@ static inline void irqd_mark_affinity_was_set(struct irq_data *d) | |||
260 | __irqd_to_state(d) |= IRQD_AFFINITY_SET; | 262 | __irqd_to_state(d) |= IRQD_AFFINITY_SET; |
261 | } | 263 | } |
262 | 264 | ||
265 | static inline bool irqd_trigger_type_was_set(struct irq_data *d) | ||
266 | { | ||
267 | return __irqd_to_state(d) & IRQD_DEFAULT_TRIGGER_SET; | ||
268 | } | ||
269 | |||
263 | static inline u32 irqd_get_trigger_type(struct irq_data *d) | 270 | static inline u32 irqd_get_trigger_type(struct irq_data *d) |
264 | { | 271 | { |
265 | return __irqd_to_state(d) & IRQD_TRIGGER_MASK; | 272 | return __irqd_to_state(d) & IRQD_TRIGGER_MASK; |
266 | } | 273 | } |
267 | 274 | ||
268 | /* | 275 | /* |
269 | * Must only be called inside irq_chip.irq_set_type() functions. | 276 | * Must only be called inside irq_chip.irq_set_type() functions or |
277 | * from the DT/ACPI setup code. | ||
270 | */ | 278 | */ |
271 | static inline void irqd_set_trigger_type(struct irq_data *d, u32 type) | 279 | static inline void irqd_set_trigger_type(struct irq_data *d, u32 type) |
272 | { | 280 | { |
273 | __irqd_to_state(d) &= ~IRQD_TRIGGER_MASK; | 281 | __irqd_to_state(d) &= ~IRQD_TRIGGER_MASK; |
274 | __irqd_to_state(d) |= type & IRQD_TRIGGER_MASK; | 282 | __irqd_to_state(d) |= type & IRQD_TRIGGER_MASK; |
283 | __irqd_to_state(d) |= IRQD_DEFAULT_TRIGGER_SET; | ||
275 | } | 284 | } |
276 | 285 | ||
277 | static inline bool irqd_is_level_type(struct irq_data *d) | 286 | static inline bool irqd_is_level_type(struct irq_data *d) |
diff --git a/include/linux/irqchip/arm-gic-v4.h b/include/linux/irqchip/arm-gic-v4.h index 447da8ca2156..fa683ea5c769 100644 --- a/include/linux/irqchip/arm-gic-v4.h +++ b/include/linux/irqchip/arm-gic-v4.h | |||
@@ -109,6 +109,7 @@ int its_get_vlpi(int irq, struct its_vlpi_map *map); | |||
109 | int its_unmap_vlpi(int irq); | 109 | int its_unmap_vlpi(int irq); |
110 | int its_prop_update_vlpi(int irq, u8 config, bool inv); | 110 | int its_prop_update_vlpi(int irq, u8 config, bool inv); |
111 | 111 | ||
112 | struct irq_domain_ops; | ||
112 | int its_init_v4(struct irq_domain *domain, const struct irq_domain_ops *ops); | 113 | int its_init_v4(struct irq_domain *domain, const struct irq_domain_ops *ops); |
113 | 114 | ||
114 | #endif | 115 | #endif |
diff --git a/kernel/irq/manage.c b/kernel/irq/manage.c index 2ff1c0c82fc9..0f922729bab9 100644 --- a/kernel/irq/manage.c +++ b/kernel/irq/manage.c | |||
@@ -1246,7 +1246,18 @@ __setup_irq(unsigned int irq, struct irq_desc *desc, struct irqaction *new) | |||
1246 | * set the trigger type must match. Also all must | 1246 | * set the trigger type must match. Also all must |
1247 | * agree on ONESHOT. | 1247 | * agree on ONESHOT. |
1248 | */ | 1248 | */ |
1249 | unsigned int oldtype = irqd_get_trigger_type(&desc->irq_data); | 1249 | unsigned int oldtype; |
1250 | |||
1251 | /* | ||
1252 | * If nobody did set the configuration before, inherit | ||
1253 | * the one provided by the requester. | ||
1254 | */ | ||
1255 | if (irqd_trigger_type_was_set(&desc->irq_data)) { | ||
1256 | oldtype = irqd_get_trigger_type(&desc->irq_data); | ||
1257 | } else { | ||
1258 | oldtype = new->flags & IRQF_TRIGGER_MASK; | ||
1259 | irqd_set_trigger_type(&desc->irq_data, oldtype); | ||
1260 | } | ||
1250 | 1261 | ||
1251 | if (!((old->flags & new->flags) & IRQF_SHARED) || | 1262 | if (!((old->flags & new->flags) & IRQF_SHARED) || |
1252 | (oldtype != (new->flags & IRQF_TRIGGER_MASK)) || | 1263 | (oldtype != (new->flags & IRQF_TRIGGER_MASK)) || |