Skip to content

Commit e44fca4

Browse files
committed
nbd: wait for udev to report nbd block device ready
This makes sure "casync mkdev" will not notify readiness until the NBD block device it allocated has been probed by udev, and thus is guaranteed to be fully initialized, probed and registered. Replaces: #114
1 parent 3423186 commit e44fca4

7 files changed

+201
-11
lines changed

meson.build

+8-1
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,6 @@ libzstd = dependency('libzstd',
114114

115115
libacl = cc.find_library('acl')
116116

117-
118117
if get_option('fuse')
119118
libfuse = dependency('fuse',
120119
version : '>= 2.6')
@@ -130,6 +129,13 @@ else
130129
endif
131130
conf.set10('HAVE_SELINUX', get_option('selinux'))
132131

132+
if get_option('udev')
133+
libudev = dependency('libudev')
134+
else
135+
libudev = []
136+
endif
137+
conf.set10('HAVE_UDEV', get_option('udev'))
138+
133139
threads = dependency('threads')
134140
math = cc.find_library('m')
135141

@@ -159,6 +165,7 @@ casync = executable(
159165
libfuse,
160166
liblzma,
161167
libselinux,
168+
libudev,
162169
libz,
163170
libzstd,
164171
math,

meson_options.txt

+2
Original file line numberDiff line numberDiff line change
@@ -5,5 +5,7 @@ option('fuse', type : 'boolean', value : true,
55
description : 'build the FUSE integration (requires fuse-devel)')
66
option('selinux', type : 'boolean', value : true,
77
description : 'build the SELinux backend (requires libselinux-devel)')
8+
option('udev', type : 'boolean', value : true,
9+
description : 'build the libudev integration (requires libudev-devel)')
810
option('man', type : 'boolean', value : true,
911
description : 'build and install man pages (requires spinx-build')

src/canbd.c

+51
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,8 @@ struct CaBlockDevice {
3535
int friendly_name_fd;
3636
char *friendly_name;
3737
char *friendly_name_path;
38+
39+
dev_t devnum;
3840
};
3941

4042
CaBlockDevice *ca_block_device_new(void) {
@@ -270,6 +272,7 @@ static int ca_block_device_establish_friendly_name(CaBlockDevice *d) {
270272
int ca_block_device_open(CaBlockDevice *d) {
271273
static const int one = 1;
272274
bool free_device_path = false;
275+
struct stat st;
273276
int r;
274277

275278
if (!d)
@@ -295,6 +298,16 @@ int ca_block_device_open(CaBlockDevice *d) {
295298
if (d->device_fd < 0)
296299
return -errno;
297300

301+
if (fstat(d->device_fd, &st) < 0) {
302+
r = -errno;
303+
goto fail;
304+
}
305+
306+
if (!S_ISBLK(st.st_mode)) {
307+
r = -ENOTBLK;
308+
goto fail;
309+
}
310+
298311
if (ioctl(d->device_fd, NBD_SET_SOCK, d->socket_fd[1]) < 0) {
299312
r = -errno;
300313
goto fail;
@@ -320,6 +333,16 @@ int ca_block_device_open(CaBlockDevice *d) {
320333
goto fail;
321334
}
322335

336+
if (fstat(d->device_fd, &st) < 0) {
337+
r = -errno;
338+
goto fail;
339+
}
340+
341+
if (!S_ISBLK(st.st_mode)) {
342+
r = -ENOTBLK;
343+
goto fail;
344+
}
345+
323346
if (ioctl(d->device_fd, NBD_SET_SOCK, d->socket_fd[1]) >= 0) {
324347
d->device_path = path;
325348
free_device_path = true;
@@ -338,6 +361,8 @@ int ca_block_device_open(CaBlockDevice *d) {
338361
}
339362
}
340363

364+
d->devnum = st.st_rdev;
365+
341366
r = ca_block_device_establish_friendly_name(d);
342367
if (r < 0)
343368
goto fail;
@@ -489,6 +514,17 @@ int ca_block_device_put_data(CaBlockDevice *d, uint64_t offset, const void *data
489514
return 0;
490515
}
491516

517+
int ca_block_device_get_poll_fd(CaBlockDevice *d) {
518+
if (!d)
519+
return -EINVAL;
520+
if (d->device_fd < 0)
521+
return -EUNATCH;
522+
if (d->socket_fd[0] < 0)
523+
return -EUNATCH;
524+
525+
return d->socket_fd[0];
526+
}
527+
492528
int ca_block_device_poll(CaBlockDevice *d, uint64_t timeout_nsec, const sigset_t *ss) {
493529
struct pollfd pollfd;
494530
int r;
@@ -556,6 +592,21 @@ int ca_block_device_set_path(CaBlockDevice *d, const char *node) {
556592
return 1;
557593
}
558594

595+
596+
int ca_block_device_get_devnum(CaBlockDevice *d, dev_t *ret) {
597+
if (!d)
598+
return -EINVAL;
599+
600+
if (d->device_fd < 0)
601+
return -EUNATCH;
602+
603+
if (d->devnum == 0)
604+
return -EUNATCH;
605+
606+
*ret = d->devnum;
607+
return 0;
608+
}
609+
559610
int ca_block_device_test_nbd(const char *name) {
560611
unsigned u;
561612
size_t n;

src/canbd.h

+3
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,9 @@ int ca_block_device_get_path(CaBlockDevice *d, const char **ret);
3838

3939
int ca_block_device_set_friendly_name(CaBlockDevice *d, const char *name);
4040

41+
int ca_block_device_get_devnum(CaBlockDevice *d, dev_t *ret);
42+
int ca_block_device_get_poll_fd(CaBlockDevice *d);
43+
4144
int ca_block_device_test_nbd(const char *name);
4245

4346
#endif

src/casync-tool.c

+122-9
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
#include <errno.h>
55
#include <fcntl.h>
66
#include <getopt.h>
7+
#include <poll.h>
78
#include <signal.h>
89
#include <stdio.h>
910
#include <stdlib.h>
@@ -27,6 +28,11 @@
2728
#include "signal-handler.h"
2829
#include "util.h"
2930

31+
#ifdef HAVE_UDEV
32+
#include <libudev.h>
33+
#include "udev-util.h"
34+
#endif
35+
3036
static enum arg_what {
3137
WHAT_ARCHIVE,
3238
WHAT_ARCHIVE_INDEX,
@@ -2556,15 +2562,24 @@ static int verb_mkdev(int argc, char *argv[]) {
25562562
MKDEV_BLOB_INDEX,
25572563
_MKDEV_OPERATION_INVALID = -1,
25582564
} MkDevOperation;
2565+
25592566
MkDevOperation operation = _MKDEV_OPERATION_INVALID;
2560-
_cleanup_(realloc_buffer_free) ReallocBuffer buffer = {};
25612567
_cleanup_(ca_block_device_unrefp) CaBlockDevice *nbd = NULL;
2562-
const char *path = NULL, *name = NULL;
2563-
bool make_symlink = false, rm_symlink = false;
2564-
int r;
2568+
_cleanup_(realloc_buffer_free) ReallocBuffer buffer = {};
25652569
_cleanup_(safe_close_nonstdp) int input_fd = -1;
2566-
_cleanup_free_ char *input = NULL;
2570+
bool make_symlink = false, rm_symlink = false;
25672571
_cleanup_(ca_sync_unrefp) CaSync *s = NULL;
2572+
const char *path = NULL, *name = NULL;
2573+
_cleanup_free_ char *input = NULL;
2574+
bool initialized, sent_ready = false;
2575+
dev_t devnum;
2576+
int r;
2577+
2578+
#if HAVE_UDEV
2579+
_cleanup_(udev_monitor_unrefp) struct udev_monitor *monitor = NULL;
2580+
_cleanup_(udev_device_unrefp) struct udev_device *d = NULL;
2581+
_cleanup_(udev_unrefp) struct udev *udev = NULL;
2582+
#endif
25682583

25692584
if (argc > 3) {
25702585
log_error("An blob path/URL expected, possibly followed by a device or symlink name.");
@@ -2762,6 +2777,49 @@ static int verb_mkdev(int argc, char *argv[]) {
27622777
goto finish;
27632778
}
27642779

2780+
r = ca_block_device_get_devnum(nbd, &devnum);
2781+
if (r < 0) {
2782+
log_error_errno(r, "Failed to get device ID: %m");
2783+
goto finish;
2784+
}
2785+
2786+
#if HAVE_UDEV
2787+
udev = udev_new();
2788+
if (!udev) {
2789+
r = log_error_errno(errno, "Failed to allocate udev context: %m");
2790+
goto finish;
2791+
}
2792+
2793+
monitor = udev_monitor_new_from_netlink(udev, "udev");
2794+
if (!monitor) {
2795+
r = log_error_errno(errno, "Failed to acquire udev monitor: %m");
2796+
goto finish;
2797+
}
2798+
2799+
r = udev_monitor_filter_add_match_subsystem_devtype(monitor, "block", NULL);
2800+
if (r < 0) {
2801+
log_error_errno(r, "Failed to addd udev match: %m");
2802+
goto finish;
2803+
}
2804+
2805+
r = udev_monitor_enable_receiving(monitor);
2806+
if (r < 0) {
2807+
log_error_errno(r, "Failed to start udev monitor: %m");
2808+
goto finish;
2809+
}
2810+
2811+
d = udev_device_new_from_devnum(udev, 'b', devnum);
2812+
if (!d) {
2813+
r = log_error_errno(errno, "Failed to get NBD udev device: %m");
2814+
goto finish;
2815+
}
2816+
2817+
initialized = udev_device_get_is_initialized(d) != 0;
2818+
d = udev_device_unref(d);
2819+
#else
2820+
initialized = true;
2821+
#endif
2822+
27652823
if (make_symlink) {
27662824
if (symlink(path, name) < 0) {
27672825
r = log_error_errno(errno, "Failed to create symlink %s → %s: %m", name, path);
@@ -2773,8 +2831,6 @@ static int verb_mkdev(int argc, char *argv[]) {
27732831

27742832
printf("Attached: %s\n", name ?: path);
27752833

2776-
(void) send_notify("READY=1");
2777-
27782834
for (;;) {
27792835
uint64_t req_offset = 0, req_size = 0;
27802836

@@ -2783,6 +2839,35 @@ static int verb_mkdev(int argc, char *argv[]) {
27832839
goto finish;
27842840
}
27852841

2842+
#if HAVE_UDEV
2843+
if (!initialized) {
2844+
_cleanup_(udev_device_unrefp) struct udev_device *t = NULL;
2845+
2846+
t = udev_monitor_receive_device(monitor);
2847+
if (t && udev_device_get_devnum(t) == devnum)
2848+
initialized = true;
2849+
}
2850+
#endif
2851+
2852+
if (initialized && !sent_ready) {
2853+
_cleanup_free_ char *t = NULL;
2854+
2855+
#if HAVE_UDEV
2856+
monitor = udev_monitor_unref(monitor);
2857+
udev = udev_unref(udev);
2858+
#endif
2859+
2860+
t = strjoin("READY=1\n"
2861+
"DEVICE=", path, "\n");
2862+
if (!t) {
2863+
r = log_oom();
2864+
goto finish;
2865+
}
2866+
2867+
(void) send_notify(t);
2868+
sent_ready = true;
2869+
}
2870+
27862871
r = ca_block_device_step(nbd);
27872872
if (r < 0) {
27882873
log_error_errno(r, "Failed to read NBD request: %m");
@@ -2794,15 +2879,43 @@ static int verb_mkdev(int argc, char *argv[]) {
27942879

27952880
if (r == CA_BLOCK_DEVICE_POLL) {
27962881
sigset_t ss;
2882+
int nbd_poll_fd, udev_fd;
2883+
2884+
nbd_poll_fd = ca_block_device_get_poll_fd(nbd);
2885+
if (nbd_poll_fd < 0) {
2886+
r = log_error_errno(nbd_poll_fd, "Failed to acquire NBD poll file descriptor: %m");
2887+
goto finish;
2888+
}
2889+
2890+
2891+
#if HAVE_UDEV
2892+
if (monitor) {
2893+
udev_fd = udev_monitor_get_fd(monitor);
2894+
if (udev_fd < 0) {
2895+
r = log_error_errno(udev_fd, "Failed to acquire udev monitor fd: %m");
2896+
goto finish;
2897+
}
2898+
} else
2899+
#endif
2900+
udev_fd = -1;
27972901

27982902
block_exit_handler(SIG_BLOCK, &ss);
27992903

28002904
if (quit)
28012905
r = -ESHUTDOWN;
28022906
else {
2803-
r = ca_block_device_poll(nbd, UINT64_MAX, &ss);
2804-
if ((r == -EINTR || r >= 0) && quit)
2907+
struct pollfd p[2] = {
2908+
[0] = { .fd = nbd_poll_fd, .events = POLLIN },
2909+
[1] = { .fd = udev_fd, .events = POLLIN },
2910+
};
2911+
2912+
r = ppoll(p, udev_fd < 0 ? 1 : 2, NULL, &ss);
2913+
if ((r >= 0 || errno == EINTR) && quit)
28052914
r = -ESHUTDOWN;
2915+
else if (r < 0)
2916+
r = -errno;
2917+
else
2918+
r = 0;
28062919
}
28072920

28082921
block_exit_handler(SIG_UNBLOCK, NULL);

src/udev-util.h

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
#ifndef fooudevutilhfoo
2+
#define fooudevutilhfoo
3+
4+
/* SPDX-License-Identifier: LGPL-2.1+ */
5+
6+
#include "util.h"
7+
8+
#include <libudev.h>
9+
10+
DEFINE_TRIVIAL_CLEANUP_FUNC(struct udev*, udev_unref);
11+
DEFINE_TRIVIAL_CLEANUP_FUNC(struct udev_device*, udev_device_unref);
12+
DEFINE_TRIVIAL_CLEANUP_FUNC(struct udev_monitor*, udev_monitor_unref);
13+
14+
#endif

test/semaphore-run

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
#!/bin/sh
22
# SPDX-License-Identifier: LGPL-2.1+
33

4-
DEVPKGS="pkg-config liblzma-dev libcurl4-openssl-dev libssl-dev libacl1-dev libfuse-dev zlib1g-dev libzstd-dev"
4+
DEVPKGS="pkg-config liblzma-dev libcurl4-openssl-dev libssl-dev libacl1-dev libfuse-dev zlib1g-dev libzstd-dev libudev-dev"
55

66
echo
77
echo "============= Installing amd64 build dependencies =============="

0 commit comments

Comments
 (0)