mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 16:48:26 +02:00 
			
		
		
		
	genirq/msi: Make msi_add_simple_msi_descs() device domain aware
Allocating simple interrupt descriptors in the core code has to be multi device irqdomain aware for the upcoming PCI/IMS support. Change the interfaces to take a domain id into account. Use the internal control struct for transport of arguments. Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Reviewed-by: Kevin Tian <kevin.tian@intel.com> Acked-by: Marc Zyngier <maz@kernel.org> Link: https://lore.kernel.org/r/20221124230314.279112474@linutronix.de
This commit is contained in:
		
							parent
							
								
									377712c5a4
								
							
						
					
					
						commit
						40742716f2
					
				
					 1 changed files with 57 additions and 41 deletions
				
			
		|  | @ -116,39 +116,6 @@ int msi_domain_insert_msi_desc(struct device *dev, unsigned int domid, | ||||||
| 	return msi_insert_desc(dev->msi.data, desc, domid, init_desc->msi_index); | 	return msi_insert_desc(dev->msi.data, desc, domid, init_desc->msi_index); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * msi_add_simple_msi_descs - Allocate and initialize MSI descriptors |  | ||||||
|  * @dev:	Pointer to the device for which the descriptors are allocated |  | ||||||
|  * @index:	Index for the first MSI descriptor |  | ||||||
|  * @ndesc:	Number of descriptors to allocate |  | ||||||
|  * |  | ||||||
|  * Return: 0 on success or an appropriate failure code. |  | ||||||
|  */ |  | ||||||
| static int msi_add_simple_msi_descs(struct device *dev, unsigned int index, unsigned int ndesc) |  | ||||||
| { |  | ||||||
| 	unsigned int idx, last = index + ndesc - 1; |  | ||||||
| 	struct msi_desc *desc; |  | ||||||
| 	int ret; |  | ||||||
| 
 |  | ||||||
| 	lockdep_assert_held(&dev->msi.data->mutex); |  | ||||||
| 
 |  | ||||||
| 	for (idx = index; idx <= last; idx++) { |  | ||||||
| 		desc = msi_alloc_desc(dev, 1, NULL); |  | ||||||
| 		if (!desc) |  | ||||||
| 			goto fail_mem; |  | ||||||
| 		ret = msi_insert_desc(dev->msi.data, desc, MSI_DEFAULT_DOMAIN, idx); |  | ||||||
| 		if (ret) |  | ||||||
| 			goto fail; |  | ||||||
| 	} |  | ||||||
| 	return 0; |  | ||||||
| 
 |  | ||||||
| fail_mem: |  | ||||||
| 	ret = -ENOMEM; |  | ||||||
| fail: |  | ||||||
| 	msi_free_msi_descs_range(dev, index, last); |  | ||||||
| 	return ret; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static bool msi_desc_match(struct msi_desc *desc, enum msi_desc_filter filter) | static bool msi_desc_match(struct msi_desc *desc, enum msi_desc_filter filter) | ||||||
| { | { | ||||||
| 	switch (filter) { | 	switch (filter) { | ||||||
|  | @ -166,6 +133,7 @@ static bool msi_desc_match(struct msi_desc *desc, enum msi_desc_filter filter) | ||||||
| static bool msi_ctrl_valid(struct device *dev, struct msi_ctrl *ctrl) | static bool msi_ctrl_valid(struct device *dev, struct msi_ctrl *ctrl) | ||||||
| { | { | ||||||
| 	if (WARN_ON_ONCE(ctrl->domid >= MSI_MAX_DEVICE_IRQDOMAINS || | 	if (WARN_ON_ONCE(ctrl->domid >= MSI_MAX_DEVICE_IRQDOMAINS || | ||||||
|  | 			 !dev->msi.data->__domains[ctrl->domid].domain || | ||||||
| 			 ctrl->first > ctrl->last || | 			 ctrl->first > ctrl->last || | ||||||
| 			 ctrl->first > MSI_MAX_INDEX || | 			 ctrl->first > MSI_MAX_INDEX || | ||||||
| 			 ctrl->last > MSI_MAX_INDEX)) | 			 ctrl->last > MSI_MAX_INDEX)) | ||||||
|  | @ -214,6 +182,41 @@ void msi_domain_free_msi_descs_range(struct device *dev, unsigned int domid, | ||||||
| 	msi_domain_free_descs(dev, &ctrl); | 	msi_domain_free_descs(dev, &ctrl); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * msi_domain_add_simple_msi_descs - Allocate and initialize MSI descriptors | ||||||
|  |  * @dev:	Pointer to the device for which the descriptors are allocated | ||||||
|  |  * @ctrl:	Allocation control struct | ||||||
|  |  * | ||||||
|  |  * Return: 0 on success or an appropriate failure code. | ||||||
|  |  */ | ||||||
|  | static int msi_domain_add_simple_msi_descs(struct device *dev, struct msi_ctrl *ctrl) | ||||||
|  | { | ||||||
|  | 	struct msi_desc *desc; | ||||||
|  | 	unsigned int idx; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	lockdep_assert_held(&dev->msi.data->mutex); | ||||||
|  | 
 | ||||||
|  | 	if (!msi_ctrl_valid(dev, ctrl)) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	for (idx = ctrl->first; idx <= ctrl->last; idx++) { | ||||||
|  | 		desc = msi_alloc_desc(dev, 1, NULL); | ||||||
|  | 		if (!desc) | ||||||
|  | 			goto fail_mem; | ||||||
|  | 		ret = msi_insert_desc(dev->msi.data, desc, ctrl->domid, idx); | ||||||
|  | 		if (ret) | ||||||
|  | 			goto fail; | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  | fail_mem: | ||||||
|  | 	ret = -ENOMEM; | ||||||
|  | fail: | ||||||
|  | 	msi_domain_free_descs(dev, ctrl); | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg) | void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg) | ||||||
| { | { | ||||||
| 	*msg = entry->msg; | 	*msg = entry->msg; | ||||||
|  | @ -786,16 +789,24 @@ int msi_domain_populate_irqs(struct irq_domain *domain, struct device *dev, | ||||||
| { | { | ||||||
| 	struct msi_domain_info *info = domain->host_data; | 	struct msi_domain_info *info = domain->host_data; | ||||||
| 	struct msi_domain_ops *ops = info->ops; | 	struct msi_domain_ops *ops = info->ops; | ||||||
|  | 	struct msi_ctrl ctrl = { | ||||||
|  | 		.domid	= MSI_DEFAULT_DOMAIN, | ||||||
|  | 		.first  = virq_base, | ||||||
|  | 		.last	= virq_base + nvec - 1, | ||||||
|  | 	}; | ||||||
| 	struct msi_desc *desc; | 	struct msi_desc *desc; | ||||||
| 	struct xarray *xa; | 	struct xarray *xa; | ||||||
| 	int ret, virq; | 	int ret, virq; | ||||||
| 
 | 
 | ||||||
|  | 	if (!msi_ctrl_valid(dev, &ctrl)) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
| 	msi_lock_descs(dev); | 	msi_lock_descs(dev); | ||||||
| 	ret = msi_add_simple_msi_descs(dev, virq_base, nvec); | 	ret = msi_domain_add_simple_msi_descs(dev, &ctrl); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		goto unlock; | 		goto unlock; | ||||||
| 
 | 
 | ||||||
| 	xa = &dev->msi.data->__domains[MSI_DEFAULT_DOMAIN].store; | 	xa = &dev->msi.data->__domains[ctrl.domid].store; | ||||||
| 
 | 
 | ||||||
| 	for (virq = virq_base; virq < virq_base + nvec; virq++) { | 	for (virq = virq_base; virq < virq_base + nvec; virq++) { | ||||||
| 		desc = xa_load(xa, virq); | 		desc = xa_load(xa, virq); | ||||||
|  | @ -814,7 +825,7 @@ int msi_domain_populate_irqs(struct irq_domain *domain, struct device *dev, | ||||||
| fail: | fail: | ||||||
| 	for (--virq; virq >= virq_base; virq--) | 	for (--virq; virq >= virq_base; virq--) | ||||||
| 		irq_domain_free_irqs_common(domain, virq, 1); | 		irq_domain_free_irqs_common(domain, virq, 1); | ||||||
| 	msi_free_msi_descs_range(dev, virq_base, virq_base + nvec - 1); | 	msi_domain_free_descs(dev, &ctrl); | ||||||
| unlock: | unlock: | ||||||
| 	msi_unlock_descs(dev); | 	msi_unlock_descs(dev); | ||||||
| 	return ret; | 	return ret; | ||||||
|  | @ -988,14 +999,19 @@ static int __msi_domain_alloc_irqs(struct irq_domain *domain, struct device *dev | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int msi_domain_add_simple_msi_descs(struct msi_domain_info *info, | static int msi_domain_alloc_simple_msi_descs(struct device *dev, | ||||||
| 					   struct device *dev, | 					     struct msi_domain_info *info, | ||||||
| 					   unsigned int num_descs) | 					     unsigned int num_descs) | ||||||
| { | { | ||||||
|  | 	struct msi_ctrl ctrl = { | ||||||
|  | 		.domid	= MSI_DEFAULT_DOMAIN, | ||||||
|  | 		.last	= num_descs - 1, | ||||||
|  | 	}; | ||||||
|  | 
 | ||||||
| 	if (!(info->flags & MSI_FLAG_ALLOC_SIMPLE_MSI_DESCS)) | 	if (!(info->flags & MSI_FLAG_ALLOC_SIMPLE_MSI_DESCS)) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	return msi_add_simple_msi_descs(dev, 0, num_descs); | 	return msi_domain_add_simple_msi_descs(dev, &ctrl); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -1026,7 +1042,7 @@ int msi_domain_alloc_irqs_descs_locked(struct irq_domain *domain, struct device | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Frees allocated descriptors in case of failure. */ | 	/* Frees allocated descriptors in case of failure. */ | ||||||
| 	ret = msi_domain_add_simple_msi_descs(info, dev, nvec); | 	ret = msi_domain_alloc_simple_msi_descs(dev, info, nvec); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		goto free; | 		goto free; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Thomas Gleixner
						Thomas Gleixner