aboutsummaryrefslogtreecommitdiff
path: root/libtrivfs/protid-clean.c
diff options
context:
space:
mode:
Diffstat (limited to 'libtrivfs/protid-clean.c')
-rw-r--r--libtrivfs/protid-clean.c54
1 files changed, 54 insertions, 0 deletions
diff --git a/libtrivfs/protid-clean.c b/libtrivfs/protid-clean.c
new file mode 100644
index 00000000..b76b202e
--- /dev/null
+++ b/libtrivfs/protid-clean.c
@@ -0,0 +1,54 @@
+/*
+ Copyright (C) 1994, 1995, 1996 Free Software Foundation
+
+ This program 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.
+
+ This program 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., 675 Mass Ave, Cambridge, MA 02139, USA. */
+
+#include "priv.h"
+
+/* Clean pointers in a struct trivfs_protid when its last reference vanishes
+ before it's freed. */
+void
+trivfs_clean_protid (void *arg)
+{
+ struct trivfs_protid *cred = arg;
+
+ if (trivfs_protid_destroy_hook && cred->realnode != MACH_PORT_NULL)
+ /* Allow the user to clean up; If the realnode field is null, then CRED
+ wasn't initialized to the point of needing user cleanup. */
+ (*trivfs_protid_destroy_hook) (cred);
+
+ /* If we hold the only reference to the peropen, try to get rid of it. */
+ mutex_lock (&cred->po->cntl->lock);
+ if (cred->po->refcnt == 1 && trivfs_peropen_destroy_hook)
+ {
+ mutex_unlock (&cred->po->cntl->lock);
+ (*trivfs_peropen_destroy_hook) (cred->po);
+ mutex_lock (&cred->po->cntl->lock);
+ }
+ if (--cred->po->refcnt == 0)
+ {
+ ports_port_deref (cred->po->cntl);
+ free (cred->po);
+ }
+ mutex_unlock (&cred->po->cntl->lock);
+
+ iohelp_free_iouser (cred->user);
+
+ if (cred->realnode != MACH_PORT_NULL)
+ mach_port_deallocate (mach_task_self (), cred->realnode);
+}
+
+
+