提交 18ac6185 编写于 作者: D David Howells

afs: Fix callback handling

In some circumstances, the callback interest pointer is NULL, so in such a
case we can't dereference it when checking to see if the callback is
broken.  This causes an oops in some circumstances.

Fix this by replacing the function that worked out the aggregate break
counter with one that actually does the comparison, and then make that
return true (ie. broken) if there is no callback interest as yet (ie. the
pointer is NULL).

Fixes: 68251f0a ("afs: Fix whole-volume callback handling")
Signed-off-by: NDavid Howells <dhowells@redhat.com>
上级 2feeaf84
...@@ -269,7 +269,7 @@ static void xdr_decode_AFSCallBack(struct afs_call *call, ...@@ -269,7 +269,7 @@ static void xdr_decode_AFSCallBack(struct afs_call *call,
write_seqlock(&vnode->cb_lock); write_seqlock(&vnode->cb_lock);
if (call->cb_break == afs_cb_break_sum(vnode, cbi)) { if (!afs_cb_is_broken(call->cb_break, vnode, cbi)) {
vnode->cb_version = ntohl(*bp++); vnode->cb_version = ntohl(*bp++);
cb_expiry = ntohl(*bp++); cb_expiry = ntohl(*bp++);
vnode->cb_type = ntohl(*bp++); vnode->cb_type = ntohl(*bp++);
......
...@@ -776,10 +776,13 @@ static inline unsigned int afs_calc_vnode_cb_break(struct afs_vnode *vnode) ...@@ -776,10 +776,13 @@ static inline unsigned int afs_calc_vnode_cb_break(struct afs_vnode *vnode)
return vnode->cb_break + vnode->cb_s_break + vnode->cb_v_break; return vnode->cb_break + vnode->cb_s_break + vnode->cb_v_break;
} }
static inline unsigned int afs_cb_break_sum(struct afs_vnode *vnode, static inline bool afs_cb_is_broken(unsigned int cb_break,
struct afs_cb_interest *cbi) const struct afs_vnode *vnode,
const struct afs_cb_interest *cbi)
{ {
return vnode->cb_break + cbi->server->cb_s_break + vnode->volume->cb_v_break; return !cbi || cb_break != (vnode->cb_break +
cbi->server->cb_s_break +
vnode->volume->cb_v_break);
} }
/* /*
......
...@@ -147,7 +147,8 @@ void afs_cache_permit(struct afs_vnode *vnode, struct key *key, ...@@ -147,7 +147,8 @@ void afs_cache_permit(struct afs_vnode *vnode, struct key *key,
break; break;
} }
if (cb_break != afs_cb_break_sum(vnode, vnode->cb_interest)) { if (afs_cb_is_broken(cb_break, vnode,
vnode->cb_interest)) {
changed = true; changed = true;
break; break;
} }
...@@ -177,7 +178,7 @@ void afs_cache_permit(struct afs_vnode *vnode, struct key *key, ...@@ -177,7 +178,7 @@ void afs_cache_permit(struct afs_vnode *vnode, struct key *key,
} }
} }
if (cb_break != afs_cb_break_sum(vnode, vnode->cb_interest)) if (afs_cb_is_broken(cb_break, vnode, vnode->cb_interest))
goto someone_else_changed_it; goto someone_else_changed_it;
/* We need a ref on any permits list we want to copy as we'll have to /* We need a ref on any permits list we want to copy as we'll have to
...@@ -256,7 +257,7 @@ void afs_cache_permit(struct afs_vnode *vnode, struct key *key, ...@@ -256,7 +257,7 @@ void afs_cache_permit(struct afs_vnode *vnode, struct key *key,
spin_lock(&vnode->lock); spin_lock(&vnode->lock);
zap = rcu_access_pointer(vnode->permit_cache); zap = rcu_access_pointer(vnode->permit_cache);
if (cb_break == afs_cb_break_sum(vnode, vnode->cb_interest) && if (!afs_cb_is_broken(cb_break, vnode, vnode->cb_interest) &&
zap == permits) zap == permits)
rcu_assign_pointer(vnode->permit_cache, replacement); rcu_assign_pointer(vnode->permit_cache, replacement);
else else
......
...@@ -324,7 +324,7 @@ static void xdr_decode_YFSCallBack(struct afs_call *call, ...@@ -324,7 +324,7 @@ static void xdr_decode_YFSCallBack(struct afs_call *call,
write_seqlock(&vnode->cb_lock); write_seqlock(&vnode->cb_lock);
if (call->cb_break == afs_cb_break_sum(vnode, cbi)) { if (!afs_cb_is_broken(call->cb_break, vnode, cbi)) {
cb_expiry = xdr_to_u64(xdr->expiration_time); cb_expiry = xdr_to_u64(xdr->expiration_time);
do_div(cb_expiry, 10 * 1000 * 1000); do_div(cb_expiry, 10 * 1000 * 1000);
vnode->cb_version = ntohl(xdr->version); vnode->cb_version = ntohl(xdr->version);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册