Rewrite grub-install, grub-mkrescue, grub-mkstandalone and grub-mknetdir
the function of these files exceeds what can be sanely handled in shell in posix-comaptible way. Also writing it in C extends the functionality to non-UNIX-like OS and minimal environments.
This commit is contained in:
		
							parent
							
								
									9ef81064a3
								
							
						
					
					
						commit
						cd46aa6cef
					
				
					 52 changed files with 5811 additions and 2101 deletions
				
			
		
							
								
								
									
										827
									
								
								util/grub-mkrescue.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										827
									
								
								util/grub-mkrescue.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,827 @@ | |||
| /*
 | ||||
|  *  Make GRUB rescue image | ||||
|  * | ||||
|  *  GRUB  --  GRand Unified Bootloader | ||||
|  *  Copyright (C) 1999,2000,2001,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/util/install.h> | ||||
| #include <grub/util/misc.h> | ||||
| #include <grub/emu/exec.h> | ||||
| #include <grub/emu/config.h> | ||||
| #include <argp.h> | ||||
| 
 | ||||
| #include <sys/types.h> | ||||
| #include <sys/wait.h> | ||||
| 
 | ||||
| #include <string.h> | ||||
| #include <time.h> | ||||
| 
 | ||||
| static char *source_dirs[GRUB_INSTALL_PLATFORM_MAX]; | ||||
| static char *rom_directory; | ||||
| static char *label_font; | ||||
| static char *label_color; | ||||
| static char *label_bgcolor; | ||||
| static char *product_name; | ||||
| static char *product_version; | ||||
| static int xorriso_tail_argc; | ||||
| static int xorriso_tail_arg_alloc; | ||||
| static char **xorriso_tail_argv; | ||||
| static char *output_image; | ||||
| static char *xorriso; | ||||
| static char *boot_grub; | ||||
| static int xorriso_argc; | ||||
| static int xorriso_arg_alloc; | ||||
| static char **xorriso_argv; | ||||
| static char *iso_uuid; | ||||
| static char *iso9660_dir; | ||||
| 
 | ||||
| static void | ||||
| xorriso_push (const char *val) | ||||
| { | ||||
|   if (xorriso_arg_alloc <= xorriso_argc + 1) | ||||
|     { | ||||
|       xorriso_arg_alloc = 2 * (4 + xorriso_argc); | ||||
|       xorriso_argv = xrealloc (xorriso_argv, | ||||
| 			       sizeof (xorriso_argv[0]) | ||||
| 			       * xorriso_arg_alloc); | ||||
|     } | ||||
|   xorriso_argv[xorriso_argc++] = xstrdup (val); | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| xorriso_link (const char *from, const char *to) | ||||
| { | ||||
|   char *tof = grub_util_path_concat (2, iso9660_dir, to); | ||||
|   char *val = xasprintf ("%s=%s", from, tof); | ||||
|   xorriso_push (val); | ||||
|   free (val); | ||||
|   free (tof); | ||||
| } | ||||
| 
 | ||||
| enum | ||||
|   { | ||||
|     OPTION_OUTPUT = 'o', | ||||
|     OPTION_ROM_DIRECTORY = 0x301, | ||||
|     OPTION_XORRISO, | ||||
|     OPTION_GLUE_EFI, | ||||
|     OPTION_RENDER_LABEL, | ||||
|     OPTION_LABEL_FONT, | ||||
|     OPTION_LABEL_COLOR, | ||||
|     OPTION_LABEL_BGCOLOR, | ||||
|     OPTION_PRODUCT_NAME, | ||||
|     OPTION_PRODUCT_VERSION, | ||||
|     OPTION_SPARC_BOOT, | ||||
|     OPTION_ARCS_BOOT | ||||
|   }; | ||||
| 
 | ||||
| static struct argp_option options[] = { | ||||
|   GRUB_INSTALL_OPTIONS, | ||||
|   {"output", 'o', N_("FILE"), | ||||
|    0, N_("save output in FILE [required]"), 2}, | ||||
|   {"rom-directory", OPTION_ROM_DIRECTORY, N_("DIR"), | ||||
|    0, N_("save ROM images in DIR [optional]"), 2}, | ||||
|   {"xorriso", OPTION_XORRISO, N_("FILE"), | ||||
|    /* TRANSLATORS: xorriso is a program for creating ISOs and burning CDs.  */ | ||||
|    0, N_("use FILE as xorriso [optional]"), 2}, | ||||
|   {"grub-glue-efi", OPTION_GLUE_EFI, N_("FILE"), OPTION_HIDDEN, 0, 2}, | ||||
|   {"grub-render-label", OPTION_RENDER_LABEL, N_("FILE"), OPTION_HIDDEN, 0, 2}, | ||||
|   {"label-font", OPTION_LABEL_FONT, N_("FILE"), 0, N_("use FILE as font for label"), 2}, | ||||
|   {"label-color", OPTION_LABEL_COLOR, N_("COLOR"), 0, N_("use COLOR for label"), 2}, | ||||
|   {"label-bgcolor", OPTION_LABEL_BGCOLOR, N_("COLOR"), 0, N_("use COLOR for label background"), 2}, | ||||
|   {"product-name", OPTION_PRODUCT_NAME, N_("STRING"), 0, N_("use STRING as product name"), 2}, | ||||
|   {"product-version", OPTION_PRODUCT_VERSION, N_("STRING"), 0, N_("use STRING as product version"), 2}, | ||||
|   {"sparc-boot", OPTION_SPARC_BOOT, 0, 0, N_("enable sparc boot. Disables HFS+, APM, ARCS and boot as disk image for i386-pc"), 2}, | ||||
|   {"arcs-boot", OPTION_ARCS_BOOT, 0, 0, N_("enable ARCS (big-endian mips machines, mostly SGI) boot. Disables HFS+, APM, sparc64 and boot as disk image for i386-pc"), 2}, | ||||
|   {0, 0, 0, 0, 0, 0} | ||||
| }; | ||||
| 
 | ||||
