grub-mknetdir.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. /*
  2. * Copyright (C) 1999,2000,2001,2002,2003,2004,2005,2006,2007,2008,2009,2010,2013 Free Software Foundation, Inc.
  3. *
  4. * GRUB is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * GRUB is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. #include <config.h>
  18. #include <grub/util/install.h>
  19. #include <grub/emu/config.h>
  20. #include <grub/util/misc.h>
  21. #include <string.h>
  22. #include <errno.h>
  23. #pragma GCC diagnostic ignored "-Wmissing-prototypes"
  24. #pragma GCC diagnostic ignored "-Wmissing-declarations"
  25. #include <argp.h>
  26. #pragma GCC diagnostic error "-Wmissing-prototypes"
  27. #pragma GCC diagnostic error "-Wmissing-declarations"
  28. static char *rootdir = NULL, *subdir = NULL;
  29. static char *debug_image = NULL;
  30. enum
  31. {
  32. OPTION_NET_DIRECTORY = 0x301,
  33. OPTION_SUBDIR,
  34. OPTION_DEBUG,
  35. OPTION_DEBUG_IMAGE
  36. };
  37. static struct argp_option options[] = {
  38. GRUB_INSTALL_OPTIONS,
  39. {"net-directory", OPTION_NET_DIRECTORY, N_("DIR"),
  40. 0, N_("root directory of TFTP server"), 2},
  41. {"subdir", OPTION_SUBDIR, N_("DIR"),
  42. 0, N_("relative subdirectory on network server"), 2},
  43. {"debug", OPTION_DEBUG, 0, OPTION_HIDDEN, 0, 2},
  44. {"debug-image", OPTION_DEBUG_IMAGE, N_("STRING"), OPTION_HIDDEN, 0, 2},
  45. {0, 0, 0, 0, 0, 0}
  46. };
  47. static error_t
  48. argp_parser (int key, char *arg, struct argp_state *state)
  49. {
  50. if (grub_install_parse (key, arg))
  51. return 0;
  52. switch (key)
  53. {
  54. case OPTION_NET_DIRECTORY:
  55. free (rootdir);
  56. rootdir = xstrdup (arg);
  57. return 0;
  58. case OPTION_SUBDIR:
  59. free (subdir);
  60. subdir = xstrdup (arg);
  61. return 0;
  62. /* This is an undocumented feature... */
  63. case OPTION_DEBUG:
  64. verbosity++;
  65. return 0;
  66. case OPTION_DEBUG_IMAGE:
  67. free (debug_image);
  68. debug_image = xstrdup (arg);
  69. return 0;
  70. case ARGP_KEY_ARG:
  71. default:
  72. return ARGP_ERR_UNKNOWN;
  73. }
  74. }
  75. struct argp argp = {
  76. options, argp_parser, NULL,
  77. "\v"N_("Prepares GRUB network boot images at net_directory/subdir "
  78. "assuming net_directory being TFTP root."),
  79. NULL, grub_install_help_filter, NULL
  80. };
  81. static char *base;
  82. static const struct
  83. {
  84. const char *mkimage_target;
  85. const char *netmodule;
  86. const char *ext;
  87. } targets[GRUB_INSTALL_PLATFORM_MAX] =
  88. {
  89. [GRUB_INSTALL_PLATFORM_I386_PC] = { "i386-pc-pxe", "pxe", ".0" },
  90. [GRUB_INSTALL_PLATFORM_SPARC64_IEEE1275] = { "sparc64-ieee1275-aout", "ofnet", ".img" },
  91. [GRUB_INSTALL_PLATFORM_I386_IEEE1275] = { "i386-ieee1275", "ofnet", ".elf" },
  92. [GRUB_INSTALL_PLATFORM_POWERPC_IEEE1275] = { "powerpc-ieee1275", "ofnet", ".elf" },
  93. [GRUB_INSTALL_PLATFORM_I386_EFI] = { "i386-efi", "efinet", ".efi" },
  94. [GRUB_INSTALL_PLATFORM_X86_64_EFI] = { "x86_64-efi", "efinet", ".efi" },
  95. [GRUB_INSTALL_PLATFORM_IA64_EFI] = { "ia64-efi", "efinet", ".efi" },
  96. [GRUB_INSTALL_PLATFORM_ARM_EFI] = { "arm-efi", "efinet", ".efi" },
  97. [GRUB_INSTALL_PLATFORM_ARM64_EFI] = { "arm64-efi", "efinet", ".efi" },
  98. [GRUB_INSTALL_PLATFORM_LOONGARCH64_EFI] = { "loongarch64-efi", "efinet", ".efi" },
  99. [GRUB_INSTALL_PLATFORM_RISCV32_EFI] = { "riscv32-efi", "efinet", ".efi" },
  100. [GRUB_INSTALL_PLATFORM_RISCV64_EFI] = { "riscv64-efi", "efinet", ".efi" },
  101. };
  102. static void
  103. process_input_dir (const char *input_dir, enum grub_install_plat platform)
  104. {
  105. char *platsub = grub_install_get_platform_name (platform);
  106. char *grubdir = grub_util_path_concat (3, rootdir, subdir, platsub);
  107. char *load_cfg = grub_util_path_concat (2, grubdir, "load.cfg");
  108. char *prefix;
  109. char *output;
  110. char *grub_cfg;
  111. FILE *cfg;
  112. grub_install_copy_files (input_dir, base, platform);
  113. grub_util_unlink (load_cfg);
  114. if (debug_image)
  115. {
  116. FILE *f = grub_util_fopen (load_cfg, "wb");
  117. if (!f)
  118. grub_util_error (_("cannot open `%s': %s"), load_cfg,
  119. strerror (errno));
  120. fprintf (f, "set debug='%s'\n", debug_image);
  121. fclose (f);
  122. }
  123. else
  124. {
  125. free (load_cfg);
  126. load_cfg = 0;
  127. }
  128. prefix = xasprintf ("/%s", subdir);
  129. if (!targets[platform].mkimage_target)
  130. grub_util_error (_("unsupported platform %s"), platsub);
  131. grub_cfg = grub_util_path_concat (2, grubdir, "grub.cfg");
  132. cfg = grub_util_fopen (grub_cfg, "wb");
  133. if (!cfg)
  134. grub_util_error (_("cannot open `%s': %s"), grub_cfg,
  135. strerror (errno));
  136. fprintf (cfg, "source %s/grub.cfg", subdir);
  137. fclose (cfg);
  138. grub_install_push_module (targets[platform].netmodule);
  139. output = grub_util_path_concat_ext (2, grubdir, "core", targets[platform].ext);
  140. grub_install_make_image_wrap (input_dir, prefix, output,
  141. 0, load_cfg,
  142. targets[platform].mkimage_target, 0);
  143. grub_set_install_backup_ponr ();
  144. grub_install_pop_module ();
  145. /* TRANSLATORS: First %s is replaced by platform name. Second one by filename. */
  146. printf (_("Netboot directory for %s created. Configure your DHCP server to point to %s\n"),
  147. platsub, output);
  148. free (platsub);
  149. free (output);
  150. free (prefix);
  151. free (grub_cfg);
  152. free (grubdir);
  153. }
  154. int
  155. main (int argc, char *argv[])
  156. {
  157. const char *pkglibdir;
  158. grub_util_host_init (&argc, &argv);
  159. grub_util_disable_fd_syncs ();
  160. rootdir = xstrdup ("/srv/tftp");
  161. pkglibdir = grub_util_get_pkglibdir ();
  162. subdir = grub_util_path_concat (2, GRUB_BOOT_DIR_NAME, GRUB_DIR_NAME);
  163. argp_parse (&argp, argc, argv, 0, 0, 0);
  164. base = grub_util_path_concat (2, rootdir, subdir);
  165. /* Create the GRUB directory if it is not present. */
  166. grub_install_mkdir_p (base);
  167. grub_install_push_module ("tftp");
  168. if (!grub_install_source_directory)
  169. {
  170. enum grub_install_plat plat;
  171. for (plat = 0; plat < GRUB_INSTALL_PLATFORM_MAX; plat++)
  172. if (targets[plat].mkimage_target)
  173. {
  174. char *platdir = grub_util_path_concat (2, pkglibdir,
  175. grub_install_get_platform_name (plat));
  176. grub_util_info ("Looking for `%s'", platdir);
  177. if (!grub_util_is_directory (platdir))
  178. {
  179. free (platdir);
  180. continue;
  181. }
  182. process_input_dir (platdir, plat);
  183. }
  184. }
  185. else
  186. {
  187. enum grub_install_plat plat;
  188. plat = grub_install_get_target (grub_install_source_directory);
  189. process_input_dir (grub_install_source_directory, plat);
  190. }
  191. return 0;
  192. }