2007-09-06 12:34:25 -04:00
/*
* Common code for control of lockd and nfsv4 grace periods .
2014-09-12 16:40:20 -04:00
*
* Transplanted from lockd code
2007-09-06 12:34:25 -04:00
*/
# include <linux/module.h>
2012-07-25 16:57:13 +04:00
# include <net/net_namespace.h>
2014-09-12 16:40:20 -04:00
# include <net/netns/generic.h>
# include <linux/fs.h>
2012-07-25 16:57:13 +04:00
2014-09-12 16:40:20 -04:00
static int grace_net_id ;
2007-09-06 12:34:25 -04:00
static DEFINE_SPINLOCK ( grace_lock ) ;
/**
* locks_start_grace
2014-09-12 16:40:20 -04:00
* @ net : net namespace that this lock manager belongs to
2007-09-06 12:34:25 -04:00
* @ lm : who this grace period is for
*
* A grace period is a period during which locks should not be given
* out . Currently grace periods are only enforced by the two lock
* managers ( lockd and nfsd ) , using the locks_in_grace ( ) function to
* check when they are in a grace period .
*
* This function is called to start a grace period .
*/
2014-09-12 16:40:20 -04:00
void
locks_start_grace ( struct net * net , struct lock_manager * lm )
2007-09-06 12:34:25 -04:00
{
2014-09-12 16:40:20 -04:00
struct list_head * grace_list = net_generic ( net , grace_net_id ) ;
2012-07-25 16:57:13 +04:00
2007-09-06 12:34:25 -04:00
spin_lock ( & grace_lock ) ;
2014-09-12 16:40:20 -04:00
list_add ( & lm - > list , grace_list ) ;
2007-09-06 12:34:25 -04:00
spin_unlock ( & grace_lock ) ;
}
EXPORT_SYMBOL_GPL ( locks_start_grace ) ;
/**
* locks_end_grace
2014-09-12 16:40:20 -04:00
* @ net : net namespace that this lock manager belongs to
2007-09-06 12:34:25 -04:00
* @ lm : who this grace period is for
*
* Call this function to state that the given lock manager is ready to
* resume regular locking . The grace period will not end until all lock
* managers that called locks_start_grace ( ) also call locks_end_grace ( ) .
* Note that callers count on it being safe to call this more than once ,
* and the second call should be a no - op .
*/
2014-09-12 16:40:20 -04:00
void
locks_end_grace ( struct lock_manager * lm )
2007-09-06 12:34:25 -04:00
{
spin_lock ( & grace_lock ) ;
list_del_init ( & lm - > list ) ;
spin_unlock ( & grace_lock ) ;
}
EXPORT_SYMBOL_GPL ( locks_end_grace ) ;
/**
* locks_in_grace
*
* Lock managers call this function to determine when it is OK for them
* to answer ordinary lock requests , and when they should accept only
* lock reclaims .
*/
2014-09-12 16:40:20 -04:00
int
2015-08-06 12:47:02 -04:00
__state_in_grace ( struct net * net , bool open )
2007-09-06 12:34:25 -04:00
{
2014-09-12 16:40:20 -04:00
struct list_head * grace_list = net_generic ( net , grace_net_id ) ;
2015-08-06 12:47:02 -04:00
struct lock_manager * lm ;
2012-07-25 16:57:13 +04:00
2015-08-06 12:47:02 -04:00
if ( ! open )
return ! list_empty ( grace_list ) ;
list_for_each_entry ( lm , grace_list , list ) {
if ( lm - > block_opens )
return true ;
}
return false ;
}
int locks_in_grace ( struct net * net )
{
return __state_in_grace ( net , 0 ) ;
2007-09-06 12:34:25 -04:00
}
EXPORT_SYMBOL_GPL ( locks_in_grace ) ;
2014-09-12 16:40:20 -04:00
2015-08-06 12:47:02 -04:00
int opens_in_grace ( struct net * net )
{
return __state_in_grace ( net , 1 ) ;
}
EXPORT_SYMBOL_GPL ( opens_in_grace ) ;
2014-09-12 16:40:20 -04:00
static int __net_init
grace_init_net ( struct net * net )
{
struct list_head * grace_list = net_generic ( net , grace_net_id ) ;
INIT_LIST_HEAD ( grace_list ) ;
return 0 ;
}
static void __net_exit
grace_exit_net ( struct net * net )
{
struct list_head * grace_list = net_generic ( net , grace_net_id ) ;
BUG_ON ( ! list_empty ( grace_list ) ) ;
}
static struct pernet_operations grace_net_ops = {
. init = grace_init_net ,
. exit = grace_exit_net ,
. id = & grace_net_id ,
. size = sizeof ( struct list_head ) ,
} ;
static int __init
init_grace ( void )
{
return register_pernet_subsys ( & grace_net_ops ) ;
}
static void __exit
exit_grace ( void )
{
unregister_pernet_subsys ( & grace_net_ops ) ;
}
MODULE_AUTHOR ( " Jeff Layton <jlayton@primarydata.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( init_grace )
module_exit ( exit_grace )