From: "J. Bruce Fields" <bfields-uC3wQj2KruNg9hUCZPvPmw@public.gmane.org>
To: NeilBrown <neilb-IBi9RG/b67k@public.gmane.org>
Cc: Kinglong Mee
<kinglongmee-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>,
Al Viro <viro-RmSDqhL/yNMiFSDQTTA3OLVCufUGDwFn@public.gmane.org>,
"linux-nfs-u79uwXL29TY76Z2rM5mHXA@public.gmane.org"
<linux-nfs-u79uwXL29TY76Z2rM5mHXA@public.gmane.org>,
linux-fsdevel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
Trond Myklebust
<trond.myklebust-7I+n7zu2hftEKMMhf/gKZA@public.gmane.org>
Subject: Re: [PATCH 7/9 v8] sunrpc: Switch to using hash list instead single list
Date: Wed, 29 Jul 2015 15:51:51 -0400 [thread overview]
Message-ID: <20150729195151.GD21949@fieldses.org> (raw)
In-Reply-To: <20150729121939.517008f4@noble>
On Wed, Jul 29, 2015 at 12:19:39PM +1000, NeilBrown wrote:
> On Mon, 27 Jul 2015 11:10:15 +0800 Kinglong Mee <kinglongmee-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
> wrote:
>
> > 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: Kinglong Mee <kinglongmee-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
>
> Reviewed-by: NeilBrown <neilb-IBi9RG/b67k@public.gmane.org>
Thanks, applying this and previous 2 patches.
--b.
>
> Thanks,
> NeilBrown
>
> > ---
> > include/linux/sunrpc/cache.h | 4 +--
> > net/sunrpc/cache.c | 60 +++++++++++++++++++++++---------------------
> > 2 files changed, 33 insertions(+), 31 deletions(-)
> >
> > diff --git a/include/linux/sunrpc/cache.h b/include/linux/sunrpc/cache.h
> > index 04ee5a2..03d3b4c 100644
> > --- a/include/linux/sunrpc/cache.h
> > +++ b/include/linux/sunrpc/cache.h
> > @@ -46,7 +46,7 @@
> > *
> > */
> > 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 last_refresh; /* If CACHE_PENDING, this is when upcall
> > * was sent, else this is when update was received
> > @@ -73,7 +73,7 @@ struct cache_detail_pipefs {
> > struct cache_detail {
> > struct module * owner;
> > int hash_size;
> > - struct cache_head ** hash_table;
> > + struct hlist_head * hash_table;
> > rwlock_t hash_lock;
> >
> > atomic_t inuse; /* active user-space update or lookup */
> > diff --git a/net/sunrpc/cache.c b/net/sunrpc/cache.c
> > index 673c2fa..4a2340a 100644
> > --- a/net/sunrpc/cache.c
> > +++ b/net/sunrpc/cache.c
> > @@ -44,7 +44,7 @@ static void cache_revisit_request(struct cache_head *item);
> > static void cache_init(struct cache_head *h)
> > {
> > time_t now = seconds_since_boot();
> > - h->next = NULL;
> > + INIT_HLIST_NODE(&h->cache_list);
> > h->flags = 0;
> > kref_init(&h->ref);
> > h->expiry_time = now + CACHE_NEW_EXPIRY;
> > @@ -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 *key, int hash)
> > {
> > - struct cache_head **head, **hp;
> > - struct cache_head *new = NULL, *freeme = NULL;
> > + struct cache_head *new = NULL, *freeme = NULL, *tmp = NULL;
> > + struct hlist_head *head;
> >
> > head = &detail->hash_table[hash];
> >
> > read_lock(&detail->hash_lock);
> >
> > - for (hp=head; *hp != NULL ; hp = &(*hp)->next) {
> > - struct cache_head *tmp = *hp;
> > + hlist_for_each_entry(tmp, head, cache_list) {
> > if (detail->match(tmp, key)) {
> > if (cache_is_expired(detail, tmp))
> > /* This entry is expired, we will discard it. */
> > @@ -88,12 +87,10 @@ struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail,
> > write_lock(&detail->hash_lock);
> >
> > /* check if entry appeared while we slept */
> > - for (hp=head; *hp != NULL ; hp = &(*hp)->next) {
> > - struct cache_head *tmp = *hp;
> > + hlist_for_each_entry(tmp, head, cache_list) {
> > if (detail->match(tmp, key)) {
> > if (cache_is_expired(detail, tmp)) {
> > - *hp = tmp->next;
> > - tmp->next = NULL;
> > + hlist_del_init(&tmp->cache_list);
> > detail->entries --;
> > freeme = tmp;
> > break;
> > @@ -104,8 +101,8 @@ struct cache_head *sunrpc_cache_lookup(struct cache_detail *detail,
> > return tmp;
> > }
> > }
> > - new->next = *head;
> > - *head = new;
> > +
> > + hlist_add_head(&new->cache_list, head);
> > detail->entries++;
> > cache_get(new);
> > write_unlock(&detail->hash_lock);
> > @@ -143,7 +140,6 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail,
> > * If 'old' is not VALID, we update it directly,
> > * otherwise we need to replace it
> > */
> > - struct cache_head **head;
> > struct cache_head *tmp;
> >
> > if (!test_bit(CACHE_VALID, &old->flags)) {
> > @@ -168,15 +164,13 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail,
> > }
> > cache_init(tmp);
> > detail->init(tmp, old);
> > - head = &detail->hash_table[hash];
> >
> > write_lock(&detail->hash_lock);
> > if (test_bit(CACHE_NEGATIVE, &new->flags))
> > set_bit(CACHE_NEGATIVE, &tmp->flags);
> > else
> > detail->update(tmp, new);
> > - tmp->next = *head;
> > - *head = tmp;
> > + hlist_add_head(&tmp->cache_list, &detail->hash_table[hash]);
> > detail->entries++;
> > cache_get(tmp);
> > cache_fresh_locked(tmp, new->expiry_time);
> > @@ -416,28 +410,29 @@ static int cache_clean(void)
> > /* find a non-empty bucket in the table */
> > while (current_detail &&
> > current_index < current_detail->hash_size &&
> > - current_detail->hash_table[current_index] == NULL)
> > + hlist_empty(¤t_detail->hash_table[current_index]))
> > current_index++;
> >
> > /* find a cleanable entry in the bucket and clean it, or set to next bucket */
> >
> > if (current_detail && current_index < current_detail->hash_size) {
> > - struct cache_head *ch, **cp;
> > + struct cache_head *ch = NULL;
> > struct cache_detail *d;
> > + struct hlist_head *head;
> > + struct hlist_node *tmp;
> >
> > write_lock(¤t_detail->hash_lock);
> >
> > /* Ok, now to clean this strand */
> >
> > - cp = & current_detail->hash_table[current_index];
> > - for (ch = *cp ; ch ; cp = & ch->next, ch = *cp) {
> > + head = ¤t_detail->hash_table[current_index];
> > + hlist_for_each_entry_safe(ch, tmp, head, cache_list) {
> > if (current_detail->nextcheck > ch->expiry_time)
> > current_detail->nextcheck = ch->expiry_time+1;
> > if (!cache_is_expired(current_detail, ch))
> > continue;
> >
> > - *cp = ch->next;
> > - ch->next = NULL;
> > + hlist_del_init(&ch->cache_list);
> > current_detail->entries--;
> > rv = 1;
> > break;
> > @@ -1284,7 +1279,7 @@ void *cache_seq_start(struct seq_file *m, loff_t *pos)
> > hash = n >> 32;
> > 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--)
> > return ch;
> > n &= ~((1LL<<32) - 1);
> > @@ -1292,11 +1287,12 @@ void *cache_seq_start(struct seq_file *m, loff_t *pos)
> > hash++;
> > n += 1LL<<32;
> > } while(hash < cd->hash_size &&
> > - cd->hash_table[hash]==NULL);
> > + hlist_empty(&cd->hash_table[hash]));
> > if (hash >= cd->hash_size)
> > return NULL;
> > *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);
> >
> > @@ -1308,23 +1304,25 @@ void *cache_seq_next(struct seq_file *m, void *p, loff_t *pos)
> >
> > if (p == SEQ_START_TOKEN)
> > hash = 0;
> > - else if (ch->next == NULL) {
> > + else if (ch->cache_list.next == NULL) {
> > hash++;
> > *pos += 1LL<<32;
> > } else {
> > ++*pos;
> > - return ch->next;
> > + return hlist_entry_safe(ch->cache_list.next,
> > + struct cache_head, cache_list);
> > }
> > *pos &= ~((1LL<<32) - 1);
> > while (hash < cd->hash_size &&
> > - cd->hash_table[hash] == NULL) {
> > + hlist_empty(&cd->hash_table[hash])) {
> > hash++;
> > *pos += 1LL<<32;
> > }
> > if (hash >= cd->hash_size)
> > return NULL;
> > ++*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);
> >
> > @@ -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 *cd;
> > + int i;
> >
> > cd = kmemdup(tmpl, sizeof(struct cache_detail), GFP_KERNEL);
> > if (cd == NULL)
> > 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);
> > if (cd->hash_table == NULL) {
> > kfree(cd);
> > return ERR_PTR(-ENOMEM);
> > }
> > +
> > + for (i = 0; i < cd->hash_size; i++)
> > + INIT_HLIST_HEAD(&cd->hash_table[i]);
> > cd->net = net;
> > return cd;
> > }
--
To unsubscribe from this list: send the line "unsubscribe linux-nfs" in
the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
next prev parent reply other threads:[~2015-07-29 19:51 UTC|newest]
Thread overview: 33+ messages / expand[flat|nested] mbox.gz Atom feed top
2015-07-27 3:05 [PATCH 0/9 v8] NFSD: Pin to vfsmount for nfsd exports cache Kinglong Mee
[not found] ` <55B5A012.1030006-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2015-07-27 3:06 ` [PATCH 1/9 v8] fs_pin: Initialize value for fs_pin explicitly Kinglong Mee
2015-07-29 0:25 ` NeilBrown
2015-07-29 19:41 ` J. Bruce Fields
[not found] ` <20150729194155.GC21949-uC3wQj2KruNg9hUCZPvPmw@public.gmane.org>
2015-07-29 21:48 ` NeilBrown
2015-07-30 0:36 ` J. Bruce Fields
2015-07-30 12:28 ` Kinglong Mee
2015-07-27 3:07 ` [PATCH 2/9 v8] fs_pin: Export functions for specific filesystem Kinglong Mee
2015-07-29 0:30 ` NeilBrown
2015-07-30 12:31 ` Kinglong Mee
2015-07-27 3:07 ` [PATCH 3/9 v8] path: New helpers path_get_pin/path_put_unpin for path pin Kinglong Mee
2015-07-27 3:09 ` [PATCH 6/9 v8] sunrpc/nfsd: Remove redundant code by exports seq_operations functions Kinglong Mee
2015-07-29 2:15 ` NeilBrown
2015-07-27 3:12 ` [PATCH 9/9 v8] nfsd: Allows user un-mounting filesystem where nfsd exports base on Kinglong Mee
[not found] ` <55B5A186.7040004-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2015-07-29 3:56 ` NeilBrown
2015-07-30 13:30 ` Kinglong Mee
2015-07-29 3:59 ` [PATCH] fs-pin: allow pin_remove() to be called other than from ->kill() NeilBrown
2015-08-10 11:37 ` Kinglong Mee
2015-08-18 6:07 ` Kinglong Mee
2015-08-18 6:21 ` NeilBrown
2015-08-18 6:37 ` Kinglong Mee
2015-07-27 3:08 ` [PATCH 4/9 v8] fs: New helper legitimize_mntget() for getting a legitimize mnt Kinglong Mee
[not found] ` <55B5A0B0.7060604-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2015-07-29 2:06 ` NeilBrown
2015-07-30 13:17 ` Kinglong Mee
2015-07-27 3:09 ` [PATCH 5/9 v8] sunrpc: Store cache_detail in seq_file's private directly Kinglong Mee
2015-07-29 2:11 ` NeilBrown
2015-07-27 3:10 ` [PATCH 7/9 v8] sunrpc: Switch to using hash list instead single list Kinglong Mee
2015-07-29 2:19 ` NeilBrown
2015-07-29 19:51 ` J. Bruce Fields [this message]
[not found] ` <20150729195151.GD21949-uC3wQj2KruNg9hUCZPvPmw@public.gmane.org>
2015-07-30 13:01 ` Kinglong Mee
2015-07-27 3:10 ` [PATCH 8/9 v8] sunrpc: New helper cache_delete_entry for deleting cache_head directly Kinglong Mee
[not found] ` <55B5A135.9050800-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2015-07-29 2:29 ` NeilBrown
2015-07-30 13:14 ` Kinglong Mee
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20150729195151.GD21949@fieldses.org \
--to=bfields-uc3wqj2krung9huczpvpmw@public.gmane.org \
--cc=kinglongmee-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org \
--cc=linux-fsdevel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=linux-nfs-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=neilb-IBi9RG/b67k@public.gmane.org \
--cc=trond.myklebust-7I+n7zu2hftEKMMhf/gKZA@public.gmane.org \
--cc=viro-RmSDqhL/yNMiFSDQTTA3OLVCufUGDwFn@public.gmane.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).