c460ee
From d7665cf3249310c5faf87368f395b4e25cb86b48 Mon Sep 17 00:00:00 2001
c460ee
From: karthik-us <ksubrahm@redhat.com>
c460ee
Date: Thu, 15 Apr 2021 10:29:06 +0530
c460ee
Subject: [PATCH 577/584] protocol/client: don't reopen fds on which POSIX
c460ee
 locks are held after a reconnect
c460ee
c460ee
XXXXXXXXXXXXXXXXXXX
c460ee
    IMPORTANT:
c460ee
XXXXXXXXXXXXXXXXXXX
c460ee
As a best pratice, with this patch we are bumping up the op-version
c460ee
from GD_OP_VERSION_7_1 to GD_OP_VERSION_7_2 since it introduces a
c460ee
new volume option. Enabling the new option will have effect only
c460ee
after all the servers and clients are upgraded to this version.
c460ee
----------------------------------------------------------------------
c460ee
c460ee
Bricks cleanup any granted locks after a client disconnects and
c460ee
currently these locks are not healed after a reconnect. This means
c460ee
post reconnect a competing process could be granted a lock even though
c460ee
the first process which was granted locks has not unlocked. By not
c460ee
re-opening fds, subsequent operations on such fds will fail forcing
c460ee
the application to close the current fd and reopen a new one. This way
c460ee
we prevent any silent corruption.
c460ee
c460ee
A new option "client.strict-locks" is introduced to control this
c460ee
behaviour. This option is set to "off" by default.
c460ee
c460ee
> Upstream patch: https://review.gluster.org/#/c/glusterfs/+/22712/
c460ee
> Change-Id: Ieed545efea466cb5e8f5a36199aa26380c301b9e
c460ee
> Signed-off-by: Raghavendra G <rgowdapp@redhat.com>
c460ee
> updates: bz#1694920
c460ee
c460ee
BUG: 1689375
c460ee
Change-Id: Ieed545efea466cb5e8f5a36199aa26380c301b9e
c460ee
Signed-off-by: karthik-us <ksubrahm@redhat.com>
c460ee
Reviewed-on: https://code.engineering.redhat.com/gerrit/c/rhs-glusterfs/+/244909
c460ee
Tested-by: RHGS Build Bot <nigelb@redhat.com>
c460ee
Reviewed-by: Sunil Kumar Heggodu Gopala Acharya <sheggodu@redhat.com>
c460ee
Reviewed-by: Ravishankar Narayanankutty <ravishankar@redhat.com>
c460ee
---
c460ee
 libglusterfs/src/glusterfs/globals.h             |  4 +-
c460ee
 tests/bugs/bug-1694920.t                         | 63 ++++++++++++++++++++++++
c460ee
 xlators/mgmt/glusterd/src/glusterd-volume-set.c  | 14 ++++++
c460ee
 xlators/protocol/client/src/client-handshake.c   |  3 +-
c460ee
 xlators/protocol/client/src/client-helpers.c     |  5 +-
c460ee
 xlators/protocol/client/src/client-lk.c          |  2 +-
c460ee
 xlators/protocol/client/src/client-rpc-fops.c    | 45 ++++++++++++++++-
c460ee
 xlators/protocol/client/src/client-rpc-fops_v2.c | 32 +++++++++++-
c460ee
 xlators/protocol/client/src/client.c             | 13 +++++
c460ee
 xlators/protocol/client/src/client.h             | 16 ++++++
c460ee
 10 files changed, 190 insertions(+), 7 deletions(-)
c460ee
 create mode 100644 tests/bugs/bug-1694920.t
c460ee
c460ee
diff --git a/libglusterfs/src/glusterfs/globals.h b/libglusterfs/src/glusterfs/globals.h
c460ee
index 33fb023..ce2d110 100644
c460ee
--- a/libglusterfs/src/glusterfs/globals.h
c460ee
+++ b/libglusterfs/src/glusterfs/globals.h
c460ee
@@ -50,7 +50,7 @@
c460ee
     1 /* MIN is the fresh start op-version, mostly                             \
c460ee
          should not change */