| static char * | ||||
| help_filter (int key, const char *text, void *input __attribute__ ((unused))) | ||||
| { | ||||
|   switch (key) | ||||
|     { | ||||
|     case ARGP_KEY_HELP_POST_DOC: | ||||
|       return xasprintf (text, "xorriso -as mkisofs -help"); | ||||
|     default: | ||||
|       return grub_install_help_filter (key, text, input); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| enum { | ||||
|   SYS_AREA_AUTO, | ||||
|   SYS_AREA_COMMON, | ||||
|   SYS_AREA_SPARC, | ||||
|   SYS_AREA_ARCS | ||||
| } system_area = SYS_AREA_AUTO; | ||||
| 
 | ||||
| static error_t  | ||||
| argp_parser (int key, char *arg, struct argp_state *state) | ||||
| { | ||||
|   if (grub_install_parse (key, arg)) | ||||
|     return 0; | ||||
|   switch (key) | ||||
|     { | ||||
|     case OPTION_OUTPUT: | ||||
|       free (output_image); | ||||
|       output_image = xstrdup (arg); | ||||
|       return 0; | ||||
|     case OPTION_ROM_DIRECTORY: | ||||
|       free (rom_directory); | ||||
|       rom_directory = xstrdup (arg); | ||||
|       return 0; | ||||
| 
 | ||||
|       /*
 | ||||
|        FIXME: | ||||
|     # Intentionally undocumented | ||||
|     --grub-mkimage-extra) | ||||
| 	mkimage_extra_arg="$mkimage_extra_arg `argument $option "$@"`"; shift ;; | ||||
|     --grub-mkimage-extra=*) | ||||
| 	mkimage_extra_arg="$mkimage_extra_arg `echo "$option" | sed 's/--grub-mkimage-extra=//'`" ;; | ||||
|       */ | ||||
|     case OPTION_SPARC_BOOT: | ||||
|       system_area = SYS_AREA_SPARC; | ||||
|       return 0; | ||||
|     case OPTION_ARCS_BOOT: | ||||
|       system_area = SYS_AREA_ARCS; | ||||
|       return 0; | ||||
|     case OPTION_PRODUCT_NAME: | ||||
|       free (product_name); | ||||
|       product_name = xstrdup (arg); | ||||
|       return 0; | ||||
|     case OPTION_PRODUCT_VERSION: | ||||
|       free (product_version); | ||||
|       product_version = xstrdup (arg); | ||||
|       return 0; | ||||
|       /* Accept and ignore for compatibility.  */ | ||||
|     case OPTION_GLUE_EFI: | ||||
|     case OPTION_RENDER_LABEL: | ||||
|       return 0; | ||||
|     case OPTION_LABEL_FONT: | ||||
|       free (label_font); | ||||
|       label_font = xstrdup (arg); | ||||
|       return 0; | ||||
| 
 | ||||
|     case OPTION_LABEL_COLOR: | ||||
|       free (label_color); | ||||
|       label_color = xstrdup (arg); | ||||
|       return 0; | ||||
| 
 | ||||
|     case OPTION_LABEL_BGCOLOR: | ||||
|       free (label_bgcolor); | ||||
|       label_bgcolor = xstrdup (arg); | ||||
|       return 0; | ||||
| 
 | ||||
|     case OPTION_XORRISO: | ||||
|       free (xorriso); | ||||
|       xorriso = xstrdup (arg); | ||||
|       return 0; | ||||
| 
 | ||||
|     case ARGP_KEY_ARG: | ||||
|       if (xorriso_tail_arg_alloc <= xorriso_tail_argc) | ||||
| 	{ | ||||
| 	  xorriso_tail_arg_alloc = 2 * (4 + xorriso_tail_argc); | ||||
| 	  xorriso_tail_argv = xrealloc (xorriso_tail_argv, | ||||
| 				   sizeof (xorriso_tail_argv[0]) | ||||
| 				   * xorriso_tail_arg_alloc); | ||||
| 	} | ||||
|       xorriso_tail_argv[xorriso_tail_argc++] = xstrdup (arg); | ||||
|       return 0; | ||||
|     default: | ||||
|       return ARGP_ERR_UNKNOWN; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| struct argp argp = { | ||||
|   options, argp_parser, N_("[OPTION] SOURCE..."), | ||||
|   /* TRANSLATORS: it generates one single image which is bootable through any method. */ | ||||
|   N_("Make GRUB CD-ROM, disk, pendrive and floppy bootable image.")"\v" | ||||
|   N_("Generates a bootable rescue image with specified source files, source directories, or mkisofs options listed by the output of `%s'.\n\n" | ||||
|      "Option -- switches to native xorriso command mode.\n\n" | ||||
|      "Mail xorriso support requests to <bug-xorriso@gnu.org>."),  | ||||
|   NULL, help_filter, NULL | ||||
| }; | ||||
| 
 | ||||
| static void | ||||
| write_part (FILE *f, const char *srcdir) | ||||
| { | ||||
|   FILE *in; | ||||
|   char *inname = grub_util_path_concat (2, srcdir, "partmap.lst"); | ||||
|   char buf[260]; | ||||
|   in = grub_util_fopen (inname, "rb"); | ||||
|   if (!in) | ||||
|     return; | ||||
|   while (fgets (buf, 256, in)) | ||||
|     { | ||||
|       char *ptr; | ||||
|       for (ptr = buf + strlen (buf) - 1; | ||||
| 	   ptr >= buf && (*ptr == '\n' || *ptr == '\r'); | ||||
| 	   ptr--); | ||||
|       ptr[1] = '\0'; | ||||
|       fprintf (f, "insmod %s\n", buf); | ||||
|     } | ||||
|   fclose (in); | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| make_image_abs (enum grub_install_plat plat, | ||||
| 		const char *mkimage_target, | ||||
| 		const char *output, | ||||
| 		grub_compression_t compress) | ||||
| { | ||||
|   char *load_cfg; | ||||
|   FILE *load_cfg_f; | ||||
| 
 | ||||
|   if (!source_dirs[plat]) | ||||
|     return; | ||||
| 
 | ||||
|   grub_util_info (N_("enabling %s support ..."), | ||||
| 		  mkimage_target); | ||||
| 
 | ||||
|   load_cfg = grub_util_make_temporary_file (); | ||||
| 
 | ||||
|   load_cfg_f = grub_util_fopen (load_cfg, "wb"); | ||||
|   fprintf (load_cfg_f, "search --fs-uuid --set=root %s\n", iso_uuid); | ||||
|   fprintf (load_cfg_f, "set prefix=(${root})/boot/grub\n"); | ||||
| 
 | ||||
|   write_part (load_cfg_f, source_dirs[plat]); | ||||
|   fclose (load_cfg_f); | ||||
| 
 | ||||
|   grub_install_push_module ("search"); | ||||
|   grub_install_push_module ("iso9660"); | ||||
|   grub_install_make_image_wrap (source_dirs[plat], "/boot/grub", output, | ||||
| 				0, load_cfg, | ||||
| 				mkimage_target, 0, | ||||
| 				compress); | ||||
|   grub_install_pop_module (); | ||||
|   grub_install_pop_module (); | ||||
|   grub_util_unlink (load_cfg); | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| make_image (enum grub_install_plat plat, | ||||
| 	    const char *mkimage_target, | ||||
| 	    const char *output_sub, | ||||
| 	    grub_compression_t compress) | ||||
| { | ||||
|   char *out = grub_util_path_concat (2, boot_grub, output_sub); | ||||
|   make_image_abs (plat, mkimage_target, | ||||
| 		  out, GRUB_COMPRESSION_AUTO); | ||||
|   free (out); | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| make_image_fwdisk_abs (enum grub_install_plat plat, | ||||
| 		       const char *mkimage_target, | ||||
| 		       const char *output) | ||||
| { | ||||
|   if (!source_dirs[plat]) | ||||
|     return; | ||||
| 
 | ||||
|   grub_install_push_module ("iso9660"); | ||||
|   grub_install_make_image_wrap (source_dirs[plat], "()/boot/grub", output, | ||||
| 				0, 0, mkimage_target, 0, | ||||
| 				GRUB_COMPRESSION_AUTO); | ||||
|   grub_install_pop_module (); | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| check_xorriso (const char *val) | ||||
| { | ||||
|   const char *argv[5]; | ||||
|   int fd; | ||||
|   pid_t pid; | ||||
|   FILE *mdadm; | ||||
|   char *buf = NULL; | ||||
|   size_t len = 0; | ||||
|   int ret = 0; | ||||
| 
 | ||||
|   argv[0] = xorriso; | ||||
|   argv[1] = "-as"; | ||||
|   argv[2] = "mkisofs"; | ||||
|   argv[3] = "-help"; | ||||
|   argv[4] = NULL; | ||||
| 
 | ||||
|   pid = grub_util_exec_pipe_stderr (argv, &fd); | ||||
| 
 | ||||
|   if (!pid) | ||||
|     return 0; | ||||
| 
 | ||||
|   /* Parent.  Read mdadm's output.  */ | ||||
|   mdadm = fdopen (fd, "r"); | ||||
|   if (! mdadm) | ||||
|     return 0; | ||||
| 
 | ||||
|   while (getline (&buf, &len, mdadm) > 0) | ||||
|     { | ||||
|       if (grub_strstr (buf, val)) | ||||
| 	ret = 1; | ||||
|     } | ||||
| 
 | ||||
|   close (fd); | ||||
|   waitpid (pid, NULL, 0); | ||||
|   free (buf); | ||||
|   return ret; | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| make_image_fwdisk (enum grub_install_plat plat, | ||||
| 		   const char *mkimage_target, | ||||
| 		   const char *output_sub) | ||||
| { | ||||
|   char *out = grub_util_path_concat (2, boot_grub, output_sub); | ||||
|   make_image_fwdisk_abs (plat, mkimage_target, out); | ||||
|   free (out); | ||||
| } | ||||
| 
 | ||||
| int | ||||
| main (int argc, char *argv[]) | ||||
| { | ||||
|   char *romdir; | ||||
|   char *sysarea_img = NULL; | ||||
|   const char *pkgdatadir; | ||||
| 
 | ||||
|   grub_util_host_init (&argc, &argv); | ||||
| 
 | ||||
|   pkgdatadir = grub_util_get_pkgdatadir (); | ||||
| 
 | ||||
|   product_name = xstrdup (PACKAGE_NAME); | ||||
|   product_version = xstrdup (PACKAGE_VERSION); | ||||
|   xorriso = xstrdup ("xorriso"); | ||||
|   label_font = grub_util_path_concat (2, pkgdatadir, "unicode.pf2"); | ||||
| 
 | ||||
|   argp_parse (&argp, argc, argv, 0, 0, 0); | ||||
| 
 | ||||
|   if (!output_image) | ||||
|     grub_util_error ("%s", _("output file must be specified")); | ||||
| 
 | ||||
|   xorriso_push (xorriso); | ||||
|   xorriso_push ("-as"); | ||||
|   xorriso_push ("mkisofs"); | ||||
|   xorriso_push ("-graft-points"); | ||||
|    | ||||
|   iso9660_dir = grub_util_make_temporary_dir (); | ||||
|   grub_util_info ("temporaray iso9660 dir is `%s'",  | ||||
| 		  iso9660_dir); | ||||
|   boot_grub = grub_util_path_concat (3, iso9660_dir, "boot", "grub"); | ||||
|   grub_install_mkdir_p (boot_grub); | ||||
|   romdir = grub_util_path_concat (2, boot_grub, "roms"); | ||||
|   grub_util_mkdir (romdir); | ||||
| 
 | ||||
|   if (!grub_install_source_directory) | ||||
|     { | ||||
|       enum grub_install_plat plat; | ||||
| 
 | ||||
|       for (plat = 0; plat < GRUB_INSTALL_PLATFORM_MAX; plat++) | ||||
| 	{ | ||||
| 	  char *platdir = grub_util_path_concat (2, pkgdatadir, | ||||
| 						 grub_install_get_platform_name (plat)); | ||||
| 
 | ||||
| 	  if (!grub_util_is_directory (platdir)) | ||||
| 	    { | ||||
| 	      free (platdir); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	  source_dirs[plat] = platdir; | ||||
| 	  grub_install_copy_files (platdir, | ||||
| 				   boot_grub, plat); | ||||
| 	} | ||||
|     } | ||||
|   else | ||||
|     { | ||||
|       enum grub_install_plat plat; | ||||
|       plat = grub_install_get_target (grub_install_source_directory); | ||||
|       grub_install_copy_files (grub_install_source_directory, | ||||
| 			       boot_grub, plat); | ||||
|       source_dirs[plat] = xstrdup (grub_install_source_directory); | ||||
|     } | ||||
|   if (system_area == SYS_AREA_AUTO || grub_install_source_directory) | ||||
|     { | ||||
|       if (source_dirs[GRUB_INSTALL_PLATFORM_I386_PC] | ||||
| 	  || source_dirs[GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275] | ||||
| 	  || source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI] | ||||
| 	  || source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI]) | ||||
| 	system_area = SYS_AREA_COMMON; | ||||
|       else if (source_dirs[GRUB_INSTALL_PLATFORM_SPARC64_IEEE1275]) | ||||
| 	system_area = SYS_AREA_SPARC; | ||||
|       else if (source_dirs[GRUB_INSTALL_PLATFORM_MIPS_ARC]) | ||||
| 	system_area = SYS_AREA_ARCS; | ||||
|     } | ||||
| 
 | ||||
|   /* obtain date-based UUID.  */ | ||||
|   { | ||||
|     time_t tim; | ||||
|     struct tm *tmm; | ||||
|     tim = time (NULL); | ||||
|     tmm = gmtime (&tim); | ||||
|     iso_uuid = xmalloc (55); | ||||
|     grub_snprintf (iso_uuid, 50, | ||||
| 		   "%04d-%02d-%02d-%02d-%02d-%02d-00", | ||||
| 		   tmm->tm_year + 1900, | ||||
| 		   tmm->tm_mon + 1, | ||||
| 		   tmm->tm_mday, | ||||
| 		   tmm->tm_hour, | ||||
| 		   tmm->tm_min, | ||||
| 		   tmm->tm_sec); | ||||
|   } | ||||
|   { | ||||
|     char *uuid_out = xmalloc (strlen (iso_uuid) + 1 + 40); | ||||
|     char *optr; | ||||
|     const char *iptr; | ||||
|     optr = grub_stpcpy (uuid_out, "--modification-date="); | ||||
|     for (iptr = iso_uuid; *iptr; iptr++) | ||||
|       if (*iptr != '-') | ||||
| 	*optr++ = *iptr; | ||||
|     *optr = '\0'; | ||||
|     xorriso_push (uuid_out); | ||||
|     free (uuid_out); | ||||
|   } | ||||
| 
 | ||||
|   /* build BIOS core.img.  */ | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_I386_PC]) | ||||
|     { | ||||
|       char *load_cfg; | ||||
|       FILE *load_cfg_f; | ||||
|       char *output = grub_util_path_concat (3, boot_grub, "i386-pc", "eltorito.img"); | ||||
|       load_cfg = grub_util_make_temporary_file (); | ||||
| 
 | ||||
|       grub_util_info (N_("enabling %s support ..."), "BIOS"); | ||||
|       load_cfg_f = grub_util_fopen (load_cfg, "wb"); | ||||
|       write_part (load_cfg_f, source_dirs[GRUB_INSTALL_PLATFORM_I386_PC]); | ||||
|       fclose (load_cfg_f); | ||||
| 
 | ||||
|       grub_install_push_module ("biosdisk"); | ||||
|       grub_install_push_module ("iso9660"); | ||||
|       grub_install_make_image_wrap (source_dirs[GRUB_INSTALL_PLATFORM_I386_PC], | ||||
| 				    "/boot/grub", output, | ||||
| 				    0, load_cfg, | ||||
| 				    "i386-pc-eltorito", 0, | ||||
| 				    GRUB_COMPRESSION_AUTO); | ||||
| 
 | ||||
|       xorriso_push ("-b"); | ||||
|       xorriso_push ("boot/grub/i386-pc/eltorito.img"); | ||||
|       xorriso_push ("-no-emul-boot"); | ||||
|       xorriso_push ("-boot-load-size"); | ||||
|       xorriso_push ("4"); | ||||
|       xorriso_push ("-boot-info-table"); | ||||
|       if (system_area == SYS_AREA_COMMON) | ||||
| 	{ | ||||
| 	  if (check_xorriso ("grub2-boot-info")) | ||||
| 	    { | ||||
| 	      char *boot_hybrid = grub_util_path_concat (2, source_dirs[GRUB_INSTALL_PLATFORM_I386_PC], | ||||
| 							 "boot_hybrid.img"); | ||||
| 	      xorriso_push ("--grub2-boot-info"); | ||||
| 	      xorriso_push ("--grub2-mbr"); | ||||
| 	      xorriso_push (boot_hybrid); | ||||
| 	    } | ||||
| 	  else | ||||
| 	    { | ||||
| 	      FILE *sa, *bi; | ||||
| 	      size_t sz; | ||||
| 	      char buf[512]; | ||||
| 	      char *bin = grub_util_path_concat (2, source_dirs[GRUB_INSTALL_PLATFORM_I386_PC], | ||||
| 						 "boot.img"); | ||||
| 	      grub_util_warn ("%s", _("Your xorriso doesn't support `--grub2-boot-info'. Some features are disabled. Please use xorriso 1.2.9 or later.")); | ||||
| 	      sysarea_img = grub_util_make_temporary_file (); | ||||
| 	      sa = grub_util_fopen (sysarea_img, "wb"); | ||||
| 	      if (!sa) | ||||
| 		grub_util_error (_("cannot open `%s': %s"), sysarea_img, | ||||
| 				 strerror (errno)); | ||||
| 	      bi = grub_util_fopen (sysarea_img, "wb"); | ||||
| 	      if (!bi) | ||||
| 		grub_util_error (_("cannot open `%s': %s"), bin, | ||||
| 				 strerror (errno)); | ||||
| 	      fread (buf, 1, 512, bi); | ||||
| 	      fclose (bi); | ||||
| 	      fwrite (buf, 1, 512, sa); | ||||
| 	       | ||||
| 	      grub_install_make_image_wrap (source_dirs[GRUB_INSTALL_PLATFORM_I386_PC], | ||||
| 					    "/boot/grub", output, | ||||
| 					    0, load_cfg, | ||||
| 					    "i386-pc", 0, | ||||
| 					    GRUB_COMPRESSION_AUTO); | ||||
| 	      sz = ftello (sa); | ||||
| 	      fflush (sa); | ||||
| 	      fsync (fileno (sa)); | ||||
| 	      fclose (sa); | ||||
| 	       | ||||
| 	      if (sz > 32768) | ||||
| 		{ | ||||
| 		  grub_util_warn ("%s", _("Your xorriso doesn't support `--grub2-boot-info'. Your core image is too big. Boot as disk is disabled. Please use xorriso 1.2.9 or later.")); | ||||
| 		} | ||||
| 	      else | ||||
| 		{ | ||||
| 		  xorriso_push ("-G"); | ||||
| 		  xorriso_push (sysarea_img); | ||||
| 		} | ||||
| 	    } | ||||
| 	} | ||||
|       grub_install_pop_module (); | ||||
|       grub_install_pop_module (); | ||||
|     } | ||||
| 
 | ||||
|   /** build multiboot core.img */ | ||||
|   grub_install_push_module ("pata"); | ||||
|   grub_install_push_module ("ahci"); | ||||
|   grub_install_push_module ("at_keyboard"); | ||||
|   make_image (GRUB_INSTALL_PLATFORM_I386_MULTIBOOT, "i386-multiboot", "i386-multiboot/core.elf", GRUB_COMPRESSION_AUTO); | ||||
|   grub_install_pop_module (); | ||||
|   grub_install_pop_module (); | ||||
|   grub_install_pop_module (); | ||||
| 
 | ||||
|   make_image_fwdisk (GRUB_INSTALL_PLATFORM_I386_IEEE1275, "i386-ieee1275", "ofwx86.elf"); | ||||
| 
 | ||||
|   char *core_services = NULL; | ||||
| 
 | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI] | ||||
|       || source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI] | ||||
|       || source_dirs[GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275]) | ||||
|     { | ||||
|       char *mach_ker, *sv, *label, *label_text; | ||||
|       FILE *f; | ||||
|       core_services = grub_util_path_concat (4, iso9660_dir, "System", "Library", "CoreServices"); | ||||
|       grub_install_mkdir_p (core_services); | ||||
| 
 | ||||
|       mach_ker = grub_util_path_concat (2, iso9660_dir, "mach_kernel"); | ||||
|       f = grub_util_fopen (mach_ker, "wb"); | ||||
|       fclose (f); | ||||
|       free (mach_ker); | ||||
| 
 | ||||
|       sv = grub_util_path_concat (2, core_services, "SystemVersion.plist"); | ||||
|       f = grub_util_fopen (sv, "wb"); | ||||
|       fprintf (f, "<plist version=\"1.0\">\n" | ||||
| 	       "<dict>\n" | ||||
| 	       "        <key>ProductBuildVersion</key>\n" | ||||
| 	       "        <string></string>\n" | ||||
| 	       "        <key>ProductName</key>\n" | ||||
| 	       "        <string>%s</string>\n" | ||||
| 	       "        <key>ProductVersion</key>\n" | ||||
| 	       "        <string>%s</string>\n" | ||||
| 	       "</dict>\n" | ||||
| 	       "</plist>\n", product_name, product_version); | ||||
|       fclose (f); | ||||
|       free (sv); | ||||
|       label = grub_util_path_concat (2, core_services, ".disk_label"); | ||||
|       char *label_string = xasprintf ("%s %s", product_name, product_version); | ||||
|       grub_util_render_label (label_font, label_bgcolor ? : "white", | ||||
| 			      label_color ? : "black", label_string, label); | ||||
|       free (label); | ||||
|       label_text = grub_util_path_concat (2, core_services, ".disk_label.contentDetails"); | ||||
|       f = grub_util_fopen (label_text, "wb"); | ||||
|       fprintf (f, "%s", label_string); | ||||
|       fclose (f); | ||||
|       free (label_string); | ||||
|       free (label_text); | ||||
|       if (system_area == SYS_AREA_COMMON) | ||||
| 	{ | ||||
| 	  xorriso_push ("-hfsplus"); | ||||
| 	  xorriso_push ("-apm-block-size"); | ||||
| 	  xorriso_push ("2048"); | ||||
| 	  xorriso_push ("-hfsplus-file-creator-type"); | ||||
| 	  xorriso_push ("chrp"); | ||||
| 	  xorriso_push ("tbxj"); | ||||
| 	  xorriso_push ("/System/Library/CoreServices/.disk_label"); | ||||
| 
 | ||||
| 	  if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI] | ||||
| 	      || source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI]) | ||||
| 	    { | ||||
| 	      xorriso_push ("-hfs-bless-by"); | ||||
| 	      xorriso_push ("i"); | ||||
| 	      xorriso_push ("/System/Library/CoreServices/boot.efi"); | ||||
| 	    } | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI] | ||||
|       || source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI] | ||||
|       || source_dirs[GRUB_INSTALL_PLATFORM_IA64_EFI] | ||||
|       || source_dirs[GRUB_INSTALL_PLATFORM_ARM_EFI]) | ||||
|     { | ||||
|       char *efidir = grub_util_make_temporary_dir (); | ||||
|       char *efidir_efi = grub_util_path_concat (2, efidir, "efi"); | ||||
|       char *efidir_efi_boot = grub_util_path_concat (3, efidir, "efi", "boot"); | ||||
|       char *imgname, *img32, *img64, *img_mac = NULL; | ||||
|       char *efiimgfat; | ||||
|       grub_install_mkdir_p (efidir_efi_boot); | ||||
| 
 | ||||
|       imgname = grub_util_path_concat (2, efidir_efi_boot, "bootia64.efi"); | ||||
|       make_image_fwdisk_abs (GRUB_INSTALL_PLATFORM_IA64_EFI, "ia64-efi", imgname); | ||||
|       free (imgname); | ||||
| 
 | ||||
|       img64 = grub_util_path_concat (2, efidir_efi_boot, "bootx64.efi"); | ||||
|       make_image_fwdisk_abs (GRUB_INSTALL_PLATFORM_X86_64_EFI, "x86_64-efi", img64); | ||||
| 
 | ||||
|       img32 = grub_util_path_concat (2, efidir_efi_boot, "bootia32.efi"); | ||||
|       make_image_fwdisk_abs (GRUB_INSTALL_PLATFORM_I386_EFI, "i386-efi", img32); | ||||
| 
 | ||||
|       imgname = grub_util_path_concat (2, efidir_efi_boot, "bootarm.efi"); | ||||
|       make_image_fwdisk_abs (GRUB_INSTALL_PLATFORM_ARM_EFI, "arm-efi", imgname); | ||||
|       free (imgname); | ||||
| 
 | ||||
|       if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI]) | ||||
| 	{ | ||||
| 	  imgname = grub_util_path_concat (2, efidir_efi_boot, "boot.efi"); | ||||
| 	  /* For old macs. Suggested by Peter Jones.  */ | ||||
| 	  grub_install_copy_file (img32, imgname, 1); | ||||
| 	} | ||||
| 
 | ||||
