247 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  GRUB  --  GRand Unified Bootloader
 | |
|  *  Copyright (C) 1999,2000,2001,2002,2003,2006,2007,2008,2009,2010,2011,2012,2013  Free Software Foundation, Inc.
 | |
|  *
 | |
|  *  GRUB is free software: you can redistribute it and/or modify
 | |
|  *  it under the terms of the GNU General Public License as published by
 | |
|  *  the Free Software Foundation, either version 3 of the License, or
 | |
|  *  (at your option) any later version.
 | |
|  *
 | |
|  *  GRUB is distributed in the hope that it will be useful,
 | |
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  *  GNU General Public License for more details.
 | |
|  *
 | |
|  *  You should have received a copy of the GNU General Public License
 | |
|  *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include <config-util.h>
 | |
| #include <config.h>
 | |
| 
 | |
| #include <sys/stat.h>
 | |
| #include <sys/types.h>
 | |
| #include <assert.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| #include <string.h>
 | |
| #include <dirent.h>
 | |
| #include <errno.h>
 | |
| #include <error.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdint.h>
 | |
| #ifdef HAVE_LIMITS_H
 | |
| #include <limits.h>
 | |
| #endif
 | |
| 
 | |
| #include <grub/types.h>
 | |
| 
 | |
| #include <grub/util/misc.h>
 | |
| 
 | |
| #include <grub/mm.h>
 | |
| #include <grub/misc.h>
 | |
| #include <grub/emu/misc.h>
 | |
| #include <grub/emu/hostdisk.h>
 | |
| #include <grub/emu/getroot.h>
 | |
| 
 | |
| #include <sys/wait.h>
 | |
| 
 | |
| #include <hurd.h>
 | |
| #include <hurd/lookup.h>
 | |
| #include <hurd/fs.h>
 | |
| #include <sys/mman.h>
 | |
| 
 | |
| static char *
 | |
| grub_util_find_hurd_root_device (const char *path)
 | |
