Blame SPECS/0439-cluster-afr-Delay-post-op-for-fsync.patch

Packit Service 150ff0
From 3ed98fc9dcb39223032e343fd5b0ad17fa3cae14 Mon Sep 17 00:00:00 2001
Packit Service 150ff0
From: Pranith Kumar K <pkarampu@redhat.com>
Packit Service 150ff0
Date: Fri, 29 May 2020 14:24:53 +0530
Packit Service 150ff0
Subject: [PATCH 439/449] cluster/afr: Delay post-op for fsync
Packit Service 150ff0
Packit Service 150ff0
Problem:
Packit Service 150ff0
AFR doesn't delay post-op for fsync fop. For fsync heavy workloads
Packit Service 150ff0
this leads to un-necessary fxattrop/finodelk for every fsync leading
Packit Service 150ff0
to bad performance.
Packit Service 150ff0
Packit Service 150ff0
Fix:
Packit Service 150ff0
Have delayed post-op for fsync. Add special flag in xdata to indicate
Packit Service 150ff0
that afr shouldn't delay post-op in cases where either the
Packit Service 150ff0
process will terminate or graph-switch would happen. Otherwise it leads
Packit Service 150ff0
to un-necessary heals when the graph-switch/process-termination
Packit Service 150ff0
happens before delayed-post-op completes.
Packit Service 150ff0
Packit Service 150ff0
> Upstream-patch: https://review.gluster.org/c/glusterfs/+/24473
Packit Service 150ff0
> Fixes: #1253
Packit Service 150ff0
Packit Service 150ff0
BUG: 1838479
Packit Service 150ff0
Change-Id: I531940d13269a111c49e0510d49514dc169f4577
Packit Service 150ff0
Signed-off-by: Pranith Kumar K <pkarampu@redhat.com>
Packit Service 150ff0
Reviewed-on: https://code.engineering.redhat.com/gerrit/202676
Packit Service 150ff0
Tested-by: RHGS Build Bot <nigelb@redhat.com>
Packit Service 150ff0
Reviewed-by: Sunil Kumar Heggodu Gopala Acharya <sheggodu@redhat.com>
Packit Service 150ff0
---
Packit Service 150ff0
 api/src/glfs-resolve.c                         |  14 ++-
Packit Service 150ff0
 tests/basic/afr/durability-off.t               |   2 +
Packit Service 150ff0
 tests/basic/gfapi/gfapi-graph-switch-open-fd.t |  44 +++++++++
Packit Service 150ff0
 tests/basic/gfapi/gfapi-keep-writing.c         | 129 +++++++++++++++++++++++++
Packit Service 150ff0
 xlators/cluster/afr/src/afr-inode-write.c      |  11 ++-
Packit Service 150ff0
 xlators/cluster/afr/src/afr-transaction.c      |   9 +-
Packit Service 150ff0
 xlators/cluster/afr/src/afr.h                  |   2 +-
Packit Service 150ff0
 xlators/cluster/dht/src/dht-rebalance.c        |  15 ++-
Packit Service 150ff0
 xlators/mount/fuse/src/fuse-bridge.c           |  23 ++++-
Packit Service 150ff0
 9 files changed, 239 insertions(+), 10 deletions(-)
Packit Service 150ff0
 create mode 100644 tests/basic/gfapi/gfapi-graph-switch-open-fd.t
Packit Service 150ff0
 create mode 100644 tests/basic/gfapi/gfapi-keep-writing.c
Packit Service 150ff0
Packit Service 150ff0
diff --git a/api/src/glfs-resolve.c b/api/src/glfs-resolve.c
Packit Service 150ff0
index a79f490..062b7dc 100644
Packit Service 150ff0
--- a/api/src/glfs-resolve.c
Packit Service 150ff0
+++ b/api/src/glfs-resolve.c
Packit Service 150ff0
@@ -722,6 +722,7 @@ glfs_migrate_fd_safe(struct glfs *fs, xlator_t *newsubvol, fd_t *oldfd)
Packit Service 150ff0
         0,
Packit Service 150ff0
     };
Packit Service 150ff0
     char uuid1[64];
Packit Service 150ff0
+    dict_t *xdata = NULL;
Packit Service 150ff0
 