|       if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI] | ||||
| 	  || source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI]) | ||||
| 	img_mac = grub_util_path_concat (2, core_services, "boot.efi"); | ||||
| 
 | ||||
|       if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI] | ||||
| 	  && source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI]) | ||||
| 	grub_util_glue_efi (img32, img64, img_mac); | ||||
|       else if (source_dirs[GRUB_INSTALL_PLATFORM_X86_64_EFI]) | ||||
| 	grub_install_copy_file (img64, img_mac, 1); | ||||
|       else if (source_dirs[GRUB_INSTALL_PLATFORM_I386_EFI]) | ||||
| 	grub_install_copy_file (img32, img_mac, 1); | ||||
| 
 | ||||
|       free (img_mac); | ||||
|       free (img32); | ||||
|       free (img64); | ||||
|       free (efidir_efi_boot); | ||||
| 
 | ||||
|       efiimgfat = grub_util_path_concat (2, iso9660_dir, "efi.img"); | ||||
|       grub_util_exec ((const char * []) { "mformat", "-C", "-f", "2880", "-L", "16", "-i", | ||||
| 	    efiimgfat, "::", NULL }); | ||||
|       grub_util_exec ((const char * []) { "mcopy", "-s", "-i", efiimgfat, efidir_efi, "::/", NULL }); | ||||
|       xorriso_push ("--efi-boot"); | ||||
|       xorriso_push ("efi.img"); | ||||
|       xorriso_push ("-efi-boot-part"); | ||||
|       xorriso_push ("--efi-boot-image"); | ||||
| 
 | ||||
