* grub-core/commands/wildcard.c (+check_file): New function.
(wildcard_expand): Don't expand to non-existing files, expand with suffix and not attempt to expand if not needed.
This commit is contained in:
parent
3d2de9a7ba
commit
f4d5820f28
2 changed files with 132 additions and 0 deletions
|
@ -1,3 +1,9 @@
|
||||||
|
2012-06-07 Vladimir Serbinenko <phcoder@gmail.com>
|
||||||
|
|
||||||
|
* grub-core/commands/wildcard.c (+check_file): New function.
|
||||||
|
(wildcard_expand): Don't expand to non-existing files, expand with
|
||||||
|
suffix and not attempt to expand if not needed.
|
||||||
|
|
||||||
2012-06-07 Vladimir Serbinenko <phcoder@gmail.com>
|
2012-06-07 Vladimir Serbinenko <phcoder@gmail.com>
|
||||||
|
|
||||||
* grub-core/disk/efi/efidisk.c (name_devices): Don't make disks
|
* grub-core/disk/efi/efidisk.c (name_devices): Don't make disks
|
||||||
|
|
|
@ -363,6 +363,57 @@ match_files (const char *prefix, const char *suffix, const char *end,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_file (const char *dir, const char *basename)
|
||||||
|
{
|
||||||
|
grub_fs_t fs;
|
||||||
|
grub_device_t dev;
|
||||||
|
int found = 0;
|
||||||
|
const char *device_name, *path;
|
||||||
|
|
||||||
|
auto int match (const char *name, const struct grub_dirhook_info *info);
|
||||||
|
int match (const char *name, const struct grub_dirhook_info *info)
|
||||||
|
{
|
||||||
|
if (basename[0] == 0
|
||||||
|
&& (info->case_insensitive ? grub_strcasecmp (name, basename) == 0
|
||||||
|
: grub_strcmp (name, basename) == 0))
|
||||||
|
{
|
||||||
|
found = 1;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
device_name = grub_file_get_device_name (dir);
|
||||||
|
dev = grub_device_open (device_name);
|
||||||
|
if (! dev)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
fs = grub_fs_probe (dev);
|
||||||
|
if (! fs)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (dir[0] == '(')
|
||||||
|
{
|
||||||
|
path = grub_strchr (dir, ')');
|
||||||
|
if (!path)
|
||||||
|
goto fail;
|
||||||
|
path++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
path = dir;
|
||||||
|
|
||||||
|
fs->dir (dev, path, match);
|
||||||
|
if (grub_errno == 0 && basename[0] == 0)
|
||||||
|
found = 1;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
grub_errno = 0;
|
||||||
|
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
static char*
|
static char*
|
||||||
wildcard_escape (const char *s)
|
wildcard_escape (const char *s)
|
||||||
{
|
{
|
||||||
|
@ -419,18 +470,91 @@ wildcard_expand (const char *s, char ***strs)
|
||||||
const char *regexop;
|
const char *regexop;
|
||||||
const char *noregexop;
|
const char *noregexop;
|
||||||
char **paths = 0;
|
char **paths = 0;
|
||||||
|
int had_regexp = 0;
|
||||||
|
|
||||||
unsigned i;
|
unsigned i;
|
||||||
regex_t regexp;
|
regex_t regexp;
|
||||||
|
|
||||||
|
if (s[0] != '/' && s[0] != '(' && s[0] != '*')
|
||||||
|
return 0;
|
||||||
|
|
||||||
start = s;
|
start = s;
|
||||||
while (*start)
|
while (*start)
|
||||||
{
|
{
|
||||||
split_path (start, &noregexop, ®exop);
|
split_path (start, &noregexop, ®exop);
|
||||||
|
|
||||||
|
if (noregexop == regexop)
|
||||||
|
{
|
||||||
|
grub_dprintf ("expand", "no expansion needed\n");
|
||||||
|
if (paths == 0)
|
||||||
|
{
|
||||||
|
paths = grub_malloc (sizeof (char *) * 2);
|
||||||
|
if (!paths)
|
||||||
|
goto fail;
|
||||||
|
paths[0] = grub_malloc (regexop - start + 1);
|
||||||
|
if (!paths[0])
|
||||||
|
goto fail;
|
||||||
|
grub_memcpy (paths[0], start, regexop - start);
|
||||||
|
paths[0][regexop - start] = '\0';
|
||||||
|
paths[1] = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
int j = 0;
|
||||||
|
for (i = 0; paths[i]; i++)
|
||||||
|
{
|
||||||
|
char *o, *oend;
|
||||||
|
char *n;
|
||||||
|
char *p;
|
||||||
|
o = paths[i];
|
||||||
|
oend = o + grub_strlen (o);
|
||||||
|
n = grub_malloc ((oend - o) + (regexop - start) + 1);
|
||||||
|
if (!n)
|
||||||
|
goto fail;
|
||||||
|
grub_memcpy (n, o, oend - o);
|
||||||
|
grub_memcpy (n + (oend - o), start, regexop - start);
|
||||||
|
n[(oend - o) + (regexop - start)] = '\0';
|
||||||
|
if (had_regexp)
|
||||||
|
p = grub_strrchr (n, '/');
|
||||||
|
else
|
||||||
|
p = 0;
|
||||||
|
if (!p)
|
||||||
|
{
|
||||||
|
grub_free (o);
|
||||||
|
paths[j++] = n;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
*p = 0;
|
||||||
|
if (!check_file (n, p + 1))
|
||||||
|
{
|
||||||
|
grub_dprintf ("expand", "file <%s> not found\n",
|
||||||
|
start);
|
||||||
|
grub_free (o);
|
||||||
|
grub_free (n);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
*p = '/';
|
||||||
|
grub_free (o);
|
||||||
|
paths[j++] = n;
|
||||||
|
}
|
||||||
|
if (j == 0)
|
||||||
|
{
|
||||||
|
grub_free (paths);
|
||||||
|
paths = 0;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
paths[j] = 0;
|
||||||
|
}
|
||||||
|
grub_dprintf ("expand", "paths[0] = `%s'\n", paths[0]);
|
||||||
|
start = regexop;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (make_regex (noregexop, regexop, ®exp))
|
if (make_regex (noregexop, regexop, ®exp))
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
|
had_regexp = 1;
|
||||||
|
|
||||||
if (paths == 0)
|
if (paths == 0)
|
||||||
{
|
{
|
||||||
if (start == noregexop) /* device part has regexop */
|
if (start == noregexop) /* device part has regexop */
|
||||||
|
@ -448,6 +572,7 @@ wildcard_expand (const char *s, char ***strs)
|
||||||
char **p;
|
char **p;
|
||||||
|
|
||||||
p = match_files (paths[i], start, noregexop, ®exp);
|
p = match_files (paths[i], start, noregexop, ®exp);
|
||||||
|
grub_free (paths[i]);
|
||||||
if (! p)
|
if (! p)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -455,6 +580,7 @@ wildcard_expand (const char *s, char ***strs)
|
||||||
if (! r)
|
if (! r)
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
grub_free (paths);
|
||||||
paths = r;
|
paths = r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue