2019-05-24 12:04:05 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-12-15 14:31:23 -08:00
/* -*- mode: c; c-basic-offset: 8; -*-
* vim : noexpandtab sw = 8 ts = 8 sts = 0 :
*
* dlmthread . c
*
* standalone DLM module
*
* Copyright ( C ) 2004 Oracle . All rights reserved .
*/
# include <linux/module.h>
# include <linux/fs.h>
# include <linux/types.h>
# include <linux/highmem.h>
# include <linux/init.h>
# include <linux/sysctl.h>
# include <linux/random.h>
# include <linux/blkdev.h>
# include <linux/socket.h>
# include <linux/inet.h>
# include <linux/timer.h>
# include <linux/kthread.h>
2006-04-27 17:58:23 -07:00
# include <linux/delay.h>
2005-12-15 14:31:23 -08:00
2020-01-30 22:11:40 -08:00
# include "../cluster/heartbeat.h"
# include "../cluster/nodemanager.h"
# include "../cluster/tcp.h"
2005-12-15 14:31:23 -08:00
# include "dlmapi.h"
# include "dlmcommon.h"
# include "dlmdomain.h"
# define MLOG_MASK_PREFIX (ML_DLM|ML_DLM_THREAD)
2020-01-30 22:11:40 -08:00
# include "../cluster/masklog.h"
2005-12-15 14:31:23 -08:00
static int dlm_thread ( void * data ) ;
static void dlm_flush_asts ( struct dlm_ctxt * dlm ) ;
/* will exit holding res->spinlock, but may drop in function */
/* waits until flags are cleared on res->state */
void __dlm_wait_on_lockres_flags ( struct dlm_lock_resource * res , int flags )
{
DECLARE_WAITQUEUE ( wait , current ) ;
assert_spin_locked ( & res - > spinlock ) ;
add_wait_queue ( & res - > wq , & wait ) ;
repeat :
set_current_state ( TASK_UNINTERRUPTIBLE ) ;
if ( res - > state & flags ) {
spin_unlock ( & res - > spinlock ) ;
schedule ( ) ;
spin_lock ( & res - > spinlock ) ;
goto repeat ;
}
remove_wait_queue ( & res - > wq , & wait ) ;
2007-04-26 00:29:35 -07:00
__set_current_state ( TASK_RUNNING ) ;
2005-12-15 14:31:23 -08:00
}
2006-12-01 14:47:20 -08:00
int __dlm_lockres_has_locks ( struct dlm_lock_resource * res )
2005-12-15 14:31:23 -08:00
{
if ( list_empty ( & res - > granted ) & &
list_empty ( & res - > converting ) & &
2006-12-01 14:47:20 -08:00
list_empty ( & res - > blocked ) )
return 0 ;
return 1 ;
}
/* "unused": the lockres has no locks, is not on the dirty list,
* has no inflight locks ( in the gap between mastery and acquiring
* the first lock ) , and has no bits in its refmap .
* truly ready to be freed . */
int __dlm_lockres_unused ( struct dlm_lock_resource * res )
{
2010-07-30 16:14:44 +08:00
int bit ;
2011-07-24 10:29:54 -07:00
assert_spin_locked ( & res - > spinlock ) ;
2010-07-30 16:14:44 +08:00
if ( __dlm_lockres_has_locks ( res ) )
return 0 ;
2011-07-24 10:29:54 -07:00
/* Locks are in the process of being created */
if ( res - > inflight_locks )
return 0 ;
2010-07-30 16:14:44 +08:00
if ( ! list_empty ( & res - > dirty ) | | res - > state & DLM_LOCK_RES_DIRTY )
return 0 ;
2016-03-15 14:53:20 -07:00
if ( res - > state & ( DLM_LOCK_RES_RECOVERING |
DLM_LOCK_RES_RECOVERY_WAITING ) )
2010-07-30 16:14:44 +08:00
return 0 ;
2011-07-24 10:29:54 -07:00
/* Another node has this resource with this node as the master */
2010-07-30 16:14:44 +08:00
bit = find_next_bit ( res - > refmap , O2NM_MAX_NODES , 0 ) ;
if ( bit < O2NM_MAX_NODES )
return 0 ;
return 1 ;
2005-12-15 14:31:23 -08:00
}
/* Call whenever you may have added or deleted something from one of
* the lockres queue ' s . This will figure out whether it belongs on the
* unused list or not and does the appropriate thing . */
void __dlm_lockres_calc_usage ( struct dlm_ctxt * dlm ,
struct dlm_lock_resource * res )
{
assert_spin_locked ( & dlm - > spinlock ) ;
assert_spin_locked ( & res - > spinlock ) ;
if ( __dlm_lockres_unused ( res ) ) {
if ( list_empty ( & res - > purge ) ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: Adding res %.*s to purge list \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
res - > last_used = jiffies ;
2006-12-01 14:47:20 -08:00
dlm_lockres_get ( res ) ;
2005-12-15 14:31:23 -08:00
list_add_tail ( & res - > purge , & dlm - > purge_list ) ;
dlm - > purge_count + + ;
}
} else if ( ! list_empty ( & res - > purge ) ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: Removing res %.*s from purge list \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
list_del_init ( & res - > purge ) ;
2006-12-01 14:47:20 -08:00
dlm_lockres_put ( res ) ;
2005-12-15 14:31:23 -08:00
dlm - > purge_count - - ;
}
}
void dlm_lockres_calc_usage ( struct dlm_ctxt * dlm ,
struct dlm_lock_resource * res )
{
spin_lock ( & dlm - > spinlock ) ;
spin_lock ( & res - > spinlock ) ;
__dlm_lockres_calc_usage ( dlm , res ) ;
spin_unlock ( & res - > spinlock ) ;
spin_unlock ( & dlm - > spinlock ) ;
}
2016-08-02 14:02:19 -07:00
/*
* Do the real purge work :
* unhash the lockres , and
* clear flag DLM_LOCK_RES_DROPPING_REF .
* It requires dlm and lockres spinlock to be taken .
*/
void __dlm_do_purge_lockres ( struct dlm_ctxt * dlm ,
struct dlm_lock_resource * res )
{
assert_spin_locked ( & dlm - > spinlock ) ;
assert_spin_locked ( & res - > spinlock ) ;
if ( ! list_empty ( & res - > purge ) ) {
mlog ( 0 , " %s: Removing res %.*s from purgelist \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
list_del_init ( & res - > purge ) ;
dlm_lockres_put ( res ) ;
dlm - > purge_count - - ;
}
if ( ! __dlm_lockres_unused ( res ) ) {
mlog ( ML_ERROR , " %s: res %.*s in use after deref \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
__dlm_print_one_lock_resource ( res ) ;
BUG ( ) ;
}
__dlm_unhash_lockres ( dlm , res ) ;
spin_lock ( & dlm - > track_lock ) ;
if ( ! list_empty ( & res - > tracking ) )
list_del_init ( & res - > tracking ) ;
else {
mlog ( ML_ERROR , " %s: Resource %.*s not on the Tracking list \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
__dlm_print_one_lock_resource ( res ) ;
}
spin_unlock ( & dlm - > track_lock ) ;
/*
* lockres is not in the hash now . drop the flag and wake up
* any processes waiting in dlm_get_lock_resource .
*/
res - > state & = ~ DLM_LOCK_RES_DROPPING_REF ;
}
2010-07-19 16:04:12 -07:00
static void dlm_purge_lockres ( struct dlm_ctxt * dlm ,
2006-12-14 00:17:32 +01:00
struct dlm_lock_resource * res )
2005-12-15 14:31:23 -08:00
{
int master ;
2006-12-01 14:47:20 -08:00
int ret = 0 ;
2005-12-15 14:31:23 -08:00
2010-07-19 16:04:12 -07:00
assert_spin_locked ( & dlm - > spinlock ) ;
assert_spin_locked ( & res - > spinlock ) ;
2009-02-26 15:00:48 -08:00
2006-12-01 14:47:20 -08:00
master = ( res - > owner = = dlm - > node_num ) ;
2009-02-26 15:00:48 -08:00
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: Purging res %.*s, master %d \n " , dlm - > name ,
res - > lockname . len , res - > lockname . name , master ) ;
2006-12-01 14:47:20 -08:00
if ( ! master ) {
2016-08-02 14:02:16 -07:00
if ( res - > state & DLM_LOCK_RES_DROPPING_REF ) {
2016-08-02 14:02:19 -07:00
mlog ( ML_NOTICE , " %s: res %.*s already in DLM_LOCK_RES_DROPPING_REF state \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
2016-08-02 14:02:16 -07:00
spin_unlock ( & res - > spinlock ) ;
return ;
}
2010-07-19 16:04:12 -07:00
res - > state | = DLM_LOCK_RES_DROPPING_REF ;
2008-03-01 14:04:25 -08:00
/* drop spinlock... retake below */
2010-07-19 16:04:12 -07:00
spin_unlock ( & res - > spinlock ) ;
2008-03-01 14:04:25 -08:00
spin_unlock ( & dlm - > spinlock ) ;
2007-01-17 17:05:53 -08:00
spin_lock ( & res - > spinlock ) ;
/* This ensures that clear refmap is sent after the set */
2009-02-03 12:37:13 -08:00
__dlm_wait_on_lockres_flags ( res , DLM_LOCK_RES_SETREF_INPROG ) ;
2007-01-17 17:05:53 -08:00
spin_unlock ( & res - > spinlock ) ;
2008-03-01 14:04:25 -08:00
2006-12-01 14:47:20 -08:00
/* clear our bit from the master's refmap, ignore errors */
ret = dlm_drop_lockres_ref ( dlm , res ) ;
if ( ret < 0 ) {
if ( ! dlm_is_host_down ( ret ) )
BUG ( ) ;
}
spin_lock ( & dlm - > spinlock ) ;
2010-07-19 16:04:12 -07:00
spin_lock ( & res - > spinlock ) ;
2005-12-15 14:31:23 -08:00
}
2006-12-01 14:47:20 -08:00
if ( ! list_empty ( & res - > purge ) ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: Removing res %.*s from purgelist, master %d \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name , master ) ;
2006-12-01 14:47:20 -08:00
list_del_init ( & res - > purge ) ;
dlm_lockres_put ( res ) ;
2005-12-15 14:31:23 -08:00
dlm - > purge_count - - ;
2010-07-19 16:04:12 -07:00
}
2016-08-02 14:02:16 -07:00
if ( ! master & & ret = = DLM_DEREF_RESPONSE_INPROG ) {
mlog ( 0 , " %s: deref %.*s in progress \n " ,
2016-03-15 14:53:11 -07:00
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
spin_unlock ( & res - > spinlock ) ;
return ;
}
2010-07-19 16:04:12 -07:00
if ( ! __dlm_lockres_unused ( res ) ) {
2010-11-19 15:06:49 -08:00
mlog ( ML_ERROR , " %s: res %.*s in use after deref \n " ,
2010-07-19 16:04:12 -07:00
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
__dlm_print_one_lock_resource ( res ) ;
BUG ( ) ;
}
2009-09-03 15:56:33 +08:00
2011-07-24 10:27:54 -07:00
__dlm_unhash_lockres ( dlm , res ) ;
2005-12-15 14:31:23 -08:00
2015-09-04 15:44:37 -07:00
spin_lock ( & dlm - > track_lock ) ;
if ( ! list_empty ( & res - > tracking ) )
list_del_init ( & res - > tracking ) ;
else {
mlog ( ML_ERROR , " Resource %.*s not on the Tracking list \n " ,
res - > lockname . len , res - > lockname . name ) ;
__dlm_print_one_lock_resource ( res ) ;
}
spin_unlock ( & dlm - > track_lock ) ;
2006-12-01 14:47:20 -08:00
/* lockres is not in the hash now. drop the flag and wake up
* any processes waiting in dlm_get_lock_resource . */
if ( ! master ) {
res - > state & = ~ DLM_LOCK_RES_DROPPING_REF ;
spin_unlock ( & res - > spinlock ) ;
wake_up ( & res - > wq ) ;
2010-07-19 16:04:12 -07:00
} else
spin_unlock ( & res - > spinlock ) ;
2006-05-01 11:16:45 -07:00
}
2005-12-15 14:31:23 -08:00
static void dlm_run_purge_list ( struct dlm_ctxt * dlm ,
int purge_now )
{
unsigned int run_max , unused ;
unsigned long purge_jiffies ;
struct dlm_lock_resource * lockres ;
spin_lock ( & dlm - > spinlock ) ;
run_max = dlm - > purge_count ;
while ( run_max & & ! list_empty ( & dlm - > purge_list ) ) {
run_max - - ;
lockres = list_entry ( dlm - > purge_list . next ,
struct dlm_lock_resource , purge ) ;
spin_lock ( & lockres - > spinlock ) ;
purge_jiffies = lockres - > last_used +
msecs_to_jiffies ( DLM_PURGE_INTERVAL_MS ) ;
/* Make sure that we want to be processing this guy at
* this time . */
if ( ! purge_now & & time_after ( purge_jiffies , jiffies ) ) {
/* Since resources are added to the purge list
* in tail order , we can stop at the first
* unpurgable resource - - anyone added after
* him will have a greater last_used value */
2010-07-19 16:04:12 -07:00
spin_unlock ( & lockres - > spinlock ) ;
2005-12-15 14:31:23 -08:00
break ;
}
2010-07-19 16:04:12 -07:00
/* Status of the lockres *might* change so double
* check . If the lockres is unused , holding the dlm
* spinlock will prevent people from getting and more
* refs on it . */
unused = __dlm_lockres_unused ( lockres ) ;
if ( ! unused | |
2014-06-23 13:22:09 -07:00
( lockres - > state & DLM_LOCK_RES_MIGRATING ) | |
( lockres - > inflight_assert_workers ! = 0 ) ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s is in use or being remastered, "
2014-06-23 13:22:09 -07:00
" used %d, state %d, assert master workers %u \n " ,
dlm - > name , lockres - > lockname . len ,
lockres - > lockname . name ,
! unused , lockres - > state ,
lockres - > inflight_assert_workers ) ;
2014-06-23 13:22:08 -07:00
list_move_tail ( & lockres - > purge , & dlm - > purge_list ) ;
2010-07-19 16:04:12 -07:00
spin_unlock ( & lockres - > spinlock ) ;
continue ;
}
2007-03-22 17:01:07 -07:00
dlm_lockres_get ( lockres ) ;
2005-12-15 14:31:23 -08:00
2010-07-19 16:04:12 -07:00
dlm_purge_lockres ( dlm , lockres ) ;
2007-03-22 17:01:07 -07:00
2007-03-12 13:24:34 -07:00
dlm_lockres_put ( lockres ) ;
2005-12-15 14:31:23 -08:00
/* Avoid adding any scheduling latencies */
cond_resched_lock ( & dlm - > spinlock ) ;
}
spin_unlock ( & dlm - > spinlock ) ;
}
static void dlm_shuffle_lists ( struct dlm_ctxt * dlm ,
struct dlm_lock_resource * res )
{
struct dlm_lock * lock , * target ;
int can_grant = 1 ;
2010-11-19 15:06:49 -08:00
/*
* Because this function is called with the lockres
2005-12-15 14:31:23 -08:00
* spinlock , and because we know that it is not migrating /
* recovering / in - progress , it is fine to reserve asts and
2010-11-19 15:06:49 -08:00
* basts right before queueing them all throughout
*/
2010-05-17 20:20:44 +08:00
assert_spin_locked ( & dlm - > ast_lock ) ;
2005-12-15 14:31:23 -08:00
assert_spin_locked ( & res - > spinlock ) ;
BUG_ON ( ( res - > state & ( DLM_LOCK_RES_MIGRATING |
DLM_LOCK_RES_RECOVERING |
DLM_LOCK_RES_IN_PROGRESS ) ) ) ;
converting :
if ( list_empty ( & res - > converting ) )
goto blocked ;
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s has locks on the convert queue \n " , dlm - > name ,
res - > lockname . len , res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
target = list_entry ( res - > converting . next , struct dlm_lock , list ) ;
if ( target - > ml . convert_type = = LKM_IVMODE ) {
2010-11-19 15:06:49 -08:00
mlog ( ML_ERROR , " %s: res %.*s converting lock to invalid mode \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
BUG ( ) ;
}
2013-09-11 14:19:50 -07:00
list_for_each_entry ( lock , & res - > granted , list ) {
2005-12-15 14:31:23 -08:00
if ( lock = = target )
continue ;
if ( ! dlm_lock_compatible ( lock - > ml . type ,
target - > ml . convert_type ) ) {
can_grant = 0 ;
/* queue the BAST if not already */
if ( lock - > ml . highest_blocked = = LKM_IVMODE ) {
__dlm_lockres_reserve_ast ( res ) ;
2010-05-17 20:20:44 +08:00
__dlm_queue_bast ( dlm , lock ) ;
2005-12-15 14:31:23 -08:00
}
/* update the highest_blocked if needed */
if ( lock - > ml . highest_blocked < target - > ml . convert_type )
lock - > ml . highest_blocked =
target - > ml . convert_type ;
}
}
2013-09-11 14:19:50 -07:00
list_for_each_entry ( lock , & res - > converting , list ) {
2005-12-15 14:31:23 -08:00
if ( lock = = target )
continue ;
if ( ! dlm_lock_compatible ( lock - > ml . type ,
target - > ml . convert_type ) ) {
can_grant = 0 ;
if ( lock - > ml . highest_blocked = = LKM_IVMODE ) {
__dlm_lockres_reserve_ast ( res ) ;
2010-05-17 20:20:44 +08:00
__dlm_queue_bast ( dlm , lock ) ;
2005-12-15 14:31:23 -08:00
}
if ( lock - > ml . highest_blocked < target - > ml . convert_type )
lock - > ml . highest_blocked =
target - > ml . convert_type ;
}
}
/* we can convert the lock */
if ( can_grant ) {
spin_lock ( & target - > spinlock ) ;
BUG_ON ( target - > ml . highest_blocked ! = LKM_IVMODE ) ;
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, AST for Converting lock %u:%llu, type "
" %d => %d, node %u \n " , dlm - > name , res - > lockname . len ,
res - > lockname . name ,
dlm_get_lock_cookie_node ( be64_to_cpu ( target - > ml . cookie ) ) ,
dlm_get_lock_cookie_seq ( be64_to_cpu ( target - > ml . cookie ) ) ,
target - > ml . type ,
2005-12-15 14:31:23 -08:00
target - > ml . convert_type , target - > ml . node ) ;
target - > ml . type = target - > ml . convert_type ;
target - > ml . convert_type = LKM_IVMODE ;
2006-06-26 00:24:46 -07:00
list_move_tail ( & target - > list , & res - > granted ) ;
2005-12-15 14:31:23 -08:00
BUG_ON ( ! target - > lksb ) ;
target - > lksb - > status = DLM_NORMAL ;
spin_unlock ( & target - > spinlock ) ;
__dlm_lockres_reserve_ast ( res ) ;
2010-05-17 20:20:44 +08:00
__dlm_queue_ast ( dlm , target ) ;
2005-12-15 14:31:23 -08:00
/* go back and check for more */
goto converting ;
}
blocked :
if ( list_empty ( & res - > blocked ) )
goto leave ;
target = list_entry ( res - > blocked . next , struct dlm_lock , list ) ;
2013-09-11 14:19:50 -07:00
list_for_each_entry ( lock , & res - > granted , list ) {
2005-12-15 14:31:23 -08:00
if ( lock = = target )
continue ;
if ( ! dlm_lock_compatible ( lock - > ml . type , target - > ml . type ) ) {
can_grant = 0 ;
if ( lock - > ml . highest_blocked = = LKM_IVMODE ) {
__dlm_lockres_reserve_ast ( res ) ;
2010-05-17 20:20:44 +08:00
__dlm_queue_bast ( dlm , lock ) ;
2005-12-15 14:31:23 -08:00
}
if ( lock - > ml . highest_blocked < target - > ml . type )
lock - > ml . highest_blocked = target - > ml . type ;
}
}
2013-09-11 14:19:50 -07:00
list_for_each_entry ( lock , & res - > converting , list ) {
2005-12-15 14:31:23 -08:00
if ( lock = = target )
continue ;
if ( ! dlm_lock_compatible ( lock - > ml . type , target - > ml . type ) ) {
can_grant = 0 ;
if ( lock - > ml . highest_blocked = = LKM_IVMODE ) {
__dlm_lockres_reserve_ast ( res ) ;
2010-05-17 20:20:44 +08:00
__dlm_queue_bast ( dlm , lock ) ;
2005-12-15 14:31:23 -08:00
}
if ( lock - > ml . highest_blocked < target - > ml . type )
lock - > ml . highest_blocked = target - > ml . type ;
}
}
/* we can grant the blocked lock (only
* possible if converting list empty ) */
if ( can_grant ) {
spin_lock ( & target - > spinlock ) ;
BUG_ON ( target - > ml . highest_blocked ! = LKM_IVMODE ) ;
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, AST for Blocked lock %u:%llu, type %d, "
" node %u \n " , dlm - > name , res - > lockname . len ,
res - > lockname . name ,
dlm_get_lock_cookie_node ( be64_to_cpu ( target - > ml . cookie ) ) ,
dlm_get_lock_cookie_seq ( be64_to_cpu ( target - > ml . cookie ) ) ,
2005-12-15 14:31:23 -08:00
target - > ml . type , target - > ml . node ) ;
2010-11-19 15:06:49 -08:00
/* target->ml.type is already correct */
2006-06-26 00:24:46 -07:00
list_move_tail ( & target - > list , & res - > granted ) ;
2005-12-15 14:31:23 -08:00
BUG_ON ( ! target - > lksb ) ;
target - > lksb - > status = DLM_NORMAL ;
spin_unlock ( & target - > spinlock ) ;
__dlm_lockres_reserve_ast ( res ) ;
2010-05-17 20:20:44 +08:00
__dlm_queue_ast ( dlm , target ) ;
2005-12-15 14:31:23 -08:00
/* go back and check for more */
goto converting ;
}
leave :
return ;
}
/* must have NO locks when calling this with res !=NULL * */
void dlm_kick_thread ( struct dlm_ctxt * dlm , struct dlm_lock_resource * res )
{
if ( res ) {
spin_lock ( & dlm - > spinlock ) ;
spin_lock ( & res - > spinlock ) ;
__dlm_dirty_lockres ( dlm , res ) ;
spin_unlock ( & res - > spinlock ) ;
spin_unlock ( & dlm - > spinlock ) ;
}
wake_up ( & dlm - > dlm_thread_wq ) ;
}
void __dlm_dirty_lockres ( struct dlm_ctxt * dlm , struct dlm_lock_resource * res )
{
assert_spin_locked ( & dlm - > spinlock ) ;
assert_spin_locked ( & res - > spinlock ) ;
/* don't shuffle secondary queues */
2018-10-26 15:02:41 -07:00
if ( res - > owner = = dlm - > node_num ) {
2007-01-05 15:00:17 -08:00
if ( res - > state & ( DLM_LOCK_RES_MIGRATING |
DLM_LOCK_RES_BLOCK_DIRTY ) )
return ;
if ( list_empty ( & res - > dirty ) ) {
/* ref for dirty_list */
dlm_lockres_get ( res ) ;
list_add_tail ( & res - > dirty , & dlm - > dirty_list ) ;
res - > state | = DLM_LOCK_RES_DIRTY ;
}
2005-12-15 14:31:23 -08:00
}
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s \n " , dlm - > name , res - > lockname . len ,
res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
}
/* Launch the NM thread for the mounted volume */
int dlm_launch_thread ( struct dlm_ctxt * dlm )
{
2010-11-19 15:06:49 -08:00
mlog ( 0 , " Starting dlm_thread... \n " ) ;
2005-12-15 14:31:23 -08:00
2015-11-05 18:44:13 -08:00
dlm - > dlm_thread_task = kthread_run ( dlm_thread , dlm , " dlm-%s " ,
dlm - > name ) ;
2005-12-15 14:31:23 -08:00
if ( IS_ERR ( dlm - > dlm_thread_task ) ) {
mlog_errno ( PTR_ERR ( dlm - > dlm_thread_task ) ) ;
dlm - > dlm_thread_task = NULL ;
return - EINVAL ;
}
return 0 ;
}
void dlm_complete_thread ( struct dlm_ctxt * dlm )
{
if ( dlm - > dlm_thread_task ) {
2010-11-19 15:06:49 -08:00
mlog ( ML_KTHREAD , " Waiting for dlm thread to exit \n " ) ;
2005-12-15 14:31:23 -08:00
kthread_stop ( dlm - > dlm_thread_task ) ;
dlm - > dlm_thread_task = NULL ;
}
}
static int dlm_dirty_list_empty ( struct dlm_ctxt * dlm )
{
int empty ;
spin_lock ( & dlm - > spinlock ) ;
empty = list_empty ( & dlm - > dirty_list ) ;
spin_unlock ( & dlm - > spinlock ) ;
return empty ;
}
static void dlm_flush_asts ( struct dlm_ctxt * dlm )
{
int ret ;
struct dlm_lock * lock ;
struct dlm_lock_resource * res ;
u8 hi ;
spin_lock ( & dlm - > ast_lock ) ;
while ( ! list_empty ( & dlm - > pending_asts ) ) {
lock = list_entry ( dlm - > pending_asts . next ,
struct dlm_lock , ast_list ) ;
/* get an extra ref on lock */
dlm_lock_get ( lock ) ;
res = lock - > lockres ;
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, Flush AST for lock %u:%llu, type %d, "
" node %u \n " , dlm - > name , res - > lockname . len ,
res - > lockname . name ,
dlm_get_lock_cookie_node ( be64_to_cpu ( lock - > ml . cookie ) ) ,
dlm_get_lock_cookie_seq ( be64_to_cpu ( lock - > ml . cookie ) ) ,
lock - > ml . type , lock - > ml . node ) ;
2005-12-15 14:31:23 -08:00
BUG_ON ( ! lock - > ast_pending ) ;
/* remove from list (including ref) */
list_del_init ( & lock - > ast_list ) ;
dlm_lock_put ( lock ) ;
spin_unlock ( & dlm - > ast_lock ) ;
if ( lock - > ml . node ! = dlm - > node_num ) {
ret = dlm_do_remote_ast ( dlm , res , lock ) ;
if ( ret < 0 )
mlog_errno ( ret ) ;
} else
dlm_do_local_ast ( dlm , res , lock ) ;
spin_lock ( & dlm - > ast_lock ) ;
/* possible that another ast was queued while
* we were delivering the last one */
if ( ! list_empty ( & lock - > ast_list ) ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, AST queued while flushing last "
" one \n " , dlm - > name , res - > lockname . len ,
res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
} else
lock - > ast_pending = 0 ;
/* drop the extra ref.
* this may drop it completely . */
dlm_lock_put ( lock ) ;
dlm_lockres_release_ast ( dlm , res ) ;
}
while ( ! list_empty ( & dlm - > pending_basts ) ) {
lock = list_entry ( dlm - > pending_basts . next ,
struct dlm_lock , bast_list ) ;
/* get an extra ref on lock */
dlm_lock_get ( lock ) ;
res = lock - > lockres ;
BUG_ON ( ! lock - > bast_pending ) ;
/* get the highest blocked lock, and reset */
spin_lock ( & lock - > spinlock ) ;
BUG_ON ( lock - > ml . highest_blocked < = LKM_IVMODE ) ;
hi = lock - > ml . highest_blocked ;
lock - > ml . highest_blocked = LKM_IVMODE ;
spin_unlock ( & lock - > spinlock ) ;
/* remove from list (including ref) */
list_del_init ( & lock - > bast_list ) ;
dlm_lock_put ( lock ) ;
spin_unlock ( & dlm - > ast_lock ) ;
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, Flush BAST for lock %u:%llu, "
" blocked %d, node %u \n " ,
dlm - > name , res - > lockname . len , res - > lockname . name ,
dlm_get_lock_cookie_node ( be64_to_cpu ( lock - > ml . cookie ) ) ,
dlm_get_lock_cookie_seq ( be64_to_cpu ( lock - > ml . cookie ) ) ,
hi , lock - > ml . node ) ;
2005-12-15 14:31:23 -08:00
if ( lock - > ml . node ! = dlm - > node_num ) {
ret = dlm_send_proxy_bast ( dlm , res , lock , hi ) ;
if ( ret < 0 )
mlog_errno ( ret ) ;
} else
dlm_do_local_bast ( dlm , res , lock , hi ) ;
spin_lock ( & dlm - > ast_lock ) ;
/* possible that another bast was queued while
* we were delivering the last one */
if ( ! list_empty ( & lock - > bast_list ) ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, BAST queued while flushing last "
" one \n " , dlm - > name , res - > lockname . len ,
res - > lockname . name ) ;
2005-12-15 14:31:23 -08:00
} else
lock - > bast_pending = 0 ;
/* drop the extra ref.
* this may drop it completely . */
dlm_lock_put ( lock ) ;
dlm_lockres_release_ast ( dlm , res ) ;
}
wake_up ( & dlm - > ast_wq ) ;
spin_unlock ( & dlm - > ast_lock ) ;
}
# define DLM_THREAD_TIMEOUT_MS (4 * 1000)
# define DLM_THREAD_MAX_DIRTY 100
static int dlm_thread ( void * data )
{
struct dlm_lock_resource * res ;
struct dlm_ctxt * dlm = data ;
unsigned long timeout = msecs_to_jiffies ( DLM_THREAD_TIMEOUT_MS ) ;
mlog ( 0 , " dlm thread running for %s... \n " , dlm - > name ) ;
while ( ! kthread_should_stop ( ) ) {
int n = DLM_THREAD_MAX_DIRTY ;
/* dlm_shutting_down is very point-in-time, but that
* doesn ' t matter as we ' ll just loop back around if we
* get false on the leading edge of a state
* transition . */
dlm_run_purge_list ( dlm , dlm_shutting_down ( dlm ) ) ;
/* We really don't want to hold dlm->spinlock while
* calling dlm_shuffle_lists on each lockres that
* needs to have its queues adjusted and AST / BASTs
* run . So let ' s pull each entry off the dirty_list
* and drop dlm - > spinlock ASAP . Once off the list ,
* res - > spinlock needs to be taken again to protect
* the queues while calling dlm_shuffle_lists . */
spin_lock ( & dlm - > spinlock ) ;
while ( ! list_empty ( & dlm - > dirty_list ) ) {
int delay = 0 ;
res = list_entry ( dlm - > dirty_list . next ,
struct dlm_lock_resource , dirty ) ;
/* peel a lockres off, remove it from the list,
* unset the dirty flag and drop the dlm lock */
BUG_ON ( ! res ) ;
dlm_lockres_get ( res ) ;
spin_lock ( & res - > spinlock ) ;
2007-01-05 15:00:17 -08:00
/* We clear the DLM_LOCK_RES_DIRTY state once we shuffle lists below */
2005-12-15 14:31:23 -08:00
list_del_init ( & res - > dirty ) ;
spin_unlock ( & res - > spinlock ) ;
spin_unlock ( & dlm - > spinlock ) ;
2006-05-01 11:51:45 -07:00
/* Drop dirty_list ref */
dlm_lockres_put ( res ) ;
2005-12-15 14:31:23 -08:00
/* lockres can be re-dirtied/re-added to the
* dirty_list in this gap , but that is ok */
2010-05-17 20:20:44 +08:00
spin_lock ( & dlm - > ast_lock ) ;
2005-12-15 14:31:23 -08:00
spin_lock ( & res - > spinlock ) ;
if ( res - > owner ! = dlm - > node_num ) {
__dlm_print_one_lock_resource ( res ) ;
2010-11-19 15:06:49 -08:00
mlog ( ML_ERROR , " %s: inprog %d, mig %d, reco %d, "
" dirty %d \n " , dlm - > name ,
! ! ( res - > state & DLM_LOCK_RES_IN_PROGRESS ) ,
! ! ( res - > state & DLM_LOCK_RES_MIGRATING ) ,
! ! ( res - > state & DLM_LOCK_RES_RECOVERING ) ,
! ! ( res - > state & DLM_LOCK_RES_DIRTY ) ) ;
2005-12-15 14:31:23 -08:00
}
BUG_ON ( res - > owner ! = dlm - > node_num ) ;
/* it is now ok to move lockreses in these states
* to the dirty list , assuming that they will only be
* dirty for a short while . */
2007-01-05 15:00:17 -08:00
BUG_ON ( res - > state & DLM_LOCK_RES_MIGRATING ) ;
2005-12-15 14:31:23 -08:00
if ( res - > state & ( DLM_LOCK_RES_IN_PROGRESS |
2016-03-15 14:53:20 -07:00
DLM_LOCK_RES_RECOVERING |
DLM_LOCK_RES_RECOVERY_WAITING ) ) {
2005-12-15 14:31:23 -08:00
/* move it to the tail and keep going */
2007-01-05 15:00:17 -08:00
res - > state & = ~ DLM_LOCK_RES_DIRTY ;
2005-12-15 14:31:23 -08:00
spin_unlock ( & res - > spinlock ) ;
2010-05-17 20:20:44 +08:00
spin_unlock ( & dlm - > ast_lock ) ;
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: res %.*s, inprogress, delay list "
" shuffle, state %d \n " , dlm - > name ,
2005-12-15 14:31:23 -08:00
res - > lockname . len , res - > lockname . name ,
res - > state ) ;
delay = 1 ;
goto in_progress ;
}
/* at this point the lockres is not migrating/
* recovering / in - progress . we have the lockres
* spinlock and do NOT have the dlm lock .
* safe to reserve / queue asts and run the lists . */
/* called while holding lockres lock */
dlm_shuffle_lists ( dlm , res ) ;
2007-01-05 15:00:17 -08:00
res - > state & = ~ DLM_LOCK_RES_DIRTY ;
2005-12-15 14:31:23 -08:00
spin_unlock ( & res - > spinlock ) ;
2010-05-17 20:20:44 +08:00
spin_unlock ( & dlm - > ast_lock ) ;
2005-12-15 14:31:23 -08:00
dlm_lockres_calc_usage ( dlm , res ) ;
in_progress :
spin_lock ( & dlm - > spinlock ) ;
/* if the lock was in-progress, stick
* it on the back of the list */
if ( delay ) {
spin_lock ( & res - > spinlock ) ;
2007-01-05 15:00:17 -08:00
__dlm_dirty_lockres ( dlm , res ) ;
2005-12-15 14:31:23 -08:00
spin_unlock ( & res - > spinlock ) ;
}
dlm_lockres_put ( res ) ;
/* unlikely, but we may need to give time to
* other tasks */
if ( ! - - n ) {
2010-11-19 15:06:49 -08:00
mlog ( 0 , " %s: Throttling dlm thread \n " ,
dlm - > name ) ;
2005-12-15 14:31:23 -08:00
break ;
}
}
spin_unlock ( & dlm - > spinlock ) ;
dlm_flush_asts ( dlm ) ;
/* yield and continue right away if there is more work to do */
if ( ! n ) {
2006-05-01 14:27:41 -07:00
cond_resched ( ) ;
2005-12-15 14:31:23 -08:00
continue ;
}
wait_event_interruptible_timeout ( dlm - > dlm_thread_wq ,
! dlm_dirty_list_empty ( dlm ) | |
kthread_should_stop ( ) ,
timeout ) ;
}
mlog ( 0 , " quitting DLM thread \n " ) ;
return 0 ;
}