Blame SOURCES/0013-New-filter-scan.patch

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