2007-04-21 07:23:42 +00:00
/*
wait for a tdb chain lock
Copyright ( C ) Andrew Tridgell 2006
2007-07-10 08:06:51 +00:00
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 ,
2007-04-21 07:23:42 +00:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2007-07-10 08:06:51 +00:00
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/>.
2007-04-21 07:23:42 +00:00
*/
# include "includes.h"
# include "lib/events/events.h"
# include "system/filesys.h"
# include "system/wait.h"
# include "lib/tdb/include/tdb.h"
# include "../include/ctdb_private.h"
struct lockwait_handle {
struct ctdb_context * ctdb ;
2007-07-10 08:06:51 +00:00
struct ctdb_db_context * ctdb_db ;
2007-04-21 07:23:42 +00:00
struct fd_event * fde ;
int fd [ 2 ] ;
pid_t child ;
void * private_data ;
void ( * callback ) ( void * ) ;
2007-07-10 08:06:51 +00:00
TDB_DATA key ;
2007-04-21 07:23:42 +00:00
struct timeval start_time ;
} ;
static void lockwait_handler ( struct event_context * ev , struct fd_event * fde ,
uint16_t flags , void * private_data )
{
struct lockwait_handle * h = talloc_get_type ( private_data ,
struct lockwait_handle ) ;
void ( * callback ) ( void * ) = h - > callback ;
void * p = h - > private_data ;
pid_t child = h - > child ;
2007-07-10 08:06:51 +00:00
TDB_DATA key = h - > key ;
struct tdb_context * tdb = h - > ctdb_db - > ltdb - > tdb ;
TALLOC_CTX * tmp_ctx = talloc_new ( ev ) ;
key . dptr = talloc_memdup ( tmp_ctx , key . dptr , key . dsize ) ;
2007-04-21 07:23:42 +00:00
talloc_set_destructor ( h , NULL ) ;
2007-07-10 08:06:51 +00:00
ctdb_latency ( & h - > ctdb - > statistics . max_lockwait_latency , h - > start_time ) ;
h - > ctdb - > statistics . pending_lockwait_calls - - ;
/* the handle needs to go away when the context is gone - when
the handle goes away this implicitly closes the pipe , which
kills the child holding the lock */
talloc_steal ( tmp_ctx , h ) ;
if ( h - > ctdb - > flags & CTDB_FLAG_TORTURE ) {
if ( tdb_chainlock_nonblock ( tdb , key ) = = 0 ) {
ctdb_fatal ( h - > ctdb , " got chain lock while lockwait child active " ) ;
}
}
tdb_chainlock_mark ( tdb , key ) ;
2007-04-21 07:23:42 +00:00
callback ( p ) ;
2007-07-10 08:06:51 +00:00
tdb_chainlock_unmark ( tdb , key ) ;
kill ( child , SIGKILL ) ;
2007-04-21 07:23:42 +00:00
waitpid ( child , NULL , 0 ) ;
2007-07-10 08:06:51 +00:00
talloc_free ( tmp_ctx ) ;
2007-04-21 07:23:42 +00:00
}
static int lockwait_destructor ( struct lockwait_handle * h )
{
2007-07-10 08:06:51 +00:00
h - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-21 07:23:42 +00:00
kill ( h - > child , SIGKILL ) ;
waitpid ( h - > child , NULL , 0 ) ;
return 0 ;
}
/*
setup a non - blocking chainlock on a tdb record . If this function
returns NULL then it could not get the chainlock . Otherwise it
returns a opaque handle , and will call callback ( ) once it has
managed to get the chainlock . You can cancel it by using talloc_free
on the returned handle .
It is the callers responsibility to unlock the chainlock once
acquired
*/
struct lockwait_handle * ctdb_lockwait ( struct ctdb_db_context * ctdb_db ,
TDB_DATA key ,
void ( * callback ) ( void * private_data ) ,
void * private_data )
{
struct lockwait_handle * result ;
int ret ;
2007-07-10 08:06:51 +00:00
pid_t parent = getpid ( ) ;
2007-04-21 07:23:42 +00:00
2007-07-10 08:06:51 +00:00
ctdb_db - > ctdb - > statistics . lockwait_calls + + ;
ctdb_db - > ctdb - > statistics . pending_lockwait_calls + + ;
2007-04-21 07:23:42 +00:00
2007-07-10 08:06:51 +00:00
if ( ! ( result = talloc_zero ( private_data , struct lockwait_handle ) ) ) {
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-21 07:23:42 +00:00
return NULL ;
}
ret = pipe ( result - > fd ) ;
if ( ret ! = 0 ) {
talloc_free ( result ) ;
2007-07-10 08:06:51 +00:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-21 07:23:42 +00:00
return NULL ;
}
result - > child = fork ( ) ;
if ( result - > child = = ( pid_t ) - 1 ) {
close ( result - > fd [ 0 ] ) ;
close ( result - > fd [ 1 ] ) ;
talloc_free ( result ) ;
2007-07-10 08:06:51 +00:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-21 07:23:42 +00:00
return NULL ;
}
result - > callback = callback ;
result - > private_data = private_data ;
result - > ctdb = ctdb_db - > ctdb ;
2007-07-10 08:06:51 +00:00
result - > ctdb_db = ctdb_db ;
result - > key = key ;
2007-04-21 07:23:42 +00:00
if ( result - > child = = 0 ) {
2007-07-10 08:06:51 +00:00
char c = 0 ;
2007-04-21 07:23:42 +00:00
close ( result - > fd [ 0 ] ) ;
tdb_chainlock ( ctdb_db - > ltdb - > tdb , key ) ;
2007-07-10 08:06:51 +00:00
write ( result - > fd [ 1 ] , & c , 1 ) ;
/* make sure we die when our parent dies */
while ( kill ( parent , 0 ) = = 0 | | errno ! = ESRCH ) {
sleep ( 5 ) ;
}
2007-04-21 07:23:42 +00:00
_exit ( 0 ) ;
}
close ( result - > fd [ 1 ] ) ;
talloc_set_destructor ( result , lockwait_destructor ) ;
result - > fde = event_add_fd ( ctdb_db - > ctdb - > ev , result , result - > fd [ 0 ] ,
2007-07-10 08:06:51 +00:00
EVENT_FD_READ | EVENT_FD_AUTOCLOSE , lockwait_handler ,
2007-04-21 07:23:42 +00:00
( void * ) result ) ;
if ( result - > fde = = NULL ) {
talloc_free ( result ) ;
2007-07-10 08:06:51 +00:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-21 07:23:42 +00:00
return NULL ;
}
result - > start_time = timeval_current ( ) ;
return result ;
}