c460ee
 #define GD_OP_VERSION_MAX                                                      \
c460ee
-    GD_OP_VERSION_7_1 /* MAX VERSION is the maximum                            \
c460ee
+    GD_OP_VERSION_7_2 /* MAX VERSION is the maximum                            \
c460ee
                          count in VME table, should                            \
c460ee
                          keep changing with                                    \
c460ee
                          introduction of newer                                 \
c460ee
@@ -140,6 +140,8 @@
c460ee
 
c460ee
 #define GD_OP_VERSION_7_1 70100 /* Op-version for GlusterFS 7.1 */
c460ee
 
c460ee
+#define GD_OP_VERSION_7_2 70200 /* Op-version for GlusterFS 7.2 */
c460ee
+
c460ee
 #include "glusterfs/xlator.h"
c460ee
 #include "glusterfs/options.h"
c460ee
 
c460ee
diff --git a/tests/bugs/bug-1694920.t b/tests/bugs/bug-1694920.t
c460ee
new file mode 100644
c460ee
index 0000000..5bf93c9
c460ee
--- /dev/null
c460ee
+++ b/tests/bugs/bug-1694920.t
c460ee
@@ -0,0 +1,63 @@
c460ee
+#!/bin/bash
c460ee
+
c460ee
+SCRIPT_TIMEOUT=300
c460ee
+
c460ee
+. $(dirname $0)/../include.rc
c460ee
+. $(dirname $0)/../volume.rc
c460ee
+. $(dirname $0)/../fileio.rc
c460ee
+cleanup;
c460ee
+
c460ee
+TEST glusterd;
c460ee
+TEST pidof glusterd
c460ee
+
c460ee
+TEST $CLI volume create $V0 $H0:$B0/${V0};
c460ee
+TEST $CLI volume set $V0 performance.quick-read off
c460ee
+TEST $CLI volume set $V0 performance.io-cache off
c460ee
+TEST $CLI volume set $V0 performance.write-behind off
c460ee
+TEST $CLI volume set $V0 performance.open-behind off
c460ee
+TEST $CLI volume set $V0 performance.stat-prefetch off
c460ee
+TEST $CLI volume set $V0 performance.read-ahead off
c460ee
+TEST $CLI volume start $V0
c460ee
+TEST $GFS -s $H0 --volfile-id=$V0  $M0;
c460ee
+
c460ee
+TEST touch $M0/a
c460ee
+
c460ee
+#When all bricks are up, lock and unlock should succeed
c460ee
+TEST fd1=`fd_available`
c460ee
+TEST fd_open $fd1 'w' $M0/a
c460ee
+TEST flock -x $fd1
c460ee
+TEST fd_close $fd1
c460ee
+
c460ee
+#When all bricks are down, lock/unlock should fail
c460ee
+TEST fd1=`fd_available`
c460ee
+TEST fd_open $fd1 'w' $M0/a
c460ee
+TEST $CLI volume stop $V0
c460ee
+TEST ! flock -x $fd1
c460ee
+TEST $CLI volume start $V0
c460ee
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" client_connected_status_meta $M0 $V0-client-0
c460ee
+TEST fd_close $fd1
c460ee
+
c460ee
+#When a brick goes down and comes back up operations on fd which had locks on it should succeed by default
c460ee
+TEST fd1=`fd_available`
c460ee
+TEST fd_open $fd1 'w' $M0/a
c460ee
+TEST flock -x $fd1
c460ee
+TEST $CLI volume stop $V0
c460ee
+sleep 2
c460ee
+TEST $CLI volume start $V0
c460ee
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" client_connected_status_meta $M0 $V0-client-0
c460ee
+TEST fd_write $fd1 "data"
c460ee
+TEST fd_close $fd1
c460ee
+
c460ee
+#When a brick goes down and comes back up operations on fd which had locks on it should fail when client.strict-locks is on
c460ee
+TEST $CLI volume set $V0 client.strict-locks on
c460ee
+TEST fd1=`fd_available`
c460ee
+TEST fd_open $fd1 'w' $M0/a
c460ee
+TEST flock -x $fd1
c460ee
+TEST $CLI volume stop $V0
c460ee
+sleep 2
c460ee
+TEST $CLI volume start $V0
c460ee
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" client_connected_status_meta $M0 $V0-client-0
c460ee
+TEST ! fd_write $fd1 "data"
c460ee
+TEST fd_close $fd1
c460ee
+
c460ee
+cleanup
c460ee
diff --git a/xlators/mgmt/glusterd/src/glusterd-volume-set.c b/xlators/mgmt/glusterd/src/glusterd-volume-set.c
c460ee
index c1ca190..01f3912 100644
c460ee
--- a/xlators/mgmt/glusterd/src/glusterd-volume-set.c
c460ee
+++ b/xlators/mgmt/glusterd/src/glusterd-volume-set.c
c460ee
@@ -2022,6 +2022,20 @@ struct volopt_map_entry glusterd_volopt_map[] = {
c460ee
      .value = "9",
c460ee
      .flags = VOLOPT_FLAG_CLIENT_OPT},