|       grub_util_unlink_recursive (efidir); | ||||
|       free (efiimgfat); | ||||
|       free (efidir_efi); | ||||
|       free (efidir); | ||||
|     } | ||||
| 
 | ||||
|   make_image_fwdisk (GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275, "powerpc-ieee1275", "powerpc-ieee1275/core.elf"); | ||||
| 
 | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275]) | ||||
|     { | ||||
|       char *grub_chrp = grub_util_path_concat (2, source_dirs[GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275], | ||||
| 					       "grub.chrp"); | ||||
|       char *bisrc = grub_util_path_concat (2, source_dirs[GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275], | ||||
| 					   "grub.chrp"); | ||||
|       char *bootx = grub_util_path_concat (2, core_services, "BootX"); | ||||
|       char *ppc_chrp = grub_util_path_concat (3, iso9660_dir, "ppc", "chrp"); | ||||
|       char *bitgt = grub_util_path_concat (3, iso9660_dir, "ppc", "bootinfo.txt"); | ||||
|       grub_install_copy_file (grub_chrp, bootx, 1); | ||||
|       grub_install_mkdir_p (ppc_chrp); | ||||
|       grub_install_copy_file (bisrc, bitgt, 1); | ||||
|       xorriso_link ("/System/Library/CoreServices/grub.elf", "/boot/grub/powerpc-ieee1275/core.elf"); | ||||
|       xorriso_link ("/boot/grub/powerpc.elf", "/boot/grub/powerpc-ieee1275/core.elf"); | ||||
|       /* FIXME: add PreP */ | ||||
|       if (system_area == SYS_AREA_COMMON) | ||||
| 	{ | ||||
| 	  xorriso_push ("-hfsplus-file-creator-type"); | ||||
| 	  xorriso_push ("chrp"); | ||||
| 	  xorriso_push ("tbxi"); | ||||
| 	  xorriso_push ("/System/Library/CoreServices/BootX"); | ||||
| 	  xorriso_push ("-hfs-bless-by"); | ||||
| 	  xorriso_push ("p"); | ||||
| 	  xorriso_push ("/System/Library/CoreServices"); | ||||
| 	} | ||||
|       xorriso_push ("-sysid"); | ||||
|       xorriso_push ("PPC"); | ||||
|     } | ||||
| 
 | ||||
