aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDavid Howells <dhowells@redhat.com>2019-05-09 09:15:11 -0400
committerDavid Howells <dhowells@redhat.com>2019-05-16 17:23:21 -0400
commit61c347ba551162edb1c6abfa60ce6907baf7a1af (patch)
tree9cc9f932c217021951719743663d9ff9b16ac43c
parentf642404a0436a50912c218009ccc7856d48d784c (diff)
afs: Clear AFS_VNODE_CB_PROMISED if we detect callback expiry
Fix afs_validate() to clear AFS_VNODE_CB_PROMISED on a vnode if we detect any condition that causes the callback promise to be broken implicitly, including server break (cb_s_break), volume break (cb_v_break) or callback expiry. Fixes: ae3b7361dc0e ("afs: Fix validation/callback interaction") Reported-by: Marc Dionne <marc.dionne@auristor.com> Signed-off-by: David Howells <dhowells@redhat.com>
-rw-r--r--fs/afs/inode.c14
1 files changed, 13 insertions, 1 deletions
diff --git a/fs/afs/inode.c b/fs/afs/inode.c
index ba35b4824408..37c5de793353 100644
--- a/fs/afs/inode.c
+++ b/fs/afs/inode.c
@@ -571,7 +571,7 @@ bool afs_check_validity(struct afs_vnode *vnode)
571 struct afs_server *server; 571 struct afs_server *server;
572 struct afs_volume *volume = vnode->volume; 572 struct afs_volume *volume = vnode->volume;
573 time64_t now = ktime_get_real_seconds(); 573 time64_t now = ktime_get_real_seconds();
574 bool valid; 574 bool valid, need_clear = false;
575 unsigned int cb_break, cb_s_break, cb_v_break; 575 unsigned int cb_break, cb_s_break, cb_v_break;
576 int seq = 0; 576 int seq = 0;
577 577
@@ -589,10 +589,13 @@ bool afs_check_validity(struct afs_vnode *vnode)
589 vnode->cb_v_break != cb_v_break) { 589 vnode->cb_v_break != cb_v_break) {
590 vnode->cb_s_break = cb_s_break; 590 vnode->cb_s_break = cb_s_break;
591 vnode->cb_v_break = cb_v_break; 591 vnode->cb_v_break = cb_v_break;
592 need_clear = true;
592 valid = false; 593 valid = false;
593 } else if (test_bit(AFS_VNODE_ZAP_DATA, &vnode->flags)) { 594 } else if (test_bit(AFS_VNODE_ZAP_DATA, &vnode->flags)) {
595 need_clear = true;
594 valid = false; 596 valid = false;
595 } else if (vnode->cb_expires_at - 10 <= now) { 597 } else if (vnode->cb_expires_at - 10 <= now) {
598 need_clear = true;
596 valid = false; 599 valid = false;
597 } else { 600 } else {
598 valid = true; 601 valid = true;
@@ -607,6 +610,15 @@ bool afs_check_validity(struct afs_vnode *vnode)
607 } while (need_seqretry(&vnode->cb_lock, seq)); 610 } while (need_seqretry(&vnode->cb_lock, seq));
608 611
609 done_seqretry(&vnode->cb_lock, seq); 612 done_seqretry(&vnode->cb_lock, seq);
613
614 if (need_clear) {
615 write_seqlock(&vnode->cb_lock);
616 if (cb_break == vnode->cb_break)
617 __afs_break_callback(vnode);
618 write_sequnlock(&vnode->cb_lock);
619 valid = false;
620 }
621
610 return valid; 622 return valid;
611} 623}
612 624