]> nv-tegra.nvidia Code Review - linux-2.6.git/blobdiff - init/do_mounts.c
Merge branch 'fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/jejb/parisc-2.6
[linux-2.6.git] / init / do_mounts.c
index b27c11064409a35812d9d35ba359e06f6d8c0ffb..c0851a8e030cbcf38701f77a5d79e253d81e0327 100644 (file)
@@ -7,7 +7,15 @@
 #include <linux/root_dev.h>
 #include <linux/security.h>
 #include <linux/delay.h>
+#include <linux/genhd.h>
 #include <linux/mount.h>
+#include <linux/device.h>
+#include <linux/init.h>
+#include <linux/fs.h>
+#include <linux/initrd.h>
+#include <linux/async.h>
+#include <linux/fs_struct.h>
+#include <linux/slab.h>
 
 #include <linux/nfs_fs.h>
 #include <linux/nfs_fs_sb.h>
 
 #include "do_mounts.h"
 
-extern int get_filesystem_list(char * buf);
-
 int __initdata rd_doload;      /* 1 = load RAM disk, 0 = don't load */
 
-int root_mountflags = MS_RDONLY | MS_VERBOSE;
-char * __initdata root_device_name;
+int root_mountflags = MS_RDONLY | MS_SILENT;
+static char * __initdata root_device_name;
 static char __initdata saved_root_name[64];
+static int __initdata root_wait;
 
-/* this is initialized in init/main.c */
 dev_t ROOT_DEV;
 
 static int __init load_ramdisk(char *str)
@@ -52,68 +58,61 @@ static int __init readwrite(char *str)
 __setup("ro", readonly);
 __setup("rw", readwrite);
 
-static dev_t try_name(char *name, int part)
+#ifdef CONFIG_BLOCK
+/**
+ * match_dev_by_uuid - callback for finding a partition using its uuid
+ * @dev:       device passed in by the caller
+ * @data:      opaque pointer to a 36 byte char array with a UUID
+ *
+ * Returns 1 if the device matches, and 0 otherwise.
+ */
+static int match_dev_by_uuid(struct device *dev, void *data)
 {
-       char path[64];
-       char buf[32];
-       int range;
-       dev_t res;
-       char *s;
-       int len;
-       int fd;
-       unsigned int maj, min;
+       u8 *uuid = data;
+       struct hd_struct *part = dev_to_part(dev);
 
-       /* read device number from .../dev */
+       if (!part->info)
+               goto no_match;
 
-       sprintf(path, "/sys/block/%s/dev", name);
-       fd = sys_open(path, 0, 0);
-       if (fd < 0)
-               goto fail;
-       len = sys_read(fd, buf, 32);
-       sys_close(fd);
-       if (len <= 0 || len == 32 || buf[len - 1] != '\n')
-               goto fail;
-       buf[len - 1] = '\0';
-       if (sscanf(buf, "%u:%u", &maj, &min) == 2) {
-               /*
-                * Try the %u:%u format -- see print_dev_t()
-                */
-               res = MKDEV(maj, min);
-               if (maj != MAJOR(res) || min != MINOR(res))
-                       goto fail;
-       } else {
-               /*
-                * Nope.  Try old-style "0321"
-                */
-               res = new_decode_dev(simple_strtoul(buf, &s, 16));
-               if (*s)
-                       goto fail;
-       }
+       if (memcmp(uuid, part->info->uuid, sizeof(part->info->uuid)))
+                       goto no_match;
 
-       /* if it's there and we are not looking for a partition - that's it */
-       if (!part)
-               return res;
+       return 1;
+no_match:
+       return 0;
+}
 
-       /* otherwise read range from .../range */
-       sprintf(path, "/sys/block/%s/range", name);
-       fd = sys_open(path, 0, 0);
-       if (fd < 0)
-               goto fail;
-       len = sys_read(fd, buf, 32);
-       sys_close(fd);
-       if (len <= 0 || len == 32 || buf[len - 1] != '\n')
-               goto fail;
-       buf[len - 1] = '\0';
-       range = simple_strtoul(buf, &s, 10);
-       if (*s)
-               goto fail;
 
-       /* if partition is within range - we got it */
-       if (part < range)
-               return res + part;
-fail:
-       return 0;
+/**
+ * devt_from_partuuid - looks up the dev_t of a partition by its UUID
+ * @uuid:      36 byte char array containing a hex ascii UUID
+ *
+ * The function will return the first partition which contains a matching
+ * UUID value in its partition_meta_info struct.  This does not search
+ * by filesystem UUIDs.
+ *
+ * Returns the matching dev_t on success or 0 on failure.
+ */
+static dev_t devt_from_partuuid(char *uuid_str)
+{
+       dev_t res = 0;
+       struct device *dev = NULL;
+       u8 uuid[16];
+
+       /* Pack the requested UUID in the expected format. */
+       part_pack_uuid(uuid_str, uuid);
+
+       dev = class_find_device(&block_class, NULL, uuid, &match_dev_by_uuid);
+       if (!dev)
+               goto done;
+
+       res = dev->devt;
+       put_device(dev);
+
+done:
+       return res;
 }
