+ if (!udev_enumerate->devices_sorted)
+ devices_sort(udev_enumerate);
+ return udev_list_get_entry(&udev_enumerate->devices_list);
+}
+
+int udev_enumerate_add_match_subsystem(struct udev_enumerate *udev_enumerate, const char *subsystem)
+{
+ if (udev_enumerate == NULL)
+ return -EINVAL;
+ if (subsystem == NULL)
+ return 0;
+ if (udev_list_entry_add(udev_enumerate_get_udev(udev_enumerate),
+ &udev_enumerate->subsystem_match_list, subsystem, NULL, 1, 0) == NULL)
+ return -ENOMEM;
+ return 0;
+}
+
+int udev_enumerate_add_nomatch_subsystem(struct udev_enumerate *udev_enumerate, const char *subsystem)
+{
+ if (udev_enumerate == NULL)
+ return -EINVAL;
+ if (subsystem == NULL)
+ return 0;
+ if (udev_list_entry_add(udev_enumerate_get_udev(udev_enumerate),
+ &udev_enumerate->subsystem_nomatch_list, subsystem, NULL, 1, 0) == NULL)
+ return -ENOMEM;
+ return 0;
+}
+
+int udev_enumerate_add_match_sysattr(struct udev_enumerate *udev_enumerate, const char *sysattr, const char *value)
+{
+ if (udev_enumerate == NULL)
+ return -EINVAL;
+ if (sysattr == NULL)
+ return 0;
+ if (udev_list_entry_add(udev_enumerate_get_udev(udev_enumerate),
+ &udev_enumerate->sysattr_match_list, sysattr, value, 1, 0) == NULL)
+ return -ENOMEM;
+ return 0;
+}
+
+int udev_enumerate_add_nomatch_sysattr(struct udev_enumerate *udev_enumerate, const char *sysattr, const char *value)
+{
+ if (udev_enumerate == NULL)
+ return -EINVAL;
+ if (sysattr == NULL)
+ return 0;
+ if (udev_list_entry_add(udev_enumerate_get_udev(udev_enumerate),
+ &udev_enumerate->sysattr_nomatch_list, sysattr, value, 1, 0) == NULL)
+ return -ENOMEM;
+ return 0;
+}
+
+static int match_sysattr_value(struct udev *udev, const char *syspath, const char *sysattr, const char *match_val)
+{
+ struct udev_device *device;
+ const char *val = NULL;
+ int match = 0;
+
+ device = udev_device_new_from_syspath(udev, syspath);
+ if (device == NULL)
+ return -EINVAL;
+ val = udev_device_get_sysattr_value(device, sysattr);
+ if (val == NULL)
+ goto exit;
+ if (match_val == NULL) {
+ match = 1;
+ goto exit;
+ }
+ if (fnmatch(match_val, val, 0) == 0) {
+ match = 1;
+ goto exit;
+ }
+exit:
+ udev_device_unref(device);
+ return match;
+}
+
+static int match_sysattr(struct udev_enumerate *udev_enumerate, const char *syspath)
+{
+ struct udev *udev = udev_enumerate_get_udev(udev_enumerate);
+ struct udev_list_entry *list_entry;
+
+ /* skip list */
+ udev_list_entry_foreach(list_entry, udev_list_get_entry(&udev_enumerate->sysattr_nomatch_list)) {
+ if (match_sysattr_value(udev, syspath,
+ udev_list_entry_get_name(list_entry),
+ udev_list_entry_get_value(list_entry)))
+ return 0;
+ }
+ /* include list */
+ if (udev_list_get_entry(&udev_enumerate->sysattr_match_list) != NULL) {
+ udev_list_entry_foreach(list_entry, udev_list_get_entry(&udev_enumerate->sysattr_match_list)) {
+ /* anything that does not match, will make it FALSE */
+ if (!match_sysattr_value(udev, syspath,
+ udev_list_entry_get_name(list_entry),
+ udev_list_entry_get_value(list_entry)))
+ return 0;
+ }
+ return 1;
+ }
+ return 1;