diff options
-rw-r--r-- | libtrivfs/open.c | 102 | ||||
-rw-r--r-- | libtrivfs/protid-dup.c | 71 |
2 files changed, 173 insertions, 0 deletions
diff --git a/libtrivfs/open.c b/libtrivfs/open.c new file mode 100644 index 00000000..a9bc62fe --- /dev/null +++ b/libtrivfs/open.c @@ -0,0 +1,102 @@ +/* Make a new trivfs peropen/protid + + Copyright (C) 1993, 1994, 1995 Free Software Foundation, Inc. + + 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 the GNU Hurd; see the file COPYING. If not, write to + the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ + +#include <string.h> /* For bcopy() */ + +#include "priv.h" + +/* Return a new protid pointing to a new peropen in CRED, with REALNODE as + the underlying node reference, with the given identity, and open flags in + FLAGS. CNTL is the trivfs control object. */ +error_t +trivfs_open (struct trivfs_control *cntl, + uid_t *uids, unsigned num_uids, gid_t *gids, unsigned num_gids, + unsigned flags, + mach_port_t realnode, + struct trivfs_protid **cred) +{ + error_t err = 0; + struct trivfs_peropen *po = malloc (sizeof (struct trivfs_peropen)); + + if (!po) + return ENOMEM; + + ports_port_ref (cntl); + + po->refcnt = 1; + po->cntl = cntl; + po->openmodes = flags; + po->hook = 0; + + if (trivfs_peropen_create_hook) + err = (*trivfs_peropen_create_hook) (po); + if (!err) + { + struct trivfs_protid *new = + ports_allocate_port (cntl->protid_bucket, + sizeof (struct trivfs_protid), + cntl->protid_class); + + if (new) + { + int i; + + new->isroot = 0; + for (i = 0; i < num_uids; i++) + if (uids[i] == 0) + new->isroot = 1; + + new->uids = malloc (num_uids * sizeof (uid_t)); + bcopy (uids, new->uids, num_uids * sizeof (uid_t)); + new->nuids = num_uids; + + new->gids = malloc (num_gids * sizeof (uid_t)); + bcopy (gids, new->gids, num_gids * sizeof (uid_t)); + new->ngids = num_gids; + + new->po = po; + new->hook = 0; + new->realnode = realnode; + + if (!err && trivfs_protid_create_hook) + err = (*trivfs_protid_create_hook) (new); + + if (err) + { + /* Setting REALNODE to null signals the clean routine not to + call the destroy hook, which we deallocate below anyway. */ + new->realnode = MACH_PORT_NULL; + ports_port_deref (new); + } + else + *cred = new; + } + else + err = ENOMEM; + } + + if (err) + { + ports_port_deref (cntl); + free (po); + } + + return err; +} diff --git a/libtrivfs/protid-dup.c b/libtrivfs/protid-dup.c new file mode 100644 index 00000000..3ac6f24b --- /dev/null +++ b/libtrivfs/protid-dup.c @@ -0,0 +1,71 @@ +/* Duplicate a protid + + Copyright (C) 1993, 1994, 1995 Free Software Foundation, Inc. + + 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 the GNU Hurd; see the file COPYING. If not, write to + the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ + +#include <string.h> /* For bcopy() */ + +#include "priv.h" + +/* Return a duplicate of CRED in DUP, sharing the same peropen and hook. A + non-null hook may be used to detect that this is a duplicate by + trivfs_peropen_create_hook. */ +error_t +trivfs_protid_dup (struct trivfs_protid *cred, struct trivfs_protid **dup) +{ + error_t err = 0; + struct trivfs_protid *new = + ports_allocate_port (cred->po->cntl->protid_bucket, + sizeof (struct trivfs_protid), + cred->po->cntl->protid_class); + + mutex_lock (&cred->po->cntl->lock); + new->po = cred->po; + new->po->refcnt++; + mutex_unlock (&cred->po->cntl->lock); + + new->isroot = cred->isroot; + + new->uids = malloc (cred->nuids * sizeof (uid_t)); + bcopy (cred->uids, new->uids, cred->nuids * sizeof (uid_t)); + new->nuids = cred->nuids; + + new->gids = malloc (cred->ngids * sizeof (gid_t)); + bcopy (cred->gids, new->gids, cred->ngids * sizeof (uid_t)); + new->ngids = cred->ngids; + + new->realnode = cred->realnode; + mach_port_mod_refs (mach_task_self (), new->realnode, + MACH_PORT_RIGHT_SEND, 1); + + new->hook = cred->hook; + + if (trivfs_protid_create_hook) + err = (*trivfs_protid_create_hook) (new); + if (err) + { + mach_port_deallocate (mach_task_self (), new->realnode); + /* Signal that the user destroy hook shouldn't be called on NEW. */ + new->realnode = MACH_PORT_NULL; + ports_port_deref (new); + } + else + *dup = new; + + return err; +} |