214 lines
5.6 KiB
C
214 lines
5.6 KiB
C
/*
|
|
* GRUB -- GRand Unified Bootloader
|
|
* Copyright (C) 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 <grub/types.h>
|
|
#include <grub/misc.h>
|
|
#include <grub/extcmd.h>
|
|
#include <grub/mm.h>
|
|
#include <grub/err.h>
|
|
#include <grub/dl.h>
|
|
#include <grub/file.h>
|
|
#include <grub/normal.h>
|
|
#include <grub/script_sh.h>
|
|
#include <grub/i18n.h>
|
|
#include <grub/term.h>
|
|
#include <grub/syslinux_parse.h>
|
|
#include <grub/crypto.h>
|
|
#include <grub/auth.h>
|
|
#include <grub/disk.h>
|
|
#include <grub/partition.h>
|
|
|
|
GRUB_MOD_LICENSE ("GPLv3+");
|
|
|
|
/* Helper for syslinux_file. */
|
|
static grub_err_t
|
|
syslinux_file_getline (char **line, int cont __attribute__ ((unused)),
|
|
void *data __attribute__ ((unused)))
|
|
{
|
|
*line = 0;
|
|
return GRUB_ERR_NONE;
|
|
}
|
|
|
|
static const struct grub_arg_option options[] =
|
|
{
|
|
{"root", 'r', 0,
|
|
N_("root directory of the syslinux disk [default=/]."),
|
|
N_("DIR"), ARG_TYPE_STRING},
|
|
{"cwd", 'c', 0,
|
|
N_("current directory of the syslinux [default is parent directory of input file]."),
|
|
N_("DIR"), ARG_TYPE_STRING},
|
|
{"isolinux", 'i', 0, N_("assume input is an isolinux configuration file."), 0, 0},
|
|
{"pxelinux", 'p', 0, N_("assume input is a pxelinux configuration file."), 0, 0},
|
|
{"syslinux", 's', 0, N_("assume input is a syslinux configuration file."), 0, 0},
|
|
{0, 0, 0, 0, 0, 0}
|
|
};
|
|
|
|
enum
|
|
{
|
|
OPTION_ROOT,
|
|
OPTION_CWD,
|
|
OPTION_ISOLINUX,
|
|
OPTION_PXELINUX,
|
|
OPTION_SYSLINUX
|
|
};
|
|
|
|
static grub_err_t
|
|
syslinux_file (grub_extcmd_context_t ctxt, const char *filename)
|
|
{
|
|
char *result;
|
|
const char *root = ctxt->state[OPTION_ROOT].set ? ctxt->state[OPTION_ROOT].arg : "/";
|
|
const char *cwd = ctxt->state[OPTION_CWD].set ? ctxt->state[OPTION_CWD].arg : NULL;
|
|
grub_syslinux_flavour_t flav = GRUB_SYSLINUX_UNKNOWN;
|
|
char *cwdf = NULL;
|
|
grub_menu_t menu;
|
|
|
|
if (ctxt->state[OPTION_ISOLINUX].set)
|
|
flav = GRUB_SYSLINUX_ISOLINUX;
|
|
if (ctxt->state[OPTION_PXELINUX].set)
|
|
flav = GRUB_SYSLINUX_PXELINUX;
|
|
if (ctxt->state[OPTION_SYSLINUX].set)
|
|
flav = GRUB_SYSLINUX_SYSLINUX;
|
|
|
|
if (!cwd)
|
|
{
|
|
char *p;
|
|
cwdf = grub_strdup (filename);
|
|
if (!cwdf)
|
|
return grub_errno;
|
|
p = grub_strrchr (cwdf, '/');
|
|
if (!p)
|
|
{
|
|
grub_free (cwdf);
|
|
cwd = "/";
|
|
cwdf = NULL;
|
|
}
|
|
else
|
|
{
|
|
*p = '\0';
|
|
cwd = cwdf;
|
|
}
|
|
}
|
|
|
|
grub_dprintf ("syslinux",
|
|
"transforming syslinux config %s, root = %s, cwd = %s\n",
|
|
filename, root, cwd);
|
|
|
|
result = grub_syslinux_config_file (root, root, cwd, cwd, filename, flav);
|
|
if (!result)
|
|
return grub_errno;
|
|
|
|
grub_dprintf ("syslinux", "syslinux config transformed\n");
|
|
|
|
menu = grub_env_get_menu ();
|
|
if (! menu)
|
|
{
|
|
menu = grub_zalloc (sizeof (*menu));
|
|
if (! menu)
|
|
return grub_errno;
|
|
|
|
grub_env_set_menu (menu);
|
|
}
|
|
|
|
grub_normal_parse_line (result, syslinux_file_getline, NULL);
|
|
grub_print_error ();
|
|
grub_free (result);
|
|
grub_free (cwdf);
|
|
|
|
return GRUB_ERR_NONE;
|
|
}
|
|
|
|
static grub_err_t
|
|
grub_cmd_syslinux_source (grub_extcmd_context_t ctxt,
|
|
int argc, char **args)
|
|
{
|
|
int new_env, extractor;
|
|
grub_err_t ret;
|
|
|
|
if (argc != 1)
|
|
return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected"));
|
|
|
|
extractor = (ctxt->extcmd->cmd->name[0] == 'e');
|
|
new_env = (ctxt->extcmd->cmd->name[extractor ? (sizeof ("extract_syslinux_entries_") - 1)
|
|
: (sizeof ("syslinux_") - 1)] == 'c');
|
|
|
|
if (new_env)
|
|
grub_cls ();
|
|
|
|
if (new_env && !extractor)
|
|
grub_env_context_open ();
|
|
if (extractor)
|
|
grub_env_extractor_open (!new_env);
|
|
|
|
ret = syslinux_file (ctxt, args[0]);
|
|
|
|
if (new_env)
|
|
{
|
|
grub_menu_t menu;
|
|
menu = grub_env_get_menu ();
|
|
if (menu && menu->size)
|
|
grub_show_menu (menu, 1, 0);
|
|
if (!extractor)
|
|
grub_env_context_close ();
|
|
}
|
|
if (extractor)
|
|
grub_env_extractor_close (!new_env);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
static grub_extcmd_t cmd_source, cmd_configfile;
|
|
static grub_extcmd_t cmd_source_extract, cmd_configfile_extract;
|
|
|
|
GRUB_MOD_INIT(syslinuxcfg)
|
|
{
|
|
cmd_source
|
|
= grub_register_extcmd ("syslinux_source",
|
|
grub_cmd_syslinux_source, 0,
|
|
N_("FILE"),
|
|
/* TRANSLATORS: "syslinux config" means
|
|
"config as used by syslinux". */
|
|
N_("Execute syslinux config in same context"),
|
|
options);
|
|
cmd_configfile
|
|
= grub_register_extcmd ("syslinux_configfile",
|
|
grub_cmd_syslinux_source, 0,
|
|
N_("FILE"),
|
|
N_("Execute syslinux config in new context"),
|
|
options);
|
|
cmd_source_extract
|
|
= grub_register_extcmd ("extract_syslinux_entries_source",
|
|
grub_cmd_syslinux_source, 0,
|
|
N_("FILE"),
|
|
N_("Execute syslinux config in same context taking only menu entries"),
|
|
options);
|
|
cmd_configfile_extract
|
|
= grub_register_extcmd ("extract_syslinux_entries_configfile",
|
|
grub_cmd_syslinux_source, 0,
|
|
N_("FILE"),
|
|
N_("Execute syslinux config in new context taking only menu entries"),
|
|
options);
|
|
}
|
|
|
|
GRUB_MOD_FINI(syslinuxcfg)
|
|
{
|
|
grub_unregister_extcmd (cmd_source);
|
|
grub_unregister_extcmd (cmd_configfile);
|
|
grub_unregister_extcmd (cmd_source_extract);
|
|
grub_unregister_extcmd (cmd_configfile_extract);
|
|
}
|