forked from mirrors/linux
		
	mm: factor out a devm_request_free_mem_region helper
Keep the physical address allocation that hmm_add_device does with the rest of the resource code, and allow future reuse of it without the hmm wrapper. Signed-off-by: Christoph Hellwig <hch@lst.de> Reviewed-by: Jason Gunthorpe <jgg@mellanox.com> Reviewed-by: John Hubbard <jhubbard@nvidia.com> Reviewed-by: Dan Williams <dan.j.williams@intel.com> Signed-off-by: Jason Gunthorpe <jgg@mellanox.com>
This commit is contained in:
		
							parent
							
								
									692622157b
								
							
						
					
					
						commit
						0092908d16
					
				
					 3 changed files with 45 additions and 29 deletions
				
			
		| 
						 | 
					@ -285,6 +285,8 @@ static inline bool resource_overlaps(struct resource *r1, struct resource *r2)
 | 
				
			||||||
       return (r1->start <= r2->end && r1->end >= r2->start);
 | 
					       return (r1->start <= r2->end && r1->end >= r2->start);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct resource *devm_request_free_mem_region(struct device *dev,
 | 
				
			||||||
 | 
							struct resource *base, unsigned long size);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif /* __ASSEMBLY__ */
 | 
					#endif /* __ASSEMBLY__ */
 | 
				
			||||||
#endif	/* _LINUX_IOPORT_H */
 | 
					#endif	/* _LINUX_IOPORT_H */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1628,6 +1628,45 @@ void resource_list_free(struct list_head *head)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
