*/
 static int __get_inode_info(struct btrfs_root *root, struct btrfs_path *path,
                          u64 ino, u64 *size, u64 *gen, u64 *mode, u64 *uid,
-                         u64 *gid, u64 *rdev)
+                         u64 *gid, u64 *rdev, u64 *fileattr)
 {
        int ret;
        struct btrfs_inode_item *ii;
                *gid = btrfs_inode_gid(path->nodes[0], ii);
        if (rdev)
                *rdev = btrfs_inode_rdev(path->nodes[0], ii);
+       /*
+        * Transfer the unchanged u64 value of btrfs_inode_item::flags, that's
+        * otherwise logically split to 32/32 parts.
+        */
+       if (fileattr)
+               *fileattr = btrfs_inode_flags(path->nodes[0], ii);
 
        return ret;
 }
 static int get_inode_info(struct btrfs_root *root,
                          u64 ino, u64 *size, u64 *gen,
                          u64 *mode, u64 *uid, u64 *gid,
-                         u64 *rdev)
+                         u64 *rdev, u64 *fileattr)
 {
        struct btrfs_path *path;
        int ret;
        if (!path)
                return -ENOMEM;
        ret = __get_inode_info(root, path, ino, size, gen, mode, uid, gid,
-                              rdev);
+                              rdev, fileattr);
        btrfs_free_path(path);
        return ret;
 }
        u64 right_gen;
 
        ret = get_inode_info(sctx->send_root, ino, NULL, &left_gen, NULL, NULL,
-                       NULL, NULL);
+                       NULL, NULL, NULL);
        if (ret < 0 && ret != -ENOENT)
                goto out;
        left_ret = ret;
                right_ret = -ENOENT;
        } else {
                ret = get_inode_info(sctx->parent_root, ino, NULL, &right_gen,
-                               NULL, NULL, NULL, NULL);
+                               NULL, NULL, NULL, NULL, NULL);
                if (ret < 0 && ret != -ENOENT)
                        goto out;
                right_ret = ret;
 
        if (dir_gen) {
                ret = get_inode_info(root, parent_dir, NULL, dir_gen, NULL,
-                                    NULL, NULL, NULL);
+                                    NULL, NULL, NULL, NULL);
                if (ret < 0)
                        goto out;
        }
         */
        if (sctx->parent_root && dir != BTRFS_FIRST_FREE_OBJECTID) {
                ret = get_inode_info(sctx->parent_root, dir, NULL, &gen, NULL,
-                                    NULL, NULL, NULL);
+                                    NULL, NULL, NULL, NULL);
                if (ret < 0 && ret != -ENOENT)
                        goto out;
                if (ret) {
        if (other_inode > sctx->send_progress ||
            is_waiting_for_move(sctx, other_inode)) {
                ret = get_inode_info(sctx->parent_root, other_inode, NULL,
-                               who_gen, who_mode, NULL, NULL, NULL);
+                               who_gen, who_mode, NULL, NULL, NULL, NULL);
                if (ret < 0)
                        goto out;
 
 
        if (dir != BTRFS_FIRST_FREE_OBJECTID) {
                ret = get_inode_info(sctx->send_root, dir, NULL, &gen, NULL,
-                                    NULL, NULL, NULL);
+                                    NULL, NULL, NULL, NULL);
                if (ret < 0 && ret != -ENOENT)
                        goto out;
                if (ret) {
        }
 
        ret = get_inode_info(sctx->send_root, ow_inode, NULL, &gen, NULL, NULL,
-                       NULL, NULL);
+                       NULL, NULL, NULL);
        if (ret < 0)
                goto out;
 
        return ret;
 }
 
