diff options
Diffstat (limited to 'mm/vmalloc.c')
-rw-r--r-- | mm/vmalloc.c | 13 |
1 files changed, 12 insertions, 1 deletions
diff --git a/mm/vmalloc.c b/mm/vmalloc.c index fb6f59935fb2..af58324c361a 100644 --- a/mm/vmalloc.c +++ b/mm/vmalloc.c | |||
@@ -333,6 +333,7 @@ static struct vmap_area *alloc_vmap_area(unsigned long size, | |||
333 | unsigned long addr; | 333 | unsigned long addr; |
334 | int purged = 0; | 334 | int purged = 0; |
335 | 335 | ||
336 | BUG_ON(!size); | ||
336 | BUG_ON(size & ~PAGE_MASK); | 337 | BUG_ON(size & ~PAGE_MASK); |
337 | 338 | ||
338 | va = kmalloc_node(sizeof(struct vmap_area), | 339 | va = kmalloc_node(sizeof(struct vmap_area), |
@@ -344,6 +345,9 @@ retry: | |||
344 | addr = ALIGN(vstart, align); | 345 | addr = ALIGN(vstart, align); |
345 | 346 | ||
346 | spin_lock(&vmap_area_lock); | 347 | spin_lock(&vmap_area_lock); |
348 | if (addr + size - 1 < addr) | ||
349 | goto overflow; | ||
350 | |||
347 | /* XXX: could have a last_hole cache */ | 351 | /* XXX: could have a last_hole cache */ |
348 | n = vmap_area_root.rb_node; | 352 | n = vmap_area_root.rb_node; |
349 | if (n) { | 353 | if (n) { |
@@ -375,6 +379,8 @@ retry: | |||
375 | 379 | ||
376 | while (addr + size > first->va_start && addr + size <= vend) { | 380 | while (addr + size > first->va_start && addr + size <= vend) { |
377 | addr = ALIGN(first->va_end + PAGE_SIZE, align); | 381 | addr = ALIGN(first->va_end + PAGE_SIZE, align); |
382 | if (addr + size - 1 < addr) | ||
383 | goto overflow; | ||
378 | 384 | ||
379 | n = rb_next(&first->rb_node); | 385 | n = rb_next(&first->rb_node); |
380 | if (n) | 386 | if (n) |
@@ -385,6 +391,7 @@ retry: | |||
385 | } | 391 | } |
386 | found: | 392 | found: |
387 | if (addr + size > vend) { | 393 | if (addr + size > vend) { |
394 | overflow: | ||
388 | spin_unlock(&vmap_area_lock); | 395 | spin_unlock(&vmap_area_lock); |
389 | if (!purged) { | 396 | if (!purged) { |
390 | purge_vmap_area_lazy(); | 397 | purge_vmap_area_lazy(); |
@@ -508,6 +515,7 @@ static void __purge_vmap_area_lazy(unsigned long *start, unsigned long *end, | |||
508 | static DEFINE_SPINLOCK(purge_lock); | 515 | static DEFINE_SPINLOCK(purge_lock); |
509 | LIST_HEAD(valist); | 516 | LIST_HEAD(valist); |
510 | struct vmap_area *va; | 517 | struct vmap_area *va; |
518 | struct vmap_area *n_va; | ||
511 | int nr = 0; | 519 | int nr = 0; |
512 | 520 | ||
513 | /* | 521 | /* |
@@ -547,7 +555,7 @@ static void __purge_vmap_area_lazy(unsigned long *start, unsigned long *end, | |||
547 | 555 | ||
548 | if (nr) { | 556 | if (nr) { |
549 | spin_lock(&vmap_area_lock); | 557 | spin_lock(&vmap_area_lock); |
550 | list_for_each_entry(va, &valist, purge_list) | 558 | list_for_each_entry_safe(va, n_va, &valist, purge_list) |
551 | __free_vmap_area(va); | 559 | __free_vmap_area(va); |
552 | spin_unlock(&vmap_area_lock); | 560 | spin_unlock(&vmap_area_lock); |
553 | } | 561 | } |
@@ -1347,6 +1355,7 @@ EXPORT_SYMBOL(vfree); | |||
1347 | void vunmap(const void *addr) | 1355 | void vunmap(const void *addr) |
1348 | { | 1356 | { |
1349 | BUG_ON(in_interrupt()); | 1357 | BUG_ON(in_interrupt()); |
1358 | might_sleep(); | ||
1350 | __vunmap(addr, 0); | 1359 | __vunmap(addr, 0); |
1351 | } | 1360 | } |
1352 | EXPORT_SYMBOL(vunmap); | 1361 | EXPORT_SYMBOL(vunmap); |
@@ -1366,6 +1375,8 @@ void *vmap(struct page **pages, unsigned int count, | |||
1366 | { | 1375 | { |
1367 | struct vm_struct *area; | 1376 | struct vm_struct *area; |
1368 | 1377 | ||
1378 | might_sleep(); | ||
1379 | |||
1369 | if (count > num_physpages) | 1380 | if (count > num_physpages) |
1370 | return NULL; | 1381 | return NULL; |
1371 | 1382 | ||