* grub-core/fs/jfs.c: Remove variable length arrays. Reduces jfs.mod

by 364 bytes (169 compressed).
This commit is contained in:
Vladimir 'phcoder' Serbinenko 2013-10-20 13:20:12 +02:00
parent d28e1163c3
commit 2359bf887c
2 changed files with 56 additions and 105 deletions

View file

@ -1,3 +1,8 @@
2013-10-20 Vladimir Serbinenko <phcoder@gmail.com>
* grub-core/fs/jfs.c: Remove variable length arrays. Reduces jfs.mod
by 364 bytes (169 compressed).
2013-10-20 Vladimir Serbinenko <phcoder@gmail.com> 2013-10-20 Vladimir Serbinenko <phcoder@gmail.com>
* grub-core/fs/bfs.c: Remove variable length arrays. Reduces afs.mod and * grub-core/fs/bfs.c: Remove variable length arrays. Reduces afs.mod and

View file

@ -639,136 +639,74 @@ static grub_err_t
grub_jfs_find_file (struct grub_jfs_data *data, const char *path, grub_jfs_find_file (struct grub_jfs_data *data, const char *path,
grub_uint32_t start_ino) grub_uint32_t start_ino)
{ {
char fpath[grub_strlen (path)]; const char *name;
char *name = fpath; const char *next = path;
char *next; struct grub_jfs_diropen *diro = NULL;
struct grub_jfs_diropen *diro;
grub_strncpy (fpath, path, grub_strlen (path) + 1);
if (grub_jfs_read_inode (data, start_ino, &data->currinode)) if (grub_jfs_read_inode (data, start_ino, &data->currinode))
return grub_errno; return grub_errno;
/* Skip the first slashes. */ while (1)
while (*name == '/')
{
name++;
if (!*name)
return 0;
}
/* Extract the actual part from the pathname. */
next = grub_strchr (name, '/');
if (next)
{
while (*next == '/')
{
next[0] = '\0';
next++;
}
}
diro = grub_jfs_opendir (data, &data->currinode);
if (!diro)
return grub_errno;
for (;;)
{ {
name = next;
while (*name == '/')
name++;
if (name[0] == 0) if (name[0] == 0)
return GRUB_ERR_NONE; return GRUB_ERR_NONE;
for (next = name; *next && *next != '/'; next++);
if (name[0] == '.' && name[1] == 0) if (name[0] == '.' && name + 1 == next)
{ continue;
if (!next)
return 0;
name = next; if (name[0] == '.' && name[1] == '.' && name + 2 == next)
next = grub_strchr (name, '/');
while (next && *next == '/')
{
next[0] = '\0';
next++;
}
continue;
}
if (name[0] == '.' && name[1] == '.' && name[2] == 0)
{ {
grub_uint32_t ino = grub_le_to_cpu32 (data->currinode.dir.header.idotdot); grub_uint32_t ino = grub_le_to_cpu32 (data->currinode.dir.header.idotdot);
grub_jfs_closedir (diro);
diro = 0;
if (grub_jfs_read_inode (data, ino, &data->currinode)) if (grub_jfs_read_inode (data, ino, &data->currinode))
break;
if (!next)
return 0;
name = next;
next = grub_strchr (name, '/');
while (next && *next == '/')
{
next[0] = '\0';
next++;
}
/* Open this directory for reading dirents. */
diro = grub_jfs_opendir (data, &data->currinode);
if (!diro)
return grub_errno; return grub_errno;
continue; continue;
} }
if (grub_jfs_getent (diro) == GRUB_ERR_OUT_OF_RANGE) diro = grub_jfs_opendir (data, &data->currinode);
break; if (!diro)
return grub_errno;
/* Check if the current direntry matches the current part of the for (;;)
pathname. */
if (data->caseins ? grub_strcasecmp (name, diro->name) == 0
: grub_strcmp (name, diro->name) == 0)
{ {
grub_uint32_t ino = diro->ino; if (grub_jfs_getent (diro) == GRUB_ERR_OUT_OF_RANGE)
grub_uint32_t dirino = grub_le_to_cpu32 (data->currinode.inode);
grub_jfs_closedir (diro);
diro = 0;
if (grub_jfs_read_inode (data, ino, &data->currinode))
break;
/* Check if this is a symlink. */
if ((grub_le_to_cpu32 (data->currinode.mode)
& GRUB_JFS_FILETYPE_MASK) == GRUB_JFS_FILETYPE_LNK)
{ {
grub_jfs_lookup_symlink (data, dirino); grub_jfs_closedir (diro);
if (grub_errno) return grub_error (GRUB_ERR_FILE_NOT_FOUND, N_("file `%s' not found"), path);
return grub_errno;
} }
if (!next) /* Check if the current direntry matches the current part of the
return 0; pathname. */
if ((data->caseins ? grub_strncasecmp (name, diro->name, next - name) == 0
name = next; : grub_strncmp (name, diro->name, next - name) == 0) && !diro->name[next - name])
next = grub_strchr (name, '/');
while (next && *next == '/')
{ {
next[0] = '\0'; grub_uint32_t ino = diro->ino;
next++; grub_uint32_t dirino = grub_le_to_cpu32 (data->currinode.inode);
grub_jfs_closedir (diro);
diro = 0;
if (grub_jfs_read_inode (data, ino, &data->currinode))
break;
/* Check if this is a symlink. */
if ((grub_le_to_cpu32 (data->currinode.mode)
& GRUB_JFS_FILETYPE_MASK) == GRUB_JFS_FILETYPE_LNK)
{
grub_jfs_lookup_symlink (data, dirino);
if (grub_errno)
return grub_errno;
}
break;
} }
/* Open this directory for reading dirents. */
diro = grub_jfs_opendir (data, &data->currinode);
if (!diro)
return grub_errno;
continue;
} }
} }
grub_jfs_closedir (diro);
grub_error (GRUB_ERR_FILE_NOT_FOUND, N_("file `%s' not found"), path);
return grub_errno;
} }
@ -776,15 +714,21 @@ static grub_err_t
grub_jfs_lookup_symlink (struct grub_jfs_data *data, grub_uint32_t ino) grub_jfs_lookup_symlink (struct grub_jfs_data *data, grub_uint32_t ino)
{ {
grub_size_t size = grub_le_to_cpu64 (data->currinode.size); grub_size_t size = grub_le_to_cpu64 (data->currinode.size);
char symlink[size + 1]; char *symlink;
if (++data->linknest > GRUB_JFS_MAX_SYMLNK_CNT) if (++data->linknest > GRUB_JFS_MAX_SYMLNK_CNT)
return grub_error (GRUB_ERR_SYMLINK_LOOP, N_("too deep nesting of symlinks")); return grub_error (GRUB_ERR_SYMLINK_LOOP, N_("too deep nesting of symlinks"));
symlink = grub_malloc (size + 1);
if (!symlink)
return grub_errno;
if (size <= sizeof (data->currinode.symlink.path)) if (size <= sizeof (data->currinode.symlink.path))
grub_strncpy (symlink, (char *) (data->currinode.symlink.path), size); grub_strncpy (symlink, (char *) (data->currinode.symlink.path), size);
else if (grub_jfs_read_file (data, 0, 0, 0, size, symlink) < 0) else if (grub_jfs_read_file (data, 0, 0, 0, size, symlink) < 0)
return grub_errno; {
grub_free (symlink);
return grub_errno;
}
symlink[size] = '\0'; symlink[size] = '\0';
@ -794,6 +738,8 @@ grub_jfs_lookup_symlink (struct grub_jfs_data *data, grub_uint32_t ino)
grub_jfs_find_file (data, symlink, ino); grub_jfs_find_file (data, symlink, ino);
grub_free (symlink);
return grub_errno; return grub_errno;
} }