From: Yu Watanabe Date: Sun, 10 Apr 2022 13:23:34 +0000 (+0900) Subject: sd-device: validate input parameter after creating sd-device object X-Git-Tag: v251-rc2~143^2~2 X-Git-Url: http://git.ipfire.org/gitweb.cgi?a=commitdiff_plain;h=a46f9cd049a484a50d8c874c86bf6d29d3dff45a;p=thirdparty%2Fsystemd.git sd-device: validate input parameter after creating sd-device object --- diff --git a/src/libsystemd/sd-device/sd-device.c b/src/libsystemd/sd-device/sd-device.c index 22200c0eea2..2bddd81c082 100644 --- a/src/libsystemd/sd-device/sd-device.c +++ b/src/libsystemd/sd-device/sd-device.c @@ -242,22 +242,55 @@ _public_ int sd_device_new_from_syspath(sd_device **ret, const char *syspath) { return 0; } -_public_ int sd_device_new_from_devnum(sd_device **ret, char type, dev_t devnum) { - char id[DECIMAL_STR_MAX(unsigned) * 2 + 1], *syspath; +static int device_new_from_mode_and_devnum(sd_device **ret, mode_t mode, dev_t devnum) { + _cleanup_(sd_device_unrefp) sd_device *dev = NULL; + _cleanup_free_ char *syspath = NULL; + const char *t, *subsystem; + dev_t n; + int r; - assert_return(ret, -EINVAL); - assert_return(IN_SET(type, 'b', 'c'), -EINVAL); + assert(ret); - if (devnum == 0) - return log_debug_errno(SYNTHETIC_ERRNO(ENODEV), - "sd-device: Attempted to allocate device by zero major/minor, refusing."); + if (S_ISCHR(mode)) + t = "char"; + else if (S_ISBLK(mode)) + t = "block"; + else + return -ENOTTY; - /* use /sys/dev/{block,char}/: link */ - xsprintf(id, "%u:%u", major(devnum), minor(devnum)); + if (major(devnum) == 0) + return -ENODEV; - syspath = strjoina("/sys/dev/", (type == 'b' ? "block" : "char"), "/", id); + if (asprintf(&syspath, "/sys/dev/%s/%u:%u", t, major(devnum), minor(devnum)) < 0) + return -ENOMEM; - return sd_device_new_from_syspath(ret, syspath); + r = sd_device_new_from_syspath(&dev, syspath); + if (r < 0) + return r; + + r = sd_device_get_devnum(dev, &n); + if (r == -ENOENT) + return -ENXIO; + if (r < 0) + return r; + if (n != devnum) + return -ENXIO; + + r = sd_device_get_subsystem(dev, &subsystem); + if (r < 0 && r != -ENOENT) + return r; + if (r >= 0 && streq(subsystem, "block") != !!S_ISBLK(mode)) + return -ENXIO; + + *ret = TAKE_PTR(dev); + return 0; +} + +_public_ int sd_device_new_from_devnum(sd_device **ret, char type, dev_t devnum) { + assert_return(ret, -EINVAL); + assert_return(IN_SET(type, 'b', 'c'), -EINVAL); + + return device_new_from_mode_and_devnum(ret, type == 'b' ? S_IFBLK : S_IFCHR, devnum); } static int device_new_from_main_ifname(sd_device **ret, const char *ifname) { @@ -295,7 +328,9 @@ _public_ int sd_device_new_from_ifname(sd_device **ret, const char *ifname) { } _public_ int sd_device_new_from_ifindex(sd_device **ret, int ifindex) { + _cleanup_(sd_device_unrefp) sd_device *dev = NULL; char ifname[IF_NAMESIZE]; + int r, i; assert_return(ret, -EINVAL); assert_return(ifindex > 0, -EINVAL); @@ -303,7 +338,20 @@ _public_ int sd_device_new_from_ifindex(sd_device **ret, int ifindex) { if (format_ifname(ifindex, ifname) < 0) return -ENODEV; - return device_new_from_main_ifname(ret, ifname); + r = device_new_from_main_ifname(&dev, ifname); + if (r < 0) + return r; + + r = sd_device_get_ifindex(dev, &i); + if (r == -ENOENT) + return -ENXIO; + if (r < 0) + return r; + if (i != ifindex) + return -ENXIO; + + *ret = TAKE_PTR(dev); + return 0; } static int device_strjoin_new( @@ -400,23 +448,16 @@ _public_ int sd_device_new_from_subsystem_sysname( } _public_ int sd_device_new_from_stat_rdev(sd_device **ret, const struct stat *st) { - char type; - assert_return(ret, -EINVAL); assert_return(st, -EINVAL); - if (S_ISBLK(st->st_mode)) - type = 'b'; - else if (S_ISCHR(st->st_mode)) - type = 'c'; - else - return -ENOTTY; - - return sd_device_new_from_devnum(ret, type, st->st_rdev); + return device_new_from_mode_and_devnum(ret, st->st_mode, st->st_rdev); } _public_ int sd_device_new_from_devname(sd_device **ret, const char *devname) { struct stat st; + dev_t devnum; + mode_t mode; assert_return(ret, -EINVAL); assert_return(devname, -EINVAL); @@ -428,16 +469,11 @@ _public_ int sd_device_new_from_devname(sd_device **ret, const char *devname) { if (isempty(path_startswith(devname, "/dev"))) return -EINVAL; - if (device_path_parse_major_minor(devname, NULL, NULL) >= 0) { - _cleanup_free_ char *syspath = NULL; - + if (device_path_parse_major_minor(devname, &mode, &devnum) >= 0) /* Let's shortcut when "/dev/block/maj:min" or "/dev/char/maj:min" is specified. - * In that case, we directly convert the path to syspath, hence it is not necessary + * In that case, we can directly convert the path to syspath, hence it is not necessary * that the specified path exists. So, this works fine without udevd being running. */ - - syspath = path_join("/sys", devname); - return sd_device_new_from_syspath(ret, syspath); - } + return device_new_from_mode_and_devnum(ret, mode, devnum); if (stat(devname, &st) < 0) return ERRNO_IS_DEVICE_ABSENT(errno) ? -ENODEV : -errno;