aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAndrew Morton <akpm@osdl.org>2005-11-07 03:58:00 -0500
committerLinus Torvalds <torvalds@g5.osdl.org>2005-11-07 10:53:24 -0500
commit4f12bb4f7715f418a9c80f89447948790f476958 (patch)
treef08c7e26b19687d26b29082e50101daf2e5a4b75
parent7fd93cf30c531fd8b014e827e7a85fcfc010b2c6 (diff)
[PATCH] slab: don't BUG on duplicated cache
slab presently goes BUG if someone tries to register an already-registered cache. But this can happen if the user accidentally loads a module which is already statically linked into the kernel. Nuking the kernel is rather a harsh reaction. Change it into a warning, and just fail the kmem_cache_alloc() attempt. If the module is well-behaved, the modprobe will fail and all is well. Notes: - Swaps the ranking of cache_chain_sem and lock_cpu_hotplug(). Doesn't seem important. Acked-by: Manfred Spraul <manfred@colorfullife.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
-rw-r--r--mm/slab.c67
1 files changed, 34 insertions, 33 deletions
diff --git a/mm/slab.c b/mm/slab.c
index 22bfb0b2ac8b..41e91794aa50 100644
--- a/mm/slab.c
+++ b/mm/slab.c
@@ -1502,6 +1502,7 @@ kmem_cache_create (const char *name, size_t size, size_t align,
1502{ 1502{
1503 size_t left_over, slab_size, ralign; 1503 size_t left_over, slab_size, ralign;
1504 kmem_cache_t *cachep = NULL; 1504 kmem_cache_t *cachep = NULL;
1505 struct list_head *p;
1505 1506
1506 /* 1507 /*
1507 * Sanity checks... these are all serious usage bugs. 1508 * Sanity checks... these are all serious usage bugs.
@@ -1516,6 +1517,35 @@ kmem_cache_create (const char *name, size_t size, size_t align,
1516 BUG(); 1517 BUG();
1517 } 1518 }
1518 1519
1520 down(&cache_chain_sem);
1521
1522 list_for_each(p, &cache_chain) {
1523 kmem_cache_t *pc = list_entry(p, kmem_cache_t, next);
1524 mm_segment_t old_fs = get_fs();
1525 char tmp;
1526 int res;
1527
1528 /*
1529 * This happens when the module gets unloaded and doesn't
1530 * destroy its slab cache and no-one else reuses the vmalloc
1531 * area of the module. Print a warning.
1532 */
1533 set_fs(KERNEL_DS);
1534 res = __get_user(tmp, pc->name);
1535 set_fs(old_fs);
1536 if (res) {
1537 printk("SLAB: cache with size %d has lost its name\n",
1538 pc->objsize);
1539 continue;
1540 }
1541
1542 if (!strcmp(pc->name,name)) {
1543 printk("kmem_cache_create: duplicate cache %s\n", name);
1544 dump_stack();
1545 goto oops;
1546 }
1547 }
1548
1519#if DEBUG 1549#if DEBUG
1520 WARN_ON(strchr(name, ' ')); /* It confuses parsers */ 1550 WARN_ON(strchr(name, ' ')); /* It confuses parsers */
1521 if ((flags & SLAB_DEBUG_INITIAL) && !ctor) { 1551 if ((flags & SLAB_DEBUG_INITIAL) && !ctor) {
@@ -1592,7 +1622,7 @@ kmem_cache_create (const char *name, size_t size, size_t align,
1592 /* Get cache's description obj. */ 1622 /* Get cache's description obj. */
1593 cachep = (kmem_cache_t *) kmem_cache_alloc(&cache_cache, SLAB_KERNEL); 1623 cachep = (kmem_cache_t *) kmem_cache_alloc(&cache_cache, SLAB_KERNEL);
1594 if (!cachep) 1624 if (!cachep)
1595 goto opps; 1625 goto oops;
1596 memset(cachep, 0, sizeof(kmem_cache_t)); 1626 memset(cachep, 0, sizeof(kmem_cache_t));
1597 1627
1598#if DEBUG 1628#if DEBUG
@@ -1686,7 +1716,7 @@ next:
1686 printk("kmem_cache_create: couldn't create cache %s.\n", name); 1716 printk("kmem_cache_create: couldn't create cache %s.\n", name);
1687 kmem_cache_free(&cache_cache, cachep); 1717 kmem_cache_free(&cache_cache, cachep);
1688 cachep = NULL; 1718 cachep = NULL;
1689 goto opps; 1719 goto oops;
1690 } 1720 }
1691 slab_size = ALIGN(cachep->num*sizeof(kmem_bufctl_t) 1721 slab_size = ALIGN(cachep->num*sizeof(kmem_bufctl_t)
1692 + sizeof(struct slab), align); 1722 + sizeof(struct slab), align);
@@ -1781,43 +1811,14 @@ next:
1781 cachep->limit = BOOT_CPUCACHE_ENTRIES; 1811 cachep->limit = BOOT_CPUCACHE_ENTRIES;
1782 } 1812 }
1783 1813
1784 /* Need the semaphore to access the chain. */
1785 down(&cache_chain_sem);
1786 {
1787 struct list_head *p;
1788 mm_segment_t old_fs;
1789
1790 old_fs = get_fs();
1791 set_fs(KERNEL_DS);
1792 list_for_each(p, &cache_chain) {
1793 kmem_cache_t *pc = list_entry(p, kmem_cache_t, next);
1794 char tmp;
1795 /* This happens when the module gets unloaded and doesn't
1796 destroy its slab cache and noone else reuses the vmalloc
1797 area of the module. Print a warning. */
1798 if (__get_user(tmp,pc->name)) {
1799 printk("SLAB: cache with size %d has lost its name\n",
1800 pc->objsize);
1801 continue;
1802 }
1803 if (!strcmp(pc->name,name)) {
1804 printk("kmem_cache_create: duplicate cache %s\n",name);
1805 up(&cache_chain_sem);
1806 unlock_cpu_hotplug();
1807 BUG();
1808 }
1809 }
1810 set_fs(old_fs);
1811 }
1812
1813 /* cache setup completed, link it into the list */ 1814 /* cache setup completed, link it into the list */
1814 list_add(&cachep->next, &cache_chain); 1815 list_add(&cachep->next, &cache_chain);
1815 up(&cache_chain_sem);
1816 unlock_cpu_hotplug(); 1816 unlock_cpu_hotplug();
1817opps: 1817oops:
1818 if (!cachep && (flags & SLAB_PANIC)) 1818 if (!cachep && (flags & SLAB_PANIC))
1819 panic("kmem_cache_create(): failed to create slab `%s'\n", 1819 panic("kmem_cache_create(): failed to create slab `%s'\n",
1820 name); 1820 name);
1821 up(&cache_chain_sem);
1821 return cachep; 1822 return cachep;
1822} 1823}
1823EXPORT_SYMBOL(kmem_cache_create); 1824EXPORT_SYMBOL(kmem_cache_create);