mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2026-05-22 18:40:43 +03:00
Fix dsl_props_set_sync_impl to work with nested nvlist
When iterating over the input nvlist in dsl_props_set_sync_impl() when we don't preserve the nvpair name before looking up ZPROP_VALUE, so when we later go to process it nvpair_name() is always "value" and not the actual property name. This fixes a couple of bugs in zfs_ioc_recv(): * Received properties were not restored correctly when failing to receive an incremental send stream * Received properties were not completely replaced by the new ones when successfully receiving an incremental send stream Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: loli10K <ezomori.nozomu@gmail.com> Closes #5497
This commit is contained in:
@@ -892,11 +892,15 @@ dsl_props_set_sync_impl(dsl_dataset_t *ds, zprop_source_t source,
|
||||
|
||||
while ((elem = nvlist_next_nvpair(props, elem)) != NULL) {
|
||||
nvpair_t *pair = elem;
|
||||
const char *name = nvpair_name(pair);
|
||||
|
||||
if (nvpair_type(pair) == DATA_TYPE_NVLIST) {
|
||||
/*
|
||||
* dsl_prop_get_all_impl() returns properties in this
|
||||
* format.
|
||||
* This usually happens when we reuse the nvlist_t data
|
||||
* returned by the counterpart dsl_prop_get_all_impl().
|
||||
* For instance we do this to restore the original
|
||||
* received properties when an error occurs in the
|
||||
* zfs_ioc_recv() codepath.
|
||||
*/
|
||||
nvlist_t *attrs = fnvpair_value_nvlist(pair);
|
||||
pair = fnvlist_lookup_nvpair(attrs, ZPROP_VALUE);
|
||||
@@ -904,14 +908,14 @@ dsl_props_set_sync_impl(dsl_dataset_t *ds, zprop_source_t source,
|
||||
|
||||
if (nvpair_type(pair) == DATA_TYPE_STRING) {
|
||||
const char *value = fnvpair_value_string(pair);
|
||||
dsl_prop_set_sync_impl(ds, nvpair_name(pair),
|
||||
dsl_prop_set_sync_impl(ds, name,
|
||||
source, 1, strlen(value) + 1, value, tx);
|
||||
} else if (nvpair_type(pair) == DATA_TYPE_UINT64) {
|
||||
uint64_t intval = fnvpair_value_uint64(pair);
|
||||
dsl_prop_set_sync_impl(ds, nvpair_name(pair),
|
||||
dsl_prop_set_sync_impl(ds, name,
|
||||
source, sizeof (intval), 1, &intval, tx);
|
||||
} else if (nvpair_type(pair) == DATA_TYPE_BOOLEAN) {
|
||||
dsl_prop_set_sync_impl(ds, nvpair_name(pair),
|
||||
dsl_prop_set_sync_impl(ds, name,
|
||||
source, 0, 0, NULL, tx);
|
||||
} else {
|
||||
panic("invalid nvpair type");
|
||||
|
||||
Reference in New Issue
Block a user