Commit 0ad2d3d0 authored by Mitko Haralanov's avatar Mitko Haralanov Committed by Doug Ledford

IB/hfi1: Fix memory leak in user ExpRcv and SDMA

The driver had two memory leaks - one in the user
expected receive code and one in SDMA buffer cache.

The leak in the expected receive code only showed up
when the user/admin had set ulimit sufficiently low
and the driver did not have enough room in the cache
before hitting the limit of allowed cachable memory.

When this condition occurred, the driver returned
early signaling userland that it needed to free some
buffers to free up room in the cache.

The bug was that the driver was not cleaning up
allocated memory prior to returning early.

The leak in the SDMA buffer cache could occur (even
though it never did), when the insertion of a buffer
node in the interval RB tree failed. In this case, the
driver failed to unpin the pages of the node instead
erroneously returning success.
Reviewed-by: default avatarDean Luick <dean.luick@intel.com>
Signed-off-by: default avatarMitko Haralanov <mitko.haralanov@intel.com>
Signed-off-by: default avatarDoug Ledford <dledford@redhat.com>
parent 4787bc5e
...@@ -399,8 +399,11 @@ int hfi1_user_exp_rcv_setup(struct file *fp, struct hfi1_tid_info *tinfo) ...@@ -399,8 +399,11 @@ int hfi1_user_exp_rcv_setup(struct file *fp, struct hfi1_tid_info *tinfo)
* pages, accept the amount pinned so far and program only that. * pages, accept the amount pinned so far and program only that.
* User space knows how to deal with partially programmed buffers. * User space knows how to deal with partially programmed buffers.
*/ */
if (!hfi1_can_pin_pages(dd, fd->tid_n_pinned, npages)) if (!hfi1_can_pin_pages(dd, fd->tid_n_pinned, npages)) {
return -ENOMEM; ret = -ENOMEM;
goto bail;
}
pinned = hfi1_acquire_user_pages(vaddr, npages, true, pages); pinned = hfi1_acquire_user_pages(vaddr, npages, true, pages);
if (pinned <= 0) { if (pinned <= 0) {
ret = pinned; ret = pinned;
......
...@@ -1139,7 +1139,8 @@ static int pin_vector_pages(struct user_sdma_request *req, ...@@ -1139,7 +1139,8 @@ static int pin_vector_pages(struct user_sdma_request *req,
list_del(&node->list); list_del(&node->list);
pq->n_locked -= node->npages; pq->n_locked -= node->npages;
spin_unlock(&pq->evict_lock); spin_unlock(&pq->evict_lock);
ret = 0; unpin_vector_pages(current->mm, node->pages, 0,
node->npages);
goto bail; goto bail;
} }
} else { } else {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment