Lines Matching +full:parent +full:- +full:interrupt +full:- +full:base
1 // SPDX-License-Identifier: GPL-2.0
3 * Texas Instruments' K3 Interrupt Router irqchip driver
5 * Copyright (C) 2018-2019 Texas Instruments Incorporated - https://www.ti.com/
21 * struct ti_sci_intr_irq_domain - Structure representing a TISCI based
22 * Interrupt Router IRQ domain.
26 * @ti_sci_id: TI-SCI device identifier
27 * @type: Specifies the trigger type supported by this Interrupt Router
48 * ti_sci_intr_irq_domain_translate() - Retrieve hwirq and type from
62 struct ti_sci_intr_irq_domain *intr = domain->host_data; in ti_sci_intr_irq_domain_translate()
64 if (fwspec->param_count != 1) in ti_sci_intr_irq_domain_translate()
65 return -EINVAL; in ti_sci_intr_irq_domain_translate()
67 *hwirq = fwspec->param[0]; in ti_sci_intr_irq_domain_translate()
68 *type = intr->type; in ti_sci_intr_irq_domain_translate()
74 * ti_sci_intr_xlate_irq() - Translate hwirq to parent's hwirq.
75 * @intr: IRQ domain corresponding to Interrupt Router
78 * Return parent irq number if translation is available else -ENOENT.
82 struct device_node *np = dev_of_node(intr->dev); in ti_sci_intr_xlate_irq()
83 u32 base, pbase, size, len; in ti_sci_intr_xlate_irq() local
86 range = of_get_property(np, "ti,interrupt-ranges", &len); in ti_sci_intr_xlate_irq()
90 for (len /= sizeof(*range); len >= 3; len -= 3) { in ti_sci_intr_xlate_irq()
91 base = be32_to_cpu(*range++); in ti_sci_intr_xlate_irq()
95 if (base <= irq && irq < base + size) in ti_sci_intr_xlate_irq()
96 return irq - base + pbase; in ti_sci_intr_xlate_irq()
99 return -ENOENT; in ti_sci_intr_xlate_irq()
103 * ti_sci_intr_irq_domain_free() - Free the specified IRQs from the domain.
111 struct ti_sci_intr_irq_domain *intr = domain->host_data; in ti_sci_intr_irq_domain_free()
116 out_irq = (uintptr_t)data->chip_data; in ti_sci_intr_irq_domain_free()
118 intr->sci->ops.rm_irq_ops.free_irq(intr->sci, in ti_sci_intr_irq_domain_free()
119 intr->ti_sci_id, data->hwirq, in ti_sci_intr_irq_domain_free()
120 intr->ti_sci_id, out_irq); in ti_sci_intr_irq_domain_free()
121 ti_sci_release_resource(intr->out_irqs, out_irq); in ti_sci_intr_irq_domain_free()
127 * ti_sci_intr_alloc_parent_irq() - Allocate parent IRQ
128 * @domain: Pointer to the interrupt router IRQ domain
137 struct ti_sci_intr_irq_domain *intr = domain->host_data; in ti_sci_intr_alloc_parent_irq()
143 out_irq = ti_sci_get_free_resource(intr->out_irqs); in ti_sci_intr_alloc_parent_irq()
145 return -EINVAL; in ti_sci_intr_alloc_parent_irq()
151 parent_node = of_irq_find_parent(dev_of_node(intr->dev)); in ti_sci_intr_alloc_parent_irq()
154 if (of_device_is_compatible(parent_node, "arm,gic-v3")) { in ti_sci_intr_alloc_parent_irq()
155 /* Parent is GIC */ in ti_sci_intr_alloc_parent_irq()
158 fwspec.param[1] = p_hwirq - 32; /* SPI offset */ in ti_sci_intr_alloc_parent_irq()
159 fwspec.param[2] = intr->type; in ti_sci_intr_alloc_parent_irq()
161 /* Parent is Interrupt Router */ in ti_sci_intr_alloc_parent_irq()
170 err = intr->sci->ops.rm_irq_ops.set_irq(intr->sci, in ti_sci_intr_alloc_parent_irq()
171 intr->ti_sci_id, hwirq, in ti_sci_intr_alloc_parent_irq()
172 intr->ti_sci_id, out_irq); in ti_sci_intr_alloc_parent_irq()
181 ti_sci_release_resource(intr->out_irqs, out_irq); in ti_sci_intr_alloc_parent_irq()
186 * ti_sci_intr_irq_domain_alloc() - Allocate Interrupt router IRQs
187 * @domain: Point to the interrupt router IRQ domain
229 struct device *dev = &pdev->dev; in ti_sci_intr_irq_domain_probe()
234 dev_err(dev, "Failed to get IRQ parent node\n"); in ti_sci_intr_irq_domain_probe()
235 return -ENODEV; in ti_sci_intr_irq_domain_probe()
241 dev_err(dev, "Failed to find IRQ parent domain\n"); in ti_sci_intr_irq_domain_probe()
242 return -ENODEV; in ti_sci_intr_irq_domain_probe()
247 return -ENOMEM; in ti_sci_intr_irq_domain_probe()
249 intr->dev = dev; in ti_sci_intr_irq_domain_probe()
250 ret = of_property_read_u32(dev_of_node(dev), "ti,intr-trigger-type", in ti_sci_intr_irq_domain_probe()
251 &intr->type); in ti_sci_intr_irq_domain_probe()
253 dev_err(dev, "missing ti,intr-trigger-type property\n"); in ti_sci_intr_irq_domain_probe()
254 return -EINVAL; in ti_sci_intr_irq_domain_probe()
257 intr->sci = devm_ti_sci_get_by_phandle(dev, "ti,sci"); in ti_sci_intr_irq_domain_probe()
258 if (IS_ERR(intr->sci)) in ti_sci_intr_irq_domain_probe()
259 return dev_err_probe(dev, PTR_ERR(intr->sci), in ti_sci_intr_irq_domain_probe()
262 ret = of_property_read_u32(dev_of_node(dev), "ti,sci-dev-id", in ti_sci_intr_irq_domain_probe()
263 &intr->ti_sci_id); in ti_sci_intr_irq_domain_probe()
265 dev_err(dev, "missing 'ti,sci-dev-id' property\n"); in ti_sci_intr_irq_domain_probe()
266 return -EINVAL; in ti_sci_intr_irq_domain_probe()
269 intr->out_irqs = devm_ti_sci_get_resource(intr->sci, dev, in ti_sci_intr_irq_domain_probe()
270 intr->ti_sci_id, in ti_sci_intr_irq_domain_probe()
272 if (IS_ERR(intr->out_irqs)) { in ti_sci_intr_irq_domain_probe()
274 return PTR_ERR(intr->out_irqs); in ti_sci_intr_irq_domain_probe()
281 return -ENOMEM; in ti_sci_intr_irq_domain_probe()
284 dev_info(dev, "Interrupt Router %d domain created\n", intr->ti_sci_id); in ti_sci_intr_irq_domain_probe()
290 { .compatible = "ti,sci-intr", },
298 .name = "ti-sci-intr",
305 MODULE_DESCRIPTION("K3 Interrupt Router driver over TI SCI protocol");