4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * * Neither the name of Red Hat nor the names of its contributors may be
16 * used to endorse or promote products derived from this software without
17 * specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
22 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
23 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
26 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
27 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
29 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
40 #include <nbdkit-filter.h>
47 static pthread_mutex_t clock_lock
;
48 static uint64_t clock_
= 0;
51 adjust_clock (uint64_t offset
)
53 ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&clock_lock
);
59 reset_clock (uint64_t offset
)
61 ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&clock_lock
);
66 get_starting_offset (void)
68 ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&clock_lock
);
69 return scan_clock
? clock_
: 0;
73 scan_thread (void *vp
)
75 struct bgthread_ctrl
*ctrl
= vp
;
76 uint64_t offset
, size
;
79 assert (ctrl
->next
!= NULL
);
81 /* Get the size of the underlying plugin. Exit the thread on error
82 * because there's not much we can do without knowing the size.
84 r
= ctrl
->next
->get_size (ctrl
->next
);
91 for (offset
= get_starting_offset (); offset
< size
; offset
+= scan_size
) {
94 /* Execute any commands in the queue. */
96 ACQUIRE_LOCK_FOR_CURRENT_SCOPE (&ctrl
->lock
);
99 while (ctrl
->cmds
.len
) {
100 cmd
= ctrl
->cmds
.ptr
[0];
101 command_queue_remove (&ctrl
->cmds
, 0);
105 nbdkit_debug ("scan: exiting background thread on connection close");
108 case CMD_NOTIFY_PREAD
:
109 if (offset
< cmd
.offset
)
115 adjust_clock (offset
);
118 /* Issue the next prefetch. */
119 n
= MIN (scan_size
, size
- offset
);
120 ctrl
->next
->cache (ctrl
->next
, n
, offset
, 0, NULL
);
125 reset_clock (offset
);
129 nbdkit_debug ("scan: finished scanning the plugin");