zfs should optionally send holds

Add -h switch to zfs send command to send dataset holds. If
holds are present in the stream, zfs receive will create them
on the target dataset, unless the zfs receive -h option is used
to skip receive of holds.

Reviewed-by: Alek Pinchuk <apinchuk@datto.com>
Reviewed-by: loli10K <ezomori.nozomu@gmail.com>
Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov>
Reviewed by: Paul Dagnelie <pcd@delphix.com>
Signed-off-by: Paul Zuchowski <pzuchowski@datto.com>
Closes #7513
This commit is contained in:
Paul Zuchowski 2019-02-15 15:41:38 -05:00 committed by Brian Behlendorf
parent e73ab1b38c
commit 9c5e88b1de
10 changed files with 324 additions and 28 deletions

View File

@ -278,10 +278,10 @@ get_usage(zfs_help_t idx)
case HELP_PROMOTE: case HELP_PROMOTE:
return (gettext("\tpromote <clone-filesystem>\n")); return (gettext("\tpromote <clone-filesystem>\n"));
case HELP_RECEIVE: case HELP_RECEIVE:
return (gettext("\treceive [-vnsFu] " return (gettext("\treceive [-vnsFhu] "
"[-o <property>=<value>] ... [-x <property>] ...\n" "[-o <property>=<value>] ... [-x <property>] ...\n"
"\t <filesystem|volume|snapshot>\n" "\t <filesystem|volume|snapshot>\n"
"\treceive [-vnsFu] [-o <property>=<value>] ... " "\treceive [-vnsFhu] [-o <property>=<value>] ... "
"[-x <property>] ... \n" "[-x <property>] ... \n"
"\t [-d | -e] <filesystem>\n" "\t [-d | -e] <filesystem>\n"
"\treceive -A <filesystem|volume>\n")); "\treceive -A <filesystem|volume>\n"));
@ -293,7 +293,7 @@ get_usage(zfs_help_t idx)
case HELP_ROLLBACK: case HELP_ROLLBACK:
return (gettext("\trollback [-rRf] <snapshot>\n")); return (gettext("\trollback [-rRf] <snapshot>\n"));
case HELP_SEND: case HELP_SEND:
return (gettext("\tsend [-DnPpRvLecwb] [-[i|I] snapshot] " return (gettext("\tsend [-DnPpRvLecwhb] [-[i|I] snapshot] "
"<snapshot>\n" "<snapshot>\n"
"\tsend [-nvPLecw] [-i snapshot|bookmark] " "\tsend [-nvPLecw] [-i snapshot|bookmark] "
"<filesystem|volume|snapshot>\n" "<filesystem|volume|snapshot>\n"
@ -3981,11 +3981,12 @@ zfs_do_send(int argc, char **argv)
{"compressed", no_argument, NULL, 'c'}, {"compressed", no_argument, NULL, 'c'},
{"raw", no_argument, NULL, 'w'}, {"raw", no_argument, NULL, 'w'},
{"backup", no_argument, NULL, 'b'}, {"backup", no_argument, NULL, 'b'},
{"holds", no_argument, NULL, 'h'},
{0, 0, 0, 0} {0, 0, 0, 0}
}; };
/* check options */ /* check options */
while ((c = getopt_long(argc, argv, ":i:I:RDpvnPLet:cwb", long_options, while ((c = getopt_long(argc, argv, ":i:I:RDpvnPLeht:cwb", long_options,
NULL)) != -1) { NULL)) != -1) {
switch (c) { switch (c) {
case 'i': case 'i':
@ -4008,6 +4009,9 @@ zfs_do_send(int argc, char **argv)
case 'b': case 'b':
flags.backup = B_TRUE; flags.backup = B_TRUE;
break; break;
case 'h':
flags.holds = B_TRUE;
break;
case 'P': case 'P':
flags.parsable = B_TRUE; flags.parsable = B_TRUE;
flags.verbose = B_TRUE; flags.verbose = B_TRUE;
@ -4130,7 +4134,7 @@ zfs_do_send(int argc, char **argv)
char frombuf[ZFS_MAX_DATASET_NAME_LEN]; char frombuf[ZFS_MAX_DATASET_NAME_LEN];
if (flags.replicate || flags.doall || flags.props || if (flags.replicate || flags.doall || flags.props ||
flags.backup || flags.dedup || flags.backup || flags.dedup || flags.holds ||
(strchr(argv[0], '@') == NULL && (strchr(argv[0], '@') == NULL &&
(flags.dryrun || flags.verbose || flags.progress))) { (flags.dryrun || flags.verbose || flags.progress))) {
(void) fprintf(stderr, gettext("Error: " (void) fprintf(stderr, gettext("Error: "
@ -4235,7 +4239,7 @@ zfs_do_receive(int argc, char **argv)
nomem(); nomem();
/* check options */ /* check options */
while ((c = getopt(argc, argv, ":o:x:denuvFsA")) != -1) { while ((c = getopt(argc, argv, ":o:x:dehnuvFsA")) != -1) {
switch (c) { switch (c) {
case 'o': case 'o':
if (!parseprop(props, optarg)) { if (!parseprop(props, optarg)) {
@ -4267,6 +4271,9 @@ zfs_do_receive(int argc, char **argv)
} }
flags.istail = B_TRUE; flags.istail = B_TRUE;
break; break;
case 'h':
flags.skipholds = B_TRUE;
break;
case 'n': case 'n':
flags.dryrun = B_TRUE; flags.dryrun = B_TRUE;
break; break;

View File

@ -645,6 +645,9 @@ typedef struct sendflags {
/* only send received properties (ie. -b) */ /* only send received properties (ie. -b) */
boolean_t backup; boolean_t backup;
/* include snapshot holds in send stream */
boolean_t holds;
} sendflags_t; } sendflags_t;
typedef boolean_t (snapfilter_cb_t)(zfs_handle_t *, void *); typedef boolean_t (snapfilter_cb_t)(zfs_handle_t *, void *);
@ -707,6 +710,12 @@ typedef struct recvflags {
/* do not mount file systems as they are extracted (private) */ /* do not mount file systems as they are extracted (private) */
boolean_t nomount; boolean_t nomount;
/* Was holds flag set in the compound header? */
boolean_t holds;
/* skip receive of snapshot holds */
boolean_t skipholds;
} recvflags_t; } recvflags_t;
extern int zfs_receive(libzfs_handle_t *, const char *, nvlist_t *, extern int zfs_receive(libzfs_handle_t *, const char *, nvlist_t *,

View File

@ -106,6 +106,7 @@ typedef enum drr_headertype {
#define DMU_BACKUP_FEATURE_LARGE_DNODE (1 << 23) #define DMU_BACKUP_FEATURE_LARGE_DNODE (1 << 23)
#define DMU_BACKUP_FEATURE_RAW (1 << 24) #define DMU_BACKUP_FEATURE_RAW (1 << 24)
/* flag #25 is reserved for the ZSTD compression feature */ /* flag #25 is reserved for the ZSTD compression feature */
#define DMU_BACKUP_FEATURE_HOLDS (1 << 26)
/* /*
* Mask of all supported backup features * Mask of all supported backup features
@ -115,7 +116,7 @@ typedef enum drr_headertype {
DMU_BACKUP_FEATURE_EMBED_DATA | DMU_BACKUP_FEATURE_LZ4 | \ DMU_BACKUP_FEATURE_EMBED_DATA | DMU_BACKUP_FEATURE_LZ4 | \
DMU_BACKUP_FEATURE_RESUMING | DMU_BACKUP_FEATURE_LARGE_BLOCKS | \ DMU_BACKUP_FEATURE_RESUMING | DMU_BACKUP_FEATURE_LARGE_BLOCKS | \
DMU_BACKUP_FEATURE_COMPRESSED | DMU_BACKUP_FEATURE_LARGE_DNODE | \ DMU_BACKUP_FEATURE_COMPRESSED | DMU_BACKUP_FEATURE_LARGE_DNODE | \
DMU_BACKUP_FEATURE_RAW) DMU_BACKUP_FEATURE_RAW | DMU_BACKUP_FEATURE_HOLDS)
/* Are all features in the given flag word currently supported? */ /* Are all features in the given flag word currently supported? */
#define DMU_STREAM_SUPPORTED(x) (!((x) & ~DMU_BACKUP_FEATURE_MASK)) #define DMU_STREAM_SUPPORTED(x) (!((x) & ~DMU_BACKUP_FEATURE_MASK))

View File

@ -29,6 +29,7 @@
* Copyright 2015, OmniTI Computer Consulting, Inc. All rights reserved. * Copyright 2015, OmniTI Computer Consulting, Inc. All rights reserved.
* Copyright 2016 Igor Kozhukhov <ikozhukhov@gmail.com> * Copyright 2016 Igor Kozhukhov <ikozhukhov@gmail.com>
* Copyright (c) 2018, loli10K <ezomori.nozomu@gmail.com>. All rights reserved. * Copyright (c) 2018, loli10K <ezomori.nozomu@gmail.com>. All rights reserved.
* Copyright (c) 2018 Datto Inc.
*/ */
#include <assert.h> #include <assert.h>
@ -616,6 +617,7 @@ typedef struct send_data {
nvlist_t *parent_snaps; nvlist_t *parent_snaps;
nvlist_t *fss; nvlist_t *fss;
nvlist_t *snapprops; nvlist_t *snapprops;
nvlist_t *snapholds; /* user holds */
/* send-receive configuration, does not change during traversal */ /* send-receive configuration, does not change during traversal */
const char *fsname; const char *fsname;
@ -627,6 +629,8 @@ typedef struct send_data {
boolean_t verbose; boolean_t verbose;
boolean_t seenfrom; boolean_t seenfrom;
boolean_t seento; boolean_t seento;
boolean_t holds; /* were holds requested with send -h */
boolean_t props;
/* /*
* The header nvlist is of the following format: * The header nvlist is of the following format:
@ -642,6 +646,7 @@ typedef struct send_data {
* "props" -> { name -> value (only if set here) } * "props" -> { name -> value (only if set here) }
* "snaps" -> { name (lastname) -> number (guid) } * "snaps" -> { name (lastname) -> number (guid) }
* "snapprops" -> { name (lastname) -> { name -> value } } * "snapprops" -> { name (lastname) -> { name -> value } }
* "snapholds" -> { name (lastname) -> { holdname -> crtime } }
* *
* "origin" -> number (guid) (if clone) * "origin" -> number (guid) (if clone)
* "is_encroot" -> boolean * "is_encroot" -> boolean
@ -712,6 +717,15 @@ send_iterate_snap(zfs_handle_t *zhp, void *arg)
send_iterate_prop(zhp, sd->backup, nv); send_iterate_prop(zhp, sd->backup, nv);
VERIFY(0 == nvlist_add_nvlist(sd->snapprops, snapname, nv)); VERIFY(0 == nvlist_add_nvlist(sd->snapprops, snapname, nv));
nvlist_free(nv); nvlist_free(nv);
if (sd->holds) {
nvlist_t *holds = fnvlist_alloc();
int err = lzc_get_holds(zhp->zfs_name, &holds);
if (err == 0) {
VERIFY(0 == nvlist_add_nvlist(sd->snapholds,
snapname, holds));
}
fnvlist_free(holds);
}
zfs_close(zhp); zfs_close(zhp);
return (0); return (0);
@ -893,9 +907,10 @@ send_iterate_fs(zfs_handle_t *zhp, void *arg)
} }
/* iterate over props */ /* iterate over props */
if (sd->props || sd->backup || sd->recursive) {
VERIFY(0 == nvlist_alloc(&nv, NV_UNIQUE_NAME, 0)); VERIFY(0 == nvlist_alloc(&nv, NV_UNIQUE_NAME, 0));
send_iterate_prop(zhp, sd->backup, nv); send_iterate_prop(zhp, sd->backup, nv);
}
if (zfs_prop_get_int(zhp, ZFS_PROP_ENCRYPTION) != ZIO_CRYPT_OFF) { if (zfs_prop_get_int(zhp, ZFS_PROP_ENCRYPTION) != ZIO_CRYPT_OFF) {
boolean_t encroot; boolean_t encroot;
@ -925,17 +940,24 @@ send_iterate_fs(zfs_handle_t *zhp, void *arg)
} }
if (nv != NULL)
VERIFY(0 == nvlist_add_nvlist(nvfs, "props", nv)); VERIFY(0 == nvlist_add_nvlist(nvfs, "props", nv));
/* iterate over snaps, and set sd->parent_fromsnap_guid */ /* iterate over snaps, and set sd->parent_fromsnap_guid */
sd->parent_fromsnap_guid = 0; sd->parent_fromsnap_guid = 0;
VERIFY(0 == nvlist_alloc(&sd->parent_snaps, NV_UNIQUE_NAME, 0)); VERIFY(0 == nvlist_alloc(&sd->parent_snaps, NV_UNIQUE_NAME, 0));
VERIFY(0 == nvlist_alloc(&sd->snapprops, NV_UNIQUE_NAME, 0)); VERIFY(0 == nvlist_alloc(&sd->snapprops, NV_UNIQUE_NAME, 0));
if (sd->holds)
VERIFY(0 == nvlist_alloc(&sd->snapholds, NV_UNIQUE_NAME, 0));
(void) zfs_iter_snapshots_sorted(zhp, send_iterate_snap, sd); (void) zfs_iter_snapshots_sorted(zhp, send_iterate_snap, sd);
VERIFY(0 == nvlist_add_nvlist(nvfs, "snaps", sd->parent_snaps)); VERIFY(0 == nvlist_add_nvlist(nvfs, "snaps", sd->parent_snaps));
VERIFY(0 == nvlist_add_nvlist(nvfs, "snapprops", sd->snapprops)); VERIFY(0 == nvlist_add_nvlist(nvfs, "snapprops", sd->snapprops));
if (sd->holds)
VERIFY(0 == nvlist_add_nvlist(nvfs, "snapholds",
sd->snapholds));
nvlist_free(sd->parent_snaps); nvlist_free(sd->parent_snaps);
nvlist_free(sd->snapprops); nvlist_free(sd->snapprops);
nvlist_free(sd->snapholds);
/* add this fs to nvlist */ /* add this fs to nvlist */
(void) snprintf(guidstring, sizeof (guidstring), (void) snprintf(guidstring, sizeof (guidstring),
@ -960,7 +982,8 @@ out:
static int static int
gather_nvlist(libzfs_handle_t *hdl, const char *fsname, const char *fromsnap, gather_nvlist(libzfs_handle_t *hdl, const char *fsname, const char *fromsnap,
const char *tosnap, boolean_t recursive, boolean_t raw, boolean_t verbose, const char *tosnap, boolean_t recursive, boolean_t raw, boolean_t verbose,
boolean_t backup, nvlist_t **nvlp, avl_tree_t **avlp) boolean_t backup, boolean_t holds, boolean_t props, nvlist_t **nvlp,
avl_tree_t **avlp)
{ {
zfs_handle_t *zhp; zfs_handle_t *zhp;
send_data_t sd = { 0 }; send_data_t sd = { 0 };
@ -978,6 +1001,8 @@ gather_nvlist(libzfs_handle_t *hdl, const char *fsname, const char *fromsnap,
sd.raw = raw; sd.raw = raw;
sd.verbose = verbose; sd.verbose = verbose;
sd.backup = backup; sd.backup = backup;
sd.holds = holds;
sd.props = props;
if ((error = send_iterate_fs(zhp, &sd)) != 0) { if ((error = send_iterate_fs(zhp, &sd)) != 0) {
nvlist_free(sd.fss); nvlist_free(sd.fss);
@ -1008,7 +1033,7 @@ typedef struct send_dump_data {
uint64_t prevsnap_obj; uint64_t prevsnap_obj;
boolean_t seenfrom, seento, replicate, doall, fromorigin; boolean_t seenfrom, seento, replicate, doall, fromorigin;
boolean_t verbose, dryrun, parsable, progress, embed_data, std_out; boolean_t verbose, dryrun, parsable, progress, embed_data, std_out;
boolean_t large_block, compress, raw; boolean_t large_block, compress, raw, holds;
int outfd; int outfd;
boolean_t err; boolean_t err;
nvlist_t *fss; nvlist_t *fss;
@ -1864,6 +1889,9 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
} }
} }
if (flags->holds)
featureflags |= DMU_BACKUP_FEATURE_HOLDS;
/* /*
* Start the dedup thread if this is a dedup stream. We do not bother * Start the dedup thread if this is a dedup stream. We do not bother
* doing this if this a raw send of an encrypted dataset with dedup off * doing this if this a raw send of an encrypted dataset with dedup off
@ -1891,7 +1919,8 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
} }
} }
if (flags->replicate || flags->doall || flags->props || flags->backup) { if (flags->replicate || flags->doall || flags->props ||
flags->holds || flags->backup) {
dmu_replay_record_t drr = { 0 }; dmu_replay_record_t drr = { 0 };
char *packbuf = NULL; char *packbuf = NULL;
size_t buflen = 0; size_t buflen = 0;
@ -1899,7 +1928,8 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
ZIO_SET_CHECKSUM(&zc, 0, 0, 0, 0); ZIO_SET_CHECKSUM(&zc, 0, 0, 0, 0);
if (flags->replicate || flags->props || flags->backup) { if (flags->replicate || flags->props || flags->backup ||
flags->holds) {
nvlist_t *hdrnv; nvlist_t *hdrnv;
VERIFY(0 == nvlist_alloc(&hdrnv, NV_UNIQUE_NAME, 0)); VERIFY(0 == nvlist_alloc(&hdrnv, NV_UNIQUE_NAME, 0));
@ -1918,7 +1948,8 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
err = gather_nvlist(zhp->zfs_hdl, zhp->zfs_name, err = gather_nvlist(zhp->zfs_hdl, zhp->zfs_name,
fromsnap, tosnap, flags->replicate, flags->raw, fromsnap, tosnap, flags->replicate, flags->raw,
flags->verbose, flags->backup, &fss, &fsavl); flags->verbose, flags->backup, flags->holds,
flags->props, &fss, &fsavl);
if (err) if (err)
goto err_out; goto err_out;
VERIFY(0 == nvlist_add_nvlist(hdrnv, "fss", fss)); VERIFY(0 == nvlist_add_nvlist(hdrnv, "fss", fss));
@ -1988,6 +2019,7 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
sdd.embed_data = flags->embed_data; sdd.embed_data = flags->embed_data;
sdd.compress = flags->compress; sdd.compress = flags->compress;
sdd.raw = flags->raw; sdd.raw = flags->raw;
sdd.holds = flags->holds;
sdd.filter_cb = filter_func; sdd.filter_cb = filter_func;
sdd.filter_cb_arg = cb_arg; sdd.filter_cb_arg = cb_arg;
if (debugnvp) if (debugnvp)
@ -2020,6 +2052,7 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
sdd.cleanup_fd = -1; sdd.cleanup_fd = -1;
sdd.snapholds = NULL; sdd.snapholds = NULL;
} }
if (flags->verbose || sdd.snapholds != NULL) { if (flags->verbose || sdd.snapholds != NULL) {
/* /*
* Do a verbose no-op dry run to get all the verbose output * Do a verbose no-op dry run to get all the verbose output
@ -2088,7 +2121,7 @@ zfs_send(zfs_handle_t *zhp, const char *fromsnap, const char *tosnap,
} }
if (!flags->dryrun && (flags->replicate || flags->doall || if (!flags->dryrun && (flags->replicate || flags->doall ||
flags->props || flags->backup)) { flags->props || flags->backup || flags->holds)) {
/* /*
* write final end record. NB: want to do this even if * write final end record. NB: want to do this even if
* there was some error, because it might not be totally * there was some error, because it might not be totally
@ -2820,7 +2853,8 @@ again:
VERIFY(0 == nvlist_alloc(&deleted, NV_UNIQUE_NAME, 0)); VERIFY(0 == nvlist_alloc(&deleted, NV_UNIQUE_NAME, 0));
if ((error = gather_nvlist(hdl, tofs, fromsnap, NULL, if ((error = gather_nvlist(hdl, tofs, fromsnap, NULL,
recursive, B_TRUE, B_FALSE, B_FALSE, &local_nv, &local_avl)) != 0) recursive, B_TRUE, B_FALSE, B_FALSE, B_FALSE, B_TRUE, &local_nv,
&local_avl)) != 0)
return (error); return (error);
/* /*
@ -3655,6 +3689,7 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap,
uint64_t parent_snapguid = 0; uint64_t parent_snapguid = 0;
prop_changelist_t *clp = NULL; prop_changelist_t *clp = NULL;
nvlist_t *snapprops_nvlist = NULL; nvlist_t *snapprops_nvlist = NULL;
nvlist_t *snapholds_nvlist = NULL;
zprop_errflags_t prop_errflags; zprop_errflags_t prop_errflags;
nvlist_t *prop_errors = NULL; nvlist_t *prop_errors = NULL;
boolean_t recursive; boolean_t recursive;
@ -3683,6 +3718,9 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap,
recursive = (nvlist_lookup_boolean(stream_nv, "not_recursive") == recursive = (nvlist_lookup_boolean(stream_nv, "not_recursive") ==
ENOENT); ENOENT);
/* Did the user request holds be skipped via zfs recv -k? */
boolean_t holds = flags->holds && !flags->skipholds;
if (stream_avl != NULL) { if (stream_avl != NULL) {
char *keylocation = NULL; char *keylocation = NULL;
nvlist_t *lookup = NULL; nvlist_t *lookup = NULL;
@ -3716,11 +3754,22 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap,
if (flags->canmountoff) { if (flags->canmountoff) {
VERIFY(0 == nvlist_add_uint64(rcvprops, VERIFY(0 == nvlist_add_uint64(rcvprops,
zfs_prop_to_name(ZFS_PROP_CANMOUNT), 0)); zfs_prop_to_name(ZFS_PROP_CANMOUNT), 0));
} else if (newprops) { /* nothing in rcvprops, eliminate it */
nvlist_free(rcvprops);
rcvprops = NULL;
newprops = B_FALSE;
} }
if (0 == nvlist_lookup_nvlist(fs, "snapprops", &lookup)) { if (0 == nvlist_lookup_nvlist(fs, "snapprops", &lookup)) {
VERIFY(0 == nvlist_lookup_nvlist(lookup, VERIFY(0 == nvlist_lookup_nvlist(lookup,
snapname, &snapprops_nvlist)); snapname, &snapprops_nvlist));
} }
if (holds) {
if (0 == nvlist_lookup_nvlist(fs, "snapholds",
&lookup)) {
VERIFY(0 == nvlist_lookup_nvlist(lookup,
snapname, &snapholds_nvlist));
}
}
} }
cp = NULL; cp = NULL;
@ -4204,6 +4253,22 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap,
zcmd_free_nvlists(&zc); zcmd_free_nvlists(&zc);
} }
} }
if (err == 0 && snapholds_nvlist) {
nvpair_t *pair;
nvlist_t *holds, *errors = NULL;
int cleanup_fd = -1;
VERIFY(0 == nvlist_alloc(&holds, 0, KM_SLEEP));
for (pair = nvlist_next_nvpair(snapholds_nvlist, NULL);
pair != NULL;
pair = nvlist_next_nvpair(snapholds_nvlist, pair)) {
VERIFY(0 == nvlist_add_string(holds, destsnap,
nvpair_name(pair)));
}
(void) lzc_hold(holds, cleanup_fd, &errors);
nvlist_free(snapholds_nvlist);
nvlist_free(holds);
}
if (err && (ioctl_errno == ENOENT || ioctl_errno == EEXIST)) { if (err && (ioctl_errno == ENOENT || ioctl_errno == EEXIST)) {
/* /*
@ -4222,7 +4287,8 @@ zfs_receive_one(libzfs_handle_t *hdl, int infd, const char *tosnap,
*/ */
*cp = '\0'; *cp = '\0';
if (gather_nvlist(hdl, destsnap, NULL, NULL, B_FALSE, B_TRUE, if (gather_nvlist(hdl, destsnap, NULL, NULL, B_FALSE, B_TRUE,
B_FALSE, B_FALSE, &local_nv, &local_avl) == 0) { B_FALSE, B_FALSE, B_FALSE, B_TRUE, &local_nv, &local_avl)
== 0) {
*cp = '@'; *cp = '@';
fs = fsavl_find(local_avl, drrb->drr_toguid, NULL); fs = fsavl_find(local_avl, drrb->drr_toguid, NULL);
fsavl_destroy(local_avl); fsavl_destroy(local_avl);
@ -4544,6 +4610,12 @@ zfs_receive_impl(libzfs_handle_t *hdl, const char *tosnap,
return (zfs_error(hdl, EZFS_BADSTREAM, errbuf)); return (zfs_error(hdl, EZFS_BADSTREAM, errbuf));
} }
/* Holds feature is set once in the compound stream header. */
boolean_t holds = (DMU_GET_FEATUREFLAGS(drrb->drr_versioninfo) &
DMU_BACKUP_FEATURE_HOLDS);
if (holds)
flags->holds = B_TRUE;
if (strchr(drrb->drr_toname, '@') == NULL) { if (strchr(drrb->drr_toname, '@') == NULL) {
zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "invalid " zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "invalid "
"stream (bad snapshot name)")); "stream (bad snapshot name)"));

View File

@ -195,7 +195,7 @@
.Ar snapshot bookmark .Ar snapshot bookmark
.Nm .Nm
.Cm send .Cm send
.Op Fl DLPRbcenpvw .Op Fl DLPRbcehnpvw
.Op Oo Fl I Ns | Ns Fl i Oc Ar snapshot .Op Oo Fl I Ns | Ns Fl i Oc Ar snapshot
.Ar snapshot .Ar snapshot
.Nm .Nm
@ -209,14 +209,14 @@
.Fl t Ar receive_resume_token .Fl t Ar receive_resume_token
.Nm .Nm
.Cm receive .Cm receive
.Op Fl Fnsuv .Op Fl Fhnsuv
.Op Fl o Sy origin Ns = Ns Ar snapshot .Op Fl o Sy origin Ns = Ns Ar snapshot
.Op Fl o Ar property Ns = Ns Ar value .Op Fl o Ar property Ns = Ns Ar value
.Op Fl x Ar property .Op Fl x Ar property
.Ar filesystem Ns | Ns Ar volume Ns | Ns Ar snapshot .Ar filesystem Ns | Ns Ar volume Ns | Ns Ar snapshot
.Nm .Nm
.Cm receive .Cm receive
.Op Fl Fnsuv .Op Fl Fhnsuv
.Op Fl d Ns | Ns Fl e .Op Fl d Ns | Ns Fl e
.Op Fl o Sy origin Ns = Ns Ar snapshot .Op Fl o Sy origin Ns = Ns Ar snapshot
.Op Fl o Ar property Ns = Ns Ar value .Op Fl o Ar property Ns = Ns Ar value
@ -3408,7 +3408,7 @@ feature.
.It Xo .It Xo
.Nm .Nm
.Cm send .Cm send
.Op Fl DLPRbcenpvw .Op Fl DLPRbcehnpvw
.Op Oo Fl I Ns | Ns Fl i Oc Ar snapshot .Op Oo Fl I Ns | Ns Fl i Oc Ar snapshot
.Ar snapshot .Ar snapshot
.Xc .Xc
@ -3548,6 +3548,12 @@ Note that if you do not use this flag for sending encrypted datasets, data will
be sent unencrypted and may be re-encrypted with a different encryption key on be sent unencrypted and may be re-encrypted with a different encryption key on
the receiving system, which will disable the ability to do a raw send to that the receiving system, which will disable the ability to do a raw send to that
system for incrementals. system for incrementals.
.It Fl h, -holds
Generate a stream package that includes any snapshot holds (created with the
.Sy zfs hold
command), and indicating to
.Sy zfs receive
that the holds be applied to the dataset on the receiving system.
.It Fl i Ar snapshot .It Fl i Ar snapshot
Generate an incremental stream from the first Generate an incremental stream from the first
.Ar snapshot .Ar snapshot
@ -3743,7 +3749,7 @@ for more details.
.It Xo .It Xo
.Nm .Nm
.Cm receive .Cm receive
.Op Fl Fnsuv .Op Fl Fhnsuv
.Op Fl o Sy origin Ns = Ns Ar snapshot .Op Fl o Sy origin Ns = Ns Ar snapshot
.Op Fl o Ar property Ns = Ns Ar value .Op Fl o Ar property Ns = Ns Ar value
.Op Fl x Ar property .Op Fl x Ar property
@ -3752,7 +3758,7 @@ for more details.
.It Xo .It Xo
.Nm .Nm
.Cm receive .Cm receive
.Op Fl Fnsuv .Op Fl Fhnsuv
.Op Fl d Ns | Ns Fl e .Op Fl d Ns | Ns Fl e
.Op Fl o Sy origin Ns = Ns Ar snapshot .Op Fl o Sy origin Ns = Ns Ar snapshot
.Op Fl o Ar property Ns = Ns Ar value .Op Fl o Ar property Ns = Ns Ar value
@ -3878,6 +3884,8 @@ snapshot as described in the paragraph above.
Discard all but the last element of the sent snapshot's file system name, using Discard all but the last element of the sent snapshot's file system name, using
that element to determine the name of the target file system for the new that element to determine the name of the target file system for the new
snapshot as described in the paragraph above. snapshot as described in the paragraph above.
.It Fl h
Skip the receive of holds. There is no effect if holds are not sent.
.It Fl n .It Fl n
Do not actually receive the stream. Do not actually receive the stream.
This can be useful in conjunction with the This can be useful in conjunction with the

View File

@ -1282,7 +1282,6 @@ dmu_send(const char *tosnap, const char *fromsnap, boolean_t embedok,
err = dsl_pool_hold(tosnap, FTAG, &dp); err = dsl_pool_hold(tosnap, FTAG, &dp);
if (err != 0) if (err != 0)
return (err); return (err);
if (strchr(tosnap, '@') == NULL && spa_writeable(dp->dp_spa)) { if (strchr(tosnap, '@') == NULL && spa_writeable(dp->dp_spa)) {
/* /*
* We are sending a filesystem or volume. Ensure * We are sending a filesystem or volume. Ensure

View File

@ -83,6 +83,7 @@ dsl_dataset_user_hold_check(void *arg, dmu_tx_t *tx)
{ {
dsl_dataset_user_hold_arg_t *dduha = arg; dsl_dataset_user_hold_arg_t *dduha = arg;
dsl_pool_t *dp = dmu_tx_pool(tx); dsl_pool_t *dp = dmu_tx_pool(tx);
nvlist_t *tmp_holds;
if (spa_version(dp->dp_spa) < SPA_VERSION_USERREFS) if (spa_version(dp->dp_spa) < SPA_VERSION_USERREFS)
return (SET_ERROR(ENOTSUP)); return (SET_ERROR(ENOTSUP));
@ -90,6 +91,26 @@ dsl_dataset_user_hold_check(void *arg, dmu_tx_t *tx)
if (!dmu_tx_is_syncing(tx)) if (!dmu_tx_is_syncing(tx))
return (0); return (0);
/*
* Ensure the list has no duplicates by copying name/values from
* non-unique dduha_holds to unique tmp_holds, and comparing counts.
*/
tmp_holds = fnvlist_alloc();
for (nvpair_t *pair = nvlist_next_nvpair(dduha->dduha_holds, NULL);
pair != NULL; pair = nvlist_next_nvpair(dduha->dduha_holds, pair)) {
size_t len = strlen(nvpair_name(pair)) +
strlen(fnvpair_value_string(pair));
char *nameval = kmem_zalloc(len + 2, KM_SLEEP);
(void) strcpy(nameval, nvpair_name(pair));
(void) strcat(nameval, "@");
(void) strcat(nameval, fnvpair_value_string(pair));
fnvlist_add_string(tmp_holds, nameval, "");
kmem_free(nameval, len + 2);
}
size_t tmp_count = fnvlist_num_pairs(tmp_holds);
fnvlist_free(tmp_holds);
if (tmp_count != fnvlist_num_pairs(dduha->dduha_holds))
return (SET_ERROR(EEXIST));
for (nvpair_t *pair = nvlist_next_nvpair(dduha->dduha_holds, NULL); for (nvpair_t *pair = nvlist_next_nvpair(dduha->dduha_holds, NULL);
pair != NULL; pair = nvlist_next_nvpair(dduha->dduha_holds, pair)) { pair != NULL; pair = nvlist_next_nvpair(dduha->dduha_holds, pair)) {
dsl_dataset_t *ds; dsl_dataset_t *ds;
@ -312,7 +333,8 @@ dsl_dataset_user_hold(nvlist_t *holds, minor_t cleanup_minor, nvlist_t *errlist)
return (0); return (0);
dduha.dduha_holds = holds; dduha.dduha_holds = holds;
dduha.dduha_chkholds = fnvlist_alloc(); /* chkholds can have non-unique name */
VERIFY(0 == nvlist_alloc(&dduha.dduha_chkholds, 0, KM_SLEEP));
dduha.dduha_errlist = errlist; dduha.dduha_errlist = errlist;
dduha.dduha_minor = cleanup_minor; dduha.dduha_minor = cleanup_minor;

View File

@ -793,7 +793,7 @@ tests = ['rsend_001_pos', 'rsend_002_pos', 'rsend_003_pos', 'rsend_004_pos',
'send-c_embedded_blocks', 'send-c_resume', 'send-cpL_varied_recsize', 'send-c_embedded_blocks', 'send-c_resume', 'send-cpL_varied_recsize',
'send-c_recv_dedup', 'send_encrypted_files', 'send_encrypted_heirarchy', 'send-c_recv_dedup', 'send_encrypted_files', 'send_encrypted_heirarchy',
'send_encrypted_props', 'send_freeobjects', 'send_realloc_dnode_size', 'send_encrypted_props', 'send_freeobjects', 'send_realloc_dnode_size',
'send_hole_birth', 'send-wDR_encrypted_zvol'] 'send_holds', 'send_hole_birth', 'send-wDR_encrypted_zvol']
tags = ['functional', 'rsend'] tags = ['functional', 'rsend']
[tests/functional/scrub_mirror] [tests/functional/scrub_mirror]

View File

@ -41,6 +41,7 @@ dist_pkgdata_SCRIPTS = \
send-cpL_varied_recsize.ksh \ send-cpL_varied_recsize.ksh \
send_freeobjects.ksh \ send_freeobjects.ksh \
send_realloc_dnode_size.ksh \ send_realloc_dnode_size.ksh \
send_holds.ksh \
send_hole_birth.ksh \ send_hole_birth.ksh \
send-wDR_encrypted_zvol.ksh send-wDR_encrypted_zvol.ksh

View File

@ -0,0 +1,177 @@
#!/bin/ksh -p
#
# CDDL HEADER START
#
# This file and its contents are supplied under the terms of the
# Common Development and Distribution License ("CDDL"), version 1.0.
# You may only use this file in accordance with the terms of version
# 1.0 of the CDDL.
#
# A full copy of the text of the CDDL should have accompanied this
# source. A copy of the CDDL is also available via the Internet at
# http://www.illumos.org/license/CDDL.
#
# CDDL HEADER END
#
#
# Copyright (c) 2012, 2016 by Delphix. All rights reserved.
# Copyright (c) 2018 Datto, Inc. All rights reserved.
#
. $STF_SUITE/tests/functional/rsend/rsend.kshlib
. $STF_SUITE/tests/functional/cli_root/cli_common.kshlib
#
# DESCRIPTION:
# Verify 'zfs send' can send dataset holds.
# Verify 'zfs receive' can receive dataset holds.
#
# STRATEGY:
# 1. Create a snapshot.
# 2. Create a full send stream with the fs, including holds.
# 3. Receive the send stream and verify the data integrity.
# 4. Fill in fs with some new data.
# 5. Create an incremental send stream with the fs, including holds.
# 6. Receive the incremental send stream and verify the data integrity.
# 7. Verify the holds have been received as expected.
# 8. Create an incremental snap with no holds, and send that with -h.
# 9. Confirm the snapshot was received as expected.
# 10. Create an incremental snapshot and place a hold on it.
# 11. Receive the incremental stream with -h and verify holds skipped.
#
verify_runnable "both"
function cleanup
{
eval "zfs holds $init_snap |grep -q hold1-1" &&
log_must zfs release hold1-1 $init_snap
eval "zfs holds $init_snap |grep -q hold1-2" &&
log_must zfs release hold1-2 $init_snap
eval "zfs holds $recv_snap |grep -q hold1-1" &&
log_must zfs release hold1-1 $recv_snap
eval "zfs holds $recv_snap |grep -q hold1-2" &&
log_must zfs release hold1-2 $recv_snap
eval "zfs holds $inc_snap |grep -q hold2-1" &&
log_must zfs release hold2-1 $inc_snap
eval "zfs holds $recv_inc_snap |grep -q hold2-1" &&
log_must zfs release hold2-1 $recv_inc_snap
eval "zfs holds $inc_snap3 |grep -q hold3-1" &&
log_must zfs release hold3-1 $inc_snap3
# destroy datasets
datasetexists $recv_root/$TESTFS1 &&
log_must destroy_dataset "$recv_root/$TESTFS1" "-Rf"
datasetexists $recv_root && log_must destroy_dataset "$recv_root" "-Rf"
datasetexists $TESTPOOL/$TESTFS1 && log_must destroy_dataset "$TESTPOOL/$TESTFS1" "-Rf"
[[ -e $full_bkup ]] && log_must rm -f $full_bkup
[[ -e $inc_bkup ]] && log_must rm -f $inc_bkup
[[ -e $inc_data2 ]] && log_must rm -f $inc_data2
[[ -d $TESTDIR1 ]] && log_must rm -rf $TESTDIR1
}
#
# Check if hold exists on the specified dataset.
#
function check_hold #<snapshot> <hold>
{
typeset dataset=$1
typeset hold=$2
log_note "checking $dataset for $hold"
eval "zfs holds $dataset |grep -q $hold"
}
log_assert "Verify 'zfs send/recv' can send and receive dataset holds."
log_onexit cleanup
init_snap=$TESTPOOL/$TESTFS1@init_snap
inc_snap=$TESTPOOL/$TESTFS1@inc_snap
inc_snap2=$TESTPOOL/$TESTFS1@inc_snap2
inc_snap3=$TESTPOOL/$TESTFS1@inc_snap3
full_bkup=$TEST_BASE_DIR/fullbkup.$$
inc_bkup=$TEST_BASE_DIR/incbkup.$$
init_data=$TESTDIR/$TESTFILE1
inc_data=$TESTDIR/$TESTFILE2
inc_data2=$TESTDIR/testfile3
recv_root=$TESTPOOL/rst_ctr
recv_snap=$recv_root/$TESTFS1@init_snap
recv_inc_snap=$recv_root/$TESTFS1@inc_snap
recv_inc_snap2=$recv_root/$TESTFS1@inc_snap2
recv_inc_snap3=$recv_root/$TESTFS1@inc_snap3
recv_data=$TESTDIR1/$TESTFS1/$TESTFILE1
recv_inc_data=$TESTDIR1/$TESTFS1/$TESTFILE2
recv_inc_data2=$TESTDIR1/$TESTFS1/testfile3
log_note "Verify 'zfs send' can create full send stream."
# Preparation
if ! datasetexists $TESTPOOL/$TESTFS1; then
log_must zfs create $TESTPOOL/$TESTFS1
fi
[[ -e $init_data ]] && log_must rm -f $init_data
log_must zfs create $recv_root
[[ ! -d $TESTDIR1 ]] && log_must mkdir -p $TESTDIR1
[[ ! -d $TESTDIR ]] && log_must mkdir -p $TESTDIR
log_must zfs set mountpoint=$TESTDIR $TESTPOOL/$TESTFS1
log_must zfs set mountpoint=$TESTDIR1 $recv_root
file_write -o create -f $init_data -b $BLOCK_SIZE -c $WRITE_COUNT
log_must zfs snapshot $init_snap
log_must zfs hold hold1-1 $init_snap
log_must zfs hold hold1-2 $init_snap
log_must eval "zfs send -h $init_snap > $full_bkup"
log_note "Verify the send stream is valid to receive."
log_must zfs recv -F $recv_snap <$full_bkup
log_must datasetexists $recv_snap
receive_check $recv_snap ${recv_snap%%@*}
log_note "Verify the holds were received."
log_must check_hold $recv_snap hold1-1
log_must check_hold $recv_snap hold1-2
compare_cksum $init_data $recv_data
log_note "Verify 'zfs send -i' can create incremental send stream."
file_write -o create -f $inc_data -b $BLOCK_SIZE -c $WRITE_COUNT -d 0
log_must zfs snapshot $inc_snap
log_must zfs hold hold2-1 $inc_snap
log_must eval "zfs send -h -i $init_snap $inc_snap > $inc_bkup"
log_note "Verify the incremental send stream is valid to receive."
log_must zfs recv -F $recv_inc_snap <$inc_bkup
log_must datasetexists $recv_inc_snap
log_note "Verify the hold was received from the incremental send."
log_must check_hold $recv_inc_snap hold2-1
compare_cksum $inc_data $recv_inc_data
log_note "Verify send -h works when there are no holds."
file_write -o create -f $inc_data2 -b $BLOCK_SIZE -c $WRITE_COUNT -d 0
log_must zfs snapshot $inc_snap2
log_must eval "zfs send -h -i $inc_snap $inc_snap2 > $inc_bkup"
log_must zfs recv -F $recv_inc_snap2 <$inc_bkup
log_must datasetexists $recv_inc_snap2
compare_cksum $inc_data2 $recv_inc_data2
log_note "Verify send -h fails properly when receive dataset already exists"
log_must zfs recv -F $recv_inc_snap2 <$inc_bkup
log_note "Verify recv -h skips the receive of holds"
log_must zfs snapshot $inc_snap3
log_must zfs hold hold3-1 $inc_snap3
log_must eval "zfs send -h -i $inc_snap2 $inc_snap3 > $inc_bkup"
log_must zfs recv -F -h $recv_inc_snap3 <$inc_bkup
log_must datasetexists $recv_inc_snap3
log_mustnot check_hold $recv_inc_snap3 hold3-1
log_pass "'zfs send/recv' can send and receive dataset holds."