2006-01-18 09:30:29 +00:00
/******************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* *
* * Copyright ( C ) Sistina Software , Inc . 1997 - 2003 All rights reserved .
2010-02-24 11:08:18 -06:00
* * Copyright ( C ) 2004 - 2010 Red Hat , Inc . All rights reserved .
2006-01-18 09:30:29 +00:00
* *
* * This copyrighted material is made available to anyone wishing to use ,
* * modify , copy , or redistribute it subject to the terms and conditions
* * of the GNU General Public License v .2 .
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# include "dlm_internal.h"
# include "lock.h"
2006-07-12 16:44:04 -05:00
# include "user.h"
2006-01-18 09:30:29 +00:00
2011-04-05 13:16:24 -05:00
static uint64_t dlm_cb_seq ;
static spinlock_t dlm_cb_seq_spin ;
2006-01-18 09:30:29 +00:00
2011-02-21 14:58:21 -06:00
static void dlm_dump_lkb_callbacks ( struct dlm_lkb * lkb )
{
int i ;
log_print ( " last_bast %x %llu flags %x mode %d sb %d %x " ,
lkb - > lkb_id ,
( unsigned long long ) lkb - > lkb_last_bast . seq ,
lkb - > lkb_last_bast . flags ,
lkb - > lkb_last_bast . mode ,
lkb - > lkb_last_bast . sb_status ,
lkb - > lkb_last_bast . sb_flags ) ;
log_print ( " last_cast %x %llu flags %x mode %d sb %d %x " ,
lkb - > lkb_id ,
( unsigned long long ) lkb - > lkb_last_cast . seq ,
lkb - > lkb_last_cast . flags ,
lkb - > lkb_last_cast . mode ,
lkb - > lkb_last_cast . sb_status ,
lkb - > lkb_last_cast . sb_flags ) ;
for ( i = 0 ; i < DLM_CALLBACKS_SIZE ; i + + ) {
log_print ( " cb %x %llu flags %x mode %d sb %d %x " ,
lkb - > lkb_id ,
( unsigned long long ) lkb - > lkb_callbacks [ i ] . seq ,
lkb - > lkb_callbacks [ i ] . flags ,
lkb - > lkb_callbacks [ i ] . mode ,
lkb - > lkb_callbacks [ i ] . sb_status ,
lkb - > lkb_callbacks [ i ] . sb_flags ) ;
}
}
int dlm_add_lkb_callback ( struct dlm_lkb * lkb , uint32_t flags , int mode ,
int status , uint32_t sbflags , uint64_t seq )
{
struct dlm_ls * ls = lkb - > lkb_resource - > res_ls ;
uint64_t prev_seq ;
int prev_mode ;
2011-04-05 13:16:24 -05:00
int i , rv ;
2011-02-21 14:58:21 -06:00
for ( i = 0 ; i < DLM_CALLBACKS_SIZE ; i + + ) {
if ( lkb - > lkb_callbacks [ i ] . seq )
continue ;
/*
* Suppress some redundant basts here , do more on removal .
* Don ' t even add a bast if the callback just before it
* is a bast for the same mode or a more restrictive mode .
* ( the addional > PR check is needed for PR / CW inversion )
*/
if ( ( i > 0 ) & & ( flags & DLM_CB_BAST ) & &
( lkb - > lkb_callbacks [ i - 1 ] . flags & DLM_CB_BAST ) ) {
prev_seq = lkb - > lkb_callbacks [ i - 1 ] . seq ;
prev_mode = lkb - > lkb_callbacks [ i - 1 ] . mode ;
if ( ( prev_mode = = mode ) | |
( prev_mode > mode & & prev_mode > DLM_LOCK_PR ) ) {
log_debug ( ls , " skip %x add bast %llu mode %d "
" for bast %llu mode %d " ,
lkb - > lkb_id ,
( unsigned long long ) seq ,
mode ,
( unsigned long long ) prev_seq ,
prev_mode ) ;
2011-04-05 13:16:24 -05:00
rv = 0 ;
goto out ;
2011-02-21 14:58:21 -06:00
}
}
lkb - > lkb_callbacks [ i ] . seq = seq ;
lkb - > lkb_callbacks [ i ] . flags = flags ;
lkb - > lkb_callbacks [ i ] . mode = mode ;
lkb - > lkb_callbacks [ i ] . sb_status = status ;
lkb - > lkb_callbacks [ i ] . sb_flags = ( sbflags & 0x000000FF ) ;
2011-04-05 13:16:24 -05:00
rv = 0 ;
2011-02-21 14:58:21 -06:00
break ;
}
if ( i = = DLM_CALLBACKS_SIZE ) {
log_error ( ls , " no callbacks %x %llu flags %x mode %d sb %d %x " ,
lkb - > lkb_id , ( unsigned long long ) seq ,
flags , mode , status , sbflags ) ;
dlm_dump_lkb_callbacks ( lkb ) ;
2011-04-05 13:16:24 -05:00
rv = - 1 ;
goto out ;
2011-02-21 14:58:21 -06:00
}
2011-04-05 13:16:24 -05:00
out :
return rv ;
2011-02-21 14:58:21 -06:00
}
int dlm_rem_lkb_callback ( struct dlm_ls * ls , struct dlm_lkb * lkb ,
struct dlm_callback * cb , int * resid )
2006-01-18 09:30:29 +00:00
{
2011-04-05 13:16:24 -05:00
int i , rv ;
2011-02-21 14:58:21 -06:00
* resid = 0 ;
2011-04-05 13:16:24 -05:00
if ( ! lkb - > lkb_callbacks [ 0 ] . seq ) {
rv = - ENOENT ;
goto out ;
}
2011-02-21 14:58:21 -06:00
/* oldest undelivered cb is callbacks[0] */
memcpy ( cb , & lkb - > lkb_callbacks [ 0 ] , sizeof ( struct dlm_callback ) ) ;
memset ( & lkb - > lkb_callbacks [ 0 ] , 0 , sizeof ( struct dlm_callback ) ) ;
/* shift others down */
for ( i = 1 ; i < DLM_CALLBACKS_SIZE ; i + + ) {
if ( ! lkb - > lkb_callbacks [ i ] . seq )
break ;
memcpy ( & lkb - > lkb_callbacks [ i - 1 ] , & lkb - > lkb_callbacks [ i ] ,
sizeof ( struct dlm_callback ) ) ;
memset ( & lkb - > lkb_callbacks [ i ] , 0 , sizeof ( struct dlm_callback ) ) ;
( * resid ) + + ;
}
/* if cb is a bast, it should be skipped if the blocking mode is
compatible with the last granted mode */
if ( ( cb - > flags & DLM_CB_BAST ) & & lkb - > lkb_last_cast . seq ) {
if ( dlm_modes_compat ( cb - > mode , lkb - > lkb_last_cast . mode ) ) {
cb - > flags | = DLM_CB_SKIP ;
log_debug ( ls , " skip %x bast %llu mode %d "
" for cast %llu mode %d " ,
lkb - > lkb_id ,
( unsigned long long ) cb - > seq ,
cb - > mode ,
( unsigned long long ) lkb - > lkb_last_cast . seq ,
lkb - > lkb_last_cast . mode ) ;
2011-04-05 13:16:24 -05:00
rv = 0 ;
goto out ;
2011-02-21 14:58:21 -06:00
}
}
if ( cb - > flags & DLM_CB_CAST ) {
memcpy ( & lkb - > lkb_last_cast , cb , sizeof ( struct dlm_callback ) ) ;
lkb - > lkb_last_cast_time = ktime_get ( ) ;
}
if ( cb - > flags & DLM_CB_BAST ) {
memcpy ( & lkb - > lkb_last_bast , cb , sizeof ( struct dlm_callback ) ) ;
lkb - > lkb_last_bast_time = ktime_get ( ) ;
}
2011-04-05 13:16:24 -05:00
rv = 0 ;
out :
return rv ;
2011-02-21 14:58:21 -06:00
}
2011-04-05 13:16:24 -05:00
void dlm_add_cb ( struct dlm_lkb * lkb , uint32_t flags , int mode , int status ,
uint32_t sbflags )
2011-02-21 14:58:21 -06:00
{
2011-04-05 13:16:24 -05:00
struct dlm_ls * ls = lkb - > lkb_resource - > res_ls ;
uint64_t new_seq , prev_seq ;
2011-02-21 14:58:21 -06:00
int rv ;
2011-04-05 13:16:24 -05:00
spin_lock ( & dlm_cb_seq_spin ) ;
new_seq = + + dlm_cb_seq ;
spin_unlock ( & dlm_cb_seq_spin ) ;
2011-02-21 14:58:21 -06:00
2006-07-12 16:44:04 -05:00
if ( lkb - > lkb_flags & DLM_IFL_USER ) {
2011-04-05 13:16:24 -05:00
dlm_user_add_ast ( lkb , flags , mode , status , sbflags , new_seq ) ;
2006-07-12 16:44:04 -05:00
return ;
}
2011-04-05 13:16:24 -05:00
mutex_lock ( & lkb - > lkb_cb_mutex ) ;
prev_seq = lkb - > lkb_callbacks [ 0 ] . seq ;
2011-02-21 14:58:21 -06:00
2011-04-05 13:16:24 -05:00
rv = dlm_add_lkb_callback ( lkb , flags , mode , status , sbflags , new_seq ) ;
if ( rv < 0 )
goto out ;
if ( ! prev_seq ) {
2006-01-18 09:30:29 +00:00
kref_get ( & lkb - > lkb_ref ) ;
2011-04-05 13:16:24 -05:00
if ( test_bit ( LSFL_CB_DELAY , & ls - > ls_flags ) ) {
mutex_lock ( & ls - > ls_cb_mutex ) ;
list_add ( & lkb - > lkb_cb_list , & ls - > ls_cb_delay ) ;
mutex_unlock ( & ls - > ls_cb_mutex ) ;
} else {
queue_work ( ls - > ls_callback_wq , & lkb - > lkb_cb_work ) ;
}
}
out :
mutex_unlock ( & lkb - > lkb_cb_mutex ) ;
2006-01-18 09:30:29 +00:00
}
2011-04-05 13:16:24 -05:00
void dlm_callback_work ( struct work_struct * work )
2006-01-18 09:30:29 +00:00
{
2011-04-05 13:16:24 -05:00
struct dlm_lkb * lkb = container_of ( work , struct dlm_lkb , lkb_cb_work ) ;
struct dlm_ls * ls = lkb - > lkb_resource - > res_ls ;
2010-02-24 11:08:18 -06:00
void ( * castfn ) ( void * astparam ) ;
void ( * bastfn ) ( void * astparam , int mode ) ;
2011-02-21 14:58:21 -06:00
struct dlm_callback callbacks [ DLM_CALLBACKS_SIZE ] ;
int i , rv , resid ;
2008-12-23 10:22:56 -06:00
2011-04-05 13:16:24 -05:00
memset ( & callbacks , 0 , sizeof ( callbacks ) ) ;
2006-01-18 09:30:29 +00:00
2011-04-05 13:16:24 -05:00
mutex_lock ( & lkb - > lkb_cb_mutex ) ;
if ( ! lkb - > lkb_callbacks [ 0 ] . seq ) {
/* no callback work exists, shouldn't happen */
log_error ( ls , " dlm_callback_work %x no work " , lkb - > lkb_id ) ;
dlm_print_lkb ( lkb ) ;
dlm_dump_lkb_callbacks ( lkb ) ;
}
2006-01-18 09:30:29 +00:00
2011-04-05 13:16:24 -05:00
for ( i = 0 ; i < DLM_CALLBACKS_SIZE ; i + + ) {
rv = dlm_rem_lkb_callback ( ls , lkb , & callbacks [ i ] , & resid ) ;
if ( rv < 0 )
break ;
}
2010-02-24 11:08:18 -06:00
2011-04-05 13:16:24 -05:00
if ( resid ) {
/* cbs remain, loop should have removed all, shouldn't happen */
log_error ( ls , " dlm_callback_work %x resid %d " , lkb - > lkb_id ,
resid ) ;
dlm_print_lkb ( lkb ) ;
dlm_dump_lkb_callbacks ( lkb ) ;
}
mutex_unlock ( & lkb - > lkb_cb_mutex ) ;
2010-02-24 11:08:18 -06:00
2011-04-05 13:16:24 -05:00
castfn = lkb - > lkb_astfn ;
bastfn = lkb - > lkb_bastfn ;
2010-02-24 11:08:18 -06:00
2011-04-05 13:16:24 -05:00
for ( i = 0 ; i < DLM_CALLBACKS_SIZE ; i + + ) {
if ( ! callbacks [ i ] . seq )
break ;
if ( callbacks [ i ] . flags & DLM_CB_SKIP ) {
continue ;
} else if ( callbacks [ i ] . flags & DLM_CB_BAST ) {
bastfn ( lkb - > lkb_astparam , callbacks [ i ] . mode ) ;
} else if ( callbacks [ i ] . flags & DLM_CB_CAST ) {
lkb - > lkb_lksb - > sb_status = callbacks [ i ] . sb_status ;
lkb - > lkb_lksb - > sb_flags = callbacks [ i ] . sb_flags ;
castfn ( lkb - > lkb_astparam ) ;
2010-02-24 11:08:18 -06:00
}
2006-01-18 09:30:29 +00:00
}
2011-04-05 13:16:24 -05:00
/* undo kref_get from dlm_add_callback, may cause lkb to be freed */
dlm_put_lkb ( lkb ) ;
2006-01-18 09:30:29 +00:00
}
2011-04-05 13:16:24 -05:00
int dlm_callback_start ( struct dlm_ls * ls )
2006-01-18 09:30:29 +00:00
{
2011-04-05 13:16:24 -05:00
ls - > ls_callback_wq = alloc_workqueue ( " dlm_callback " ,
2011-07-18 15:28:34 -05:00
WQ_UNBOUND |
WQ_MEM_RECLAIM |
WQ_NON_REENTRANT ,
0 ) ;
2011-04-05 13:16:24 -05:00
if ( ! ls - > ls_callback_wq ) {
log_print ( " can't start dlm_callback workqueue " ) ;
return - ENOMEM ;
2006-01-18 09:30:29 +00:00
}
return 0 ;
}
2011-04-05 13:16:24 -05:00
void dlm_callback_stop ( struct dlm_ls * ls )
2006-01-18 09:30:29 +00:00
{
2011-04-05 13:16:24 -05:00
if ( ls - > ls_callback_wq )
destroy_workqueue ( ls - > ls_callback_wq ) ;
2006-01-18 09:30:29 +00:00
}
2011-04-05 13:16:24 -05:00
void dlm_callback_suspend ( struct dlm_ls * ls )
2006-01-18 09:30:29 +00:00
{
2011-04-05 13:16:24 -05:00
set_bit ( LSFL_CB_DELAY , & ls - > ls_flags ) ;
2006-01-18 09:30:29 +00:00
2011-04-05 13:16:24 -05:00
if ( ls - > ls_callback_wq )
flush_workqueue ( ls - > ls_callback_wq ) ;
2006-01-18 09:30:29 +00:00
}
2011-04-05 13:16:24 -05:00
void dlm_callback_resume ( struct dlm_ls * ls )
2006-01-18 09:30:29 +00:00
{
2011-04-05 13:16:24 -05:00
struct dlm_lkb * lkb , * safe ;
int count = 0 ;
2006-01-18 09:30:29 +00:00
2011-04-05 13:16:24 -05:00
clear_bit ( LSFL_CB_DELAY , & ls - > ls_flags ) ;
if ( ! ls - > ls_callback_wq )
return ;
mutex_lock ( & ls - > ls_cb_mutex ) ;
list_for_each_entry_safe ( lkb , safe , & ls - > ls_cb_delay , lkb_cb_list ) {
list_del_init ( & lkb - > lkb_cb_list ) ;
queue_work ( ls - > ls_callback_wq , & lkb - > lkb_cb_work ) ;
count + + ;
}
mutex_unlock ( & ls - > ls_cb_mutex ) ;
log_debug ( ls , " dlm_callback_resume %d " , count ) ;
2006-01-18 09:30:29 +00:00
}