328 lines
		
	
	
	
		
			8.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			328 lines
		
	
	
	
		
			8.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* grub-mkimage.c - make a bootable image */
 | ||
| /*
 | ||
|  *  GRUB  --  GRand Unified Bootloader
 | ||
|  *  Copyright (C) 2002,2003,2004,2005,2006,2007,2008,2009,2010  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.h>
 | ||
| #include <grub/types.h>
 | ||
| #include <grub/elf.h>
 | ||
| #include <grub/aout.h>
 | ||
| #include <grub/i18n.h>
 | ||
| #include <grub/kernel.h>
 | ||
| #include <grub/disk.h>
 | ||
| #include <grub/emu/misc.h>
 | ||
| #include <grub/util/misc.h>
 | ||
| #include <grub/util/resolve.h>
 | ||
| #include <grub/misc.h>
 | ||
| #include <grub/offsets.h>
 | ||
| #include <grub/crypto.h>
 | ||
| #include <grub/dl.h>
 | ||
| #include <time.h>
 | ||
| #include <multiboot.h>
 | ||
| 
 | ||
| #include <stdio.h>
 | ||
| #include <unistd.h>
 | ||
| #include <string.h>
 | ||
| #include <stdlib.h>
 | ||
| #include <assert.h>
 | ||
| #include <grub/efi/pe32.h>
 | ||
| #include <grub/uboot/image.h>
 | ||
| #include <grub/arm/reloc.h>
 | ||
| #include <grub/ia64/reloc.h>
 | ||
| #include <grub/osdep/hostfile.h>
 | ||
| #include <grub/util/install.h>
 | ||
| #include <grub/emu/config.h>
 | ||
| 
 | ||
| #define _GNU_SOURCE	1
 | ||
| 
 | ||
| #pragma GCC diagnostic ignored "-Wmissing-prototypes"
 | ||
| #pragma GCC diagnostic ignored "-Wmissing-declarations"
 | ||
| #include <argp.h>
 | ||
| #pragma GCC diagnostic error "-Wmissing-prototypes"
 | ||
| #pragma GCC diagnostic error "-Wmissing-declarations"
 | ||
| 
 | ||
| 
 | ||
| #include "progname.h"
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| static struct argp_option options[] = {
 | ||
|   {"directory",  'd', N_("DIR"), 0,
 | ||
|    /* TRANSLATORS: platform here isn't identifier. It can be translated.  */
 | ||
|    N_("use images and modules under DIR [default=%s/<platform>]"), 0},
 | ||
|   {"prefix",  'p', N_("DIR"), 0, N_("set prefix directory"), 0},
 | ||
|   {"memdisk",  'm', N_("FILE"), 0,
 | ||
|    /* TRANSLATORS: "memdisk" here isn't an identifier, it can be translated.
 | ||
|     "embed" is a verb (command description).  "*/
 | ||
|    N_("embed FILE as a memdisk image\n"
 | ||
|       "Implies `-p (memdisk)/boot/grub' and overrides any prefix supplied previously,"
 | ||
|       " but the prefix itself can be overridden by later options"), 0},
 | ||
|   {"dtb",  'D', N_("FILE"), 0, N_("embed FILE as a device tree (DTB)\n"), 0},
 | ||
|    /* TRANSLATORS: "embed" is a verb (command description).  "*/
 | ||
|   {"config",   'c', N_("FILE"), 0, N_("embed FILE as an early config"), 0},
 | ||
|    /* TRANSLATORS: "embed" is a verb (command description).  "*/
 | ||
|   {"pubkey",   'k', N_("FILE"), 0, N_("embed FILE as public key for signature checking"), 0},
 | ||
|   /* TRANSLATORS: NOTE is a name of segment.  */
 | ||
|   {"note",   'n', 0, 0, N_("add NOTE segment for CHRP IEEE1275"), 0},
 | ||
|   {"output",  'o', N_("FILE"), 0, N_("output a generated image to FILE [default=stdout]"), 0},
 | ||
|   {"format",  'O', N_("FORMAT"), 0, 0, 0},
 | ||
|   {"compression",  'C', "(xz|none|auto)", 0, N_("choose the compression to use for core image"), 0},
 | ||
|   {"verbose",     'v', 0,      0, N_("print verbose messages."), 0},
 | ||
|   { 0, 0, 0, 0, 0, 0 }
 | ||
| };
 | ||
| 
 | ||
| #pragma GCC diagnostic ignored "-Wformat-nonliteral"
 | ||
| 
 | ||
| static char *
 | ||
| help_filter (int key, const char *text, void *input __attribute__ ((unused)))
 | ||
