提交 989a19b9 编写于 作者: N NeilBrown 提交者: J. Bruce Fields

sunrpc/cache: recheck cache validity after cache_defer_req

If cache_defer_req did not leave the request on a queue, then it could
possibly have waited long enough that the cache became valid.  So check the
status after the call.
Signed-off-by: NNeilBrown <neilb@suse.de>
Signed-off-by: NJ. Bruce Fields <bfields@citi.umich.edu>
上级 5c4d2639
...@@ -176,6 +176,22 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail, ...@@ -176,6 +176,22 @@ struct cache_head *sunrpc_cache_update(struct cache_detail *detail,
EXPORT_SYMBOL_GPL(sunrpc_cache_update); EXPORT_SYMBOL_GPL(sunrpc_cache_update);
static int cache_make_upcall(struct cache_detail *detail, struct cache_head *h); static int cache_make_upcall(struct cache_detail *detail, struct cache_head *h);
static inline int cache_is_valid(struct cache_detail *detail, struct cache_head *h)
{
if (!test_bit(CACHE_VALID, &h->flags) ||
h->expiry_time < get_seconds())
return -EAGAIN;
else if (detail->flush_time > h->last_refresh)
return -EAGAIN;
else {
/* entry is valid */
if (test_bit(CACHE_NEGATIVE, &h->flags))
return -ENOENT;
else
return 0;
}
}
/* /*
* This is the generic cache management routine for all * This is the generic cache management routine for all
* the authentication caches. * the authentication caches.
...@@ -184,8 +200,10 @@ static int cache_make_upcall(struct cache_detail *detail, struct cache_head *h); ...@@ -184,8 +200,10 @@ static int cache_make_upcall(struct cache_detail *detail, struct cache_head *h);
* *
* *
* Returns 0 if the cache_head can be used, or cache_puts it and returns * Returns 0 if the cache_head can be used, or cache_puts it and returns
* -EAGAIN if upcall is pending, * -EAGAIN if upcall is pending and request has been queued
* -ETIMEDOUT if upcall failed and should be retried, * -ETIMEDOUT if upcall failed or request could not be queue or
* upcall completed but item is still invalid (implying that
* the cache item has been replaced with a newer one).
* -ENOENT if cache entry was negative * -ENOENT if cache entry was negative
*/ */
int cache_check(struct cache_detail *detail, int cache_check(struct cache_detail *detail,
...@@ -195,17 +213,7 @@ int cache_check(struct cache_detail *detail, ...@@ -195,17 +213,7 @@ int cache_check(struct cache_detail *detail,
long refresh_age, age; long refresh_age, age;
/* First decide return status as best we can */ /* First decide return status as best we can */
if (!test_bit(CACHE_VALID, &h->flags) || rv = cache_is_valid(detail, h);
h->expiry_time < get_seconds())
rv = -EAGAIN;
else if (detail->flush_time > h->last_refresh)
rv = -EAGAIN;
else {
/* entry is valid */
if (test_bit(CACHE_NEGATIVE, &h->flags))
rv = -ENOENT;
else rv = 0;
}
/* now see if we want to start an upcall */ /* now see if we want to start an upcall */
refresh_age = (h->expiry_time - h->last_refresh); refresh_age = (h->expiry_time - h->last_refresh);
...@@ -238,10 +246,14 @@ int cache_check(struct cache_detail *detail, ...@@ -238,10 +246,14 @@ int cache_check(struct cache_detail *detail,
} }
} }
if (rv == -EAGAIN) if (rv == -EAGAIN) {
if (cache_defer_req(rqstp, h) != 0) if (cache_defer_req(rqstp, h) == 0) {
rv = -ETIMEDOUT; /* Request is not deferred */
rv = cache_is_valid(detail, h);
if (rv == -EAGAIN)
rv = -ETIMEDOUT;
}
}
if (rv) if (rv)
cache_put(h, detail); cache_put(h, detail);
return rv; return rv;
...@@ -560,11 +572,11 @@ static int cache_defer_req(struct cache_req *req, struct cache_head *item) ...@@ -560,11 +572,11 @@ static int cache_defer_req(struct cache_req *req, struct cache_head *item)
* or continue and drop the oldest below * or continue and drop the oldest below
*/ */
if (net_random()&1) if (net_random()&1)
return -ETIMEDOUT; return 0;
} }
dreq = req->defer(req); dreq = req->defer(req);
if (dreq == NULL) if (dreq == NULL)
return -ETIMEDOUT; return 0;
dreq->item = item; dreq->item = item;
...@@ -594,8 +606,9 @@ static int cache_defer_req(struct cache_req *req, struct cache_head *item) ...@@ -594,8 +606,9 @@ static int cache_defer_req(struct cache_req *req, struct cache_head *item)
if (!test_bit(CACHE_PENDING, &item->flags)) { if (!test_bit(CACHE_PENDING, &item->flags)) {
/* must have just been validated... */ /* must have just been validated... */
cache_revisit_request(item); cache_revisit_request(item);
return 0;
} }
return 0; return 1;
} }
static void cache_revisit_request(struct cache_head *item) static void cache_revisit_request(struct cache_head *item)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册