c460ee
 
c460ee
+    {.key = "client.strict-locks",
c460ee
+     .voltype = "protocol/client",
c460ee
+     .option = "strict-locks",
c460ee
+     .value = "off",
c460ee
+     .op_version = GD_OP_VERSION_7_2,
c460ee
+     .validate_fn = validate_boolean,
c460ee
+     .type = GLOBAL_DOC,
c460ee
+     .description = "When set, doesn't reopen saved fds after reconnect "
c460ee
+                    "if POSIX locks are held on them. Hence subsequent "
c460ee
+                    "operations on these fds will fail. This is "
c460ee
+                    "necessary for stricter lock complaince as bricks "
c460ee
+                    "cleanup any granted locks when a client "
c460ee
+                    "disconnects."},
c460ee
+
c460ee
     /* Server xlator options */
c460ee
     {.key = "network.tcp-window-size",
c460ee
      .voltype = "protocol/server",
c460ee
diff --git a/xlators/protocol/client/src/client-handshake.c b/xlators/protocol/client/src/client-handshake.c
c460ee
index 6b20d92..a12472b 100644
c460ee
--- a/xlators/protocol/client/src/client-handshake.c
c460ee
+++ b/xlators/protocol/client/src/client-handshake.c
c460ee
@@ -910,7 +910,8 @@ client_post_handshake(call_frame_t *frame, xlator_t *this)
c460ee
     {
c460ee
         list_for_each_entry_safe(fdctx, tmp, &conf->saved_fds, sfd_pos)
c460ee
         {
c460ee
-            if (fdctx->remote_fd != -1)
c460ee
+            if (fdctx->remote_fd != -1 ||
c460ee
+                (!list_empty(&fdctx->lock_list) && conf->strict_locks))
c460ee
                 continue;
c460ee
 
c460ee
             fdctx->reopen_done = client_child_up_reopen_done;
c460ee
diff --git a/xlators/protocol/client/src/client-helpers.c b/xlators/protocol/client/src/client-helpers.c
c460ee
index 53b4484..6543100 100644
c460ee
--- a/xlators/protocol/client/src/client-helpers.c
c460ee
+++ b/xlators/protocol/client/src/client-helpers.c
c460ee
@@ -410,6 +410,7 @@ client_get_remote_fd(xlator_t *this, fd_t *fd, int flags, int64_t *remote_fd)
c460ee
 {
c460ee
     clnt_fd_ctx_t *fdctx = NULL;
c460ee
     clnt_conf_t *conf = NULL;
c460ee
+    gf_boolean_t locks_held = _gf_false;
c460ee
 
c460ee
     GF_VALIDATE_OR_GOTO(this->name, fd, out);
c460ee
     GF_VALIDATE_OR_GOTO(this->name, remote_fd, out);
c460ee
@@ -431,11 +432,13 @@ client_get_remote_fd(xlator_t *this, fd_t *fd, int flags, int64_t *remote_fd)
c460ee
                 *remote_fd = -1;
c460ee
             else
c460ee
                 *remote_fd = fdctx->remote_fd;
c460ee
+
c460ee
+            locks_held = !list_empty(&fdctx->lock_list);
c460ee
         }
c460ee
     }
c460ee
     pthread_spin_unlock(&conf->fd_lock);
c460ee
 
c460ee
-    if ((flags & FALLBACK_TO_ANON_FD) && (*remote_fd == -1))
c460ee
+    if ((flags & FALLBACK_TO_ANON_FD) && (*remote_fd == -1) && (!locks_held))
c460ee
         *remote_fd = GF_ANON_FD_NO;
c460ee
 
c460ee
     return 0;
c460ee
diff --git a/xlators/protocol/client/src/client-lk.c b/xlators/protocol/client/src/client-lk.c
c460ee
index 679e198..c1fb055 100644
c460ee
--- a/xlators/protocol/client/src/client-lk.c
c460ee
+++ b/xlators/protocol/client/src/client-lk.c
c460ee
@@ -351,7 +351,7 @@ delete_granted_locks_owner(fd_t *fd, gf_lkowner_t *owner)
c460ee
 
c460ee
     list_for_each_entry_safe(lock, tmp, &fdctx->lock_list, list)
c460ee
     {
c460ee
-        if (!is_same_lkowner(&lock->owner, owner)) {
c460ee
+        if (is_same_lkowner(&lock->owner, owner)) {
c460ee
             list_del_init(&lock->list);
c460ee
             list_add_tail(&lock->list, &delete_list);
c460ee
             count++;
c460ee
diff --git a/xlators/protocol/client/src/client-rpc-fops.c b/xlators/protocol/client/src/client-rpc-fops.c
c460ee
index 1c8b31b..3110c78 100644
c460ee
--- a/xlators/protocol/client/src/client-rpc-fops.c
c460ee
+++ b/xlators/protocol/client/src/client-rpc-fops.c
c460ee
@@ -22,8 +22,18 @@ int32_t
c460ee
 client3_getspec(call_frame_t *frame, xlator_t *this, void *data);
c460ee
 rpc_clnt_prog_t clnt3_3_fop_prog;
c460ee
 
c460ee
-/* CBK */
c460ee
+int
c460ee
+client_is_setlk(int32_t cmd)
c460ee
+{
c460ee
+    if ((cmd == F_SETLK) || (cmd == F_SETLK64) || (cmd == F_SETLKW) ||
c460ee
+        (cmd == F_SETLKW64)) {
c460ee
+        return 1;
c460ee
+    }
c460ee
 
c460ee
+    return 0;
c460ee
+}
c460ee
+
c460ee
+/* CBK */
c460ee
 int
c460ee
 client3_3_symlink_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
                       void *myframe)
c460ee
@@ -816,7 +826,8 @@ client3_3_flush_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
         goto out;
c460ee
     }
c460ee
 
c460ee
-    if (rsp.op_ret >= 0 && !fd_is_anonymous(local->fd)) {
c460ee
+    if ((rsp.op_ret >= 0 || (rsp.op_errno == ENOTCONN)) &&
c460ee
+        !fd_is_anonymous(local->fd)) {
c460ee
         /* Delete all saved locks of the owner issuing flush */
c460ee
         ret = delete_granted_locks_owner(local->fd, &local->owner);
c460ee
         gf_msg_trace(this->name, 0, "deleting locks of owner (%s) returned %d",
c460ee
@@ -2388,10 +2399,12 @@ client3_3_lk_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
     int ret = 0;
c460ee
     xlator_t *this = NULL;
c460ee
     dict_t *xdata = NULL;
c460ee
+    clnt_local_t *local = NULL;
c460ee
 
c460ee
     this = THIS;
c460ee
 
c460ee
     frame = myframe;
c460ee
+    local = frame->local;
c460ee
 
c460ee
     if (-1 == req->rpc_status) {
c460ee
         rsp.op_ret = -1;
c460ee
@@ -2412,6 +2425,18 @@ client3_3_lk_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
         ret = client_post_lk(this, &rsp, &lock, &xdata);
c460ee
         if (ret < 0)
c460ee
             goto out;
c460ee
+
c460ee
+        /* Save the lock to the client lock cache to be able
c460ee
+           to recover in the case of server reboot.*/
c460ee
+
c460ee
+        if (client_is_setlk(local->cmd)) {
c460ee
+            ret = client_add_lock_for_recovery(local->fd, &lock, &local->owner,
c460ee
+                                               local->cmd);
c460ee
+            if (ret < 0) {
c460ee
+                rsp.op_ret = -1;
c460ee
+                rsp.op_errno = -ret;
c460ee
+            }
c460ee
+        }
c460ee
     }
c460ee
 
c460ee
 out:
c460ee
@@ -4263,8 +4288,16 @@ client3_3_flush(call_frame_t *frame, xlator_t *this, void *data)
c460ee
     ret = client_pre_flush(this, &req, args->fd, args->xdata);
c460ee
     if (ret) {
c460ee
         op_errno = -ret;
c460ee
+        if (op_errno == EBADF) {
c460ee
+            ret = delete_granted_locks_owner(local->fd, &local->owner);
c460ee
+            gf_msg_trace(this->name, 0,
c460ee
+                         "deleting locks of owner (%s) returned %d",
c460ee
+                         lkowner_utoa(&local->owner), ret);
c460ee
+        }
c460ee
+
c460ee
         goto unwind;
c460ee
     }
c460ee
+
c460ee
     ret = client_submit_request(this, &req, frame, conf->fops, GFS3_OP_FLUSH,
c460ee
                                 client3_3_flush_cbk, NULL,
c460ee
                                 (xdrproc_t)xdr_gfs3_flush_req);
c460ee
@@ -5199,8 +5232,16 @@ client3_3_lk(call_frame_t *frame, xlator_t *this, void *data)
c460ee
                         args->xdata);
c460ee
     if (ret) {
c460ee
         op_errno = -ret;
c460ee
+
c460ee
+        if ((op_errno == EBADF) && (args->flock->l_type == F_UNLCK) &&
c460ee
+            client_is_setlk(local->cmd)) {
c460ee
+            client_add_lock_for_recovery(local->fd, args->flock, &local->owner,
c460ee
+                                         local->cmd);
c460ee
+        }
c460ee
+
c460ee
         goto unwind;
c460ee
     }
c460ee
+
c460ee
     ret = client_submit_request(this, &req, frame, conf->fops, GFS3_OP_LK,
c460ee
                                 client3_3_lk_cbk, NULL,
c460ee
                                 (xdrproc_t)xdr_gfs3_lk_req);
c460ee
diff --git a/xlators/protocol/client/src/client-rpc-fops_v2.c b/xlators/protocol/client/src/client-rpc-fops_v2.c
c460ee
index 613dda8..954fc58 100644
c460ee
--- a/xlators/protocol/client/src/client-rpc-fops_v2.c
c460ee
+++ b/xlators/protocol/client/src/client-rpc-fops_v2.c
c460ee
@@ -723,7 +723,8 @@ client4_0_flush_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
         goto out;
c460ee
     }
c460ee
 
c460ee
-    if (rsp.op_ret >= 0 && !fd_is_anonymous(local->fd)) {
c460ee
+    if ((rsp.op_ret >= 0 || (rsp.op_errno == ENOTCONN)) &&
c460ee
+        !fd_is_anonymous(local->fd)) {
c460ee
         /* Delete all saved locks of the owner issuing flush */
c460ee
         ret = delete_granted_locks_owner(local->fd, &local->owner);
c460ee
         gf_msg_trace(this->name, 0, "deleting locks of owner (%s) returned %d",
c460ee
@@ -2193,10 +2194,12 @@ client4_0_lk_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
     int ret = 0;
c460ee
     xlator_t *this = NULL;
c460ee
     dict_t *xdata = NULL;
c460ee
+    clnt_local_t *local = NULL;
c460ee
 
c460ee
     this = THIS;
c460ee
 
c460ee
     frame = myframe;
c460ee
+    local = frame->local;
c460ee
 
c460ee
     if (-1 == req->rpc_status) {
c460ee
         rsp.op_ret = -1;
c460ee
@@ -2217,6 +2220,18 @@ client4_0_lk_cbk(struct rpc_req *req, struct iovec *iov, int count,
c460ee
         ret = client_post_lk_v2(this, &rsp, &lock, &xdata);
c460ee
         if (ret < 0)
c460ee
             goto out;
c460ee
+
c460ee
+        /* Save the lock to the client lock cache to be able
c460ee
+           to recover in the case of server reboot.*/
c460ee
+
c460ee
+        if (client_is_setlk(local->cmd)) {
c460ee
+            ret = client_add_lock_for_recovery(local->fd, &lock, &local->owner,
c460ee
+                                               local->cmd);
c460ee
+            if (ret < 0) {
c460ee
+                rsp.op_ret = -1;
c460ee
+                rsp.op_errno = -ret;
c460ee
+            }
c460ee
+        }
c460ee
     }
c460ee
 
c460ee
 out:
c460ee
@@ -3998,6 +4013,13 @@ client4_0_flush(call_frame_t *frame, xlator_t *this, void *data)
c460ee
     ret = client_pre_flush_v2(this, &req, args->fd, args->xdata);
c460ee
     if (ret) {
c460ee
         op_errno = -ret;
c460ee
+        if (op_errno == EBADF) {
c460ee
+            ret = delete_granted_locks_owner(local->fd, &local->owner);
c460ee
+            gf_msg_trace(this->name, 0,
c460ee
+                         "deleting locks of owner (%s) returned %d",
c460ee
+                         lkowner_utoa(&local->owner), ret);
c460ee
+        }
c460ee
+
c460ee
         goto unwind;
c460ee
     }
c460ee
     ret = client_submit_request(this, &req, frame, conf->fops, GFS3_OP_FLUSH,
c460ee
@@ -4771,8 +4793,16 @@ client4_0_lk(call_frame_t *frame, xlator_t *this, void *data)
c460ee
                            args->xdata);
c460ee
     if (ret) {
c460ee
         op_errno = -ret;
c460ee
+
c460ee
+        if ((op_errno == EBADF) && (args->flock->l_type == F_UNLCK) &&
c460ee
+            client_is_setlk(local->cmd)) {
c460ee
+            client_add_lock_for_recovery(local->fd, args->flock, &local->owner,
c460ee
+                                         local->cmd);
c460ee
+        }
c460ee
+
c460ee
         goto unwind;
c460ee
     }
c460ee
+
c460ee
     ret = client_submit_request(this, &req, frame, conf->fops, GFS3_OP_LK,
c460ee
                                 client4_0_lk_cbk, NULL,
c460ee
                                 (xdrproc_t)xdr_gfx_lk_req);
c460ee
diff --git a/xlators/protocol/client/src/client.c b/xlators/protocol/client/src/client.c
c460ee
index ed855ca..63c90ea 100644
c460ee
--- a/xlators/protocol/client/src/client.c
c460ee
+++ b/xlators/protocol/client/src/client.c
c460ee
@@ -2491,6 +2491,7 @@ build_client_config(xlator_t *this, clnt_conf_t *conf)
c460ee
     GF_OPTION_INIT("filter-O_DIRECT", conf->filter_o_direct, bool, out);
c460ee
 
c460ee
     GF_OPTION_INIT("send-gids", conf->send_gids, bool, out);
c460ee
+    GF_OPTION_INIT("strict-locks", conf->strict_locks, bool, out);
c460ee
 
c460ee
     conf->client_id = glusterfs_leaf_position(this);
c460ee
 
c460ee
@@ -2676,6 +2677,7 @@ reconfigure(xlator_t *this, dict_t *options)
c460ee
                      out);
c460ee
 
c460ee
     GF_OPTION_RECONF("send-gids", conf->send_gids, options, bool, out);
c460ee
+    GF_OPTION_RECONF("strict-locks", conf->strict_locks, options, bool, out);
c460ee
 
c460ee
     ret = 0;
c460ee
 out:
c460ee
@@ -3032,6 +3034,17 @@ struct volume_options options[] = {
c460ee
                     " power. Range 1-32 threads.",
c460ee
      .op_version = {GD_OP_VERSION_RHS_3_0},
c460ee
      .flags = OPT_FLAG_SETTABLE | OPT_FLAG_DOC},
c460ee
+    {.key = {"strict-locks"},
c460ee
+     .type = GF_OPTION_TYPE_BOOL,
c460ee
+     .default_value = "off",
c460ee
+     .op_version = {GD_OP_VERSION_7_2},
c460ee
+     .flags = OPT_FLAG_SETTABLE,
c460ee
+     .description = "When set, doesn't reopen saved fds after reconnect "
c460ee
+                    "if POSIX locks are held on them. Hence subsequent "
c460ee
+                    "operations on these fds will fail. This is "
c460ee
+                    "necessary for stricter lock complaince as bricks "
c460ee
+                    "cleanup any granted locks when a client "
c460ee
+                    "disconnects."},
c460ee
     {.key = {NULL}},
c460ee
 };
c460ee
 
c460ee
diff --git a/xlators/protocol/client/src/client.h b/xlators/protocol/client/src/client.h
c460ee
index f12fa61..bde3d1a 100644
c460ee
--- a/xlators/protocol/client/src/client.h
c460ee
+++ b/xlators/protocol/client/src/client.h
c460ee
@@ -235,6 +235,15 @@ typedef struct clnt_conf {
c460ee
                                       * up, disconnects can be
c460ee
                                       * logged
c460ee
                                       */
c460ee
+
c460ee
+    gf_boolean_t strict_locks; /* When set, doesn't reopen saved fds after
c460ee
+                                  reconnect if POSIX locks are held on them.
c460ee
+                                  Hence subsequent operations on these fds will
c460ee
+                                  fail. This is necessary for stricter lock
c460ee
+                                  complaince as bricks cleanup any granted
c460ee
+                                  locks when a client disconnects.
c460ee
+                               */
c460ee
+
c460ee
 } clnt_conf_t;
c460ee
 
c460ee
 typedef struct _client_fd_ctx {
c460ee
@@ -513,4 +522,11 @@ compound_request_cleanup_v2(gfx_compound_req *req);
c460ee
 void
c460ee
 client_compound_rsp_cleanup_v2(gfx_compound_rsp *rsp, int len);
c460ee
 
c460ee
+int
c460ee
+client_add_lock_for_recovery(fd_t *fd, struct gf_flock *flock,
c460ee
+                             gf_lkowner_t *owner, int32_t cmd);
c460ee
+
c460ee
+int
c460ee
+client_is_setlk(int32_t cmd);
c460ee
+
c460ee
 #endif /* !_CLIENT_H */
c460ee
-- 
c460ee
1.8.3.1
c460ee