Skip to content

Commit 7138fe7

Browse files
jgottulatonyhutter
authored andcommitted
Print zvol_id error messages to stderr rather than stdout
The zvol_id program is invoked by udev, via a PROGRAM key in the 60-zvol.rules.in rule file, to determine the "pretty" /dev/zvol/* symlink paths paths that should be generated for each opaquely named /dev/zd* dev node. The udev rule uses the PROGRAM key, followed by a SYMLINK+= assignment containing the %c substitution, to collect the program's stdout and then "paste" it directly into the name of the symlink(s) to be created. Unfortunately, as currently written, zvol_id outputs both its intended output (a single string representing the symlink path that should be created to refer to the name of the dataset whose /dev/zd* path is given) AND its error messages (if any) to stdout. When processing PROGRAM keys (and others, such as IMPORT{program}), udev uses only the data written to stdout for functional purposes. Any data written to stderr is used solely for the purposes of logging (if udev's log_level is set to debug). The unintended consequence of this is as follows: if zvol_id encounters an error condition; and then udev fails to halt processing of the current rule (either because zvol_id didn't return a nonzero exit status, or because the PROGRAM key in the rule wasn't written properly to result in a "non-match" condition that would stop the current rule on a nonzero exit); then udev will create a space-delimited list of symlink names derived directly from the words of the error message string! I've observed this exact behavior on my own system, in a situation where the open() syscall on /dev/zd* dev nodes was failing sporadically (for reasons that aren't especially relevant here). Because the open() call failed, zvol_id printed "Unable to open device file: /dev/zd736\n" to stdout and then exited. The udev rule finished with SYMLINK+="zvol/%c %c". Assuming a volume name like pool/foo/bar, this would ordinarily expand to SYMLINK+="zvol/pool/foo/bar pool/foo/bar" and would cause symlinks to be created like this: /dev/zvol/pool/foo/bar -> /dev/zd736 /dev/pool/foo/bar -> /dev/zd736 But because of the combination of error messages being printed to stdout, and the udev syntax freely accepting a space-delimited sequence of names in this context, the error message string "Unable to open device file: /dev/zd736\n" in reality expanded to SYMLINK+="zvol/Unable to open device file: /dev/zd736" which caused the following symlinks to actually be created: /dev/zvol/Unable -> /dev/zd736 /dev/to -> /dev/zd736 /dev/open -> /dev/zd736 /dev/device -> /dev/zd736 /dev/file: -> /dev/zd736 /dev//dev/zd736 -> /dev/zd736 (And, because multiple zvols had open() syscall errors, multiple zvols attempted to claim several of those symlink names, resulting in numerous udev errors and timeouts and general chaos.) This commit rectifies all this silliness by simply printing error messages to stderr, as Dennis Ritchie originally intended. Reviewed-by: Brian Behlendorf <[email protected]> Reviewed-by: Pavel Zakharov <[email protected]> Signed-off-by: Justin Gottula <[email protected]> Closes openzfs#12302
1 parent fd2e4d1 commit 7138fe7

File tree

1 file changed

+4
-4
lines changed

1 file changed

+4
-4
lines changed

cmd/zvol_id/zvol_id_main.c

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -63,14 +63,14 @@ main(int argc, char **argv)
6363
int rc;
6464

6565
if (argc < 2) {
66-
printf("Usage: %s /dev/zvol_device_node\n", argv[0]);
66+
fprintf(stderr, "Usage: %s /dev/zvol_device_node\n", argv[0]);
6767
return (EINVAL);
6868
}
6969

7070
dev_name = argv[1];
7171
error = stat64(dev_name, &statbuf);
7272
if (error != 0) {
73-
printf("Unable to access device file: %s\n", dev_name);
73+
fprintf(stderr, "Unable to access device file: %s\n", dev_name);
7474
return (errno);
7575
}
7676

@@ -79,13 +79,13 @@ main(int argc, char **argv)
7979

8080
fd = open(dev_name, O_RDONLY);
8181
if (fd < 0) {
82-
printf("Unable to open device file: %s\n", dev_name);
82+
fprintf(stderr, "Unable to open device file: %s\n", dev_name);
8383
return (errno);
8484
}
8585

8686
error = ioctl_get_msg(zvol_name, fd);
8787
if (error < 0) {
88-
printf("ioctl_get_msg failed:%s\n", strerror(errno));
88+
fprintf(stderr, "ioctl_get_msg failed: %s\n", strerror(errno));
8989
return (errno);
9090
}
9191
if (dev_part > 0)

0 commit comments

Comments
 (0)