summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--trans/magic.c359
1 files changed, 272 insertions, 87 deletions
diff --git a/trans/magic.c b/trans/magic.c
index 2febff68..2e47f911 100644
--- a/trans/magic.c
+++ b/trans/magic.c
@@ -1,6 +1,6 @@
/* A translator for returning FS_RETRY_MAGIC strings.
- Copyright (C) 1999,2001,02 Free Software Foundation, Inc.
+ Copyright (C) 1999,2001,02, 03 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License as
@@ -19,6 +19,7 @@
#include <hurd.h>
#include <hurd/ports.h>
#include <hurd/trivfs.h>
+#include <hurd/fshelp.h>
#include <hurd/fsys.h>
#include <version.h>
@@ -36,23 +37,38 @@
#include <assert.h>
const char *argp_program_version = STANDARD_HURD_VERSION (magic);
-static char args_doc[] = "MAGIC";
-static char doc[] = "A translator that returns the magic retry result MAGIC.";
-static const struct argp_option options[] =
+
+/* This structure has all the state about one filesystem.
+ It hangs off trivfs_control->hook. */
+struct magic
{
- {"directory", 'd', 0, 0, "Provide virtual (empty) directory node"},
- {0}
-};
+ /* We chain all filesystems together so we can tell easily when they are
+ all unused. */
+ struct trivfs_control *next;
+
+ /* The magic string we return for lookups. */
+ char *magic;
-/* The magic string we return for lookups. */
-static char *magic;
+ int directory; /* --directory flag */
-static int directory; /* --directory flag */
+ /* Pre-fab contents of dummy directory for dir_readdir.
+ Set up only under --directory. */
+ void *dirbuf;
+ size_t dirbufsize;
-/* Pre-fab contents of dummy directory for dir_readdir.
- Set up only under --directory. */
-static void *dirbuf;
-static size_t dirbufsize;
+ unsigned int nusers; /* Count of users, only with --directory. */
+};
+
+static inline void
+free_magic (struct magic *m)
+{
+ free (m->magic);
+ if (m->dirbuf)
+ munmap (m->dirbuf, m->dirbufsize);
+ free (m);
+}
+
+static struct trivfs_control *all_fsys;
/* Trivfs hooks */
@@ -68,7 +84,9 @@ int trivfs_allow_open = O_READ;
void
trivfs_modify_stat (struct trivfs_protid *cred, struct stat *st)
{
- st->st_size = dirbufsize;
+ struct magic *const m = cred->po->cntl->hook;
+
+ st->st_size = m->dirbufsize;
st->st_blocks = getpagesize () / S_BLKSIZE;
st->st_mode = ((st->st_mode & ~S_IFMT & ~ALLPERMS)
@@ -79,9 +97,45 @@ trivfs_modify_stat (struct trivfs_protid *cred, struct stat *st)
error_t
trivfs_goaway (struct trivfs_control *fsys, int flags)
{
- exit (0);
+ struct magic *const m = fsys->hook;
+
+ /* We are single-threaded, so no fancy stuff is needed here. */
+
+ if (m->nusers > 0 && !(flags & FSYS_GOAWAY_FORCE))
+ return EBUSY;
+
+ /* No more communication with the parent filesystem.
+ This running RPC should now be the only ref keeping FSYS alive. */
+ ports_destroy_right (fsys);
+ return 0;
}
+
+/* Clean pointers in a struct trivfs_control when its last reference
+ vanishes before it's freed. This overrides the libtrivfs version
+ so we can clean up our hook data. */
+void
+trivfs_clean_cntl (void *arg)
+{
+ struct trivfs_control *cntl = arg;
+ struct magic *const m = cntl->hook;
+
+ /* Remove us from the list of all filesystems. */
+ struct trivfs_control **lastp = &all_fsys;
+ while (*lastp != cntl)
+ lastp = &((struct magic *) (*lastp)->hook)->next;
+ *lastp = m->next;
+
+ if (all_fsys == 0)
+ /* Nothing more to do in this life. */
+ exit (0);
+
+ mach_port_destroy (mach_task_self (), cntl->filesys_id);
+ mach_port_destroy (mach_task_self (), cntl->file_id);
+ mach_port_deallocate (mach_task_self (), cntl->underlying);
+
+ free_magic (m);
+}
/* This hook is used when running without --directory;
it circumvents basically all the trivfs machinery. */
@@ -96,7 +150,12 @@ magic_getroot (struct trivfs_control *cntl,
retry_type *do_retry, char *retry_name,
mach_port_t *node, mach_msg_type_name_t *node_type)
{
- strcpy (retry_name, magic);
+ struct magic *const m = cntl->hook;
+
+ if (m->directory)
+ return EAGAIN; /* Do normal trivfs getroot processing. */
+
+ strcpy (retry_name, m->magic);
*do_retry = FS_RETRY_MAGICAL;
*node = MACH_PORT_NULL;
*node_type = MACH_MSG_TYPE_COPY_SEND;
@@ -125,10 +184,20 @@ magic_open (struct trivfs_control *cntl,
assert_perror (err);
err = mach_port_deallocate (mach_task_self (), dotdot);
assert_perror (err);
+ struct magic *const m = cntl->hook;
+ m->nusers++;
}
return err;
}
+/* We have this hook only for simple tracking of the live user ports. */
+static void
+magic_protid_destroy (struct trivfs_protid *cred)
+{
+ struct magic *const m = cred->po->cntl->hook;
+ m->nusers--;
+}
+
/* Do a directory lookup. */
@@ -154,44 +223,49 @@ trivfs_S_dir_lookup (struct trivfs_protid *cred,
if (name[0] != '\0')
{
- if (!directory)
+ struct magic *const m = cred->po->cntl->hook;
+
+ if (!m->directory)
return ENOTDIR;
- else
- {
- /* We have a real lookup in the dummy directory.
- Handle `.' and `..' specially, and anything else
- gets redirected to the magical retry. */
+ /* We have a real lookup in the dummy directory.
+ Handle `.' and `..' specially, and anything else
+ gets redirected to the magical retry. */
+
+ while (*name == '/')
+ ++name;
+ while (!strncmp (name, "./", 2))
+ {
+ name += 2;
while (*name == '/')
++name;
- while (!strncmp (name, "./", 2))
- {
- name += 2;
- while (*name == '/')
- ++name;
- }
+ }
- if (!strcmp (name, "..") || !strncmp (name, "../", 3))
- {
- name += 2;
- while (*name == '/')
- ++name;
- strcpy (retry_name, name);
- *retry_type = FS_RETRY_REAUTH;
- *retrypt = (mach_port_t) cred->po->hook;
- *retrypt_type = MACH_MSG_TYPE_COPY_SEND;
- return 0;
- }
- else if (name[0] != '\0' && strcmp (name, "."))
+ if (!strcmp (name, "..") || !strncmp (name, "../", 3))
+ {
+ name += 2;
+ while (*name == '/')
+ ++name;
+ strcpy (retry_name, name);
+ *retry_type = FS_RETRY_REAUTH;
+ *retrypt = (mach_port_t) cred->po->hook;
+ *retrypt_type = MACH_MSG_TYPE_COPY_SEND;
+ return 0;
+ }
+ else if (name[0] != '\0' && strcmp (name, "."))
+ {
+ if (m->magic == 0)
+ strcpy (retry_name, name);
+ else
{
- char *p = stpcpy (retry_name, magic);
+ char *p = stpcpy (retry_name, m->magic);
*p++ = '/';
strcpy (p, name);
- *retry_type = FS_RETRY_MAGICAL;
- *retrypt = MACH_PORT_NULL;
- *retrypt_type = MACH_MSG_TYPE_COPY_SEND;
- return 0;
}
+ *retry_type = FS_RETRY_MAGICAL;
+ *retrypt = MACH_PORT_NULL;
+ *retrypt_type = MACH_MSG_TYPE_COPY_SEND;
+ return 0;
}
}
@@ -253,23 +327,25 @@ trivfs_S_dir_readdir (struct trivfs_protid *cred,
if (!cred)
return EOPNOTSUPP;
+ struct magic *const m = cred->po->cntl->hook;
+
if (entry > 0)
{
void *p;
int i;
i = 0;
- for (p = dirbuf; p < dirbuf + dirbufsize;
+ for (p = m->dirbuf; p < m->dirbuf + m->dirbufsize;
p += ((struct dirent *) p)->d_reclen)
if (i++ == entry)
break;
*data = p;
- *datalen = dirbuf + dirbufsize - p;
+ *datalen = m->dirbuf + m->dirbufsize - p;
*amount = 2 - entry;
}
else
{
- *data = dirbuf;
- *datalen = dirbufsize;
+ *data = m->dirbuf;
+ *datalen = m->dirbufsize;
*amount = 2;
}
@@ -278,32 +354,71 @@ trivfs_S_dir_readdir (struct trivfs_protid *cred,
}
+#include <hurd/paths.h>
+#define _SERVERS_MAGIC _SERVERS "magic"
+
+/* To whom should we try to delegate on startup? */
+static const char *delegate = _SERVERS_MAGIC;
+
+static const struct argp_option options[] =
+{
+ {"directory", 'd', 0, 0, "Provide virtual (empty) directory node"},
+ {"use-server", 'U', "NAME", 0,
+ "Delegate to server NAME instead of " _SERVERS_MAGIC},
+ {0}
+};
static error_t
parse_opt (int opt, char *arg, struct argp_state *state)
{
+ struct magic *const m = state->input;
+
switch (opt)
{
- case 'd':
- directory = 1;
+ case 'U':
+ /* This is only valid for the startup options, not delegates. */
+ if (all_fsys != 0)
+ return EINVAL;
+ delegate = arg;
return 0;
+ case 'd':
case ARGP_KEY_NO_ARGS:
- argp_usage (state);
- return EINVAL;
+ m->directory = 1;
+ return 0;
- case ARGP_KEY_ARGS:
- if (state->next != state->argc - 1)
+ case ARGP_KEY_ARG:
+ if (m->magic != 0)
{
argp_usage (state);
return EINVAL;
}
- else
+ m->magic = strdup (arg);
+ return m->magic == 0 ? ENOMEM : 0;
+
+ case ARGP_KEY_SUCCESS:
+ if (m->directory)
{
- magic = state->argv[state->next];
- return 0;
+ inline struct dirent *add (struct dirent *d, const char *name)
+ {
+ d->d_fileno = 2; /* random */
+ d->d_type = DT_DIR;
+ d->d_namlen = strlen (name);
+ strcpy (d->d_name, name);
+ d->d_name[d->d_namlen] = '\0';
+ d->d_reclen = &d->d_name[d->d_namlen + 1] - (char *) d;
+ d->d_reclen = ((d->d_reclen + __alignof (struct dirent) - 1)
+ & ~(__alignof (struct dirent) - 1));
+ return (struct dirent *) ((char *) d + d->d_reclen);
+ }
+ struct dirent *d;
+ m->dirbuf = mmap (0, getpagesize (), PROT_READ|PROT_WRITE,
+ MAP_ANON, 0, 0);
+ d = add (m->dirbuf, ".");
+ d = add (d, "..");
+ m->dirbufsize = (char *) d - (char *) m->dirbuf;
}
- break;
+ return 0;
}
return ARGP_ERR_UNKNOWN;
@@ -313,9 +428,16 @@ error_t
trivfs_append_args (struct trivfs_control *fsys,
char **argz, size_t *argz_len)
{
- return ((directory ? argz_add (argz, argz_len, "--directory") : 0)
- ?: argz_add (argz, argz_len, magic));
+ struct magic *const m = fsys->hook;
+ return ((m->directory ? argz_add (argz, argz_len, "--directory") : 0)
+ ?: (m->magic ? argz_add (argz, argz_len, m->magic) : 0));
}
+
+static struct argp argp =
+ {
+ options, parse_opt, "MAGIC",
+ "A translator that returns the magic retry result MAGIC."
+ };
int
main (int argc, char **argv)
@@ -323,48 +445,111 @@ main (int argc, char **argv)
error_t err;
mach_port_t bootstrap;
struct trivfs_control *fsys;
- struct argp argp = { options, parse_opt, args_doc, doc };
+ struct magic *m = calloc (1, sizeof *m);
- argp_parse (&argp, argc, argv, 0, 0, 0);
+ argp_parse (&argp, argc, argv, 0, 0, m);
task_get_bootstrap_port (mach_task_self (), &bootstrap);
if (bootstrap == MACH_PORT_NULL)
error (1, 0, "Must be started as a translator");
+ if (delegate != 0)
+ {
+ /* First, try to have the canonical server sitting on /servers/magic
+ take over for us. */
+ err = fshelp_delegate_translation (delegate, bootstrap, argv);
+ if (err == 0)
+ return 0;
+ }
+
+ /* Nope, we are doing it ourselves. */
+
+ trivfs_getroot_hook = &magic_getroot;
+ trivfs_open_hook = &magic_open;
+ trivfs_protid_destroy_hook = &magic_protid_destroy;
+
/* Reply to our parent */
err = trivfs_startup (bootstrap, 0, 0, 0, 0, 0, &fsys);
mach_port_deallocate (mach_task_self (), bootstrap);
if (err)
error (3, err, "Contacting parent");
+ fsys->hook = m;
+ all_fsys = fsys;
- if (directory)
+ /* Launch. */
+ while (1)
{
- inline struct dirent *add (struct dirent *d, const char *name)
+ ports_manage_port_operations_one_thread (fsys->pi.bucket, trivfs_demuxer,
+ 10 * 60 * 1000);
+
+ /* That returns when 10 minutes pass without an RPC. Try shutting down
+ as if sent fsys_goaway; if we have any users who need us to stay
+ around, this returns EBUSY and we loop to service more RPCs. */
+
+ struct trivfs_control *fs = all_fsys;
+ do
{
- d->d_fileno = 2; /* random */
- d->d_type = DT_DIR;
- d->d_namlen = strlen (name);
- strcpy (d->d_name, name);
- d->d_name[d->d_namlen] = '\0';
- d->d_reclen = &d->d_name[d->d_namlen + 1] - (char *) d;
- d->d_reclen = ((d->d_reclen + __alignof (struct dirent) - 1)
- & ~(__alignof (struct dirent) - 1));
- return (struct dirent *) ((char *) d + d->d_reclen);
- }
- struct dirent *d;
- dirbuf = mmap (0, getpagesize (), PROT_READ|PROT_WRITE, MAP_ANON, 0, 0);
- d = add (dirbuf, ".");
- d = add (d, "..");
- dirbufsize = (char *) d - (char *) dirbuf;
+ struct magic *const m = fs->hook;
+ struct trivfs_control *const next = m->next;
+ trivfs_goaway (fs, 0);
+ fs = next;
+ } while (fs != 0);
+ }
+
+ return 0;
+}
+
- trivfs_open_hook = &magic_open;
+/* Handle delegated filesystems. */
+error_t
+trivfs_S_fsys_forward (mach_port_t server,
+ mach_port_t reply,
+ mach_msg_type_name_t replytype,
+ mach_port_t requestor,
+ char *argz, size_t argz_len)
+{
+ struct trivfs_protid *cred
+ = ports_lookup_port (all_fsys->pi.bucket, server,
+ trivfs_protid_portclasses[0]);
+ if (!cred)
+ return EOPNOTSUPP;
+ ports_port_deref (cred);
+
+ /* Allocate a new structure for parameters, and parse the arguments
+ to fill it in. */
+ struct magic *m = calloc (1, sizeof *m);
+ if (!m)
+ return ENOMEM;
+
+ int argc = argz_count (argz, argz_len);
+ char *argv[argc + 1];
+ argz_extract (argz, argz_len, argv);
+ error_t err = argp_parse (&argp, argc, argv,
+ ARGP_NO_ERRS | ARGP_NO_HELP, 0, m);
+ if (err)
+ {
+ free_magic (m);
+ return err;
}
- else
- trivfs_getroot_hook = &magic_getroot;
- /* Launch. */
- ports_manage_port_operations_one_thread (fsys->pi.bucket, trivfs_demuxer,
- 2 * 60 * 1000);
+ /* Now we are ready to start up the filesystem. Contact the parent. */
+ struct trivfs_control *fsys;
+ err = trivfs_startup (requestor, 0,
+ trivfs_cntl_portclasses[0], all_fsys->pi.bucket,
+ trivfs_protid_portclasses[0], all_fsys->pi.bucket,
+ &fsys);
+ if (err)
+ {
+ free_magic (m);
+ return err;
+ }
+ mach_port_deallocate (mach_task_self (), requestor);
+
+ /* The new filesystem is all hooked up.
+ Put it on the list of all filesystems we are serving. */
+ m->next = all_fsys;
+ fsys->hook = m;
+ all_fsys = fsys;
return 0;
}