diff options
author | Shaohua Li <shaohua.li@intel.com> | 2011-11-14 00:34:13 -0500 |
---|---|---|
committer | Pekka Enberg <penberg@kernel.org> | 2011-11-15 13:41:00 -0500 |
commit | 9ada19342b2441f290f0043ed7c562682c8c4ede (patch) | |
tree | 702599b3b616d4bce7415012d8d4d18c7c43e5ec | |
parent | f64ae042d94d376b54e7a343d93c48561e9d2e16 (diff) |
slub: move discard_slab out of node lock
Lockdep reports there is potential deadlock for slub node list_lock.
discard_slab() is called with the lock hold in unfreeze_partials(),
which could trigger a slab allocation, which could hold the lock again.
discard_slab() doesn't need hold the lock actually, if the slab is
already removed from partial list.
Acked-by: Christoph Lameter <cl@linux.com>
Reported-and-tested-by: Yong Zhang <yong.zhang0@gmail.com>
Reported-and-tested-by: Julie Sullivan <kernelmail.jms@gmail.com>
Signed-off-by: Shaohua Li <shaohua.li@intel.com>
Signed-off-by: Pekka Enberg <penberg@kernel.org>
-rw-r--r-- | mm/slub.c | 16 |
1 files changed, 12 insertions, 4 deletions
@@ -1862,7 +1862,7 @@ static void unfreeze_partials(struct kmem_cache *s) | |||
1862 | { | 1862 | { |
1863 | struct kmem_cache_node *n = NULL; | 1863 | struct kmem_cache_node *n = NULL; |
1864 | struct kmem_cache_cpu *c = this_cpu_ptr(s->cpu_slab); | 1864 | struct kmem_cache_cpu *c = this_cpu_ptr(s->cpu_slab); |
1865 | struct page *page; | 1865 | struct page *page, *discard_page = NULL; |
1866 | 1866 | ||
1867 | while ((page = c->partial)) { | 1867 | while ((page = c->partial)) { |
1868 | enum slab_modes { M_PARTIAL, M_FREE }; | 1868 | enum slab_modes { M_PARTIAL, M_FREE }; |
@@ -1916,14 +1916,22 @@ static void unfreeze_partials(struct kmem_cache *s) | |||
1916 | "unfreezing slab")); | 1916 | "unfreezing slab")); |
1917 | 1917 | ||
1918 | if (m == M_FREE) { | 1918 | if (m == M_FREE) { |
1919 | stat(s, DEACTIVATE_EMPTY); | 1919 | page->next = discard_page; |
1920 | discard_slab(s, page); | 1920 | discard_page = page; |
1921 | stat(s, FREE_SLAB); | ||
1922 | } | 1921 | } |
1923 | } | 1922 | } |
1924 | 1923 | ||
1925 | if (n) | 1924 | if (n) |
1926 | spin_unlock(&n->list_lock); | 1925 | spin_unlock(&n->list_lock); |
1926 | |||
1927 | while (discard_page) { | ||
1928 | page = discard_page; | ||
1929 | discard_page = discard_page->next; | ||
1930 | |||
1931 | stat(s, DEACTIVATE_EMPTY); | ||
1932 | discard_slab(s, page); | ||
1933 | stat(s, FREE_SLAB); | ||
1934 | } | ||
1927 | } | 1935 | } |
1928 | 1936 | ||
1929 | /* | 1937 | /* |