aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlex Williamson <alex.williamson@redhat.com>2013-07-01 10:28:58 -0400
committerAlex Williamson <alex.williamson@redhat.com>2013-07-01 10:28:58 -0400
commit8d38ef1948bd415a5cb653a5c0ec16f3402aaca1 (patch)
treea28c39a7e8f7b4855a4ff867051fd90297aad670
parent6d6768c61b39a2184bc11bb0342e4f7f35642bcc (diff)
vfio/type1: Fix leak on error path
We also don't handle unpinning zero pages as an error on other exits so we can fix that inconsistency by rolling in the next conditional return. Reported-by: Dan Carpenter <dan.carpenter@oracle.com> Signed-off-by: Alex Williamson <alex.williamson@redhat.com>
-rw-r--r--drivers/vfio/vfio_iommu_type1.c13
1 files changed, 8 insertions, 5 deletions
diff --git a/drivers/vfio/vfio_iommu_type1.c b/drivers/vfio/vfio_iommu_type1.c
index 98231d10890c..a9807dea3887 100644
--- a/drivers/vfio/vfio_iommu_type1.c
+++ b/drivers/vfio/vfio_iommu_type1.c
@@ -436,6 +436,12 @@ static int vfio_remove_dma_overlap(struct vfio_iommu *iommu, dma_addr_t start,
436 } 436 }
437 437
438 /* Split existing */ 438 /* Split existing */
439
440 /*
441 * Allocate our tracking structure early even though it may not
442 * be used. An Allocation failure later loses track of pages and
443 * is more difficult to unwind.
444 */
439 split = kzalloc(sizeof(*split), GFP_KERNEL); 445 split = kzalloc(sizeof(*split), GFP_KERNEL);
440 if (!split) 446 if (!split)
441 return -ENOMEM; 447 return -ENOMEM;
@@ -443,12 +449,9 @@ static int vfio_remove_dma_overlap(struct vfio_iommu *iommu, dma_addr_t start,
443 offset = start - dma->iova; 449 offset = start - dma->iova;
444 450
445 ret = vfio_unmap_unpin(iommu, dma, start, size); 451 ret = vfio_unmap_unpin(iommu, dma, start, size);
446 if (ret) 452 if (ret || !*size) {
447 return ret;
448
449 if (!*size) {
450 kfree(split); 453 kfree(split);
451 return -EINVAL; 454 return ret;
452 } 455 }
453 456
454 tmp = dma->size; 457 tmp = dma->size;