| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * Copyright (C) 2023-2024, Ventana Micro Systems Inc |
| * Author: Sunil V L <sunilvl@ventanamicro.com> |
| */ |
| |
| #include <linux/acpi.h> |
| #include <linux/sort.h> |
| #include <linux/irq.h> |
| |
| #include "init.h" |
| |
| struct riscv_ext_intc_list { |
| acpi_handle handle; |
| u32 gsi_base; |
| u32 nr_irqs; |
| u32 nr_idcs; |
| u32 id; |
| u32 type; |
| struct list_head list; |
| }; |
| |
| struct acpi_irq_dep_ctx { |
| int rc; |
| unsigned int index; |
| acpi_handle handle; |
| }; |
| |
| LIST_HEAD(ext_intc_list); |
| |
| static int irqchip_cmp_func(const void *in0, const void *in1) |
| { |
| struct acpi_probe_entry *elem0 = (struct acpi_probe_entry *)in0; |
| struct acpi_probe_entry *elem1 = (struct acpi_probe_entry *)in1; |
| |
| return (elem0->type > elem1->type) - (elem0->type < elem1->type); |
| } |
| |
| /* |
| * On RISC-V, RINTC structures in MADT should be probed before any other |
| * interrupt controller structures and IMSIC before APLIC. The interrupt |
| * controller subtypes in MADT of ACPI spec for RISC-V are defined in |
| * the incremental order like RINTC(24)->IMSIC(25)->APLIC(26)->PLIC(27). |
| * Hence, simply sorting the subtypes in incremental order will |
| * establish the required order. |
| */ |
| void arch_sort_irqchip_probe(struct acpi_probe_entry *ap_head, int nr) |
| { |
| struct acpi_probe_entry *ape = ap_head; |
| |
| if (nr == 1 || !ACPI_COMPARE_NAMESEG(ACPI_SIG_MADT, ape->id)) |
| return; |
| sort(ape, nr, sizeof(*ape), irqchip_cmp_func, NULL); |
| } |
| |
| static acpi_status riscv_acpi_update_gsi_handle(u32 gsi_base, acpi_handle handle) |
| { |
| struct riscv_ext_intc_list *ext_intc_element; |
| struct list_head *i, *tmp; |
| |
| list_for_each_safe(i, tmp, &ext_intc_list) { |
| ext_intc_element = list_entry(i, struct riscv_ext_intc_list, list); |
| if (gsi_base == ext_intc_element->gsi_base) { |
| ext_intc_element->handle = handle; |
| return AE_OK; |
| } |
| } |
| |
| return AE_NOT_FOUND; |
| } |
| |
| int riscv_acpi_get_gsi_info(struct fwnode_handle *fwnode, u32 *gsi_base, |
| u32 *id, u32 *nr_irqs, u32 *nr_idcs) |
| { |
| struct riscv_ext_intc_list *ext_intc_element; |
| struct list_head *i; |
| |
| list_for_each(i, &ext_intc_list) { |
| ext_intc_element = list_entry(i, struct riscv_ext_intc_list, list); |
| if (ext_intc_element->handle == ACPI_HANDLE_FWNODE(fwnode)) { |
| *gsi_base = ext_intc_element->gsi_base; |
| *id = ext_intc_element->id; |
| *nr_irqs = ext_intc_element->nr_irqs; |
| if (nr_idcs) |
| *nr_idcs = ext_intc_element->nr_idcs; |
| |
| return 0; |
| } |
| } |
| |
| return -ENODEV; |
| } |
| |
| struct fwnode_handle *riscv_acpi_get_gsi_domain_id(u32 gsi) |
| { |
| struct riscv_ext_intc_list *ext_intc_element; |
| struct acpi_device *adev; |
| struct list_head *i; |
| |
| list_for_each(i, &ext_intc_list) { |
| ext_intc_element = list_entry(i, struct riscv_ext_intc_list, list); |
| if (gsi >= ext_intc_element->gsi_base && |
| gsi < (ext_intc_element->gsi_base + ext_intc_element->nr_irqs)) { |
| adev = acpi_fetch_acpi_dev(ext_intc_element->handle); |
| if (!adev) |
| return NULL; |
| |
| return acpi_fwnode_handle(adev); |
| } |
| } |
| |
| return NULL; |
| } |
| |
| static int __init riscv_acpi_register_ext_intc(u32 gsi_base, u32 nr_irqs, u32 nr_idcs, |
| u32 id, u32 type) |
| { |
| struct riscv_ext_intc_list *ext_intc_element; |
| |
| ext_intc_element = kzalloc(sizeof(*ext_intc_element), GFP_KERNEL); |
| if (!ext_intc_element) |
| return -ENOMEM; |
| |
| ext_intc_element->gsi_base = gsi_base; |
| ext_intc_element->nr_irqs = nr_irqs; |
| ext_intc_element->nr_idcs = nr_idcs; |
| ext_intc_element->id = id; |
| list_add_tail(&ext_intc_element->list, &ext_intc_list); |
| return 0; |
| } |
| |
| static acpi_status __init riscv_acpi_create_gsi_map(acpi_handle handle, u32 level, |
| void *context, void **return_value) |
| { |
| acpi_status status; |
| u64 gbase; |
| |
| if (!acpi_has_method(handle, "_GSB")) { |
| acpi_handle_err(handle, "_GSB method not found\n"); |
| return AE_ERROR; |
| } |
| |
| status = acpi_evaluate_integer(handle, "_GSB", NULL, &gbase); |
| if (ACPI_FAILURE(status)) { |
| acpi_handle_err(handle, "failed to evaluate _GSB method\n"); |
| return status; |
| } |
| |
| status = riscv_acpi_update_gsi_handle((u32)gbase, handle); |
| if (ACPI_FAILURE(status)) { |
| acpi_handle_err(handle, "failed to find the GSI mapping entry\n"); |
| return status; |
| } |
| |
| return AE_OK; |
| } |
| |
| static int __init riscv_acpi_aplic_parse_madt(union acpi_subtable_headers *header, |
| const unsigned long end) |
| { |
| struct acpi_madt_aplic *aplic = (struct acpi_madt_aplic *)header; |
| |
| return riscv_acpi_register_ext_intc(aplic->gsi_base, aplic->num_sources, aplic->num_idcs, |
| aplic->id, ACPI_RISCV_IRQCHIP_APLIC); |
| } |
| |
| static int __init riscv_acpi_plic_parse_madt(union acpi_subtable_headers *header, |
| const unsigned long end) |
| { |
| struct acpi_madt_plic *plic = (struct acpi_madt_plic *)header; |
| |
| return riscv_acpi_register_ext_intc(plic->gsi_base, plic->num_irqs, 0, |
| plic->id, ACPI_RISCV_IRQCHIP_PLIC); |
| } |
| |
| void __init riscv_acpi_init_gsi_mapping(void) |
| { |
| /* There can be either PLIC or APLIC */ |
| if (acpi_table_parse_madt(ACPI_MADT_TYPE_PLIC, riscv_acpi_plic_parse_madt, 0) > 0) { |
| acpi_get_devices("RSCV0001", riscv_acpi_create_gsi_map, NULL, NULL); |
| return; |
| } |
| |
| if (acpi_table_parse_madt(ACPI_MADT_TYPE_APLIC, riscv_acpi_aplic_parse_madt, 0) > 0) |
| acpi_get_devices("RSCV0002", riscv_acpi_create_gsi_map, NULL, NULL); |
| } |
| |
| static acpi_handle riscv_acpi_get_gsi_handle(u32 gsi) |
| { |
| struct riscv_ext_intc_list *ext_intc_element; |
| struct list_head *i; |
| |
| list_for_each(i, &ext_intc_list) { |
| ext_intc_element = list_entry(i, struct riscv_ext_intc_list, list); |
| if (gsi >= ext_intc_element->gsi_base && |
| gsi < (ext_intc_element->gsi_base + ext_intc_element->nr_irqs)) |
| return ext_intc_element->handle; |
| } |
| |
| return NULL; |
| } |
| |
| static acpi_status riscv_acpi_irq_get_parent(struct acpi_resource *ares, void *context) |
| { |
| struct acpi_irq_dep_ctx *ctx = context; |
| struct acpi_resource_irq *irq; |
| struct acpi_resource_extended_irq *eirq; |
| |
| switch (ares->type) { |
| case ACPI_RESOURCE_TYPE_IRQ: |
| irq = &ares->data.irq; |
| if (ctx->index >= irq->interrupt_count) { |
| ctx->index -= irq->interrupt_count; |
| return AE_OK; |
| } |
| ctx->handle = riscv_acpi_get_gsi_handle(irq->interrupts[ctx->index]); |
| return AE_CTRL_TERMINATE; |
| case ACPI_RESOURCE_TYPE_EXTENDED_IRQ: |
| eirq = &ares->data.extended_irq; |
| if (eirq->producer_consumer == ACPI_PRODUCER) |
| return AE_OK; |
| |
| if (ctx->index >= eirq->interrupt_count) { |
| ctx->index -= eirq->interrupt_count; |
| return AE_OK; |
| } |
| |
| /* Support GSIs only */ |
| if (eirq->resource_source.string_length) |
| return AE_OK; |
| |
| ctx->handle = riscv_acpi_get_gsi_handle(eirq->interrupts[ctx->index]); |
| return AE_CTRL_TERMINATE; |
| } |
| |
| return AE_OK; |
| } |
| |
| static int riscv_acpi_irq_get_dep(acpi_handle handle, unsigned int index, acpi_handle *gsi_handle) |
| { |
| struct acpi_irq_dep_ctx ctx = {-EINVAL, index, NULL}; |
| |
| if (!gsi_handle) |
| return 0; |
| |
| acpi_walk_resources(handle, METHOD_NAME__CRS, riscv_acpi_irq_get_parent, &ctx); |
| *gsi_handle = ctx.handle; |
| if (*gsi_handle) |
| return 1; |
| |
| return 0; |
| } |
| |
| static u32 riscv_acpi_add_prt_dep(acpi_handle handle) |
| { |
| struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; |
| struct acpi_pci_routing_table *entry; |
| struct acpi_handle_list dep_devices; |
| acpi_handle gsi_handle; |
| acpi_handle link_handle; |
| acpi_status status; |
| u32 count = 0; |
| |
| status = acpi_get_irq_routing_table(handle, &buffer); |
| if (ACPI_FAILURE(status)) { |
| acpi_handle_err(handle, "failed to get IRQ routing table\n"); |
| kfree(buffer.pointer); |
| return 0; |
| } |
| |
| entry = buffer.pointer; |
| while (entry && (entry->length > 0)) { |
| if (entry->source[0]) { |
| acpi_get_handle(handle, entry->source, &link_handle); |
| dep_devices.count = 1; |
| dep_devices.handles = kcalloc(1, sizeof(*dep_devices.handles), GFP_KERNEL); |
| if (!dep_devices.handles) { |
| acpi_handle_err(handle, "failed to allocate memory\n"); |
| continue; |
| } |
| |
| dep_devices.handles[0] = link_handle; |
| count += acpi_scan_add_dep(handle, &dep_devices); |
| } else { |
| gsi_handle = riscv_acpi_get_gsi_handle(entry->source_index); |
| dep_devices.count = 1; |
| dep_devices.handles = kcalloc(1, sizeof(*dep_devices.handles), GFP_KERNEL); |
| if (!dep_devices.handles) { |
| acpi_handle_err(handle, "failed to allocate memory\n"); |
| continue; |
| } |
| |
| dep_devices.handles[0] = gsi_handle; |
| count += acpi_scan_add_dep(handle, &dep_devices); |
| } |
| |
| entry = (struct acpi_pci_routing_table *) |
| ((unsigned long)entry + entry->length); |
| } |
| |
| kfree(buffer.pointer); |
| return count; |
| } |
| |
| static u32 riscv_acpi_add_irq_dep(acpi_handle handle) |
| { |
| struct acpi_handle_list dep_devices; |
| acpi_handle gsi_handle; |
| u32 count = 0; |
| int i; |
| |
| for (i = 0; |
| riscv_acpi_irq_get_dep(handle, i, &gsi_handle); |
| i++) { |
| dep_devices.count = 1; |
| dep_devices.handles = kcalloc(1, sizeof(*dep_devices.handles), GFP_KERNEL); |
| if (!dep_devices.handles) { |
| acpi_handle_err(handle, "failed to allocate memory\n"); |
| continue; |
| } |
| |
| dep_devices.handles[0] = gsi_handle; |
| count += acpi_scan_add_dep(handle, &dep_devices); |
| } |
| |
| return count; |
| } |
| |
| u32 arch_acpi_add_auto_dep(acpi_handle handle) |
| { |
| if (acpi_has_method(handle, "_PRT")) |
| return riscv_acpi_add_prt_dep(handle); |
| |
| return riscv_acpi_add_irq_dep(handle); |
| } |