e3c68b
From 6c004c6c8b8f98f56e186740881520b8364e6f85 Mon Sep 17 00:00:00 2001
e3c68b
From: Atin Mukherjee <amukherj@redhat.com>
e3c68b
Date: Mon, 18 Mar 2019 16:08:04 +0530
e3c68b
Subject: [PATCH 45/52] glusterd: fix txn-id mem leak
e3c68b
e3c68b
This commit ensures the following:
e3c68b
1. Don't send commit op request to the remote nodes when gluster v
e3c68b
status all is executed as for the status all transaction the local
e3c68b
commit gets the name of the volumes and remote commit ops are
e3c68b
technically a no-op. So no need for additional rpc requests.
e3c68b
2. In op state machine flow, if the transaction is in staged state and
e3c68b
op_info.skip_locking is true, then no need to set the txn id in the
e3c68b
priv->glusterd_txn_opinfo dictionary which never gets freed.
e3c68b
e3c68b
> Fixes: bz#1691164
e3c68b
> Change-Id: Ib6a9300ea29633f501abac2ba53fb72ff648c822
e3c68b
> Signed-off-by: Atin Mukherjee <amukherj@redhat.com>
e3c68b
e3c68b
upstream patch: https://review.gluster.org/#/c/glusterfs/+/22388/
e3c68b
e3c68b
BUG: 1670415
e3c68b
Change-Id: Ib6a9300ea29633f501abac2ba53fb72ff648c822
e3c68b
Signed-off-by: Sanju Rakonde <srakonde@redhat.com>
e3c68b
Reviewed-on: https://code.engineering.redhat.com/gerrit/166449
e3c68b
Tested-by: RHGS Build Bot <nigelb@redhat.com>
e3c68b
Reviewed-by: Sunil Kumar Heggodu Gopala Acharya <sheggodu@redhat.com>
e3c68b
---
e3c68b
 xlators/mgmt/glusterd/src/glusterd-op-sm.c  | 26 ++++++++++++++++++++------
e3c68b
 xlators/mgmt/glusterd/src/glusterd-syncop.c | 16 ++++++++++++++++
e3c68b
 2 files changed, 36 insertions(+), 6 deletions(-)
e3c68b
e3c68b
diff --git a/xlators/mgmt/glusterd/src/glusterd-op-sm.c b/xlators/mgmt/glusterd/src/glusterd-op-sm.c
e3c68b
index cbbb5d9..12d857a 100644
e3c68b
--- a/xlators/mgmt/glusterd/src/glusterd-op-sm.c
e3c68b
+++ b/xlators/mgmt/glusterd/src/glusterd-op-sm.c
e3c68b
@@ -5652,6 +5652,9 @@ glusterd_op_ac_stage_op(glusterd_op_sm_event_t *event, void *ctx)
e3c68b
     dict_t *dict = NULL;
e3c68b
     xlator_t *this = NULL;
e3c68b
     uuid_t *txn_id = NULL;
e3c68b
+    glusterd_op_info_t txn_op_info = {
e3c68b
+        {0},
e3c68b
+    };
e3c68b
 
e3c68b
     this = THIS;
e3c68b
     GF_ASSERT(this);
e3c68b
@@ -5686,6 +5689,7 @@ glusterd_op_ac_stage_op(glusterd_op_sm_event_t *event, void *ctx)
e3c68b
         ret = -1;
e3c68b
         goto out;
e3c68b
     }
e3c68b
+    ret = glusterd_get_txn_opinfo(&event->txn_id, &txn_op_info);
e3c68b
 
e3c68b
     ret = dict_set_bin(rsp_dict, "transaction_id", txn_id, sizeof(*txn_id));
