2009-10-25 18:12:12 +03:00
/*
Unix SMB / CIFS implementation .
global locks based on dbwrap and messaging
Copyright ( C ) 2009 by Volker Lendecke
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-02-26 01:20:06 +03:00
# include "system/filesys.h"
2011-07-07 19:42:08 +04:00
# include "dbwrap/dbwrap.h"
2011-07-06 18:40:21 +04:00
# include "dbwrap/dbwrap_open.h"
2009-10-25 18:12:12 +03:00
# include "g_lock.h"
2011-05-05 13:25:29 +04:00
# include "util_tdb.h"
2010-08-26 12:14:07 +04:00
# include "ctdbd_conn.h"
2010-10-01 12:08:15 +04:00
# include "../lib/util/select.h"
2011-02-08 13:59:04 +03:00
# include "system/select.h"
2011-03-24 17:31:06 +03:00
# include "messages.h"
2009-10-25 18:12:12 +03:00
static NTSTATUS g_lock_force_unlock ( struct g_lock_ctx * ctx , const char * name ,
struct server_id pid ) ;
struct g_lock_ctx {
struct db_context * db ;
struct messaging_context * msg ;
} ;
/*
* The " g_lock.tdb " file contains records , indexed by the 0 - terminated
* lockname . The record contains an array of " struct g_lock_rec "
* structures . Waiters have the lock_type with G_LOCK_PENDING or ' ed .
*/
struct g_lock_rec {
enum g_lock_type lock_type ;
struct server_id pid ;
} ;
struct g_lock_ctx * g_lock_ctx_init ( TALLOC_CTX * mem_ctx ,
struct messaging_context * msg )
{
struct g_lock_ctx * result ;
result = talloc ( mem_ctx , struct g_lock_ctx ) ;
if ( result = = NULL ) {
return NULL ;
}
result - > msg = msg ;
result - > db = db_open ( result , lock_path ( " g_lock.tdb " ) , 0 ,
2011-06-27 17:44:10 +04:00
TDB_CLEAR_IF_FIRST | TDB_INCOMPATIBLE_HASH ,
2012-01-06 20:19:54 +04:00
O_RDWR | O_CREAT , 0600 ,
DBWRAP_LOCK_ORDER_2 ) ;
2009-10-25 18:12:12 +03:00
if ( result - > db = = NULL ) {
2011-09-20 00:30:57 +04:00
DEBUG ( 1 , ( " g_lock_init: Could not open g_lock.tdb \n " ) ) ;
2009-10-25 18:12:12 +03:00
TALLOC_FREE ( result ) ;
return NULL ;
}
return result ;
}
static bool g_lock_conflicts ( enum g_lock_type lock_type ,
const struct g_lock_rec * rec )
{
enum g_lock_type rec_lock = rec - > lock_type ;
if ( ( rec_lock & G_LOCK_PENDING ) ! = 0 ) {
return false ;
}
/*
* Only tested write locks so far . Very likely this routine
* needs to be fixed for read locks . . . .
*/
if ( ( lock_type = = G_LOCK_READ ) & & ( rec_lock = = G_LOCK_READ ) ) {
return false ;
}
return true ;
}
static bool g_lock_parse ( TALLOC_CTX * mem_ctx , TDB_DATA data ,
int * pnum_locks , struct g_lock_rec * * plocks )
{
int i , num_locks ;
struct g_lock_rec * locks ;
if ( ( data . dsize % sizeof ( struct g_lock_rec ) ) ! = 0 ) {
DEBUG ( 1 , ( " invalid lock record length %d \n " , ( int ) data . dsize ) ) ;
return false ;
}
num_locks = data . dsize / sizeof ( struct g_lock_rec ) ;
locks = talloc_array ( mem_ctx , struct g_lock_rec , num_locks ) ;
if ( locks = = NULL ) {
DEBUG ( 1 , ( " talloc failed \n " ) ) ;
return false ;
}
memcpy ( locks , data . dptr , data . dsize ) ;
DEBUG ( 10 , ( " locks: \n " ) ) ;
for ( i = 0 ; i < num_locks ; i + + ) {
DEBUGADD ( 10 , ( " %s: %s %s \n " ,
2011-06-08 08:05:55 +04:00
server_id_str ( talloc_tos ( ) , & locks [ i ] . pid ) ,
2009-10-25 18:12:12 +03:00
( ( locks [ i ] . lock_type & 1 ) = = G_LOCK_READ ) ?
" read " : " write " ,
( locks [ i ] . lock_type & G_LOCK_PENDING ) ?
" (pending) " : " (owner) " ) ) ;
2010-02-15 18:57:16 +03:00
if ( ( ( locks [ i ] . lock_type & G_LOCK_PENDING ) = = 0 )
& & ! process_exists ( locks [ i ] . pid ) ) {
DEBUGADD ( 10 , ( " lock owner %s died -- discarding \n " ,
2011-06-08 08:05:55 +04:00
server_id_str ( talloc_tos ( ) ,
2010-02-15 18:57:16 +03:00
& locks [ i ] . pid ) ) ) ;
if ( i < ( num_locks - 1 ) ) {
locks [ i ] = locks [ num_locks - 1 ] ;
}
num_locks - = 1 ;
}
}
* plocks = locks ;
* pnum_locks = num_locks ;
return true ;
}
static void g_lock_cleanup ( int * pnum_locks , struct g_lock_rec * locks )
{
int i , num_locks ;
num_locks = * pnum_locks ;
DEBUG ( 10 , ( " g_lock_cleanup: %d locks \n " , num_locks ) ) ;
for ( i = 0 ; i < num_locks ; i + + ) {
2009-10-25 18:12:12 +03:00
if ( process_exists ( locks [ i ] . pid ) ) {
continue ;
}
DEBUGADD ( 10 , ( " %s does not exist -- discarding \n " ,
2011-06-08 08:05:55 +04:00
server_id_str ( talloc_tos ( ) , & locks [ i ] . pid ) ) ) ;
2009-10-25 18:12:12 +03:00
if ( i < ( num_locks - 1 ) ) {
locks [ i ] = locks [ num_locks - 1 ] ;
}
num_locks - = 1 ;
}
* pnum_locks = num_locks ;
2010-02-15 18:57:16 +03:00
return ;
2009-10-25 18:12:12 +03:00
}
static struct g_lock_rec * g_lock_addrec ( TALLOC_CTX * mem_ctx ,
struct g_lock_rec * locks ,
2010-02-15 18:49:46 +03:00
int * pnum_locks ,
2009-10-25 18:12:12 +03:00
const struct server_id pid ,
enum g_lock_type lock_type )
{
struct g_lock_rec * result ;
2010-02-15 18:49:46 +03:00
int num_locks = * pnum_locks ;
2009-10-25 18:12:12 +03:00
result = talloc_realloc ( mem_ctx , locks , struct g_lock_rec ,
num_locks + 1 ) ;
if ( result = = NULL ) {
return NULL ;
}
result [ num_locks ] . pid = pid ;
result [ num_locks ] . lock_type = lock_type ;
2010-02-15 18:49:46 +03:00
* pnum_locks + = 1 ;
2009-10-25 18:12:12 +03:00
return result ;
}
static void g_lock_got_retry ( struct messaging_context * msg ,
void * private_data ,
uint32_t msg_type ,
struct server_id server_id ,
DATA_BLOB * data ) ;
static NTSTATUS g_lock_trylock ( struct g_lock_ctx * ctx , const char * name ,
enum g_lock_type lock_type )
{
struct db_record * rec = NULL ;
struct g_lock_rec * locks = NULL ;
int i , num_locks ;
struct server_id self ;
int our_index ;
TDB_DATA data ;
NTSTATUS status = NT_STATUS_OK ;
NTSTATUS store_status ;
2011-08-17 13:21:31 +04:00
TDB_DATA value ;
2009-10-25 18:12:12 +03:00
again :
2011-08-17 13:21:31 +04:00
rec = dbwrap_fetch_locked ( ctx - > db , talloc_tos ( ) ,
string_term_tdb_data ( name ) ) ;
2009-10-25 18:12:12 +03:00
if ( rec = = NULL ) {
DEBUG ( 10 , ( " fetch_locked( \" %s \" ) failed \n " , name ) ) ;
status = NT_STATUS_LOCK_NOT_GRANTED ;
goto done ;
}
2011-08-17 13:21:31 +04:00
value = dbwrap_record_get_value ( rec ) ;
if ( ! g_lock_parse ( talloc_tos ( ) , value , & num_locks , & locks ) ) {
2009-10-25 18:12:12 +03:00
DEBUG ( 10 , ( " g_lock_parse for %s failed \n " , name ) ) ;
status = NT_STATUS_INTERNAL_ERROR ;
goto done ;
}
2010-06-27 17:36:38 +04:00
self = messaging_server_id ( ctx - > msg ) ;
2009-10-25 18:12:12 +03:00
our_index = - 1 ;
for ( i = 0 ; i < num_locks ; i + + ) {
if ( procid_equal ( & self , & locks [ i ] . pid ) ) {
if ( our_index ! = - 1 ) {
DEBUG ( 1 , ( " g_lock_trylock: Added ourself "
" twice! \n " ) ) ;
status = NT_STATUS_INTERNAL_ERROR ;
goto done ;
}
if ( ( locks [ i ] . lock_type & G_LOCK_PENDING ) = = 0 ) {
DEBUG ( 1 , ( " g_lock_trylock: Found ourself not "
" pending! \n " ) ) ;
status = NT_STATUS_INTERNAL_ERROR ;
goto done ;
}
our_index = i ;
/* never conflict with ourself */
continue ;
}
if ( g_lock_conflicts ( lock_type , & locks [ i ] ) ) {
struct server_id pid = locks [ i ] . pid ;
if ( ! process_exists ( pid ) ) {
TALLOC_FREE ( locks ) ;
TALLOC_FREE ( rec ) ;
status = g_lock_force_unlock ( ctx , name , pid ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Could not unlock dead lock "
" holder! \n " ) ) ;
goto done ;
}
goto again ;
}
lock_type | = G_LOCK_PENDING ;
}
}
if ( our_index = = - 1 ) {
/* First round, add ourself */
2010-02-15 18:49:46 +03:00
locks = g_lock_addrec ( talloc_tos ( ) , locks , & num_locks ,
2009-10-25 18:12:12 +03:00
self , lock_type ) ;
if ( locks = = NULL ) {
DEBUG ( 10 , ( " g_lock_addrec failed \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
} else {
/*
* Retry . We were pending last time . Overwrite the
* stored lock_type with what we calculated , we might
* have acquired the lock this time .
*/
locks [ our_index ] . lock_type = lock_type ;
}
2010-02-15 18:57:16 +03:00
if ( NT_STATUS_IS_OK ( status ) & & ( ( lock_type & G_LOCK_PENDING ) = = 0 ) ) {
/*
* Walk through the list of locks , search for dead entries
*/
g_lock_cleanup ( & num_locks , locks ) ;
}
2010-02-15 18:49:46 +03:00
data = make_tdb_data ( ( uint8_t * ) locks , num_locks * sizeof ( * locks ) ) ;
2011-08-17 13:21:31 +04:00
store_status = dbwrap_record_store ( rec , data , 0 ) ;
2009-10-25 18:12:12 +03:00
if ( ! NT_STATUS_IS_OK ( store_status ) ) {
DEBUG ( 1 , ( " rec->store failed: %s \n " ,
nt_errstr ( store_status ) ) ) ;
status = store_status ;
}
done :
TALLOC_FREE ( locks ) ;
TALLOC_FREE ( rec ) ;
if ( NT_STATUS_IS_OK ( status ) & & ( lock_type & G_LOCK_PENDING ) ! = 0 ) {
return STATUS_PENDING ;
}
return NT_STATUS_OK ;
}
NTSTATUS g_lock_lock ( struct g_lock_ctx * ctx , const char * name ,
enum g_lock_type lock_type , struct timeval timeout )
{
struct tevent_timer * te = NULL ;
NTSTATUS status ;
bool retry = false ;
2010-01-23 03:17:06 +03:00
struct timeval timeout_end ;
struct timeval time_now ;
2009-10-25 18:12:12 +03:00
DEBUG ( 10 , ( " Trying to acquire lock %d for %s \n " , ( int ) lock_type ,
name ) ) ;
if ( lock_type & ~ 1 ) {
DEBUG ( 1 , ( " Got invalid lock type %d for %s \n " ,
( int ) lock_type , name ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
# ifdef CLUSTER_SUPPORT
if ( lp_clustering ( ) ) {
2010-08-31 18:52:56 +04:00
status = ctdb_watch_us ( messaging_ctdbd_connection ( ) ) ;
2009-10-25 18:12:12 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " could not register retry with ctdb: %s \n " ,
nt_errstr ( status ) ) ) ;
goto done ;
}
}
# endif
status = messaging_register ( ctx - > msg , & retry , MSG_DBWRAP_G_LOCK_RETRY ,
g_lock_got_retry ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " messaging_register failed: %s \n " ,
nt_errstr ( status ) ) ) ;
return status ;
}
2010-01-23 03:17:06 +03:00
time_now = timeval_current ( ) ;
timeout_end = timeval_sum ( & time_now , & timeout ) ;
2009-10-25 18:12:12 +03:00
while ( true ) {
2011-02-08 13:59:04 +03:00
struct pollfd * pollfds ;
int num_pollfds ;
int saved_errno ;
2010-01-23 03:17:06 +03:00
int ret ;
2010-02-16 17:21:25 +03:00
struct timeval timeout_remaining , select_timeout ;
2010-01-23 03:17:06 +03:00
status = g_lock_trylock ( ctx , name , lock_type ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " Got lock %s \n " , name ) ) ;
break ;
2009-10-25 18:12:12 +03:00
}
2010-01-23 03:17:06 +03:00
if ( ! NT_STATUS_EQUAL ( status , STATUS_PENDING ) ) {
DEBUG ( 10 , ( " g_lock_trylock failed: %s \n " ,
nt_errstr ( status ) ) ) ;
break ;
2009-10-25 18:12:12 +03:00
}
2010-01-23 03:17:06 +03:00
DEBUG ( 10 , ( " g_lock_trylock: Did not get lock, waiting... \n " ) ) ;
/* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* ! ! ! HACK ALERT - - - FIX ME ! ! !
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
* What we really want to do here is to react to
* MSG_DBWRAP_G_LOCK_RETRY messages that are either sent
* by a client doing g_lock_unlock or by ourselves when
* we receive a CTDB_SRVID_SAMBA_NOTIFY or
* CTDB_SRVID_RECONFIGURE message from ctdbd , i . e . when
* either a client holding a lock or a complete node
* has died .
*
* Doing this properly involves calling tevent_loop_once ( ) ,
* but doing this here with the main ctdbd messaging context
* creates a nested event loop when g_lock_lock ( ) is called
* from the main event loop , e . g . in a tcon_and_X where the
* share_info . tdb needs to be initialized and is locked by
* another process , or when the remore registry is accessed
* for writing and some other process already holds a lock
* on the registry . tdb .
*
2010-02-15 18:35:06 +03:00
* So as a quick fix , we act a little coarsely here : we do
2010-01-23 03:17:06 +03:00
* a select on the ctdb connection fd and when it is readable
* or we get EINTR , then we retry without actually parsing
* any ctdb packages or dispatching messages . This means that
* we retry more often than intended by design , but this does
* not harm and it is unobtrusive . When we have finished ,
* the main loop will pick up all the messages and ctdb
* packets . The only extra twist is that we cannot use timed
* events here but have to handcode a timeout .
*/
2009-10-25 18:12:12 +03:00
2011-02-08 13:59:04 +03:00
/*
* We allocate 2 entries here . One is needed anyway for
* sys_poll and in the clustering case we might have to add
* the ctdb fd . This avoids the realloc then .
*/
2011-06-07 05:30:12 +04:00
pollfds = talloc_array ( talloc_tos ( ) , struct pollfd , 2 ) ;
2011-02-08 13:59:04 +03:00
if ( pollfds = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
break ;
}
2011-03-30 18:38:31 +04:00
num_pollfds = 0 ;
2011-02-08 13:59:04 +03:00
2010-01-23 03:17:06 +03:00
# ifdef CLUSTER_SUPPORT
if ( lp_clustering ( ) ) {
2010-07-04 18:54:04 +04:00
struct ctdbd_connection * conn ;
2010-08-31 18:52:56 +04:00
conn = messaging_ctdbd_connection ( ) ;
2010-01-23 03:17:06 +03:00
2011-02-08 13:59:04 +03:00
pollfds [ 0 ] . fd = ctdbd_conn_get_fd ( conn ) ;
pollfds [ 0 ] . events = POLLIN | POLLHUP ;
num_pollfds + = 1 ;
2009-10-25 18:12:12 +03:00
}
2010-01-23 03:17:06 +03:00
# endif
2010-02-16 17:21:25 +03:00
time_now = timeval_current ( ) ;
timeout_remaining = timeval_until ( & time_now , & timeout_end ) ;
2010-02-16 14:28:53 +03:00
select_timeout = timeval_set ( 60 , 0 ) ;
2010-01-23 03:17:06 +03:00
2010-02-16 17:21:25 +03:00
select_timeout = timeval_min ( & select_timeout ,
& timeout_remaining ) ;
2011-02-08 13:59:04 +03:00
ret = sys_poll ( pollfds , num_pollfds ,
timeval_to_msec ( select_timeout ) ) ;
/*
* We ' re not * really interested in the actual flags . We just
* need to retry this whole thing .
*/
saved_errno = errno ;
TALLOC_FREE ( pollfds ) ;
errno = saved_errno ;
2010-01-23 03:17:06 +03:00
if ( ret = = - 1 ) {
if ( errno ! = EINTR ) {
DEBUG ( 1 , ( " error calling select: %s \n " ,
strerror ( errno ) ) ) ;
status = NT_STATUS_INTERNAL_ERROR ;
break ;
}
/*
* errno = = EINTR :
* This means a signal was received .
* It might have been a MSG_DBWRAP_G_LOCK_RETRY message .
* = = > retry
*/
} else if ( ret = = 0 ) {
if ( timeval_expired ( & timeout_end ) ) {
DEBUG ( 10 , ( " g_lock_lock timed out \n " ) ) ;
status = NT_STATUS_LOCK_NOT_GRANTED ;
break ;
} else {
DEBUG ( 10 , ( " select returned 0 but timeout not "
2010-02-16 14:28:53 +03:00
" not expired, retrying \n " ) ) ;
2010-01-23 03:17:06 +03:00
}
} else if ( ret ! = 1 ) {
DEBUG ( 1 , ( " invalid return code of select: %d \n " , ret ) ) ;
status = NT_STATUS_INTERNAL_ERROR ;
break ;
}
/*
* ret = = 1 :
* This means ctdbd has sent us some data .
* Might be a CTDB_SRVID_RECONFIGURE or a
* CTDB_SRVID_SAMBA_NOTIFY message .
* = = > retry
*/
2009-10-25 18:12:12 +03:00
}
2010-01-23 03:17:06 +03:00
2010-02-13 09:21:19 +03:00
# ifdef CLUSTER_SUPPORT
2009-10-25 18:12:12 +03:00
done :
2010-02-13 09:21:19 +03:00
# endif
2009-10-25 18:12:12 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
NTSTATUS unlock_status ;
unlock_status = g_lock_unlock ( ctx , name ) ;
if ( ! NT_STATUS_IS_OK ( unlock_status ) ) {
DEBUG ( 1 , ( " Could not remove ourself from the locking "
" db: %s \n " , nt_errstr ( status ) ) ) ;
}
}
messaging_deregister ( ctx - > msg , MSG_DBWRAP_G_LOCK_RETRY , & retry ) ;
TALLOC_FREE ( te ) ;
return status ;
}
static void g_lock_got_retry ( struct messaging_context * msg ,
void * private_data ,
uint32_t msg_type ,
struct server_id server_id ,
DATA_BLOB * data )
{
bool * pretry = ( bool * ) private_data ;
DEBUG ( 10 , ( " Got retry message from pid %s \n " ,
2011-06-08 08:05:55 +04:00
server_id_str ( talloc_tos ( ) , & server_id ) ) ) ;
2009-10-25 18:12:12 +03:00
* pretry = true ;
}
static NTSTATUS g_lock_force_unlock ( struct g_lock_ctx * ctx , const char * name ,
struct server_id pid )
{
struct db_record * rec = NULL ;
struct g_lock_rec * locks = NULL ;
int i , num_locks ;
enum g_lock_type lock_type ;
NTSTATUS status ;
2011-08-17 13:21:31 +04:00
TDB_DATA value ;
2009-10-25 18:12:12 +03:00
2011-08-17 13:21:31 +04:00
rec = dbwrap_fetch_locked ( ctx - > db , talloc_tos ( ) ,
string_term_tdb_data ( name ) ) ;
2009-10-25 18:12:12 +03:00
if ( rec = = NULL ) {
DEBUG ( 10 , ( " fetch_locked( \" %s \" ) failed \n " , name ) ) ;
status = NT_STATUS_INTERNAL_ERROR ;
goto done ;
}
2011-08-17 13:21:31 +04:00
value = dbwrap_record_get_value ( rec ) ;
if ( ! g_lock_parse ( talloc_tos ( ) , value , & num_locks , & locks ) ) {
2009-10-25 18:12:12 +03:00
DEBUG ( 10 , ( " g_lock_parse for %s failed \n " , name ) ) ;
2011-08-31 12:19:18 +04:00
status = NT_STATUS_FILE_INVALID ;
2009-10-25 18:12:12 +03:00
goto done ;
}
for ( i = 0 ; i < num_locks ; i + + ) {
if ( procid_equal ( & pid , & locks [ i ] . pid ) ) {
break ;
}
}
if ( i = = num_locks ) {
DEBUG ( 10 , ( " g_lock_force_unlock: Lock not found \n " ) ) ;
2011-08-31 12:19:18 +04:00
status = NT_STATUS_NOT_FOUND ;
2009-10-25 18:12:12 +03:00
goto done ;
}
lock_type = locks [ i ] . lock_type ;
if ( i < ( num_locks - 1 ) ) {
locks [ i ] = locks [ num_locks - 1 ] ;
}
num_locks - = 1 ;
if ( num_locks = = 0 ) {
2011-08-17 13:21:31 +04:00
status = dbwrap_record_delete ( rec ) ;
2009-10-25 18:12:12 +03:00
} else {
TDB_DATA data ;
data = make_tdb_data ( ( uint8_t * ) locks ,
sizeof ( struct g_lock_rec ) * num_locks ) ;
2011-08-17 13:21:31 +04:00
status = dbwrap_record_store ( rec , data , 0 ) ;
2009-10-25 18:12:12 +03:00
}
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " g_lock_force_unlock: Could not store record: %s \n " ,
nt_errstr ( status ) ) ) ;
goto done ;
}
2010-02-16 14:31:58 +03:00
TALLOC_FREE ( rec ) ;
2009-10-25 18:12:12 +03:00
if ( ( lock_type & G_LOCK_PENDING ) = = 0 ) {
2010-02-16 14:22:08 +03:00
int num_wakeups = 0 ;
2009-10-25 18:12:12 +03:00
/*
2010-02-16 14:22:08 +03:00
* We ' ve been the lock holder . Others to retry . Don ' t
* tell all others to avoid a thundering herd . In case
* this leads to a complete stall because we miss some
* processes , the loop in g_lock_lock tries at least
* once a minute .
2009-10-25 18:12:12 +03:00
*/
2010-02-16 14:22:08 +03:00
2009-10-25 18:12:12 +03:00
for ( i = 0 ; i < num_locks ; i + + ) {
if ( ( locks [ i ] . lock_type & G_LOCK_PENDING ) = = 0 ) {
continue ;
}
2010-02-16 14:22:08 +03:00
if ( ! process_exists ( locks [ i ] . pid ) ) {
continue ;
}
2009-10-25 18:12:12 +03:00
/*
* Ping all waiters to retry
*/
status = messaging_send ( ctx - > msg , locks [ i ] . pid ,
MSG_DBWRAP_G_LOCK_RETRY ,
& data_blob_null ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " sending retry to %s failed: %s \n " ,
2011-06-08 08:05:55 +04:00
server_id_str ( talloc_tos ( ) ,
2009-10-25 18:12:12 +03:00
& locks [ i ] . pid ) ,
nt_errstr ( status ) ) ) ;
2010-02-16 14:22:08 +03:00
} else {
num_wakeups + = 1 ;
}
if ( num_wakeups > 5 ) {
break ;
2009-10-25 18:12:12 +03:00
}
}
}
done :
2010-02-16 14:31:58 +03:00
/*
* For the error path , TALLOC_FREE ( rec ) as well . In the good
* path we have already freed it .
*/
TALLOC_FREE ( rec ) ;
2009-10-25 18:12:12 +03:00
TALLOC_FREE ( locks ) ;
return status ;
}
NTSTATUS g_lock_unlock ( struct g_lock_ctx * ctx , const char * name )
{
NTSTATUS status ;
2010-06-27 17:36:38 +04:00
status = g_lock_force_unlock ( ctx , name , messaging_server_id ( ctx - > msg ) ) ;
2009-10-25 18:12:12 +03:00
# ifdef CLUSTER_SUPPORT
if ( lp_clustering ( ) ) {
2010-08-31 18:52:56 +04:00
ctdb_unwatch ( messaging_ctdbd_connection ( ) ) ;
2009-10-25 18:12:12 +03:00
}
# endif
return status ;
}
struct g_lock_locks_state {
int ( * fn ) ( const char * name , void * private_data ) ;
void * private_data ;
} ;
static int g_lock_locks_fn ( struct db_record * rec , void * priv )
{
2011-08-17 13:21:31 +04:00
TDB_DATA key ;
2009-10-25 18:12:12 +03:00
struct g_lock_locks_state * state = ( struct g_lock_locks_state * ) priv ;
2011-08-17 13:21:31 +04:00
key = dbwrap_record_get_key ( rec ) ;
if ( ( key . dsize = = 0 ) | | ( key . dptr [ key . dsize - 1 ] ! = 0 ) ) {
2009-10-25 18:12:12 +03:00
DEBUG ( 1 , ( " invalid key in g_lock.tdb, ignoring \n " ) ) ;
return 0 ;
}
2011-08-17 13:21:31 +04:00
return state - > fn ( ( char * ) key . dptr , state - > private_data ) ;
2009-10-25 18:12:12 +03:00
}
int g_lock_locks ( struct g_lock_ctx * ctx ,
int ( * fn ) ( const char * name , void * private_data ) ,
void * private_data )
{
struct g_lock_locks_state state ;
2011-08-17 13:21:31 +04:00
NTSTATUS status ;
int count ;
2009-10-25 18:12:12 +03:00
state . fn = fn ;
state . private_data = private_data ;
2011-08-17 13:21:31 +04:00
status = dbwrap_traverse_read ( ctx - > db , g_lock_locks_fn , & state , & count ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return - 1 ;
} else {
return count ;
}
2009-10-25 18:12:12 +03:00
}
NTSTATUS g_lock_dump ( struct g_lock_ctx * ctx , const char * name ,
int ( * fn ) ( struct server_id pid ,
enum g_lock_type lock_type ,
void * private_data ) ,
void * private_data )
{
TDB_DATA data ;
int i , num_locks ;
struct g_lock_rec * locks = NULL ;
bool ret ;
2011-08-17 13:21:31 +04:00
NTSTATUS status ;
2009-10-25 18:12:12 +03:00
2011-08-17 13:21:31 +04:00
status = dbwrap_fetch_bystring ( ctx - > db , talloc_tos ( ) , name , & data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2009-10-25 18:12:12 +03:00
}
if ( ( data . dsize = = 0 ) | | ( data . dptr = = NULL ) ) {
return NT_STATUS_OK ;
}
ret = g_lock_parse ( talloc_tos ( ) , data , & num_locks , & locks ) ;
TALLOC_FREE ( data . dptr ) ;
if ( ! ret ) {
DEBUG ( 10 , ( " g_lock_parse for %s failed \n " , name ) ) ;
return NT_STATUS_INTERNAL_ERROR ;
}
for ( i = 0 ; i < num_locks ; i + + ) {
if ( fn ( locks [ i ] . pid , locks [ i ] . lock_type , private_data ) ! = 0 ) {
break ;
}
}
TALLOC_FREE ( locks ) ;
return NT_STATUS_OK ;
}
struct g_lock_get_state {
bool found ;
struct server_id * pid ;
} ;
static int g_lock_get_fn ( struct server_id pid , enum g_lock_type lock_type ,
void * priv )
{
struct g_lock_get_state * state = ( struct g_lock_get_state * ) priv ;
if ( ( lock_type & G_LOCK_PENDING ) ! = 0 ) {
return 0 ;
}
state - > found = true ;
* state - > pid = pid ;
return 1 ;
}
NTSTATUS g_lock_get ( struct g_lock_ctx * ctx , const char * name ,
struct server_id * pid )
{
struct g_lock_get_state state ;
NTSTATUS status ;
state . found = false ;
state . pid = pid ;
status = g_lock_dump ( ctx , name , g_lock_get_fn , & state ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( ! state . found ) {
return NT_STATUS_NOT_FOUND ;
}
return NT_STATUS_OK ;
}
2010-03-12 16:22:54 +03:00
static bool g_lock_init_all ( TALLOC_CTX * mem_ctx ,
struct tevent_context * * pev ,
struct messaging_context * * pmsg ,
struct g_lock_ctx * * pg_ctx )
{
struct tevent_context * ev = NULL ;
struct messaging_context * msg = NULL ;
struct g_lock_ctx * g_ctx = NULL ;
ev = tevent_context_init ( mem_ctx ) ;
if ( ev = = NULL ) {
d_fprintf ( stderr , " ERROR: could not init event context \n " ) ;
goto fail ;
}
2011-12-12 17:55:54 +04:00
msg = messaging_init ( mem_ctx , ev ) ;
2010-03-12 16:22:54 +03:00
if ( msg = = NULL ) {
d_fprintf ( stderr , " ERROR: could not init messaging context \n " ) ;
goto fail ;
}
g_ctx = g_lock_ctx_init ( mem_ctx , msg ) ;
if ( g_ctx = = NULL ) {
d_fprintf ( stderr , " ERROR: could not init g_lock context \n " ) ;
goto fail ;
}
* pev = ev ;
* pmsg = msg ;
* pg_ctx = g_ctx ;
return true ;
fail :
TALLOC_FREE ( g_ctx ) ;
TALLOC_FREE ( msg ) ;
TALLOC_FREE ( ev ) ;
return false ;
}
NTSTATUS g_lock_do ( const char * name , enum g_lock_type lock_type ,
2011-12-12 17:55:54 +04:00
struct timeval timeout ,
2010-03-12 16:22:54 +03:00
void ( * fn ) ( void * private_data ) , void * private_data )
{
struct tevent_context * ev = NULL ;
struct messaging_context * msg = NULL ;
struct g_lock_ctx * g_ctx = NULL ;
NTSTATUS status ;
2011-12-12 17:55:54 +04:00
if ( ! g_lock_init_all ( talloc_tos ( ) , & ev , & msg , & g_ctx ) ) {
2010-03-12 16:22:54 +03:00
status = NT_STATUS_ACCESS_DENIED ;
goto done ;
}
status = g_lock_lock ( g_ctx , name , lock_type , timeout ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
fn ( private_data ) ;
g_lock_unlock ( g_ctx , name ) ;
done :
TALLOC_FREE ( g_ctx ) ;
TALLOC_FREE ( msg ) ;
TALLOC_FREE ( ev ) ;
return status ;
}