changeset 1460:94f7ec50ef50 draft

Debugging pass on mount. Not quite done yet, but the basics seem to work now.
author Rob Landley <rob@landley.net>
date Sun, 07 Sep 2014 14:42:51 -0500
parents 240bd13db413
children 67c1402e9723
files lib/getmountlist.c toys/pending/mount.c
diffstat 2 files changed, 39 insertions(+), 36 deletions(-) [+]
line wrap: on
line diff
--- a/lib/getmountlist.c	Sat Sep 06 13:24:58 2014 -0500
+++ b/lib/getmountlist.c	Sun Sep 07 14:42:51 2014 -0500
@@ -31,7 +31,7 @@
 {
   char *start = *list, *end;
 
-  if (!*list) return 0;
+  if (!*list || !**list) return 0;
 
   if (!(end = strchr(*list, ','))) {
     *len = strlen(*list);
@@ -76,8 +76,10 @@
 
     if (!s) break;
     no = 2*(*s == 'n' && s[1] == 'o');
-    if (optlen == len+no && !strcmp(opt, s+no)) got = !no;
-    if (clean) memmove(s, optlist, strlen(optlist)+1);
+    if (optlen == len+no && !strcmp(opt, s+no)) {
+      got = !no;
+      if (clean) memmove(s, optlist, strlen(optlist)+1);
+    }
   }
 
   return got;
--- a/toys/pending/mount.c	Sat Sep 06 13:24:58 2014 -0500
+++ b/toys/pending/mount.c	Sun Sep 07 14:42:51 2014 -0500
@@ -139,7 +139,7 @@
   }
 
   // Autodetect bind mount or filesystem type
-  if (!type || !strcmp(type, "auto")) {
+  if (!(flags & MS_MOVE) && (!type || !strcmp(type, "auto"))) {
     struct stat stdev, stdir;
 
     // file on file or dir on dir is a --bind mount.
@@ -182,43 +182,43 @@
     }
     free(buf);
 
-    if (rc) {
-      // Trying to autodetect loop mounts like bind mounts above (file on dir)
-      // isn't good enough because "mount -t ext2 fs.img dir" is valid, but if
-      // you _do_ accept loop mounts with -t how do you tell "-t cifs" isn't
-      // looking for a block device if it's not in /proc/filesystems yet
-      // because the module that won't be loaded until you try the mount, and
-      // if you can't then DEVICE existing as a file would cause a false
-      // positive loopback mount (so "touch servername" becomes a potential
-      // denial of service attack...)
-      //
-      // Solution: try the mount, let the kernel tell us it wanted a block
-      // device, then do the loopback setup and retry the mount.
+    if (!rc) break;
 
-      if (fp && errno == EINVAL) continue;;
+    // Trying to autodetect loop mounts like bind mounts above (file on dir)
+    // isn't good enough because "mount -t ext2 fs.img dir" is valid, but if
+    // you _do_ accept loop mounts with -t how do you tell "-t cifs" isn't
+    // looking for a block device if it's not in /proc/filesystems yet
+    // because the module that won't be loaded until you try the mount, and
+    // if you can't then DEVICE existing as a file would cause a false
+    // positive loopback mount (so "touch servername" becomes a potential
+    // denial of service attack...)
+    //
+    // Solution: try the mount, let the kernel tell us it wanted a block
+    // device, then do the loopback setup and retry the mount.
+
+    if (fp && errno == EINVAL) continue;
 
-      if (errno == ENOTBLK) {
-        char *losetup[] = {"losetup", "-fs", dev, 0};
-        int pipes[2], len;
-        pid_t pid;
+    if (errno == ENOTBLK) {
+      char *losetup[] = {"losetup", "-fs", dev, 0};
+      int pipes[2], len;
+      pid_t pid;
 
-        if (flags & MS_RDONLY) losetup[1] = "-fsr";
-        pid = xpopen(losetup, pipes);
-        len = readall(pipes[1], toybuf, sizeof(toybuf)-1);
-        rc = xpclose(pid, pipes);
-        if (!rc && len > 1) {
-          if (toybuf[len-1] == '\n') --len;
-          toybuf[len] = 0;
-          dev = toybuf;
+      if (flags & MS_RDONLY) losetup[1] = "-fsr";
+      pid = xpopen(losetup, pipes);
+      len = readall(pipes[1], toybuf, sizeof(toybuf)-1);
+      rc = xpclose(pid, pipes);
+      if (!rc && len > 1) {
+        if (toybuf[len-1] == '\n') --len;
+        toybuf[len] = 0;
+        dev = toybuf;
 
-          continue;
-        } else error_msg("losetup failed %d", rc);
-      } else perror_msg("'%s'->'%s'", dev, dir);
-    }
+        continue;
+      } else error_msg("losetup failed %d", rc);
+    } else perror_msg("'%s'->'%s'", dev, dir);
+
+    break;
   }
   if (fp) fclose(fp);
-
-  if (rc) perror_msg("'%s' on '%s'", dev, dir);
 }
 
 void mount_main(void)
@@ -261,8 +261,9 @@
 
   // Do we need to do an /etc/fstab trawl?
   // This covers -a, -o remount, one argument, all user mounts
-  if ((toys.optflags & FLAG_a) || remount || !dir || getuid()) {
+  if ((toys.optflags & FLAG_a) || (dev && (!dir || getuid() || remount))) {
     if (!remount) mtl = xgetmountlist("/etc/fstab");
+
     for (mm = remount ? remount : mtl; mm; mm = (remount ? mm->prev : mm->next))
     {
       int aflags, noauto, len;