Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

soc: ti: ti_sci_inta_msi: Rework MSI descriptor allocation

Protect the allocation properly and use the core allocation and free
mechanism.

No functional change intended.

Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Nishanth Menon <nm@ti.com>
Reviewed-by: Jason Gunthorpe <jgg@nvidia.com>
Link: https://lore.kernel.org/r/20211206210748.737904583@linutronix.de

+25 -45
+25 -45
drivers/soc/ti/ti_sci_inta_msi.c
··· 51 51 struct irq_domain *domain; 52 52 53 53 ti_sci_inta_msi_update_chip_ops(info); 54 + info->flags |= MSI_FLAG_FREE_MSI_DESCS; 54 55 55 56 domain = msi_create_irq_domain(fwnode, info, parent); 56 57 if (domain) ··· 61 60 } 62 61 EXPORT_SYMBOL_GPL(ti_sci_inta_msi_create_irq_domain); 63 62 64 - static void ti_sci_inta_msi_free_descs(struct device *dev) 65 - { 66 - struct msi_desc *desc, *tmp; 67 - 68 - list_for_each_entry_safe(desc, tmp, dev_to_msi_list(dev), list) { 69 - list_del(&desc->list); 70 - free_msi_entry(desc); 71 - } 72 - } 73 - 74 63 static int ti_sci_inta_msi_alloc_descs(struct device *dev, 75 64 struct ti_sci_resource *res) 76 65 { 77 - struct msi_desc *msi_desc; 66 + struct msi_desc msi_desc; 78 67 int set, i, count = 0; 79 68 69 + memset(&msi_desc, 0, sizeof(msi_desc)); 70 + msi_desc.nvec_used = 1; 71 + 80 72 for (set = 0; set < res->sets; set++) { 81 - for (i = 0; i < res->desc[set].num; i++) { 82 - msi_desc = alloc_msi_entry(dev, 1, NULL); 83 - if (!msi_desc) { 84 - ti_sci_inta_msi_free_descs(dev); 85 - return -ENOMEM; 86 - } 87 - 88 - msi_desc->msi_index = res->desc[set].start + i; 89 - INIT_LIST_HEAD(&msi_desc->list); 90 - list_add_tail(&msi_desc->list, dev_to_msi_list(dev)); 91 - count++; 73 + for (i = 0; i < res->desc[set].num; i++, count++) { 74 + msi_desc.msi_index = res->desc[set].start + i; 75 + if (msi_add_msi_desc(dev, &msi_desc)) 76 + goto fail; 92 77 } 93 - for (i = 0; i < res->desc[set].num_sec; i++) { 94 - msi_desc = alloc_msi_entry(dev, 1, NULL); 95 - if (!msi_desc) { 96 - ti_sci_inta_msi_free_descs(dev); 97 - return -ENOMEM; 98 - } 99 78 100 - msi_desc->msi_index = res->desc[set].start_sec + i; 101 - INIT_LIST_HEAD(&msi_desc->list); 102 - list_add_tail(&msi_desc->list, dev_to_msi_list(dev)); 103 - count++; 79 + for (i = 0; i < res->desc[set].num_sec; i++, count++) { 80 + msi_desc.msi_index = res->desc[set].start_sec + i; 81 + if (msi_add_msi_desc(dev, &msi_desc)) 82 + goto fail; 104 83 } 105 84 } 106 - 107 85 return count; 86 + fail: 87 + msi_free_msi_descs(dev); 88 + return -ENOMEM; 108 89 } 109 90 110 91 int ti_sci_inta_msi_domain_alloc_irqs(struct device *dev, ··· 107 124 if (ret) 108 125 return ret; 109 126 127 + msi_lock_descs(dev); 110 128 nvec = ti_sci_inta_msi_alloc_descs(dev, res); 111 - if (nvec <= 0) 112 - return nvec; 113 - 114 - ret = msi_domain_alloc_irqs(msi_domain, dev, nvec); 115 - if (ret) { 116 - dev_err(dev, "Failed to allocate IRQs %d\n", ret); 117 - goto cleanup; 129 + if (nvec <= 0) { 130 + ret = nvec; 131 + goto unlock; 118 132 } 119 133 120 - return 0; 121 - 122 - cleanup: 123 - ti_sci_inta_msi_free_descs(&pdev->dev); 134 + ret = msi_domain_alloc_irqs_descs_locked(msi_domain, dev, nvec); 135 + if (ret) 136 + dev_err(dev, "Failed to allocate IRQs %d\n", ret); 137 + unlock: 138 + msi_unlock_descs(dev); 124 139 return ret; 125 140 } 126 141 EXPORT_SYMBOL_GPL(ti_sci_inta_msi_domain_alloc_irqs); ··· 126 145 void ti_sci_inta_msi_domain_free_irqs(struct device *dev) 127 146 { 128 147 msi_domain_free_irqs(dev->msi.domain, dev); 129 - ti_sci_inta_msi_free_descs(dev); 130 148 } 131 149 EXPORT_SYMBOL_GPL(ti_sci_inta_msi_domain_free_irqs);