|   make_image_fwdisk (GRUB_INSTALL_PLATFORM_SPARC64_IEEE1275, | ||||
| 		     "sparc64-ieee1275-cdcore", "sparc64-ieee1275/core.img"); | ||||
| 
 | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_SPARC64_IEEE1275] | ||||
|       && system_area == SYS_AREA_SPARC) | ||||
|     { | ||||
|       char *cdboot; | ||||
|       FILE *in, *out; | ||||
|       char buf[512]; | ||||
|       sysarea_img = grub_util_make_temporary_file (); | ||||
|       cdboot = grub_util_path_concat (2, source_dirs[GRUB_INSTALL_PLATFORM_SPARC64_IEEE1275], | ||||
| 				      "cdboot.img"); | ||||
|       in = grub_util_fopen (cdboot, "rb"); | ||||
|       out = grub_util_fopen (sysarea_img, "wb"); | ||||
|       memset (buf, 0, 512); | ||||
|       fwrite (buf, 1, 512, out); | ||||
|       fread (buf, 1, 512, in); | ||||
|       fwrite (buf, 1, 512, out); | ||||
|       fclose (in); | ||||
|       fclose (out); | ||||
|       xorriso_push ("-G"); | ||||
|       xorriso_push (sysarea_img); | ||||
|       xorriso_push ("-B"); | ||||
|       xorriso_push (","); | ||||
|       xorriso_push ("--grub2-sparc-core"); | ||||
|       xorriso_push ("/boot/grub/sparc64-ieee1275/core.img"); | ||||
|     } | ||||
| 
 | ||||