| {
 | |
|   file_t file;
 | |
|   error_t err;
 | |
|   char *argz = NULL, *name = NULL, *ret;
 | |
|   size_t argz_len = 0;
 | |
|   int i;
 | |
| 
 | |
|   file = file_name_lookup (path, 0, 0);
 | |
|   if (file == MACH_PORT_NULL)
 | |
|     /* TRANSLATORS: The first %s is the file being looked at, the second %s is
 | |
|        the error message.  */
 | |
|     grub_util_error (_("cannot open `%s': %s"), path, strerror (errno));
 | |
| 
 | |
|   /* This returns catenated 0-terminated strings.  */
 | |
|   err = file_get_fs_options (file, &argz, &argz_len);
 | |
|   if (err)
 | |
|     /* TRANSLATORS: On GNU/Hurd, a "translator" is similar to a filesystem
 | |
|        mount, but handled by a userland daemon, whose invocation command line
 | |
|        is being fetched here.  First %s is the file being looked at (for which
 | |
|        we are fetching the "translator" command line), second %s is the error
 | |
|        message.
 | |
|        */
 | |
|     grub_util_error (_("cannot get translator command line "
 | |
|                        "for path `%s': %s"), path, strerror(err));
 | |
|   if (argz_len == 0)
 | |
|     grub_util_error (_("translator command line is empty for path `%s'"), path);
 | |
| 
 | |
|   /* Make sure the string is terminated.  */
 | |
|   argz[argz_len-1] = 0;
 | |
| 
 | |
|   /* Skip first word (translator path) and options.  */
 | |
|   for (i = strlen (argz) + 1; i < argz_len; i += strlen (argz + i) + 1)
 | |
|     {
 | |
|       if (argz[i] != '-')
 | |
|         {
 | |
|           /* Non-option.  Only accept one, assumed to be the FS path.  */
 | |
|           /* XXX: this should be replaced by an RPC to the translator.  */
 | |
|           if (name)
 | |
|             /* TRANSLATORS: we expect to get something like
 | |
|                /hurd/foobar --option1 --option2=baz /dev/something
 | |
|              */
 | |
|             grub_util_error (_("translator `%s' for path `%s' has several "
 | |
|                                "non-option words, at least `%s' and `%s'"),
 | |
|                                argz, path, name, argz + i);
 | |
|           name = argz + i;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   if (!name)
 | |
|     /* TRANSLATORS: we expect to get something like
 | |
|        /hurd/foobar --option1 --option2=baz /dev/something
 | |
|      */
 | |
|     grub_util_error (_("translator `%s' for path `%s' is given only options, "
 | |
|                        "cannot find device part"), argz, path);
 | |
| 
 | |
|   if (strncmp (name, "device:", sizeof ("device:") - 1) == 0)
 | |
|     {
 | |
|       char *dev_name = name + sizeof ("device:") - 1;
 | |
|       size_t size = sizeof ("/dev/") - 1 + strlen (dev_name) + 1;
 | |
|       char *next;
 | |
|       ret = malloc (size);
 | |
|       next = stpncpy (ret, "/dev/", size);
 | |
|       stpncpy (next, dev_name, size - (next - ret));
 | |
|     }
 | |
|   else if (!strncmp (name, "file:", sizeof ("file:") - 1))
 | |
|     ret = strdup (name + sizeof ("file:") - 1);
 | |
|   else
 | |
|     ret = strdup (name);
 | |
| 
 | |
|   munmap (argz, argz_len);
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| static int
 | |
| is_fulldisk (const char *child, const char *parent)
 | |
| {
 | |
|   if (strcmp (parent, child) == 0)
 | |
|     return 1;
 | |
|   if (strncmp (parent, "/dev/", sizeof ("/dev/") - 1) == 0
 | |
|       && child[0] !=0 && strcmp (parent + sizeof ("/dev/") - 1, child) == 0)
 | |
|     return 1;
 | |
|   if (strncmp (child, "/dev/", sizeof ("/dev/") - 1) == 0
 | |
|       && parent[0] != 0 && strcmp (child + sizeof ("/dev/") - 1, parent) == 0)
 | |
|     return 1;
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| char *
 | |
| grub_util_part_to_disk (const char *os_dev,
 | |
| 			struct stat *st,
 | |
| 			int *is_part)
 | |
| {
 | |
|   char *path;
 | |
|   grub_disk_addr_t offset;
 | |
|   char *p;
 | |
| 
 | |
|   if (! S_ISBLK (st->st_mode))
 | |
|     {
 | |
|       *is_part = 0;
 | |
|       return xstrdup (os_dev);
 | |
|     }
 | |
| 
 | |
|   if (!grub_util_hurd_get_disk_info (os_dev, NULL, &offset, NULL, &path))
 | |
|     return xstrdup (os_dev);
 | |
| 
 | |
|   /* Some versions of Hurd use badly glued Linux code to handle partitions
 | |
|      resulting in partitions being promoted to disks.  */
 | |
|   if (path && !(offset == 0 && is_fulldisk (path, os_dev)
 | |
| 		&& (strncmp ("/dev/sd", os_dev, 7) == 0
 | |
| 		    || strncmp ("/dev/hd", os_dev, 7) == 0)))
 | |
|     {
 | |
|       *is_part = !is_fulldisk (path, os_dev);
 | |
|       if (path[0] != '/')
 | |
| 	{
 | |
| 	  char *n = xasprintf ("/dev/%s", path);
 | |
| 	  free (path);
 | |
| 	  path = n;
 | |
| 	}
 | |
|       return path;
 | |
|     }
 | |
|   free (path);
 | |
| 
 | |
|   path = xstrdup (os_dev);
 | |
| 
 | |
|   p = strchr (path + 7, 's');
 | |
|   if (p)
 | |
|     {
 | |
|       *is_part = 1;
 | |
|       *p = '\0';
 | |
|     }
 | |
|   return path;
 | |
| }
 | |
| 
 | |
| enum grub_dev_abstraction_types
 | |
| grub_util_get_dev_abstraction_os (const char *os_dev __attribute__((unused)))
 | |
| {
 | |
|   return GRUB_DEV_ABSTRACTION_NONE;
 | |
| }
 | |
| 
 | |
| int
 | |
| grub_util_pull_device_os (const char *os_dev __attribute__ ((unused)),
 | |
| 			  enum grub_dev_abstraction_types ab __attribute__ ((unused)))
 | |
| {
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| char *
 | |
| grub_util_get_grub_dev_os (const char *os_dev __attribute__ ((unused)))
 | |
| {
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| grub_disk_addr_t
 | |
| grub_util_find_partition_start_os (const char *dev)
 | |
| {
 | |
|   grub_uint32_t secsize;
 | |
|   grub_disk_addr_t offset;
 | |
|   char *path;
 | |
|   if (!grub_util_hurd_get_disk_info (dev, &secsize, &offset, NULL, &path))
 | |
|     return 0;
 | |
|   if (path && !(offset == 0 && is_fulldisk (path, dev)
 | |
| 		&& (strncmp ("/dev/sd", dev, 7) == 0
 | |
| 		    || strncmp ("/dev/hd", dev, 7) == 0)))
 | |
|     {
 | |
|       free (path);
 | |
|       return (secsize / 512) * offset;
 | |
|     }
 | |
|   free (path);
 | |
|   return -1;
 | |
| }
 | |
| 
 | |
| char **
 | |
| grub_guess_root_devices (const char *dir)
 | |
| {
 | |
|   char **os_dev = NULL;
 | |
| 
 | |
|   os_dev = xmalloc (2 * sizeof (os_dev[0]));
 | |
| 
 | |
|   /* GNU/Hurd specific function.  */
 | |
|   os_dev[0] = grub_util_find_hurd_root_device (dir);
 | |
| 
 | |
|   if (!os_dev[0])
 | |
|     {
 | |
|       free (os_dev);
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|   os_dev[1] = 0;
 | |
| 
 | |
|   return os_dev;
 | |
| }
 |