diff options
author | stevel@tonic-gate <none@none> | 2005-06-14 00:00:00 -0700 |
---|---|---|
committer | stevel@tonic-gate <none@none> | 2005-06-14 00:00:00 -0700 |
commit | 7c478bd95313f5f23a4c958a745db2134aa03244 (patch) | |
tree | c871e58545497667cbb4b0a4f2daf204743e1fe7 /usr/src/uts/common/fs/cachefs/cachefs_module.c | |
download | illumos-gate-7c478bd95313f5f23a4c958a745db2134aa03244.tar.gz |
OpenSolaris Launch
Diffstat (limited to 'usr/src/uts/common/fs/cachefs/cachefs_module.c')
-rw-r--r-- | usr/src/uts/common/fs/cachefs/cachefs_module.c | 314 |
1 files changed, 314 insertions, 0 deletions
diff --git a/usr/src/uts/common/fs/cachefs/cachefs_module.c b/usr/src/uts/common/fs/cachefs/cachefs_module.c new file mode 100644 index 0000000000..5e68d9f4ce --- /dev/null +++ b/usr/src/uts/common/fs/cachefs/cachefs_module.c @@ -0,0 +1,314 @@ +/* + * CDDL HEADER START + * + * The contents of this file are subject to the terms of the + * Common Development and Distribution License, Version 1.0 only + * (the "License"). You may not use this file except in compliance + * with the License. + * + * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE + * or http://www.opensolaris.org/os/licensing. + * See the License for the specific language governing permissions + * and limitations under the License. + * + * When distributing Covered Code, include this CDDL HEADER in each + * file and include the License file at usr/src/OPENSOLARIS.LICENSE. + * If applicable, add the following below this CDDL HEADER, with the + * fields enclosed by brackets "[]" replaced with your own identifying + * information: Portions Copyright [yyyy] [name of copyright owner] + * + * CDDL HEADER END + */ +/* + * Copyright 2004 Sun Microsystems, Inc. All rights reserved. + * Use is subject to license terms. + */ + +#pragma ident "%Z%%M% %I% %E% SMI" + +#include <sys/errno.h> +#include <sys/param.h> +#include <sys/types.h> +#include <sys/user.h> +#include <sys/stat.h> +#include <sys/time.h> +#include <sys/vfs.h> +#include <sys/vnode.h> +#include <rpc/types.h> +#include <sys/mode.h> +#include <sys/cmn_err.h> +#include <sys/debug.h> +#include <sys/fs/cachefs_fs.h> + +/* + * This is the loadable module wrapper. + */ +#include <sys/systm.h> +#include <sys/modctl.h> +#include <sys/syscall.h> + +extern time_t time; + +static int cachefs_init(int, char *); +static void cachefs_fini(); + +static int cachefs_unloadable = 0; /* tunable */ +static boolean_t cachefs_up = B_FALSE; + +uint_t cachefs_max_apop_inqueue = CACHEFS_MAX_APOP_INQUEUE; + +/* + * this is a list of possible hash table sizes, for the `double + * hashing' algorithm described in rosen's `elementary number theory + * and its applications'. minimally, this needs to be a list of + * increasing prime integers, terminated by a 0. ideally, they should + * be the larger of twin primes; i.e. P and P-2 are both prime. + */ + +int cachefs_hash_sizes[] = {5, 2029, 4093, 8089, 16363, 32719, 0}; + +/* + * Module linkage information for the kernel. + */ + +static vfsdef_t vfs_z = { + VFSDEF_VERSION, + CACHEFS_BASETYPE, + cachefs_init, + VSW_CANREMOUNT, + NULL +}; + +static struct modlfs modlfs = { + &mod_fsops, + "cache filesystem", + &vfs_z +}; + +static struct modlinkage modlinkage = { + MODREV_1, (void *)&modlfs, NULL +}; + +char _depends_on[] = "strmod/rpcmod"; + +int +_init(void) +{ + int status; + + status = mod_install(&modlinkage); + if (status != 0) { + /* + * Could not load module, clean up the work performed + * by cachefs_init() which was indirectly called by + * mod_installfs() which in turn was called by mod_install(). + */ + cachefs_fini(); + } + + return (status); +} + +int +_info(struct modinfo *modinfop) +{ + return (mod_info(&modlinkage, modinfop)); +} + +int +_fini(void) +{ + int status; + + if (!cachefs_unloadable) + return (EBUSY); + + if ((status = mod_remove(&modlinkage)) == 0) { + /* + * Module has been unloaded, now clean up + */ + cachefs_fini(); + } + + return (status); +} + +extern kmutex_t cachefs_cachelock; /* Cache list mutex */ +extern kmutex_t cachefs_newnum_lock; +extern kmutex_t cachefs_kstat_key_lock; +extern kmutex_t cachefs_rename_lock; +extern kmutex_t cachefs_minor_lock; /* Lock for minor device map */ +extern kmutex_t cachefs_kmem_lock; +extern kmutex_t cachefs_async_lock; /* global async work count */ +extern major_t cachefs_major; + +/* + * Cache initialization routine. This routine should only be called + * once. It performs the following tasks: + * - Initalize all global locks + * - Call sub-initialization routines (localize access to variables) + */ +static int +cachefs_init(int fstyp, char *name) +{ + kstat_t *ksp; + int error; + + ASSERT(cachefs_up == B_FALSE); + + error = cachefs_init_vfsops(fstyp); + if (error != 0) + return (error); + + error = cachefs_init_vnops(name); + if (error != 0) + return (error); + + mutex_init(&cachefs_cachelock, NULL, MUTEX_DEFAULT, NULL); + mutex_init(&cachefs_newnum_lock, NULL, MUTEX_DEFAULT, NULL); + mutex_init(&cachefs_kstat_key_lock, NULL, MUTEX_DEFAULT, NULL); + mutex_init(&cachefs_kmem_lock, NULL, MUTEX_DEFAULT, NULL); + mutex_init(&cachefs_rename_lock, NULL, MUTEX_DEFAULT, NULL); + mutex_init(&cachefs_minor_lock, NULL, MUTEX_DEFAULT, NULL); + mutex_init(&cachefs_async_lock, NULL, MUTEX_DEFAULT, NULL); +#ifdef CFSRLDEBUG + mutex_init(&cachefs_rl_debug_mutex, NULL, MUTEX_DEFAULT, NULL); +#endif /* CFSRLDEBUG */ + + /* + * set up kmem_cache entities + */ + + cachefs_cnode_cache = kmem_cache_create("cachefs_cnode_cache", + sizeof (struct cnode), 0, NULL, NULL, NULL, NULL, NULL, 0); + cachefs_req_cache = kmem_cache_create("cachefs_async_request", + sizeof (struct cachefs_req), 0, + cachefs_req_create, cachefs_req_destroy, NULL, NULL, NULL, 0); + cachefs_fscache_cache = kmem_cache_create("cachefs_fscache", + sizeof (fscache_t), 0, NULL, NULL, NULL, NULL, NULL, 0); + cachefs_filegrp_cache = kmem_cache_create("cachefs_filegrp", + sizeof (filegrp_t), 0, + filegrp_cache_create, filegrp_cache_destroy, NULL, NULL, NULL, 0); + cachefs_cache_kmcache = kmem_cache_create("cachefs_cache_t", + sizeof (cachefscache_t), 0, NULL, NULL, NULL, NULL, NULL, 0); + + /* + * set up the cachefs.0.key kstat + */ + + cachefs_kstat_key = NULL; + cachefs_kstat_key_n = 0; + ksp = kstat_create("cachefs", 0, "key", "misc", KSTAT_TYPE_RAW, 1, + KSTAT_FLAG_VIRTUAL | KSTAT_FLAG_VAR_SIZE); + if (ksp != NULL) { + ksp->ks_data = &cachefs_kstat_key; + ksp->ks_update = cachefs_kstat_key_update; + ksp->ks_snapshot = cachefs_kstat_key_snapshot; + ksp->ks_lock = &cachefs_kstat_key_lock; + kstat_install(ksp); + } + + /* + * Assign unique major number for all nfs mounts + */ + + if ((cachefs_major = getudev()) == -1) { + cmn_err(CE_WARN, + "cachefs: init: can't get unique device number"); + cachefs_major = 0; + } + cachefs_up = B_TRUE; +#ifdef CFSRLDEBUG + cachefs_dbvalid = time; +#endif /* CFSRLDEBUG */ + + return (0); +} + +/* + * Cache clean up routine. This routine is called if mod_install() failed + * and we have to clean up because the module could not be installed, + * or by _fini() when we're unloading the module. + */ +static void +cachefs_fini() +{ + extern int cachefsfstyp; + extern struct vnodeops *cachefs_vnodeops; + + if (cachefs_up == B_FALSE) { + /* + * cachefs_init() was not called on _init(), + * nothing to deallocate. + */ + return; + } + + /* + * Clean up cachefs.0.key kstat. + * Currently, you can only do a + * modunload if cachefs_unloadable is nonzero, and that's + * pretty much just for debugging. however, if there ever + * comes a day when cachefs is more freely unloadable + * (e.g. the modunload daemon can do it normally), then we'll + * have to make changes in the stats_ API. this is because a + * stats_cookie_t holds the id # derived from here, and it + * will all go away at modunload time. thus, the API will + * need to somehow be more robust than is currently necessary. + */ + kstat_delete_byname("cachefs", 0, "key"); + + if (cachefs_kstat_key != NULL) { + cachefs_kstat_key_t *key; + int i; + + for (i = 0; i < cachefs_kstat_key_n; i++) { + key = cachefs_kstat_key + i; + + cachefs_kmem_free((void *)(uintptr_t)key->ks_mountpoint, + strlen((char *)(uintptr_t)key->ks_mountpoint) + 1); + cachefs_kmem_free((void *)(uintptr_t)key->ks_backfs, + strlen((char *)(uintptr_t)key->ks_backfs) + 1); + cachefs_kmem_free((void *)(uintptr_t)key->ks_cachedir, + strlen((char *)(uintptr_t)key->ks_cachedir) + 1); + cachefs_kmem_free((void *)(uintptr_t)key->ks_cacheid, + strlen((char *)(uintptr_t)key->ks_cacheid) + 1); + } + + cachefs_kmem_free(cachefs_kstat_key, + cachefs_kstat_key_n * sizeof (*cachefs_kstat_key)); + } + + /* + * Clean up kmem_cache entities + */ + kmem_cache_destroy(cachefs_cache_kmcache); + kmem_cache_destroy(cachefs_filegrp_cache); + kmem_cache_destroy(cachefs_fscache_cache); + kmem_cache_destroy(cachefs_req_cache); + kmem_cache_destroy(cachefs_cnode_cache); +#ifdef CFSRLDEBUG + if (cachefs_rl_debug_cache != NULL) + kmem_cache_destroy(cachefs_rl_debug_cache); +#endif /* CFSRLDEBUG */ + + /* + * Clean up the operations structures + */ + (void) vfs_freevfsops_by_type(cachefsfstyp); + vn_freevnodeops(cachefs_vnodeops); + + /* + * Destroy mutexes + */ +#ifdef CFSRLDEBUG + mutex_destroy(&cachefs_rl_debug_mutex); +#endif /* CFSRLDEBUG */ + mutex_destroy(&cachefs_async_lock); + mutex_destroy(&cachefs_minor_lock); + mutex_destroy(&cachefs_rename_lock); + mutex_destroy(&cachefs_kmem_lock); + mutex_destroy(&cachefs_kstat_key_lock); + mutex_destroy(&cachefs_newnum_lock); + mutex_destroy(&cachefs_cachelock); +} |