Author: mav
Date: Tue Jan 26 13:20:31 2016
New Revision: 294816
URL: https://svnweb.freebsd.org/changeset/base/294816

Log:
  4986 receiving replication stream fails if any snapshot exceeds refquota
  
  Reviewed by: John Kennedy <[email protected]>
  Reviewed by: Matthew Ahrens <[email protected]>
  Approved by: Gordon Ross <[email protected]>
  Author: Dan McDonald <[email protected]>
  
  illumos/illumos-gate@5878fad70d76d8711f6608c1f80b0447601261c6

Modified:
  vendor-sys/illumos/dist/uts/common/fs/zfs/zfs_ioctl.c

Changes in other areas also in this revision:
Modified:
  vendor/illumos/dist/lib/libzfs/common/libzfs_sendrecv.c

Modified: vendor-sys/illumos/dist/uts/common/fs/zfs/zfs_ioctl.c
==============================================================================
--- vendor-sys/illumos/dist/uts/common/fs/zfs/zfs_ioctl.c       Tue Jan 26 
13:14:39 2016        (r294815)
+++ vendor-sys/illumos/dist/uts/common/fs/zfs/zfs_ioctl.c       Tue Jan 26 
13:20:31 2016        (r294816)
@@ -22,6 +22,7 @@
 /*
  * Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
  * Portions Copyright 2011 Martin Matuska
+ * Copyright 2015, OmniTI Computer Consulting, Inc. All rights reserved.
  * Copyright 2015 Nexenta Systems, Inc.  All rights reserved.
  * Copyright (c) 2014, Joyent, Inc. All rights reserved.
  * Copyright (c) 2011, 2015 by Delphix. All rights reserved.
@@ -4090,6 +4091,56 @@ next:
        }
 }
 
+/*
+ * Extract properties that cannot be set PRIOR to the receipt of a dataset.
+ * For example, refquota cannot be set until after the receipt of a dataset,
+ * because in replication streams, an older/earlier snapshot may exceed the
+ * refquota.  We want to receive the older/earlier snapshot, but setting
+ * refquota pre-receipt will set the dsl's ACTUAL quota, which will prevent
+ * the older/earlier snapshot from being received (with EDQUOT).
+ *
+ * The ZFS test "zfs_receive_011_pos" demonstrates such a scenario.
+ *
+ * libzfs will need to be judicious handling errors encountered by props
+ * extracted by this function.
+ */
+static nvlist_t *
+extract_delay_props(nvlist_t *props)
+{
+       nvlist_t *delayprops;
+       nvpair_t *nvp, *tmp;
+       static const zfs_prop_t delayable[] = { ZFS_PROP_REFQUOTA, 0 };
+       int i;
+
+       VERIFY(nvlist_alloc(&delayprops, NV_UNIQUE_NAME, KM_SLEEP) == 0);
+
+       for (nvp = nvlist_next_nvpair(props, NULL); nvp != NULL;
+           nvp = nvlist_next_nvpair(props, nvp)) {
+               /*
+                * strcmp() is safe because zfs_prop_to_name() always returns
+                * a bounded string.
+                */
+               for (i = 0; delayable[i] != 0; i++) {
+                       if (strcmp(zfs_prop_to_name(delayable[i]),
+                           nvpair_name(nvp)) == 0) {
+                               break;
+                       }
+               }
+               if (delayable[i] != 0) {
+                       tmp = nvlist_prev_nvpair(props, nvp);
+                       VERIFY(nvlist_add_nvpair(delayprops, nvp) == 0);
+                       VERIFY(nvlist_remove_nvpair(props, nvp) == 0);
+                       nvp = tmp;
+               }
+       }
+
+       if (nvlist_empty(delayprops)) {
+               nvlist_free(delayprops);
+               delayprops = NULL;
+       }
+       return (delayprops);
+}
+
 #ifdef DEBUG
 static boolean_t zfs_ioc_recv_inject_err;
 #endif
@@ -4126,6 +4177,7 @@ zfs_ioc_recv(zfs_cmd_t *zc)
        offset_t off;
        nvlist_t *props = NULL; /* sent properties */
        nvlist_t *origprops = NULL; /* existing properties */
+       nvlist_t *delayprops = NULL; /* sent properties applied post-receive */
        char *origin = NULL;
        char *tosnap;
        char tofs[ZFS_MAXNAMELEN];
@@ -4206,21 +4258,12 @@ zfs_ioc_recv(zfs_cmd_t *zc)
                props_error = dsl_prop_set_hasrecvd(tofs);
 
                if (props_error == 0) {
+                       delayprops = extract_delay_props(props);
                        (void) zfs_set_prop_nvlist(tofs, ZPROP_SRC_RECEIVED,
                            props, errors);
                }
        }
 
-       if (zc->zc_nvlist_dst_size != 0 &&
-           (nvlist_smush(errors, zc->zc_nvlist_dst_size) != 0 ||
-           put_nvlist(zc, errors) != 0)) {
-               /*
-                * Caller made zc->zc_nvlist_dst less than the minimum expected
-                * size or supplied an invalid address.
-                */
-               props_error = SET_ERROR(EINVAL);
-       }
-
        off = fp->f_offset;
        error = dmu_recv_stream(&drc, fp->f_vnode, &off, zc->zc_cleanup_fd,
            &zc->zc_action_handle);
@@ -4245,6 +4288,40 @@ zfs_ioc_recv(zfs_cmd_t *zc)
                } else {
                        error = dmu_recv_end(&drc, NULL);
                }
+
+               /* Set delayed properties now, after we're done receiving. */
+               if (delayprops != NULL && error == 0) {
+                       (void) zfs_set_prop_nvlist(tofs, ZPROP_SRC_RECEIVED,
+                           delayprops, errors);
+               }
+       }
+
+       if (delayprops != NULL) {
+               /*
+                * Merge delayed props back in with initial props, in case
+                * we're DEBUG and zfs_ioc_recv_inject_err is set (which means
+                * we have to make sure clear_received_props() includes
+                * the delayed properties).
+                *
+                * Since zfs_ioc_recv_inject_err is only in DEBUG kernels,
+                * using ASSERT() will be just like a VERIFY.
+                */
+               ASSERT(nvlist_merge(props, delayprops, 0) == 0);
+               nvlist_free(delayprops);
+       }
+
+       /*
+        * Now that all props, initial and delayed, are set, report the prop
+        * errors to the caller.
+        */
+       if (zc->zc_nvlist_dst_size != 0 &&
+           (nvlist_smush(errors, zc->zc_nvlist_dst_size) != 0 ||
+           put_nvlist(zc, errors) != 0)) {
+               /*
+                * Caller made zc->zc_nvlist_dst less than the minimum expected
+                * size or supplied an invalid address.
+                */
+               props_error = SET_ERROR(EINVAL);
        }
 
        zc->zc_cookie = off - fp->f_offset;
_______________________________________________
[email protected] mailing list
https://lists.freebsd.org/mailman/listinfo/svn-src-all
To unsubscribe, send any mail to "[email protected]"

Reply via email to