summaryrefslogtreecommitdiffstats
path: root/sys/fs/pseudofs/pseudofs_fileno.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/fs/pseudofs/pseudofs_fileno.c')
-rw-r--r--sys/fs/pseudofs/pseudofs_fileno.c285
1 files changed, 285 insertions, 0 deletions
diff --git a/sys/fs/pseudofs/pseudofs_fileno.c b/sys/fs/pseudofs/pseudofs_fileno.c
new file mode 100644
index 0000000..877b7b8
--- /dev/null
+++ b/sys/fs/pseudofs/pseudofs_fileno.c
@@ -0,0 +1,285 @@
+/*-
+ * Copyright (c) 2001 Dag-Erling Coïdan Smørgrav
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer
+ * in this position and unchanged.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ *
+ * $FreeBSD$
+ */
+
+#include <sys/param.h>
+#include <sys/kernel.h>
+#include <sys/systm.h>
+#include <sys/malloc.h>
+#include <sys/mount.h>
+#include <sys/proc.h>
+#include <sys/sbuf.h>
+#include <sys/sysctl.h>
+
+#include <machine/limits.h>
+
+#include <fs/pseudofs/pseudofs.h>
+#include <fs/pseudofs/pseudofs_internal.h>
+
+static MALLOC_DEFINE(M_PFSFILENO, "pseudofs_fileno", "pseudofs fileno bitmap");
+
+static struct mtx pfs_fileno_mutex;
+
+#define PFS_BITMAP_SIZE 4096
+#define PFS_SLOT_BITS (sizeof(unsigned int) * CHAR_BIT)
+#define PFS_BITMAP_BITS (PFS_BITMAP_SIZE * PFS_SLOT_BITS)
+struct pfs_bitmap {
+ u_int32_t pb_offset;
+ int pb_used;
+ unsigned int pb_bitmap[PFS_BITMAP_SIZE];
+ struct pfs_bitmap *pb_next;
+};
+
+/*
+ * Initialization
+ */
+void
+pfs_fileno_load(void)
+{
+ mtx_init(&pfs_fileno_mutex, "pseudofs_fileno", MTX_DEF);
+}
+
+/*
+ * Teardown
+ */
+void
+pfs_fileno_unload(void)
+{
+ mtx_destroy(&pfs_fileno_mutex);
+}
+
+/*
+ * Initialize fileno bitmap
+ */
+void
+pfs_fileno_init(struct pfs_info *pi)
+{
+ struct pfs_bitmap *pb;
+
+ MALLOC(pb, struct pfs_bitmap *, sizeof *pb,
+ M_PFSFILENO, M_WAITOK|M_ZERO);
+
+ mtx_lock(&pi->pi_mutex);
+
+ pb->pb_bitmap[0] = 07;
+ pb->pb_used = 3;
+ pi->pi_bitmap = pb;
+ pi->pi_root->pn_fileno = 2;
+
+ mtx_unlock(&pi->pi_mutex);
+}
+
+/*
+ * Tear down fileno bitmap
+ */
+void
+pfs_fileno_uninit(struct pfs_info *pi)
+{
+ struct pfs_bitmap *pb, *npb;
+ int used;
+
+ mtx_lock(&pi->pi_mutex);
+
+ pb = pi->pi_bitmap;
+ pi->pi_bitmap = NULL;
+
+ mtx_unlock(&pi->pi_mutex);
+
+ for (used = 0; pb; pb = npb) {
+ npb = pb->pb_next;
+ used += pb->pb_used;
+ FREE(pb, M_PFSFILENO);
+ }
+ if (used > 2)
+ printf("WARNING: %d file numbers still in use\n", used);
+}
+
+/*
+ * Get the next available file number
+ */
+static u_int32_t
+pfs_get_fileno(struct pfs_info *pi)
+{
+ struct pfs_bitmap *pb, *ppb;
+ u_int32_t fileno;
+ unsigned int *p;
+ int i;
+
+ mtx_lock(&pi->pi_mutex);
+
+ /* look for the first page with free bits */
+ for (ppb = NULL, pb = pi->pi_bitmap; pb; ppb = pb, pb = pb->pb_next)
+ if (pb->pb_used != PFS_BITMAP_BITS)
+ break;
+
+ /* out of pages? */
+ if (pb == NULL) {
+ mtx_unlock(&pi->pi_mutex);
+ MALLOC(pb, struct pfs_bitmap *, sizeof *pb,
+ M_PFSFILENO, M_WAITOK|M_ZERO);
+ mtx_lock(&pi->pi_mutex);
+ /* protect against possible race */
+ while (ppb->pb_next)
+ ppb = ppb->pb_next;
+ pb->pb_offset = ppb->pb_offset + PFS_BITMAP_BITS;
+ ppb->pb_next = pb;
+ }
+
+ /* find the first free slot */
+ for (i = 0; i < PFS_BITMAP_SIZE; ++i)
+ if (pb->pb_bitmap[i] != UINT_MAX)
+ break;
+
+ /* find the first available bit and flip it */
+ fileno = pb->pb_offset + i * PFS_SLOT_BITS;
+ p = &pb->pb_bitmap[i];
+ for (i = 0; i < PFS_SLOT_BITS; ++i, ++fileno)
+ if ((*p & (unsigned int)(1 << i)) == 0)
+ break;
+ KASSERT(i < PFS_SLOT_BITS,
+ ("slot has free bits, yet doesn't"));
+ *p |= (unsigned int)(1 << i);
+ ++pb->pb_used;
+
+ mtx_unlock(&pi->pi_mutex);
+
+ return fileno;
+}
+
+/*
+ * Free a file number
+ */
+static void
+pfs_free_fileno(struct pfs_info *pi, u_int32_t fileno)
+{
+ struct pfs_bitmap *pb;
+ unsigned int *p;
+ int i;
+
+ mtx_lock(&pi->pi_mutex);
+
+ /* find the right page */
+ for (pb = pi->pi_bitmap;
+ pb && fileno >= PFS_BITMAP_BITS;
+ pb = pb->pb_next, fileno -= PFS_BITMAP_BITS)
+ /* nothing */ ;
+ KASSERT(pb,
+ ("fileno isn't in any bitmap"));
+
+ /* find the right bit in the right slot and flip it */
+ p = &pb->pb_bitmap[fileno / PFS_SLOT_BITS];
+ i = fileno % PFS_SLOT_BITS;
+ KASSERT(*p & (unsigned int)(1 << i),
+ ("fileno is already free"));
+ *p &= ~((unsigned int)(1 << i));
+ --pb->pb_used;
+
+ mtx_unlock(&pi->pi_mutex);
+}
+
+/*
+ * Allocate a file number
+ */
+void
+pfs_fileno_alloc(struct pfs_info *pi, struct pfs_node *pn)
+{
+ /* make sure our parent has a file number */
+ if (pn->pn_parent && !pn->pn_parent->pn_fileno)
+ pfs_fileno_alloc(pi, pn->pn_parent);
+
+ switch (pn->pn_type) {
+ case pfstype_root:
+ case pfstype_dir:
+ case pfstype_file:
+ case pfstype_symlink:
+ pn->pn_fileno = pfs_get_fileno(pi);
+ break;
+ case pfstype_this:
+ KASSERT(pn->pn_parent != NULL,
+ ("pfstype_this node has no parent"));
+ pn->pn_fileno = pn->pn_parent->pn_fileno;
+ break;
+ case pfstype_parent:
+ KASSERT(pn->pn_parent != NULL,
+ ("pfstype_parent node has no parent"));
+ if (pn->pn_parent == pi->pi_root) {
+ pn->pn_fileno = pn->pn_parent->pn_fileno;
+ break;
+ }
+ KASSERT(pn->pn_parent->pn_parent != NULL,
+ ("pfstype_parent node has no grandparent"));
+ pn->pn_fileno = pn->pn_parent->pn_parent->pn_fileno;
+ break;
+ case pfstype_procdep:
+ KASSERT(1,
+ ("pfs_fileno_alloc() called for pfstype_procdep node"));
+ break;
+ case pfstype_none:
+ KASSERT(1,
+ ("pfs_fileno_alloc() called for pfstype_none node"));
+ break;
+ }
+
+ printf("pfs_fileno_alloc(): %s: ", pi->pi_name);
+ if (pn->pn_parent) {
+ if (pn->pn_parent->pn_parent) {
+ printf("%s/", pn->pn_parent->pn_parent->pn_name);
+ }
+ printf("%s/", pn->pn_parent->pn_name);
+ }
+ printf("%s -> %d\n", pn->pn_name, pn->pn_fileno);
+}
+
+/*
+ * Release a file number
+ */
+void
+pfs_fileno_free(struct pfs_info *pi, struct pfs_node *pn)
+{
+ switch (pn->pn_type) {
+ case pfstype_root:
+ case pfstype_dir:
+ case pfstype_file:
+ case pfstype_symlink:
+ pfs_free_fileno(pi, pn->pn_fileno);
+ break;
+ case pfstype_this:
+ case pfstype_parent:
+ /* ignore these, as they don't "own" their file number */
+ break;
+ case pfstype_procdep:
+ KASSERT(1,
+ ("pfs_fileno_free() called for pfstype_procdep node"));
+ break;
+ case pfstype_none:
+ KASSERT(1,
+ ("pfs_fileno_free() called for pfstype_none node"));
+ break;
+ }
+}
OpenPOWER on IntegriCloud