|   make_image_fwdisk (GRUB_INSTALL_PLATFORM_MIPS_ARC, "mips-arc", "mips-arc/core.img"); | ||||
| 
 | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_MIPS_ARC]) | ||||
|     { | ||||
|       xorriso_link ("/boot/grub/mips-arc/grub", "/boot/grub/mips-arc/core.img"); | ||||
|       xorriso_link ("/boot/grub/mips-arc/sashARCS", "/boot/grub/mips-arc/core.img"); | ||||
|       xorriso_link ("/boot/grub/mips-arc/sash", "/boot/grub/mips-arc/core.img"); | ||||
|     } | ||||
|   if (source_dirs[GRUB_INSTALL_PLATFORM_MIPS_ARC] && system_area == SYS_AREA_ARCS) | ||||
|     { | ||||
|       xorriso_push ("-mips-boot"); | ||||
|       xorriso_push ("/boot/grub/mips-arc/sashARCS"); | ||||
|       xorriso_push ("-mips-boot"); | ||||
|       xorriso_push ("/boot/grub/mips-arc/sash"); | ||||
|       xorriso_push ("-mips-boot"); | ||||
|       xorriso_push ("/boot/grub/mips-arc/grub"); | ||||
|     } | ||||
| 
 | ||||
|   make_image_fwdisk (GRUB_INSTALL_PLATFORM_MIPSEL_ARC, "mipsel-arc", "arc.exe"); | ||||
| 
 | ||||