e3c68b
     if (ret) {
e3c68b
@@ -5704,6 +5708,12 @@ out:
e3c68b
 
e3c68b
     gf_msg_debug(this->name, 0, "Returning with %d", ret);
e3c68b
 
e3c68b
+    /* for no volname transactions, the txn_opinfo needs to be cleaned up
e3c68b
+     * as there's no unlock event triggered
e3c68b
+     */
e3c68b
+    if (txn_op_info.skip_locking)
e3c68b
+        ret = glusterd_clear_txn_opinfo(txn_id);
e3c68b
+
e3c68b
     if (rsp_dict)
e3c68b
         dict_unref(rsp_dict);
e3c68b
 
e3c68b
@@ -8159,12 +8169,16 @@ glusterd_op_sm()
e3c68b
                            "Unable to clear "
e3c68b
                            "transaction's opinfo");
e3c68b
             } else {
e3c68b
-                ret = glusterd_set_txn_opinfo(&event->txn_id, &opinfo);
e3c68b
-                if (ret)
e3c68b
-                    gf_msg(this->name, GF_LOG_ERROR, 0,
e3c68b
-                           GD_MSG_TRANS_OPINFO_SET_FAIL,
e3c68b
-                           "Unable to set "
e3c68b
-                           "transaction's opinfo");
e3c68b
+                if (!(event_type == GD_OP_EVENT_STAGE_OP &&
e3c68b
+                      opinfo.state.state == GD_OP_STATE_STAGED &&
e3c68b
+                      opinfo.skip_locking)) {
e3c68b
+                    ret = glusterd_set_txn_opinfo(&event->txn_id, &opinfo);
e3c68b
+                    if (ret)
e3c68b
+                        gf_msg(this->name, GF_LOG_ERROR, 0,
e3c68b
+                               GD_MSG_TRANS_OPINFO_SET_FAIL,
e3c68b
+                               "Unable to set "
e3c68b
+                               "transaction's opinfo");
e3c68b
+                }
e3c68b
             }
e3c68b
 
e3c68b
             glusterd_destroy_op_event_ctx(event);
e3c68b
diff --git a/xlators/mgmt/glusterd/src/glusterd-syncop.c b/xlators/mgmt/glusterd/src/glusterd-syncop.c
e3c68b
index 1741cf8..618d8bc 100644
e3c68b
--- a/xlators/mgmt/glusterd/src/glusterd-syncop.c
e3c68b
+++ b/xlators/mgmt/glusterd/src/glusterd-syncop.c
e3c68b
@@ -1392,6 +1392,8 @@ gd_commit_op_phase(glusterd_op_t op, dict_t *op_ctx, dict_t *req_dict,
e3c68b
     char *errstr = NULL;
e3c68b
     struct syncargs args = {0};
e3c68b
     int type = GF_QUOTA_OPTION_TYPE_NONE;
e3c68b
+    uint32_t cmd = 0;
e3c68b
+    gf_boolean_t origin_glusterd = _gf_false;
e3c68b
 
e3c68b
     this = THIS;
e3c68b
     GF_ASSERT(this);
e3c68b
@@ -1449,6 +1451,20 @@ commit_done:
e3c68b
     gd_syncargs_init(&args, op_ctx);
e3c68b
     synctask_barrier_init((&args));
e3c68b
     peer_cnt = 0;
e3c68b
+    origin_glusterd = is_origin_glusterd(req_dict);
e3c68b
+
e3c68b
+    if (op == GD_OP_STATUS_VOLUME) {
e3c68b
+        ret = dict_get_uint32(req_dict, "cmd", &cmd);
e3c68b
+        if (ret)
e3c68b
+            goto out;
e3c68b
+
e3c68b
+        if (origin_glusterd) {
e3c68b
+            if ((cmd & GF_CLI_STATUS_ALL)) {
e3c68b
+                ret = 0;
e3c68b
+                goto out;
e3c68b
+            }
e3c68b
+        }
e3c68b
+    }
e3c68b
 
e3c68b
     RCU_READ_LOCK;
e3c68b
     cds_list_for_each_entry_rcu(peerinfo, &conf->peers, uuid_list)
e3c68b
-- 
e3c68b
1.8.3.1
e3c68b