2015-07-17 17:38:13 +03:00
/*
* ( C ) 2001 Clemson University and The University of Chicago
*
* Changes by Acxiom Corporation to add proc file handler for pvfs2 client
* parameters , Copyright Acxiom Corporation , 2005.
*
* See COPYING in top - level directory .
*/
# include "protocol.h"
2015-12-04 20:56:14 +03:00
# include "orangefs-kernel.h"
# include "orangefs-debugfs.h"
# include "orangefs-sysfs.h"
2015-07-17 17:38:13 +03:00
2015-11-24 23:12:14 +03:00
/* ORANGEFS_VERSION is a ./configure define */
# ifndef ORANGEFS_VERSION
2016-01-13 19:34:59 +03:00
# define ORANGEFS_VERSION "upstream"
2015-07-17 17:38:13 +03:00
# endif
/*
* global variables declared here
*/
/* array of client debug keyword/mask values */
struct client_debug_mask * cdm_array ;
int cdm_element_count ;
2015-11-24 23:12:14 +03:00
char kernel_debug_string [ ORANGEFS_MAX_DEBUG_STRING_LEN ] = " none " ;
char client_debug_string [ ORANGEFS_MAX_DEBUG_STRING_LEN ] ;
char client_debug_array_string [ ORANGEFS_MAX_DEBUG_STRING_LEN ] ;
2015-07-17 17:38:13 +03:00
char * debug_help_string ;
int help_string_initialized ;
struct dentry * help_file_dentry ;
struct dentry * client_debug_dentry ;
struct dentry * debug_dir ;
int client_verbose_index ;
int client_all_index ;
2015-11-24 23:12:14 +03:00
struct orangefs_stats g_orangefs_stats ;
2015-07-17 17:38:13 +03:00
/* the size of the hash tables for ops in progress */
int hash_table_size = 509 ;
static ulong module_parm_debug_mask ;
__u64 gossip_debug_mask ;
struct client_debug_mask client_debug_mask = { NULL , 0 , 0 } ;
unsigned int kernel_mask_set_mod_init ; /* implicitly false */
2015-11-24 23:12:14 +03:00
int op_timeout_secs = ORANGEFS_DEFAULT_OP_TIMEOUT_SECS ;
int slot_timeout_secs = ORANGEFS_DEFAULT_SLOT_TIMEOUT_SECS ;
2015-07-17 17:38:13 +03:00
MODULE_LICENSE ( " GPL " ) ;
2015-11-24 23:12:14 +03:00
MODULE_AUTHOR ( " ORANGEFS Development Team " ) ;
MODULE_DESCRIPTION ( " The Linux Kernel VFS interface to ORANGEFS " ) ;
MODULE_PARM_DESC ( module_parm_debug_mask , " debugging level (see orangefs-debug.h for values) " ) ;
2015-07-17 17:38:13 +03:00
MODULE_PARM_DESC ( op_timeout_secs , " Operation timeout in seconds " ) ;
MODULE_PARM_DESC ( slot_timeout_secs , " Slot timeout in seconds " ) ;
MODULE_PARM_DESC ( hash_table_size ,
" size of hash table for operations in progress " ) ;
2015-11-24 23:12:14 +03:00
static struct file_system_type orangefs_fs_type = {
2015-07-17 17:38:13 +03:00
. name = " pvfs2 " ,
2015-11-24 23:12:14 +03:00
. mount = orangefs_mount ,
. kill_sb = orangefs_kill_sb ,
2015-07-17 17:38:13 +03:00
. owner = THIS_MODULE ,
} ;
module_param ( hash_table_size , int , 0 ) ;
2015-08-03 07:08:59 +03:00
module_param ( module_parm_debug_mask , ulong , 0644 ) ;
2015-07-17 17:38:13 +03:00
module_param ( op_timeout_secs , int , 0 ) ;
module_param ( slot_timeout_secs , int , 0 ) ;
/* synchronizes the request device file */
2016-01-19 19:33:40 +03:00
DEFINE_MUTEX ( devreq_mutex ) ;
2015-07-17 17:38:13 +03:00
/*
2015-10-05 20:44:24 +03:00
* Blocks non - priority requests from being queued for servicing . This
* could be used for protecting the request list data structure , but
* for now it ' s only being used to stall the op addition to the request
* list
*/
2016-01-19 19:33:40 +03:00
DEFINE_MUTEX ( request_mutex ) ;
2015-07-17 17:38:13 +03:00
/* hash table for storing operations waiting for matching downcall */
struct list_head * htable_ops_in_progress ;
DEFINE_SPINLOCK ( htable_ops_in_progress_lock ) ;
/* list for queueing upcall operations */
2015-11-24 23:12:14 +03:00
LIST_HEAD ( orangefs_request_list ) ;
2015-07-17 17:38:13 +03:00
2015-11-24 23:12:14 +03:00
/* used to protect the above orangefs_request_list */
DEFINE_SPINLOCK ( orangefs_request_list_lock ) ;
2015-07-17 17:38:13 +03:00
/* used for incoming request notification */
2015-11-24 23:12:14 +03:00
DECLARE_WAIT_QUEUE_HEAD ( orangefs_request_list_waitq ) ;
2015-07-17 17:38:13 +03:00
2015-11-24 23:12:14 +03:00
static int __init orangefs_init ( void )
2015-07-17 17:38:13 +03:00
{
int ret = - 1 ;
__u32 i = 0 ;
/* convert input debug mask to a 64-bit unsigned integer */
gossip_debug_mask = ( unsigned long long ) module_parm_debug_mask ;
/*
* set the kernel ' s gossip debug string ; invalid mask values will
* be ignored .
*/
debug_mask_to_string ( & gossip_debug_mask , 0 ) ;
/* remove any invalid values from the mask */
debug_string_to_mask ( kernel_debug_string , & gossip_debug_mask , 0 ) ;
/*
* if the mask has a non - zero value , then indicate that the mask
2015-11-24 23:12:14 +03:00
* was set when the kernel module was loaded . The orangefs dev ioctl
2015-07-17 17:38:13 +03:00
* command will look at this boolean to determine if the kernel ' s
* debug mask should be overwritten when the client - core is started .
*/
if ( gossip_debug_mask ! = 0 )
kernel_mask_set_mod_init = true ;
/* print information message to the system log */
2015-11-24 23:12:14 +03:00
pr_info ( " orangefs: orangefs_init called with debug mask: :%s: :%llx: \n " ,
2015-07-17 17:38:13 +03:00
kernel_debug_string ,
( unsigned long long ) gossip_debug_mask ) ;
2015-11-24 23:12:14 +03:00
ret = bdi_init ( & orangefs_backing_dev_info ) ;
2015-07-17 17:38:13 +03:00
if ( ret )
return ret ;
if ( op_timeout_secs < 0 )
op_timeout_secs = 0 ;
if ( slot_timeout_secs < 0 )
slot_timeout_secs = 0 ;
/* initialize global book keeping data structures */
ret = op_cache_initialize ( ) ;
if ( ret < 0 )
goto err ;
ret = dev_req_cache_initialize ( ) ;
if ( ret < 0 )
goto cleanup_op ;
2015-11-24 23:12:14 +03:00
ret = orangefs_inode_cache_initialize ( ) ;
2015-07-17 17:38:13 +03:00
if ( ret < 0 )
goto cleanup_req ;
ret = kiocb_cache_initialize ( ) ;
if ( ret < 0 )
goto cleanup_inode ;
2015-12-04 20:56:14 +03:00
/* Initialize the orangefsdev subsystem. */
2015-11-24 23:12:14 +03:00
ret = orangefs_dev_init ( ) ;
2015-07-17 17:38:13 +03:00
if ( ret < 0 ) {
2015-11-24 23:12:14 +03:00
gossip_err ( " orangefs: could not initialize device subsystem %d! \n " ,
2015-07-17 17:38:13 +03:00
ret ) ;
goto cleanup_kiocb ;
}
htable_ops_in_progress =
kcalloc ( hash_table_size , sizeof ( struct list_head ) , GFP_KERNEL ) ;
if ( ! htable_ops_in_progress ) {
gossip_err ( " Failed to initialize op hashtable " ) ;
ret = - ENOMEM ;
goto cleanup_device ;
}
/* initialize a doubly linked at each hash table index */
for ( i = 0 ; i < hash_table_size ; i + + )
INIT_LIST_HEAD ( & htable_ops_in_progress [ i ] ) ;
ret = fsid_key_table_initialize ( ) ;
if ( ret < 0 )
goto cleanup_progress_table ;
/*
* Build the contents of / sys / kernel / debug / orangefs / debug - help
* from the keywords in the kernel keyword / mask array .
*
* The keywords in the client keyword / mask array are
* unknown at boot time .
*
* orangefs_prepare_debugfs_help_string will be used again
* later to rebuild the debug - help file after the client starts
* and passes along the needed info . The argument signifies
* which time orangefs_prepare_debugfs_help_string is being
* called .
*
*/
ret = orangefs_prepare_debugfs_help_string ( 1 ) ;
if ( ret )
goto out ;
2015-11-24 23:12:14 +03:00
orangefs_debugfs_init ( ) ;
orangefs_kernel_debug_init ( ) ;
2015-07-17 17:38:13 +03:00
orangefs_sysfs_init ( ) ;
2015-11-24 23:12:14 +03:00
ret = register_filesystem ( & orangefs_fs_type ) ;
2015-07-17 17:38:13 +03:00
if ( ret = = 0 ) {
2015-11-24 23:12:14 +03:00
pr_info ( " orangefs: module version %s loaded \n " , ORANGEFS_VERSION ) ;
2015-07-17 17:38:13 +03:00
return 0 ;
}
2015-11-24 23:12:14 +03:00
orangefs_debugfs_cleanup ( ) ;
2015-07-17 17:38:13 +03:00
orangefs_sysfs_exit ( ) ;
fsid_key_table_finalize ( ) ;
cleanup_progress_table :
kfree ( htable_ops_in_progress ) ;
cleanup_device :
2015-11-24 23:12:14 +03:00
orangefs_dev_cleanup ( ) ;
2015-07-17 17:38:13 +03:00
cleanup_kiocb :
kiocb_cache_finalize ( ) ;
cleanup_inode :
2015-11-24 23:12:14 +03:00
orangefs_inode_cache_finalize ( ) ;
2015-07-17 17:38:13 +03:00
cleanup_req :
dev_req_cache_finalize ( ) ;
cleanup_op :
op_cache_finalize ( ) ;
err :
2015-11-24 23:12:14 +03:00
bdi_destroy ( & orangefs_backing_dev_info ) ;
2015-07-17 17:38:13 +03:00
out :
return ret ;
}
2015-11-24 23:12:14 +03:00
static void __exit orangefs_exit ( void )
2015-07-17 17:38:13 +03:00
{
int i = 0 ;
2015-11-24 23:12:14 +03:00
gossip_debug ( GOSSIP_INIT_DEBUG , " orangefs: orangefs_exit called \n " ) ;
2015-07-17 17:38:13 +03:00
2015-11-24 23:12:14 +03:00
unregister_filesystem ( & orangefs_fs_type ) ;
orangefs_debugfs_cleanup ( ) ;
2015-07-17 17:38:13 +03:00
orangefs_sysfs_exit ( ) ;
fsid_key_table_finalize ( ) ;
2015-11-24 23:12:14 +03:00
orangefs_dev_cleanup ( ) ;
2016-01-22 21:34:32 +03:00
BUG_ON ( ! list_empty ( & orangefs_request_list ) ) ;
2015-07-17 17:38:13 +03:00
for ( i = 0 ; i < hash_table_size ; i + + )
2016-01-22 21:34:32 +03:00
BUG_ON ( ! list_empty ( & htable_ops_in_progress [ i ] ) ) ;
2015-07-17 17:38:13 +03:00
kiocb_cache_finalize ( ) ;
2015-11-24 23:12:14 +03:00
orangefs_inode_cache_finalize ( ) ;
2015-07-17 17:38:13 +03:00
dev_req_cache_finalize ( ) ;
op_cache_finalize ( ) ;
kfree ( htable_ops_in_progress ) ;
2015-11-24 23:12:14 +03:00
bdi_destroy ( & orangefs_backing_dev_info ) ;
2015-07-17 17:38:13 +03:00
2015-11-24 23:12:14 +03:00
pr_info ( " orangefs: module version %s unloaded \n " , ORANGEFS_VERSION ) ;
2015-07-17 17:38:13 +03:00
}
/*
* What we do in this function is to walk the list of operations
* that are in progress in the hash table and mark them as purged as well .
*/
void purge_inprogress_ops ( void )
{
int i ;
for ( i = 0 ; i < hash_table_size ; i + + ) {
2015-11-24 23:12:14 +03:00
struct orangefs_kernel_op_s * op ;
struct orangefs_kernel_op_s * next ;
2015-07-17 17:38:13 +03:00
2016-01-23 03:47:47 +03:00
spin_lock ( & htable_ops_in_progress_lock ) ;
2015-07-17 17:38:13 +03:00
list_for_each_entry_safe ( op ,
next ,
& htable_ops_in_progress [ i ] ,
list ) {
spin_lock ( & op - > lock ) ;
gossip_debug ( GOSSIP_INIT_DEBUG ,
" pvfs2-client-core: purging in-progress op tag "
" %llu %s \n " ,
llu ( op - > tag ) ,
get_opname_string ( op ) ) ;
set_op_state_purged ( op ) ;
spin_unlock ( & op - > lock ) ;
}
2016-01-23 03:47:47 +03:00
spin_unlock ( & htable_ops_in_progress_lock ) ;
2015-07-17 17:38:13 +03:00
}
}
2015-11-24 23:12:14 +03:00
module_init ( orangefs_init ) ;
module_exit ( orangefs_exit ) ;