|
|
3cdd4c |
From 8bfe6512d07caf778fd001425435b048c45513eb Mon Sep 17 00:00:00 2001
|
|
|
3cdd4c |
From: "Richard W.M. Jones" <rjones@redhat.com>
|
|
|
3cdd4c |
Date: Sat, 14 May 2022 13:46:56 +0100
|
|
|
3cdd4c |
Subject: [PATCH] New filter: scan
|
|
|
3cdd4c |
|
|
|
3cdd4c |
This filter will simply scan across the disk issuing a series of cache
|
|
|
3cdd4c |
requests to the underlying plugin. It is similar in scope and usage
|
|
|
3cdd4c |
to the new nbdkit-readahead-filter.
|
|
|
3cdd4c |
|
|
|
3cdd4c |
(cherry picked from commit 65c20a09ceacb4431986a2982f2c2e746df63fcb)
|
|
|
3cdd4c |
---
|
|
|
3cdd4c |
TODO | 8 -
|
|
|
3cdd4c |
configure.ac | 2 +
|
|
|
3cdd4c |
filters/cache/nbdkit-cache-filter.pod | 4 +-
|
|
|
3cdd4c |
.../nbdkit-cacheextents-filter.pod | 1 +
|
|
|
3cdd4c |
filters/readahead/nbdkit-readahead-filter.pod | 5 +
|
|
|
3cdd4c |
filters/scan/Makefile.am | 72 +++++
|
|
|
3cdd4c |
filters/scan/bgthread.c | 131 ++++++++
|
|
|
3cdd4c |
filters/scan/nbdkit-scan-filter.pod | 159 ++++++++++
|
|
|
3cdd4c |
filters/scan/scan.c | 280 ++++++++++++++++++
|
|
|
3cdd4c |
filters/scan/scan.h | 64 ++++
|
|
|
3cdd4c |
plugins/ssh/nbdkit-ssh-plugin.pod | 1 +
|
|
|
3cdd4c |
plugins/torrent/nbdkit-torrent-plugin.pod | 1 +
|
|
|
3cdd4c |
plugins/vddk/nbdkit-vddk-plugin.pod | 1 +
|
|
|
3cdd4c |
tests/Makefile.am | 10 +
|
|
|
3cdd4c |
tests/test-scan-copy.sh | 42 +++
|
|
|
3cdd4c |
tests/test-scan-info.sh | 46 +++
|
|
|
3cdd4c |
16 files changed, 817 insertions(+), 10 deletions(-)
|
|
|
3cdd4c |
create mode 100644 filters/scan/Makefile.am
|
|
|
3cdd4c |
create mode 100644 filters/scan/bgthread.c
|
|
|
3cdd4c |
create mode 100644 filters/scan/nbdkit-scan-filter.pod
|
|
|
3cdd4c |
create mode 100644 filters/scan/scan.c
|
|
|
3cdd4c |
create mode 100644 filters/scan/scan.h
|
|
|
3cdd4c |
create mode 100755 tests/test-scan-copy.sh
|
|
|
3cdd4c |
create mode 100755 tests/test-scan-info.sh
|
|
|
3cdd4c |
|
|
|
3cdd4c |
diff --git a/TODO b/TODO
|
|
|
3cdd4c |
index 0f5dc41d..8600d9e4 100644
|
|
|
3cdd4c |
--- a/TODO
|
|
|
3cdd4c |
+++ b/TODO
|
|
|
3cdd4c |
@@ -182,14 +182,6 @@ Python:
|
|
|
3cdd4c |
Suggestions for filters
|
|
|
3cdd4c |
-----------------------
|
|
|
3cdd4c |
|
|
|
3cdd4c |
-* Add scan filter. This would be placed on top of cache filters and
|
|
|
3cdd4c |
- would scan (read) the whole disk in the background, ensuring it is
|
|
|
3cdd4c |
- copied into the cache. Useful if you have a slow plugin, limited
|
|
|
3cdd4c |
- size device, and lots of local disk space, especially if you know
|
|
|
3cdd4c |
- that the NBD clients will eventually read all of the device. RWMJ
|
|
|
3cdd4c |
- wrote an implementation of this but it doesn't work well without a
|
|
|
3cdd4c |
- background thread.
|
|
|
3cdd4c |
-
|
|
|
3cdd4c |
* Add shared filter. Take advantage of filter context APIs to open a
|
|
|
3cdd4c |
single context into the backend shared among multiple client
|
|
|
3cdd4c |
connections. This may even allow a filter to offer a more parallel
|
|
|
3cdd4c |
diff --git a/configure.ac b/configure.ac
|
|
|
3cdd4c |
index 1d209f67..466dbd9b 100644
|
|
|
3cdd4c |
--- a/configure.ac
|
|
|
3cdd4c |
+++ b/configure.ac
|
|
|
3cdd4c |
@@ -142,6 +142,7 @@ filters="\
|
|
|
3cdd4c |
readahead \
|
|
|
3cdd4c |
retry \
|
|
|
3cdd4c |
retry-request \
|
|
|
3cdd4c |
+ scan \
|
|
|
3cdd4c |
stats \
|
|
|
3cdd4c |
swab \
|
|
|
3cdd4c |
tar \
|
|
|
3cdd4c |
@@ -1403,6 +1404,7 @@ AC_CONFIG_FILES([Makefile
|
|
|
3cdd4c |
filters/readahead/Makefile
|
|
|
3cdd4c |
filters/retry/Makefile
|
|
|
3cdd4c |
filters/retry-request/Makefile
|
|
|
3cdd4c |
+ filters/scan/Makefile
|
|
|
3cdd4c |
filters/stats/Makefile
|
|
|
3cdd4c |
filters/swab/Makefile
|
|
|
3cdd4c |
filters/tar/Makefile
|
|
|
3cdd4c |
diff --git a/filters/cache/nbdkit-cache-filter.pod b/filters/cache/nbdkit-cache-filter.pod
|
|
|
3cdd4c |
index f4234e1a..935804b5 100644
|
|
|
3cdd4c |
--- a/filters/cache/nbdkit-cache-filter.pod
|
|
|
3cdd4c |
+++ b/filters/cache/nbdkit-cache-filter.pod
|
|
|
3cdd4c |
@@ -28,8 +28,8 @@ loss, as the name suggests).
|
|
|
3cdd4c |
|
|
|
3cdd4c |
This filter only caches image contents. To cache image metadata, use
|
|
|
3cdd4c |
L<nbdkit-cacheextents-filter(1)> between this filter and the plugin.
|
|
|
3cdd4c |
-To accelerate sequential reads, use L<nbdkit-readahead-filter(1)> on
|
|
|
3cdd4c |
-top of this filter.
|
|
|
3cdd4c |
+To accelerate sequential reads, use L<nbdkit-readahead-filter(1)> or
|
|
|
3cdd4c |
+L<nbdkit-scan-filter(1)> on top of this filter.
|
|
|
3cdd4c |
|
|
|
3cdd4c |
=head1 PARAMETERS
|
|
|
3cdd4c |
|
|
|
3cdd4c |
diff --git a/filters/cacheextents/nbdkit-cacheextents-filter.pod b/filters/cacheextents/nbdkit-cacheextents-filter.pod
|
|
|
3cdd4c |
index bb2514a4..6464eac2 100644
|
|
|
3cdd4c |
--- a/filters/cacheextents/nbdkit-cacheextents-filter.pod
|
|
|
3cdd4c |
+++ b/filters/cacheextents/nbdkit-cacheextents-filter.pod
|
|
|
3cdd4c |
@@ -54,6 +54,7 @@ L<nbdkit(1)>,
|
|
|
3cdd4c |
L<nbdkit-cache-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-extentlist-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-readahead-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-scan-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-vddk-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-filter(3)>,
|
|
|
3cdd4c |
L<qemu-img(1)>.
|
|
|
3cdd4c |
diff --git a/filters/readahead/nbdkit-readahead-filter.pod b/filters/readahead/nbdkit-readahead-filter.pod
|
|
|
3cdd4c |
index 630e5924..99d64dfb 100644
|
|
|
3cdd4c |
--- a/filters/readahead/nbdkit-readahead-filter.pod
|
|
|
3cdd4c |
+++ b/filters/readahead/nbdkit-readahead-filter.pod
|
|
|
3cdd4c |
@@ -27,6 +27,10 @@ option.
|
|
|
3cdd4c |
The filter uses a simple adaptive algorithm which accelerates
|
|
|
3cdd4c |
sequential reads and requires no further configuration.
|
|
|
3cdd4c |
|
|
|
3cdd4c |
+A similar filter is L<nbdkit-scan-filter(1)> which reads ahead over
|
|
|
3cdd4c |
+the whole disk, useful if you know that the client will be reading
|
|
|
3cdd4c |
+sequentially across most or all of the disk.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
=head2 Limitations
|
|
|
3cdd4c |
|
|
|
3cdd4c |
In a number of significant cases this filter will do nothing. The
|
|
|
3cdd4c |
@@ -91,6 +95,7 @@ L<nbdkit-cache-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-cow-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-file-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-retry-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-scan-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-torrent-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-vddk-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-filter(3)>,
|
|
|
3cdd4c |
diff --git a/filters/scan/Makefile.am b/filters/scan/Makefile.am
|
|
|
3cdd4c |
new file mode 100644
|
|
|
3cdd4c |
index 00000000..d4aabfc6
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/filters/scan/Makefile.am
|
|
|
3cdd4c |
@@ -0,0 +1,72 @@
|
|
|
3cdd4c |
+# nbdkit
|
|
|
3cdd4c |
+# Copyright (C) 2019-2021 Red Hat Inc.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# Redistribution and use in source and binary forms, with or without
|
|
|
3cdd4c |
+# modification, are permitted provided that the following conditions are
|
|
|
3cdd4c |
+# met:
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Redistributions of source code must retain the above copyright
|
|
|
3cdd4c |
+# notice, this list of conditions and the following disclaimer.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Redistributions in binary form must reproduce the above copyright
|
|
|
3cdd4c |
+# notice, this list of conditions and the following disclaimer in the
|
|
|
3cdd4c |
+# documentation and/or other materials provided with the distribution.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Neither the name of Red Hat nor the names of its contributors may be
|
|
|
3cdd4c |
+# used to endorse or promote products derived from this software without
|
|
|
3cdd4c |
+# specific prior written permission.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
|
|
|
3cdd4c |
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
3cdd4c |
+# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
|
3cdd4c |
+# PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
|
|
|
3cdd4c |
+# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
3cdd4c |
+# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
3cdd4c |
+# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
3cdd4c |
+# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
3cdd4c |
+# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
3cdd4c |
+# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
3cdd4c |
+# OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
3cdd4c |
+# SUCH DAMAGE.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+include $(top_srcdir)/common-rules.mk
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+EXTRA_DIST = nbdkit-scan-filter.pod
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+filter_LTLIBRARIES = nbdkit-scan-filter.la
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+nbdkit_scan_filter_la_SOURCES = \
|
|
|
3cdd4c |
+ scan.c \
|
|
|
3cdd4c |
+ scan.h \
|
|
|
3cdd4c |
+ bgthread.c \
|
|
|
3cdd4c |
+ $(top_srcdir)/include/nbdkit-filter.h \
|
|
|
3cdd4c |
+ $(NULL)
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+nbdkit_scan_filter_la_CPPFLAGS = \
|
|
|
3cdd4c |
+ -I$(top_srcdir)/include \
|
|
|
3cdd4c |
+ -I$(top_srcdir)/common/include \
|
|
|
3cdd4c |
+ -I$(top_srcdir)/common/utils \
|
|
|
3cdd4c |
+ $(NULL)
|
|
|
3cdd4c |
+nbdkit_scan_filter_la_CFLAGS = $(WARNINGS_CFLAGS)
|
|
|
3cdd4c |
+nbdkit_scan_filter_la_LDFLAGS = \
|
|
|
3cdd4c |
+ -module -avoid-version -shared $(NO_UNDEFINED_ON_WINDOWS) \
|
|
|
3cdd4c |
+ -Wl,--version-script=$(top_srcdir)/filters/filters.syms \
|
|
|
3cdd4c |
+ $(NULL)
|
|
|
3cdd4c |
+nbdkit_scan_filter_la_LIBADD = \
|
|
|
3cdd4c |
+ $(top_builddir)/common/utils/libutils.la \
|
|
|
3cdd4c |
+ $(top_builddir)/common/replacements/libcompat.la \
|
|
|
3cdd4c |
+ $(IMPORT_LIBRARY_ON_WINDOWS) \
|
|
|
3cdd4c |
+ $(NULL)
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+if HAVE_POD
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+man_MANS = nbdkit-scan-filter.1
|
|
|
3cdd4c |
+CLEANFILES += $(man_MANS)
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+nbdkit-scan-filter.1: nbdkit-scan-filter.pod \
|
|
|
3cdd4c |
+ $(top_builddir)/podwrapper.pl
|
|
|
3cdd4c |
+ $(PODWRAPPER) --section=1 --man $@ \
|
|
|
3cdd4c |
+ --html $(top_builddir)/html/$@.html \
|
|
|
3cdd4c |
+ $<
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+endif HAVE_POD
|
|
|
3cdd4c |
diff --git a/filters/scan/bgthread.c b/filters/scan/bgthread.c
|
|
|
3cdd4c |
new file mode 100644
|
|
|
3cdd4c |
index 00000000..384e79b6
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/filters/scan/bgthread.c
|
|
|
3cdd4c |
@@ -0,0 +1,131 @@
|
|
|
3cdd4c |
+/* nbdkit
|
|
|
3cdd4c |
+ * Copyright (C) 2019-2022 Red Hat Inc.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * Redistribution and use in source and binary forms, with or without
|
|
|
3cdd4c |
+ * modification, are permitted provided that the following conditions are
|
|
|
3cdd4c |
+ * met:
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Redistributions of source code must retain the above copyright
|
|
|
3cdd4c |
+ * notice, this list of conditions and the following disclaimer.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Redistributions in binary form must reproduce the above copyright
|
|
|
3cdd4c |
+ * notice, this list of conditions and the following disclaimer in the
|
|
|
3cdd4c |
+ * documentation and/or other materials provided with the distribution.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Neither the name of Red Hat nor the names of its contributors may be
|
|
|
3cdd4c |
+ * used to endorse or promote products derived from this software without
|
|
|
3cdd4c |
+ * specific prior written permission.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
|
|
|
3cdd4c |
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
3cdd4c |
+ * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
|
3cdd4c |
+ * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
|
|
|
3cdd4c |
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
3cdd4c |
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
3cdd4c |
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
3cdd4c |
+ * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
3cdd4c |
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
3cdd4c |
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
3cdd4c |
+ * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
3cdd4c |
+ * SUCH DAMAGE.
|
|
|
3cdd4c |
+ */
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <config.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <stdio.h>
|
|
|
3cdd4c |
+#include <stdlib.h>
|
|
|
3cdd4c |
+#include <stdint.h>
|
|
|
3cdd4c |
+#include <pthread.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <nbdkit-filter.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include "scan.h"
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include "cleanup.h"
|
|
|
3cdd4c |
+#include "minmax.h"
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static pthread_mutex_t clock_lock;
|
|
|
3cdd4c |
+static uint64_t clock_ = 0;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static void
|
|
|
3cdd4c |
+adjust_clock (uint64_t offset)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&clock_lock);
|
|
|
3cdd4c |
+ if (clock_ < offset)
|
|
|
3cdd4c |
+ clock_ = offset;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static void
|
|
|
3cdd4c |
+reset_clock (uint64_t offset)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&clock_lock);
|
|
|
3cdd4c |
+ clock_ = 0;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static uint64_t
|
|
|
3cdd4c |
+get_starting_offset (void)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&clock_lock);
|
|
|
3cdd4c |
+ return scan_clock ? clock_ : 0;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+void *
|
|
|
3cdd4c |
+scan_thread (void *vp)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ struct bgthread_ctrl *ctrl = vp;
|
|
|
3cdd4c |
+ uint64_t offset, size;
|
|
|
3cdd4c |
+ int64_t r;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ assert (ctrl->next != NULL);
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Get the size of the underlying plugin. Exit the thread on error
|
|
|
3cdd4c |
+ * because there's not much we can do without knowing the size.
|
|
|
3cdd4c |
+ */
|
|
|
3cdd4c |
+ r = ctrl->next->get_size (ctrl->next);
|
|
|
3cdd4c |
+ if (r == -1)
|
|
|
3cdd4c |
+ return NULL;
|
|
|
3cdd4c |
+ size = r;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Start scanning. */
|
|
|
3cdd4c |
+ start:
|
|
|
3cdd4c |
+ for (offset = get_starting_offset (); offset < size; offset += scan_size) {
|
|
|
3cdd4c |
+ uint64_t n;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Execute any commands in the queue. */
|
|
|
3cdd4c |
+ {
|
|
|
3cdd4c |
+ ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&ctrl->lock);
|
|
|
3cdd4c |
+ struct command cmd;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ while (ctrl->cmds.len) {
|
|
|
3cdd4c |
+ cmd = ctrl->cmds.ptr[0];
|
|
|
3cdd4c |
+ command_queue_remove (&ctrl->cmds, 0);
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ switch (cmd.type) {
|
|
|
3cdd4c |
+ case CMD_QUIT:
|
|
|
3cdd4c |
+ nbdkit_debug ("scan: exiting background thread on connection close");
|
|
|
3cdd4c |
+ return NULL;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ case CMD_NOTIFY_PREAD:
|
|
|
3cdd4c |
+ if (offset < cmd.offset)
|
|
|
3cdd4c |
+ offset = cmd.offset;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ adjust_clock (offset);
|
|
|
3cdd4c |
+ if (offset > size)
|
|
|
3cdd4c |
+ continue;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Issue the next prefetch. */
|
|
|
3cdd4c |
+ n = MIN (scan_size, size - offset);
|
|
|
3cdd4c |
+ ctrl->next->cache (ctrl->next, n, offset, 0, NULL);
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (scan_forever) {
|
|
|
3cdd4c |
+ reset_clock (offset);
|
|
|
3cdd4c |
+ goto start;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ nbdkit_debug ("scan: finished scanning the plugin");
|
|
|
3cdd4c |
+ return NULL;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
diff --git a/filters/scan/nbdkit-scan-filter.pod b/filters/scan/nbdkit-scan-filter.pod
|
|
|
3cdd4c |
new file mode 100644
|
|
|
3cdd4c |
index 00000000..4a8d0ef9
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/filters/scan/nbdkit-scan-filter.pod
|
|
|
3cdd4c |
@@ -0,0 +1,159 @@
|
|
|
3cdd4c |
+=head1 NAME
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+nbdkit-scan-filter - scan disk prefetching data ahead of sequential reads
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 SYNOPSIS
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ nbdkit --filter=scan PLUGIN [scan-ahead=false] [scan-clock=false]
|
|
|
3cdd4c |
+ [scan-forever=true] [scan-size=]NN
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ nbdkit --filter=scan --filter=cache PLUGIN
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ nbdkit --filter=scan --filter=cow PLUGIN cow-on-cache=true
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 DESCRIPTION
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+C<nbdkit-scan-filter> is a filter that scans the disk prefetching
|
|
|
3cdd4c |
+data. It is sometimes useful if you expect that the client will read
|
|
|
3cdd4c |
+the disk sequentially.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+The basic operation of the filter is that when a client connects, the
|
|
|
3cdd4c |
+filter will start issuing C<.cache> (prefetch) requests to the plugin
|
|
|
3cdd4c |
+across the whole disk. Plugins which support this command will
|
|
|
3cdd4c |
+prefetch the data, making subsequent reads faster. For plugins which
|
|
|
3cdd4c |
+do not support this command, you can inject L<nbdkit-cache-filter(1)>
|
|
|
3cdd4c |
+below (after) this filter, giving approximately the same effect.
|
|
|
3cdd4c |
+L<nbdkit-cow-filter(1)> can be used instead of nbdkit-cache-filter, if
|
|
|
3cdd4c |
+you add the C<cow-on-cache=true> option.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+Various C<scan-*> parameters can be used to tune scanning, although
|
|
|
3cdd4c |
+the defaults should be suitable in most cases.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+A similar filter is L<nbdkit-readahead-filter(1)>.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head2 Limitations
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+In a number of significant cases this filter will do nothing. The
|
|
|
3cdd4c |
+filter will print a warning message if this happens.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=over 4
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item Thread model must be parallel *
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+For example L<nbdkit-curl-plugin(1)> only supports
|
|
|
3cdd4c |
+C<serialize_requests>, and so this filter cannot perform prefetches in
|
|
|
3cdd4c |
+parallel with the read requests.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item Only scans while clients are connected *
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+The current filter only scans while there is at least one client
|
|
|
3cdd4c |
+connected.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item Only scans the default export *
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+The current filter only scans the default export and ignores all
|
|
|
3cdd4c |
+clients connecting to the non-default export name.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+* We may be able to lift these restrictions in future.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item Underlying filters or plugin must support C<.cache> (prefetch)
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+Very many plugins do not have the concept of prefetching and/or
|
|
|
3cdd4c |
+do not implement the C<.cache> callback, and so there is no
|
|
|
3cdd4c |
+way for this filter to issue prefetches.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+You can usually get around this by adding I<--filter=cache> after this
|
|
|
3cdd4c |
+filter as explained above.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item Prefetching the whole disk may load it all into cache
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+In particular if you use this filter together with
|
|
|
3cdd4c |
+L<nbdkit-cache-filter(1)> or L<nbdkit-cow-filter(1)>, they will cache
|
|
|
3cdd4c |
+the whole content of the plugin into a temporary file. This may be
|
|
|
3cdd4c |
+many gigabytes of data, consuming all space in F. Of course
|
|
|
3cdd4c |
+this is the whole point of using this filter, but you should be aware
|
|
|
3cdd4c |
+of it.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+If using the cache filter, the total size of the cache can be limited
|
|
|
3cdd4c |
+(see L<nbdkit-cache-filter(1)/CACHE MAXIMUM SIZE>).
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=back
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 PARAMETERS
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=over 4
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item B<scan-ahead=false>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+By default the filter tries to stay ahead of incoming read requests.
|
|
|
3cdd4c |
+That is to say, it starts prefetching at the beginning of the disk and
|
|
|
3cdd4c |
+continues incrementally, but if the client issues a read beyond the
|
|
|
3cdd4c |
+current prefetch point then the filter skips forward and begins
|
|
|
3cdd4c |
+prefetching after the read.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+However if you set this parameter to false, then this behaviour is
|
|
|
3cdd4c |
+disabled. The filter simply prefetches sequentially regardless of
|
|
|
3cdd4c |
+client requests.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item B<scan-clock=false>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+By default, if all clients disconnect and then another client
|
|
|
3cdd4c |
+connects, prefetching resumes at the same place in the disk. (Like
|
|
|
3cdd4c |
+stopping and starting a clock.)
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+If you set this parameter to false, then the filter starts prefetching
|
|
|
3cdd4c |
+from the beginning of the disk again.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item B<scan-forever=true>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+By default the filter scans over the disk once and then stops.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+If you set this parameter to true, then after the disk has been
|
|
|
3cdd4c |
+prefetched completely, the filter goes back to the beginning and
|
|
|
3cdd4c |
+starts over, repeating this for as long as nbdkit is running and there
|
|
|
3cdd4c |
+are clients connected.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item B<scan-size=>NN
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+This parameter controls the prefetch block size. The default is
|
|
|
3cdd4c |
+C<2M>. This must be a power of 2 and most plugins will have their own
|
|
|
3cdd4c |
+limits on the amount of data they can prefetch in a single request.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=back
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 FILES
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=over 4
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=item F<$filterdir/nbdkit-scan-filter.so>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+The filter.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+Use C<nbdkit --dump-config> to find the location of C<$filterdir>.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=back
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 VERSION
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+C<nbdkit-scan-filter> first appeared in nbdkit 1.32.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 SEE ALSO
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+L<nbdkit(1)>,
|
|
|
3cdd4c |
+L<nbdkit-cache-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-cow-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-file-plugin(1)>,
|
|
|
3cdd4c |
+L<nbdkit-readahead-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-retry-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-torrent-plugin(1)>,
|
|
|
3cdd4c |
+L<nbdkit-vddk-plugin(1)>,
|
|
|
3cdd4c |
+L<nbdkit-filter(3)>,
|
|
|
3cdd4c |
+L<qemu-img(1)>.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 AUTHORS
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+Richard W.M. Jones
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+=head1 COPYRIGHT
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+Copyright (C) 2019-2022 Red Hat Inc.
|
|
|
3cdd4c |
diff --git a/filters/scan/scan.c b/filters/scan/scan.c
|
|
|
3cdd4c |
new file mode 100644
|
|
|
3cdd4c |
index 00000000..ac5b18d2
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/filters/scan/scan.c
|
|
|
3cdd4c |
@@ -0,0 +1,280 @@
|
|
|
3cdd4c |
+/* nbdkit
|
|
|
3cdd4c |
+ * Copyright (C) 2019-2022 Red Hat Inc.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * Redistribution and use in source and binary forms, with or without
|
|
|
3cdd4c |
+ * modification, are permitted provided that the following conditions are
|
|
|
3cdd4c |
+ * met:
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Redistributions of source code must retain the above copyright
|
|
|
3cdd4c |
+ * notice, this list of conditions and the following disclaimer.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Redistributions in binary form must reproduce the above copyright
|
|
|
3cdd4c |
+ * notice, this list of conditions and the following disclaimer in the
|
|
|
3cdd4c |
+ * documentation and/or other materials provided with the distribution.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Neither the name of Red Hat nor the names of its contributors may be
|
|
|
3cdd4c |
+ * used to endorse or promote products derived from this software without
|
|
|
3cdd4c |
+ * specific prior written permission.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
|
|
|
3cdd4c |
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
3cdd4c |
+ * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
|
3cdd4c |
+ * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
|
|
|
3cdd4c |
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
3cdd4c |
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
3cdd4c |
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
3cdd4c |
+ * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
3cdd4c |
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
3cdd4c |
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
3cdd4c |
+ * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
3cdd4c |
+ * SUCH DAMAGE.
|
|
|
3cdd4c |
+ */
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <config.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <stdio.h>
|
|
|
3cdd4c |
+#include <stdlib.h>
|
|
|
3cdd4c |
+#include <stdbool.h>
|
|
|
3cdd4c |
+#include <stdint.h>
|
|
|
3cdd4c |
+#include <string.h>
|
|
|
3cdd4c |
+#include <errno.h>
|
|
|
3cdd4c |
+#include <pthread.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <nbdkit-filter.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include "scan.h"
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include "cleanup.h"
|
|
|
3cdd4c |
+#include "ispowerof2.h"
|
|
|
3cdd4c |
+#include "vector.h"
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static bool scan_ahead = true;
|
|
|
3cdd4c |
+bool scan_clock = true;
|
|
|
3cdd4c |
+bool scan_forever = false;
|
|
|
3cdd4c |
+unsigned scan_size = 2*1024*1024;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static int thread_model = -1; /* Thread model of the underlying plugin. */
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* Per-connection data. */
|
|
|
3cdd4c |
+struct scan_handle {
|
|
|
3cdd4c |
+ bool is_default_export; /* If exportname == "". */
|
|
|
3cdd4c |
+ bool running; /* True if background thread is running. */
|
|
|
3cdd4c |
+ pthread_t thread; /* The background thread, one per connection. */
|
|
|
3cdd4c |
+ struct bgthread_ctrl ctrl;
|
|
|
3cdd4c |
+};
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+scan_config (nbdkit_next_config *next, nbdkit_backend *nxdata,
|
|
|
3cdd4c |
+ const char *key, const char *value)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ int r;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (strcmp (key, "scan-ahead") == 0) {
|
|
|
3cdd4c |
+ r = nbdkit_parse_bool (value);
|
|
|
3cdd4c |
+ if (r == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ scan_ahead = r;
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+ else if (strcmp (key, "scan-clock") == 0) {
|
|
|
3cdd4c |
+ r = nbdkit_parse_bool (value);
|
|
|
3cdd4c |
+ if (r == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ scan_clock = r;
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+ else if (strcmp (key, "scan-forever") == 0) {
|
|
|
3cdd4c |
+ r = nbdkit_parse_bool (value);
|
|
|
3cdd4c |
+ if (r == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ scan_forever = r;
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+ else if (strcmp (key, "scan-size") == 0) {
|
|
|
3cdd4c |
+ scan_size = nbdkit_parse_size (value);
|
|
|
3cdd4c |
+ if (scan_size == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ return next (nxdata, key, value);
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+scan_config_complete (nbdkit_next_config_complete *next, nbdkit_backend *nxdata)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ if (scan_size < 512 || scan_size > 32*1024*1024 ||
|
|
|
3cdd4c |
+ !is_power_of_2 (scan_size)) {
|
|
|
3cdd4c |
+ nbdkit_error ("scan-size parameter should be [512..32M] "
|
|
|
3cdd4c |
+ "and a power of two");
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ return next (nxdata);
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#define scan_config_help \
|
|
|
3cdd4c |
+ "scan-ahead=false Skip ahead when client reads faster.\n" \
|
|
|
3cdd4c |
+ "scan-clock=false Always start prefetching from beginning.\n" \
|
|
|
3cdd4c |
+ "scan-forever=true Scan in a loop while clients connected.\n" \
|
|
|
3cdd4c |
+ "scan-size=NN Set scan block size."
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* We need to hook into .get_ready() so we can read the final thread
|
|
|
3cdd4c |
+ * model (of the whole server).
|
|
|
3cdd4c |
+ */
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+scan_get_ready (int final_thread_model)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ thread_model = final_thread_model;
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+send_command_to_background_thread (struct bgthread_ctrl *ctrl,
|
|
|
3cdd4c |
+ const struct command cmd)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&ctrl->lock);
|
|
|
3cdd4c |
+ if (command_queue_append (&ctrl->cmds, cmd) == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ /* Signal the thread if it could be sleeping on an empty queue. */
|
|
|
3cdd4c |
+ if (ctrl->cmds.len == 1)
|
|
|
3cdd4c |
+ pthread_cond_signal (&ctrl->cond);
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static void *
|
|
|
3cdd4c |
+scan_open (nbdkit_next_open *next, nbdkit_context *nxdata,
|
|
|
3cdd4c |
+ int readonly, const char *exportname, int is_tls)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ struct scan_handle *h;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (next (nxdata, readonly, exportname) == -1)
|
|
|
3cdd4c |
+ return NULL;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ h = calloc (1, sizeof *h);
|
|
|
3cdd4c |
+ if (h == NULL) {
|
|
|
3cdd4c |
+ nbdkit_error ("malloc: %m");
|
|
|
3cdd4c |
+ return NULL;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ h->is_default_export = strcmp (exportname, "") == 0;
|
|
|
3cdd4c |
+ return h;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* In prepare we check if it's possible to support the scan filter on
|
|
|
3cdd4c |
+ * this connection (or print a warning), and start the background
|
|
|
3cdd4c |
+ * thread.
|
|
|
3cdd4c |
+ */
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+scan_prepare (nbdkit_next *next, void *handle, int readonly)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ struct scan_handle *h = handle;
|
|
|
3cdd4c |
+ int r, err;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (!h->is_default_export) {
|
|
|
3cdd4c |
+ nbdkit_error ("scan: warning: not the default export, not scanning");
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (thread_model != NBDKIT_THREAD_MODEL_PARALLEL) {
|
|
|
3cdd4c |
+ nbdkit_error ("scan: warning: underlying plugin does not support "
|
|
|
3cdd4c |
+ "the PARALLEL thread model, not scanning");
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Call next->can_cache to read the underlying 'can_cache'. */
|
|
|
3cdd4c |
+ r = next->can_cache (next);
|
|
|
3cdd4c |
+ if (r == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ if (r != NBDKIT_CACHE_NATIVE) {
|
|
|
3cdd4c |
+ nbdkit_error ("scan: warning: underlying plugin does not support "
|
|
|
3cdd4c |
+ "NBD_CMD_CACHE, not scanning; try adding --filter=cache "
|
|
|
3cdd4c |
+ "after this filter");
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Save the connection in the handle, for the background thread to use. */
|
|
|
3cdd4c |
+ h->ctrl.next = next;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Create the background thread. */
|
|
|
3cdd4c |
+ h->ctrl.cmds = (command_queue) empty_vector;
|
|
|
3cdd4c |
+ pthread_mutex_init (&h->ctrl.lock, NULL);
|
|
|
3cdd4c |
+ pthread_cond_init (&h->ctrl.cond, NULL);
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ err = pthread_create (&h->thread, NULL, scan_thread, &h->ctrl);
|
|
|
3cdd4c |
+ if (err != 0) {
|
|
|
3cdd4c |
+ errno = err;
|
|
|
3cdd4c |
+ nbdkit_error ("pthread_create: %m");
|
|
|
3cdd4c |
+ pthread_cond_destroy (&h->ctrl.cond);
|
|
|
3cdd4c |
+ pthread_mutex_destroy (&h->ctrl.lock);
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ h->running = true;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* Finalize cleans up the thread if it is running. */
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+scan_finalize (nbdkit_next *next, void *handle)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ struct scan_handle *h = handle;
|
|
|
3cdd4c |
+ const struct command quit_cmd = { .type = CMD_QUIT };
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (!h->running)
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ send_command_to_background_thread (&h->ctrl, quit_cmd);
|
|
|
3cdd4c |
+ pthread_join (h->thread, NULL);
|
|
|
3cdd4c |
+ pthread_cond_destroy (&h->ctrl.cond);
|
|
|
3cdd4c |
+ pthread_mutex_destroy (&h->ctrl.lock);
|
|
|
3cdd4c |
+ command_queue_reset (&h->ctrl.cmds);
|
|
|
3cdd4c |
+ h->running = false;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ return 0;
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static void
|
|
|
3cdd4c |
+scan_close (void *handle)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ struct scan_handle *h = handle;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ free (h);
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* Read data. */
|
|
|
3cdd4c |
+static int
|
|
|
3cdd4c |
+scan_pread (nbdkit_next *next,
|
|
|
3cdd4c |
+ void *handle, void *buf, uint32_t count, uint64_t offset,
|
|
|
3cdd4c |
+ uint32_t flags, int *err)
|
|
|
3cdd4c |
+{
|
|
|
3cdd4c |
+ struct scan_handle *h = handle;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (scan_ahead && h->running) {
|
|
|
3cdd4c |
+ const struct command cmd =
|
|
|
3cdd4c |
+ { .type = CMD_NOTIFY_PREAD, .offset = offset + count };
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ if (send_command_to_background_thread (&h->ctrl, cmd) == -1)
|
|
|
3cdd4c |
+ return -1;
|
|
|
3cdd4c |
+ }
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+ /* Issue the normal read. */
|
|
|
3cdd4c |
+ return next->pread (next, buf, count, offset, flags, err);
|
|
|
3cdd4c |
+}
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+static struct nbdkit_filter filter = {
|
|
|
3cdd4c |
+ .name = "scan",
|
|
|
3cdd4c |
+ .longname = "nbdkit scan filter",
|
|
|
3cdd4c |
+ .get_ready = scan_get_ready,
|
|
|
3cdd4c |
+ .config = scan_config,
|
|
|
3cdd4c |
+ .config_complete = scan_config_complete,
|
|
|
3cdd4c |
+ .config_help = scan_config_help,
|
|
|
3cdd4c |
+ .open = scan_open,
|
|
|
3cdd4c |
+ .prepare = scan_prepare,
|
|
|
3cdd4c |
+ .finalize = scan_finalize,
|
|
|
3cdd4c |
+ .close = scan_close,
|
|
|
3cdd4c |
+ .pread = scan_pread,
|
|
|
3cdd4c |
+};
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+NBDKIT_REGISTER_FILTER(filter)
|
|
|
3cdd4c |
diff --git a/filters/scan/scan.h b/filters/scan/scan.h
|
|
|
3cdd4c |
new file mode 100644
|
|
|
3cdd4c |
index 00000000..7ff39310
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/filters/scan/scan.h
|
|
|
3cdd4c |
@@ -0,0 +1,64 @@
|
|
|
3cdd4c |
+/* nbdkit
|
|
|
3cdd4c |
+ * Copyright (C) 2019-2022 Red Hat Inc.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * Redistribution and use in source and binary forms, with or without
|
|
|
3cdd4c |
+ * modification, are permitted provided that the following conditions are
|
|
|
3cdd4c |
+ * met:
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Redistributions of source code must retain the above copyright
|
|
|
3cdd4c |
+ * notice, this list of conditions and the following disclaimer.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Redistributions in binary form must reproduce the above copyright
|
|
|
3cdd4c |
+ * notice, this list of conditions and the following disclaimer in the
|
|
|
3cdd4c |
+ * documentation and/or other materials provided with the distribution.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * * Neither the name of Red Hat nor the names of its contributors may be
|
|
|
3cdd4c |
+ * used to endorse or promote products derived from this software without
|
|
|
3cdd4c |
+ * specific prior written permission.
|
|
|
3cdd4c |
+ *
|
|
|
3cdd4c |
+ * THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
|
|
|
3cdd4c |
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
3cdd4c |
+ * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
|
3cdd4c |
+ * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
|
|
|
3cdd4c |
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
3cdd4c |
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
3cdd4c |
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
3cdd4c |
+ * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
3cdd4c |
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
3cdd4c |
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
3cdd4c |
+ * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
3cdd4c |
+ * SUCH DAMAGE.
|
|
|
3cdd4c |
+ */
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#ifndef NBDKIT_SCAN_H
|
|
|
3cdd4c |
+#define NBDKIT_SCAN_H
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <stdbool.h>
|
|
|
3cdd4c |
+#include <pthread.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include <nbdkit-filter.h>
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#include "vector.h"
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+extern bool scan_clock;
|
|
|
3cdd4c |
+extern bool scan_forever;
|
|
|
3cdd4c |
+extern unsigned scan_size;
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* List of commands issued to the background thread. */
|
|
|
3cdd4c |
+struct command {
|
|
|
3cdd4c |
+ enum { CMD_QUIT, CMD_NOTIFY_PREAD } type;
|
|
|
3cdd4c |
+ uint64_t offset;
|
|
|
3cdd4c |
+};
|
|
|
3cdd4c |
+DEFINE_VECTOR_TYPE(command_queue, struct command);
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+struct bgthread_ctrl {
|
|
|
3cdd4c |
+ command_queue cmds; /* Command queue. */
|
|
|
3cdd4c |
+ pthread_mutex_t lock; /* Lock for queue. */
|
|
|
3cdd4c |
+ pthread_cond_t cond; /* Condition queue size 0 -> 1. */
|
|
|
3cdd4c |
+ nbdkit_next *next; /* For sending cache operations. */
|
|
|
3cdd4c |
+};
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+/* Start background thread (one per connection). */
|
|
|
3cdd4c |
+extern void *scan_thread (void *vp);
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+#endif /* NBDKIT_SCAN_H */
|
|
|
3cdd4c |
diff --git a/plugins/ssh/nbdkit-ssh-plugin.pod b/plugins/ssh/nbdkit-ssh-plugin.pod
|
|
|
3cdd4c |
index 2bc2c4a7..214957d6 100644
|
|
|
3cdd4c |
--- a/plugins/ssh/nbdkit-ssh-plugin.pod
|
|
|
3cdd4c |
+++ b/plugins/ssh/nbdkit-ssh-plugin.pod
|
|
|
3cdd4c |
@@ -349,6 +349,7 @@ L<nbdkit-curl-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-extentlist-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-readahead-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-retry-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-scan-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-plugin(3)>,
|
|
|
3cdd4c |
L<ssh(1)>,
|
|
|
3cdd4c |
L<ssh-agent(1)>,
|
|
|
3cdd4c |
diff --git a/plugins/torrent/nbdkit-torrent-plugin.pod b/plugins/torrent/nbdkit-torrent-plugin.pod
|
|
|
3cdd4c |
index 196ce4e9..f09ac3d2 100644
|
|
|
3cdd4c |
--- a/plugins/torrent/nbdkit-torrent-plugin.pod
|
|
|
3cdd4c |
+++ b/plugins/torrent/nbdkit-torrent-plugin.pod
|
|
|
3cdd4c |
@@ -175,6 +175,7 @@ L<nbdkit-curl-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-file-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-iso-plugin(1)>,
|
|
|
3cdd4c |
L<nbdkit-readahead-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-scan-filter(1)>,
|
|
|
3cdd4c |
L<transmission-show(1)>,
|
|
|
3cdd4c |
L<https://en.wikipedia.org/wiki/BitTorrent>,
|
|
|
3cdd4c |
L<http://libtorrent.org/>.
|
|
|
3cdd4c |
diff --git a/plugins/vddk/nbdkit-vddk-plugin.pod b/plugins/vddk/nbdkit-vddk-plugin.pod
|
|
|
3cdd4c |
index ea5899dc..3991e86b 100644
|
|
|
3cdd4c |
--- a/plugins/vddk/nbdkit-vddk-plugin.pod
|
|
|
3cdd4c |
+++ b/plugins/vddk/nbdkit-vddk-plugin.pod
|
|
|
3cdd4c |
@@ -733,6 +733,7 @@ L<nbdkit-plugin(3)>,
|
|
|
3cdd4c |
L<nbdkit-blocksize-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-readahead-filter(1)>,
|
|
|
3cdd4c |
L<nbdkit-retry-filter(1)>,
|
|
|
3cdd4c |
+L<nbdkit-scan-filter(1)>,
|
|
|
3cdd4c |
L<virsh(1)>,
|
|
|
3cdd4c |
L<https://libvirt.org/drvesx.html>,
|
|
|
3cdd4c |
L<https://www.vmware.com/support/developer/vddk/>,
|
|
|
3cdd4c |
diff --git a/tests/Makefile.am b/tests/Makefile.am
|
|
|
3cdd4c |
index 5585b3b7..799aa6c2 100644
|
|
|
3cdd4c |
--- a/tests/Makefile.am
|
|
|
3cdd4c |
+++ b/tests/Makefile.am
|
|
|
3cdd4c |
@@ -1754,6 +1754,16 @@ test_retry_request_mirror_LDADD = \
|
|
|
3cdd4c |
$(LIBNBD_LIBS) \
|
|
|
3cdd4c |
$(NULL)
|
|
|
3cdd4c |
|
|
|
3cdd4c |
+# scan filter test.
|
|
|
3cdd4c |
+TESTS += \
|
|
|
3cdd4c |
+ test-scan-copy.sh \
|
|
|
3cdd4c |
+ test-scan-info.sh \
|
|
|
3cdd4c |
+ $(NULL)
|
|
|
3cdd4c |
+EXTRA_DIST += \
|
|
|
3cdd4c |
+ test-scan-copy.sh \
|
|
|
3cdd4c |
+ test-scan-info.sh \
|
|
|
3cdd4c |
+ $(NULL)
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
# swab filter test.
|
|
|
3cdd4c |
TESTS += \
|
|
|
3cdd4c |
test-swab-8.sh \
|
|
|
3cdd4c |
diff --git a/tests/test-scan-copy.sh b/tests/test-scan-copy.sh
|
|
|
3cdd4c |
new file mode 100755
|
|
|
3cdd4c |
index 00000000..227ad7b2
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/tests/test-scan-copy.sh
|
|
|
3cdd4c |
@@ -0,0 +1,42 @@
|
|
|
3cdd4c |
+#!/usr/bin/env bash
|
|
|
3cdd4c |
+# nbdkit
|
|
|
3cdd4c |
+# Copyright (C) 2018-2022 Red Hat Inc.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# Redistribution and use in source and binary forms, with or without
|
|
|
3cdd4c |
+# modification, are permitted provided that the following conditions are
|
|
|
3cdd4c |
+# met:
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Redistributions of source code must retain the above copyright
|
|
|
3cdd4c |
+# notice, this list of conditions and the following disclaimer.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Redistributions in binary form must reproduce the above copyright
|
|
|
3cdd4c |
+# notice, this list of conditions and the following disclaimer in the
|
|
|
3cdd4c |
+# documentation and/or other materials provided with the distribution.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Neither the name of Red Hat nor the names of its contributors may be
|
|
|
3cdd4c |
+# used to endorse or promote products derived from this software without
|
|
|
3cdd4c |
+# specific prior written permission.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
|
|
|
3cdd4c |
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
3cdd4c |
+# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
|
3cdd4c |
+# PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
|
|
|
3cdd4c |
+# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
3cdd4c |
+# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
3cdd4c |
+# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
3cdd4c |
+# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
3cdd4c |
+# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
3cdd4c |
+# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
3cdd4c |
+# OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
3cdd4c |
+# SUCH DAMAGE.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+source ./functions.sh
|
|
|
3cdd4c |
+set -e
|
|
|
3cdd4c |
+set -x
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+requires nbdcopy --version
|
|
|
3cdd4c |
+requires_plugin sparse-random
|
|
|
3cdd4c |
+requires_filter scan
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+nbdkit -fv -U - sparse-random 1M --filter=scan --run 'nbdcopy "$uri" "$uri"'
|
|
|
3cdd4c |
+nbdkit -fv -U - sparse-random 1G --filter=scan --run 'nbdcopy "$uri" "$uri"'
|
|
|
3cdd4c |
diff --git a/tests/test-scan-info.sh b/tests/test-scan-info.sh
|
|
|
3cdd4c |
new file mode 100755
|
|
|
3cdd4c |
index 00000000..6b109ca8
|
|
|
3cdd4c |
--- /dev/null
|
|
|
3cdd4c |
+++ b/tests/test-scan-info.sh
|
|
|
3cdd4c |
@@ -0,0 +1,46 @@
|
|
|
3cdd4c |
+#!/usr/bin/env bash
|
|
|
3cdd4c |
+# nbdkit
|
|
|
3cdd4c |
+# Copyright (C) 2018-2022 Red Hat Inc.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# Redistribution and use in source and binary forms, with or without
|
|
|
3cdd4c |
+# modification, are permitted provided that the following conditions are
|
|
|
3cdd4c |
+# met:
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Redistributions of source code must retain the above copyright
|
|
|
3cdd4c |
+# notice, this list of conditions and the following disclaimer.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Redistributions in binary form must reproduce the above copyright
|
|
|
3cdd4c |
+# notice, this list of conditions and the following disclaimer in the
|
|
|
3cdd4c |
+# documentation and/or other materials provided with the distribution.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# * Neither the name of Red Hat nor the names of its contributors may be
|
|
|
3cdd4c |
+# used to endorse or promote products derived from this software without
|
|
|
3cdd4c |
+# specific prior written permission.
|
|
|
3cdd4c |
+#
|
|
|
3cdd4c |
+# THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
|
|
|
3cdd4c |
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
3cdd4c |
+# THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
|
3cdd4c |
+# PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
|
|
|
3cdd4c |
+# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
3cdd4c |
+# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
3cdd4c |
+# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
|
3cdd4c |
+# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
3cdd4c |
+# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
3cdd4c |
+# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
|
3cdd4c |
+# OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
3cdd4c |
+# SUCH DAMAGE.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+source ./functions.sh
|
|
|
3cdd4c |
+set -e
|
|
|
3cdd4c |
+set -x
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+requires nbdinfo --version
|
|
|
3cdd4c |
+requires_filter scan
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+# We're just testing that there are no problematic races with the
|
|
|
3cdd4c |
+# background thread.
|
|
|
3cdd4c |
+
|
|
|
3cdd4c |
+nbdkit -fv -U - memory 1 --filter=scan --run 'nbdinfo $uri'
|
|
|
3cdd4c |
+nbdkit -fv -U - memory 1M --filter=scan --run 'nbdinfo $uri'
|
|
|
3cdd4c |
+nbdkit -fv -U - memory 1G --filter=scan --run 'nbdinfo $uri'
|
|
|
3cdd4c |
+nbdkit -fv -U - memory 1G --filter=scan -e test --run 'nbdinfo $uri'
|
|
|
3cdd4c |
--
|
|
|
3cdd4c |
2.31.1
|
|
|
3cdd4c |
|