|
Luis Henriques |
59d091 |
From: "Yan, Zheng" <zyan@redhat.com>
|
|
Luis Henriques |
59d091 |
Date: Thu, 23 Nov 2017 17:47:15 +0800
|
|
Luis Henriques |
59d091 |
Subject: ceph: voluntarily drop Ax cap for requests that create new inode
|
|
Luis Henriques |
59d091 |
Git-commit: 222b7f90ba3825728fd27e9105aeee7af9576819
|
|
Luis Henriques |
59d091 |
Patch-mainline: v4.16-rc1
|
|
Luis Henriques |
59d091 |
References: FATE#324714
|
|
Luis Henriques |
59d091 |
|
|
Luis Henriques |
59d091 |
MDS need to rdlock directory inode's authlock when handling these
|
|
Luis Henriques |
59d091 |
requests. Voluntarily dropping CEPH_CAP_AUTH_EXCL avoids a cap revoke
|
|
Luis Henriques |
59d091 |
message.
|
|
Luis Henriques |
59d091 |
|
|
Luis Henriques |
59d091 |
Signed-off-by: "Yan, Zheng" <zyan@redhat.com>
|
|
Luis Henriques |
59d091 |
Signed-off-by: Ilya Dryomov <idryomov@gmail.com>
|
|
Luis Henriques |
59d091 |
Acked-by: Luis Henriques <lhenriques@suse.com>
|
|
Luis Henriques |
59d091 |
---
|
|
Luis Henriques |
59d091 |
fs/ceph/caps.c | 21 +++++++++++++++------
|
|
Luis Henriques |
59d091 |
fs/ceph/dir.c | 6 +++---
|
|
Luis Henriques |
59d091 |
fs/ceph/file.c | 2 +-
|
|
Luis Henriques |
59d091 |
3 files changed, 19 insertions(+), 10 deletions(-)
|
|
Luis Henriques |
59d091 |
|
|
Luis Henriques |
59d091 |
--- a/fs/ceph/caps.c
|
|
Luis Henriques |
59d091 |
+++ b/fs/ceph/caps.c
|
|
Luis Henriques |
59d091 |
@@ -3938,11 +3938,20 @@ int ceph_encode_inode_release(void **p,
|
|
Luis Henriques |
59d091 |
|
|
Luis Henriques |
59d091 |
cap = __get_cap_for_mds(ci, mds);
|
|
Luis Henriques |
59d091 |
if (cap && __cap_is_valid(cap)) {
|
|
Luis Henriques |
59d091 |
- if (force ||
|
|
Luis Henriques |
59d091 |
- ((cap->issued & drop) &&
|
|
Luis Henriques |
59d091 |
- (cap->issued & unless) == 0)) {
|
|
Luis Henriques |
59d091 |
- if ((cap->issued & drop) &&
|
|
Luis Henriques |
59d091 |
- (cap->issued & unless) == 0) {
|
|
Luis Henriques |
59d091 |
+ unless &= cap->issued;
|
|
Luis Henriques |
59d091 |
+ if (unless) {
|
|
Luis Henriques |
59d091 |
+ if (unless & CEPH_CAP_AUTH_EXCL)
|
|
Luis Henriques |
59d091 |
+ drop &= ~CEPH_CAP_AUTH_SHARED;
|
|
Luis Henriques |
59d091 |
+ if (unless & CEPH_CAP_LINK_EXCL)
|
|
Luis Henriques |
59d091 |
+ drop &= ~CEPH_CAP_LINK_SHARED;
|
|
Luis Henriques |
59d091 |
+ if (unless & CEPH_CAP_XATTR_EXCL)
|
|
Luis Henriques |
59d091 |
+ drop &= ~CEPH_CAP_XATTR_SHARED;
|
|
Luis Henriques |
59d091 |
+ if (unless & CEPH_CAP_FILE_EXCL)
|
|
Luis Henriques |
59d091 |
+ drop &= ~CEPH_CAP_FILE_SHARED;
|
|
Luis Henriques |
59d091 |
+ }
|
|
Luis Henriques |
59d091 |
+
|
|
Luis Henriques |
59d091 |
+ if (force || (cap->issued & drop)) {
|
|
Luis Henriques |
59d091 |
+ if (cap->issued & drop) {
|
|
Luis Henriques |
59d091 |
int wanted = __ceph_caps_wanted(ci);
|
|
Luis Henriques |
59d091 |
if ((ci->i_ceph_flags & CEPH_I_NODELAY) == 0)
|
|
Luis Henriques |
59d091 |
wanted |= cap->mds_wanted;
|
|
Luis Henriques |
59d091 |
@@ -3974,7 +3983,7 @@ int ceph_encode_inode_release(void **p,
|
|
Luis Henriques |
59d091 |
*p += sizeof(*rel);
|
|
Luis Henriques |
59d091 |
ret = 1;
|
|
Luis Henriques |
59d091 |
} else {
|
|
Luis Henriques |
59d091 |
- dout("encode_inode_release %p cap %p %s\n",
|
|
Luis Henriques |
59d091 |
+ dout("encode_inode_release %p cap %p %s (noop)\n",
|
|
Luis Henriques |
59d091 |
inode, cap, ceph_cap_string(cap->issued));
|
|
Luis Henriques |
59d091 |
}
|
|
Luis Henriques |
59d091 |
}
|
|
Luis Henriques |
59d091 |
--- a/fs/ceph/dir.c
|
|
Luis Henriques |
59d091 |
+++ b/fs/ceph/dir.c
|
|
Luis Henriques |
59d091 |
@@ -834,7 +834,7 @@ static int ceph_mknod(struct inode *dir,
|
|
Luis Henriques |
59d091 |
set_bit(CEPH_MDS_R_PARENT_LOCKED, &req->r_req_flags);
|
|
Luis Henriques |
59d091 |
req->r_args.mknod.mode = cpu_to_le32(mode);
|
|
Luis Henriques |
59d091 |
req->r_args.mknod.rdev = cpu_to_le32(rdev);
|
|
Luis Henriques |
59d091 |
- req->r_dentry_drop = CEPH_CAP_FILE_SHARED;
|
|
Luis Henriques |
59d091 |
+ req->r_dentry_drop = CEPH_CAP_FILE_SHARED | CEPH_CAP_AUTH_EXCL;
|
|
Luis Henriques |
59d091 |
req->r_dentry_unless = CEPH_CAP_FILE_EXCL;
|
|
Luis Henriques |
59d091 |
if (acls.pagelist) {
|
|
Luis Henriques |
59d091 |
req->r_pagelist = acls.pagelist;
|
|
Luis Henriques |
59d091 |
@@ -886,7 +886,7 @@ static int ceph_symlink(struct inode *di
|
|
Luis Henriques |
59d091 |
set_bit(CEPH_MDS_R_PARENT_LOCKED, &req->r_req_flags);
|
|
Luis Henriques |
59d091 |
req->r_dentry = dget(dentry);
|
|
Luis Henriques |
59d091 |
req->r_num_caps = 2;
|
|
Luis Henriques |
59d091 |
- req->r_dentry_drop = CEPH_CAP_FILE_SHARED;
|
|
Luis Henriques |
59d091 |
+ req->r_dentry_drop = CEPH_CAP_FILE_SHARED | CEPH_CAP_AUTH_EXCL;
|
|
Luis Henriques |
59d091 |
req->r_dentry_unless = CEPH_CAP_FILE_EXCL;
|
|
Luis Henriques |
59d091 |
err = ceph_mdsc_do_request(mdsc, dir, req);
|
|
Luis Henriques |
59d091 |
if (!err && !req->r_reply_info.head->is_dentry)
|
|
Luis Henriques |
59d091 |
@@ -935,7 +935,7 @@ static int ceph_mkdir(struct inode *dir,
|
|
Luis Henriques |
59d091 |
req->r_parent = dir;
|
|
Luis Henriques |
59d091 |
set_bit(CEPH_MDS_R_PARENT_LOCKED, &req->r_req_flags);
|
|
Luis Henriques |
59d091 |
req->r_args.mkdir.mode = cpu_to_le32(mode);
|
|
Luis Henriques |
59d091 |
- req->r_dentry_drop = CEPH_CAP_FILE_SHARED;
|
|
Luis Henriques |
59d091 |
+ req->r_dentry_drop = CEPH_CAP_FILE_SHARED | CEPH_CAP_AUTH_EXCL;
|
|
Luis Henriques |
59d091 |
req->r_dentry_unless = CEPH_CAP_FILE_EXCL;
|
|
Luis Henriques |
59d091 |
if (acls.pagelist) {
|
|
Luis Henriques |
59d091 |
req->r_pagelist = acls.pagelist;
|
|
Luis Henriques |
59d091 |
--- a/fs/ceph/file.c
|
|
Luis Henriques |
59d091 |
+++ b/fs/ceph/file.c
|
|
Luis Henriques |
59d091 |
@@ -395,7 +395,7 @@ int ceph_atomic_open(struct inode *dir,
|
|
Luis Henriques |
59d091 |
req->r_dentry = dget(dentry);
|
|
Luis Henriques |
59d091 |
req->r_num_caps = 2;
|
|
Luis Henriques |
59d091 |
if (flags & O_CREAT) {
|
|
Luis Henriques |
59d091 |
- req->r_dentry_drop = CEPH_CAP_FILE_SHARED;
|
|
Luis Henriques |
59d091 |
+ req->r_dentry_drop = CEPH_CAP_FILE_SHARED | CEPH_CAP_AUTH_EXCL;
|
|
Luis Henriques |
59d091 |
req->r_dentry_unless = CEPH_CAP_FILE_EXCL;
|
|
Luis Henriques |
59d091 |
if (acls.pagelist) {
|
|
Luis Henriques |
59d091 |
req->r_pagelist = acls.pagelist;
|