Commit 18093d1c authored by Herbert Xu's avatar Herbert Xu Committed by David S. Miller

rhashtable: Shrink to fit

This patch changes rhashtable_shrink to shrink to the smallest
size possible rather than halving the table.  This is needed
because with multiple rehashing we will defer shrinking until
all other rehashing is done, meaning that when we do shrink
we may be able to shrink a lot.
Signed-off-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
Acked-by: default avatarThomas Graf <tgraf@suug.ch>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 6d022949
...@@ -261,8 +261,8 @@ EXPORT_SYMBOL_GPL(rhashtable_expand); ...@@ -261,8 +261,8 @@ EXPORT_SYMBOL_GPL(rhashtable_expand);
* rhashtable_shrink - Shrink hash table while allowing concurrent lookups * rhashtable_shrink - Shrink hash table while allowing concurrent lookups
* @ht: the hash table to shrink * @ht: the hash table to shrink
* *
* This function may only be called in a context where it is safe to call * This function shrinks the hash table to fit, i.e., the smallest
* synchronize_rcu(), e.g. not within a rcu_read_lock() section. * size would not cause it to expand right away automatically.
* *
* The caller must ensure that no concurrent resizing occurs by holding * The caller must ensure that no concurrent resizing occurs by holding
* ht->mutex. * ht->mutex.
...@@ -276,10 +276,17 @@ EXPORT_SYMBOL_GPL(rhashtable_expand); ...@@ -276,10 +276,17 @@ EXPORT_SYMBOL_GPL(rhashtable_expand);
int rhashtable_shrink(struct rhashtable *ht) int rhashtable_shrink(struct rhashtable *ht)
{ {
struct bucket_table *new_tbl, *old_tbl = rht_dereference(ht->tbl, ht); struct bucket_table *new_tbl, *old_tbl = rht_dereference(ht->tbl, ht);
unsigned size = roundup_pow_of_two(atomic_read(&ht->nelems) * 3 / 2);
ASSERT_RHT_MUTEX(ht); ASSERT_RHT_MUTEX(ht);
new_tbl = bucket_table_alloc(ht, old_tbl->size / 2); if (size < ht->p.min_size)
size = ht->p.min_size;
if (old_tbl->size <= size)
return 0;
new_tbl = bucket_table_alloc(ht, size);
if (new_tbl == NULL) if (new_tbl == NULL)
return -ENOMEM; return -ENOMEM;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment