Fix few bugs and put a cleaner way to handle kernel command
This commit is contained in:
parent
092fd48a25
commit
68cc4355f8
1 changed files with 87 additions and 64 deletions
|
@ -27,24 +27,26 @@
|
||||||
#include <grub/normal.h>
|
#include <grub/normal.h>
|
||||||
#include <grub/script_sh.h>
|
#include <grub/script_sh.h>
|
||||||
#include <grub/i18n.h>
|
#include <grub/i18n.h>
|
||||||
|
#include <grub/term.h>
|
||||||
|
|
||||||
struct legacy_command
|
struct legacy_command
|
||||||
{
|
{
|
||||||
const char *name;
|
const char *name;
|
||||||
const char *map;
|
const char *map;
|
||||||
unsigned argc;
|
unsigned argc;
|
||||||
enum {
|
enum arg_type {
|
||||||
TYPE_VERBATIM,
|
TYPE_VERBATIM,
|
||||||
TYPE_FORCE_OPTION,
|
TYPE_FORCE_OPTION,
|
||||||
TYPE_NOAPM_OPTION,
|
TYPE_NOAPM_OPTION,
|
||||||
|
TYPE_TYPE_OPTION,
|
||||||
TYPE_FILE,
|
TYPE_FILE,
|
||||||
TYPE_PARTITION,
|
TYPE_PARTITION,
|
||||||
TYPE_BOOL,
|
TYPE_BOOL,
|
||||||
TYPE_INT
|
TYPE_INT,
|
||||||
|
TYPE_REST_VERBATIM
|
||||||
} argt[3];
|
} argt[3];
|
||||||
enum {
|
enum {
|
||||||
FLAG_IGNORE_REST = 1,
|
FLAG_IGNORE_REST = 1
|
||||||
FLAG_ALL_VERBATIM = 2
|
|
||||||
} flags;
|
} flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -85,7 +87,8 @@ struct legacy_command legacy_commands[] =
|
||||||
/* install unsupported. */
|
/* install unsupported. */
|
||||||
/* ioprobe unsupported. */
|
/* ioprobe unsupported. */
|
||||||
/* FIXME: implement command. */
|
/* FIXME: implement command. */
|
||||||
{"kernel", "legacy_kernel %s\n", 0, {}, FLAG_ALL_VERBATIM},
|
{"kernel", "legacy_kernel %s '%s' %s\n", 3, {TYPE_TYPE_OPTION, TYPE_FILE,
|
||||||
|
TYPE_REST_VERBATIM}, 0},
|
||||||
/* lock is handled separately. */
|
/* lock is handled separately. */
|
||||||
{"makeactive", "parttool '%s' boot+\n", 1, {TYPE_PARTITION}, 0},
|
{"makeactive", "parttool '%s' boot+\n", 1, {TYPE_PARTITION}, 0},
|
||||||
{"map", "drivemap '%s' '%s'\n", 2, {TYPE_PARTITION, TYPE_PARTITION},
|
{"map", "drivemap '%s' '%s'\n", 2, {TYPE_PARTITION, TYPE_PARTITION},
|
||||||
|
@ -116,6 +119,7 @@ struct legacy_command legacy_commands[] =
|
||||||
/* title is handled separately. */
|
/* title is handled separately. */
|
||||||
{"unhide", "parttool '%s' hidden-\n", 1, {TYPE_PARTITION}, 0},
|
{"unhide", "parttool '%s' hidden-\n", 1, {TYPE_PARTITION}, 0},
|
||||||
/* uppermem unsupported. */
|
/* uppermem unsupported. */
|
||||||
|
{"uuid", "search -u '%s'\n", 1, {TYPE_VERBATIM}, 0},
|
||||||
/* vbeprobe unsupported. */
|
/* vbeprobe unsupported. */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -139,6 +143,7 @@ escape (const char *in)
|
||||||
|
|
||||||
*outptr++ = *ptr;
|
*outptr++ = *ptr;
|
||||||
}
|
}
|
||||||
|
*outptr++ = 0;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -188,6 +193,27 @@ adjust_file (const char *in)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
is_option (enum arg_type opt, const char *curarg)
|
||||||
|
{
|
||||||
|
switch (opt)
|
||||||
|
{
|
||||||
|
case TYPE_NOAPM_OPTION:
|
||||||
|
return grub_strcmp (curarg, "--no-apm") == 0;
|
||||||
|
case TYPE_FORCE_OPTION:
|
||||||
|
return grub_strcmp (curarg, "--force") == 0;
|
||||||
|
case TYPE_TYPE_OPTION:
|
||||||
|
return grub_strcmp (curarg, "--type=netbsd") == 0
|
||||||
|
|| grub_strcmp (curarg, "--type=freebsd") == 0
|
||||||
|
|| grub_strcmp (curarg, "--type=openbsd") == 0
|
||||||
|
|| grub_strcmp (curarg, "--type=linux") == 0
|
||||||
|
|| grub_strcmp (curarg, "--type=biglinux") == 0
|
||||||
|
|| grub_strcmp (curarg, "--type=multiboot") == 0;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static char *
|
static char *
|
||||||
legacy_parse (char *buf, char **entryname)
|
legacy_parse (char *buf, char **entryname)
|
||||||
{
|
{
|
||||||
|
@ -197,7 +223,11 @@ legacy_parse (char *buf, char **entryname)
|
||||||
|
|
||||||
for (ptr = buf; *ptr && grub_isspace (*ptr); ptr++);
|
for (ptr = buf; *ptr && grub_isspace (*ptr); ptr++);
|
||||||
if ((!*ptr || *ptr == '#') && entryname && *entryname)
|
if ((!*ptr || *ptr == '#') && entryname && *entryname)
|
||||||
return buf;
|
{
|
||||||
|
char *ret = grub_xasprintf ("%s\n", buf);
|
||||||
|
grub_free (buf);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
if (!*ptr || *ptr == '#')
|
if (!*ptr || *ptr == '#')
|
||||||
{
|
{
|
||||||
grub_free (buf);
|
grub_free (buf);
|
||||||
|
@ -234,51 +264,6 @@ legacy_parse (char *buf, char **entryname)
|
||||||
char *args[ARRAY_SIZE (legacy_commands[0].argt)];
|
char *args[ARRAY_SIZE (legacy_commands[0].argt)];
|
||||||
memset (args, 0, sizeof (args));
|
memset (args, 0, sizeof (args));
|
||||||
|
|
||||||
if (legacy_commands[cmdnum].flags & FLAG_ALL_VERBATIM)
|
|
||||||
{
|
|
||||||
char *arg0 = ptr, *outptr;
|
|
||||||
int overhead = 3;
|
|
||||||
while (*ptr)
|
|
||||||
{
|
|
||||||
char *curarg;
|
|
||||||
for (; grub_isspace (*ptr); ptr++);
|
|
||||||
curarg = ptr;
|
|
||||||
for (; *ptr && !grub_isspace (*ptr); ptr++)
|
|
||||||
if (*ptr == '\\' || *ptr == '\'')
|
|
||||||
overhead++;
|
|
||||||
if (ptr)
|
|
||||||
ptr++;
|
|
||||||
overhead += 3;
|
|
||||||
}
|
|
||||||
args[0] = grub_malloc (overhead + (ptr - arg0));
|
|
||||||
if (!args[0])
|
|
||||||
{
|
|
||||||
grub_free (buf);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
ptr = arg0;
|
|
||||||
outptr = args[0];
|
|
||||||
while (*ptr)
|
|
||||||
{
|
|
||||||
char *curarg;
|
|
||||||
for (; grub_isspace (*ptr); ptr++);
|
|
||||||
curarg = ptr;
|
|
||||||
if (outptr != args[0])
|
|
||||||
*outptr++ = ' ';
|
|
||||||
*outptr++ = '\'';
|
|
||||||
for (; *ptr && !grub_isspace (*ptr); ptr++)
|
|
||||||
{
|
|
||||||
if (*ptr == '\\' || *ptr == '\'')
|
|
||||||
*outptr++ = '\\';
|
|
||||||
*outptr++ = *ptr;
|
|
||||||
}
|
|
||||||
*outptr++ = '\'';
|
|
||||||
if (ptr)
|
|
||||||
ptr++;
|
|
||||||
overhead += 3;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
{
|
||||||
unsigned j = 0;
|
unsigned j = 0;
|
||||||
for (i = 0; i < legacy_commands[cmdnum].argc; i++)
|
for (i = 0; i < legacy_commands[cmdnum].argc; i++)
|
||||||
|
@ -294,6 +279,7 @@ legacy_parse (char *buf, char **entryname)
|
||||||
c = *cptr;
|
c = *cptr;
|
||||||
*ptr = 0;
|
*ptr = 0;
|
||||||
}
|
}
|
||||||
|
if (*ptr)
|
||||||
ptr++;
|
ptr++;
|
||||||
switch (legacy_commands[cmdnum].argt[i])
|
switch (legacy_commands[cmdnum].argt[i])
|
||||||
{
|
{
|
||||||
|
@ -302,28 +288,65 @@ legacy_parse (char *buf, char **entryname)
|
||||||
args[j++] = adjust_file (curarg);
|
args[j++] = adjust_file (curarg);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case TYPE_REST_VERBATIM:
|
||||||
|
{
|
||||||
|
char *outptr, *outptr0;
|
||||||
|
int overhead = 3;
|
||||||
|
ptr = curarg;
|
||||||
|
while (*ptr)
|
||||||
|
{
|
||||||
|
for (; grub_isspace (*ptr); ptr++);
|
||||||
|
for (; *ptr && !grub_isspace (*ptr); ptr++)
|
||||||
|
if (*ptr == '\\' || *ptr == '\'')
|
||||||
|
overhead++;
|
||||||
|
if (*ptr)
|
||||||
|
ptr++;
|
||||||
|
overhead += 3;
|
||||||
|
}
|
||||||
|
outptr0 = args[j++] = grub_malloc (overhead + (ptr - curarg));
|
||||||
|
if (!outptr0)
|
||||||
|
{
|
||||||
|
grub_free (buf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
ptr = curarg;
|
||||||
|
outptr = outptr0;
|
||||||
|
while (*ptr)
|
||||||
|
{
|
||||||
|
for (; grub_isspace (*ptr); ptr++);
|
||||||
|
if (outptr != outptr0)
|
||||||
|
*outptr++ = ' ';
|
||||||
|
*outptr++ = '\'';
|
||||||
|
for (; *ptr && !grub_isspace (*ptr); ptr++)
|
||||||
|
{
|
||||||
|
if (*ptr == '\\' || *ptr == '\'')
|
||||||
|
*outptr++ = '\\';
|
||||||
|
*outptr++ = *ptr;
|
||||||
|
}
|
||||||
|
*outptr++ = '\'';
|
||||||
|
if (*ptr)
|
||||||
|
ptr++;
|
||||||
|
overhead += 3;
|
||||||
|
}
|
||||||
|
*outptr++ = 0;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case TYPE_VERBATIM:
|
case TYPE_VERBATIM:
|
||||||
args[j++] = escape (curarg);
|
args[j++] = escape (curarg);
|
||||||
break;
|
break;
|
||||||
case TYPE_FORCE_OPTION:
|
case TYPE_FORCE_OPTION:
|
||||||
if (grub_strcmp (curarg, "--force") == 0)
|
|
||||||
{
|
|
||||||
args[j++] = grub_strdup ("--force");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (cptr)
|
|
||||||
*cptr = c;
|
|
||||||
ptr = curarg;
|
|
||||||
break;
|
|
||||||
case TYPE_NOAPM_OPTION:
|
case TYPE_NOAPM_OPTION:
|
||||||
if (grub_strcmp (curarg, "--no-apm") == 0)
|
case TYPE_TYPE_OPTION:
|
||||||
|
if (is_option (legacy_commands[cmdnum].argt[i], curarg))
|
||||||
{
|
{
|
||||||
args[j++] = grub_strdup ("--no-apm");
|
args[j++] = grub_strdup (curarg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (cptr)
|
if (cptr)
|
||||||
*cptr = c;
|
*cptr = c;
|
||||||
ptr = curarg;
|
ptr = curarg;
|
||||||
|
args[j++] = "";
|
||||||
break;
|
break;
|
||||||
case TYPE_INT:
|
case TYPE_INT:
|
||||||
{
|
{
|
||||||
|
@ -448,7 +471,7 @@ legacy_file (const char *filename)
|
||||||
grub_free (parsed);
|
grub_free (parsed);
|
||||||
return grub_errno;
|
return grub_errno;
|
||||||
}
|
}
|
||||||
grub_memcpy (entrysrc + grub_strlen (entrysrc), buf,
|
grub_memcpy (entrysrc + grub_strlen (entrysrc), parsed,
|
||||||
grub_strlen (parsed) + 1);
|
grub_strlen (parsed) + 1);
|
||||||
grub_free (parsed);
|
grub_free (parsed);
|
||||||
parsed = NULL;
|
parsed = NULL;
|
||||||
|
|
Loading…
Reference in a new issue