diff options
author | Yan, Zheng <zheng.z.yan@intel.com> | 2013-06-03 06:22:17 -0400 |
---|---|---|
committer | Sage Weil <sage@inktank.com> | 2013-07-03 18:32:46 -0400 |
commit | bb137f84d1d8f692233b590f7cae14abbdc1e0c1 (patch) | |
tree | 393f01adbf4b55b590f8831b5a7aa3ab96ccbf06 /fs/ceph/caps.c | |
parent | ccca4e37b1a912da3db68aee826557ea66145273 (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.c | 22 |
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, |