diff options
author | David Rientjes <rientjes@google.com> | 2009-09-21 20:04:31 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-09-22 10:17:47 -0400 |
commit | 4af5a2f770cc8575840ccb1514ec76ecb592985c (patch) | |
tree | 7af18234d7d171a946925e65e987fd51648be3aa /lib | |
parent | 19da3dd157f8db6fe727ff268dab4791d55a6371 (diff) |
flex_array: add flex_array_shrink function
Add a new function to the flex_array API:
int flex_array_shrink(struct flex_array *fa)
This function will free all unused second-level pages. Since elements are
now poisoned if they are not allocated with __GFP_ZERO, it's possible to
identify parts that consist solely of unused elements.
flex_array_shrink() returns the number of pages freed.
Signed-off-by: David Rientjes <rientjes@google.com>
Cc: Dave Hansen <dave@linux.vnet.ibm.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'lib')
-rw-r--r-- | lib/flex_array.c | 40 |
1 files changed, 40 insertions, 0 deletions
diff --git a/lib/flex_array.c b/lib/flex_array.c index e22d0e9776aa..1b03bb553410 100644 --- a/lib/flex_array.c +++ b/lib/flex_array.c | |||
@@ -291,3 +291,43 @@ void *flex_array_get(struct flex_array *fa, unsigned int element_nr) | |||
291 | } | 291 | } |
292 | return &part->elements[index_inside_part(fa, element_nr)]; | 292 | return &part->elements[index_inside_part(fa, element_nr)]; |
293 | } | 293 | } |
294 | |||
295 | static int part_is_free(struct flex_array_part *part) | ||
296 | { | ||
297 | int i; | ||
298 | |||
299 | for (i = 0; i < sizeof(struct flex_array_part); i++) | ||
300 | if (part->elements[i] != FLEX_ARRAY_FREE) | ||
301 | return 0; | ||
302 | return 1; | ||
303 | } | ||
304 | |||
305 | /** | ||
306 | * flex_array_shrink - free unused second-level pages | ||
307 | * | ||
308 | * Frees all second-level pages that consist solely of unused | ||
309 | * elements. Returns the number of pages freed. | ||
310 | * | ||
311 | * Locking must be provided by the caller. | ||
312 | */ | ||
313 | int flex_array_shrink(struct flex_array *fa) | ||
314 | { | ||
315 | struct flex_array_part *part; | ||
316 | int max_part = nr_base_part_ptrs(); | ||
317 | int part_nr; | ||
318 | int ret = 0; | ||
319 | |||
320 | if (elements_fit_in_base(fa)) | ||
321 | return ret; | ||
322 | for (part_nr = 0; part_nr < max_part; part_nr++) { | ||
323 | part = fa->parts[part_nr]; | ||
324 | if (!part) | ||
325 | continue; | ||
326 | if (part_is_free(part)) { | ||
327 | fa->parts[part_nr] = NULL; | ||
328 | kfree(part); | ||
329 | ret++; | ||
330 | } | ||
331 | } | ||
332 | return ret; | ||
333 | } | ||