diff options
| author | David Howells <dhowells@redhat.com> | 2019-05-09 09:15:11 -0400 |
|---|---|---|
| committer | David Howells <dhowells@redhat.com> | 2019-05-16 17:23:21 -0400 |
| commit | 61c347ba551162edb1c6abfa60ce6907baf7a1af (patch) | |
| tree | 9cc9f932c217021951719743663d9ff9b16ac43c | |
| parent | f642404a0436a50912c218009ccc7856d48d784c (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.c | 14 |
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 | ||