+static int send_fileattr(struct send_ctx *sctx, u64 ino, u64 gen, u64 fileattr)
+{
+       struct btrfs_fs_info *fs_info = sctx->send_root->fs_info;
+       int ret = 0;
+       struct fs_path *p;
+
+       if (sctx->proto < 2)
+               return 0;
+
+       btrfs_debug(fs_info, "send_fileattr %llu fileattr=%llu", ino, fileattr);
+
+       p = fs_path_alloc();
+       if (!p)
+               return -ENOMEM;
+
+       ret = begin_cmd(sctx, BTRFS_SEND_C_FILEATTR);
+       if (ret < 0)
+               goto out;
+
+       ret = get_cur_path(sctx, ino, gen, p);
+       if (ret < 0)
+               goto out;
+       TLV_PUT_PATH(sctx, BTRFS_SEND_A_PATH, p);
+       TLV_PUT_U64(sctx, BTRFS_SEND_A_FILEATTR, fileattr);
+
+       ret = send_cmd(sctx);
+
+tlv_put_failure:
+out:
+       fs_path_free(p);
+       return ret;
+}
+
 static int send_chown(struct send_ctx *sctx, u64 ino, u64 gen, u64 uid, u64 gid)
 {
        struct btrfs_fs_info *fs_info = sctx->send_root->fs_info;
 
        if (ino != sctx->cur_ino) {
                ret = get_inode_info(sctx->send_root, ino, NULL, &gen, &mode,
-                                    NULL, NULL, &rdev);
+                                    NULL, NULL, &rdev, NULL);
                if (ret < 0)
                        goto out;
        } else {
                 * The parent inode might have been deleted in the send snapshot
                 */
                ret = get_inode_info(sctx->send_root, cur->dir, NULL,
-                                    NULL, NULL, NULL, NULL, NULL);
+                                    NULL, NULL, NULL, NULL, NULL, NULL);
                if (ret == -ENOENT) {
                        ret = 0;
                        continue;
        }
 
        ret = get_inode_info(sctx->parent_root, di_key.objectid, NULL,
-                            &left_gen, NULL, NULL, NULL, NULL);
+                            &left_gen, NULL, NULL, NULL, NULL, NULL);
        if (ret < 0)
                goto out;
        ret = get_inode_info(sctx->send_root, di_key.objectid, NULL,
-                            &right_gen, NULL, NULL, NULL, NULL);
+                            &right_gen, NULL, NULL, NULL, NULL, NULL);
        if (ret < 0) {
                if (ret == -ENOENT)
                        ret = 0;
                        }
 
                        ret = get_inode_info(root, parent, NULL, &parent_gen,
-                                            NULL, NULL, NULL, NULL);
+                                            NULL, NULL, NULL, NULL, NULL);
                        if (ret < 0)
                                goto out;
                        ret = check_ino_in_path(root, ino1, ino1_gen,
 
                        ret = get_inode_info(sctx->parent_root, ino, NULL,
                                             &parent_ino_gen, NULL, NULL, NULL,
-                                            NULL);
+                                            NULL, NULL);
                        if (ret < 0)
                                goto out;
                        if (ino_gen == parent_ino_gen) {
        if (!p)
                return -ENOMEM;
 
-       ret = get_inode_info(root, dir, NULL, &gen, NULL, NULL,
-                       NULL, NULL);
+       ret = get_inode_info(root, dir, NULL, &gen, NULL, NULL, NULL, NULL, NULL);
        if (ret < 0)
                goto out;
 
                 * else matches.
                 */
                ret = get_inode_info(ctx->root, dir, NULL, &dir_gen, NULL,
-                                    NULL, NULL, NULL);
+                                    NULL, NULL, NULL, NULL);
                if (ret)
                        return ret;
                if (dir_gen != ctx->dir_gen)
        struct send_ctx *sctx = ctx;
 
        ret = get_inode_info(sctx->send_root, dir, NULL, &dir_gen, NULL,
-                            NULL, NULL, NULL);
+                            NULL, NULL, NULL, NULL);
        if (ret)
                return ret;
 
        struct send_ctx *sctx = ctx;
 
        ret = get_inode_info(sctx->parent_root, dir, NULL, &dir_gen, NULL,
-                            NULL, NULL, NULL);
+                            NULL, NULL, NULL, NULL);
        if (ret)
                return ret;
 
 
        if (clone_root->root == sctx->send_root) {
                ret = get_inode_info(sctx->send_root, clone_root->ino, NULL,
-                               &gen, NULL, NULL, NULL, NULL);
+                               &gen, NULL, NULL, NULL, NULL, NULL);
                if (ret < 0)
                        goto out;
                ret = get_cur_path(sctx, clone_root->ino, gen, p);
         * accept clones from these extents.
         */
        ret = __get_inode_info(clone_root->root, path, clone_root->ino,
-                              &clone_src_i_size, NULL, NULL, NULL, NULL, NULL);
+                              &clone_src_i_size, NULL, NULL, NULL, NULL, NULL,
+                              NULL);
        btrfs_release_path(path);
        if (ret < 0)
                goto out;
        u64 left_mode;
        u64 left_uid;
        u64 left_gid;
+       u64 left_fileattr;
        u64 right_mode;
        u64 right_uid;
        u64 right_gid;
+       u64 right_fileattr;
        int need_chmod = 0;
        int need_chown = 0;
+       bool need_fileattr = false;
        int need_truncate = 1;
        int pending_move = 0;
        int refs_processed = 0;
                goto out;
 
        ret = get_inode_info(sctx->send_root, sctx->cur_ino, NULL, NULL,
-                       &left_mode, &left_uid, &left_gid, NULL);
+                       &left_mode, &left_uid, &left_gid, NULL, &left_fileattr);
        if (ret < 0)
                goto out;
 
 
                ret = get_inode_info(sctx->parent_root, sctx->cur_ino,
                                &old_size, NULL, &right_mode, &right_uid,
-                               &right_gid, NULL);
+                               &right_gid, NULL, &right_fileattr);
                if (ret < 0)
                        goto out;
 
                        need_chown = 1;
                if (!S_ISLNK(sctx->cur_inode_mode) && left_mode != right_mode)
                        need_chmod = 1;
+               if (!S_ISLNK(sctx->cur_inode_mode) && left_fileattr != right_fileattr)
+                       need_fileattr = true;
                if ((old_size == sctx->cur_inode_size) ||
                    (sctx->cur_inode_size > old_size &&
                     sctx->cur_inode_next_write_offset == sctx->cur_inode_size))
                if (ret < 0)
                        goto out;
        }
+       if (need_fileattr) {
+               ret = send_fileattr(sctx, sctx->cur_ino, sctx->cur_inode_gen,
+                                   left_fileattr);
+               if (ret < 0)
+                       goto out;
+       }
 
        ret = send_capabilities(sctx);
        if (ret < 0)
        int ret;
 
        ret = get_inode_info(sctx->send_root, dir, NULL, &new_gen, NULL, NULL,
-                            NULL, NULL);
+                            NULL, NULL, NULL);
        if (ret)
                return ret;
 
        ret = get_inode_info(sctx->parent_root, dir, NULL, &orig_gen, NULL,
-                            NULL, NULL, NULL);
+                            NULL, NULL, NULL, NULL);
        if (ret)
                return ret;