libudev: handle "device" link as parent, handle "class" "block" as "subsystem"

This commit is contained in:
Kay Sievers 2008-09-15 11:19:56 -07:00
parent 93b0f38458
commit 0518da3b74
4 changed files with 76 additions and 26 deletions

3
TODO
View file

@ -11,6 +11,5 @@ These things will change in future udev versions:
o symlink names to udevadm will no longer be resolved to old command names
before next release:
o handle old "device" link in get_parent (CONFIG_SYSFS_DEPRECATED*=y)
o handle "subsystem", "driver", as device
o switch libudev API to syspath instead of devpath
o udev_* prefix for private methods (ctrl_*)?

View file

@ -142,7 +142,7 @@ static int device_read_db(struct udev_device *udev_device)
device_set_ignore_remove(udev_device, atoi(val));
break;
case 'E':
device_add_property(udev_device, val);
device_add_property_from_string(udev_device, val);
break;
}
}
@ -236,15 +236,29 @@ static struct udev_device *device_new_from_parent(struct udev_device *udev_devic
pos[0] = '\0';
udev_device_parent = udev_device_new_from_devpath(udev_device->udev, path);
if (udev_device_parent != NULL)
break;
return udev_device_parent;
}
return udev_device_parent;
/* follow "device" link in deprecated sysfs /sys/class/ layout */
if (strncmp(udev_device->devpath, "/class/", 7) == 0) {
util_strlcpy(path, udev_device->devpath, sizeof(path));
util_strlcat(path, "/device", sizeof(path));
if (util_resolve_sys_link(udev_device->udev, path, sizeof(path)) == 0) {
udev_device_parent = udev_device_new_from_devpath(udev_device->udev, path);
if (udev_device_parent != NULL)
return udev_device_parent;
}
}
return NULL;
}
struct udev_device *udev_device_get_parent(struct udev_device *udev_device)
{
if (udev_device->parent_device == NULL)
udev_device->parent_device = device_new_from_parent(udev_device);
if (udev_device->parent_device != NULL) {
info(udev_device->udev, "returning existing parent %p\n", udev_device->parent_device);
return udev_device->parent_device;
}
udev_device->parent_device = device_new_from_parent(udev_device);
return udev_device->parent_device;
}
@ -382,10 +396,29 @@ const char *udev_device_get_subsystem(struct udev_device *udev_device)
return NULL;
if (udev_device->subsystem != NULL)
return udev_device->subsystem;
if (util_get_sys_subsystem(udev_device->udev, udev_device->devpath, subsystem, sizeof(subsystem)) < 2)
return NULL;
udev_device->subsystem = strdup(subsystem);
return udev_device->subsystem;
/* read "subsytem" link */
if (util_get_sys_subsystem(udev_device->udev, udev_device->devpath, subsystem, sizeof(subsystem)) == 0) {
udev_device->subsystem = strdup(subsystem);
return udev_device->subsystem;
}
/* implicit names */
if (strncmp(udev_device->devpath, "/module/", 8) == 0) {
udev_device->subsystem = strdup("module");
return udev_device->subsystem;
}
if (strstr(udev_device->devpath, "/drivers/") != NULL) {
udev_device->subsystem = strdup("drivers");
return udev_device->subsystem;
}
if (strncmp(udev_device->devpath, "/subsystem/", 11) == 0 ||
strncmp(udev_device->devpath, "/class/", 7) == 0 ||
strncmp(udev_device->devpath, "/bus/", 5) == 0) {
udev_device->subsystem = strdup("subsystem");
return udev_device->subsystem;
}
return NULL;
}
/**
@ -442,17 +475,8 @@ int udev_device_get_properties(struct udev_device *udev_device,
if (udev_device == NULL)
return -1;
list_for_each_entry(name_loop, &udev_device->env_list, node) {
char name[UTIL_NAME_SIZE];
char *val;
strncpy(name, name_loop->name, sizeof(name));
val = strchr(name, '=');
if (val == NULL)
continue;
val[0] = '\0';
val = &val[1];
count++;
if (cb(udev_device, name, val, data) != 0)
if (cb(udev_device, name_loop->name, name_loop->value, data) != 0)
break;
}
return count;
@ -495,6 +519,7 @@ unsigned long long int udev_device_get_seqnum(struct udev_device *udev_device)
const char *udev_device_get_attr_value(struct udev_device *udev_device, const char *attr)
{
struct util_name_entry *name_loop;
char path[UTIL_PATH_SIZE];
char value[UTIL_NAME_SIZE];
struct stat statbuf;
@ -502,6 +527,14 @@ const char *udev_device_get_attr_value(struct udev_device *udev_device, const ch
ssize_t size;
const char *val = NULL;
/* look for possibly already cached result */
list_for_each_entry(name_loop, &udev_device->attr_list, node) {
if (strcmp(name_loop->name, attr) == 0) {
info(udev_device->udev, "'%s' in cache '%s'\n", attr, name_loop->value);
return name_loop->value;
}
}
util_strlcpy(path, udev_device_get_syspath(udev_device), sizeof(path));
util_strlcat(path, "/", sizeof(path));
util_strlcat(path, attr, sizeof(path));
@ -551,7 +584,7 @@ const char *udev_device_get_attr_value(struct udev_device *udev_device, const ch
if (size == sizeof(value))
goto out;
/* got a valid value, store and return it */
/* got a valid value, store it in cache and return it */
value[size] = '\0';
util_remove_trailing_chars(value, '\n');
info(udev_device->udev, "'%s' has attribute value '%s'\n", path, value);
@ -593,13 +626,30 @@ int device_add_devlink(struct udev_device *udev_device, const char *devlink)
return 0;
}
int device_add_property(struct udev_device *udev_device, const char *property)
int device_add_property(struct udev_device *udev_device, const char *key, const char *value)
{
if (util_name_list_add(udev_device->udev, &udev_device->env_list, property, NULL, 0) == NULL)
if (util_name_list_add(udev_device->udev, &udev_device->env_list, key, value, 0) == NULL)
return -ENOMEM;
return 0;
}
int device_add_property_from_string(struct udev_device *udev_device, const char *property)
{
char name[UTIL_PATH_SIZE];
char *val;
strncpy(name, property, sizeof(name));
val = strchr(name, '=');
if (val == NULL)
return -1;
val[0] = '\0';
val = &val[1];
if (val[0] == '\0')
val = NULL;
device_add_property(udev_device, name, val);
return 0;
}
int device_set_action(struct udev_device *udev_device, const char *action)
{
udev_device->action = strdup(action);

View file

@ -336,7 +336,7 @@ struct udev_device *udev_monitor_receive_device(struct udev_monitor *udev_monito
} else if (strncmp(key, "TIMEOUT=", 8) == 0) {
device_set_timeout(udev_device, strtoull(&key[8], NULL, 10));
}
device_add_property(udev_device, key);
device_add_property_from_string(udev_device, key);
}
device_set_devnum(udev_device, makedev(maj, min));

View file

@ -59,7 +59,8 @@ extern int device_set_devpath(struct udev_device *udev_device, const char *devpa
extern int device_set_subsystem(struct udev_device *udev_device, const char *subsystem);
extern int device_set_devname(struct udev_device *udev_device, const char *devname);
extern int device_add_devlink(struct udev_device *udev_device, const char *devlink);
extern int device_add_property(struct udev_device *udev_device, const char *property);
extern int device_add_property(struct udev_device *udev_device, const char *key, const char *value);
extern int device_add_property_from_string(struct udev_device *udev_device, const char *property);
extern int device_set_action(struct udev_device *udev_device, const char *action);
extern int device_set_driver(struct udev_device *udev_device, const char *driver);
extern const char *device_get_devpath_old(struct udev_device *udev_device);