|   grub_install_push_module ("pata"); | ||||
|   make_image (GRUB_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS, "mipsel-qemu_mips-elf", "roms/mipsel-qemu_mips.elf", GRUB_COMPRESSION_AUTO); | ||||
| 
 | ||||
|   make_image (GRUB_INSTALL_PLATFORM_MIPSEL_LOONGSON, "mipsel-loongson-elf", "loongson.elf", GRUB_COMPRESSION_XZ); | ||||
| 
 | ||||
|   make_image (GRUB_INSTALL_PLATFORM_MIPSEL_LOONGSON, "mipsel-yeeloong-flash", "mipsel-yeeloong.bin", GRUB_COMPRESSION_XZ); | ||||
|   make_image (GRUB_INSTALL_PLATFORM_MIPSEL_LOONGSON, "mipsel-fulong2f-flash", "mipsel-fuloong2f.bin", GRUB_COMPRESSION_XZ); | ||||
| 
 | ||||
|   make_image (GRUB_INSTALL_PLATFORM_MIPS_QEMU_MIPS, "mips-qemu_mips-elf", "roms/mips-qemu_mips.elf", GRUB_COMPRESSION_AUTO); | ||||
| 
 | ||||
|   grub_install_push_module ("at_keyboard"); | ||||
| 
 | ||||
