|
@@ -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,
|
|
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,
|
|
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)
|
|
|
|
|
|
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);
|