aboutsummaryrefslogtreecommitdiffstats
path: root/fs/ceph/caps.c
diff options
context:
space:
mode:
authorYan, Zheng <zheng.z.yan@intel.com>2013-06-03 06:22:17 -0400
committerSage Weil <sage@inktank.com>2013-07-03 18:32:46 -0400
commitbb137f84d1d8f692233b590f7cae14abbdc1e0c1 (patch)
tree393f01adbf4b55b590f8831b5a7aa3ab96ccbf06 /fs/ceph/caps.c
parentccca4e37b1a912da3db68aee826557ea66145273 (diff)
ceph: fix cap release race
ceph_encode_inode_release() can race with ceph_open() and release caps wanted by open files. So it should call __ceph_caps_wanted() to get the wanted caps. Signed-off-by: Yan, Zheng <zheng.z.yan@intel.com> Reviewed-by: Sage Weil <sage@inktank.com>
Diffstat (limited to 'fs/ceph/caps.c')
-rw-r--r--fs/ceph/caps.c22
1 files changed, 10 insertions, 12 deletions
diff --git a/fs/ceph/caps.c b/fs/ceph/caps.c
index da0f9b8a3bcb..54c290b083ab 100644
--- a/fs/ceph/caps.c
+++ b/fs/ceph/caps.c
@@ -3042,21 +3042,19 @@ int ceph_encode_inode_release(void **p, struct inode *inode,
3042 (cap->issued & unless) == 0)) { 3042 (cap->issued & unless) == 0)) {
3043 if ((cap->issued & drop) && 3043 if ((cap->issued & drop) &&
3044 (cap->issued & unless) == 0) { 3044 (cap->issued & unless) == 0) {
3045 dout("encode_inode_release %p cap %p %s -> " 3045 int wanted = __ceph_caps_wanted(ci);
3046 "%s\n", inode, cap, 3046 if ((ci->i_ceph_flags & CEPH_I_NODELAY) == 0)
3047 wanted |= cap->mds_wanted;
3048 dout("encode_inode_release %p cap %p "
3049 "%s -> %s, wanted %s -> %s\n", inode, cap,
3047 ceph_cap_string(cap->issued), 3050 ceph_cap_string(cap->issued),
3048 ceph_cap_string(cap->issued & ~drop)); 3051 ceph_cap_string(cap->issued & ~drop),
3052 ceph_cap_string(cap->mds_wanted),
3053 ceph_cap_string(wanted));
3054
3049 cap->issued &= ~drop; 3055 cap->issued &= ~drop;
3050 cap->implemented &= ~drop; 3056 cap->implemented &= ~drop;
3051 if (ci->i_ceph_flags & CEPH_I_NODELAY) { 3057 cap->mds_wanted = wanted;
3052 int wanted = __ceph_caps_wanted(ci);
3053 dout(" wanted %s -> %s (act %s)\n",
3054 ceph_cap_string(cap->mds_wanted),
3055 ceph_cap_string(cap->mds_wanted &
3056 ~wanted),
3057 ceph_cap_string(wanted));
3058 cap->mds_wanted &= wanted;
3059 }
3060 } else { 3058 } else {
3061 dout("encode_inode_release %p cap %p %s" 3059 dout("encode_inode_release %p cap %p %s"
3062 " (force)\n", inode, cap, 3060 " (force)\n", inode, cap,