| {
 | ||
|   switch (key)
 | ||
|     {
 | ||
|     case 'd':
 | ||
|       return xasprintf (text, grub_util_get_pkglibdir ());
 | ||
|     case 'O':
 | ||
|       {
 | ||
| 	char *formats = grub_install_get_image_targets_string (), *ret;
 | ||
| 	ret = xasprintf ("%s\n%s %s", _("generate an image in FORMAT"),
 | ||
| 			 _("available formats:"), formats);
 | ||
| 	free (formats);
 | ||
| 	return ret;
 | ||
|       }
 | ||
|     default:
 | ||
|       return (char *) text;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| #pragma GCC diagnostic error "-Wformat-nonliteral"
 | ||
| 
 | ||
| struct arguments
 | ||
| {
 | ||
|   size_t nmodules;
 | ||
|   size_t modules_max;
 | ||
|   char **modules;
 | ||
|   char *output;
 | ||
|   char *dir;
 | ||
|   char *prefix;
 | ||
|   char *memdisk;
 | ||
|   char *dtb;
 | ||
|   char **pubkeys;
 | ||
|   size_t npubkeys;
 | ||
|   char *font;
 | ||
|   char *config;
 | ||
|   int note;
 | ||
|   const struct grub_install_image_target_desc *image_target;
 | ||
|   grub_compression_t comp;
 | ||
| };
 | ||
| 
 | ||
| static error_t
 | ||
| argp_parser (int key, char *arg, struct argp_state *state)
 | ||
| {
 | ||
|   /* Get the input argument from argp_parse, which we
 | ||
|      know is a pointer to our arguments structure. */
 | ||
|   struct arguments *arguments = state->input;
 | ||
| 
 | ||
|   switch (key)
 | ||
|     {
 | ||
|     case 'o':
 | ||
|       if (arguments->output)
 | ||
| 	free (arguments->output);
 | ||
| 
 | ||
|       arguments->output = xstrdup (arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'O':
 | ||
|       {
 | ||
| 	arguments->image_target = grub_install_get_image_target (arg);
 | ||
| 	if (!arguments->image_target)
 | ||
| 	  {
 | ||
| 	    printf (_("unknown target format %s\n"), arg);
 | ||
| 	    argp_usage (state);
 | ||
| 	    exit (1);
 | ||
| 	  }
 | ||
| 	break;
 | ||
|       }
 | ||
|     case 'd':
 | ||
|       if (arguments->dir)
 | ||
| 	free (arguments->dir);
 | ||
| 
 | ||
|       arguments->dir = xstrdup (arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'n':
 | ||
|       arguments->note = 1;
 | ||
|       break;
 | ||
| 
 | ||
|     case 'm':
 | ||
|       if (arguments->memdisk)
 | ||
| 	free (arguments->memdisk);
 | ||
| 
 | ||
|       arguments->memdisk = xstrdup (arg);
 | ||
| 
 | ||
|       if (arguments->prefix)
 | ||
| 	free (arguments->prefix);
 | ||
| 
 | ||
|       arguments->prefix = xstrdup ("(memdisk)/boot/grub");
 | ||
|       break;
 | ||
| 
 | ||
|     case 'D':
 | ||
|       if (arguments->dtb)
 | ||
| 	free (arguments->dtb);
 | ||
| 
 | ||
|       arguments->dtb = xstrdup (arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'k':
 | ||
|       arguments->pubkeys = xrealloc (arguments->pubkeys,
 | ||
| 				     sizeof (arguments->pubkeys[0])
 | ||
| 				     * (arguments->npubkeys + 1));
 | ||
|       arguments->pubkeys[arguments->npubkeys++] = xstrdup (arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'c':
 | ||
|       if (arguments->config)
 | ||
| 	free (arguments->config);
 | ||
| 
 | ||
|       arguments->config = xstrdup (arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'C':
 | ||
|       if (grub_strcmp (arg, "xz") == 0)
 | ||
| 	{
 | ||
| #ifdef HAVE_LIBLZMA
 | ||
| 	  arguments->comp = GRUB_COMPRESSION_XZ;
 | ||
| #else
 | ||
| 	  grub_util_error ("%s",
 | ||
| 			   _("grub-mkimage is compiled without XZ support"));
 | ||
| #endif
 | ||
| 	}
 | ||
|       else if (grub_strcmp (arg, "none") == 0)
 | ||
| 	arguments->comp = GRUB_COMPRESSION_NONE;
 | ||
|       else if (grub_strcmp (arg, "auto") == 0)
 | ||
| 	arguments->comp = GRUB_COMPRESSION_AUTO;
 | ||
|       else
 | ||
| 	grub_util_error (_("Unknown compression format %s"), arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'p':
 | ||
|       if (arguments->prefix)
 | ||
| 	free (arguments->prefix);
 | ||
| 
 | ||
|       arguments->prefix = xstrdup (arg);
 | ||
|       break;
 | ||
| 
 | ||
|     case 'v':
 | ||
|       verbosity++;
 | ||
|       break;
 | ||
|     case ARGP_KEY_ARG:
 | ||
|       assert (arguments->nmodules < arguments->modules_max);
 | ||
|       arguments->modules[arguments->nmodules++] = xstrdup(arg);
 | ||
|       break;
 | ||
| 
 | ||
|     default:
 | ||
|       return ARGP_ERR_UNKNOWN;
 | ||
|     }
 | ||
|   return 0;
 | ||
| }
 | ||
| 
 | ||
| static struct argp argp = {
 | ||
|   options, argp_parser, N_("[OPTION]... [MODULES]"),
 | ||
|   N_("Make a bootable image of GRUB."),
 | ||
|   NULL, help_filter, NULL
 | ||
| };
 | ||
| 
 | ||
| int
 | ||
| main (int argc, char *argv[])
 | ||
| {
 | ||
|   FILE *fp = stdout;
 | ||
|   struct arguments arguments;
 | ||
|   unsigned i;
 | ||
| 
 | ||
|   grub_util_host_init (&argc, &argv);
 | ||
| 
 | ||
|   memset (&arguments, 0, sizeof (struct arguments));
 | ||
|   arguments.comp = GRUB_COMPRESSION_AUTO;
 | ||
|   arguments.modules_max = argc + 1;
 | ||
|   arguments.modules = xmalloc ((arguments.modules_max + 1)
 | ||
| 			     * sizeof (arguments.modules[0]));
 | ||
|   memset (arguments.modules, 0, (arguments.modules_max + 1)
 | ||
| 	  * sizeof (arguments.modules[0]));
 | ||
| 
 | ||
|   if (argp_parse (&argp, argc, argv, 0, 0, &arguments) != 0)
 | ||
|     {
 | ||
|       fprintf (stderr, "%s", _("Error in parsing command line arguments\n"));
 | ||
|       exit(1);
 | ||
|     }
 | ||
| 
 | ||
|   if (!arguments.image_target)
 | ||
|     {
 | ||
|       char *program = xstrdup(program_name);
 | ||
|       printf ("%s\n", _("Target format not specified (use the -O option)."));
 | ||
|       argp_help (&argp, stderr, ARGP_HELP_STD_USAGE, program);
 | ||
|       free (program);
 | ||
|       exit(1);
 | ||
|     }
 | ||
| 
 | ||
|   if (!arguments.prefix)
 | ||
|     {
 | ||
|       char *program = xstrdup(program_name);
 | ||
|       printf ("%s\n", _("Prefix not specified (use the -p option)."));
 | ||
|       argp_help (&argp, stderr, ARGP_HELP_STD_USAGE, program);
 | ||
|       free (program);
 | ||
|       exit(1);
 | ||
|     }
 | ||
| 
 | ||
|   if (arguments.output)
 | ||
|     {
 | ||
|       fp = grub_util_fopen (arguments.output, "wb");
 | ||
|       if (! fp)
 | ||
| 	grub_util_error (_("cannot open `%s': %s"), arguments.output,
 | ||
| 			 strerror (errno));
 | ||
|     }
 | ||
| 
 | ||
|   if (!arguments.dir)
 | ||
|     {
 | ||
|       const char *dn = grub_util_get_target_dirname (arguments.image_target);
 | ||
|       const char *pkglibdir = grub_util_get_pkglibdir ();
 | ||
|       char *ptr;
 | ||
|       arguments.dir = xmalloc (grub_strlen (pkglibdir) + grub_strlen (dn) + 2);
 | ||
|       ptr = grub_stpcpy (arguments.dir, pkglibdir);
 | ||
|       *ptr++ = '/';
 | ||
|       strcpy (ptr, dn);
 | ||
|     }
 | ||
| 
 | ||
|   grub_install_generate_image (arguments.dir, arguments.prefix, fp,
 | ||
| 			       arguments.output, arguments.modules,
 | ||
| 			       arguments.memdisk, arguments.pubkeys,
 | ||
| 			       arguments.npubkeys, arguments.config,
 | ||
| 			       arguments.image_target, arguments.note,
 | ||
| 			       arguments.comp, arguments.dtb);
 | ||
| 
 | ||
|   grub_util_file_sync  (fp);
 | ||
|   fclose (fp);
 | ||
| 
 | ||
|   for (i = 0; i < arguments.nmodules; i++)
 | ||
|     free (arguments.modules[i]);
 | ||
| 
 | ||
|   free (arguments.dir);
 | ||
|   free (arguments.prefix);
 | ||
|   free (arguments.modules);
 | ||
| 
 | ||
|   if (arguments.output)
 | ||
|     free (arguments.output);
 | ||
| 
 | ||
|   return 0;
 | ||
| }
 |