|   make_image (GRUB_INSTALL_PLATFORM_I386_QEMU, "i386-qemu", "roms/qemu.img", GRUB_COMPRESSION_AUTO); | ||||
| 
 | ||||
|   grub_install_push_module ("ahci"); | ||||
| 
 | ||||
|   make_image (GRUB_INSTALL_PLATFORM_I386_COREBOOT, "i386-coreboot", "roms/coreboot.elf", GRUB_COMPRESSION_AUTO); | ||||
|   grub_install_pop_module (); | ||||
|   grub_install_pop_module (); | ||||
|   grub_install_pop_module (); | ||||
| 
 | ||||
|   if (rom_directory) | ||||
|     { | ||||
|       const struct | ||||
|       { | ||||
| 	enum grub_install_plat plat; | ||||
| 	const char *from, *to; | ||||
|       } roms[] = | ||||
| 	  { | ||||
| 	    {GRUB_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS, "roms/mipsel-qemu_mips.elf", "mipsel-qemu_mips.elf"}, | ||||
| 	    {GRUB_INSTALL_PLATFORM_MIPSEL_LOONGSON, "loongson.elf", "mipsel-loongson.elf"}, | ||||
| 	    {GRUB_INSTALL_PLATFORM_MIPSEL_LOONGSON, "roms/mipsel-yeeloong.bin", "mipsel-yeeloong.bin"}, | ||||
| 	    {GRUB_INSTALL_PLATFORM_MIPSEL_LOONGSON, "roms/mipsel-fulong.bin", "mipsel-fulong.bin"}, | ||||
| 	    {GRUB_INSTALL_PLATFORM_MIPS_QEMU_MIPS, "roms/mips-qemu_mips.elf", "mips-qemu_mips.elf"}, | ||||
| 	    {GRUB_INSTALL_PLATFORM_I386_QEMU, "roms/qemu.img", "qemu.img"}, | ||||
| 	    {GRUB_INSTALL_PLATFORM_I386_COREBOOT, "roms/coreboot.elf", "coreboot.elf"}, | ||||
| 	  }; | ||||
|       grub_size_t i; | ||||
|       for (i = 0; i < ARRAY_SIZE (roms); i++) | ||||
| 	{ | ||||
| 	  char *from = grub_util_path_concat (2, boot_grub, roms[i].from); | ||||
| 	  char *to = grub_util_path_concat (2, rom_directory, roms[i].to); | ||||
| 	  grub_install_copy_file (from, to, 0); | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|   xorriso_push ("--protective-msdos-label"); | ||||
|   xorriso_push ("-o"); | ||||
|   xorriso_push (output_image); | ||||
|   xorriso_push ("-r"); | ||||
|   xorriso_push (iso9660_dir); | ||||
|   xorriso_push ("--sort-weight"); | ||||
|   xorriso_push ("0"); | ||||
|   xorriso_push ("/"); | ||||
|   xorriso_push ("--sort-weight"); | ||||
|   xorriso_push ("1"); | ||||
|   xorriso_push ("/boot"); | ||||
|   int i; | ||||
|   for (i = 0; i < xorriso_tail_argc; i++) | ||||
|     xorriso_push (xorriso_tail_argv[i]); | ||||
| 
 | ||||
|   xorriso_argv[xorriso_argc] = NULL; | ||||
| 
 | ||||
|   grub_util_exec ((const char *const *)xorriso_argv); | ||||
| 
 | ||||
|   grub_util_unlink_recursive (iso9660_dir); | ||||
| 
 | ||||
|   if (sysarea_img) | ||||
|     grub_util_unlink (sysarea_img); | ||||
| 
 | ||||
|   free (core_services); | ||||
|   free (romdir); | ||||
|   return 0; | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue