summaryrefslogtreecommitdiff
path: root/libfshelp
diff options
context:
space:
mode:
Diffstat (limited to 'libfshelp')
-rw-r--r--libfshelp/fetch-root.c158
1 files changed, 118 insertions, 40 deletions
diff --git a/libfshelp/fetch-root.c b/libfshelp/fetch-root.c
index b4a88e12..7a48be42 100644
--- a/libfshelp/fetch-root.c
+++ b/libfshelp/fetch-root.c
@@ -1,5 +1,5 @@
/*
- Copyright (C) 1994 Free Software Foundation, Inc.
+ Copyright (C) 1995 Free Software Foundation, Inc.
Written by Michael I. Bushnell.
This file is part of the GNU Hurd.
@@ -19,53 +19,131 @@
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
#include "trans.h"
+#include <unistd.h>
+#include <assert.h>
+#include <string.h>
+#include <hurd/fsys.h>
-/* Implement fshelp_fetch_root; see <hurd/fshelp.h> for the description. */
error_t
-fshelp_fetch_root (struct trans_link *link, mach_port_t *cntl,
- int passive, error_t (*passive_fn) (char **, u_int *),
- void *dirpt, void *nodept,
- mach_port_t (*genpt_fn) (mach_port_t),
- struct mutex *unlock, uid_t uid, gid_t gid)
+fshelp_fetch_root (struct transbox *box, file_t dotdot,
+ uid_t *uids, int uids_len,
+ uid_t *gids, int gids_len,
+ int flags, fshelp_callback_t callback,
+ retry_type *retry, char *retryname,
+ file_t *root)
{
- mach_port_t cwdirpt, nodept;
- char buf[1000];
- char *bufp = buf;
- u_int buflen = 1000;
+ error_t err;
+ mach_port_t control;
+
+ do
+ {
+ if (box->active == MACH_PORT_NULL)
+ do
+ {
+ mach_port_t underlying;
+ uid_t uid, gid;
+ char *argz;
+ int argz_len;
+ error_t err;
+ mach_port_t ports[INIT_PORT_MAX];
+ int ints[INIT_INT_MAX];
+ mach_port_t fds[STDERR_FILENO + 1];
+ auth_t ourauth, newauth;
+ int uidarray[2], gidarray[2];
+
+ mach_port_t
+ reauth (mach_port_t port)
+ {
+ mach_port_t rend, ret;
+ error_t err;
- mutex_lock (&link->lock);
+ rend = mach_reply_port ();
+ err = io_reauthenticate (port, rend,
+ MACH_MSG_TYPE_MAKE_SEND);
+ assert_perror (err);
- if (!passive && link->control == MACH_PORT_NULL)
- {
- mutex_unlock (&link->lock);
- return 0;
- }
-
- cwdirpt = (*genpt_fn)(dirpt);
-
- if (link->control == MACH_PORT_NULL)
- {
- /* Start passive translator */
- nodept = (*genpt_fn)(nodept);
- error = (*passive_fn) (&bufp, buflen);
-
- if (error)
- {
- mutex_unlock (&link->lock);
- return error;
- }
+ err = auth_user_authenticate (newauth, port, rend,
+ MACH_MSG_TYPE_MAKE_SEND,
+ &ret);
+ if (err)
+ ret = MACH_PORT_NULL;
+
+ mach_port_destroy (mach_task_self (), rend);
+ mach_port_deallocate (mach_task_self (), port);
+ return ret;
+ }
+
+ err = (*callback) (box->cookie, &underlying, &uid,
+ &gid, &argz, &argz_len);
+ if (err)
+ return err;
- mutex_unlock (unlock);
-
- error = fshelp_start_translator (link, bufp, buflen, cwdirpt,
- nodept, uid, gid);
+ mutex_lock (&box->innerlock); /* cancellation point XXX */
+ mutex_unlock (box->lock);
+
+ ourauth = getauth ();
+ uidarray[0] = uidarray[1] = uid;
+ gidarray[0] = gidarray[1] = gid;
+ err = auth_makeauth (ourauth, MACH_PORT_NULL, 0,
+ MACH_MSG_TYPE_MAKE_SEND,
+ uidarray, 1, uidarray, 2,
+ gidarray, 1, gidarray, 2, &newauth);
+ assert_perror (err);
- if (bufp != buf)
- vm_deallocate (mach_task_self (), (vm_address_t) bufp, buflen);
+ bzero (ports, INIT_PORT_MAX * sizeof (mach_port_t));
+ bzero (fds, (STDERR_FILENO + 1) * sizeof (mach_port_t));
+ bzero (ints, INIT_INT_MAX * sizeof (int));
- if (error)
- return error;
+ ports[INIT_PORT_CWDIR] = reauth (getcwdir ());
+ ports[INIT_PORT_CRDIR] = reauth (getcrdir ());
+ ports[INIT_PORT_AUTH] = newauth;
+
+ fds[STDERR_FILENO] = reauth (getdport (STDERR_FILENO));
+
+ err = fshelp_start_translator_long (reauth (underlying),
+ MACH_MSG_TYPE_MOVE_SEND,
+ argz, argz, argz_len,
+ fds, STDERR_FILENO + 1,
+ MACH_MSG_TYPE_MOVE_SEND,
+ ports, INIT_PORT_MAX,
+ MACH_MSG_TYPE_MOVE_SEND,
+ ints, INIT_INT_MAX,
+ 0, &box->active);
+ mutex_unlock (&box->innerlock);
+
+ if (err)
+ return err;
+
+ mutex_lock (box->lock);
+
+ /* The loop here is because between the interlock unlock and
+ the reacquire of the node lock, someone might have deallocated
+ the port we just added. */
+ }
+ while (box->active == MACH_PORT_NULL);
+
+ control = box->active;
+ mach_port_mod_refs (mach_task_self (), control,
+ MACH_PORT_RIGHT_SEND, 1);
+ mutex_unlock (box->lock);
+
+ /* Cancellation point XXX */
+ err = fsys_getroot (control, dotdot, MACH_MSG_TYPE_COPY_SEND,
+ uids, uids_len, gids, gids_len, flags,
+ retry, retryname, root);
+
+ mutex_lock (box->lock);
+
+ if ((err == MACH_SEND_INVALID_DEST || err == MIG_SERVER_DIED)
+ && control == box->active)
+ fshelp_set_active (box, MACH_PORT_NULL);
+ mach_port_deallocate (mach_task_self (), control);
}
+ while (err == MACH_SEND_INVALID_DEST || err == MIG_SERVER_DIED);
+ mach_port_deallocate (mach_task_self (), dotdot);
+ return err;
+}
-
+
+