IB/core: Improve ODP to use hmm_range_fault()

Move to use hmm_range_fault() instead of get_user_pags_remote() to improve
performance in a few aspects:

This includes:
- Dropping the need to allocate and free memory to hold its output

- No need any more to use put_page() to unpin the pages

- The logic to detect contiguous pages is done based on the returned
  order, no need to run per page and evaluate.

In addition, moving to use hmm_range_fault() enables to reduce page faults
in the system with it's snapshot mode, this will be introduced in next
patches from this series.

As part of this, cleanup some flows and use the required data structures
to work with hmm_range_fault().

Link: https://lore.kernel.org/r/20200930163828.1336747-2-leon@kernel.org
Signed-off-by: Yishai Hadas <yishaih@nvidia.com>
Signed-off-by: Leon Romanovsky <leonro@nvidia.com>
Signed-off-by: Jason Gunthorpe <jgg@nvidia.com>
This commit is contained in:
Yishai Hadas 2020-09-30 19:38:25 +03:00 committed by Jason Gunthorpe
parent 2ee9bf346f
commit 36f30e486d
4 changed files with 130 additions and 198 deletions

View File

@ -48,6 +48,7 @@ config INFINIBAND_ON_DEMAND_PAGING
depends on INFINIBAND_USER_MEM depends on INFINIBAND_USER_MEM
select MMU_NOTIFIER select MMU_NOTIFIER
select INTERVAL_TREE select INTERVAL_TREE
select HMM_MIRROR
default y default y
help help
On demand paging support for the InfiniBand subsystem. On demand paging support for the InfiniBand subsystem.

View File

