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: NeilBrown <neilb@suse.de> Signed-off-by: J. Bruce Fields <bfields@citi.umich.edu>
This commit is contained in:
committed by
J. Bruce Fields
parent
5c4d263903
commit
989a19b9b1
@@ -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);
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* 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,
|
|||||||
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,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
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)
|
|||||||
* 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)
|
|||||||
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)
|
||||||
|
Reference in New Issue
Block a user