+#endif
 
 /*
  *     Convert a name into device number.  We accept the following variants:
@@ -125,13 +124,13 @@ fail:
  *         of partition - device number of disk plus the partition number
  *     5) /dev/<disk_name>p<decimal> - same as the above, that form is
  *        used when disk name of partitioned disk ends on a digit.
+ *     6) PARTUUID=00112233-4455-6677-8899-AABBCCDDEEFF representing the
+ *        unique id of a partition if the partition table provides it.
  *
- *     If name doesn't have fall into the categories above, we return 0.
- *     Sysfs is used to check if something is a disk name - it has
- *     all known disks under bus/block/devices.  If the disk name
- *     contains slashes, name of sysfs node has them replaced with
- *     bangs.  try_name() does the actual checks, assuming that sysfs
- *     is mounted on rootfs /sys.
+ *     If name doesn't have fall into the categories above, we return (0,0).
+ *     block_class is used to check if something is a disk name. If the disk
+ *     name contains slashes, the device name has them replaced with
+ *     bangs.
  */
 
 dev_t name_to_dev_t(char *name)
@@ -141,10 +140,16 @@ dev_t name_to_dev_t(char *name)
        dev_t res = 0;
        int part;
 
-#ifdef CONFIG_SYSFS
-       int mkdir_err = sys_mkdir("/sys", 0700);
-       if (sys_mount("sysfs", "/sys", "sysfs", 0, NULL) < 0)
-               goto out;
+#ifdef CONFIG_BLOCK
+       if (strncmp(name, "PARTUUID=", 9) == 0) {
+               name += 9;
+               if (strlen(name) != 36)
+                       goto fail;
+               res = devt_from_partuuid(name);
+               if (!res)
+                       goto fail;
+               goto done;
+       }
 #endif
 
        if (strncmp(name, "/dev/", 5) != 0) {
@@ -161,6 +166,7 @@ dev_t name_to_dev_t(char *name)
                }
                goto done;
        }
+
        name += 5;
        res = Root_NFS;
        if (strcmp(name, "nfs") == 0)
@@ -175,35 +181,38 @@ dev_t name_to_dev_t(char *name)
        for (p = s; *p; p++)
                if (*p == '/')
                        *p = '!';
-       res = try_name(s, 0);
+       res = blk_lookup_devt(s, 0);
        if (res)
                goto done;
 
+       /*
+        * try non-existent, but valid partition, which may only exist
+        * after revalidating the disk, like partitioned md devices
+        */
        while (p > s && isdigit(p[-1]))
                p--;
        if (p == s || !*p || *p == '0')
                goto fail;
+
+       /* try disk name without <part number> */
        part = simple_strtoul(p, NULL, 10);
        *p = '\0';
-       res = try_name(s, part);
+       res = blk_lookup_devt(s, part);
        if (res)
                goto done;
 
+       /* try disk name without p<part number> */
        if (p < s + 2 || !isdigit(p[-2]) || p[-1] != 'p')
                goto fail;
        p[-1] = '\0';
-       res = try_name(s, part);
+       res = blk_lookup_devt(s, part);
+       if (res)
+               goto done;
+
+fail:
+       return 0;
 done:
-#ifdef CONFIG_SYSFS
-       sys_umount("/sys", 0);
-out:
-       if (!mkdir_err)
-               sys_rmdir("/sys");
-#endif
        return res;
-fail:
-       res = 0;
-       goto done;
 }
 
 static int __init root_dev_setup(char *line)
@@ -214,6 +223,16 @@ static int __init root_dev_setup(char *line)
 
 __setup("root=", root_dev_setup);
 
+static int __init rootwait_setup(char *str)
+{
+       if (*str)
+               return 0;
+       root_wait = 1;
+       return 1;
+}
+
+__setup("rootwait", rootwait_setup);
+
 static char * __initdata root_mount_data;
 static int __init root_data_setup(char *str)
 {
@@ -272,20 +291,26 @@ static int __init do_mount_root(char *name, char *fs, int flags, void *data)
        if (err)
                return err;
 
-       sys_chdir("/root");
-       ROOT_DEV = current->fs->pwdmnt->mnt_sb->s_dev;
-       printk("VFS: Mounted root (%s filesystem)%s.\n",
-              current->fs->pwdmnt->mnt_sb->s_type->name,
-              current->fs->pwdmnt->mnt_sb->s_flags & MS_RDONLY ? 
-              " readonly" : "");
+       sys_chdir((const char __user __force *)"/root");
+       ROOT_DEV = current->fs->pwd.mnt->mnt_sb->s_dev;
+       printk(KERN_INFO
+              "VFS: Mounted root (%s filesystem)%s on device %u:%u.\n",
+              current->fs->pwd.mnt->mnt_sb->s_type->name,
+              current->fs->pwd.mnt->mnt_sb->s_flags & MS_RDONLY ?
+              " readonly" : "", MAJOR(ROOT_DEV), MINOR(ROOT_DEV));
        return 0;
 }
 
 void __init mount_block_root(char *name, int flags)
 {
-       char *fs_names = __getname();
+       char *fs_names = __getname_gfp(GFP_KERNEL
+               | __GFP_NOTRACK_FALSE_POSITIVE);
        char *p;
+#ifdef CONFIG_BLOCK
        char b[BDEVNAME_SIZE];
+#else
+       const char *b = name;
+#endif
 
        get_fs_names(fs_names);
 retry:
@@ -303,15 +328,33 @@ retry:
                /*
                 * Allow the user to distinguish between failed sys_open
                 * and bad superblock on root device.
+                * and give them a list of the available devices
                 */
+#ifdef CONFIG_BLOCK
                __bdevname(ROOT_DEV, b);
+#endif
                printk("VFS: Cannot open root device \"%s\" or %s\n",
                                root_device_name, b);
-               printk("Please append a correct \"root=\" boot option\n");
+               printk("Please append a correct \"root=\" boot option; here are the available partitions:\n");
 
+               printk_all_partitions();
+#ifdef CONFIG_DEBUG_BLOCK_EXT_DEVT
+               printk("DEBUG_BLOCK_EXT_DEVT is enabled, you need to specify "
+                      "explicit textual name for \"root=\" boot option.\n");
+#endif
                panic("VFS: Unable to mount root fs on %s", b);
        }