Packit Service 150ff0
     oldinode = oldfd->inode;
Packit Service 150ff0
     oldsubvol = oldinode->table->xl;
Packit Service 150ff0
@@ -730,7 +731,15 @@ glfs_migrate_fd_safe(struct glfs *fs, xlator_t *newsubvol, fd_t *oldfd)
Packit Service 150ff0
         return fd_ref(oldfd);
Packit Service 150ff0
 
Packit Service 150ff0
     if (!oldsubvol->switched) {
Packit Service 150ff0
-        ret = syncop_fsync(oldsubvol, oldfd, 0, NULL, NULL, NULL, NULL);
Packit Service 150ff0
+        xdata = dict_new();
Packit Service 150ff0
+        if (!xdata || dict_set_int8(xdata, "last-fsync", 1)) {
Packit Service 150ff0
+            gf_msg(fs->volname, GF_LOG_WARNING, ENOMEM, API_MSG_FSYNC_FAILED,
Packit Service 150ff0
+                   "last-fsync set failed on %s graph %s (%d)",
Packit Service 150ff0
+                   uuid_utoa_r(oldfd->inode->gfid, uuid1),
Packit Service 150ff0
+                   graphid_str(oldsubvol), oldsubvol->graph->id);
Packit Service 150ff0
+        }
Packit Service 150ff0
+
Packit Service 150ff0
+        ret = syncop_fsync(oldsubvol, oldfd, 0, NULL, NULL, xdata, NULL);
Packit Service 150ff0
         DECODE_SYNCOP_ERR(ret);
Packit Service 150ff0
         if (ret) {
Packit Service 150ff0
             gf_msg(fs->volname, GF_LOG_WARNING, errno, API_MSG_FSYNC_FAILED,
Packit Service 150ff0
@@ -809,6 +818,9 @@ out:
Packit Service 150ff0
         newfd = NULL;
Packit Service 150ff0
     }
Packit Service 150ff0
 
Packit Service 150ff0
+    if (xdata)
Packit Service 150ff0
+        dict_unref(xdata);
Packit Service 150ff0
+
Packit Service 150ff0
     return newfd;
Packit Service 150ff0
 }
Packit Service 150ff0
 
Packit Service 150ff0
diff --git a/tests/basic/afr/durability-off.t b/tests/basic/afr/durability-off.t
Packit Service 150ff0
index 155ffa0..6e0f18b 100644
Packit Service 150ff0
--- a/tests/basic/afr/durability-off.t
Packit Service 150ff0
+++ b/tests/basic/afr/durability-off.t
Packit Service 150ff0
@@ -26,6 +26,8 @@ TEST $CLI volume heal $V0
Packit Service 150ff0
 EXPECT_WITHIN $HEAL_TIMEOUT "0" get_pending_heal_count $V0
Packit Service 150ff0
 EXPECT "^0$" echo $($CLI volume profile $V0 info | grep -w FSYNC | wc -l)
Packit Service 150ff0
 
Packit Service 150ff0
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status $V0 0
Packit Service 150ff0
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status $V0 1
Packit Service 150ff0
 #Test that fsyncs happen when durability is on
Packit Service 150ff0
 TEST $CLI volume set $V0 cluster.ensure-durability on
Packit Service 150ff0
 TEST $CLI volume set $V0 performance.strict-write-ordering on
Packit Service 150ff0
diff --git a/tests/basic/gfapi/gfapi-graph-switch-open-fd.t b/tests/basic/gfapi/gfapi-graph-switch-open-fd.t
Packit Service 150ff0
new file mode 100644
Packit Service 150ff0
index 0000000..2e666be
Packit Service 150ff0
--- /dev/null
Packit Service 150ff0
+++ b/tests/basic/gfapi/gfapi-graph-switch-open-fd.t
Packit Service 150ff0
@@ -0,0 +1,44 @@
Packit Service 150ff0
+#!/bin/bash
Packit Service 150ff0
+
Packit Service 150ff0
+. $(dirname $0)/../../include.rc
Packit Service 150ff0
+. $(dirname $0)/../../volume.rc
Packit Service 150ff0
+
Packit Service 150ff0
+cleanup;
Packit Service 150ff0
+
Packit Service 150ff0
+TEST glusterd
Packit Service 150ff0
+
Packit Service 150ff0
+TEST $CLI volume create $V0 replica 3 ${H0}:$B0/brick{0..2};
Packit Service 150ff0
+EXPECT 'Created' volinfo_field $V0 'Status';
Packit Service 150ff0
+
Packit Service 150ff0
+TEST $CLI volume start $V0;
Packit Service 150ff0
+EXPECT 'Started' volinfo_field $V0 'Status';
Packit Service 150ff0
+
Packit Service 150ff0
+TEST $GFS --volfile-id=$V0 --volfile-server=$H0 $M0;
Packit Service 150ff0
+TEST touch $M0/sync
Packit Service 150ff0
+logdir=`gluster --print-logdir`
Packit Service 150ff0
+
Packit Service 150ff0
+TEST build_tester $(dirname $0)/gfapi-keep-writing.c -lgfapi
Packit Service 150ff0
+
Packit Service 150ff0
+
Packit Service 150ff0
+#Launch a program to keep doing writes on an fd
Packit Service 150ff0
+./$(dirname $0)/gfapi-keep-writing ${H0} $V0 $logdir/gfapi-async-calls-test.log sync &
Packit Service 150ff0
+p=$!
Packit Service 150ff0
+sleep 1 #Let some writes go through
Packit Service 150ff0
+#Check if graph switch will lead to any pending markers for ever
Packit Service 150ff0
+TEST $CLI volume set $V0 performance.quick-read off
Packit Service 150ff0
+TEST $CLI volume set $V0 performance.io-cache off
Packit Service 150ff0
+TEST $CLI volume set $V0 performance.stat-prefetch off
Packit Service 150ff0
+TEST $CLI volume set $V0 performance.read-ahead off
Packit Service 150ff0
+
Packit Service 150ff0
+
Packit Service 150ff0
+TEST rm -f $M0/sync #Make sure the glfd is closed
Packit Service 150ff0
+TEST wait #Wait for background process to die
Packit Service 150ff0
+#Goal is to check if there is permanent FOOL changelog
Packit Service 150ff0
+sleep 5
Packit Service 150ff0
+EXPECT "0x000000000000000000000000" afr_get_changelog_xattr $B0/brick0/glfs_test.txt trusted.afr.dirty
Packit Service 150ff0
+EXPECT "0x000000000000000000000000" afr_get_changelog_xattr $B0/brick1/glfs_test.txt trusted.afr.dirty
Packit Service 150ff0
+EXPECT "0x000000000000000000000000" afr_get_changelog_xattr $B0/brick2/glfs_test.txt trusted.afr.dirty
Packit Service 150ff0
+
Packit Service 150ff0
+cleanup_tester $(dirname $0)/gfapi-async-calls-test
Packit Service 150ff0
+
Packit Service 150ff0
+cleanup;
Packit Service 150ff0
diff --git a/tests/basic/gfapi/gfapi-keep-writing.c b/tests/basic/gfapi/gfapi-keep-writing.c
Packit Service 150ff0
new file mode 100644
Packit Service 150ff0
index 0000000..91b59ce
Packit Service 150ff0
--- /dev/null
Packit Service 150ff0
+++ b/tests/basic/gfapi/gfapi-keep-writing.c
Packit Service 150ff0
@@ -0,0 +1,129 @@
Packit Service 150ff0
+#include <fcntl.h>
Packit Service 150ff0
+#include <unistd.h>
Packit Service 150ff0
+#include <time.h>
Packit Service 150ff0
+#include <limits.h>
Packit Service 150ff0
+#include <string.h>
Packit Service 150ff0
+#include <stdio.h>
Packit Service 150ff0
+#include <stdlib.h>
Packit Service 150ff0
+#include <errno.h>
Packit Service 150ff0
+#include <glusterfs/api/glfs.h>
Packit Service 150ff0
+#include <glusterfs/api/glfs-handles.h>
Packit Service 150ff0
+
Packit Service 150ff0
+#define LOG_ERR(msg)                                                           \
Packit Service 150ff0
+    do {                                                                       \
Packit Service 150ff0
+        fprintf(stderr, "%s : Error (%s)\n", msg, strerror(errno));            \
Packit Service 150ff0
+    } while (0)
Packit Service 150ff0
+
Packit Service 150ff0
+glfs_t *
Packit Service 150ff0
+init_glfs(const char *hostname, const char *volname, const char *logfile)
Packit Service 150ff0
+{
Packit Service 150ff0
+    int ret = -1;
Packit Service 150ff0
+    glfs_t *fs = NULL;
Packit Service 150ff0
+
Packit Service 150ff0
+    fs = glfs_new(volname);
Packit Service 150ff0
+    if (!fs) {
Packit Service 150ff0
+        LOG_ERR("glfs_new failed");
Packit Service 150ff0
+        return NULL;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = glfs_set_volfile_server(fs, "tcp", hostname, 24007);
Packit Service 150ff0
+    if (ret < 0) {
Packit Service 150ff0
+        LOG_ERR("glfs_set_volfile_server failed");
Packit Service 150ff0
+        goto out;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = glfs_set_logging(fs, logfile, 7);
Packit Service 150ff0
+    if (ret < 0) {
Packit Service 150ff0
+        LOG_ERR("glfs_set_logging failed");
Packit Service 150ff0
+        goto out;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = glfs_init(fs);
Packit Service 150ff0
+    if (ret < 0) {
Packit Service 150ff0
+        LOG_ERR("glfs_init failed");
Packit Service 150ff0
+        goto out;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = 0;
Packit Service 150ff0
+out:
Packit Service 150ff0
+    if (ret) {
Packit Service 150ff0
+        glfs_fini(fs);
Packit Service 150ff0
+        fs = NULL;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    return fs;
Packit Service 150ff0
+}
Packit Service 150ff0
+
Packit Service 150ff0
+int
Packit Service 150ff0
+glfs_test_function(const char *hostname, const char *volname,
Packit Service 150ff0
+                   const char *logfile, const char *syncfile)
Packit Service 150ff0
+{
Packit Service 150ff0
+    int ret = -1;
Packit Service 150ff0
+    int flags = O_CREAT | O_RDWR;
Packit Service 150ff0
+    glfs_t *fs = NULL;
Packit Service 150ff0
+    glfs_fd_t *glfd = NULL;
Packit Service 150ff0
+    const char *buff = "This is from my prog\n";
Packit Service 150ff0
+    const char *filename = "glfs_test.txt";
Packit Service 150ff0
+    struct stat buf = {0};
Packit Service 150ff0
+
Packit Service 150ff0
+    fs = init_glfs(hostname, volname, logfile);
Packit Service 150ff0
+    if (fs == NULL) {
Packit Service 150ff0
+        LOG_ERR("init_glfs failed");
Packit Service 150ff0
+        return -1;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    glfd = glfs_creat(fs, filename, flags, 0644);
Packit Service 150ff0
+    if (glfd == NULL) {
Packit Service 150ff0
+        LOG_ERR("glfs_creat failed");
Packit Service 150ff0
+        goto out;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    while (glfs_stat(fs, syncfile, &buf) == 0) {
Packit Service 150ff0
+        ret = glfs_write(glfd, buff, strlen(buff), flags);
Packit Service 150ff0
+        if (ret < 0) {
Packit Service 150ff0
+            LOG_ERR("glfs_write failed");
Packit Service 150ff0
+            goto out;
Packit Service 150ff0
+        }
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = glfs_close(glfd);
Packit Service 150ff0
+    if (ret < 0) {
Packit Service 150ff0
+        LOG_ERR("glfs_write failed");
Packit Service 150ff0
+        goto out;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+out:
Packit Service 150ff0
+    ret = glfs_fini(fs);
Packit Service 150ff0
+    if (ret) {
Packit Service 150ff0
+        LOG_ERR("glfs_fini failed");
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    return ret;
Packit Service 150ff0
+}
Packit Service 150ff0
+
Packit Service 150ff0
+int
Packit Service 150ff0
+main(int argc, char *argv[])
Packit Service 150ff0
+{
Packit Service 150ff0
+    int ret = 0;
Packit Service 150ff0
+    char *hostname = NULL;
Packit Service 150ff0
+    char *volname = NULL;
Packit Service 150ff0
+    char *logfile = NULL;
Packit Service 150ff0
+    char *syncfile = NULL;
Packit Service 150ff0
+
Packit Service 150ff0
+    if (argc != 5) {
Packit Service 150ff0
+        fprintf(stderr, "Invalid argument\n");
Packit Service 150ff0
+        exit(1);
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    hostname = argv[1];
Packit Service 150ff0
+    volname = argv[2];
Packit Service 150ff0
+    logfile = argv[3];
Packit Service 150ff0
+    syncfile = argv[4];
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = glfs_test_function(hostname, volname, logfile, syncfile);
Packit Service 150ff0
+    if (ret) {
Packit Service 150ff0
+        LOG_ERR("glfs_test_function failed");
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    return ret;
Packit Service 150ff0
+}
Packit Service 150ff0
diff --git a/xlators/cluster/afr/src/afr-inode-write.c b/xlators/cluster/afr/src/afr-inode-write.c
Packit Service 150ff0
index 7fcc9d4..df82b6e 100644
Packit Service 150ff0
--- a/xlators/cluster/afr/src/afr-inode-write.c
Packit Service 150ff0
+++ b/xlators/cluster/afr/src/afr-inode-write.c
Packit Service 150ff0
@@ -2492,6 +2492,7 @@ afr_fsync(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t datasync,
Packit Service 150ff0
     call_frame_t *transaction_frame = NULL;
Packit Service 150ff0
     int ret = -1;
Packit Service 150ff0
     int32_t op_errno = ENOMEM;
Packit Service 150ff0
+    int8_t last_fsync = 0;
Packit Service 150ff0
 
Packit Service 150ff0
     transaction_frame = copy_frame(frame);
Packit Service 150ff0
     if (!transaction_frame)
Packit Service 150ff0
@@ -2501,10 +2502,16 @@ afr_fsync(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t datasync,
Packit Service 150ff0
     if (!local)
Packit Service 150ff0
         goto out;
Packit Service 150ff0
 
Packit Service 150ff0
-    if (xdata)
Packit Service 150ff0
+    if (xdata) {
Packit Service 150ff0
         local->xdata_req = dict_copy_with_ref(xdata, NULL);
Packit Service 150ff0
-    else
Packit Service 150ff0
+        if (dict_get_int8(xdata, "last-fsync", &last_fsync) == 0) {
Packit Service 150ff0
+            if (last_fsync) {
Packit Service 150ff0
+                local->transaction.disable_delayed_post_op = _gf_true;
Packit Service 150ff0
+            }
Packit Service 150ff0
+        }
Packit Service 150ff0
+    } else {
Packit Service 150ff0
         local->xdata_req = dict_new();
Packit Service 150ff0
+    }
Packit Service 150ff0
 
Packit Service 150ff0
     if (!local->xdata_req)
Packit Service 150ff0
         goto out;
Packit Service 150ff0
diff --git a/xlators/cluster/afr/src/afr-transaction.c b/xlators/cluster/afr/src/afr-transaction.c
Packit Service 150ff0
index 8e65ae2..ffd0ab8 100644
Packit Service 150ff0
--- a/xlators/cluster/afr/src/afr-transaction.c
Packit Service 150ff0
+++ b/xlators/cluster/afr/src/afr-transaction.c
Packit Service 150ff0
@@ -2385,8 +2385,13 @@ afr_is_delayed_changelog_post_op_needed(call_frame_t *frame, xlator_t *this,
Packit Service 150ff0
         goto out;
Packit Service 150ff0
     }
Packit Service 150ff0
 
Packit Service 150ff0
-    if ((local->op != GF_FOP_WRITE) && (local->op != GF_FOP_FXATTROP)) {
Packit Service 150ff0
-        /*Only allow writes but shard does [f]xattrops on writes, so
Packit Service 150ff0
+    if (local->transaction.disable_delayed_post_op) {
Packit Service 150ff0
+        goto out;
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    if ((local->op != GF_FOP_WRITE) && (local->op != GF_FOP_FXATTROP) &&
Packit Service 150ff0
+        (local->op != GF_FOP_FSYNC)) {
Packit Service 150ff0
+        /*Only allow writes/fsyncs but shard does [f]xattrops on writes, so
Packit Service 150ff0
          * they are fine too*/
Packit Service 150ff0
         goto out;
Packit Service 150ff0
     }
Packit Service 150ff0
diff --git a/xlators/cluster/afr/src/afr.h b/xlators/cluster/afr/src/afr.h
Packit Service 150ff0
index 18f1a6a..ff96246 100644
Packit Service 150ff0
--- a/xlators/cluster/afr/src/afr.h
Packit Service 150ff0
+++ b/xlators/cluster/afr/src/afr.h
Packit Service 150ff0
@@ -854,7 +854,7 @@ typedef struct _afr_local {
Packit Service 150ff0
 
Packit Service 150ff0
         int (*unwind)(call_frame_t *frame, xlator_t *this);
Packit Service 150ff0
 
Packit Service 150ff0
-        /* post-op hook */
Packit Service 150ff0
+        gf_boolean_t disable_delayed_post_op;
Packit Service 150ff0
     } transaction;
Packit Service 150ff0
 
Packit Service 150ff0
     syncbarrier_t barrier;
Packit Service 150ff0
diff --git a/xlators/cluster/dht/src/dht-rebalance.c b/xlators/cluster/dht/src/dht-rebalance.c
Packit Service 150ff0
index d0c21b4..e9974cd 100644
Packit Service 150ff0
--- a/xlators/cluster/dht/src/dht-rebalance.c
Packit Service 150ff0
+++ b/xlators/cluster/dht/src/dht-rebalance.c
Packit Service 150ff0
@@ -1550,6 +1550,7 @@ dht_migrate_file(xlator_t *this, loc_t *loc, xlator_t *from, xlator_t *to,
Packit Service 150ff0
     xlator_t *old_target = NULL;
Packit Service 150ff0
     xlator_t *hashed_subvol = NULL;
Packit Service 150ff0
     fd_t *linkto_fd = NULL;
Packit Service 150ff0
+    dict_t *xdata = NULL;
Packit Service 150ff0
 
Packit Service 150ff0
     if (from == to) {
Packit Service 150ff0
         gf_msg_debug(this->name, 0,
Packit Service 150ff0
@@ -1868,7 +1869,15 @@ dht_migrate_file(xlator_t *this, loc_t *loc, xlator_t *from, xlator_t *to,
Packit Service 150ff0
 
Packit Service 150ff0
     /* TODO: Sync the locks */
Packit Service 150ff0
 
Packit Service 150ff0
-    ret = syncop_fsync(to, dst_fd, 0, NULL, NULL, NULL, NULL);
Packit Service 150ff0
+    xdata = dict_new();
Packit Service 150ff0
+    if (!xdata || dict_set_int8(xdata, "last-fsync", 1)) {
Packit Service 150ff0
+        gf_log(this->name, GF_LOG_ERROR,
Packit Service 150ff0
+               "%s: failed to set last-fsync flag on "
Packit Service 150ff0
+               "%s (%s)",
Packit Service 150ff0
+               loc->path, to->name, strerror(ENOMEM));
Packit Service 150ff0
+    }
Packit Service 150ff0
+
Packit Service 150ff0
+    ret = syncop_fsync(to, dst_fd, 0, NULL, NULL, xdata, NULL);
Packit Service 150ff0
     if (ret) {
Packit Service 150ff0
         gf_log(this->name, GF_LOG_WARNING, "%s: failed to fsync on %s (%s)",
Packit Service 150ff0
                loc->path, to->name, strerror(-ret));
Packit Service 150ff0
@@ -2342,11 +2351,15 @@ out:
Packit Service 150ff0
 
Packit Service 150ff0
     if (dst_fd)
Packit Service 150ff0
         syncop_close(dst_fd);
Packit Service 150ff0
+
Packit Service 150ff0
     if (src_fd)
Packit Service 150ff0
         syncop_close(src_fd);
Packit Service 150ff0
     if (linkto_fd)
Packit Service 150ff0
         syncop_close(linkto_fd);
Packit Service 150ff0
 
Packit Service 150ff0
+    if (xdata)
Packit Service 150ff0
+        dict_unref(xdata);
Packit Service 150ff0
+
Packit Service 150ff0
     loc_wipe(&tmp_loc);
Packit Service 150ff0
     loc_wipe(&parent_loc);
Packit Service 150ff0
 
Packit Service 150ff0
diff --git a/xlators/mount/fuse/src/fuse-bridge.c b/xlators/mount/fuse/src/fuse-bridge.c
Packit Service 150ff0
index fdeec49..4264fad 100644
Packit Service 150ff0
--- a/xlators/mount/fuse/src/fuse-bridge.c
Packit Service 150ff0
+++ b/xlators/mount/fuse/src/fuse-bridge.c
Packit Service 150ff0
@@ -5559,6 +5559,7 @@ fuse_migrate_fd(xlator_t *this, fd_t *basefd, xlator_t *old_subvol,
Packit Service 150ff0
     char create_in_progress = 0;
Packit Service 150ff0
     fuse_fd_ctx_t *basefd_ctx = NULL;
Packit Service 150ff0
     fd_t *oldfd = NULL;
Packit Service 150ff0
+    dict_t *xdata = NULL;
Packit Service 150ff0
 
Packit Service 150ff0
     basefd_ctx = fuse_fd_ctx_get(this, basefd);
Packit Service 150ff0
     GF_VALIDATE_OR_GOTO("glusterfs-fuse", basefd_ctx, out);
Packit Service 150ff0
@@ -5595,10 +5596,23 @@ fuse_migrate_fd(xlator_t *this, fd_t *basefd, xlator_t *old_subvol,
Packit Service 150ff0
     }
Packit Service 150ff0
 
Packit Service 150ff0
     if (oldfd->inode->table->xl == old_subvol) {
Packit Service 150ff0
-        if (IA_ISDIR(oldfd->inode->ia_type))
Packit Service 150ff0
+        if (IA_ISDIR(oldfd->inode->ia_type)) {
Packit Service 150ff0
             ret = syncop_fsyncdir(old_subvol, oldfd, 0, NULL, NULL);
Packit Service 150ff0
-        else
Packit Service 150ff0
-            ret = syncop_fsync(old_subvol, oldfd, 0, NULL, NULL, NULL, NULL);
Packit Service 150ff0
+        } else {
Packit Service 150ff0
+            xdata = dict_new();
Packit Service 150ff0
+            if (!xdata || dict_set_int8(xdata, "last-fsync", 1)) {
Packit Service 150ff0
+                gf_log("glusterfs-fuse", GF_LOG_WARNING,
Packit Service 150ff0
+                       "last-fsync set failed (%s) on fd (%p)"
Packit Service 150ff0
+                       "(basefd:%p basefd-inode.gfid:%s) "
Packit Service 150ff0
+                       "(old-subvolume:%s-%d new-subvolume:%s-%d)",
Packit Service 150ff0
+                       strerror(ENOMEM), oldfd, basefd,
Packit Service 150ff0
+                       uuid_utoa(basefd->inode->gfid), old_subvol->name,
Packit Service 150ff0
+                       old_subvol->graph->id, new_subvol->name,
Packit Service 150ff0
+                       new_subvol->graph->id);
Packit Service 150ff0
+            }
Packit Service 150ff0
+
Packit Service 150ff0
+            ret = syncop_fsync(old_subvol, oldfd, 0, NULL, NULL, xdata, NULL);
Packit Service 150ff0
+        }
Packit Service 150ff0
 
Packit Service 150ff0
         if (ret < 0) {
Packit Service 150ff0
             gf_log("glusterfs-fuse", GF_LOG_WARNING,
Packit Service 150ff0
@@ -5653,6 +5667,9 @@ out:
Packit Service 150ff0
 
Packit Service 150ff0
     fd_unref(oldfd);
Packit Service 150ff0
 
Packit Service 150ff0
+    if (xdata)
Packit Service 150ff0
+        dict_unref(xdata);
Packit Service 150ff0
+
Packit Service 150ff0
     return ret;
Packit Service 150ff0
 }
Packit Service 150ff0
 
Packit Service 150ff0
-- 
Packit Service 150ff0
1.8.3.1
Packit Service 150ff0