2007-04-17 07:41:27 +10:00
/*
wait for a tdb chain lock
Copyright ( C ) Andrew Tridgell 2006
2007-05-31 13:50:53 +10: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
2007-07-10 15:29:31 +10:00
the Free Software Foundation ; either version 3 of the License , or
2007-05-31 13:50:53 +10:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
2007-04-17 07:41:27 +10:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2007-05-31 13:50:53 +10: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
2007-07-10 15:29:31 +10:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2007-04-17 07:41:27 +10:00
*/
# include "includes.h"
# include "lib/events/events.h"
# include "system/filesys.h"
# include "system/wait.h"
# include "db_wrap.h"
# include "lib/tdb/include/tdb.h"
2007-04-21 13:08:22 +10:00
# include "../include/ctdb_private.h"
2007-04-17 07:41:27 +10:00
struct lockwait_handle {
2007-04-20 20:07:47 +10:00
struct ctdb_context * ctdb ;
2007-05-05 17:19:59 +10:00
struct ctdb_db_context * ctdb_db ;
2007-04-17 07:41:27 +10:00
struct fd_event * fde ;
int fd [ 2 ] ;
pid_t child ;
void * private_data ;
void ( * callback ) ( void * ) ;
2007-05-05 17:19:59 +10:00
TDB_DATA key ;
2007-04-20 21:02:53 +10:00
struct timeval start_time ;
2007-04-17 07:41:27 +10:00
} ;
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 ;
2007-04-17 09:10:52 +10:00
pid_t child = h - > child ;
2007-05-05 17:19:59 +10: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-17 07:41:27 +10:00
talloc_set_destructor ( h , NULL ) ;
2007-05-29 12:16:59 +10:00
ctdb_latency ( & h - > ctdb - > statistics . max_lockwait_latency , h - > start_time ) ;
h - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-05-05 17:19:59 +10:00
2007-05-15 09:44:03 +10:00
/* the handle needs to go away when the context is gone - when
the handle goes away this implicitly closes the pipe , which
2007-05-14 14:01:33 +10:00
kills the child holding the lock */
2007-05-15 09:44:03 +10:00
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 " ) ;
}
}
2007-05-14 14:01:33 +10:00
2007-05-05 17:19:59 +10:00
tdb_chainlock_mark ( tdb , key ) ;
2007-04-17 07:41:27 +10:00
callback ( p ) ;
2007-05-05 17:19:59 +10:00
tdb_chainlock_unmark ( tdb , key ) ;
kill ( child , SIGKILL ) ;
talloc_free ( tmp_ctx ) ;
2007-04-17 07:41:27 +10:00
}
static int lockwait_destructor ( struct lockwait_handle * h )
{
2007-05-29 12:16:59 +10:00
h - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-17 07:41:27 +10:00
kill ( h - > child , SIGKILL ) ;
return 0 ;
}
2007-04-17 09:14:52 +10:00
/*
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
*/
2007-04-17 07:41:27 +10:00
struct lockwait_handle * ctdb_lockwait ( struct ctdb_db_context * ctdb_db ,
TDB_DATA key ,
2007-04-17 11:26:59 +10:00
void ( * callback ) ( void * private_data ) ,
void * private_data )
2007-04-17 07:41:27 +10:00
{
2007-04-17 11:26:59 +10:00
struct lockwait_handle * result ;
2007-04-17 07:41:27 +10:00
int ret ;
2007-05-15 09:44:03 +10:00
pid_t parent = getpid ( ) ;
2007-04-17 07:41:27 +10:00
2007-05-29 12:16:59 +10:00
ctdb_db - > ctdb - > statistics . lockwait_calls + + ;
ctdb_db - > ctdb - > statistics . pending_lockwait_calls + + ;
2007-04-20 20:07:47 +10:00
2007-05-05 17:19:59 +10:00
if ( ! ( result = talloc_zero ( private_data , struct lockwait_handle ) ) ) {
2007-05-29 12:16:59 +10:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-17 07:41:27 +10:00
return NULL ;
}
2007-04-17 11:26:59 +10:00
ret = pipe ( result - > fd ) ;
2007-04-17 07:41:27 +10:00
if ( ret ! = 0 ) {
2007-04-17 11:26:59 +10:00
talloc_free ( result ) ;
2007-05-29 12:16:59 +10:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-17 07:41:27 +10:00
return NULL ;
}
2007-04-17 11:26:59 +10:00
result - > child = fork ( ) ;
if ( result - > child = = ( pid_t ) - 1 ) {
close ( result - > fd [ 0 ] ) ;
close ( result - > fd [ 1 ] ) ;
talloc_free ( result ) ;
2007-05-29 12:16:59 +10:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-17 07:41:27 +10:00
return NULL ;
}
2007-04-17 11:26:59 +10:00
result - > callback = callback ;
result - > private_data = private_data ;
2007-04-20 20:07:47 +10:00
result - > ctdb = ctdb_db - > ctdb ;
2007-05-05 17:19:59 +10:00
result - > ctdb_db = ctdb_db ;
result - > key = key ;
2007-04-17 07:41:27 +10:00
2007-04-17 11:26:59 +10:00
if ( result - > child = = 0 ) {
2007-05-05 17:19:59 +10:00
char c = 0 ;
2007-04-17 11:26:59 +10:00
close ( result - > fd [ 0 ] ) ;
tdb_chainlock ( ctdb_db - > ltdb - > tdb , key ) ;
2007-05-05 17:19:59 +10:00
write ( result - > fd [ 1 ] , & c , 1 ) ;
2007-05-15 09:44:03 +10:00
/* make sure we die when our parent dies */
while ( kill ( parent , 0 ) = = 0 | | errno ! = ESRCH ) {
sleep ( 5 ) ;
}
2007-04-20 17:58:37 +10:00
_exit ( 0 ) ;
2007-04-17 07:41:27 +10:00
}
2007-04-17 11:26:59 +10:00
close ( result - > fd [ 1 ] ) ;
talloc_set_destructor ( result , lockwait_destructor ) ;
2007-04-17 07:41:27 +10:00
2007-04-17 11:26:59 +10:00
result - > fde = event_add_fd ( ctdb_db - > ctdb - > ev , result , result - > fd [ 0 ] ,
2007-05-05 17:19:59 +10:00
EVENT_FD_READ | EVENT_FD_AUTOCLOSE , lockwait_handler ,
2007-04-17 11:26:59 +10:00
( void * ) result ) ;
if ( result - > fde = = NULL ) {
talloc_free ( result ) ;
2007-05-29 12:16:59 +10:00
ctdb_db - > ctdb - > statistics . pending_lockwait_calls - - ;
2007-04-17 07:41:27 +10:00
return NULL ;
}
2007-04-20 21:02:53 +10:00
result - > start_time = timeval_current ( ) ;
2007-04-20 17:58:37 +10:00
2007-04-17 11:26:59 +10:00
return result ;
2007-04-17 07:41:27 +10:00
}