2007-08-02 17:24:06 +00:00
|
|
|
|
/* hostfs.c - Dummy filesystem to provide access to the hosts filesystem */
|
|
|
|
|
/*
|
|
|
|
|
* GRUB -- GRand Unified Bootloader
|
2010-01-27 03:18:14 +00:00
|
|
|
|
* Copyright (C) 2007,2008,2009,2010 Free Software Foundation, Inc.
|
2007-08-02 17:24:06 +00:00
|
|
|
|
*
|
|
|
|
|
* 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/>.
|
|
|
|
|
*/
|
2013-04-11 19:12:46 +00:00
|
|
|
|
|
|
|
|
|
#include <config-util.h>
|
|
|
|
|
|
2009-04-05 20:19:05 +00:00
|
|
|
|
#define _BSD_SOURCE
|
2007-08-02 17:24:06 +00:00
|
|
|
|
#include <grub/fs.h>
|
|
|
|
|
#include <grub/file.h>
|
|
|
|
|
#include <grub/disk.h>
|
|
|
|
|
#include <grub/misc.h>
|
2012-02-08 18:26:01 +00:00
|
|
|
|
#include <grub/mm.h>
|
2008-03-30 18:32:15 +00:00
|
|
|
|
#include <grub/dl.h>
|
2008-08-29 19:55:23 +00:00
|
|
|
|
#include <grub/util/misc.h>
|
2012-02-29 14:23:31 +00:00
|
|
|
|
#include <grub/emu/hostdisk.h>
|
2012-02-08 18:26:01 +00:00
|
|
|
|
#include <grub/i18n.h>
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
|
|
|
|
#include <dirent.h>
|
|
|
|
|
#include <stdio.h>
|
2010-01-27 03:18:14 +00:00
|
|
|
|
#include <errno.h>
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
2007-11-18 07:14:26 +00:00
|
|
|
|
|
|
|
|
|
/* dirent.d_type is a BSD extension, not part of POSIX */
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
is_dir (const char *path, const char *name)
|
|
|
|
|
{
|
|
|
|
|
int len1 = strlen(path);
|
|
|
|
|
int len2 = strlen(name);
|
|
|
|
|
|
|
|
|
|
char pathname[len1 + 1 + len2 + 1 + 13];
|
|
|
|
|
strcpy (pathname, path);
|
|
|
|
|
|
|
|
|
|
/* Avoid UNC-path "//name" on Cygwin. */
|
|
|
|
|
if (len1 > 0 && pathname[len1 - 1] != '/')
|
|
|
|
|
strcat (pathname, "/");
|
|
|
|
|
|
|
|
|
|
strcat (pathname, name);
|
|
|
|
|
|
|
|
|
|
struct stat st;
|
|
|
|
|
if (stat (pathname, &st))
|
|
|
|
|
return 0;
|
|
|
|
|
return S_ISDIR (st.st_mode);
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-08 18:26:01 +00:00
|
|
|
|
struct grub_hostfs_data
|
|
|
|
|
{
|
|
|
|
|
char *filename;
|
|
|
|
|
FILE *f;
|
|
|
|
|
};
|
|
|
|
|
|
2007-08-02 17:24:06 +00:00
|
|
|
|
static grub_err_t
|
2009-06-10 21:04:23 +00:00
|
|
|
|
grub_hostfs_dir (grub_device_t device, const char *path,
|
2013-01-21 01:33:46 +00:00
|
|
|
|
grub_fs_dir_hook_t hook, void *hook_data)
|
2007-08-02 17:24:06 +00:00
|
|
|
|
{
|
|
|
|
|
DIR *dir;
|
|
|
|
|
|
|
|
|
|
/* Check if the disk is our dummy disk. */
|
|
|
|
|
if (grub_strcmp (device->disk->name, "host"))
|
|
|
|
|
return grub_error (GRUB_ERR_BAD_FS, "not a hostfs");
|
|
|
|
|
|
|
|
|
|
dir = opendir (path);
|
|
|
|
|
if (! dir)
|
|
|
|
|
return grub_error (GRUB_ERR_BAD_FILENAME,
|
2012-02-08 18:26:01 +00:00
|
|
|
|
N_("can't open `%s': %s"), path,
|
|
|
|
|
strerror (errno));
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
|
|
|
|
struct dirent *de;
|
2009-04-05 20:19:05 +00:00
|
|
|
|
struct grub_dirhook_info info;
|
|
|
|
|
grub_memset (&info, 0, sizeof (info));
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
|
|
|
|
de = readdir (dir);
|
|
|
|
|
if (! de)
|
|
|
|
|
break;
|
|
|
|
|
|
2009-04-05 20:19:05 +00:00
|
|
|
|
info.dir = !! is_dir (path, de->d_name);
|
2013-01-21 01:33:46 +00:00
|
|
|
|
hook (de->d_name, &info, hook_data);
|
2009-04-05 20:19:05 +00:00
|
|
|
|
|
2007-08-02 17:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
closedir (dir);
|
|
|
|
|
|
|
|
|
|
return GRUB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Open a file named NAME and initialize FILE. */
|
|
|
|
|
static grub_err_t
|
|
|
|
|
grub_hostfs_open (struct grub_file *file, const char *name)
|
|
|
|
|
{
|
|
|
|
|
FILE *f;
|
2012-02-08 18:26:01 +00:00
|
|
|
|
struct grub_hostfs_data *data;
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
2008-08-29 19:55:23 +00:00
|
|
|
|
f = fopen (name, "rb");
|
2007-08-02 17:24:06 +00:00
|
|
|
|
if (! f)
|
|
|
|
|
return grub_error (GRUB_ERR_BAD_FILENAME,
|
2012-02-08 18:26:01 +00:00
|
|
|
|
N_("can't open `%s': %s"), name,
|
|
|
|
|
strerror (errno));
|
|
|
|
|
data = grub_malloc (sizeof (*data));
|
|
|
|
|
if (!data)
|
|
|
|
|
{
|
|
|
|
|
fclose (f);
|
|
|
|
|
return grub_errno;
|
|
|
|
|
}
|
|
|
|
|
data->filename = grub_strdup (name);
|
|
|
|
|
if (!data->filename)
|
|
|
|
|
{
|
|
|
|
|
grub_free (data);
|
|
|
|
|
fclose (f);
|
|
|
|
|
return grub_errno;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data->f = f;
|
|
|
|
|
|
|
|
|
|
file->data = data;
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
2013-09-24 17:19:31 +00:00
|
|
|
|
#if defined (__CYGWIN__) || defined (__MINGW32__) || defined (__AROS__)
|
2013-09-23 09:21:09 +00:00
|
|
|
|
fseek (f, 0, SEEK_END);
|
|
|
|
|
file->size = ftello (f);
|
|
|
|
|
fseek (f, 0, SEEK_SET);
|
|
|
|
|
#else
|
2012-02-29 14:23:31 +00:00
|
|
|
|
file->size = grub_util_get_fd_size (fileno (f), name, NULL);
|
2013-09-23 09:21:09 +00:00
|
|
|
|
#endif
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
|
|
|
|
return GRUB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static grub_ssize_t
|
|
|
|
|
grub_hostfs_read (grub_file_t file, char *buf, grub_size_t len)
|
|
|
|
|
{
|
2012-02-08 18:26:01 +00:00
|
|
|
|
struct grub_hostfs_data *data;
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
2012-02-08 18:26:01 +00:00
|
|
|
|
data = file->data;
|
|
|
|
|
if (fseeko (data->f, file->offset, SEEK_SET) != 0)
|
2010-01-27 03:18:14 +00:00
|
|
|
|
{
|
2012-02-08 18:26:01 +00:00
|
|
|
|
grub_error (GRUB_ERR_OUT_OF_RANGE, N_("cannot seek `%s': %s"),
|
|
|
|
|
data->filename, strerror (errno));
|
2010-01-27 03:18:14 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-08 18:26:01 +00:00
|
|
|
|
unsigned int s = fread (buf, 1, len, data->f);
|
2010-01-27 03:18:14 +00:00
|
|
|
|
if (s != len)
|
2012-02-08 18:26:01 +00:00
|
|
|
|
grub_error (GRUB_ERR_FILE_READ_ERROR, N_("cannot read `%s': %s"),
|
|
|
|
|
data->filename, strerror (errno));
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
2010-01-27 03:18:14 +00:00
|
|
|
|
return (signed) s;
|
2007-08-02 17:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static grub_err_t
|
|
|
|
|
grub_hostfs_close (grub_file_t file)
|
|
|
|
|
{
|
2012-02-08 18:26:01 +00:00
|
|
|
|
struct grub_hostfs_data *data;
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
2012-02-08 18:26:01 +00:00
|
|
|
|
data = file->data;
|
|
|
|
|
fclose (data->f);
|
|
|
|
|
grub_free (data->filename);
|
|
|
|
|
grub_free (data);
|
2007-08-02 17:24:06 +00:00
|
|
|
|
|
|
|
|
|
return GRUB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static grub_err_t
|
|
|
|
|
grub_hostfs_label (grub_device_t device __attribute ((unused)),
|
|
|
|
|
char **label __attribute ((unused)))
|
|
|
|
|
{
|
2007-11-18 07:14:26 +00:00
|
|
|
|
*label = 0;
|
2007-08-02 17:24:06 +00:00
|
|
|
|
return GRUB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct grub_fs grub_hostfs_fs =
|
|
|
|
|
{
|
|
|
|
|
.name = "hostfs",
|
|
|
|
|
.dir = grub_hostfs_dir,
|
|
|
|
|
.open = grub_hostfs_open,
|
|
|
|
|
.read = grub_hostfs_read,
|
|
|
|
|
.close = grub_hostfs_close,
|
|
|
|
|
.label = grub_hostfs_label,
|
|
|
|
|
.next = 0
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-03-30 18:32:15 +00:00
|
|
|
|
GRUB_MOD_INIT(hostfs)
|
2007-08-02 17:24:06 +00:00
|
|
|
|
{
|
|
|
|
|
grub_fs_register (&grub_hostfs_fs);
|
|
|
|
|
}
|
|
|
|
|
|
2008-03-30 18:32:15 +00:00
|
|
|
|
GRUB_MOD_FINI(hostfs)
|
2007-08-02 17:24:06 +00:00
|
|
|
|
{
|
|
|
|
|
grub_fs_unregister (&grub_hostfs_fs);
|
|
|
|
|
}
|