X-Git-Url: https://www.chiark.greenend.org.uk/ucgi/~ianmdlvl/git?p=elogind.git;a=blobdiff_plain;f=src%2Fudev%2Fudev-builtin-net_id.c;h=b6e7eee6d8f1af14ac4313e371ba0ba925ce9d4f;hp=eff4552778cf24a5d99e9a66b1929d5237beb7ba;hb=0035597a30d120f70df2dd7da3d6128fb8ba6051;hpb=a660c63c551b88136ac6176855b5907cc533e848 diff --git a/src/udev/udev-builtin-net_id.c b/src/udev/udev-builtin-net_id.c index eff455277..b6e7eee6d 100644 --- a/src/udev/udev-builtin-net_id.c +++ b/src/udev/udev-builtin-net_id.c @@ -17,6 +17,25 @@ along with systemd; If not, see . ***/ +/* + * prefixes: + * en -- ethernet + * wl -- wlan + * ww -- wwan + * + * types: + * o -- on-board device index + * sf -- hotplug slot number + * x -- MAC address + * psf -- PCI/physical location + * + * example: + * ID_NET_NAME_ONBOARD=eno1 + * ID_NET_NAME_SLOT=ens1f0 + * ID_NET_NAME_MAC=enxf0def180d479 + * ID_NET_NAME_PATH=enp19s0f0 + */ + #include #include #include @@ -26,25 +45,175 @@ #include "udev.h" -/* IEEE Organizationally Unique Identifier */ -static int lookup_OUI(struct udev_device *dev, bool test) { - const char *addr; - unsigned int a1, a2, a3; - char oui[16]; +/* retrieve on-board index number and label from firmware */ +static int dev_pci_onboard(struct udev_device *dev, const char *prefix, bool test) { + const char *index; + int idx; + const char *label; + char s[16]; + int err; + + /* ACPI _DSM -- device specific method for naming a PCI or PCI Express device */ + index = udev_device_get_sysattr_value(dev, "acpi_index"); + /* SMBIOS type 41 -- Onboard Devices Extended Information */ + if (!index) + index = udev_device_get_sysattr_value(dev, "index"); + if (!index) + return -ENOENT; + idx = strtoul(index, NULL, 0); + if (idx <= 0) + return -EINVAL; + snprintf(s, sizeof(s), "%so%d", prefix, idx); + err = udev_builtin_add_property(dev, test, "ID_NET_NAME_ONBOARD", s); + if (err < 0) + return err; + + label = udev_device_get_sysattr_value(dev, "label"); + if (label) { + err = udev_builtin_add_property(dev, test, "ID_NET_LABEL_ONBOARD", label); + if (err < 0) + return err; + } + return 0; +} + +static int dev_pci_slot(struct udev_device *dev, const char *prefix, bool test) { + struct udev *udev = udev_device_get_udev(dev); + unsigned int bus; + unsigned int slot; + unsigned int func; + struct udev_device *pci = NULL; + char slots[256]; + DIR *dir; + struct dirent *dent; + char str[256]; + int hotplug_slot = 0; + int err = 0; - addr = udev_device_get_sysattr_value(dev, "address"); - if (!addr) + /* compose a name based on the raw kernel's PCI bus, slot numbers */ + if (sscanf(udev_device_get_sysname(dev), "0000:%x:%x.%d", &bus, &slot, &func) != 3) return -ENOENT; + snprintf(str, sizeof(str), "%sp%ds%df%d", prefix, bus, slot, func); + err = udev_builtin_add_property(dev, test, "ID_NET_NAME_PATH", str); + if (err < 0) + return err; + + /* ACPI _SUN -- slot user number */ + pci = udev_device_new_from_subsystem_sysname(udev, "subsystem", "pci"); + if (!pci) { + err = -ENOENT; + goto out; + } + snprintf(slots, sizeof(slots), "%s/slots", udev_device_get_syspath(pci)); + dir = opendir(slots); + if (!dir) { + err = -errno; + goto out; + } + + for (dent = readdir(dir); dent != NULL; dent = readdir(dir)) { + int i; + char *rest; + char *address; + + if (dent->d_name[0] == '.') + continue; + i = strtol(dent->d_name, &rest, 10); + if (rest[0] != '\0') + continue; + if (i < 1) + continue; + snprintf(str, sizeof(str), "%s/%s/address", slots, dent->d_name); + if (read_one_line_file(str, &address) >= 0) { + /* match slot address with device by stripping the function */ + if (strncmp(address, udev_device_get_sysname(dev), strlen(address)) == 0) + hotplug_slot = i; + free(address); + } + + if (hotplug_slot > 0) + break; + } + closedir(dir); + + if (hotplug_slot > 0) { + snprintf(str, sizeof(str), "%ss%df%d", prefix, hotplug_slot, func); + err = udev_builtin_add_property(dev, test, "ID_NET_NAME_SLOT", str); + } +out: + udev_device_unref(pci); + return err; +} + +static int dev_pci(struct udev_device *dev, const char *prefix, bool test) { + struct udev_device *d; - if (sscanf(addr, "%x:%x:%x:", &a1, &a2, &a3) != 3) + /* skip other buses than direct PCI parents */ + d = udev_device_get_parent(dev); + if (!d || !streq("pci", udev_device_get_subsystem(d))) + return -ENOENT; + + dev_pci_onboard(d, prefix, test); + dev_pci_slot(d, prefix, test); + return 0; +} + +static int dev_mac(struct udev_device *dev, const char *prefix, bool test) { + const char *s; + unsigned int i; + unsigned int a1, a2, a3, a4, a5, a6; + char str[16]; + + /* check for NET_ADDR_PERM, skip random MAC addresses */ + s = udev_device_get_sysattr_value(dev, "addr_assign_type"); + if (!s) + return EXIT_FAILURE; + i = strtoul(s, NULL, 0); + if (i != 0) + return 0; + + s = udev_device_get_sysattr_value(dev, "address"); + if (!s) + return -ENOENT; + if (sscanf(s, "%x:%x:%x:%x:%x:%x", &a1, &a2, &a3, &a4, &a5, &a6) != 6) return -EINVAL; - snprintf(oui, sizeof(oui), "OUI:%X%X%X", a1, a2, a3); - return udev_builtin_hwdb_lookup(dev, oui, test); + /* skip empty MAC addresses */ + if (a1 + a2 + a3 + a4 + a5 + a6 == 0) + return -EINVAL; + + /* add IEEE Organizationally Unique Identifier */ + snprintf(str, sizeof(str), "OUI:%X%X%X", a1, a2, a3); + udev_builtin_hwdb_lookup(dev, str, test); + + snprintf(str, sizeof(str), "%sx%x%x%x%x%x%x", prefix, a1, a2, a3, a4, a5, a6); + return udev_builtin_add_property(dev, test, "ID_NET_NAME_MAC", str); } static int builtin_net_id(struct udev_device *dev, int argc, char *argv[], bool test) { - lookup_OUI(dev, test); + const char *s; + unsigned int i; + const char *devtype; + const char *prefix = "en"; + + /* handle only ARPHRD_ETHER devices */ + s = udev_device_get_sysattr_value(dev, "type"); + if (!s) + return EXIT_FAILURE; + i = strtoul(s, NULL, 0); + if (i != 1) + return 0; + + devtype = udev_device_get_devtype(dev); + if (devtype) { + if (streq("wlan", devtype)) + prefix = "wl"; + else if (streq("wwan", devtype)) + prefix = "ww"; + } + + dev_pci(dev, prefix, test); + dev_mac(dev, prefix, test); return EXIT_SUCCESS; }