summaryrefslogtreecommitdiff
path: root/usermux/mux.c
diff options
context:
space:
mode:
Diffstat (limited to 'usermux/mux.c')
-rw-r--r--usermux/mux.c492
1 files changed, 492 insertions, 0 deletions
diff --git a/usermux/mux.c b/usermux/mux.c
new file mode 100644
index 00000000..932361fd
--- /dev/null
+++ b/usermux/mux.c
@@ -0,0 +1,492 @@
+/* Root usermux node
+
+ Copyright (C) 1997 Free Software Foundation, Inc.
+ Written by Miles Bader <miles@gnu.ai.mit.edu>
+ This file is part of the GNU Hurd.
+
+ The GNU Hurd 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 2, or (at
+ your option) any later version.
+
+ The GNU Hurd 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 this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA. */
+
+#include <stddef.h>
+#include <string.h>
+#include <dirent.h>
+#include <pwd.h>
+
+#include "usermux.h"
+
+/* The granularity with which we allocate space to return our result. */
+#define DIRENTS_CHUNK_SIZE (128*1024)/* Enough for perhaps 8000 names. */
+
+/* The number seconds we cache our directory return value, in seconds. */
+#define DIRENTS_CACHE_TIME 90
+
+/* Returned directory entries are aligned to blocks this many bytes long.
+ Must be a power of two. */
+#define DIRENT_ALIGN 4
+#define DIRENT_NAME_OFFS offsetof (struct dirent, d_name)
+
+/* Length is structure before the name + the name + '\0', all
+ padded to a four-byte alignment. */
+#define DIRENT_LEN(name_len) \
+ ((DIRENT_NAME_OFFS + (name_len) + 1 + (DIRENT_ALIGN - 1)) \
+ & ~(DIRENT_ALIGN - 1))
+
+static error_t lookup_user (struct usermux *mux, const char *user,
+ struct node **node); /* fwd decl */
+
+/* [root] Directory operations. */
+
+/* Lookup NAME in DIR for USER; set *NODE to the found name upon return. If
+ the name was not found, then return ENOENT. On any error, clear *NODE.
+ (*NODE, if found, should be locked, this call should unlock DIR no matter
+ what.) */
+error_t
+netfs_attempt_lookup (struct iouser *user, struct node *dir,
+ char *name, struct node **node)
+{
+ error_t err;
+
+ if (dir->nn->name)
+ err = ENOTDIR;
+ else
+ err = lookup_user (dir->nn->mux, name, node);
+
+ fshelp_touch (&dir->nn_stat, TOUCH_ATIME, usermux_maptime);
+
+ mutex_unlock (&dir->lock);
+
+ if (! err)
+ mutex_lock (&(*node)->lock);
+
+ return err;
+}
+
+/* Fetch a directory of user entries, as for netfs_get_dirents (that function
+ is actually a wrapper that caches the results for a while). */
+static error_t
+get_dirents (struct node *dir,
+ int first_entry, int max_entries, char **data,
+ mach_msg_type_number_t *data_len,
+ vm_size_t max_data_len, int *data_entries)
+{
+ error_t err = 0;
+
+ if (dir->nn->name)
+ return ENOTDIR;
+
+ /* Start scanning. */
+ setpwent ();
+
+ /* Find the first entry. */
+ while (first_entry-- > 0)
+ if (! getpwent ())
+ {
+ max_entries = 0;
+ break;
+ }
+
+ if (max_entries != 0)
+ {
+ size_t size = (max_data_len == 0 ? DIRENTS_CHUNK_SIZE : max_data_len);
+
+ err = vm_allocate (mach_task_self (), (vm_address_t *) data, size, 1);
+
+ if (! err)
+ {
+ struct passwd *pw;
+ char *p = *data;
+ int count = 0;
+ int entry_type =
+ (S_ISLNK (dir->nn->mux->stat_template.st_mode) ? DT_LNK : DT_REG);
+
+ /* See how much space we need for the result. */
+ while ((max_entries == -1 || count < max_entries)
+ && (pw = getpwent ()))
+ {
+ struct dirent hdr;
+ size_t name_len = strlen (pw->pw_name);
+ size_t sz = DIRENT_LEN (name_len);
+
+ if ((p - *data) + sz > size)
+ if (max_data_len > 0)
+ break;
+ else
+ /* Try to grow our return buffer. */
+ {
+ vm_address_t extension = (vm_address_t)(*data + size);
+ err = vm_allocate (mach_task_self (), &extension,
+ DIRENTS_CHUNK_SIZE, 0);
+ if (err)
+ break;
+ size += DIRENTS_CHUNK_SIZE;
+ }
+
+ hdr.d_namlen = name_len;
+ hdr.d_fileno = pw->pw_uid + USERMUX_FILENO_UID_OFFSET;
+ hdr.d_reclen = sz;
+ hdr.d_type = entry_type;
+
+ memcpy (p, &hdr, DIRENT_NAME_OFFS);
+ strcpy (p + DIRENT_NAME_OFFS, pw->pw_name);
+ p += sz;
+
+ count++;
+ }
+
+ if (err)
+ vm_deallocate (mach_task_self (), (vm_address_t)*data, size);
+ else
+ {
+ vm_address_t alloc_end = (vm_address_t)(*data + size);
+ vm_address_t real_end = round_page (p);
+ if (alloc_end > real_end)
+ vm_deallocate (mach_task_self (),
+ real_end, alloc_end - real_end);
+ *data_len = p - *data;
+ *data_entries = count;
+ }
+ }
+ }
+
+ endpwent ();
+
+ return err;
+}
+
+/* Implement the netfs_get_directs callback as described in
+ <hurd/netfs.h>. */
+error_t
+netfs_get_dirents (struct iouser *cred, struct node *dir,
+ int first_entry, int max_entries, char **data,
+ mach_msg_type_number_t *data_len,
+ vm_size_t max_data_len, int *data_entries)
+{
+ error_t err;
+ static time_t cache_timestamp = 0;
+ static struct rwlock cache_lock = RWLOCK_INITIALIZER;
+ static char *cached_data = 0;
+ static mach_msg_type_number_t cached_data_len = 0;
+ static int cached_data_entries = 0;
+ struct timeval tv;
+ char *first;
+ size_t bytes_left, entries_left;
+
+ maptime_read (usermux_maptime, &tv);
+ if (tv.tv_sec > cache_timestamp + DIRENTS_CACHE_TIME)
+ {
+ rwlock_writer_lock (&cache_lock);
+
+ if (cached_data_len > 0)
+ /* Free the old cache. */
+ {
+ vm_deallocate (mach_task_self (),
+ (vm_address_t)cached_data, cached_data_len);
+ cached_data = 0;
+ cached_data_len = 0;
+ }
+
+ err = get_dirents (dir, 0, -1, &cached_data, &cached_data_len, 0,
+ &cached_data_entries);
+
+ if (! err)
+ cache_timestamp = tv.tv_sec;
+
+ rwlock_writer_unlock (&cache_lock);
+
+ if (err)
+ return err;
+ }
+
+ rwlock_reader_lock (&cache_lock);
+
+ first = cached_data;
+ bytes_left = cached_data_len;
+ entries_left = cached_data_entries;
+
+ while (first_entry > 0)
+ {
+ struct dirent *e = (struct dirent *)first;
+
+ if (entries_left == 0)
+ {
+ rwlock_reader_unlock (&cache_lock);
+ return EINVAL;
+ }
+
+ first += e->d_reclen;
+ bytes_left -= e->d_reclen;
+ entries_left--;
+ }
+
+ if ((max_data_len > 0 && max_data_len < bytes_left)
+ || (max_entries > 0 && max_entries < entries_left))
+ /* If there's some limit on the return value, we can't just use our
+ values representing the whole cache, so we have to explicitly count
+ how much we're going to return. */
+ {
+ char *lim = first;
+ int entries = 0;
+
+ while (entries_left > 0
+ && max_entries > 0
+ && max_data_len > ((struct dirent *)lim)->d_reclen)
+ {
+ size_t reclen = ((struct dirent *)lim)->d_reclen;
+ max_data_len -= reclen;
+ max_entries--;
+ entries++;
+ lim += reclen;
+ }
+
+ bytes_left = (lim - first);
+ entries_left = entries;
+ }
+
+ *data_len = bytes_left;
+ *data_entries = entries_left;
+
+ err = vm_allocate (mach_task_self (), (vm_address_t *)data, bytes_left, 1);
+ if (! err)
+ bcopy (cached_data, *data, bytes_left);
+
+ rwlock_reader_unlock (&cache_lock);
+
+ fshelp_touch (&dir->nn_stat, TOUCH_ATIME, usermux_maptime);
+
+ return err;
+}
+
+/* User lookup. */
+
+/* Free storage allocated consumed by the host mux name NM, but not the node
+ it points to. */
+static void
+free_name (struct usermux_name *nm)
+{
+ free ((char *)nm->name);
+ free (nm);
+}
+
+/* See if there's an existing entry for the name USER, and if so, return its
+ node in NODE with an additional references. True is returned iff the
+ lookup succeeds. If PURGE is true, then any nodes with a null node are
+ removed. */
+static int
+lookup_cached (struct usermux *mux, const char *user, int purge,
+ struct node **node)
+{
+ struct usermux_name *nm = mux->names, **prevl = &mux->names;
+
+ while (nm)
+ {
+ struct usermux_name *next = nm->next;
+
+ if (strcasecmp (user, nm->name) == 0)
+ {
+ spin_lock (&netfs_node_refcnt_lock);
+ if (nm->node)
+ nm->node->references++;
+ spin_unlock (&netfs_node_refcnt_lock);
+
+ if (nm->node)
+ {
+ *node = nm->node;
+ return 1;
+ }
+ }
+
+ if (purge && !nm->node)
+ {
+ *prevl = nm->next;
+ free_name (nm);
+ }
+ else
+ prevl = &nm->next;
+
+ nm = next;
+ }
+
+ return 0;
+}
+
+/* See if there's an existing entry for the name USER, and if so, return its
+ node in NODE, with an additional reference, otherwise, create a new node
+ for the user HE as referred to by USER, and return that instead, with a
+ single reference. The type of node created is either a translator node,
+ if USER refers to the official name of the user, or a symlink node to the
+ official name, if it doesn't. */
+static error_t
+lookup_pwent (struct usermux *mux, const char *user, struct passwd *pw,
+ struct node **node)
+{
+ error_t err;
+ struct usermux_name *nm = malloc (sizeof (struct usermux_name));
+
+ if (! nm)
+ return ENOMEM;
+
+ nm->name = strdup (user);
+ err = create_user_node (mux, nm, pw, node);
+ if (err)
+ {
+ free_name (nm);
+ return err;
+ }
+
+ rwlock_writer_lock (&mux->names_lock);
+ if (lookup_cached (mux, user, 1, node))
+ /* An entry for USER has already been created between the time we last
+ looked and now (which is possible because we didn't lock MUX).
+ Just throw away our version and return the one already in the cache. */
+ {
+ rwlock_writer_unlock (&mux->names_lock);
+ nm->node->nn->name = 0; /* Avoid touching the mux name list. */
+ netfs_nrele (nm->node); /* Free the tentative new node. */
+ free_name (nm); /* And the name it was under. */
+ }
+ else
+ /* Enter NM into MUX's list of names, and return the new node. */
+ {
+ nm->next = mux->names;
+ mux->names = nm;
+ rwlock_writer_unlock (&mux->names_lock);
+ }
+
+ return 0;
+}
+
+/* Lookup the user USER in MUX, and return the resulting node in NODE, with
+ an additional reference, or an error. */
+static error_t
+lookup_user (struct usermux *mux, const char *user, struct node **node)
+{
+ int was_cached;
+ struct passwd _pw, *pw;
+ char pwent_data[2048]; /* XXX what size should this be???? */
+
+ rwlock_reader_lock (&mux->names_lock);
+ was_cached = lookup_cached (mux, user, 0, node);
+ rwlock_reader_unlock (&mux->names_lock);
+
+ if (was_cached)
+ return 0;
+ else if (getpwnam_r (user, &_pw, pwent_data, sizeof pwent_data, &pw) == 0)
+ return lookup_pwent (mux, user, pw, node);
+ else
+ return ENOENT;
+}
+
+/* This should sync the entire remote filesystem. If WAIT is set, return
+ only after sync is completely finished. */
+error_t
+netfs_attempt_syncfs (struct iouser *cred, int wait)
+{
+ return 0;
+}
+
+/* This should attempt a chmod call for the user specified by CRED on node
+ NODE, to change the owner to UID and the group to GID. */
+error_t
+netfs_attempt_chown (struct iouser *cred, struct node *node, uid_t uid, uid_t gid)
+{
+ if (node->nn->name)
+ return EOPNOTSUPP;
+ else
+ {
+ struct usermux *mux = node->nn->mux;
+ error_t err = file_chown (mux->underlying, uid, gid);
+
+ if (! err)
+ {
+ struct usermux_name *nm;
+
+ /* Change NODE's owner. */
+ mux->stat_template.st_uid = uid;
+ mux->stat_template.st_gid = gid;
+ node->nn_stat.st_uid = uid;
+ node->nn_stat.st_gid = gid;
+
+ /* Change the owner of each leaf node. */
+ rwlock_reader_lock (&mux->names_lock);
+ for (nm = mux->names; nm; nm = nm->next)
+ if (nm->node)
+ {
+ nm->node->nn_stat.st_uid = uid;
+ nm->node->nn_stat.st_gid = gid;
+ }
+ rwlock_reader_unlock (&mux->names_lock);
+
+ fshelp_touch (&node->nn_stat, TOUCH_CTIME, usermux_maptime);
+ }
+
+ return err;
+ }
+}
+
+/* This should attempt a chauthor call for the user specified by CRED on node
+ NODE, to change the author to AUTHOR. */
+error_t
+netfs_attempt_chauthor (struct iouser *cred, struct node *node, uid_t author)
+{
+ if (node->nn->name)
+ return EOPNOTSUPP;
+ else
+ {
+ struct usermux *mux = node->nn->mux;
+ error_t err = file_chauthor (mux->underlying, author);
+
+ if (! err)
+ {
+ struct usermux_name *nm;
+
+ /* Change NODE's owner. */
+ mux->stat_template.st_author = author;
+ node->nn_stat.st_author = author;
+
+ /* Change the owner of each leaf node. */
+ rwlock_reader_lock (&mux->names_lock);
+ for (nm = mux->names; nm; nm = nm->next)
+ if (nm->node)
+ nm->node->nn_stat.st_author = author;
+ rwlock_reader_unlock (&mux->names_lock);
+
+ fshelp_touch (&node->nn_stat, TOUCH_CTIME, usermux_maptime);
+ }
+
+ return err;
+ }
+}
+
+/* This should attempt a chmod call for the user specified by CRED on node
+ NODE, to change the mode to MODE. Unlike the normal Unix and Hurd meaning
+ of chmod, this function is also used to attempt to change files into other
+ types. If such a transition is attempted which is impossible, then return
+ EOPNOTSUPP. */
+error_t
+netfs_attempt_chmod (struct iouser *cred, struct node *node, mode_t mode)
+{
+ mode &= ~S_ITRANS;
+ if (node->nn->name || ((mode & S_IFMT) != (node->nn_stat.st_mode & S_IFMT)))
+ return EOPNOTSUPP;
+ else
+ {
+ error_t err = file_chmod (node->nn->mux->underlying, mode & ~S_IFMT);
+ if (! err)
+ {
+ node->nn_stat.st_mode = mode;
+ fshelp_touch (&node->nn_stat, TOUCH_CTIME, usermux_maptime);
+ }
+ return err;
+ }
+}