提交 129e5824 编写于 作者: K Kinglong Mee 提交者: J. Bruce Fields

sunrpc: Switch to using hash list instead single list

Switch using list_head for cache_head in cache_detail,
it is useful of remove an cache_head entry directly from cache_detail.

v8, using hash list, not head list
Signed-off-by: NKinglong Mee <kinglongmee@gmail.com>
Reviewed-by: NNeilBrown <neilb@suse.com>
Signed-off-by: NJ. Bruce Fields <bfields@redhat.com>
上级 c8c081b7
...@@ -46,7 +46,7 @@ ...@@ -46,7 +46,7 @@
* *
*/ */
struct cache_head { struct cache_head {
struct cache_head * next; struct hlist_node cache_list;
time_t expiry_time; /* After time time, don't use the data */ time_t expiry_time; /* After time time, don't use the data */
time_t last_refresh; /* If CACHE_PENDING, this is when upcall time_t last_refresh; /* If CACHE_PENDING, this is when upcall
* was sent, else this is when update was received * was sent, else this is when update was received
...@@ -73,7 +73,7 @@ struct cache_detail_pipefs { ...@@ -73,7 +73,7 @@ struct cache_detail_pipefs {
struct cache_detail { struct cache_detail {
struct module * owner; struct module * owner;
int hash_size; int hash_size;
struct cache_head ** hash_table; struct hlist_head * hash_table;
rwlock_t hash_lock; rwlock_t hash_lock;
atomic_t inuse; /* active user-space update or lookup */ atomic_t inuse; /* active user-space update or lookup */
......
...@@ -44,7 +44,7 @@ static void cache_revisit_request(struct cache_head *item); ...@@ -44,7 +44,7 @@ static void cache_revisit_request(struct cache_head *item);
static void cache_init(struct cache_head *h) static void cache_init(struct cache_head *h)
{ {
time_t now = seconds_since_boot(); time_t now = seconds_since_boot();
h->next = NULL; INIT_HLIST_NODE(&h->cache_list);
h->flags = 0; h->flags = 0;
kref_init(&h->ref); kref_init(&h->ref);
h->expiry_time = now + CACHE_NEW_EXPIRY; h->expiry_time = now + CACHE_NEW_EXPIRY;
...@@ -54,15 +54,14 @@ static void cache_init(struct cache_head *h) ...@@ -54,15 +54,14 @@ static void cache_init(struct cache_head *h)
struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail, struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail,
struct cache_head *key, int hash) struct cache_head *key, int hash)
{ {
struct cache_head **head, **hp; struct cache_head *new = NULL, *freeme = NULL, *tmp = NULL;
struct cache_head *new = NULL, *freeme = NULL; struct hlist_head *head;
head = &detail->hash_table[hash]; head = &detail->hash_table[hash];
read_lock(&detail->hash_lock); read_lock(&detail->hash_lock);
for (hp=head; *hp != NULL ; hp = &(*hp)->next) { hlist_for_each_entry(tmp, head, cache_list) {
struct cache_head *tmp = *hp;
if (detail->match(tmp, key)) { if (detail->match(tmp, key)) {
if (cache_is_expired(detail, tmp)) if (cache_is_expired(detail, tmp))
/* This entry is expired, we will discard it. */ /* This entry is expired, we will discard it. */
...@@ -88,12 +87,10 @@ struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail, ...@@ -88,12 +87,10 @@ struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail,
write_lock(&detail->hash_lock); write_lock(&detail->hash_lock);
/* check if entry appeared while we slept */ /* check if entry appeared while we slept */
for (hp=head; *hp != NULL ; hp = &(*hp)->next) { hlist_for_each_entry(tmp, head, cache_list) {
struct cache_head *tmp = *hp;
if (detail->match(tmp, key)) { if (detail->match(tmp, key)) {
if (cache_is_expired(detail, tmp)) { if (cache_is_expired(detail, tmp)) {
*hp = tmp->next; hlist_del_init(&tmp->cache_list);
tmp->next = NULL;
detail->entries --; detail->entries --;
freeme = tmp; freeme = tmp;
break; break;
...@@ -104,8 +101,8 @@ struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail, ...@@ -104,8 +101,8 @@ struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail,
return tmp; return tmp;
} }
} }
new->next = *head;
*head = new; hlist_add_head(&new->cache_list, head);
detail->entries++; detail->entries++;
cache_get(new); cache_get(new);
write_unlock(&detail->hash_lock); write_unlock(&detail->hash_lock);
...@@ -143,7 +140,6 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail, ...@@ -143,7 +140,6 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail,
* If 'old' is not VALID, we update it directly, * If 'old' is not VALID, we update it directly,
* otherwise we need to replace it * otherwise we need to replace it
*/ */
struct cache_head **head;
struct cache_head *tmp; struct cache_head *tmp;
if (!test_bit(CACHE_VALID, &old->flags)) { if (!test_bit(CACHE_VALID, &old->flags)) {
...@@ -168,15 +164,13 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail, ...@@ -168,15 +164,13 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail,
} }
cache_init(tmp); cache_init(tmp);
detail->init(tmp, old); detail->init(tmp, old);
head = &detail->hash_table[hash];
write_lock(&detail->hash_lock); write_lock(&detail->hash_lock);
if (test_bit(CACHE_NEGATIVE, &new->flags)) if (test_bit(CACHE_NEGATIVE, &new->flags))
set_bit(CACHE_NEGATIVE, &tmp->flags); set_bit(CACHE_NEGATIVE, &tmp->flags);
else else
detail->update(tmp, new); detail->update(tmp, new);
tmp->next = *head; hlist_add_head(&tmp->cache_list, &detail->hash_table[hash]);
*head = tmp;
detail->entries++; detail->entries++;
cache_get(tmp); cache_get(tmp);
cache_fresh_locked(tmp, new->expiry_time); cache_fresh_locked(tmp, new->expiry_time);
...@@ -416,28 +410,29 @@ static int cache_clean(void) ...@@ -416,28 +410,29 @@ static int cache_clean(void)
/* find a non-empty bucket in the table */ /* find a non-empty bucket in the table */
while (current_detail && while (current_detail &&
current_index < current_detail->hash_size && current_index < current_detail->hash_size &&
current_detail->hash_table[current_index] == NULL) hlist_empty(&current_detail->hash_table[current_index]))
current_index++; current_index++;
/* find a cleanable entry in the bucket and clean it, or set to next bucket */ /* find a cleanable entry in the bucket and clean it, or set to next bucket */
if (current_detail && current_index < current_detail->hash_size) { if (current_detail && current_index < current_detail->hash_size) {
struct cache_head *ch, **cp; struct cache_head *ch = NULL;
struct cache_detail *d; struct cache_detail *d;
struct hlist_head *head;
struct hlist_node *tmp;
write_lock(&current_detail->hash_lock); write_lock(&current_detail->hash_lock);
/* Ok, now to clean this strand */ /* Ok, now to clean this strand */
cp = & current_detail->hash_table[current_index]; head = &current_detail->hash_table[current_index];
for (ch = *cp ; ch ; cp = & ch->next, ch = *cp) { hlist_for_each_entry_safe(ch, tmp, head, cache_list) {
if (current_detail->nextcheck > ch->expiry_time) if (current_detail->nextcheck > ch->expiry_time)
current_detail->nextcheck = ch->expiry_time+1; current_detail->nextcheck = ch->expiry_time+1;
if (!cache_is_expired(current_detail, ch)) if (!cache_is_expired(current_detail, ch))
continue; continue;
*cp = ch->next; hlist_del_init(&ch->cache_list);
ch->next = NULL;
current_detail->entries--; current_detail->entries--;
rv = 1; rv = 1;
break; break;
...@@ -1284,7 +1279,7 @@ void *cache_seq_start(struct seq_file *m, loff_t *pos) ...@@ -1284,7 +1279,7 @@ void *cache_seq_start(struct seq_file *m, loff_t *pos)
hash = n >> 32; hash = n >> 32;
entry = n & ((1LL<<32) - 1); entry = n & ((1LL<<32) - 1);
for (ch=cd->hash_table[hash]; ch; ch=ch->next) hlist_for_each_entry(ch, &cd->hash_table[hash], cache_list)
if (!entry--) if (!entry--)
return ch; return ch;
n &= ~((1LL<<32) - 1); n &= ~((1LL<<32) - 1);
...@@ -1292,11 +1287,12 @@ void *cache_seq_start(struct seq_file *m, loff_t *pos) ...@@ -1292,11 +1287,12 @@ void *cache_seq_start(struct seq_file *m, loff_t *pos)
hash++; hash++;
n += 1LL<<32; n += 1LL<<32;
} while(hash < cd->hash_size && } while(hash < cd->hash_size &&
cd->hash_table[hash]==NULL); hlist_empty(&cd->hash_table[hash]));
if (hash >= cd->hash_size) if (hash >= cd->hash_size)
return NULL; return NULL;
*pos = n+1; *pos = n+1;
return cd->hash_table[hash]; return hlist_entry_safe(cd->hash_table[hash].first,
struct cache_head, cache_list);
} }
EXPORT_SYMBOL_GPL(cache_seq_start); EXPORT_SYMBOL_GPL(cache_seq_start);
...@@ -1308,23 +1304,25 @@ void *cache_seq_next(struct seq_file *m, void *p, loff_t *pos) ...@@ -1308,23 +1304,25 @@ void *cache_seq_next(struct seq_file *m, void *p, loff_t *pos)
if (p == SEQ_START_TOKEN) if (p == SEQ_START_TOKEN)
hash = 0; hash = 0;
else if (ch->next == NULL) { else if (ch->cache_list.next == NULL) {
hash++; hash++;
*pos += 1LL<<32; *pos += 1LL<<32;
} else { } else {
++*pos; ++*pos;
return ch->next; return hlist_entry_safe(ch->cache_list.next,
struct cache_head, cache_list);
} }
*pos &= ~((1LL<<32) - 1); *pos &= ~((1LL<<32) - 1);
while (hash < cd->hash_size && while (hash < cd->hash_size &&
cd->hash_table[hash] == NULL) { hlist_empty(&cd->hash_table[hash])) {
hash++; hash++;
*pos += 1LL<<32; *pos += 1LL<<32;
} }
if (hash >= cd->hash_size) if (hash >= cd->hash_size)
return NULL; return NULL;
++*pos; ++*pos;
return cd->hash_table[hash]; return hlist_entry_safe(cd->hash_table[hash].first,
struct cache_head, cache_list);
} }
EXPORT_SYMBOL_GPL(cache_seq_next); EXPORT_SYMBOL_GPL(cache_seq_next);
...@@ -1666,17 +1664,21 @@ EXPORT_SYMBOL_GPL(cache_unregister_net); ...@@ -1666,17 +1664,21 @@ EXPORT_SYMBOL_GPL(cache_unregister_net);
struct cache_detail *cache_create_net(struct cache_detail *tmpl, struct net *net) struct cache_detail *cache_create_net(struct cache_detail *tmpl, struct net *net)
{ {
struct cache_detail *cd; struct cache_detail *cd;
int i;
cd = kmemdup(tmpl, sizeof(struct cache_detail), GFP_KERNEL); cd = kmemdup(tmpl, sizeof(struct cache_detail), GFP_KERNEL);
if (cd == NULL) if (cd == NULL)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
cd->hash_table = kzalloc(cd->hash_size * sizeof(struct cache_head *), cd->hash_table = kzalloc(cd->hash_size * sizeof(struct hlist_head),
GFP_KERNEL); GFP_KERNEL);
if (cd->hash_table == NULL) { if (cd->hash_table == NULL) {
kfree(cd); kfree(cd);
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
} }
for (i = 0; i < cd->hash_size; i++)
INIT_HLIST_HEAD(&cd->hash_table[i]);
cd->net = net; cd->net = net;
return cd; return cd;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册