Commit dc4716d7 authored by Gerd Hoffmann's avatar Gerd Hoffmann

udmabuf: rework limits

Create variable for the list length limit.  Serves as documentation,
also allows to make it a module parameter if needed.

Also add a total size limit.
Signed-off-by: default avatarGerd Hoffmann <kraxel@redhat.com>
Acked-by: default avatarDaniel Vetter <daniel.vetter@ffwll.ch>
Link: http://patchwork.freedesktop.org/patch/msgid/20180911134216.9760-8-kraxel@redhat.com
parent 68d2f70e
...@@ -12,6 +12,9 @@ ...@@ -12,6 +12,9 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/udmabuf.h> #include <linux/udmabuf.h>
static const u32 list_limit = 1024; /* udmabuf_create_list->count limit */
static const size_t size_limit_mb = 64; /* total dmabuf size, in megabytes */
struct udmabuf { struct udmabuf {
pgoff_t pagecount; pgoff_t pagecount;
struct page **pages; struct page **pages;
...@@ -123,7 +126,7 @@ static long udmabuf_create(const struct udmabuf_create_list *head, ...@@ -123,7 +126,7 @@ static long udmabuf_create(const struct udmabuf_create_list *head,
struct file *memfd = NULL; struct file *memfd = NULL;
struct udmabuf *ubuf; struct udmabuf *ubuf;
struct dma_buf *buf; struct dma_buf *buf;
pgoff_t pgoff, pgcnt, pgidx, pgbuf; pgoff_t pgoff, pgcnt, pgidx, pgbuf, pglimit;
struct page *page; struct page *page;
int seals, ret = -EINVAL; int seals, ret = -EINVAL;
u32 i, flags; u32 i, flags;
...@@ -132,12 +135,15 @@ static long udmabuf_create(const struct udmabuf_create_list *head, ...@@ -132,12 +135,15 @@ static long udmabuf_create(const struct udmabuf_create_list *head,
if (!ubuf) if (!ubuf)
return -ENOMEM; return -ENOMEM;
pglimit = (size_limit_mb * 1024 * 1024) >> PAGE_SHIFT;
for (i = 0; i < head->count; i++) { for (i = 0; i < head->count; i++) {
if (!IS_ALIGNED(list[i].offset, PAGE_SIZE)) if (!IS_ALIGNED(list[i].offset, PAGE_SIZE))
goto err_free_ubuf; goto err_free_ubuf;
if (!IS_ALIGNED(list[i].size, PAGE_SIZE)) if (!IS_ALIGNED(list[i].size, PAGE_SIZE))
goto err_free_ubuf; goto err_free_ubuf;
ubuf->pagecount += list[i].size >> PAGE_SHIFT; ubuf->pagecount += list[i].size >> PAGE_SHIFT;
if (ubuf->pagecount > pglimit)
goto err_free_ubuf;
} }
ubuf->pages = kmalloc_array(ubuf->pagecount, sizeof(struct page *), ubuf->pages = kmalloc_array(ubuf->pagecount, sizeof(struct page *),
GFP_KERNEL); GFP_KERNEL);
...@@ -227,7 +233,7 @@ static long udmabuf_ioctl_create_list(struct file *filp, unsigned long arg) ...@@ -227,7 +233,7 @@ static long udmabuf_ioctl_create_list(struct file *filp, unsigned long arg)
if (copy_from_user(&head, (void __user *)arg, sizeof(head))) if (copy_from_user(&head, (void __user *)arg, sizeof(head)))
return -EFAULT; return -EFAULT;
if (head.count > 1024) if (head.count > list_limit)
return -EINVAL; return -EINVAL;
lsize = sizeof(struct udmabuf_create_item) * head.count; lsize = sizeof(struct udmabuf_create_item) * head.count;
list = memdup_user((void __user *)(arg + sizeof(head)), lsize); list = memdup_user((void __user *)(arg + sizeof(head)), lsize);
......
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