diff options
-rw-r--r-- | mm/slub.c | 57 |
1 files changed, 32 insertions, 25 deletions
@@ -2127,6 +2127,37 @@ static inline void *new_slab_objects(struct kmem_cache *s, gfp_t flags, | |||
2127 | } | 2127 | } |
2128 | 2128 | ||
2129 | /* | 2129 | /* |
2130 | * Check the page->freelist of a page and either transfer the freelist to the per cpu freelist | ||
2131 | * or deactivate the page. | ||
2132 | * | ||
2133 | * The page is still frozen if the return value is not NULL. | ||
2134 | * | ||
2135 | * If this function returns NULL then the page has been unfrozen. | ||
2136 | */ | ||
2137 | static inline void *get_freelist(struct kmem_cache *s, struct page *page) | ||
2138 | { | ||
2139 | struct page new; | ||
2140 | unsigned long counters; | ||
2141 | void *freelist; | ||
2142 | |||
2143 | do { | ||
2144 | freelist = page->freelist; | ||
2145 | counters = page->counters; | ||
2146 | new.counters = counters; | ||
2147 | VM_BUG_ON(!new.frozen); | ||
2148 | |||
2149 | new.inuse = page->objects; | ||
2150 | new.frozen = freelist != NULL; | ||
2151 | |||
2152 | } while (!cmpxchg_double_slab(s, page, | ||
2153 | freelist, counters, | ||
2154 | NULL, new.counters, | ||
2155 | "get_freelist")); | ||
2156 | |||
2157 | return freelist; | ||
2158 | } | ||
2159 | |||
2160 | /* | ||
2130 | * Slow path. The lockless freelist is empty or we need to perform | 2161 | * Slow path. The lockless freelist is empty or we need to perform |
2131 | * debugging duties. | 2162 | * debugging duties. |
2132 | * | 2163 | * |
@@ -2147,8 +2178,6 @@ static void *__slab_alloc(struct kmem_cache *s, gfp_t gfpflags, int node, | |||
2147 | { | 2178 | { |
2148 | void **object; | 2179 | void **object; |
2149 | unsigned long flags; | 2180 | unsigned long flags; |
2150 | struct page new; | ||
2151 | unsigned long counters; | ||
2152 | 2181 | ||
2153 | local_irq_save(flags); | 2182 | local_irq_save(flags); |
2154 | #ifdef CONFIG_PREEMPT | 2183 | #ifdef CONFIG_PREEMPT |
@@ -2176,29 +2205,7 @@ redo: | |||
2176 | 2205 | ||
2177 | stat(s, ALLOC_SLOWPATH); | 2206 | stat(s, ALLOC_SLOWPATH); |
2178 | 2207 | ||
2179 | do { | 2208 | object = get_freelist(s, c->page); |
2180 | object = c->page->freelist; | ||
2181 | counters = c->page->counters; | ||
2182 | new.counters = counters; | ||
2183 | VM_BUG_ON(!new.frozen); | ||
2184 | |||
2185 | /* | ||
2186 | * If there is no object left then we use this loop to | ||
2187 | * deactivate the slab which is simple since no objects | ||
2188 | * are left in the slab and therefore we do not need to | ||
2189 | * put the page back onto the partial list. | ||
2190 | * | ||
2191 | * If there are objects left then we retrieve them | ||
2192 | * and use them to refill the per cpu queue. | ||
2193 | */ | ||
2194 | |||
2195 | new.inuse = c->page->objects; | ||
2196 | new.frozen = object != NULL; | ||
2197 | |||
2198 | } while (!__cmpxchg_double_slab(s, c->page, | ||
2199 | object, counters, | ||
2200 | NULL, new.counters, | ||
2201 | "__slab_alloc")); | ||
2202 | 2209 | ||
2203 | if (!object) { | 2210 | if (!object) { |
2204 | c->page = NULL; | 2211 | c->page = NULL; |