The kernel can't handle umount /dev/hdc, we have to do it through mtab,

except that we still have to work when there is no mtab.

Oh, and while we're at it, take advantage of the fact that modern processors
avoid branches via conditional assignment where possible.  ("x = a ? b : c;"
turns into "x = c; if (a) x = b;" because that way there's no branch to
potentially mispredict and thus never a bubble in the pipeline.  The if(a)
turns into an assembly test followed by a conditional assignment (rather
than a conditional jump).)  So since the compiler is going to do that _anyway_,
we might as well take advantage of it to produce a slightly smaller binary.

So there.
1_4_stable
Rob Landley 2006-08-17 19:07:20 +00:00
parent eb28ce4644
commit 4470b74e29
1 changed files with 14 additions and 8 deletions

View File

@ -78,6 +78,7 @@ int umount_main(int argc, char **argv)
// Loop through everything we're supposed to umount, and do so. // Loop through everything we're supposed to umount, and do so.
for (;;) { for (;;) {
int curstat; int curstat;
char *zapit = *argv;
// Do we already know what to umount this time through the loop? // Do we already know what to umount this time through the loop?
if (m) safe_strncpy(path, m->dir, PATH_MAX); if (m) safe_strncpy(path, m->dir, PATH_MAX);
@ -86,32 +87,37 @@ int umount_main(int argc, char **argv)
// Get next command line argument (and look it up in mtab list) // Get next command line argument (and look it up in mtab list)
else if (!argc--) break; else if (!argc--) break;
else { else {
realpath(*argv++, path); argv++;
realpath(zapit, path);
for (m = mtl; m; m = m->next) for (m = mtl; m; m = m->next)
if (!strcmp(path, m->dir) || !strcmp(path, m->device)) if (!strcmp(path, m->dir) || !strcmp(path, m->device))
break; break;
} }
// If we couldn't find this sucker in /etc/mtab, punt by passing our
// command line argument straight to the umount syscall. Otherwise,
// umount the directory even if we were given the block device.
if (m) zapit = m->dir;
// Let's ask the thing nicely to unmount. // Let's ask the thing nicely to unmount.
curstat = umount(path); curstat = umount(zapit);
// Force the unmount, if necessary. // Force the unmount, if necessary.
if (curstat && doForce) { if (curstat && doForce) {
curstat = umount2(path, doForce); curstat = umount2(zapit, doForce);
if (curstat) if (curstat)
bb_error_msg_and_die("forced umount of %s failed!", path); bb_error_msg_and_die("forced umount of %s failed!", zapit);
} }
// If still can't umount, maybe remount read-only? // If still can't umount, maybe remount read-only?
if (curstat && (opt & OPT_REMOUNT) && errno == EBUSY && m) { if (curstat && (opt & OPT_REMOUNT) && errno == EBUSY && m) {
curstat = mount(m->device, path, NULL, MS_REMOUNT|MS_RDONLY, NULL); curstat = mount(m->device, zapit, NULL, MS_REMOUNT|MS_RDONLY, NULL);
bb_error_msg(curstat ? "Cannot remount %s read-only" : bb_error_msg(curstat ? "Cannot remount %s read-only" :
"%s busy - remounted read-only", m->device); "%s busy - remounted read-only", m->device);
} }
if (curstat) { if (curstat) {
status = EXIT_FAILURE; status = EXIT_FAILURE;
bb_perror_msg("Couldn't umount %s", path); bb_perror_msg("Couldn't umount %s", zapit);
} else { } else {
/* De-allocate the loop device. This ioctl should be ignored on /* De-allocate the loop device. This ioctl should be ignored on
* any non-loop block devices. */ * any non-loop block devices. */
@ -121,9 +127,9 @@ int umount_main(int argc, char **argv)
erase_mtab(m->dir); erase_mtab(m->dir);
} }
// Find next matching mtab entry for -a or umount /dev // Find next matching mtab entry for -a or umount /dev
// Note this means that "umount /dev/blah" will unmount all instances
// of /dev/blah, not just the most recent.
while (m && (m = m->next)) while (m && (m = m->next))
if ((opt & OPT_ALL) || !strcmp(path,m->device)) if ((opt & OPT_ALL) || !strcmp(path,m->device))
break; break;