EXPORT_SYMBOL(resource_list_free);
 | 
					EXPORT_SYMBOL(resource_list_free);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef CONFIG_DEVICE_PRIVATE
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * devm_request_free_mem_region - find free region for device private memory
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * @dev: device struct to bind the resource to
 | 
				
			||||||
 | 
					 * @size: size in bytes of the device memory to add
 | 
				
			||||||
 | 
					 * @base: resource tree to look in
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This function tries to find an empty range of physical address big enough to
 | 
				
			||||||
 | 
					 * contain the new resource, so that it can later be hotplugged as ZONE_DEVICE
 | 
				
			||||||
 | 
					 * memory, which in turn allocates struct pages.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					struct resource *devm_request_free_mem_region(struct device *dev,
 | 
				
			||||||
 | 
							struct resource *base, unsigned long size)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						resource_size_t end, addr;
 | 
				
			||||||
 | 
						struct resource *res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						size = ALIGN(size, 1UL << PA_SECTION_SHIFT);
 | 
				
			||||||
 | 
						end = min_t(unsigned long, base->end, (1UL << MAX_PHYSMEM_BITS) - 1);
 | 
				
			||||||
 | 
						addr = end - size + 1UL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (; addr > size && addr >= base->start; addr -= size) {
 | 
				
			||||||
 | 
							if (region_intersects(addr, size, 0, IORES_DESC_NONE) !=
 | 
				
			||||||
 | 
									REGION_DISJOINT)
 | 
				
			||||||
 | 
								continue;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							res = devm_request_mem_region(dev, addr, size, dev_name(dev));
 | 
				
			||||||
 | 
							if (!res)
 | 
				
			||||||
 | 
								return ERR_PTR(-ENOMEM);
 | 
				
			||||||
 | 
							res->desc = IORES_DESC_DEVICE_PRIVATE_MEMORY;
 | 
				
			||||||
 | 
							return res;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return ERR_PTR(-ERANGE);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EXPORT_SYMBOL_GPL(devm_request_free_mem_region);
 | 
				
			||||||
 | 
					#endif /* CONFIG_DEVICE_PRIVATE */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int __init strict_iomem(char *str)
 | 
					static int __init strict_iomem(char *str)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	if (strstr(str, "relaxed"))
 | 
						if (strstr(str, "relaxed"))
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										33
									
								
								mm/hmm.c
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								mm/hmm.c
									
									
									
									
									
								
							| 
						 | 
					@ -25,8 +25,6 @@
 | 
				
			||||||
#include <linux/mmu_notifier.h>
 | 
					#include <linux/mmu_notifier.h>
 | 
				
			||||||
#include <linux/memory_hotplug.h>
 | 
					#include <linux/memory_hotplug.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define PA_SECTION_SIZE (1UL << PA_SECTION_SHIFT)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if IS_ENABLED(CONFIG_HMM_MIRROR)
 | 
					#if IS_ENABLED(CONFIG_HMM_MIRROR)
 | 
				
			||||||
static const struct mmu_notifier_ops hmm_mmu_notifier_ops;
 | 
					static const struct mmu_notifier_ops hmm_mmu_notifier_ops;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1408,7 +1406,6 @@ struct hmm_devmem *hmm_devmem_add(const struct hmm_devmem_ops *ops,
 | 
				
			||||||
				  unsigned long size)
 | 
									  unsigned long size)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct hmm_devmem *devmem;
 | 
						struct hmm_devmem *devmem;
 | 
				
			||||||
	resource_size_t addr;
 | 
					 | 
				
			||||||
	void *result;
 | 
						void *result;
 | 
				
			||||||
	int ret;
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1430,32 +1427,10 @@ struct hmm_devmem *hmm_devmem_add(const struct hmm_devmem_ops *ops,
 | 
				
			||||||
	if (ret)
 | 
						if (ret)
 | 
				
			||||||
		return ERR_PTR(ret);
 | 
							return ERR_PTR(ret);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	size = ALIGN(size, PA_SECTION_SIZE);
 | 
						devmem->resource = devm_request_free_mem_region(device, &iomem_resource,
 | 
				
			||||||
	addr = min((unsigned long)iomem_resource.end,
 | 
								size);
 | 
				
			||||||
		   (1UL << MAX_PHYSMEM_BITS) - 1);
 | 
						if (IS_ERR(devmem->resource))
 | 
				
			||||||
	addr = addr - size + 1UL;
 | 
							return ERR_CAST(devmem->resource);
 | 
				
			||||||
 | 
					 | 
				
			||||||
	/*
 | 
					 | 
				
			||||||
	 * FIXME add a new helper to quickly walk resource tree and find free
 | 
					 | 
				
			||||||
	 * range
 | 
					 | 
				
			||||||
	 *
 | 
					 | 
				
			||||||
	 * FIXME what about ioport_resource resource ?
 | 
					 | 
				
			||||||
	 */
 | 
					 | 
				
			||||||
	for (; addr > size && addr >= iomem_resource.start; addr -= size) {
 | 
					 | 
				
			||||||
		ret = region_intersects(addr, size, 0, IORES_DESC_NONE);
 | 
					 | 
				
			||||||
		if (ret != REGION_DISJOINT)
 | 
					 | 
				
			||||||
			continue;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		devmem->resource = devm_request_mem_region(device, addr, size,
 | 
					 | 
				
			||||||
							   dev_name(device));
 | 
					 | 
				
			||||||
		if (!devmem->resource)
 | 
					 | 
				
			||||||
			return ERR_PTR(-ENOMEM);
 | 
					 | 
				
			||||||
		break;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	if (!devmem->resource)
 | 
					 | 
				
			||||||
		return ERR_PTR(-ERANGE);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	devmem->resource->desc = IORES_DESC_DEVICE_PRIVATE_MEMORY;
 | 
					 | 
				
			||||||
	devmem->pfn_first = devmem->resource->start >> PAGE_SHIFT;
 | 
						devmem->pfn_first = devmem->resource->start >> PAGE_SHIFT;
 | 
				
			||||||
	devmem->pfn_last = devmem->pfn_first +
 | 
						devmem->pfn_last = devmem->pfn_first +
 | 
				
			||||||
			   (resource_size(devmem->resource) >> PAGE_SHIFT);
 | 
								   (resource_size(devmem->resource) >> PAGE_SHIFT);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue