xref: /openbmc/linux/drivers/pci/msi/irqdomain.c (revision 13e7accb81d6c07993385af8342238ff22b41ac8)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * PCI Message Signaled Interrupt (MSI) - irqdomain support
4  */
5 #include <linux/acpi_iort.h>
6 #include <linux/irqdomain.h>
7 #include <linux/of_irq.h>
8 
9 #include "msi.h"
10 
11 int pci_msi_setup_msi_irqs(struct pci_dev *dev, int nvec, int type)
12 {
13 	struct irq_domain *domain;
14 
15 	domain = dev_get_msi_domain(&dev->dev);
16 	if (domain && irq_domain_is_hierarchy(domain))
17 		return msi_domain_alloc_irqs_descs_locked(domain, &dev->dev, nvec);
18 
19 	return pci_msi_legacy_setup_msi_irqs(dev, nvec, type);
20 }
21 
22 void pci_msi_teardown_msi_irqs(struct pci_dev *dev)
23 {
24 	struct irq_domain *domain;
25 
26 	domain = dev_get_msi_domain(&dev->dev);
27 	if (domain && irq_domain_is_hierarchy(domain)) {
28 		msi_domain_free_irqs_descs_locked(domain, &dev->dev);
29 	} else {
30 		pci_msi_legacy_teardown_msi_irqs(dev);
31 		msi_free_msi_descs(&dev->dev);
32 	}
33 }
34 
35 /**
36  * pci_msi_domain_write_msg - Helper to write MSI message to PCI config space
37  * @irq_data:	Pointer to interrupt data of the MSI interrupt
38  * @msg:	Pointer to the message
39  */
40 static void pci_msi_domain_write_msg(struct irq_data *irq_data, struct msi_msg *msg)
41 {
42 	struct msi_desc *desc = irq_data_get_msi_desc(irq_data);
43 
44 	/*
45 	 * For MSI-X desc->irq is always equal to irq_data->irq. For
46 	 * MSI only the first interrupt of MULTI MSI passes the test.
47 	 */
48 	if (desc->irq == irq_data->irq)
49 		__pci_write_msi_msg(desc, msg);
50 }
51 
52 /**
53  * pci_msi_domain_calc_hwirq - Generate a unique ID for an MSI source
54  * @desc:	Pointer to the MSI descriptor
55  *
56  * The ID number is only used within the irqdomain.
57  */
58 static irq_hw_number_t pci_msi_domain_calc_hwirq(struct msi_desc *desc)
59 {
60 	struct pci_dev *dev = msi_desc_to_pci_dev(desc);
61 
62 	return (irq_hw_number_t)desc->msi_index |
63 		pci_dev_id(dev) << 11 |
64 		(pci_domain_nr(dev->bus) & 0xFFFFFFFF) << 27;
65 }
66 
67 static inline bool pci_msi_desc_is_multi_msi(struct msi_desc *desc)
68 {
69 	return !desc->pci.msi_attrib.is_msix && desc->nvec_used > 1;
70 }
71 
72 /**
73  * pci_msi_domain_check_cap - Verify that @domain supports the capabilities
74  *			      for @dev
75  * @domain:	The interrupt domain to check
76  * @info:	The domain info for verification
77  * @dev:	The device to check
78  *
79  * Returns:
80  *  0 if the functionality is supported
81  *  1 if Multi MSI is requested, but the domain does not support it
82  *  -ENOTSUPP otherwise
83  */
84 static int pci_msi_domain_check_cap(struct irq_domain *domain,
85 				    struct msi_domain_info *info,
86 				    struct device *dev)
87 {
88 	struct msi_desc *desc = msi_first_desc(dev, MSI_DESC_ALL);
89 
90 	/* Special handling to support __pci_enable_msi_range() */
91 	if (pci_msi_desc_is_multi_msi(desc) &&
92 	    !(info->flags & MSI_FLAG_MULTI_PCI_MSI))
93 		return 1;
94 
95 	if (desc->pci.msi_attrib.is_msix) {
96 		if (!(info->flags & MSI_FLAG_PCI_MSIX))
97 			return -ENOTSUPP;
98 
99 		if (info->flags & MSI_FLAG_MSIX_CONTIGUOUS) {
100 			unsigned int idx = 0;
101 
102 			/* Check for gaps in the entry indices */
103 			msi_for_each_desc(desc, dev, MSI_DESC_ALL) {
104 				if (desc->msi_index != idx++)
105 					return -ENOTSUPP;
106 			}
107 		}
108 	}
109 	return 0;
110 }
111 
112 static void pci_msi_domain_set_desc(msi_alloc_info_t *arg,
113 				    struct msi_desc *desc)
114 {
115 	arg->desc = desc;
116 	arg->hwirq = pci_msi_domain_calc_hwirq(desc);
117 }
118 
119 static struct msi_domain_ops pci_msi_domain_ops_default = {
120 	.set_desc	= pci_msi_domain_set_desc,
121 	.msi_check	= pci_msi_domain_check_cap,
122 };
123 
124 static void pci_msi_domain_update_dom_ops(struct msi_domain_info *info)
125 {
126 	struct msi_domain_ops *ops = info->ops;
127 
128 	if (ops == NULL) {
129 		info->ops = &pci_msi_domain_ops_default;
130 	} else {
131 		if (ops->set_desc == NULL)
132 			ops->set_desc = pci_msi_domain_set_desc;
133 		if (ops->msi_check == NULL)
134 			ops->msi_check = pci_msi_domain_check_cap;
135 	}
136 }
137 
138 static void pci_msi_domain_update_chip_ops(struct msi_domain_info *info)
139 {
140 	struct irq_chip *chip = info->chip;
141 
142 	BUG_ON(!chip);
143 	if (!chip->irq_write_msi_msg)
144 		chip->irq_write_msi_msg = pci_msi_domain_write_msg;
145 	if (!chip->irq_mask)
146 		chip->irq_mask = pci_msi_mask_irq;
147 	if (!chip->irq_unmask)
148 		chip->irq_unmask = pci_msi_unmask_irq;
149 }
150 
151 /**
152  * pci_msi_create_irq_domain - Create a MSI interrupt domain
153  * @fwnode:	Optional fwnode of the interrupt controller
154  * @info:	MSI domain info
155  * @parent:	Parent irq domain
156  *
157  * Updates the domain and chip ops and creates a MSI interrupt domain.
158  *
159  * Returns:
160  * A domain pointer or NULL in case of failure.
161  */
162 struct irq_domain *pci_msi_create_irq_domain(struct fwnode_handle *fwnode,
163 					     struct msi_domain_info *info,
164 					     struct irq_domain *parent)
165 {
166 	if (WARN_ON(info->flags & MSI_FLAG_LEVEL_CAPABLE))
167 		info->flags &= ~MSI_FLAG_LEVEL_CAPABLE;
168 
169 	if (info->flags & MSI_FLAG_USE_DEF_DOM_OPS)
170 		pci_msi_domain_update_dom_ops(info);
171 	if (info->flags & MSI_FLAG_USE_DEF_CHIP_OPS)
172 		pci_msi_domain_update_chip_ops(info);
173 
174 	/* Let the core code free MSI descriptors when freeing interrupts */
175 	info->flags |= MSI_FLAG_FREE_MSI_DESCS;
176 
177 	info->flags |= MSI_FLAG_ACTIVATE_EARLY | MSI_FLAG_DEV_SYSFS;
178 	if (IS_ENABLED(CONFIG_GENERIC_IRQ_RESERVATION_MODE))
179 		info->flags |= MSI_FLAG_MUST_REACTIVATE;
180 
181 	/* PCI-MSI is oneshot-safe */
182 	info->chip->flags |= IRQCHIP_ONESHOT_SAFE;
183 	/* Let the core update the bus token */
184 	info->bus_token = DOMAIN_BUS_PCI_MSI;
185 
186 	return msi_create_irq_domain(fwnode, info, parent);
187 }
188 EXPORT_SYMBOL_GPL(pci_msi_create_irq_domain);
189 
190 /*
191  * Users of the generic MSI infrastructure expect a device to have a single ID,
192  * so with DMA aliases we have to pick the least-worst compromise. Devices with
193  * DMA phantom functions tend to still emit MSIs from the real function number,
194  * so we ignore those and only consider topological aliases where either the
195  * alias device or RID appears on a different bus number. We also make the
196  * reasonable assumption that bridges are walked in an upstream direction (so
197  * the last one seen wins), and the much braver assumption that the most likely
198  * case is that of PCI->PCIe so we should always use the alias RID. This echoes
199  * the logic from intel_irq_remapping's set_msi_sid(), which presumably works
200  * well enough in practice; in the face of the horrible PCIe<->PCI-X conditions
201  * for taking ownership all we can really do is close our eyes and hope...
202  */
203 static int get_msi_id_cb(struct pci_dev *pdev, u16 alias, void *data)
204 {
205 	u32 *pa = data;
206 	u8 bus = PCI_BUS_NUM(*pa);
207 
208 	if (pdev->bus->number != bus || PCI_BUS_NUM(alias) != bus)
209 		*pa = alias;
210 
211 	return 0;
212 }
213 
214 /**
215  * pci_msi_domain_get_msi_rid - Get the MSI requester id (RID)
216  * @domain:	The interrupt domain
217  * @pdev:	The PCI device.
218  *
219  * The RID for a device is formed from the alias, with a firmware
220  * supplied mapping applied
221  *
222  * Returns: The RID.
223  */
224 u32 pci_msi_domain_get_msi_rid(struct irq_domain *domain, struct pci_dev *pdev)
225 {
226 	struct device_node *of_node;
227 	u32 rid = pci_dev_id(pdev);
228 
229 	pci_for_each_dma_alias(pdev, get_msi_id_cb, &rid);
230 
231 	of_node = irq_domain_get_of_node(domain);
232 	rid = of_node ? of_msi_map_id(&pdev->dev, of_node, rid) :
233 			iort_msi_map_id(&pdev->dev, rid);
234 
235 	return rid;
236 }
237 
238 /**
239  * pci_msi_get_device_domain - Get the MSI domain for a given PCI device
240  * @pdev:	The PCI device
241  *
242  * Use the firmware data to find a device-specific MSI domain
243  * (i.e. not one that is set as a default).
244  *
245  * Returns: The corresponding MSI domain or NULL if none has been found.
246  */
247 struct irq_domain *pci_msi_get_device_domain(struct pci_dev *pdev)
248 {
249 	struct irq_domain *dom;
250 	u32 rid = pci_dev_id(pdev);
251 
252 	pci_for_each_dma_alias(pdev, get_msi_id_cb, &rid);
253 	dom = of_msi_map_get_device_domain(&pdev->dev, rid, DOMAIN_BUS_PCI_MSI);
254 	if (!dom)
255 		dom = iort_get_device_domain(&pdev->dev, rid,
256 					     DOMAIN_BUS_PCI_MSI);
257 	return dom;
258 }
259 
260 /**
261  * pci_dev_has_special_msi_domain - Check whether the device is handled by
262  *				    a non-standard PCI-MSI domain
263  * @pdev:	The PCI device to check.
264  *
265  * Returns: True if the device irqdomain or the bus irqdomain is
266  * non-standard PCI/MSI.
267  */
268 bool pci_dev_has_special_msi_domain(struct pci_dev *pdev)
269 {
270 	struct irq_domain *dom = dev_get_msi_domain(&pdev->dev);
271 
272 	if (!dom)
273 		dom = dev_get_msi_domain(&pdev->bus->dev);
274 
275 	if (!dom)
276 		return true;
277 
278 	return dom->bus_token != DOMAIN_BUS_PCI_MSI;
279 }
280