-       panic("VFS: Unable to mount root fs on %s", __bdevname(ROOT_DEV, b));
+
+       printk("List of all partitions:\n");
+       printk_all_partitions();
+       printk("No filesystem could mount root, tried: ");
+       for (p = fs_names; *p; p += strlen(p)+1)
+               printk(" %s", p);
+       printk("\n");
+#ifdef CONFIG_BLOCK
+       __bdevname(ROOT_DEV, b);
+#endif
+       panic("VFS: Unable to mount root fs on %s", b);
 out:
        putname(fs_names);
 }
@@ -319,13 +362,13 @@ out:
 #ifdef CONFIG_ROOT_NFS
 static int __init mount_nfs_root(void)
 {
-       void *data = nfs_root_data();
+       char *root_dev, *root_data;
 
-       create_dev("/dev/root", ROOT_DEV, NULL);
-       if (data &&
-           do_mount_root("/dev/root", "nfs", root_mountflags, data) == 0)
-               return 1;
-       return 0;
+       if (nfs_root_data(&root_dev, &root_data) != 0)
+               return 0;
+       if (do_mount_root(root_dev, "nfs", root_mountflags, root_data) != 0)
+               return 0;
+       return 1;
 }
 #endif
 
@@ -382,8 +425,10 @@ void __init mount_root(void)
                        change_floppy("root floppy");
        }
 #endif
-       create_dev("/dev/root", ROOT_DEV, root_device_name);
+#ifdef CONFIG_BLOCK
+       create_dev("/dev/root", ROOT_DEV);
        mount_block_root("/dev/root", root_mountflags);
+#endif
 }
 
 /*
@@ -393,37 +438,56 @@ void __init prepare_namespace(void)
 {
        int is_floppy;
 
-       mount_devfs();
-
        if (root_delay) {
                printk(KERN_INFO "Waiting %dsec before mounting root device...\n",
                       root_delay);
                ssleep(root_delay);
        }
 
+       /*
+        * wait for the known devices to complete their probing
+        *
+        * Note: this is a potential source of long boot delays.
+        * For example, it is not atypical to wait 5 seconds here
+        * for the touchpad of a laptop to initialize.
+        */
+       wait_for_device_probe();
+
        md_run_setup();
 
        if (saved_root_name[0]) {
                root_device_name = saved_root_name;
+               if (!strncmp(root_device_name, "mtd", 3) ||
+                   !strncmp(root_device_name, "ubi", 3)) {
+                       mount_block_root(root_device_name, root_mountflags);
+                       goto out;
+               }
                ROOT_DEV = name_to_dev_t(root_device_name);
                if (strncmp(root_device_name, "/dev/", 5) == 0)
                        root_device_name += 5;
        }
 
-       is_floppy = MAJOR(ROOT_DEV) == FLOPPY_MAJOR;
-
        if (initrd_load())
                goto out;
 
+       /* wait for any asynchronous scanning to complete */
+       if ((ROOT_DEV == 0) && root_wait) {
+               printk(KERN_INFO "Waiting for root device %s...\n",
+                       saved_root_name);
+               while (driver_probe_done() != 0 ||
+                       (ROOT_DEV = name_to_dev_t(saved_root_name)) == 0)
+                       msleep(100);
+               async_synchronize_full();
+       }
+
+       is_floppy = MAJOR(ROOT_DEV) == FLOPPY_MAJOR;
+
        if (is_floppy && rd_doload && rd_load_disk(0))
                ROOT_DEV = Root_RAM0;
 
        mount_root();
 out:
-       umount_devfs("/dev");
+       devtmpfs_mount("dev");
        sys_mount(".", "/", NULL, MS_MOVE, NULL);
-       sys_chroot(".");
-       security_sb_post_mountroot();
-       mount_devfs_fs ();
+       sys_chroot((const char __user __force *)".");
 }
-