Lines Matching +full:use +full:- +full:broken +full:- +full:interrupts
1 // SPDX-License-Identifier: GPL-2.0-only
7 * o There is one Interrupt Distributor, which receives interrupts
10 * o There is one CPU Interface per CPU, which sends interrupts sent
11 * by the Distributor, and interrupts generated locally, to the
16 * Note that IRQs 0-31 are special - they are local to each CPU.
18 * registers are banked per-cpu for these sources.
42 #include <linux/irqchip/arm-gic.h>
50 #include "irq-gic-common.h"
59 "GICv3 system registers enabled, broken firmware!\n"); in gic_check_cpu_features()
115 * the logical CPU numbering. Let's use a mapping as returned
140 return raw_cpu_read(*base->percpu_base); in __get_base()
142 return base->common_base; in __get_base()
145 #define gic_data_dist_base(d) __get_base(&(d)->dist_base)
146 #define gic_data_cpu_base(d) __get_base(&(d)->cpu_base)
148 #define gic_data_dist_base(d) ((d)->dist_base.common_base)
149 #define gic_data_cpu_base(d) ((d)->cpu_base.common_base)
177 * Routines to acknowledge, disable and enable interrupts
261 return -EINVAL; in gic_irq_set_irqchip_state()
285 return -EINVAL; in gic_irq_get_irqchip_state()
299 return type != IRQ_TYPE_EDGE_RISING ? -EINVAL : 0; in gic_set_type()
304 return -EINVAL; in gic_set_type()
309 pr_warn("GIC: PPI%ld is secure or misconfigured\n", gicirq - 16); in gic_set_type()
318 /* Only interrupts on the primary GIC can be forwarded to a vcpu. */ in gic_irq_set_vcpu_affinity()
320 return -EINVAL; in gic_irq_set_vcpu_affinity()
370 generic_handle_domain_irq(gic->domain, irqnr); in gic_handle_irq()
391 ret = generic_handle_domain_irq(chip_data->domain, gic_irq); in gic_handle_cascade_irq()
402 if (gic->domain->pm_dev) in gic_irq_print_chip()
403 seq_printf(p, gic->domain->pm_dev->of_node->name); in gic_irq_print_chip()
405 seq_printf(p, "GIC-%d", (int)(gic - &gic_data[0])); in gic_irq_print_chip()
429 pr_crit("GIC CPU mask not found - kernel will fail to boot.\n"); in gic_get_cpumask()
468 unsigned int gic_irqs = gic->gic_irqs; in gic_dist_init()
474 * Set all global interrupts to this CPU only. in gic_dist_init()
504 return -EINVAL; in gic_cpu_init()
533 return -EINVAL; in gic_cpu_if_down()
546 * with interrupts disabled but before powering down the GIC. After calling
547 * this function, no interrupts will be delivered by the GIC, and another
548 * platform-specific wakeup source must be enabled.
559 gic_irqs = gic->gic_irqs; in gic_dist_save()
566 gic->saved_spi_conf[i] = in gic_dist_save()
570 gic->saved_spi_target[i] = in gic_dist_save()
574 gic->saved_spi_enable[i] = in gic_dist_save()
578 gic->saved_spi_active[i] = in gic_dist_save()
584 * idle. Must be called before enabling interrupts. If a level interrupt
586 * handled normally, but any edge interrupts that occurred will not be seen by
587 * the GIC and need to be handled by the platform-specific wakeup source.
598 gic_irqs = gic->gic_irqs; in gic_dist_restore()
607 writel_relaxed(gic->saved_spi_conf[i], in gic_dist_restore()
615 writel_relaxed(gic->saved_spi_target[i], in gic_dist_restore()
621 writel_relaxed(gic->saved_spi_enable[i], in gic_dist_restore()
628 writel_relaxed(gic->saved_spi_active[i], in gic_dist_restore()
651 ptr = raw_cpu_ptr(gic->saved_ppi_enable); in gic_cpu_save()
655 ptr = raw_cpu_ptr(gic->saved_ppi_active); in gic_cpu_save()
659 ptr = raw_cpu_ptr(gic->saved_ppi_conf); in gic_cpu_save()
681 ptr = raw_cpu_ptr(gic->saved_ppi_enable); in gic_cpu_restore()
688 ptr = raw_cpu_ptr(gic->saved_ppi_active); in gic_cpu_restore()
695 ptr = raw_cpu_ptr(gic->saved_ppi_conf); in gic_cpu_restore()
739 gic->saved_ppi_enable = __alloc_percpu(DIV_ROUND_UP(32, 32) * 4, in gic_pm_init()
741 if (WARN_ON(!gic->saved_ppi_enable)) in gic_pm_init()
742 return -ENOMEM; in gic_pm_init()
744 gic->saved_ppi_active = __alloc_percpu(DIV_ROUND_UP(32, 32) * 4, in gic_pm_init()
746 if (WARN_ON(!gic->saved_ppi_active)) in gic_pm_init()
749 gic->saved_ppi_conf = __alloc_percpu(DIV_ROUND_UP(32, 16) * 4, in gic_pm_init()
751 if (WARN_ON(!gic->saved_ppi_conf)) in gic_pm_init()
760 free_percpu(gic->saved_ppi_active); in gic_pm_init()
762 free_percpu(gic->saved_ppi_enable); in gic_pm_init()
764 return -ENOMEM; in gic_pm_init()
784 addr -= offset; in rmw_writeb()
801 return -EINVAL; in gic_set_affinity()
809 return -EINVAL; in gic_set_affinity()
826 /* Only one CPU? let's do a self-IPI... */ in gic_ipi_send_mask()
827 writel_relaxed(2 << 24 | d->hwirq, in gic_ipi_send_mask()
845 writel_relaxed(map << 16 | d->hwirq, gic_data_dist_base(&gic_data[0]) + GIC_DIST_SOFTINT); in gic_ipi_send_mask()
859 .fwnode = gic_data[0].domain->fwnode, in gic_smp_init()
915 * gic_send_sgi - send a SGI directly to given CPU interface number
929 * gic_get_cpu_id - get the CPU interface ID for the specified CPU
934 * or -1 if the CPU number is too large or the interface ID is
942 return -1; in gic_get_cpu_id()
944 if (cpu_bit & (cpu_bit - 1)) in gic_get_cpu_id()
945 return -1; in gic_get_cpu_id()
950 * gic_migrate_target - migrate IRQs to another CPU interface
954 * Migrate all peripheral interrupts with a target matching the current CPU
975 ror_val = (cur_cpu_id - new_cpu_id) & 31; in gic_migrate_target()
983 * Find all the peripheral interrupts targeting the current in gic_migrate_target()
985 * We skip DIST_TARGET 0 to 7 as they are read-only. in gic_migrate_target()
1004 * doesn't use that information anyway. in gic_migrate_target()
1025 * gic_get_sgir_physaddr - get the physical address for the SGI register
1055 struct gic_chip_data *gic = d->host_data; in gic_irq_domain_map()
1065 irq_domain_set_info(d, irq, hw, chip, d->host_data, in gic_irq_domain_map()
1069 irq_domain_set_info(d, irq, hw, chip, d->host_data, in gic_irq_domain_map()
1086 if (fwspec->param_count == 1 && fwspec->param[0] < 16) { in gic_irq_domain_translate()
1087 *hwirq = fwspec->param[0]; in gic_irq_domain_translate()
1092 if (is_of_node(fwspec->fwnode)) { in gic_irq_domain_translate()
1093 if (fwspec->param_count < 3) in gic_irq_domain_translate()
1094 return -EINVAL; in gic_irq_domain_translate()
1096 switch (fwspec->param[0]) { in gic_irq_domain_translate()
1098 *hwirq = fwspec->param[1] + 32; in gic_irq_domain_translate()
1101 *hwirq = fwspec->param[1] + 16; in gic_irq_domain_translate()
1104 return -EINVAL; in gic_irq_domain_translate()
1107 *type = fwspec->param[2] & IRQ_TYPE_SENSE_MASK; in gic_irq_domain_translate()
1109 /* Make it clear that broken DTs are... broken */ in gic_irq_domain_translate()
1115 if (is_fwnode_irqchip(fwspec->fwnode)) { in gic_irq_domain_translate()
1116 if(fwspec->param_count != 2) in gic_irq_domain_translate()
1117 return -EINVAL; in gic_irq_domain_translate()
1119 if (fwspec->param[0] < 16) { in gic_irq_domain_translate()
1121 fwspec->param[0]); in gic_irq_domain_translate()
1122 return -EINVAL; in gic_irq_domain_translate()
1125 *hwirq = fwspec->param[0]; in gic_irq_domain_translate()
1126 *type = fwspec->param[1]; in gic_irq_domain_translate()
1133 return -EINVAL; in gic_irq_domain_translate()
1168 if (IS_ENABLED(CONFIG_GIC_NON_BANKED) && gic->percpu_offset) { in gic_init_bases()
1169 /* Frankein-GIC without banked registers... */ in gic_init_bases()
1172 gic->dist_base.percpu_base = alloc_percpu(void __iomem *); in gic_init_bases()
1173 gic->cpu_base.percpu_base = alloc_percpu(void __iomem *); in gic_init_bases()
1174 if (WARN_ON(!gic->dist_base.percpu_base || in gic_init_bases()
1175 !gic->cpu_base.percpu_base)) { in gic_init_bases()
1176 ret = -ENOMEM; in gic_init_bases()
1183 unsigned long offset = gic->percpu_offset * core_id; in gic_init_bases()
1184 *per_cpu_ptr(gic->dist_base.percpu_base, cpu) = in gic_init_bases()
1185 gic->raw_dist_base + offset; in gic_init_bases()
1186 *per_cpu_ptr(gic->cpu_base.percpu_base, cpu) = in gic_init_bases()
1187 gic->raw_cpu_base + offset; in gic_init_bases()
1193 WARN(gic->percpu_offset, in gic_init_bases()
1195 gic->percpu_offset); in gic_init_bases()
1196 gic->dist_base.common_base = gic->raw_dist_base; in gic_init_bases()
1197 gic->cpu_base.common_base = gic->raw_cpu_base; in gic_init_bases()
1201 * Find out how many interrupts are supported. in gic_init_bases()
1208 gic->gic_irqs = gic_irqs; in gic_init_bases()
1210 gic->domain = irq_domain_create_linear(handle, gic_irqs, in gic_init_bases()
1213 if (WARN_ON(!gic->domain)) { in gic_init_bases()
1214 ret = -ENODEV; in gic_init_bases()
1230 if (IS_ENABLED(CONFIG_GIC_NON_BANKED) && gic->percpu_offset) { in gic_init_bases()
1231 free_percpu(gic->dist_base.percpu_base); in gic_init_bases()
1232 free_percpu(gic->cpu_base.percpu_base); in gic_init_bases()
1243 if (WARN_ON(!gic || gic->domain)) in __gic_init_bases()
1244 return -EINVAL; in __gic_init_bases()
1272 if (gic->raw_dist_base) in gic_teardown()
1273 iounmap(gic->raw_dist_base); in gic_teardown()
1274 if (gic->raw_cpu_base) in gic_teardown()
1275 iounmap(gic->raw_cpu_base); in gic_teardown()
1318 pr_warn("GIC: GICv2 at %pa, but range is too small (broken DT?), assuming 8kB\n", in gic_check_eoimode()
1337 cpuif_res.end = cpuif_res.start + SZ_128K -1; in gic_check_eoimode()
1353 * contiguous region, which allows us to use GICC_DIR in gic_check_eoimode()
1368 * The EMEV2 class of machines has a broken interconnect, and in gic_enable_rmw_access()
1382 .desc = "broken byte access",
1392 return -EINVAL; in gic_of_setup()
1394 gic->raw_dist_base = of_iomap(node, 0); in gic_of_setup()
1395 if (WARN(!gic->raw_dist_base, "unable to map gic dist registers\n")) in gic_of_setup()
1398 gic->raw_cpu_base = of_iomap(node, 1); in gic_of_setup()
1399 if (WARN(!gic->raw_cpu_base, "unable to map gic cpu registers\n")) in gic_of_setup()
1402 if (of_property_read_u32(node, "cpu-offset", &gic->percpu_offset)) in gic_of_setup()
1403 gic->percpu_offset = 0; in gic_of_setup()
1412 return -ENOMEM; in gic_of_setup()
1419 if (!dev || !dev->of_node || !gic || !irq) in gic_of_init_child()
1420 return -EINVAL; in gic_of_init_child()
1424 return -ENOMEM; in gic_of_init_child()
1426 ret = gic_of_setup(*gic, dev->of_node); in gic_of_init_child()
1430 ret = gic_init_bases(*gic, &dev->of_node->fwnode); in gic_of_init_child()
1436 irq_domain_set_pm_device((*gic)->domain, dev); in gic_of_init_child()
1473 return -ENODEV; in gic_of_init()
1476 return -EINVAL; in gic_of_init()
1488 if (gic_cnt == 0 && !gic_check_eoimode(node, &gic->raw_cpu_base)) in gic_of_init()
1491 ret = __gic_init_bases(gic, &node->fwnode); in gic_of_init()
1508 gicv2m_init(&node->fwnode, gic_data[gic_cnt].domain); in gic_of_init()
1513 IRQCHIP_DECLARE(gic_400, "arm,gic-400", gic_of_init);
1514 IRQCHIP_DECLARE(arm11mp_gic, "arm,arm11mp-gic", gic_of_init);
1515 IRQCHIP_DECLARE(arm1176jzf_dc_gic, "arm,arm1176jzf-devchip-gic", gic_of_init);
1516 IRQCHIP_DECLARE(cortex_a15_gic, "arm,cortex-a15-gic", gic_of_init);
1517 IRQCHIP_DECLARE(cortex_a9_gic, "arm,cortex-a9-gic", gic_of_init);
1518 IRQCHIP_DECLARE(cortex_a7_gic, "arm,cortex-a7-gic", gic_of_init);
1519 IRQCHIP_DECLARE(msm_8660_qgic, "qcom,msm-8660-qgic", gic_of_init);
1520 IRQCHIP_DECLARE(msm_qgic2, "qcom,msm-qgic2", gic_of_init);
1544 return -EINVAL; in gic_acpi_parse_madt_cpu()
1547 * There is no support for non-banked GICv1/2 register in ACPI spec. in gic_acpi_parse_madt_cpu()
1550 gic_cpu_base = processor->base_address; in gic_acpi_parse_madt_cpu()
1552 return -EINVAL; in gic_acpi_parse_madt_cpu()
1555 acpi_data.maint_irq = processor->vgic_interrupt; in gic_acpi_parse_madt_cpu()
1556 acpi_data.maint_irq_mode = (processor->flags & ACPI_MADT_VGIC_IRQ_MODE) ? in gic_acpi_parse_madt_cpu()
1558 acpi_data.vctrl_base = processor->gich_base_address; in gic_acpi_parse_madt_cpu()
1559 acpi_data.vcpu_base = processor->gicv_base_address; in gic_acpi_parse_madt_cpu()
1584 return (dist->version == ape->driver_data && in gic_validate_dist()
1585 (dist->version != ACPI_MADT_GIC_VERSION_NONE || in gic_validate_dist()
1605 vctrl_res->flags = IORESOURCE_MEM; in gic_acpi_setup_kvm_info()
1606 vctrl_res->start = acpi_data.vctrl_base; in gic_acpi_setup_kvm_info()
1607 vctrl_res->end = vctrl_res->start + ACPI_GICV2_VCTRL_MEM_SIZE - 1; in gic_acpi_setup_kvm_info()
1612 vcpu_res->flags = IORESOURCE_MEM; in gic_acpi_setup_kvm_info()
1613 vcpu_res->start = acpi_data.vcpu_base; in gic_acpi_setup_kvm_info()
1614 vcpu_res->end = vcpu_res->start + ACPI_GICV2_VCPU_MEM_SIZE - 1; in gic_acpi_setup_kvm_info()
1646 return -EINVAL; in gic_v2_acpi_init()
1649 gic->raw_cpu_base = ioremap(acpi_data.cpu_phys_base, ACPI_GIC_CPU_IF_MEM_SIZE); in gic_v2_acpi_init()
1650 if (!gic->raw_cpu_base) { in gic_v2_acpi_init()
1652 return -ENOMEM; in gic_v2_acpi_init()
1656 gic->raw_dist_base = ioremap(dist->base_address, in gic_v2_acpi_init()
1658 if (!gic->raw_dist_base) { in gic_v2_acpi_init()
1661 return -ENOMEM; in gic_v2_acpi_init()
1673 * Initialize GIC instance zero (no multi-GIC support). in gic_v2_acpi_init()
1675 gsi_domain_handle = irq_domain_alloc_fwnode(&dist->base_address); in gic_v2_acpi_init()
1679 return -ENOMEM; in gic_v2_acpi_init()