SUNRPC/cache: Allow garbage collection of invalid cache entries

If the cache entry never gets initialised, we want the garbage
collector to be able to evict it. Otherwise if the upcall daemon
fails to initialise the entry, we end up never expiring it.

Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com>
[ cel: resolved a merge conflict ]
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
This commit is contained in:
Trond Myklebust 2020-03-01 18:21:43 -05:00 committed by Chuck Lever
parent 65286b883c
commit 277f27e2f2
2 changed files with 19 additions and 20 deletions

View File

@ -209,9 +209,6 @@ static inline void cache_put(struct cache_head *h, struct cache_detail *cd)
static inline bool cache_is_expired(struct cache_detail *detail, struct cache_head *h) static inline bool cache_is_expired(struct cache_detail *detail, struct cache_head *h)
{ {
if (!test_bit(CACHE_VALID, &h->flags))
return false;
return (h->expiry_time < seconds_since_boot()) || return (h->expiry_time < seconds_since_boot()) ||
(detail->flush_time >= h->last_refresh); (detail->flush_time >= h->last_refresh);
} }

View File

@ -64,13 +64,14 @@ static struct cache_head *sunrpc_cache_find_rcu(struct cache_detail *detail,
rcu_read_lock(); rcu_read_lock();
hlist_for_each_entry_rcu(tmp, head, cache_list) { hlist_for_each_entry_rcu(tmp, head, cache_list) {
if (detail->match(tmp, key)) { if (!detail->match(tmp, key))
if (cache_is_expired(detail, tmp)) continue;
continue; if (test_bit(CACHE_VALID, &tmp->flags) &&
tmp = cache_get_rcu(tmp); cache_is_expired(detail, tmp))
rcu_read_unlock(); continue;
return tmp; tmp = cache_get_rcu(tmp);
} rcu_read_unlock();
return tmp;
} }
rcu_read_unlock(); rcu_read_unlock();
return NULL; return NULL;
@ -114,17 +115,18 @@ static struct cache_head *sunrpc_cache_add_entry(struct cache_detail *detail,
/* check if entry appeared while we slept */ /* check if entry appeared while we slept */
hlist_for_each_entry_rcu(tmp, head, cache_list, hlist_for_each_entry_rcu(tmp, head, cache_list,
lockdep_is_held(&detail->hash_lock)) { lockdep_is_held(&detail->hash_lock)) {
if (detail->match(tmp, key)) { if (!detail->match(tmp, key))
if (cache_is_expired(detail, tmp)) { continue;
sunrpc_begin_cache_remove_entry(tmp, detail); if (test_bit(CACHE_VALID, &tmp->flags) &&
freeme = tmp; cache_is_expired(detail, tmp)) {
break; sunrpc_begin_cache_remove_entry(tmp, detail);
} freeme = tmp;
cache_get(tmp); break;
spin_unlock(&detail->hash_lock);
cache_put(new, detail);
return tmp;
} }
cache_get(tmp);
spin_unlock(&detail->hash_lock);
cache_put(new, detail);
return tmp;
} }
hlist_add_head_rcu(&new->cache_list, head); hlist_add_head_rcu(&new->cache_list, head);