diff options
author | 2025-03-28 11:22:54 -0700 | |
---|---|---|
committer | 2025-03-28 11:22:54 -0700 | |
commit | 112e43e9fd3b999513b1914e2bf523ae509f4c7d (patch) | |
tree | 6f2ede50b4d57e31b00e2d23e24b096dce0aeede /kernel/irq/msi.c | |
parent | Merge tag 'm68knommu-for-v6.15' of git://git.kernel.org/pub/scm/linux/kernel/git/gerg/m68knommu (diff) | |
download | wireguard-linux-112e43e9fd3b999513b1914e2bf523ae509f4c7d.tar.xz wireguard-linux-112e43e9fd3b999513b1914e2bf523ae509f4c7d.zip |
Revert "Merge tag 'irq-msi-2025-03-23' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip"
This reverts commit 36f5f026df6c1cd8a20373adc4388d2b3401ce91, reversing
changes made to 43a7eec035a5b64546c8adefdc9cf96a116da14b.
Thomas says:
"I just noticed that for some incomprehensible reason, probably sheer
incompetemce when trying to utilize b4, I managed to merge an outdated
_and_ buggy version of that series.
Can you please revert that merge completely?"
Done.
Requested-by: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to '')
-rw-r--r-- | kernel/irq/msi.c | 161 |
1 files changed, 106 insertions, 55 deletions
diff --git a/kernel/irq/msi.c b/kernel/irq/msi.c index 1951a08f0421..5c8d43cdb0a3 100644 --- a/kernel/irq/msi.c +++ b/kernel/irq/msi.c @@ -270,11 +270,16 @@ fail: return ret; } +void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg) +{ + *msg = entry->msg; +} + void get_cached_msi_msg(unsigned int irq, struct msi_msg *msg) { struct msi_desc *entry = irq_get_msi_desc(irq); - *msg = entry->msg; + __get_cached_msi_msg(entry, msg); } EXPORT_SYMBOL_GPL(get_cached_msi_msg); @@ -338,30 +343,26 @@ int msi_setup_device_data(struct device *dev) } /** - * __msi_lock_descs - Lock the MSI descriptor storage of a device + * msi_lock_descs - Lock the MSI descriptor storage of a device * @dev: Device to operate on - * - * Internal function for guard(msi_descs_lock). Don't use in code. */ -void __msi_lock_descs(struct device *dev) +void msi_lock_descs(struct device *dev) { mutex_lock(&dev->msi.data->mutex); } -EXPORT_SYMBOL_GPL(__msi_lock_descs); +EXPORT_SYMBOL_GPL(msi_lock_descs); /** - * __msi_unlock_descs - Unlock the MSI descriptor storage of a device + * msi_unlock_descs - Unlock the MSI descriptor storage of a device * @dev: Device to operate on - * - * Internal function for guard(msi_descs_lock). Don't use in code. */ -void __msi_unlock_descs(struct device *dev) +void msi_unlock_descs(struct device *dev) { /* Invalidate the index which was cached by the iterator */ dev->msi.data->__iter_idx = MSI_XA_MAX_INDEX; mutex_unlock(&dev->msi.data->mutex); } -EXPORT_SYMBOL_GPL(__msi_unlock_descs); +EXPORT_SYMBOL_GPL(msi_unlock_descs); static struct msi_desc *msi_find_desc(struct msi_device_data *md, unsigned int domid, enum msi_desc_filter filter) @@ -447,6 +448,7 @@ EXPORT_SYMBOL_GPL(msi_next_desc); unsigned int msi_domain_get_virq(struct device *dev, unsigned int domid, unsigned int index) { struct msi_desc *desc; + unsigned int ret = 0; bool pcimsi = false; struct xarray *xa; @@ -460,7 +462,7 @@ unsigned int msi_domain_get_virq(struct device *dev, unsigned int domid, unsigne if (dev_is_pci(dev) && domid == MSI_DEFAULT_DOMAIN) pcimsi = to_pci_dev(dev)->msi_enabled; - guard(msi_descs_lock)(dev); + msi_lock_descs(dev); xa = &dev->msi.data->__domains[domid].store; desc = xa_load(xa, pcimsi ? 0 : index); if (desc && desc->irq) { @@ -469,12 +471,16 @@ unsigned int msi_domain_get_virq(struct device *dev, unsigned int domid, unsigne * PCI-MSIX and platform MSI use a descriptor per * interrupt. */ - if (!pcimsi) - return desc->irq; - if (index < desc->nvec_used) - return desc->irq + index; + if (pcimsi) { + if (index < desc->nvec_used) + ret = desc->irq + index; + } else { + ret = desc->irq; + } } - return 0; + + msi_unlock_descs(dev); + return ret; } EXPORT_SYMBOL_GPL(msi_domain_get_virq); @@ -992,8 +998,9 @@ bool msi_create_device_irq_domain(struct device *dev, unsigned int domid, void *chip_data) { struct irq_domain *domain, *parent = dev->msi.domain; + struct fwnode_handle *fwnode, *fwnalloced = NULL; + struct msi_domain_template *bundle; const struct msi_parent_ops *pops; - struct fwnode_handle *fwnode; if (!irq_domain_is_msi_parent(parent)) return false; @@ -1001,8 +1008,7 @@ bool msi_create_device_irq_domain(struct device *dev, unsigned int domid, if (domid >= MSI_MAX_DEVICE_IRQDOMAINS) return false; - struct msi_domain_template *bundle __free(kfree) = - bundle = kmemdup(template, sizeof(*bundle), GFP_KERNEL); + bundle = kmemdup(template, sizeof(*bundle), GFP_KERNEL); if (!bundle) return false; @@ -1025,36 +1031,41 @@ bool msi_create_device_irq_domain(struct device *dev, unsigned int domid, * node as they are not guaranteed to have a fwnode. They are never * looked up and always handled in the context of the device. */ - struct fwnode_handle *fwnode_alloced __free(irq_domain_free_fwnode) = NULL; - - if (!(bundle->info.flags & MSI_FLAG_USE_DEV_FWNODE)) - fwnode = fwnode_alloced = irq_domain_alloc_named_fwnode(bundle->name); - else + if (bundle->info.flags & MSI_FLAG_USE_DEV_FWNODE) fwnode = dev->fwnode; + else + fwnode = fwnalloced = irq_domain_alloc_named_fwnode(bundle->name); if (!fwnode) - return false; + goto free_bundle; if (msi_setup_device_data(dev)) - return false; + goto free_fwnode; + + msi_lock_descs(dev); - guard(msi_descs_lock)(dev); if (WARN_ON_ONCE(msi_get_device_domain(dev, domid))) - return false; + goto fail; if (!pops->init_dev_msi_info(dev, parent, parent, &bundle->info)) - return false; + goto fail; domain = __msi_create_irq_domain(fwnode, &bundle->info, IRQ_DOMAIN_FLAG_MSI_DEVICE, parent); if (!domain) - return false; + goto fail; - /* @bundle and @fwnode_alloced are now in use. Prevent cleanup */ - retain_ptr(bundle); - retain_ptr(fwnode_alloced); domain->dev = dev; dev->msi.data->__domains[domid].domain = domain; + msi_unlock_descs(dev); return true; + +fail: + msi_unlock_descs(dev); +free_fwnode: + irq_domain_free_fwnode(fwnalloced); +free_bundle: + kfree(bundle); + return false; } /** @@ -1068,10 +1079,12 @@ void msi_remove_device_irq_domain(struct device *dev, unsigned int domid) struct msi_domain_info *info; struct irq_domain *domain; - guard(msi_descs_lock)(dev); + msi_lock_descs(dev); + domain = msi_get_device_domain(dev, domid); + if (!domain || !irq_domain_is_msi_device(domain)) - return; + goto unlock; dev->msi.data->__domains[domid].domain = NULL; info = domain->host_data; @@ -1080,6 +1093,9 @@ void msi_remove_device_irq_domain(struct device *dev, unsigned int domid) irq_domain_remove(domain); irq_domain_free_fwnode(fwnode); kfree(container_of(info, struct msi_domain_template, info)); + +unlock: + msi_unlock_descs(dev); } /** @@ -1095,14 +1111,16 @@ bool msi_match_device_irq_domain(struct device *dev, unsigned int domid, { struct msi_domain_info *info; struct irq_domain *domain; + bool ret = false; - guard(msi_descs_lock)(dev); + msi_lock_descs(dev); domain = msi_get_device_domain(dev, domid); if (domain && irq_domain_is_msi_device(domain)) { info = domain->host_data; - return info->bus_token == bus_token; + ret = info->bus_token == bus_token; } - return false; + msi_unlock_descs(dev); + return ret; } static int msi_domain_prepare_irqs(struct irq_domain *domain, struct device *dev, @@ -1334,17 +1352,21 @@ static int msi_domain_alloc_locked(struct device *dev, struct msi_ctrl *ctrl) } /** - * msi_domain_alloc_irqs_range - Allocate interrupts from a MSI interrupt domain + * msi_domain_alloc_irqs_range_locked - Allocate interrupts from a MSI interrupt domain * @dev: Pointer to device struct of the device for which the interrupts * are allocated * @domid: Id of the interrupt domain to operate on * @first: First index to allocate (inclusive) * @last: Last index to allocate (inclusive) * + * Must be invoked from within a msi_lock_descs() / msi_unlock_descs() + * pair. Use this for MSI irqdomains which implement their own descriptor + * allocation/free. + * * Return: %0 on success or an error code. */ -int msi_domain_alloc_irqs_range(struct device *dev, unsigned int domid, - unsigned int first, unsigned int last) +int msi_domain_alloc_irqs_range_locked(struct device *dev, unsigned int domid, + unsigned int first, unsigned int last) { struct msi_ctrl ctrl = { .domid = domid, @@ -1353,9 +1375,29 @@ int msi_domain_alloc_irqs_range(struct device *dev, unsigned int domid, .nirqs = last + 1 - first, }; - guard(msi_descs_lock)(dev); return msi_domain_alloc_locked(dev, &ctrl); } + +/** + * msi_domain_alloc_irqs_range - Allocate interrupts from a MSI interrupt domain + * @dev: Pointer to device struct of the device for which the interrupts + * are allocated + * @domid: Id of the interrupt domain to operate on + * @first: First index to allocate (inclusive) + * @last: Last index to allocate (inclusive) + * + * Return: %0 on success or an error code. + */ +int msi_domain_alloc_irqs_range(struct device *dev, unsigned int domid, + unsigned int first, unsigned int last) +{ + int ret; + + msi_lock_descs(dev); + ret = msi_domain_alloc_irqs_range_locked(dev, domid, first, last); + msi_unlock_descs(dev); + return ret; +} EXPORT_SYMBOL_GPL(msi_domain_alloc_irqs_range); /** @@ -1458,8 +1500,12 @@ struct msi_map msi_domain_alloc_irq_at(struct device *dev, unsigned int domid, u const struct irq_affinity_desc *affdesc, union msi_instance_cookie *icookie) { - guard(msi_descs_lock)(dev); - return __msi_domain_alloc_irq_at(dev, domid, index, affdesc, icookie); + struct msi_map map; + + msi_lock_descs(dev); + map = __msi_domain_alloc_irq_at(dev, domid, index, affdesc, icookie); + msi_unlock_descs(dev); + return map; } /** @@ -1496,11 +1542,13 @@ int msi_device_domain_alloc_wired(struct irq_domain *domain, unsigned int hwirq, icookie.value = ((u64)type << 32) | hwirq; - guard(msi_descs_lock)(dev); + msi_lock_descs(dev); if (WARN_ON_ONCE(msi_get_device_domain(dev, domid) != domain)) map.index = -EINVAL; else map = __msi_domain_alloc_irq_at(dev, domid, MSI_ANY_INDEX, NULL, &icookie); + msi_unlock_descs(dev); + return map.index >= 0 ? map.virq : map.index; } @@ -1570,8 +1618,8 @@ static void msi_domain_free_locked(struct device *dev, struct msi_ctrl *ctrl) * @first: First index to free (inclusive) * @last: Last index to free (inclusive) */ -static void msi_domain_free_irqs_range_locked(struct device *dev, unsigned int domid, - unsigned int first, unsigned int last) +void msi_domain_free_irqs_range_locked(struct device *dev, unsigned int domid, + unsigned int first, unsigned int last) { struct msi_ctrl ctrl = { .domid = domid, @@ -1593,8 +1641,9 @@ static void msi_domain_free_irqs_range_locked(struct device *dev, unsigned int d void msi_domain_free_irqs_range(struct device *dev, unsigned int domid, unsigned int first, unsigned int last) { - guard(msi_descs_lock)(dev); + msi_lock_descs(dev); msi_domain_free_irqs_range_locked(dev, domid, first, last); + msi_unlock_descs(dev); } EXPORT_SYMBOL_GPL(msi_domain_free_irqs_all); @@ -1624,8 +1673,9 @@ void msi_domain_free_irqs_all_locked(struct device *dev, unsigned int domid) */ void msi_domain_free_irqs_all(struct device *dev, unsigned int domid) { - guard(msi_descs_lock)(dev); + msi_lock_descs(dev); msi_domain_free_irqs_all_locked(dev, domid); + msi_unlock_descs(dev); } /** @@ -1644,11 +1694,12 @@ void msi_device_domain_free_wired(struct irq_domain *domain, unsigned int virq) if (WARN_ON_ONCE(!dev || !desc || domain->bus_token != DOMAIN_BUS_WIRED_TO_MSI)) return; - guard(msi_descs_lock)(dev); - if (WARN_ON_ONCE(msi_get_device_domain(dev, MSI_DEFAULT_DOMAIN) != domain)) - return; - msi_domain_free_irqs_range_locked(dev, MSI_DEFAULT_DOMAIN, desc->msi_index, - desc->msi_index); + msi_lock_descs(dev); + if (!WARN_ON_ONCE(msi_get_device_domain(dev, MSI_DEFAULT_DOMAIN) != domain)) { + msi_domain_free_irqs_range_locked(dev, MSI_DEFAULT_DOMAIN, desc->msi_index, + desc->msi_index); + } + msi_unlock_descs(dev); } /** |