@ -40,6 +40,7 @@
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/hugetlb.h> #include <linux/hugetlb.h>
#include <linux/interval_tree.h> #include <linux/interval_tree.h>
#include <linux/hmm.h>
#include <linux/pagemap.h> #include <linux/pagemap.h>
#include <rdma/ib_verbs.h> #include <rdma/ib_verbs.h>
@ -60,7 +61,7 @@ static inline int ib_init_umem_odp(struct ib_umem_odp *umem_odp,
size_t page_size = 1UL << umem_odp->page_shift; size_t page_size = 1UL << umem_odp->page_shift;
unsigned long start; unsigned long start;
unsigned long end; unsigned long end;
size_t pages; size_t ndmas, npfns;
start = ALIGN_DOWN(umem_odp->umem.address, page_size); start = ALIGN_DOWN(umem_odp->umem.address, page_size);
if (check_add_overflow(umem_odp->umem.address, if (check_add_overflow(umem_odp->umem.address,
@ -71,20 +72,21 @@ static inline int ib_init_umem_odp(struct ib_umem_odp *umem_odp,
if (unlikely(end < page_size)) if (unlikely(end < page_size))
return -EOVERFLOW; return -EOVERFLOW;
pages = (end - start) >> umem_odp->page_shift; ndmas = (end - start) >> umem_odp->page_shift;
if (!pages) if (!ndmas)
return -EINVAL; return -EINVAL;
umem_odp->page_list = kvcalloc( npfns = (end - start) >> PAGE_SHIFT;
pages, sizeof(*umem_odp->page_list), GFP_KERNEL); umem_odp->pfn_list = kvcalloc(
if (!umem_odp->page_list) npfns, sizeof(*umem_odp->pfn_list), GFP_KERNEL);
if (!umem_odp->pfn_list)
return -ENOMEM; return -ENOMEM;
umem_odp->dma_list = kvcalloc( umem_odp->dma_list = kvcalloc(
pages, sizeof(*umem_odp->dma_list), GFP_KERNEL); ndmas, sizeof(*umem_odp->dma_list), GFP_KERNEL);
if (!umem_odp->dma_list) { if (!umem_odp->dma_list) {
ret = -ENOMEM; ret = -ENOMEM;
goto out_page_list; goto out_pfn_list;
} }
ret = mmu_interval_notifier_insert(&umem_odp->notifier, ret = mmu_interval_notifier_insert(&umem_odp->notifier,
@ -98,8 +100,8 @@ static inline int ib_init_umem_odp(struct ib_umem_odp *umem_odp,
out_dma_list: out_dma_list:
kvfree(umem_odp->dma_list); kvfree(umem_odp->dma_list);
out_page_list: out_pfn_list:
kvfree(umem_odp->page_list); kvfree(umem_odp->pfn_list);
return ret; return ret;
} }
@ -276,7 +278,7 @@ void ib_umem_odp_release(struct ib_umem_odp *umem_odp)
mutex_unlock(&umem_odp->umem_mutex); mutex_unlock(&umem_odp->umem_mutex);
mmu_interval_notifier_remove(&umem_odp->notifier); mmu_interval_notifier_remove(&umem_odp->notifier);
kvfree(umem_odp->dma_list); kvfree(umem_odp->dma_list);
kvfree(umem_odp->page_list); kvfree(umem_odp->pfn_list);
} }
put_pid(umem_odp->tgid); put_pid(umem_odp->tgid);
kfree(umem_odp); kfree(umem_odp);
@ -287,87 +289,56 @@ EXPORT_SYMBOL(ib_umem_odp_release);
* Map for DMA and insert a single page into the on-demand paging page tables. * Map for DMA and insert a single page into the on-demand paging page tables.
* *
* @umem: the umem to insert the page to. * @umem: the umem to insert the page to.
* @page_index: index in the umem to add the page to. * @dma_index: index in the umem to add the dma to.
* @page: the page struct to map and add. * @page: the page struct to map and add.
* @access_mask: access permissions needed for this page. * @access_mask: access permissions needed for this page.
* @current_seq: sequence number for synchronization with invalidations. * @current_seq: sequence number for synchronization with invalidations.
* the sequence number is taken from * the sequence number is taken from
* umem_odp->notifiers_seq. * umem_odp->notifiers_seq.
* *
* The function returns -EFAULT if the DMA mapping operation fails. It returns * The function returns -EFAULT if the DMA mapping operation fails.
* -EAGAIN if a concurrent invalidation prevents us from updating the page.
* *
* The page is released via put_page even if the operation failed. For on-demand
* pinning, the page is released whenever it isn't stored in the umem.
*/ */
static int ib_umem_odp_map_dma_single_page( static int ib_umem_odp_map_dma_single_page(
struct ib_umem_odp *umem_odp, struct ib_umem_odp *umem_odp,
unsigned int page_index, unsigned int dma_index,
struct page *page, struct page *page,
u64 access_mask, u64 access_mask)
unsigned long current_seq)
{ {
struct ib_device *dev = umem_odp->umem.ibdev; struct ib_device *dev = umem_odp->umem.ibdev;
dma_addr_t dma_addr; dma_addr_t *dma_addr = &umem_odp->dma_list[dma_index];
int ret = 0;
if (mmu_interval_check_retry(&umem_odp->notifier, current_seq)) { if (*dma_addr) {
ret = -EAGAIN;
goto out;
}
if (!(umem_odp->dma_list[page_index])) {
dma_addr =
ib_dma_map_page(dev, page, 0, BIT(umem_odp->page_shift),
DMA_BIDIRECTIONAL);
if (ib_dma_mapping_error(dev, dma_addr)) {
ret = -EFAULT;
goto out;
}
umem_odp->dma_list[page_index] = dma_addr | access_mask;
umem_odp->page_list[page_index] = page;
umem_odp->npages++;
} else if (umem_odp->page_list[page_index] == page) {
umem_odp->dma_list[page_index] |= access_mask;
} else {
/* /*
* This is a race here where we could have done: * If the page is already dma mapped it means it went through
* * a non-invalidating trasition, like read-only to writable.
* CPU0 CPU1 * Resync the flags.
* get_user_pages()
* invalidate()
* page_fault()
* mutex_lock(umem_mutex)
* page from GUP != page in ODP
*
* It should be prevented by the retry test above as reading
* the seq number should be reliable under the
* umem_mutex. Thus something is really not working right if
* things get here.
*/ */
WARN(true, *dma_addr = (*dma_addr & ODP_DMA_ADDR_MASK) | access_mask;
"Got different pages in IB device and from get_user_pages. IB device page: %p, gup page: %p\n", return 0;
umem_odp->page_list[page_index], page);
ret = -EAGAIN;
} }
out: *dma_addr = ib_dma_map_page(dev, page, 0, 1 << umem_odp->page_shift,
put_page(page); DMA_BIDIRECTIONAL);
return ret; if (ib_dma_mapping_error(dev, *dma_addr)) {
*dma_addr = 0;
return -EFAULT;
}
umem_odp->npages++;
*dma_addr |= access_mask;
return 0;
} }
/** /**
* ib_umem_odp_map_dma_pages - Pin and DMA map userspace memory in an ODP MR. * ib_umem_odp_map_dma_and_lock - DMA map userspace memory in an ODP MR and lock it.
* *
* Pins the range of pages passed in the argument, and maps them to * Maps the range passed in the argument to DMA addresses.
* DMA addresses. The DMA addresses of the mapped pages is updated in * The DMA addresses of the mapped pages is updated in umem_odp->dma_list.
* umem_odp->dma_list. * Upon success the ODP MR will be locked to let caller complete its device
* page table update.
* *
* Returns the number of pages mapped in success, negative error code * Returns the number of pages mapped in success, negative error code
* for failure. * for failure.
* An -EAGAIN error code is returned when a concurrent mmu notifier prevents
* the function from completing its task.
* An -ENOENT error code indicates that userspace process is being terminated
* and mm was already destroyed.
* @umem_odp: the umem to map and pin * @umem_odp: the umem to map and pin
* @user_virt: the address from which we need to map. * @user_virt: the address from which we need to map.
* @bcnt: the minimal number of bytes to pin and map. The mapping might be * @bcnt: the minimal number of bytes to pin and map. The mapping might be
@ -376,21 +347,18 @@ static int ib_umem_odp_map_dma_single_page(
* the return value. * the return value.
* @access_mask: bit mask of the requested access permissions for the given * @access_mask: bit mask of the requested access permissions for the given
* range. * range.
* @current_seq: the MMU notifiers sequance value for synchronization with
* invalidations. the sequance number is read from
* umem_odp->notifiers_seq before calling this function
*/ */
int ib_umem_odp_map_dma_pages(struct ib_umem_odp *umem_odp, u64 user_virt, int ib_umem_odp_map_dma_and_lock(struct ib_umem_odp *umem_odp, u64 user_virt,
u64 bcnt, u64 access_mask, u64 bcnt, u64 access_mask)
unsigned long current_seq) __acquires(&umem_odp->umem_mutex)
{ {
struct task_struct *owning_process = NULL; struct task_struct *owning_process = NULL;
struct mm_struct *owning_mm = umem_odp->umem.owning_mm; struct mm_struct *owning_mm = umem_odp->umem.owning_mm;
struct page **local_page_list = NULL; int pfn_index, dma_index, ret = 0, start_idx;
u64 page_mask, off; unsigned int page_shift, hmm_order, pfn_start_idx;
int j, k, ret = 0, start_idx, npages = 0; unsigned long num_pfns, current_seq;
unsigned int flags = 0, page_shift; struct hmm_range range = {};
phys_addr_t p = 0; unsigned long timeout;
if (access_mask == 0) if (access_mask == 0)
return -EINVAL; return -EINVAL;
@ -399,15 +367,7 @@ int ib_umem_odp_map_dma_pages(struct ib_umem_odp *umem_odp, u64 user_virt,
user_virt + bcnt > ib_umem_end(umem_odp)) user_virt + bcnt > ib_umem_end(umem_odp))
return -EFAULT; return -EFAULT;
local_page_list = (struct page **)__get_free_page(GFP_KERNEL);
if (!local_page_list)
return -ENOMEM;
page_shift = umem_odp->page_shift; page_shift = umem_odp->page_shift;
page_mask = ~(BIT(page_shift) - 1);
off = user_virt & (~page_mask);
user_virt = user_virt & page_mask;
bcnt += off; /* Charge for the first page offset as well. */
/* /*
* owning_process is allowed to be NULL, this means somehow the mm is * owning_process is allowed to be NULL, this means somehow the mm is
@ -420,99 +380,90 @@ int ib_umem_odp_map_dma_pages(struct ib_umem_odp *umem_odp, u64 user_virt,
goto out_put_task; goto out_put_task;
} }
range.notifier = &umem_odp->notifier;
range.start = ALIGN_DOWN(user_virt, 1UL << page_shift);
range.end = ALIGN(user_virt + bcnt, 1UL << page_shift);
pfn_start_idx = (range.start - ib_umem_start(umem_odp)) >> PAGE_SHIFT;
num_pfns = (range.end - range.start) >> PAGE_SHIFT;
range.default_flags = HMM_PFN_REQ_FAULT;
if (access_mask & ODP_WRITE_ALLOWED_BIT) if (access_mask & ODP_WRITE_ALLOWED_BIT)
flags |= FOLL_WRITE; range.default_flags |= HMM_PFN_REQ_WRITE;
start_idx = (user_virt - ib_umem_start(umem_odp)) >> page_shift; range.hmm_pfns = &(umem_odp->pfn_list[pfn_start_idx]);
k = start_idx; timeout = jiffies + msecs_to_jiffies(HMM_RANGE_DEFAULT_TIMEOUT);
while (bcnt > 0) { retry:
const size_t gup_num_pages = min_t(size_t, current_seq = range.notifier_seq =
ALIGN(bcnt, PAGE_SIZE) / PAGE_SIZE, mmu_interval_read_begin(&umem_odp->notifier);
PAGE_SIZE / sizeof(struct page *));
mmap_read_lock(owning_mm); mmap_read_lock(owning_mm);
ret = hmm_range_fault(&range);
mmap_read_unlock(owning_mm);
if (unlikely(ret)) {
if (ret == -EBUSY && !time_after(jiffies, timeout))
goto retry;
goto out_put_mm;
}
start_idx = (range.start - ib_umem_start(umem_odp)) >> page_shift;
dma_index = start_idx;
mutex_lock(&umem_odp->umem_mutex);
if (mmu_interval_read_retry(&umem_odp->notifier, current_seq)) {
mutex_unlock(&umem_odp->umem_mutex);
goto retry;
}
for (pfn_index = 0; pfn_index < num_pfns;
pfn_index += 1 << (page_shift - PAGE_SHIFT), dma_index++) {
/* /*
* Note: this might result in redundent page getting. We can * Since we asked for hmm_range_fault() to populate pages,
* avoid this by checking dma_list to be 0 before calling * it shouldn't return an error entry on success.
* get_user_pages. However, this make the code much more
* complex (and doesn't gain us much performance in most use
* cases).
*/ */
npages = get_user_pages_remote(owning_mm, WARN_ON(range.hmm_pfns[pfn_index] & HMM_PFN_ERROR);
user_virt, gup_num_pages, WARN_ON(!(range.hmm_pfns[pfn_index] & HMM_PFN_VALID));
flags, local_page_list, NULL, NULL); hmm_order = hmm_pfn_to_map_order(range.hmm_pfns[pfn_index]);
mmap_read_unlock(owning_mm); /* If a hugepage was detected and ODP wasn't set for, the umem
* page_shift will be used, the opposite case is an error.
if (npages < 0) { */
if (npages != -EAGAIN) if (hmm_order + PAGE_SHIFT < page_shift) {
pr_warn("fail to get %zu user pages with error %d\n", gup_num_pages, npages); ret = -EINVAL;
else ibdev_dbg(umem_odp->umem.ibdev,
pr_debug("fail to get %zu user pages with error %d\n", gup_num_pages, npages); "%s: un-expected hmm_order %d, page_shift %d\n",
__func__, hmm_order, page_shift);
break; break;
} }
bcnt -= min_t(size_t, npages << PAGE_SHIFT, bcnt); ret = ib_umem_odp_map_dma_single_page(
mutex_lock(&umem_odp->umem_mutex); umem_odp, dma_index, hmm_pfn_to_page(range.hmm_pfns[pfn_index]),
for (j = 0; j < npages; j++, user_virt += PAGE_SIZE) { access_mask);
if (user_virt & ~page_mask) { if (ret < 0) {
p += PAGE_SIZE; ibdev_dbg(umem_odp->umem.ibdev,
if (page_to_phys(local_page_list[j]) != p) { "ib_umem_odp_map_dma_single_page failed with error %d\n", ret);
ret = -EFAULT; break;
break;
}
put_page(local_page_list[j]);
continue;
}
ret = ib_umem_odp_map_dma_single_page(
umem_odp, k, local_page_list[j],
access_mask, current_seq);
if (ret < 0) {
if (ret != -EAGAIN)
pr_warn("ib_umem_odp_map_dma_single_page failed with error %d\n", ret);
else
pr_debug("ib_umem_odp_map_dma_single_page failed with error %d\n", ret);
break;
}
p = page_to_phys(local_page_list[j]);
k++;
} }
}
/* upon sucesss lock should stay on hold for the callee */
if (!ret)
ret = dma_index - start_idx;
else
mutex_unlock(&umem_odp->umem_mutex); mutex_unlock(&umem_odp->umem_mutex);
if (ret < 0) { out_put_mm:
/*
* Release pages, remembering that the first page
* to hit an error was already released by
* ib_umem_odp_map_dma_single_page().
*/
if (npages - (j + 1) > 0)
release_pages(&local_page_list[j+1],
npages - (j + 1));
break;
}
}
if (ret >= 0) {
if (npages < 0 && k == start_idx)
ret = npages;
else
ret = k - start_idx;
}
mmput(owning_mm); mmput(owning_mm);
out_put_task: out_put_task:
if (owning_process) if (owning_process)
put_task_struct(owning_process); put_task_struct(owning_process);
free_page((unsigned long)local_page_list);
return ret; return ret;
} }
EXPORT_SYMBOL(ib_umem_odp_map_dma_pages); EXPORT_SYMBOL(ib_umem_odp_map_dma_and_lock);
void ib_umem_odp_unmap_dma_pages(struct ib_umem_odp *umem_odp, u64 virt, void ib_umem_odp_unmap_dma_pages(struct ib_umem_odp *umem_odp, u64 virt,
u64 bound) u64 bound)
{ {
dma_addr_t dma_addr;
dma_addr_t dma;
int idx; int idx;
u64 addr; u64 addr;
struct ib_device *dev = umem_odp->umem.ibdev; struct ib_device *dev = umem_odp->umem.ibdev;
@ -521,20 +472,16 @@ void ib_umem_odp_unmap_dma_pages(struct ib_umem_odp *umem_odp, u64 virt,
virt = max_t(u64, virt, ib_umem_start(umem_odp)); virt = max_t(u64, virt, ib_umem_start(umem_odp));
bound = min_t(u64, bound, ib_umem_end(umem_odp)); bound = min_t(u64, bound, ib_umem_end(umem_odp));
/* Note that during the run of this function, the
* notifiers_count of the MR is > 0, preventing any racing
* faults from completion. We might be racing with other
* invalidations, so we must make sure we free each page only
* once. */
for (addr = virt; addr < bound; addr += BIT(umem_odp->page_shift)) { for (addr = virt; addr < bound; addr += BIT(umem_odp->page_shift)) {
idx = (addr - ib_umem_start(umem_odp)) >> umem_odp->page_shift; idx = (addr - ib_umem_start(umem_odp)) >> umem_odp->page_shift;
if (umem_odp->page_list[idx]) { dma = umem_odp->dma_list[idx];
struct page *page = umem_odp->page_list[idx];
dma_addr_t dma = umem_odp->dma_list[idx];
dma_addr_t dma_addr = dma & ODP_DMA_ADDR_MASK;
WARN_ON(!dma_addr); /* The access flags guaranteed a valid DMA address in case was NULL */
if (dma) {
unsigned long pfn_idx = (addr - ib_umem_start(umem_odp)) >> PAGE_SHIFT;
struct page *page = hmm_pfn_to_page(umem_odp->pfn_list[pfn_idx]);
dma_addr = dma & ODP_DMA_ADDR_MASK;
ib_dma_unmap_page(dev, dma_addr, ib_dma_unmap_page(dev, dma_addr,
BIT(umem_odp->page_shift), BIT(umem_odp->page_shift),
DMA_BIDIRECTIONAL); DMA_BIDIRECTIONAL);
@ -551,7 +498,6 @@ void ib_umem_odp_unmap_dma_pages(struct ib_umem_odp *umem_odp, u64 virt,
*/ */
set_page_dirty(head_page); set_page_dirty(head_page);
} }
umem_odp->page_list[idx] = NULL;
umem_odp->dma_list[idx] = 0; umem_odp->dma_list[idx] = 0;
umem_odp->npages--; umem_odp->npages--;
} }

View File

@ -671,7 +671,6 @@ static int pagefault_real_mr(struct mlx5_ib_mr *mr, struct ib_umem_odp *odp,
{ {
int page_shift, ret, np; int page_shift, ret, np;
bool downgrade = flags & MLX5_PF_FLAGS_DOWNGRADE; bool downgrade = flags & MLX5_PF_FLAGS_DOWNGRADE;
unsigned long current_seq;
u64 access_mask; u64 access_mask;
u64 start_idx; u64 start_idx;
@ -682,25 +681,16 @@ static int pagefault_real_mr(struct mlx5_ib_mr *mr, struct ib_umem_odp *odp,
if (odp->umem.writable && !downgrade) if (odp->umem.writable && !downgrade)
access_mask |= ODP_WRITE_ALLOWED_BIT; access_mask |= ODP_WRITE_ALLOWED_BIT;
current_seq = mmu_interval_read_begin(&odp->notifier); np = ib_umem_odp_map_dma_and_lock(odp, user_va, bcnt, access_mask);
np = ib_umem_odp_map_dma_pages(odp, user_va, bcnt, access_mask,
current_seq);
if (np < 0) if (np < 0)
return np; return np;
mutex_lock(&odp->umem_mutex); /*
if (!mmu_interval_read_retry(&odp->notifier, current_seq)) { * No need to check whether the MTTs really belong to this MR, since
/* * ib_umem_odp_map_dma_and_lock already checks this.
* No need to check whether the MTTs really belong to */
* this MR, since ib_umem_odp_map_dma_pages already ret = mlx5_ib_update_xlt(mr, start_idx, np, page_shift,
* checks this. MLX5_IB_UPD_XLT_ATOMIC);
*/
ret = mlx5_ib_update_xlt(mr, start_idx, np,
page_shift, MLX5_IB_UPD_XLT_ATOMIC);
} else {
ret = -EAGAIN;
}
mutex_unlock(&odp->umem_mutex); mutex_unlock(&odp->umem_mutex);
if (ret < 0) { if (ret < 0) {

View File

@ -14,17 +14,13 @@ struct ib_umem_odp {
struct mmu_interval_notifier notifier; struct mmu_interval_notifier notifier;
struct pid *tgid; struct pid *tgid;
/* An array of the pfns included in the on-demand paging umem. */
unsigned long *pfn_list;
/* /*
* An array of the pages included in the on-demand paging umem. * An array with DMA addresses mapped for pfns in pfn_list.
* Indices of pages that are currently not mapped into the device will * The lower two bits designate access permissions.
* contain NULL. * See ODP_READ_ALLOWED_BIT and ODP_WRITE_ALLOWED_BIT.
*/
struct page **page_list;
/*
* An array of the same size as page_list, with DMA addresses mapped
* for pages the pages in page_list. The lower two bits designate
* access permissions. See ODP_READ_ALLOWED_BIT and
* ODP_WRITE_ALLOWED_BIT.
*/ */
dma_addr_t *dma_list; dma_addr_t *dma_list;
/* /*
@ -97,9 +93,8 @@ ib_umem_odp_alloc_child(struct ib_umem_odp *root_umem, unsigned long addr,
const struct mmu_interval_notifier_ops *ops); const struct mmu_interval_notifier_ops *ops);
void ib_umem_odp_release(struct ib_umem_odp *umem_odp); void ib_umem_odp_release(struct ib_umem_odp *umem_odp);
int ib_umem_odp_map_dma_pages(struct ib_umem_odp *umem_odp, u64 start_offset, int ib_umem_odp_map_dma_and_lock(struct ib_umem_odp *umem_odp, u64 start_offset,
u64 bcnt, u64 access_mask, u64 bcnt, u64 access_mask);
unsigned long current_seq);
void ib_umem_odp_unmap_dma_pages(struct ib_umem_odp *umem_odp, u64 start_offset, void ib_umem_odp_unmap_dma_pages(struct ib_umem_odp *umem_odp, u64 start_offset,
u64 bound); u64 bound);