mirror of
https://github.com/systemd/systemd-stable.git
synced 2025-08-30 05:50:12 +03:00
[PATCH] experimental (very simple) SYMLINK creation
> > here is a experimental symlink creation patch - for discussion, > > in which direction we should go. > > It is possible now to define SYMLINK= after the NAME= in udev.rules. > > The link is relative to the node, but the path is not optimized now > > if the node and the link are in the same nested directory. > > Only one link is supported, cause i need to sleep now :) > > > > 06-simple-symlink-creation.diff > > simple symlink creation > > reorganized udev-remove to have access to the symlink field > > subdir creation/removal are functions now > > udev-test.pl tests for link creation/removal Here is a new version with relative link target path optimization an better tests in udev-test.pl: LABEL, BUS="scsi", vendor="IBM-ESXS", NAME="1/2/a/b/node", SYMLINK="1/2/c/d/symlink" Dec 7 06:48:34 pim udev[13789]: create_node: symlink 'udev-root/1/2/c/d/symlink' to node '1/2/a/b/node' requested Dec 7 06:48:34 pim udev[13789]: create_path: created 'udev-root/1/2/c' Dec 7 06:48:34 pim udev[13789]: create_path: created 'udev-root/1/2/c/d' Dec 7 06:48:34 pim udev[13789]: create_node: symlink(../../a/b/node, udev-root/1/2/c/d/symlink)
This commit is contained in:
committed by
Greg KH
parent
eadb1bbc2e
commit
3d150dfb28
@ -121,6 +121,7 @@ int add_config_dev(struct config_device *new_dev)
|
|||||||
copy_string(dev, new_dev, place);
|
copy_string(dev, new_dev, place);
|
||||||
copy_string(dev, new_dev, kernel_name);
|
copy_string(dev, new_dev, kernel_name);
|
||||||
copy_string(dev, new_dev, exec_program);
|
copy_string(dev, new_dev, exec_program);
|
||||||
|
copy_string(dev, new_dev, symlink);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -366,6 +367,7 @@ static int do_callout(struct sysfs_class_device *class_dev, struct udevice *udev
|
|||||||
if (strcmp_pattern(dev->id, udev->callout_value) != 0)
|
if (strcmp_pattern(dev->id, udev->callout_value) != 0)
|
||||||
continue;
|
continue;
|
||||||
strfieldcpy(udev->name, dev->name);
|
strfieldcpy(udev->name, dev->name);
|
||||||
|
strfieldcpy(udev->symlink, dev->symlink);
|
||||||
dbg("callout returned matching value '%s', '%s' becomes '%s'",
|
dbg("callout returned matching value '%s', '%s' becomes '%s'",
|
||||||
dev->id, class_dev->name, udev->name);
|
dev->id, class_dev->name, udev->name);
|
||||||
return 0;
|
return 0;
|
||||||
@ -416,6 +418,7 @@ label_found:
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
strfieldcpy(udev->name, dev->name);
|
strfieldcpy(udev->name, dev->name);
|
||||||
|
strfieldcpy(udev->symlink, dev->symlink);
|
||||||
dbg("found matching attribute '%s', '%s' becomes '%s' ",
|
dbg("found matching attribute '%s', '%s' becomes '%s' ",
|
||||||
dev->sysfs_file, class_dev->name, udev->name);
|
dev->sysfs_file, class_dev->name, udev->name);
|
||||||
|
|
||||||
@ -461,6 +464,7 @@ static int do_number(struct sysfs_class_device *class_dev, struct udevice *udev,
|
|||||||
if (!found)
|
if (!found)
|
||||||
continue;
|
continue;
|
||||||
strfieldcpy(udev->name, dev->name);
|
strfieldcpy(udev->name, dev->name);
|
||||||
|
strfieldcpy(udev->symlink, dev->symlink);
|
||||||
dbg("found matching id '%s', '%s' becomes '%s'",
|
dbg("found matching id '%s', '%s' becomes '%s'",
|
||||||
dev->id, class_dev->name, udev->name);
|
dev->id, class_dev->name, udev->name);
|
||||||
return 0;
|
return 0;
|
||||||
@ -506,6 +510,7 @@ static int do_topology(struct sysfs_class_device *class_dev, struct udevice *ude
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
strfieldcpy(udev->name, dev->name);
|
strfieldcpy(udev->name, dev->name);
|
||||||
|
strfieldcpy(udev->symlink, dev->symlink);
|
||||||
dbg("found matching place '%s', '%s' becomes '%s'",
|
dbg("found matching place '%s', '%s' becomes '%s'",
|
||||||
dev->place, class_dev->name, udev->name);
|
dev->place, class_dev->name, udev->name);
|
||||||
return 0;
|
return 0;
|
||||||
@ -528,6 +533,7 @@ static int do_replace(struct sysfs_class_device *class_dev, struct udevice *udev
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
strfieldcpy(udev->name, dev->name);
|
strfieldcpy(udev->name, dev->name);
|
||||||
|
strfieldcpy(udev->symlink, dev->symlink);
|
||||||
dbg("found name, '%s' becomes '%s'", dev->kernel_name, udev->name);
|
dbg("found name, '%s' becomes '%s'", dev->kernel_name, udev->name);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -618,8 +624,9 @@ int namedev_name_device(struct sysfs_class_device *class_dev, struct udevice *ud
|
|||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
found:
|
found:
|
||||||
/* substitute placeholder in NAME */
|
/* substitute placeholder */
|
||||||
apply_format(udev, udev->name);
|
apply_format(udev, udev->name);
|
||||||
|
apply_format(udev, udev->symlink);
|
||||||
|
|
||||||
done:
|
done:
|
||||||
perm = find_perm(udev->name);
|
perm = find_perm(udev->name);
|
||||||
|
@ -63,6 +63,7 @@ struct config_device {
|
|||||||
char kernel_name[NAME_SIZE];
|
char kernel_name[NAME_SIZE];
|
||||||
char exec_program[FILE_SIZE];
|
char exec_program[FILE_SIZE];
|
||||||
char name[NAME_SIZE];
|
char name[NAME_SIZE];
|
||||||
|
char symlink[NAME_SIZE];
|
||||||
};
|
};
|
||||||
|
|
||||||
struct perm_device {
|
struct perm_device {
|
||||||
|
@ -213,10 +213,16 @@ int namedev_init_rules(void)
|
|||||||
break;
|
break;
|
||||||
strfieldcpy(dev.name, temp3);
|
strfieldcpy(dev.name, temp3);
|
||||||
|
|
||||||
|
/* SYMLINK="name" */
|
||||||
|
temp2 = strsep(&temp, ",");
|
||||||
|
retval = get_value("SYMLINK", &temp, &temp3);
|
||||||
|
if (retval == 0)
|
||||||
|
strfieldcpy(dev.symlink, temp3);
|
||||||
|
|
||||||
dbg_parse("LABEL name='%s', bus='%s', "
|
dbg_parse("LABEL name='%s', bus='%s', "
|
||||||
"sysfs_file='%s', sysfs_value='%s'",
|
"sysfs_file='%s', sysfs_value='%s', symlink='%s'",
|
||||||
dev.name, dev.bus, dev.sysfs_file,
|
dev.name, dev.bus, dev.sysfs_file,
|
||||||
dev.sysfs_value);
|
dev.sysfs_value, dev.symlink);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strcasecmp(temp2, TYPE_NUMBER) == 0) {
|
if (strcasecmp(temp2, TYPE_NUMBER) == 0) {
|
||||||
@ -243,8 +249,14 @@ int namedev_init_rules(void)
|
|||||||
break;
|
break;
|
||||||
strfieldcpy(dev.name, temp3);
|
strfieldcpy(dev.name, temp3);
|
||||||
|
|
||||||
dbg_parse("NUMBER name='%s', bus='%s', id='%s'",
|
/* SYMLINK="name" */
|
||||||
dev.name, dev.bus, dev.id);
|
temp2 = strsep(&temp, ",");
|
||||||
|
retval = get_value("SYMLINK", &temp, &temp3);
|
||||||
|
if (retval == 0)
|
||||||
|
strfieldcpy(dev.symlink, temp3);
|
||||||
|
|
||||||
|
dbg_parse("NUMBER name='%s', bus='%s', id='%s', symlink='%s'",
|
||||||
|
dev.name, dev.bus, dev.id, dev.symlink);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strcasecmp(temp2, TYPE_TOPOLOGY) == 0) {
|
if (strcasecmp(temp2, TYPE_TOPOLOGY) == 0) {
|
||||||
@ -271,8 +283,15 @@ int namedev_init_rules(void)
|
|||||||
break;
|
break;
|
||||||
strfieldcpy(dev.name, temp3);
|
strfieldcpy(dev.name, temp3);
|
||||||
|
|
||||||
dbg_parse("TOPOLOGY name='%s', bus='%s', place='%s'",
|
/* SYMLINK="name" */
|
||||||
dev.name, dev.bus, dev.place);
|
temp2 = strsep(&temp, ",");
|
||||||
|
retval = get_value("SYMLINK", &temp, &temp3);
|
||||||
|
if (retval == 0)
|
||||||
|
strfieldcpy(dev.symlink, temp3);
|
||||||
|
|
||||||
|
dbg_parse("TOPOLOGY name='%s', bus='%s', "
|
||||||
|
"place='%s', symlink='%s'",
|
||||||
|
dev.name, dev.bus, dev.place, dev.symlink);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strcasecmp(temp2, TYPE_REPLACE) == 0) {
|
if (strcasecmp(temp2, TYPE_REPLACE) == 0) {
|
||||||
@ -291,9 +310,17 @@ int namedev_init_rules(void)
|
|||||||
if (retval)
|
if (retval)
|
||||||
break;
|
break;
|
||||||
strfieldcpy(dev.name, temp3);
|
strfieldcpy(dev.name, temp3);
|
||||||
dbg_parse("REPLACE name='%s', kernel_name='%s'",
|
|
||||||
dev.name, dev.kernel_name);
|
/* SYMLINK="name" */
|
||||||
|
temp2 = strsep(&temp, ",");
|
||||||
|
retval = get_value("SYMLINK", &temp, &temp3);
|
||||||
|
if (retval == 0)
|
||||||
|
strfieldcpy(dev.symlink, temp3);
|
||||||
|
|
||||||
|
dbg_parse("REPLACE name='%s', kernel_name='%s', symlink='%s'",
|
||||||
|
dev.name, dev.kernel_name, dev.symlink);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strcasecmp(temp2, TYPE_CALLOUT) == 0) {
|
if (strcasecmp(temp2, TYPE_CALLOUT) == 0) {
|
||||||
/* number type */
|
/* number type */
|
||||||
dev.type = CALLOUT;
|
dev.type = CALLOUT;
|
||||||
@ -324,8 +351,17 @@ int namedev_init_rules(void)
|
|||||||
if (retval)
|
if (retval)
|
||||||
break;
|
break;
|
||||||
strfieldcpy(dev.name, temp3);
|
strfieldcpy(dev.name, temp3);
|
||||||
dbg_parse("CALLOUT name='%s', program='%s'",
|
|
||||||
dev.name, dev.exec_program);
|
/* SYMLINK="name" */
|
||||||
|
temp2 = strsep(&temp, ",");
|
||||||
|
retval = get_value("SYMLINK", &temp, &temp3);
|
||||||
|
if (retval == 0)
|
||||||
|
strfieldcpy(dev.symlink, temp3);
|
||||||
|
|
||||||
|
dbg_parse("CALLOUT name='%s', bus='%s', program='%s', "
|
||||||
|
"id='%s', symlink='%s'",
|
||||||
|
dev.name, dev.bus, dev.exec_program,
|
||||||
|
dev.id, dev.symlink);
|
||||||
}
|
}
|
||||||
|
|
||||||
retval = add_config_dev(&dev);
|
retval = add_config_dev(&dev);
|
||||||
@ -414,7 +450,7 @@ int namedev_init_permissions(void)
|
|||||||
dev.mode);
|
dev.mode);
|
||||||
retval = add_perm_dev(&dev);
|
retval = add_perm_dev(&dev);
|
||||||
if (retval) {
|
if (retval) {
|
||||||
dbg("add_config_dev returned with error %d", retval);
|
dbg("add_perm_dev returned with error %d", retval);
|
||||||
goto exit;
|
goto exit;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -39,7 +39,7 @@ my @tests = (
|
|||||||
expected => "boot_disk" ,
|
expected => "boot_disk" ,
|
||||||
conf => <<EOF
|
conf => <<EOF
|
||||||
LABEL, BUS="scsi", vendor="IBM-ESXS", NAME="boot_disk%n"
|
LABEL, BUS="scsi", vendor="IBM-ESXS", NAME="boot_disk%n"
|
||||||
REPLACE, KERNEL="ttyUSB0", NAME="visor""
|
REPLACE, KERNEL="ttyUSB0", NAME="visor"
|
||||||
EOF
|
EOF
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -179,6 +179,42 @@ EOF
|
|||||||
CALLOUT, BUS="usb", PROGRAM="/bin/echo -n usb-%b", ID="*", NAME="%c"
|
CALLOUT, BUS="usb", PROGRAM="/bin/echo -n usb-%b", ID="*", NAME="%c"
|
||||||
CALLOUT, BUS="scsi", PROGRAM="/bin/echo -n scsi-%b", ID="*", NAME="%c"
|
CALLOUT, BUS="scsi", PROGRAM="/bin/echo -n scsi-%b", ID="*", NAME="%c"
|
||||||
CALLOUT, BUS="foo", PROGRAM="/bin/echo -n foo-%b", ID="*", NAME="%c"
|
CALLOUT, BUS="foo", PROGRAM="/bin/echo -n foo-%b", ID="*", NAME="%c"
|
||||||
|
EOF
|
||||||
|
},
|
||||||
|
{
|
||||||
|
desc => "symlink creation (same directory)",
|
||||||
|
subsys => "tty",
|
||||||
|
devpath => "class/tty/ttyUSB0",
|
||||||
|
expected => "visor0" ,
|
||||||
|
conf => <<EOF
|
||||||
|
REPLACE, KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="visor%n"
|
||||||
|
EOF
|
||||||
|
},
|
||||||
|
{
|
||||||
|
desc => "symlink creation (relative link back)",
|
||||||
|
subsys => "block",
|
||||||
|
devpath => "block/sda/sda2",
|
||||||
|
expected => "1/2/a/b/symlink" ,
|
||||||
|
conf => <<EOF
|
||||||
|
LABEL, BUS="scsi", vendor="IBM-ESXS", NAME="1/2/node", SYMLINK="1/2/a/b/symlink"
|
||||||
|
EOF
|
||||||
|
},
|
||||||
|
{
|
||||||
|
desc => "symlink creation (relative link forward)",
|
||||||
|
subsys => "block",
|
||||||
|
devpath => "block/sda/sda2",
|
||||||
|
expected => "1/2/symlink" ,
|
||||||
|
conf => <<EOF
|
||||||
|
LABEL, BUS="scsi", vendor="IBM-ESXS", NAME="1/2/a/b/node", SYMLINK="1/2/symlink"
|
||||||
|
EOF
|
||||||
|
},
|
||||||
|
{
|
||||||
|
desc => "symlink creation (relative link back and forward)",
|
||||||
|
subsys => "block",
|
||||||
|
devpath => "block/sda/sda2",
|
||||||
|
expected => "1/2/c/d/symlink" ,
|
||||||
|
conf => <<EOF
|
||||||
|
LABEL, BUS="scsi", vendor="IBM-ESXS", NAME="1/2/a/b/node", SYMLINK="1/2/c/d/symlink"
|
||||||
EOF
|
EOF
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@ -238,7 +274,8 @@ foreach my $config (@tests) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
udev("remove", $config->{subsys}, $config->{devpath}, \$config->{conf});
|
udev("remove", $config->{subsys}, $config->{devpath}, \$config->{conf});
|
||||||
if (-e "$PWD/$udev_root$config->{expected}") {
|
if ((-e "$PWD/$udev_root$config->{expected}") ||
|
||||||
|
(-l "$PWD/$udev_root$config->{expected}")) {
|
||||||
print "remove: error\n\n";
|
print "remove: error\n\n";
|
||||||
system("tree $udev_root");
|
system("tree $udev_root");
|
||||||
$error++;
|
$error++;
|
||||||
|
99
udev-add.c
99
udev-add.c
@ -72,6 +72,34 @@ exit:
|
|||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int create_path(char *file)
|
||||||
|
{
|
||||||
|
char p[NAME_SIZE];
|
||||||
|
char *pos;
|
||||||
|
int retval;
|
||||||
|
struct stat stats;
|
||||||
|
|
||||||
|
strncpy(p, file, sizeof(p));
|
||||||
|
pos = strchr(p+1, '/');
|
||||||
|
while (1) {
|
||||||
|
pos = strchr(pos+1, '/');
|
||||||
|
if (pos == NULL)
|
||||||
|
break;
|
||||||
|
*pos = 0x00;
|
||||||
|
if (stat(p, &stats)) {
|
||||||
|
retval = mkdir(p, 0755);
|
||||||
|
if (retval) {
|
||||||
|
dbg("mkdir(%s) failed with error '%s'",
|
||||||
|
p, strerror(errno));
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
dbg("created '%s'", p);
|
||||||
|
}
|
||||||
|
*pos = '/';
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* we possibly want to add some symlinks here
|
* we possibly want to add some symlinks here
|
||||||
* only numeric owner/group id's are supported
|
* only numeric owner/group id's are supported
|
||||||
@ -79,10 +107,14 @@ exit:
|
|||||||
static int create_node(struct udevice *dev)
|
static int create_node(struct udevice *dev)
|
||||||
{
|
{
|
||||||
char filename[255];
|
char filename[255];
|
||||||
|
char linktarget[255];
|
||||||
int retval = 0;
|
int retval = 0;
|
||||||
uid_t uid = 0;
|
uid_t uid = 0;
|
||||||
gid_t gid = 0;
|
gid_t gid = 0;
|
||||||
dev_t res;
|
dev_t res;
|
||||||
|
int i;
|
||||||
|
int tail;
|
||||||
|
|
||||||
|
|
||||||
strncpy(filename, udev_root, sizeof(filename));
|
strncpy(filename, udev_root, sizeof(filename));
|
||||||
strncat(filename, dev->name, sizeof(filename));
|
strncat(filename, dev->name, sizeof(filename));
|
||||||
@ -109,31 +141,9 @@ static int create_node(struct udevice *dev)
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* create subdirectories if requested */
|
/* create parent directories if needed */
|
||||||
if (strchr(dev->name, '/')) {
|
if (strrchr(dev->name, '/'))
|
||||||
char path[255];
|
create_path(filename);
|
||||||
char *pos;
|
|
||||||
struct stat stats;
|
|
||||||
|
|
||||||
strncpy(path, filename, sizeof(path));
|
|
||||||
pos = strchr(path+1, '/');
|
|
||||||
while (1) {
|
|
||||||
pos = strchr(pos+1, '/');
|
|
||||||
if (pos == NULL)
|
|
||||||
break;
|
|
||||||
*pos = 0x00;
|
|
||||||
if (stat(path, &stats)) {
|
|
||||||
retval = mkdir(path, 0755);
|
|
||||||
if (retval) {
|
|
||||||
dbg("mkdir(%s) failed with error '%s'",
|
|
||||||
path, strerror(errno));
|
|
||||||
return retval;
|
|
||||||
}
|
|
||||||
dbg("created '%s'", path);
|
|
||||||
}
|
|
||||||
*pos = '/';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dbg("mknod(%s, %#o, %u, %u)", filename, dev->mode, dev->major, dev->minor);
|
dbg("mknod(%s, %#o, %u, %u)", filename, dev->mode, dev->major, dev->minor);
|
||||||
retval = mknod(filename, dev->mode, res);
|
retval = mknod(filename, dev->mode, res);
|
||||||
@ -179,8 +189,43 @@ static int create_node(struct udevice *dev)
|
|||||||
dbg("chown(%s, %u, %u)", filename, uid, gid);
|
dbg("chown(%s, %u, %u)", filename, uid, gid);
|
||||||
retval = chown(filename, uid, gid);
|
retval = chown(filename, uid, gid);
|
||||||
if (retval)
|
if (retval)
|
||||||
dbg("chown(%s, %u, %u) failed with error '%s'", filename,
|
dbg("chown(%s, %u, %u) failed with error '%s'",
|
||||||
uid, gid, strerror(errno));
|
filename, uid, gid, strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* create symlink if requested */
|
||||||
|
if (*dev->symlink) {
|
||||||
|
strncpy(filename, udev_root, sizeof(filename));
|
||||||
|
strncat(filename, dev->symlink, sizeof(filename));
|
||||||
|
dbg("symlink '%s' to node '%s' requested", filename, dev->name);
|
||||||
|
if (strrchr(dev->symlink, '/'))
|
||||||
|
create_path(filename);
|
||||||
|
|
||||||
|
/* optimize relative link */
|
||||||
|
linktarget[0] = '\0';
|
||||||
|
i = 0;
|
||||||
|
tail = 0;
|
||||||
|
while ((dev->name[i] == dev->symlink[i]) && dev->name[i]) {
|
||||||
|
if (dev->name[i] == '/')
|
||||||
|
tail = i+1;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
while (dev->symlink[i]) {
|
||||||
|
if (dev->symlink[i] == '/')
|
||||||
|
strcat(linktarget, "../");
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*linktarget == '\0')
|
||||||
|
strcpy(linktarget, "./");
|
||||||
|
strcat(linktarget, &dev->name[tail]);
|
||||||
|
|
||||||
|
dbg("symlink(%s, %s)", linktarget, filename);
|
||||||
|
retval = symlink(linktarget, filename);
|
||||||
|
if (retval)
|
||||||
|
dbg("symlink(%s, %s) failed with error '%s'",
|
||||||
|
linktarget, filename, strerror(errno));
|
||||||
}
|
}
|
||||||
|
|
||||||
return retval;
|
return retval;
|
||||||
|
125
udev-remove.c
125
udev-remove.c
@ -34,47 +34,43 @@
|
|||||||
#include "udevdb.h"
|
#include "udevdb.h"
|
||||||
#include "libsysfs/libsysfs.h"
|
#include "libsysfs/libsysfs.h"
|
||||||
|
|
||||||
|
static int delete_path(char *path)
|
||||||
/*
|
|
||||||
* Look up the sysfs path in the database to see if we have named this device
|
|
||||||
* something different from the kernel name. If we have, us it. If not, use
|
|
||||||
* the default kernel name for lack of anything else to know to do.
|
|
||||||
*/
|
|
||||||
static char *get_name(char *path, int major, int minor)
|
|
||||||
{
|
{
|
||||||
static char name[100];
|
char *pos;
|
||||||
struct udevice *dev;
|
int retval;
|
||||||
char *temp;
|
|
||||||
|
|
||||||
dev = udevdb_get_dev(path);
|
pos = strrchr(path, '/');
|
||||||
if (dev != NULL) {
|
while (1) {
|
||||||
strcpy(name, dev->name);
|
*pos = '\0';
|
||||||
goto exit;
|
pos = strrchr(path, '/');
|
||||||
|
|
||||||
|
/* don't remove the last one */
|
||||||
|
if ((pos == path) || (pos == NULL))
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* remove if empty */
|
||||||
|
retval = rmdir(path);
|
||||||
|
if (retval) {
|
||||||
|
if (errno == ENOTEMPTY)
|
||||||
|
return 0;
|
||||||
|
dbg("rmdir(%s) failed with error '%s'",
|
||||||
|
path, strerror(errno));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
dbg("removed '%s'", path);
|
||||||
}
|
}
|
||||||
|
return 0;
|
||||||
dbg("'%s' not found in database, falling back on default name", path);
|
|
||||||
temp = strrchr(path, '/');
|
|
||||||
if (temp == NULL)
|
|
||||||
return NULL;
|
|
||||||
strncpy(name, &temp[1], sizeof(name));
|
|
||||||
|
|
||||||
exit:
|
|
||||||
dbg("name is '%s'", name);
|
|
||||||
return &name[0];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
static int delete_node(struct udevice *dev)
|
||||||
* We also want to clean up any symlinks that were created in create_node()
|
|
||||||
*/
|
|
||||||
static int delete_node(char *name)
|
|
||||||
{
|
{
|
||||||
char filename[255];
|
char filename[255];
|
||||||
int retval;
|
int retval;
|
||||||
|
|
||||||
strncpy(filename, udev_root, sizeof(filename));
|
strncpy(filename, udev_root, sizeof(filename));
|
||||||
strncat(filename, name, sizeof(filename));
|
strncat(filename, dev->name, sizeof(filename));
|
||||||
|
|
||||||
dbg("unlinking '%s'", filename);
|
dbg("unlinking node '%s'", filename);
|
||||||
retval = unlink(filename);
|
retval = unlink(filename);
|
||||||
if (retval) {
|
if (retval) {
|
||||||
dbg("unlink(%s) failed with error '%s'",
|
dbg("unlink(%s) failed with error '%s'",
|
||||||
@ -83,49 +79,48 @@ static int delete_node(char *name)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* remove subdirectories */
|
/* remove subdirectories */
|
||||||
if (strchr(name, '/')) {
|
if (strchr(dev->name, '/'))
|
||||||
char *pos;
|
delete_path(filename);
|
||||||
|
|
||||||
pos = strrchr(filename, '/');
|
if (*dev->symlink) {
|
||||||
while (1) {
|
strncpy(filename, udev_root, sizeof(filename));
|
||||||
*pos = 0x00;
|
strncat(filename, dev->symlink, sizeof(filename));
|
||||||
pos = strrchr(filename, '/');
|
dbg("unlinking symlink '%s'", filename);
|
||||||
|
retval = unlink(filename);
|
||||||
/* don't remove the last one */
|
if (retval) {
|
||||||
if ((pos == filename) || (pos == NULL))
|
dbg("unlink(%s) failed with error '%s'",
|
||||||
break;
|
filename, strerror(errno));
|
||||||
|
return retval;
|
||||||
/* remove if empty */
|
}
|
||||||
retval = rmdir(filename);
|
if (strchr(dev->symlink, '/')) {
|
||||||
if (retval) {
|
delete_path(filename);
|
||||||
if (errno == ENOTEMPTY)
|
|
||||||
return 0;
|
|
||||||
dbg("rmdir(%s) failed with error '%s'",
|
|
||||||
filename, strerror(errno));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
dbg("removed '%s'", filename);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
int udev_remove_device(char *device, char *subsystem)
|
/*
|
||||||
|
* Look up the sysfs path in the database to see if we have named this device
|
||||||
|
* something different from the kernel name. If we have, us it. If not, use
|
||||||
|
* the default kernel name for lack of anything else to know to do.
|
||||||
|
*/
|
||||||
|
int udev_remove_device(char *path, char *subsystem)
|
||||||
{
|
{
|
||||||
char *name;
|
char name[100];
|
||||||
int retval = 0;
|
struct udevice *dev;
|
||||||
|
char *temp;
|
||||||
|
|
||||||
name = get_name(device, 0, 0);
|
dev = udevdb_get_dev(path);
|
||||||
if (name == NULL) {
|
if (dev == NULL) {
|
||||||
dbg ("get_name failed");
|
dbg("'%s' not found in database, falling back on default name", path);
|
||||||
retval = -ENODEV;
|
temp = strrchr(path, '/');
|
||||||
goto exit;
|
if (temp == NULL)
|
||||||
|
return -ENODEV;
|
||||||
|
strncpy(name, &temp[1], sizeof(name));
|
||||||
}
|
}
|
||||||
|
|
||||||
udevdb_delete_dev(device);
|
dbg("name is '%s'", dev->name);
|
||||||
|
udevdb_delete_dev(path);
|
||||||
return delete_node(name);
|
return delete_node(dev);
|
||||||
|
|
||||||
exit:
|
|
||||||
return retval;
|
|
||||||
}
|
}
|
||||||
|
1
udev.h
1
udev.h
@ -64,6 +64,7 @@ struct udevice {
|
|||||||
int major;
|
int major;
|
||||||
int minor;
|
int minor;
|
||||||
mode_t mode;
|
mode_t mode;
|
||||||
|
char symlink[NAME_SIZE];
|
||||||
|
|
||||||
/* fields that help us in building strings */
|
/* fields that help us in building strings */
|
||||||
unsigned char bus_id[SYSFS_NAME_LEN];
|
unsigned char bus_id[SYSFS_NAME_LEN];
|
||||||
|
Reference in New Issue
Block a user