2012-07-25 21:08:25 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 Alexander Block. All rights reserved.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public
|
|
|
|
* License v2 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public
|
|
|
|
* License along with this program; if not, write to the
|
|
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 021110-1307, USA.
|
|
|
|
*/
|
|
|
|
|
2012-09-20 21:26:28 +00:00
|
|
|
#include "kerncompat.h"
|
2022-09-14 15:06:52 +00:00
|
|
|
#include <sys/ioctl.h>
|
2012-07-25 21:08:25 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <pthread.h>
|
2015-06-12 11:15:41 +00:00
|
|
|
#include <getopt.h>
|
2015-07-29 18:37:37 +00:00
|
|
|
#include <limits.h>
|
2022-09-15 11:59:39 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "kernel-lib/sizes.h"
|
2023-04-19 21:17:12 +00:00
|
|
|
#include "kernel-shared/uapi/btrfs.h"
|
2022-09-14 15:06:52 +00:00
|
|
|
#include "common/utils.h"
|
2020-08-18 13:56:04 +00:00
|
|
|
#include "common/send-utils.h"
|
2019-06-19 23:46:21 +00:00
|
|
|
#include "common/help.h"
|
2019-07-01 18:54:39 +00:00
|
|
|
#include "common/path-utils.h"
|
2022-09-15 21:15:17 +00:00
|
|
|
#include "common/string-utils.h"
|
2022-09-15 11:59:39 +00:00
|
|
|
#include "common/messages.h"
|
2022-09-14 15:06:52 +00:00
|
|
|
#include "cmds/commands.h"
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2022-10-17 03:52:31 +00:00
|
|
|
#define BTRFS_SEND_BUF_SIZE_V1 (SZ_64K)
|
|
|
|
#define BTRFS_MAX_COMPRESSED (SZ_128K)
|
|
|
|
#define BTRFS_SEND_BUF_SIZE_V2 (SZ_16K + BTRFS_MAX_COMPRESSED)
|
|
|
|
|
2016-11-02 12:27:45 +00:00
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
struct btrfs_send {
|
|
|
|
int send_fd;
|
|
|
|
int dump_fd;
|
|
|
|
int mnt_fd;
|
|
|
|
|
|
|
|
u64 *clone_sources;
|
|
|
|
u64 clone_sources_count;
|
|
|
|
|
|
|
|
char *root_path;
|
2022-03-17 17:25:52 +00:00
|
|
|
u32 proto;
|
|
|
|
u32 proto_supported;
|
2012-07-25 21:08:25 +00:00
|
|
|
};
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
static int get_root_id(struct btrfs_send *sctx, const char *path, u64 *root_id)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
|
|
|
struct subvol_info *si;
|
|
|
|
|
2021-09-30 18:06:50 +00:00
|
|
|
si = subvol_uuid_search(sctx->mnt_fd, 0, NULL, 0, path, subvol_search_by_path);
|
2016-12-15 08:37:02 +00:00
|
|
|
if (IS_ERR_OR_NULL(si)) {
|
|
|
|
if (!si)
|
|
|
|
return -ENOENT;
|
|
|
|
else
|
|
|
|
return PTR_ERR(si);
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
*root_id = si->root_id;
|
2013-06-26 15:17:57 +00:00
|
|
|
free(si->path);
|
|
|
|
free(si);
|
2012-07-25 21:08:25 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
static struct subvol_info *get_parent(struct btrfs_send *sctx, u64 root_id)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
2013-06-26 15:17:57 +00:00
|
|
|
struct subvol_info *si_tmp;
|
2012-07-25 21:08:25 +00:00
|
|
|
struct subvol_info *si;
|
|
|
|
|
2021-09-30 18:06:50 +00:00
|
|
|
si_tmp = subvol_uuid_search(sctx->mnt_fd, root_id, NULL, 0, NULL,
|
2012-07-25 21:08:25 +00:00
|
|
|
subvol_search_by_root_id);
|
2016-12-15 08:37:02 +00:00
|
|
|
if (IS_ERR_OR_NULL(si_tmp))
|
2016-12-10 13:47:43 +00:00
|
|
|
return si_tmp;
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2021-09-30 18:06:50 +00:00
|
|
|
si = subvol_uuid_search(sctx->mnt_fd, 0, si_tmp->parent_uuid, 0, NULL,
|
2012-07-25 21:08:25 +00:00
|
|
|
subvol_search_by_uuid);
|
2013-06-26 15:17:57 +00:00
|
|
|
free(si_tmp->path);
|
|
|
|
free(si_tmp);
|
2012-07-25 21:08:25 +00:00
|
|
|
return si;
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
static int find_good_parent(struct btrfs_send *sctx, u64 root_id, u64 *found)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
|
|
|
int ret;
|
2013-06-26 15:17:57 +00:00
|
|
|
struct subvol_info *parent = NULL;
|
|
|
|
struct subvol_info *parent2 = NULL;
|
2012-07-25 21:08:25 +00:00
|
|
|
struct subvol_info *best_parent = NULL;
|
|
|
|
u64 best_diff = (u64)-1;
|
|
|
|
int i;
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
parent = get_parent(sctx, root_id);
|
2016-12-15 08:37:02 +00:00
|
|
|
if (IS_ERR_OR_NULL(parent)) {
|
|
|
|
if (!parent)
|
|
|
|
ret = -ENOENT;
|
|
|
|
else
|
|
|
|
ret = PTR_ERR(parent);
|
2022-01-11 16:34:08 +00:00
|
|
|
parent = NULL;
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
for (i = 0; i < sctx->clone_sources_count; i++) {
|
|
|
|
if (sctx->clone_sources[i] == parent->root_id) {
|
2012-07-25 21:08:25 +00:00
|
|
|
best_parent = parent;
|
2013-06-26 15:17:57 +00:00
|
|
|
parent = NULL;
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out_found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
for (i = 0; i < sctx->clone_sources_count; i++) {
|
2016-11-02 13:59:07 +00:00
|
|
|
s64 tmp;
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
parent2 = get_parent(sctx, sctx->clone_sources[i]);
|
2016-12-15 08:37:02 +00:00
|
|
|
if (IS_ERR_OR_NULL(parent2))
|
2012-07-25 21:08:25 +00:00
|
|
|
continue;
|
2013-06-26 15:17:57 +00:00
|
|
|
if (parent2->root_id != parent->root_id) {
|
|
|
|
free(parent2->path);
|
|
|
|
free(parent2);
|
|
|
|
parent2 = NULL;
|
|
|
|
continue;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2013-06-26 15:17:57 +00:00
|
|
|
free(parent2->path);
|
|
|
|
free(parent2);
|
2021-09-30 18:06:50 +00:00
|
|
|
parent2 = subvol_uuid_search(sctx->mnt_fd,
|
2016-11-02 13:14:56 +00:00
|
|
|
sctx->clone_sources[i], NULL, 0, NULL,
|
|
|
|
subvol_search_by_root_id);
|
2016-12-15 08:37:02 +00:00
|
|
|
if (IS_ERR_OR_NULL(parent2)) {
|
|
|
|
if (!parent2)
|
|
|
|
ret = -ENOENT;
|
|
|
|
else
|
|
|
|
ret = PTR_ERR(parent2);
|
2022-01-11 16:34:08 +00:00
|
|
|
parent2 = NULL;
|
2014-05-12 01:39:42 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
tmp = parent2->ctransid - parent->ctransid;
|
|
|
|
if (tmp < 0)
|
2016-11-02 13:59:07 +00:00
|
|
|
tmp = -tmp;
|
2012-07-25 21:08:25 +00:00
|
|
|
if (tmp < best_diff) {
|
2013-06-26 15:17:57 +00:00
|
|
|
if (best_parent) {
|
|
|
|
free(best_parent->path);
|
|
|
|
free(best_parent);
|
|
|
|
}
|
2013-01-24 11:53:40 +00:00
|
|
|
best_parent = parent2;
|
2013-06-26 15:17:57 +00:00
|
|
|
parent2 = NULL;
|
2012-07-25 21:08:25 +00:00
|
|
|
best_diff = tmp;
|
2013-06-26 15:17:57 +00:00
|
|
|
} else {
|
|
|
|
free(parent2->path);
|
|
|
|
free(parent2);
|
|
|
|
parent2 = NULL;
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!best_parent) {
|
|
|
|
ret = -ENOENT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_found:
|
|
|
|
*found = best_parent->root_id;
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
2013-06-26 15:17:57 +00:00
|
|
|
if (parent) {
|
|
|
|
free(parent->path);
|
|
|
|
free(parent);
|
|
|
|
}
|
|
|
|
if (best_parent) {
|
|
|
|
free(best_parent->path);
|
|
|
|
free(best_parent);
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
static int add_clone_source(struct btrfs_send *sctx, u64 root_id)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
2015-08-21 08:51:52 +00:00
|
|
|
void *tmp;
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
tmp = sctx->clone_sources;
|
|
|
|
sctx->clone_sources = realloc(sctx->clone_sources,
|
|
|
|
sizeof(*sctx->clone_sources) * (sctx->clone_sources_count + 1));
|
2014-12-19 15:55:32 +00:00
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
if (!sctx->clone_sources) {
|
2015-08-21 08:51:52 +00:00
|
|
|
free(tmp);
|
2014-12-19 15:55:32 +00:00
|
|
|
return -ENOMEM;
|
2015-08-21 08:51:52 +00:00
|
|
|
}
|
2016-11-02 13:14:56 +00:00
|
|
|
sctx->clone_sources[sctx->clone_sources_count++] = root_id;
|
2014-12-19 15:55:32 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
|
|
|
|
2016-11-02 14:04:47 +00:00
|
|
|
static void *read_sent_data(void *arg)
|
2016-11-02 13:54:10 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct btrfs_send *sctx = (struct btrfs_send*)arg;
|
|
|
|
|
|
|
|
while (1) {
|
2022-10-17 03:52:31 +00:00
|
|
|
size_t splice_buf_size = BTRFS_SEND_BUF_SIZE_V1;
|
2016-11-02 13:54:10 +00:00
|
|
|
ssize_t sbytes;
|
|
|
|
|
2022-10-17 03:52:31 +00:00
|
|
|
if(sctx->proto > 1){
|
|
|
|
splice_buf_size = BTRFS_SEND_BUF_SIZE_V2;
|
|
|
|
|
|
|
|
/* Try to change pipe buffer size */
|
|
|
|
fcntl(sctx->dump_fd, F_SETPIPE_SZ, splice_buf_size);
|
|
|
|
}
|
2016-11-02 13:54:10 +00:00
|
|
|
/* Source is a pipe, output is either file or stdout */
|
|
|
|
sbytes = splice(sctx->send_fd, NULL, sctx->dump_fd,
|
2022-10-17 03:52:31 +00:00
|
|
|
NULL, splice_buf_size, SPLICE_F_MORE);
|
2016-11-02 13:54:10 +00:00
|
|
|
if (sbytes < 0) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("failed to read stream from kernel: %m");
|
2016-11-02 13:54:10 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (!sbytes) {
|
|
|
|
ret = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
out:
|
2016-11-02 13:07:53 +00:00
|
|
|
if (ret < 0)
|
2012-07-25 21:08:25 +00:00
|
|
|
exit(-ret);
|
|
|
|
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
}
|
|
|
|
|
2013-11-29 16:37:25 +00:00
|
|
|
static int do_send(struct btrfs_send *send, u64 parent_root_id,
|
2016-11-02 12:17:31 +00:00
|
|
|
int is_first_subvol, int is_last_subvol, const char *subvol,
|
2015-06-12 11:15:41 +00:00
|
|
|
u64 flags)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
pthread_t t_read;
|
|
|
|
struct btrfs_ioctl_send_args io_send;
|
|
|
|
void *t_err = NULL;
|
|
|
|
int subvol_fd = -1;
|
2013-01-30 07:11:46 +00:00
|
|
|
int pipefd[2] = {-1, -1};
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2013-11-29 16:37:25 +00:00
|
|
|
subvol_fd = openat(send->mnt_fd, subvol, O_RDONLY | O_NOATIME);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (subvol_fd < 0) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("cannot open %s: %m", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = pipe(pipefd);
|
|
|
|
if (ret < 0) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("pipe failed: %m");
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2013-01-20 21:04:18 +00:00
|
|
|
memset(&io_send, 0, sizeof(io_send));
|
2012-07-25 21:08:25 +00:00
|
|
|
io_send.send_fd = pipefd[1];
|
|
|
|
send->send_fd = pipefd[0];
|
2022-03-17 17:25:52 +00:00
|
|
|
io_send.flags = flags;
|
|
|
|
|
|
|
|
if (send->proto_supported > 1) {
|
|
|
|
/*
|
|
|
|
* Versioned stream supported, requesting default or specific
|
|
|
|
* number.
|
|
|
|
*/
|
|
|
|
io_send.version = send->proto;
|
|
|
|
io_send.flags |= BTRFS_SEND_FLAG_VERSION;
|
2022-10-17 03:52:31 +00:00
|
|
|
|
|
|
|
fcntl(pipefd[0], F_SETPIPE_SZ, BTRFS_SEND_BUF_SIZE_V2);
|
|
|
|
fcntl(pipefd[1], F_SETPIPE_SZ, BTRFS_SEND_BUF_SIZE_V2);
|
2022-03-17 17:25:52 +00:00
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
|
|
|
|
if (!ret)
|
2016-11-02 14:04:47 +00:00
|
|
|
ret = pthread_create(&t_read, NULL, read_sent_data, send);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (ret) {
|
|
|
|
ret = -ret;
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("thread setup failed: %m");
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
io_send.clone_sources = (__u64*)send->clone_sources;
|
|
|
|
io_send.clone_sources_count = send->clone_sources_count;
|
2012-11-06 20:47:07 +00:00
|
|
|
io_send.parent_root = parent_root_id;
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
if (!is_first_subvol)
|
|
|
|
io_send.flags |= BTRFS_SEND_FLAG_OMIT_STREAM_HEADER;
|
|
|
|
if (!is_last_subvol)
|
|
|
|
io_send.flags |= BTRFS_SEND_FLAG_OMIT_END_CMD;
|
2012-07-25 21:08:25 +00:00
|
|
|
ret = ioctl(subvol_fd, BTRFS_IOC_SEND, &io_send);
|
2016-01-12 12:35:50 +00:00
|
|
|
if (ret < 0) {
|
2012-07-25 21:08:25 +00:00
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("send ioctl failed with %d: %m", ret);
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
if (ret == -EINVAL && (!is_first_subvol || !is_last_subvol))
|
2022-10-05 12:27:57 +00:00
|
|
|
pr_stderr(LOG_DEFAULT,
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
"Try upgrading your kernel or don't use -e.\n");
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2022-10-05 12:27:57 +00:00
|
|
|
pr_stderr(LOG_INFO, "BTRFS_IOC_SEND returned %d\n", ret);
|
|
|
|
pr_stderr(LOG_DEBUG, "joining genl thread\n");
|
2012-07-25 21:08:25 +00:00
|
|
|
|
|
|
|
close(pipefd[1]);
|
2013-06-26 15:17:57 +00:00
|
|
|
pipefd[1] = -1;
|
2012-07-25 21:08:25 +00:00
|
|
|
|
|
|
|
ret = pthread_join(t_read, &t_err);
|
|
|
|
if (ret) {
|
|
|
|
ret = -ret;
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("pthread_join failed: %m");
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (t_err) {
|
|
|
|
ret = (long int)t_err;
|
2022-10-26 07:46:22 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("failed to process send stream, ret=%ld (%m)",
|
|
|
|
(long int)t_err);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (subvol_fd != -1)
|
|
|
|
close(subvol_fd);
|
2013-01-30 07:11:46 +00:00
|
|
|
if (pipefd[0] != -1)
|
2012-07-25 21:08:25 +00:00
|
|
|
close(pipefd[0]);
|
2013-01-30 07:11:46 +00:00
|
|
|
if (pipefd[1] != -1)
|
2012-07-25 21:08:25 +00:00
|
|
|
close(pipefd[1]);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
static int init_root_path(struct btrfs_send *sctx, const char *subvol)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
if (sctx->root_path)
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
ret = find_mount_root(subvol, &sctx->root_path);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (ret < 0) {
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("failed to determine mount point for %s: %m", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2014-07-23 05:47:35 +00:00
|
|
|
if (ret > 0) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("%s doesn't belong to btrfs mount point", subvol);
|
2014-07-23 05:47:35 +00:00
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
sctx->mnt_fd = open(sctx->root_path, O_RDONLY | O_NOATIME);
|
|
|
|
if (sctx->mnt_fd < 0) {
|
2012-07-25 21:08:25 +00:00
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("cannot open '%s': %m", sctx->root_path);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret < 0) {
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("failed to initialize subvol search: %m");
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
static int is_subvol_ro(struct btrfs_send *sctx, const char *subvol)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u64 flags;
|
|
|
|
int fd = -1;
|
|
|
|
|
2016-11-02 13:14:56 +00:00
|
|
|
fd = openat(sctx->mnt_fd, subvol, O_RDONLY | O_NOATIME);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (fd < 0) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("cannot open %s: %m", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = ioctl(fd, BTRFS_IOC_SUBVOL_GETFLAGS, &flags);
|
|
|
|
if (ret < 0) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("failed to get flags for subvolume %s: %m", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & BTRFS_SUBVOL_RDONLY)
|
|
|
|
ret = 1;
|
|
|
|
else
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (fd != -1)
|
|
|
|
close(fd);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-11-04 08:33:58 +00:00
|
|
|
static int set_root_info(struct btrfs_send *sctx, const char *subvol,
|
|
|
|
u64 *root_id)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = init_root_path(sctx, subvol);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = get_root_id(sctx, subvol_strip_mountpoint(sctx->root_path, subvol),
|
|
|
|
root_id);
|
|
|
|
if (ret < 0) {
|
|
|
|
error("cannot resolve rootid for %s", subvol);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void free_send_info(struct btrfs_send *sctx)
|
|
|
|
{
|
|
|
|
if (sctx->mnt_fd >= 0) {
|
|
|
|
close(sctx->mnt_fd);
|
|
|
|
sctx->mnt_fd = -1;
|
|
|
|
}
|
|
|
|
free(sctx->root_path);
|
|
|
|
sctx->root_path = NULL;
|
|
|
|
}
|
|
|
|
|
2022-03-17 17:25:52 +00:00
|
|
|
static u32 get_sysfs_proto_supported(void)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
int ret;
|
|
|
|
char buf[32] = {};
|
|
|
|
char *end = NULL;
|
|
|
|
u64 version;
|
|
|
|
|
|
|
|
fd = sysfs_open_file("features/send_stream_version");
|
|
|
|
if (fd < 0) {
|
|
|
|
/*
|
|
|
|
* No file is either no version support or old kernel with just
|
|
|
|
* v1.
|
|
|
|
*/
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
ret = sysfs_read_file(fd, buf, sizeof(buf));
|
|
|
|
close(fd);
|
|
|
|
if (ret <= 0)
|
|
|
|
return 1;
|
|
|
|
version = strtoull(buf, &end, 10);
|
|
|
|
if (version == ULLONG_MAX && errno == ERANGE)
|
|
|
|
return 1;
|
|
|
|
if (version > U32_MAX) {
|
|
|
|
warning("sysfs/send_stream_version too big: %llu", version);
|
|
|
|
version = 1;
|
|
|
|
}
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
2018-03-07 02:07:03 +00:00
|
|
|
static const char * const cmd_send_usage[] = {
|
2018-03-07 02:04:15 +00:00
|
|
|
"btrfs send [-ve] [-p <parent>] [-c <clone-src>] [-f <outfile>] <subvol> [<subvol>...]",
|
|
|
|
"Send the subvolume(s) to stdout.",
|
|
|
|
"Sends the subvolume(s) specified by <subvol> to stdout.",
|
|
|
|
"<subvol> should be read-only here.",
|
|
|
|
"By default, this will send the whole subvolume. To do an incremental",
|
|
|
|
"send, use '-p <parent>'. If you want to allow btrfs to clone from",
|
|
|
|
"any additional local snapshots, use '-c <clone-src>' (multiple times",
|
|
|
|
"where applicable). You must not specify clone sources unless you",
|
|
|
|
"guarantee that these snapshots are exactly in the same state on both",
|
|
|
|
"sides, the sender and the receiver. It is allowed to omit the",
|
|
|
|
"'-p <parent>' option when '-c <clone-src>' options are given, in",
|
|
|
|
"which case 'btrfs send' will determine a suitable parent among the",
|
|
|
|
"clone sources itself.",
|
2019-06-21 13:36:04 +00:00
|
|
|
"",
|
2023-02-22 00:37:18 +00:00
|
|
|
OPTLINE("-e", "if sending multiple subvols at once, use the new format and omit the end-cmd between the subvols"),
|
|
|
|
OPTLINE("-p <parent>", "send an incremental stream from <parent> to <subvol>"),
|
|
|
|
OPTLINE("-c <clone-src>", "Use this snapshot as a clone source for an incremental send (multiple allowed)"),
|
|
|
|
OPTLINE("-f <outfile>", "Output is normally written to stdout. To write to "
|
|
|
|
"a file, use this option. An alternative would be to use pipes."),
|
|
|
|
OPTLINE("--no-data", "send in NO_FILE_DATA mode, Note: the output stream "
|
|
|
|
"does not contain any file data and thus cannot be used "
|
|
|
|
"to transfer changes. This mode is faster and useful to "
|
|
|
|
"show the differences in metadata."),
|
|
|
|
OPTLINE("--proto N", "use protocol version N, or 0 to use the highest version "
|
|
|
|
"supported by the sending kernel (default: 1)"),
|
|
|
|
OPTLINE("--compressed-data", "send data that is compressed on the filesystem directly without decompressing it"),
|
|
|
|
OPTLINE("-v|--verbose", "deprecated, alias for global -v option"),
|
|
|
|
OPTLINE("-q|--quiet", "deprecated, alias for global -q option"),
|
2019-11-25 10:39:04 +00:00
|
|
|
HELPINFO_INSERT_GLOBALS,
|
|
|
|
HELPINFO_INSERT_VERBOSE,
|
|
|
|
HELPINFO_INSERT_QUIET,
|
2018-03-07 02:04:15 +00:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2018-03-07 02:24:46 +00:00
|
|
|
static int cmd_send(const struct cmd_struct *cmd, int argc, char **argv)
|
2012-07-25 21:08:25 +00:00
|
|
|
{
|
|
|
|
char *subvol = NULL;
|
|
|
|
int ret;
|
2015-06-12 14:42:41 +00:00
|
|
|
char outname[PATH_MAX];
|
2012-07-25 21:08:25 +00:00
|
|
|
struct btrfs_send send;
|
|
|
|
u32 i;
|
|
|
|
char *mount_root = NULL;
|
|
|
|
char *snapshot_parent = NULL;
|
2014-06-05 08:13:39 +00:00
|
|
|
u64 root_id = 0;
|
2012-07-25 21:08:25 +00:00
|
|
|
u64 parent_root_id = 0;
|
2022-10-05 18:02:13 +00:00
|
|
|
bool full_send = true;
|
|
|
|
bool new_end_cmd_semantic = false;
|
2015-06-12 11:15:41 +00:00
|
|
|
u64 send_flags = 0;
|
2022-03-17 17:25:52 +00:00
|
|
|
u64 proto = 0;
|
2012-07-25 21:08:25 +00:00
|
|
|
|
|
|
|
memset(&send, 0, sizeof(send));
|
|
|
|
send.dump_fd = fileno(stdout);
|
2022-03-17 17:25:52 +00:00
|
|
|
send.proto = 1;
|
2015-06-12 14:42:41 +00:00
|
|
|
outname[0] = 0;
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2019-11-25 10:39:04 +00:00
|
|
|
/*
|
|
|
|
* For send, verbose default is 1 (insteasd of 0) for historical reasons,
|
|
|
|
* changing may break scripts that expect the 'At subvol' message. But do
|
|
|
|
* it only when bconf.verbose is unset (-1) and also adjust the value,
|
|
|
|
* if global verbose is already set.
|
|
|
|
*/
|
|
|
|
if (bconf.verbose == BTRFS_BCONF_UNSET)
|
|
|
|
bconf.verbose = 1;
|
|
|
|
else if (bconf.verbose > BTRFS_BCONF_QUIET)
|
|
|
|
bconf.verbose++;
|
|
|
|
|
2018-06-20 00:38:38 +00:00
|
|
|
optind = 0;
|
2015-06-12 11:15:41 +00:00
|
|
|
while (1) {
|
2022-03-17 17:25:52 +00:00
|
|
|
enum {
|
2022-06-21 00:20:21 +00:00
|
|
|
GETOPT_VAL_SEND_NO_DATA = GETOPT_VAL_FIRST,
|
2022-03-17 17:25:52 +00:00
|
|
|
GETOPT_VAL_PROTO,
|
|
|
|
GETOPT_VAL_COMPRESSED_DATA,
|
|
|
|
};
|
2015-06-12 11:15:41 +00:00
|
|
|
static const struct option long_options[] = {
|
2016-05-12 19:49:25 +00:00
|
|
|
{ "verbose", no_argument, NULL, 'v' },
|
|
|
|
{ "quiet", no_argument, NULL, 'q' },
|
2020-11-18 19:18:44 +00:00
|
|
|
{ "no-data", no_argument, NULL, GETOPT_VAL_SEND_NO_DATA },
|
2022-03-17 17:25:52 +00:00
|
|
|
{ "proto", required_argument, NULL, GETOPT_VAL_PROTO },
|
|
|
|
{ "compressed-data", no_argument, NULL, GETOPT_VAL_COMPRESSED_DATA },
|
2020-11-18 19:18:44 +00:00
|
|
|
{ NULL, 0, NULL, 0 }
|
2015-06-12 11:15:41 +00:00
|
|
|
};
|
2016-05-12 19:49:25 +00:00
|
|
|
int c = getopt_long(argc, argv, "vqec:f:i:p:", long_options, NULL);
|
2015-06-12 11:15:41 +00:00
|
|
|
|
|
|
|
if (c < 0)
|
|
|
|
break;
|
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
switch (c) {
|
|
|
|
case 'v':
|
2019-11-25 10:39:04 +00:00
|
|
|
bconf_be_verbose();
|
2012-07-25 21:08:25 +00:00
|
|
|
break;
|
2016-05-12 19:49:25 +00:00
|
|
|
case 'q':
|
2019-11-25 10:39:04 +00:00
|
|
|
bconf_be_quiet();
|
2016-05-12 19:49:25 +00:00
|
|
|
break;
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
case 'e':
|
2022-10-05 18:02:13 +00:00
|
|
|
new_end_cmd_semantic = true;
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
break;
|
2012-11-06 20:47:07 +00:00
|
|
|
case 'c':
|
2012-07-25 21:08:25 +00:00
|
|
|
subvol = realpath(optarg, NULL);
|
|
|
|
if (!subvol) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("realpath %s failed: %m\n", optarg);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2016-11-04 08:33:58 +00:00
|
|
|
ret = set_root_info(&send, subvol, &root_id);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2013-12-06 18:00:49 +00:00
|
|
|
ret = is_subvol_ro(&send, subvol);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
if (!ret) {
|
|
|
|
ret = -EINVAL;
|
2016-01-11 10:01:47 +00:00
|
|
|
error("cloned subvolume %s is not read-only", subvol);
|
2013-12-06 18:00:49 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2014-12-19 15:55:32 +00:00
|
|
|
ret = add_clone_source(&send, root_id);
|
|
|
|
if (ret < 0) {
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("cannot add clone source: %m");
|
2014-12-19 15:55:32 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
free(subvol);
|
2013-04-23 09:25:00 +00:00
|
|
|
subvol = NULL;
|
2016-11-04 08:33:58 +00:00
|
|
|
free_send_info(&send);
|
2022-10-05 18:02:13 +00:00
|
|
|
full_send = false;
|
2012-07-25 21:08:25 +00:00
|
|
|
break;
|
|
|
|
case 'f':
|
2015-06-12 14:42:41 +00:00
|
|
|
if (arg_copy_path(outname, optarg, sizeof(outname))) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("output file path too long (%zu)", strlen(optarg));
|
2015-06-12 14:42:41 +00:00
|
|
|
ret = 1;
|
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
break;
|
|
|
|
case 'p':
|
2012-11-06 20:47:07 +00:00
|
|
|
if (snapshot_parent) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("you cannot have more than one parent (-p)");
|
2013-04-23 09:25:00 +00:00
|
|
|
ret = 1;
|
|
|
|
goto out;
|
2012-11-06 20:47:07 +00:00
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
snapshot_parent = realpath(optarg, NULL);
|
|
|
|
if (!snapshot_parent) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("realpath %s failed: %m", optarg);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2013-12-06 18:00:49 +00:00
|
|
|
|
|
|
|
ret = is_subvol_ro(&send, snapshot_parent);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
if (!ret) {
|
|
|
|
ret = -EINVAL;
|
2016-01-11 10:01:47 +00:00
|
|
|
error("parent subvolume %s is not read-only",
|
2013-12-06 18:00:49 +00:00
|
|
|
snapshot_parent);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2022-10-05 18:02:13 +00:00
|
|
|
full_send = false;
|
2012-07-25 21:08:25 +00:00
|
|
|
break;
|
2012-11-06 20:47:07 +00:00
|
|
|
case 'i':
|
2016-01-11 10:01:47 +00:00
|
|
|
error("option -i was removed, use -c instead");
|
2013-04-23 09:25:00 +00:00
|
|
|
ret = 1;
|
|
|
|
goto out;
|
2015-06-12 11:15:41 +00:00
|
|
|
case GETOPT_VAL_SEND_NO_DATA:
|
|
|
|
send_flags |= BTRFS_SEND_FLAG_NO_FILE_DATA;
|
|
|
|
break;
|
2022-03-17 17:25:52 +00:00
|
|
|
case GETOPT_VAL_PROTO:
|
|
|
|
proto = arg_strtou64(optarg);
|
|
|
|
if (proto > U32_MAX) {
|
|
|
|
error("protocol version number too big %llu", proto);
|
|
|
|
ret = 1;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
send.proto = proto;
|
|
|
|
break;
|
|
|
|
case GETOPT_VAL_COMPRESSED_DATA:
|
|
|
|
send_flags |= BTRFS_SEND_FLAG_COMPRESSED;
|
|
|
|
break;
|
2012-07-25 21:08:25 +00:00
|
|
|
default:
|
2018-03-07 02:54:38 +00:00
|
|
|
usage_unknown_option(cmd, argv);
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-16 03:59:46 +00:00
|
|
|
if (check_argc_min(argc - optind, 1))
|
2019-03-04 13:49:15 +00:00
|
|
|
return 1;
|
2012-07-25 21:08:25 +00:00
|
|
|
|
2015-06-12 14:42:41 +00:00
|
|
|
if (outname[0]) {
|
2016-11-07 15:51:20 +00:00
|
|
|
int tmpfd;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Try to use an existing file first. Even if send runs as
|
|
|
|
* root, it might not have permissions to create file (eg. on a
|
|
|
|
* NFS) but it should still be able to use a pre-created file.
|
|
|
|
*/
|
|
|
|
tmpfd = open(outname, O_WRONLY | O_TRUNC);
|
|
|
|
if (tmpfd < 0) {
|
|
|
|
if (errno == ENOENT)
|
|
|
|
tmpfd = open(outname,
|
|
|
|
O_CREAT | O_WRONLY | O_TRUNC, 0600);
|
|
|
|
}
|
|
|
|
send.dump_fd = tmpfd;
|
2012-07-25 21:08:25 +00:00
|
|
|
if (send.dump_fd == -1) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("cannot create '%s': %m", outname);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-31 08:21:24 +00:00
|
|
|
if (isatty(send.dump_fd)) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error(
|
|
|
|
"not dumping send stream into a terminal, redirect it into a file");
|
2013-04-23 09:25:00 +00:00
|
|
|
ret = 1;
|
|
|
|
goto out;
|
2013-01-31 08:21:24 +00:00
|
|
|
}
|
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
/* use first send subvol to determine mount_root */
|
2012-08-09 16:48:29 +00:00
|
|
|
subvol = realpath(argv[optind], NULL);
|
|
|
|
if (!subvol) {
|
2012-08-10 02:46:16 +00:00
|
|
|
ret = -errno;
|
2016-01-11 10:01:47 +00:00
|
|
|
error("unable to resolve %s", argv[optind]);
|
2012-08-09 16:48:29 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
ret = init_root_path(&send, subvol);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (snapshot_parent != NULL) {
|
|
|
|
ret = get_root_id(&send,
|
2016-03-21 07:21:05 +00:00
|
|
|
subvol_strip_mountpoint(send.root_path, snapshot_parent),
|
|
|
|
&parent_root_id);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (ret < 0) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("could not resolve rootid for %s", snapshot_parent);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2014-12-19 15:55:32 +00:00
|
|
|
ret = add_clone_source(&send, parent_root_id);
|
|
|
|
if (ret < 0) {
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("cannot add clone source: %m");
|
2014-12-19 15:55:32 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = optind; i < argc; i++) {
|
2013-04-23 09:25:00 +00:00
|
|
|
free(subvol);
|
2012-08-09 16:48:29 +00:00
|
|
|
subvol = realpath(argv[i], NULL);
|
|
|
|
if (!subvol) {
|
2012-08-10 02:46:16 +00:00
|
|
|
ret = -errno;
|
2016-01-11 10:01:47 +00:00
|
|
|
error("unable to resolve %s", argv[i]);
|
2012-08-09 16:48:29 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
|
|
|
|
ret = find_mount_root(subvol, &mount_root);
|
|
|
|
if (ret < 0) {
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("find_mount_root failed on %s: %m", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2014-07-23 05:47:35 +00:00
|
|
|
if (ret > 0) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("%s does not belong to btrfs mount point",
|
2014-07-23 05:47:35 +00:00
|
|
|
subvol);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
if (strcmp(send.root_path, mount_root) != 0) {
|
|
|
|
ret = -EINVAL;
|
2016-01-11 10:01:47 +00:00
|
|
|
error("all subvolumes must be from the same filesystem");
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
free(mount_root);
|
|
|
|
|
|
|
|
ret = is_subvol_ro(&send, subvol);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
if (!ret) {
|
|
|
|
ret = -EINVAL;
|
2016-05-11 23:50:36 +00:00
|
|
|
error("subvolume %s is not read-only", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-25 10:39:04 +00:00
|
|
|
if ((send_flags & BTRFS_SEND_FLAG_NO_FILE_DATA) && bconf.verbose > 1)
|
|
|
|
if (bconf.verbose > 1)
|
2022-10-05 12:27:57 +00:00
|
|
|
pr_stderr(LOG_DEFAULT, "Mode NO_FILE_DATA enabled\n");
|
2022-03-17 17:25:52 +00:00
|
|
|
send.proto_supported = get_sysfs_proto_supported();
|
|
|
|
if (send.proto_supported == 1) {
|
|
|
|
if (send.proto > send.proto_supported) {
|
|
|
|
error("requested version %u but kernel supports only %u",
|
|
|
|
send.proto, send.proto_supported);
|
|
|
|
ret = -EPROTO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (send_flags & BTRFS_SEND_FLAG_COMPRESSED) {
|
|
|
|
/*
|
|
|
|
* If no protocol version was explicitly requested, then
|
|
|
|
* --compressed-data implies --proto 2.
|
|
|
|
*/
|
|
|
|
if (send.proto == 1 && !proto)
|
|
|
|
send.proto = 2;
|
|
|
|
|
|
|
|
if (send.proto == 1) {
|
|
|
|
error("--compressed-data requires protocol version >= 2 (requested 1)");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
} else if (send.proto == 0 && send.proto_supported < 2) {
|
|
|
|
error("kernel does not support --compressed-data");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
2022-10-05 12:27:57 +00:00
|
|
|
pr_stderr(LOG_INFO, "Protocol version requested: %u (supported %u)\n",
|
|
|
|
send.proto, send.proto_supported);
|
2015-06-12 11:15:41 +00:00
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
for (i = optind; i < argc; i++) {
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
int is_first_subvol;
|
|
|
|
int is_last_subvol;
|
|
|
|
|
2013-04-23 09:25:00 +00:00
|
|
|
free(subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
subvol = argv[i];
|
|
|
|
|
2022-10-05 12:27:57 +00:00
|
|
|
pr_stderr(LOG_DEFAULT, "At subvol %s\n", subvol);
|
2012-07-25 21:08:25 +00:00
|
|
|
|
|
|
|
subvol = realpath(subvol, NULL);
|
|
|
|
if (!subvol) {
|
|
|
|
ret = -errno;
|
2018-10-25 12:10:54 +00:00
|
|
|
error("realpath %s failed: %m", argv[i]);
|
2012-07-25 21:08:25 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2016-11-15 08:16:27 +00:00
|
|
|
if (!full_send && !snapshot_parent) {
|
2016-11-04 08:33:58 +00:00
|
|
|
ret = set_root_info(&send, subvol, &root_id);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2012-07-25 21:08:25 +00:00
|
|
|
ret = find_good_parent(&send, root_id, &parent_root_id);
|
2012-11-06 20:47:07 +00:00
|
|
|
if (ret < 0) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("parent determination failed for %lld",
|
2012-11-06 20:47:07 +00:00
|
|
|
root_id);
|
|
|
|
goto out;
|
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
|
|
|
|
Btrfs-progs: add send option for using new end-cmd semantic
This commit adds a command line option to enable sending streams
which make use of the new end-cmd semantic if multiple snapshots are
sent back-to-back. The goal is to use the <end cmd> as an indication
to stop reading the input stream. So far, the receiver could only
use EOF to recognize the end.
If the new command line option '-e' is set, this commit requires a
kernel which is able to support the new flags in the send ioctl. New
bits in the flags of the send ioctl will be set which cause EINVAL
on old kernels. However, if the option '-e' is not set, it works
with old and new kernels without any errors or any changed behavior.
This used to be the encoding (with 2 snapshots in this example):
<stream header> + <sequence of commands> + <end cmd> +
<stream header> + <sequence of commands> + <end cmd> + EOF
The new format (if the two new flags are used) is this one:
<stream header> + <sequence of commands> +
<sequence of commands> + <end cmd>
Note that the currently existing receivers treat <end cmd> only as
an indication that a new <stream header> is following. This means,
you can just skip the sequence <end cmd> <stream header> without
loosing compatibility. As long as an EOF is following, the currently
existing receivers handle the new format (if the two new flags are
used) exactly as the old one.
Also note that the kernel interface was changed in a way that is
backward compatible to old btrfs-progs tools. You set one or two bits
in the flags field of the ioctl to enable the new behavior. Old tools
set these flags to zero, thus getting exactly the same as they got
with older kernels. And this is exactly what happens if the new '-e'
option is not set, the new bits in the flags are not set and thus
old kernels and new kernels are both supported.
So what is the benefit of this change? The goal is to be able to use
a single stream (one TCP connection) to multiplex a request/response
handshake plus Btrfs send streams, all in the same stream. In this
case you cannot evaluate an EOF condition as an end of the Btrfs send
stream. You need something else, and the <end cmd> is just perfect
for this purpose.
Signed-off-by: Stefan Behrens <sbehrens@giantdisaster.de>
Signed-off-by: David Sterba <dsterba@suse.cz>
Signed-off-by: Chris Mason <chris.mason@fusionio.com>
2013-04-10 17:10:56 +00:00
|
|
|
if (new_end_cmd_semantic) {
|
|
|
|
/* require new kernel */
|
|
|
|
is_first_subvol = (i == optind);
|
|
|
|
is_last_subvol = (i == argc - 1);
|
|
|
|
} else {
|
|
|
|
/* be compatible to old and new kernel */
|
|
|
|
is_first_subvol = 1;
|
|
|
|
is_last_subvol = 1;
|
|
|
|
}
|
2013-11-29 16:37:25 +00:00
|
|
|
ret = do_send(&send, parent_root_id, is_first_subvol,
|
2015-06-12 11:15:41 +00:00
|
|
|
is_last_subvol, subvol, send_flags);
|
2012-07-25 21:08:25 +00:00
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2016-11-15 08:16:27 +00:00
|
|
|
if (!full_send && !snapshot_parent) {
|
2016-03-24 07:47:28 +00:00
|
|
|
/* done with this subvol, so add it to the clone sources */
|
|
|
|
ret = add_clone_source(&send, root_id);
|
|
|
|
if (ret < 0) {
|
2018-10-25 12:10:54 +00:00
|
|
|
errno = -ret;
|
|
|
|
error("cannot add clone source: %m");
|
2016-03-24 07:47:28 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2016-11-04 08:33:58 +00:00
|
|
|
free_send_info(&send);
|
2014-12-19 15:55:32 +00:00
|
|
|
}
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
2013-04-23 09:25:00 +00:00
|
|
|
free(subvol);
|
|
|
|
free(snapshot_parent);
|
|
|
|
free(send.clone_sources);
|
2016-11-04 08:33:58 +00:00
|
|
|
free_send_info(&send);
|
2013-09-04 15:22:29 +00:00
|
|
|
return !!ret;
|
2012-07-25 21:08:25 +00:00
|
|
|
}
|
2018-03-07 02:07:03 +00:00
|
|
|
DEFINE_SIMPLE_COMMAND(send, "send");
|