2006-05-24 18:19:34 +04:00
/*
Unix SMB / CIFS implementation .
testing of the events subsystem
2009-02-16 10:52:06 +03:00
Copyright ( C ) Stefan Metzmacher 2006 - 2009
2013-02-15 02:16:31 +04:00
Copyright ( C ) Jeremy Allison 2013
2009-02-16 10:52:06 +03:00
* * NOTE ! The following LGPL license applies to the tevent
* * library . This does NOT imply that all of Samba is released
* * under the LGPL
This library is free software ; you can redistribute it and / or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation ; either
version 3 of the License , or ( at your option ) any later version .
This library is distributed in the hope that it will be useful ,
2006-05-24 18:19:34 +04:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2009-02-16 10:52:06 +03:00
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
Lesser General Public License for more details .
You should have received a copy of the GNU Lesser General Public
License along with this library ; if not , see < http : //www.gnu.org/licenses/>.
2006-05-24 18:19:34 +04:00
*/
# include "includes.h"
2019-01-08 17:25:22 +03:00
# define TEVENT_DEPRECATED 1
2017-08-15 01:54:39 +03:00
# include "tevent.h"
2006-05-24 18:19:34 +04:00
# include "system/filesys.h"
2012-07-30 11:09:46 +04:00
# include "system/select.h"
2013-02-23 00:48:44 +04:00
# include "system/network.h"
2006-05-24 18:19:34 +04:00
# include "torture/torture.h"
2014-02-27 12:08:17 +04:00
# include "torture/local/proto.h"
2012-07-30 11:09:46 +04:00
# ifdef HAVE_PTHREAD
2018-05-22 16:43:12 +03:00
# include "system/threads.h"
2012-07-30 11:09:46 +04:00
# include <assert.h>
# endif
2006-05-24 18:19:34 +04:00
2023-01-10 14:57:58 +03:00
static struct tevent_context *
test_tevent_context_init ( TALLOC_CTX * mem_ctx )
{
struct tevent_context * ev = NULL ;
ev = tevent_context_init ( mem_ctx ) ;
if ( ev ! = NULL ) {
samba_tevent_set_debug ( ev , " <default> " ) ;
}
return ev ;
}
static struct tevent_context *
test_tevent_context_init_byname ( TALLOC_CTX * mem_ctx , const char * name )
{
struct tevent_context * ev = NULL ;
ev = tevent_context_init_byname ( mem_ctx , name ) ;
if ( ev ! = NULL ) {
samba_tevent_set_debug ( ev , name ) ;
}
return ev ;
}
2006-05-24 18:19:34 +04:00
static int fde_count ;
2016-04-05 19:40:37 +03:00
static void do_read ( int fd , void * buf , size_t count )
{
ssize_t ret ;
do {
ret = read ( fd , buf , count ) ;
} while ( ret = = - 1 & & errno = = EINTR ) ;
}
2013-02-15 02:16:31 +04:00
static void fde_handler_read ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
2009-02-02 14:59:28 +03:00
uint16_t flags , void * private_data )
2006-05-24 18:19:34 +04:00
{
2009-02-02 14:59:28 +03:00
int * fd = ( int * ) private_data ;
2007-01-21 13:32:39 +03:00
char c ;
# ifdef SA_SIGINFO
kill ( getpid ( ) , SIGUSR1 ) ;
# endif
kill ( getpid ( ) , SIGALRM ) ;
2013-02-15 02:16:31 +04:00
2016-04-05 19:40:37 +03:00
do_read ( fd [ 0 ] , & c , 1 ) ;
2006-05-24 18:19:34 +04:00
fde_count + + ;
}
2016-04-05 19:51:13 +03:00
static void do_write ( int fd , void * buf , size_t count )
{
ssize_t ret ;
do {
ret = write ( fd , buf , count ) ;
} while ( ret = = - 1 & & errno = = EINTR ) ;
}
2013-02-15 02:16:31 +04:00
static void fde_handler_write ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
uint16_t flags , void * private_data )
{
int * fd = ( int * ) private_data ;
char c = 0 ;
2016-04-05 19:51:13 +03:00
do_write ( fd [ 1 ] , & c , 1 ) ;
2013-02-15 02:16:31 +04:00
}
2013-03-08 22:09:01 +04:00
/* This will only fire if the fd's returned from pipe() are bi-directional. */
2013-02-15 02:16:31 +04:00
static void fde_handler_read_1 ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
uint16_t flags , void * private_data )
{
2013-03-08 22:09:01 +04:00
int * fd = ( int * ) private_data ;
char c ;
# ifdef SA_SIGINFO
kill ( getpid ( ) , SIGUSR1 ) ;
# endif
kill ( getpid ( ) , SIGALRM ) ;
2016-04-05 19:40:37 +03:00
do_read ( fd [ 1 ] , & c , 1 ) ;
2013-03-08 22:09:01 +04:00
fde_count + + ;
2013-02-15 02:16:31 +04:00
}
2013-03-08 22:09:01 +04:00
/* This will only fire if the fd's returned from pipe() are bi-directional. */
2013-02-15 02:16:31 +04:00
static void fde_handler_write_1 ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
uint16_t flags , void * private_data )
{
2013-03-08 22:09:01 +04:00
int * fd = ( int * ) private_data ;
char c = 0 ;
2016-04-05 19:51:13 +03:00
do_write ( fd [ 0 ] , & c , 1 ) ;
2013-02-15 02:16:31 +04:00
}
2008-12-29 22:24:57 +03:00
static void finished_handler ( struct tevent_context * ev_ctx , struct tevent_timer * te ,
2009-02-02 14:59:28 +03:00
struct timeval tval , void * private_data )
2006-05-24 18:19:34 +04:00
{
2009-02-02 14:59:28 +03:00
int * finished = ( int * ) private_data ;
2007-01-21 13:32:39 +03:00
( * finished ) = 1 ;
2006-05-24 18:19:34 +04:00
}
2012-05-14 13:48:00 +04:00
static void count_handler ( struct tevent_context * ev_ctx , struct tevent_signal * te ,
2009-02-02 14:59:28 +03:00
int signum , int count , void * info , void * private_data )
2007-01-21 13:32:39 +03:00
{
2009-02-02 14:59:28 +03:00
int * countp = ( int * ) private_data ;
2007-01-21 13:32:39 +03:00
( * countp ) + = count ;
}
static bool test_event_context ( struct torture_context * test ,
const void * test_data )
2006-05-24 18:19:34 +04:00
{
2008-12-29 22:24:57 +03:00
struct tevent_context * ev_ctx ;
2006-05-24 18:19:34 +04:00
int fd [ 2 ] = { - 1 , - 1 } ;
2007-01-05 12:35:49 +03:00
const char * backend = ( const char * ) test_data ;
2007-01-21 13:32:39 +03:00
int alarm_count = 0 , info_count = 0 ;
2013-02-15 02:16:31 +04:00
struct tevent_fd * fde_read ;
struct tevent_fd * fde_read_1 ;
struct tevent_fd * fde_write ;
struct tevent_fd * fde_write_1 ;
2009-07-16 11:06:42 +04:00
# ifdef SA_RESTART
struct tevent_signal * se1 = NULL ;
# endif
2013-02-23 00:48:11 +04:00
# ifdef SA_RESETHAND
2009-07-16 11:06:42 +04:00
struct tevent_signal * se2 = NULL ;
2013-02-23 00:48:11 +04:00
# endif
2009-07-16 11:06:42 +04:00
# ifdef SA_SIGINFO
struct tevent_signal * se3 = NULL ;
# endif
2007-01-21 13:32:39 +03:00
int finished = 0 ;
struct timeval t ;
2015-03-31 22:15:53 +03:00
int ret ;
2007-01-21 13:32:39 +03:00
2023-01-10 14:57:58 +03:00
ev_ctx = test_tevent_context_init_byname ( test , backend ) ;
2007-01-05 12:35:49 +03:00
if ( ev_ctx = = NULL ) {
torture_comment ( test , " event backend '%s' not supported \n " , backend ) ;
return true ;
}
2013-02-23 00:48:11 +04:00
torture_comment ( test , " backend '%s' - %s \n " ,
backend , __FUNCTION__ ) ;
2007-01-05 12:35:49 +03:00
2006-05-24 18:19:34 +04:00
/* reset globals */
fde_count = 0 ;
/* create a pipe */
2015-03-31 22:15:53 +03:00
ret = pipe ( fd ) ;
torture_assert_int_equal ( test , ret , 0 , " pipe failed " ) ;
2006-05-24 18:19:34 +04:00
2013-02-15 02:16:31 +04:00
fde_read = tevent_add_fd ( ev_ctx , ev_ctx , fd [ 0 ] , TEVENT_FD_READ ,
fde_handler_read , fd ) ;
fde_write_1 = tevent_add_fd ( ev_ctx , ev_ctx , fd [ 0 ] , TEVENT_FD_WRITE ,
2013-03-08 22:09:01 +04:00
fde_handler_write_1 , fd ) ;
2013-02-15 02:16:31 +04:00
fde_write = tevent_add_fd ( ev_ctx , ev_ctx , fd [ 1 ] , TEVENT_FD_WRITE ,
fde_handler_write , fd ) ;
fde_read_1 = tevent_add_fd ( ev_ctx , ev_ctx , fd [ 1 ] , TEVENT_FD_READ ,
2013-03-08 22:09:01 +04:00
fde_handler_read_1 , fd ) ;
2013-02-15 02:16:31 +04:00
tevent_fd_set_auto_close ( fde_read ) ;
tevent_fd_set_auto_close ( fde_write ) ;
2007-01-21 13:32:39 +03:00
2012-05-14 13:48:00 +04:00
tevent_add_timer ( ev_ctx , ev_ctx , timeval_current_ofs ( 2 , 0 ) ,
finished_handler , & finished ) ;
2006-05-24 18:19:34 +04:00
2009-07-16 11:06:42 +04:00
# ifdef SA_RESTART
2012-05-14 13:48:00 +04:00
se1 = tevent_add_signal ( ev_ctx , ev_ctx , SIGALRM , SA_RESTART , count_handler , & alarm_count ) ;
2013-02-23 00:48:11 +04:00
torture_assert ( test , se1 ! = NULL , " failed to setup se1 " ) ;
2009-07-16 11:06:42 +04:00
# endif
2011-08-01 22:49:10 +04:00
# ifdef SA_RESETHAND
2012-05-14 13:48:00 +04:00
se2 = tevent_add_signal ( ev_ctx , ev_ctx , SIGALRM , SA_RESETHAND , count_handler , & alarm_count ) ;
2013-02-23 00:48:11 +04:00
torture_assert ( test , se2 ! = NULL , " failed to setup se2 " ) ;
2011-08-01 22:49:10 +04:00
# endif
2007-01-21 13:32:39 +03:00
# ifdef SA_SIGINFO
2012-05-14 13:48:00 +04:00
se3 = tevent_add_signal ( ev_ctx , ev_ctx , SIGUSR1 , SA_SIGINFO , count_handler , & info_count ) ;
2013-02-23 00:48:11 +04:00
torture_assert ( test , se3 ! = NULL , " failed to setup se3 " ) ;
2007-01-21 13:32:39 +03:00
# endif
2006-05-24 18:19:34 +04:00
2007-01-21 13:32:39 +03:00
t = timeval_current ( ) ;
while ( ! finished ) {
2007-08-10 11:40:50 +04:00
errno = 0 ;
2012-05-14 13:48:00 +04:00
if ( tevent_loop_once ( ev_ctx ) = = - 1 ) {
2014-10-23 08:54:10 +04:00
TALLOC_FREE ( ev_ctx ) ;
2007-08-17 09:21:05 +04:00
torture_fail ( test , talloc_asprintf ( test , " Failed event loop %s \n " , strerror ( errno ) ) ) ;
2014-10-23 08:54:10 +04:00
return false ;
2007-01-24 07:30:44 +03:00
}
2007-01-21 13:32:39 +03:00
}
2013-02-15 02:16:31 +04:00
talloc_free ( fde_read_1 ) ;
talloc_free ( fde_write_1 ) ;
2015-05-30 02:42:08 +03:00
talloc_free ( fde_read ) ;
talloc_free ( fde_write ) ;
2007-01-21 13:32:39 +03:00
while ( alarm_count < fde_count + 1 ) {
2012-05-14 13:48:00 +04:00
if ( tevent_loop_once ( ev_ctx ) = = - 1 ) {
2007-01-24 07:30:44 +03:00
break ;
}
2007-01-21 13:32:39 +03:00
}
torture_comment ( test , " Got %.2f pipe events/sec \n " , fde_count / timeval_elapsed ( & t ) ) ;
2009-07-16 11:06:42 +04:00
# ifdef SA_RESTART
2007-01-21 13:32:39 +03:00
talloc_free ( se1 ) ;
2009-07-16 11:06:42 +04:00
# endif
2007-01-21 13:32:39 +03:00
torture_assert_int_equal ( test , alarm_count , 1 + fde_count , " alarm count mismatch " ) ;
2013-02-23 00:48:11 +04:00
# ifdef SA_RESETHAND
/*
* we do not call talloc_free ( se2 )
* because it is already gone ,
* after triggering the event handler .
*/
# endif
2007-01-21 13:32:39 +03:00
# ifdef SA_SIGINFO
talloc_free ( se3 ) ;
torture_assert_int_equal ( test , info_count , fde_count , " info count mismatch " ) ;
# endif
2006-05-24 18:19:34 +04:00
2006-06-17 02:06:09 +04:00
talloc_free ( ev_ctx ) ;
2007-01-21 13:32:39 +03:00
2006-10-16 17:06:41 +04:00
return true ;
2006-05-24 18:19:34 +04:00
}
2023-01-27 14:12:45 +03:00
static void fde_handler_do_read ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
uint16_t flags , void * private_data )
{
int * fd = ( int * ) private_data ;
char c = 0 ;
do_read ( fd [ 0 ] , & c , 1 ) ;
fde_count + + ;
}
static void fde_handler_do_write ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
uint16_t flags , void * private_data )
{
int * fd = ( int * ) private_data ;
char c = 0 ;
do_write ( fd [ 1 ] , & c , 1 ) ;
}
static void fde_handler_ignore ( struct tevent_context * ev_ctx , struct tevent_fd * f ,
uint16_t flags , void * private_data )
{
}
static bool test_fd_speedX ( struct torture_context * test ,
const void * test_data ,
size_t additional_fdes )
{
struct tevent_context * ev_ctx = NULL ;
int fd [ 2 ] = { - 1 , - 1 } ;
const char * backend = ( const char * ) test_data ;
struct tevent_fd * fde_read = NULL ;
struct tevent_fd * fde_write = NULL ;
int finished = 0 ;
struct timeval t ;
size_t i ;
int ret ;
ev_ctx = test_tevent_context_init_byname ( test , backend ) ;
if ( ev_ctx = = NULL ) {
torture_comment ( test , " event backend '%s' not supported \n " , backend ) ;
return true ;
}
torture_comment ( test , " backend '%s' - test_fd_speed%zu \n " ,
backend , 1 + additional_fdes ) ;
/* reset globals */
fde_count = 0 ;
/* create a pipe */
ret = pipe ( fd ) ;
torture_assert_int_equal ( test , ret , 0 , " pipe failed " ) ;
fde_read = tevent_add_fd ( ev_ctx , ev_ctx , fd [ 0 ] , TEVENT_FD_READ ,
fde_handler_do_read , fd ) ;
fde_write = tevent_add_fd ( ev_ctx , ev_ctx , fd [ 1 ] , TEVENT_FD_WRITE ,
fde_handler_do_write , fd ) ;
for ( i = 0 ; i < additional_fdes ; i + + ) {
tevent_add_fd ( ev_ctx , ev_ctx , fd [ 0 ] , TEVENT_FD_WRITE ,
fde_handler_ignore , fd ) ;
tevent_add_fd ( ev_ctx , ev_ctx , fd [ 1 ] , TEVENT_FD_READ ,
fde_handler_ignore , fd ) ;
}
tevent_fd_set_auto_close ( fde_read ) ;
tevent_fd_set_auto_close ( fde_write ) ;
tevent_add_timer ( ev_ctx , ev_ctx , timeval_current_ofs ( 600 , 0 ) ,
finished_handler , & finished ) ;
t = timeval_current ( ) ;
while ( ! finished & & fde_count < 1000000 ) {
errno = 0 ;
if ( tevent_loop_once ( ev_ctx ) = = - 1 ) {
TALLOC_FREE ( ev_ctx ) ;
torture_fail ( test , talloc_asprintf ( test , " Failed event loop %s \n " , strerror ( errno ) ) ) ;
return false ;
}
}
talloc_free ( fde_read ) ;
talloc_free ( fde_write ) ;
torture_comment ( test , " Got %.2f pipe events \n " , ( double ) fde_count ) ;
torture_comment ( test , " Got %.2f pipe events/sec \n " , fde_count / timeval_elapsed ( & t ) ) ;
talloc_free ( ev_ctx ) ;
return true ;
}
static bool test_fd_speed1 ( struct torture_context * test ,
const void * test_data )
{
return test_fd_speedX ( test , test_data , 0 ) ;
}
static bool test_fd_speed2 ( struct torture_context * test ,
const void * test_data )
{
return test_fd_speedX ( test , test_data , 1 ) ;
}
2013-02-23 00:48:44 +04:00
struct test_event_fd1_state {
struct torture_context * tctx ;
const char * backend ;
struct tevent_context * ev ;
int sock [ 2 ] ;
struct tevent_timer * te ;
struct tevent_fd * fde0 ;
struct tevent_fd * fde1 ;
bool got_write ;
bool got_read ;
bool drain ;
bool drain_done ;
unsigned loop_count ;
bool finished ;
const char * error ;
} ;
static void test_event_fd1_fde_handler ( struct tevent_context * ev_ctx ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data )
{
struct test_event_fd1_state * state =
( struct test_event_fd1_state * ) private_data ;
if ( state - > drain_done ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
if ( state - > drain ) {
ssize_t ret ;
uint8_t c = 0 ;
if ( ! ( flags & TEVENT_FD_READ ) ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
ret = read ( state - > sock [ 0 ] , & c , 1 ) ;
if ( ret = = 1 ) {
return ;
}
/*
* end of test . . .
*/
tevent_fd_set_flags ( fde , 0 ) ;
state - > drain_done = true ;
return ;
}
if ( ! state - > got_write ) {
uint8_t c = 0 ;
if ( flags ! = TEVENT_FD_WRITE ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
state - > got_write = true ;
/*
* we write to the other socket . . .
*/
2016-04-05 19:51:13 +03:00
do_write ( state - > sock [ 1 ] , & c , 1 ) ;
2013-02-23 00:48:44 +04:00
TEVENT_FD_NOT_WRITEABLE ( fde ) ;
TEVENT_FD_READABLE ( fde ) ;
return ;
}
if ( ! state - > got_read ) {
if ( flags ! = TEVENT_FD_READ ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
state - > got_read = true ;
TEVENT_FD_NOT_READABLE ( fde ) ;
return ;
}
state - > finished = true ;
state - > error = __location__ ;
return ;
}
static void test_event_fd1_finished ( struct tevent_context * ev_ctx ,
struct tevent_timer * te ,
struct timeval tval ,
void * private_data )
{
struct test_event_fd1_state * state =
( struct test_event_fd1_state * ) private_data ;
if ( state - > drain_done ) {
state - > finished = true ;
return ;
}
if ( ! state - > got_write ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
if ( ! state - > got_read ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
state - > loop_count + + ;
if ( state - > loop_count > 3 ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
state - > got_write = false ;
state - > got_read = false ;
tevent_fd_set_flags ( state - > fde0 , TEVENT_FD_WRITE ) ;
if ( state - > loop_count > 2 ) {
state - > drain = true ;
TALLOC_FREE ( state - > fde1 ) ;
TEVENT_FD_READABLE ( state - > fde0 ) ;
}
state - > te = tevent_add_timer ( state - > ev , state - > ev ,
timeval_current_ofs ( 0 , 2000 ) ,
test_event_fd1_finished , state ) ;
}
static bool test_event_fd1 ( struct torture_context * tctx ,
const void * test_data )
{
struct test_event_fd1_state state ;
2018-04-10 21:58:11 +03:00
int ret ;
2013-02-23 00:48:44 +04:00
ZERO_STRUCT ( state ) ;
state . tctx = tctx ;
state . backend = ( const char * ) test_data ;
2023-01-10 14:57:58 +03:00
state . ev = test_tevent_context_init_byname ( tctx , state . backend ) ;
2013-02-23 00:48:44 +04:00
if ( state . ev = = NULL ) {
torture_skip ( tctx , talloc_asprintf ( tctx ,
" event backend '%s' not supported \n " ,
state . backend ) ) ;
return true ;
}
torture_comment ( tctx , " backend '%s' - %s \n " ,
state . backend , __FUNCTION__ ) ;
/*
* This tests the following :
*
* It monitors the state of state . sock [ 0 ]
* with tevent_fd , but we never read / write on state . sock [ 0 ]
* while state . sock [ 1 ] * is only used to write a few bytes .
*
* We have a loop :
* - we wait only for TEVENT_FD_WRITE on state . sock [ 0 ]
* - we write 1 byte to state . sock [ 1 ]
* - we wait only for TEVENT_FD_READ on state . sock [ 0 ]
* - we disable events on state . sock [ 0 ]
* - the timer event restarts the loop
* Then we close state . sock [ 1 ]
* We have a loop :
* - we wait for TEVENT_FD_READ / WRITE on state . sock [ 0 ]
* - we try to read 1 byte
* - if the read gets an error of returns 0
* we disable the event handler
* - the timer finishes the test
*/
state . sock [ 0 ] = - 1 ;
state . sock [ 1 ] = - 1 ;
2018-04-10 21:58:11 +03:00
ret = socketpair ( AF_UNIX , SOCK_STREAM , 0 , state . sock ) ;
torture_assert ( tctx , ret = = 0 , " socketpair() failed " ) ;
2013-02-23 00:48:44 +04:00
state . te = tevent_add_timer ( state . ev , state . ev ,
2020-05-05 16:07:49 +03:00
timeval_current_ofs ( 0 , 10000 ) ,
2013-02-23 00:48:44 +04:00
test_event_fd1_finished , & state ) ;
state . fde0 = tevent_add_fd ( state . ev , state . ev ,
state . sock [ 0 ] , TEVENT_FD_WRITE ,
test_event_fd1_fde_handler , & state ) ;
/* state.fde1 is only used to auto close */
state . fde1 = tevent_add_fd ( state . ev , state . ev ,
state . sock [ 1 ] , 0 ,
test_event_fd1_fde_handler , & state ) ;
tevent_fd_set_auto_close ( state . fde0 ) ;
tevent_fd_set_auto_close ( state . fde1 ) ;
while ( ! state . finished ) {
errno = 0 ;
if ( tevent_loop_once ( state . ev ) = = - 1 ) {
talloc_free ( state . ev ) ;
torture_fail ( tctx , talloc_asprintf ( tctx ,
" Failed event loop %s \n " ,
strerror ( errno ) ) ) ;
}
}
talloc_free ( state . ev ) ;
torture_assert ( tctx , state . error = = NULL , talloc_asprintf ( tctx ,
" %s " , state . error ) ) ;
return true ;
}
2013-02-27 19:43:44 +04:00
struct test_event_fd2_state {
struct torture_context * tctx ;
const char * backend ;
struct tevent_context * ev ;
struct tevent_timer * te ;
struct test_event_fd2_sock {
struct test_event_fd2_state * state ;
int fd ;
struct tevent_fd * fde ;
size_t num_written ;
size_t num_read ;
bool got_full ;
} sock0 , sock1 ;
bool finished ;
const char * error ;
} ;
static void test_event_fd2_sock_handler ( struct tevent_context * ev_ctx ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data )
{
struct test_event_fd2_sock * cur_sock =
( struct test_event_fd2_sock * ) private_data ;
struct test_event_fd2_state * state = cur_sock - > state ;
struct test_event_fd2_sock * oth_sock = NULL ;
uint8_t v = 0 , c ;
ssize_t ret ;
if ( cur_sock = = & state - > sock0 ) {
oth_sock = & state - > sock1 ;
} else {
oth_sock = & state - > sock0 ;
}
if ( oth_sock - > num_written = = 1 ) {
if ( flags ! = ( TEVENT_FD_READ | TEVENT_FD_WRITE ) ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
}
if ( cur_sock - > num_read = = oth_sock - > num_written ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
if ( ! ( flags & TEVENT_FD_READ ) ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
2013-03-09 01:44:08 +04:00
if ( oth_sock - > num_read > = PIPE_BUF ) {
2013-02-27 19:43:44 +04:00
/*
2013-03-09 01:44:08 +04:00
* On Linux we become writable once we ' ve read
* one byte . On Solaris we only become writable
* again once we ' ve read 4096 bytes . PIPE_BUF
* is probably a safe bet to test against .
*
2013-02-27 19:43:44 +04:00
* There should be room to write a byte again
*/
if ( ! ( flags & TEVENT_FD_WRITE ) ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
}
if ( ( flags & TEVENT_FD_WRITE ) & & ! cur_sock - > got_full ) {
v = ( uint8_t ) cur_sock - > num_written ;
ret = write ( cur_sock - > fd , & v , 1 ) ;
if ( ret ! = 1 ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
cur_sock - > num_written + + ;
if ( cur_sock - > num_written > 0x80000000 ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
return ;
}
if ( ! cur_sock - > got_full ) {
cur_sock - > got_full = true ;
if ( ! oth_sock - > got_full ) {
/*
* cur_sock is full ,
* lets wait for oth_sock
* to be filled
*/
tevent_fd_set_flags ( cur_sock - > fde , 0 ) ;
return ;
}
/*
* oth_sock waited for cur_sock ,
* lets restart it
*/
tevent_fd_set_flags ( oth_sock - > fde ,
TEVENT_FD_READ | TEVENT_FD_WRITE ) ;
}
ret = read ( cur_sock - > fd , & v , 1 ) ;
if ( ret ! = 1 ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
c = ( uint8_t ) cur_sock - > num_read ;
if ( c ! = v ) {
state - > finished = true ;
state - > error = __location__ ;
return ;
}
cur_sock - > num_read + + ;
if ( cur_sock - > num_read < oth_sock - > num_written ) {
/* there is more to read */
return ;
}
/*
* we read everything , we need to remove TEVENT_FD_WRITE
* to avoid spinning
*/
TEVENT_FD_NOT_WRITEABLE ( cur_sock - > fde ) ;
if ( oth_sock - > num_read = = cur_sock - > num_written ) {
/*
* both directions are finished
*/
state - > finished = true ;
}
return ;
}
static void test_event_fd2_finished ( struct tevent_context * ev_ctx ,
struct tevent_timer * te ,
struct timeval tval ,
void * private_data )
{
struct test_event_fd2_state * state =
( struct test_event_fd2_state * ) private_data ;
/*
* this should never be triggered
*/
state - > finished = true ;
state - > error = __location__ ;
}
static bool test_event_fd2 ( struct torture_context * tctx ,
const void * test_data )
{
struct test_event_fd2_state state ;
int sock [ 2 ] ;
uint8_t c = 0 ;
ZERO_STRUCT ( state ) ;
state . tctx = tctx ;
state . backend = ( const char * ) test_data ;
2023-01-10 14:57:58 +03:00
state . ev = test_tevent_context_init_byname ( tctx , state . backend ) ;
2013-02-27 19:43:44 +04:00
if ( state . ev = = NULL ) {
torture_skip ( tctx , talloc_asprintf ( tctx ,
" event backend '%s' not supported \n " ,
state . backend ) ) ;
return true ;
}
torture_comment ( tctx , " backend '%s' - %s \n " ,
state . backend , __FUNCTION__ ) ;
/*
* This tests the following
*
* - We write 1 byte to each socket
* - We wait for TEVENT_FD_READ / WRITE on both sockets
* - When we get TEVENT_FD_WRITE we write 1 byte
* until both socket buffers are full , which
* means both sockets only get TEVENT_FD_READ .
* - Then we read 1 byte until we have consumed
* all bytes the other end has written .
*/
sock [ 0 ] = - 1 ;
sock [ 1 ] = - 1 ;
socketpair ( AF_UNIX , SOCK_STREAM , 0 , sock ) ;
/*
* the timer should never expire
*/
state . te = tevent_add_timer ( state . ev , state . ev ,
timeval_current_ofs ( 600 , 0 ) ,
test_event_fd2_finished , & state ) ;
state . sock0 . state = & state ;
state . sock0 . fd = sock [ 0 ] ;
state . sock0 . fde = tevent_add_fd ( state . ev , state . ev ,
state . sock0 . fd ,
TEVENT_FD_READ | TEVENT_FD_WRITE ,
test_event_fd2_sock_handler ,
& state . sock0 ) ;
state . sock1 . state = & state ;
state . sock1 . fd = sock [ 1 ] ;
state . sock1 . fde = tevent_add_fd ( state . ev , state . ev ,
state . sock1 . fd ,
TEVENT_FD_READ | TEVENT_FD_WRITE ,
test_event_fd2_sock_handler ,
& state . sock1 ) ;
tevent_fd_set_auto_close ( state . sock0 . fde ) ;
tevent_fd_set_auto_close ( state . sock1 . fde ) ;
2016-04-05 19:51:13 +03:00
do_write ( state . sock0 . fd , & c , 1 ) ;
2013-02-27 19:43:44 +04:00
state . sock0 . num_written + + ;
2016-04-05 19:51:13 +03:00
do_write ( state . sock1 . fd , & c , 1 ) ;
2013-02-27 19:43:44 +04:00
state . sock1 . num_written + + ;
while ( ! state . finished ) {
errno = 0 ;
if ( tevent_loop_once ( state . ev ) = = - 1 ) {
talloc_free ( state . ev ) ;
torture_fail ( tctx , talloc_asprintf ( tctx ,
" Failed event loop %s \n " ,
strerror ( errno ) ) ) ;
}
}
talloc_free ( state . ev ) ;
torture_assert ( tctx , state . error = = NULL , talloc_asprintf ( tctx ,
" %s " , state . error ) ) ;
return true ;
}
2018-06-16 15:12:01 +03:00
struct test_wrapper_state {
struct torture_context * tctx ;
int num_events ;
int num_wrap_handlers ;
} ;
static bool test_wrapper_before_use ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
return true ;
}
static void test_wrapper_after_use ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_before_fd_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_fd * fde ,
uint16_t flags ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_after_fd_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_fd * fde ,
uint16_t flags ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_before_timer_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_timer * te ,
struct timeval requested_time ,
struct timeval trigger_time ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_after_timer_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_timer * te ,
struct timeval requested_time ,
struct timeval trigger_time ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_before_immediate_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_immediate * im ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_after_immediate_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_immediate * im ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_before_signal_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static void test_wrapper_after_signal_handler ( struct tevent_context * wrap_ev ,
void * private_data ,
struct tevent_context * main_ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
const char * handler_name ,
const char * location )
{
struct test_wrapper_state * state =
talloc_get_type_abort ( private_data ,
struct test_wrapper_state ) ;
torture_comment ( state - > tctx , " %s \n " , __func__ ) ;
state - > num_wrap_handlers + + ;
}
static const struct tevent_wrapper_ops test_wrapper_ops = {
. name = " test_wrapper " ,
. before_use = test_wrapper_before_use ,
. after_use = test_wrapper_after_use ,
. before_fd_handler = test_wrapper_before_fd_handler ,
. after_fd_handler = test_wrapper_after_fd_handler ,
. before_timer_handler = test_wrapper_before_timer_handler ,
. after_timer_handler = test_wrapper_after_timer_handler ,
. before_immediate_handler = test_wrapper_before_immediate_handler ,
. after_immediate_handler = test_wrapper_after_immediate_handler ,
. before_signal_handler = test_wrapper_before_signal_handler ,
. after_signal_handler = test_wrapper_after_signal_handler ,
} ;
static void test_wrapper_timer_handler ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval tv ,
void * private_data )
{
struct test_wrapper_state * state =
( struct test_wrapper_state * ) private_data ;
torture_comment ( state - > tctx , " timer handler \n " ) ;
state - > num_events + + ;
talloc_free ( te ) ;
return ;
}
static void test_wrapper_fd_handler ( struct tevent_context * ev ,
struct tevent_fd * fde ,
unsigned short fd_flags ,
void * private_data )
{
struct test_wrapper_state * state =
( struct test_wrapper_state * ) private_data ;
torture_comment ( state - > tctx , " fd handler \n " ) ;
state - > num_events + + ;
talloc_free ( fde ) ;
return ;
}
static void test_wrapper_immediate_handler ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_data )
{
struct test_wrapper_state * state =
( struct test_wrapper_state * ) private_data ;
state - > num_events + + ;
talloc_free ( im ) ;
torture_comment ( state - > tctx , " immediate handler \n " ) ;
return ;
}
static void test_wrapper_signal_handler ( struct tevent_context * ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
void * private_data )
{
struct test_wrapper_state * state =
( struct test_wrapper_state * ) private_data ;
torture_comment ( state - > tctx , " signal handler \n " ) ;
state - > num_events + + ;
talloc_free ( se ) ;
return ;
}
static bool test_wrapper ( struct torture_context * tctx ,
const void * test_data )
{
struct test_wrapper_state * state = NULL ;
int sock [ 2 ] = { - 1 , - 1 } ;
uint8_t c = 0 ;
const int num_events = 4 ;
const char * backend = ( const char * ) test_data ;
struct tevent_context * ev = NULL ;
struct tevent_context * wrap_ev = NULL ;
struct tevent_fd * fde = NULL ;
struct tevent_timer * te = NULL ;
struct tevent_signal * se = NULL ;
struct tevent_immediate * im = NULL ;
int ret ;
bool ok = false ;
bool ret2 ;
2023-01-10 14:57:58 +03:00
ev = test_tevent_context_init_byname ( tctx , backend ) ;
2018-06-16 15:12:01 +03:00
if ( ev = = NULL ) {
torture_skip ( tctx , talloc_asprintf ( tctx ,
" event backend '%s' not supported \n " ,
backend ) ) ;
return true ;
}
torture_comment ( tctx , " tevent backend '%s' \n " , backend ) ;
wrap_ev = tevent_context_wrapper_create (
ev , ev , & test_wrapper_ops , & state , struct test_wrapper_state ) ;
torture_assert_not_null_goto ( tctx , wrap_ev , ok , done ,
" tevent_context_wrapper_create failed \n " ) ;
* state = ( struct test_wrapper_state ) {
. tctx = tctx ,
} ;
ret = socketpair ( AF_UNIX , SOCK_STREAM , 0 , sock ) ;
torture_assert_goto ( tctx , ret = = 0 , ok , done , " socketpair failed \n " ) ;
te = tevent_add_timer ( wrap_ev , wrap_ev ,
timeval_current_ofs ( 0 , 0 ) ,
test_wrapper_timer_handler , state ) ;
torture_assert_not_null_goto ( tctx , te , ok , done ,
" tevent_add_timer failed \n " ) ;
fde = tevent_add_fd ( wrap_ev , wrap_ev ,
sock [ 1 ] ,
TEVENT_FD_READ ,
test_wrapper_fd_handler ,
state ) ;
torture_assert_not_null_goto ( tctx , fde , ok , done ,
" tevent_add_fd failed \n " ) ;
im = tevent_create_immediate ( wrap_ev ) ;
torture_assert_not_null_goto ( tctx , im , ok , done ,
" tevent_create_immediate failed \n " ) ;
se = tevent_add_signal ( wrap_ev , wrap_ev ,
SIGUSR1 ,
0 ,
test_wrapper_signal_handler ,
state ) ;
torture_assert_not_null_goto ( tctx , se , ok , done ,
" tevent_add_signal failed \n " ) ;
do_write ( sock [ 0 ] , & c , 1 ) ;
kill ( getpid ( ) , SIGUSR1 ) ;
tevent_schedule_immediate ( im ,
wrap_ev ,
test_wrapper_immediate_handler ,
state ) ;
ret2 = tevent_context_push_use ( wrap_ev ) ;
torture_assert_goto ( tctx , ret2 , ok , done , " tevent_context_push_use(wrap_ev) failed \n " ) ;
ret2 = tevent_context_push_use ( ev ) ;
torture_assert_goto ( tctx , ret2 , ok , pop_use , " tevent_context_push_use(ev) failed \n " ) ;
tevent_context_pop_use ( ev ) ;
tevent_context_pop_use ( wrap_ev ) ;
ret = tevent_loop_wait ( ev ) ;
torture_assert_int_equal_goto ( tctx , ret , 0 , ok , done , " tevent_loop_wait failed \n " ) ;
torture_comment ( tctx , " Num events: %d \n " , state - > num_events ) ;
torture_comment ( tctx , " Num wrap handlers: %d \n " ,
state - > num_wrap_handlers ) ;
torture_assert_int_equal_goto ( tctx , state - > num_events , num_events , ok , done ,
" Wrong event count \n " ) ;
torture_assert_int_equal_goto ( tctx , state - > num_wrap_handlers ,
num_events * 2 + 2 ,
ok , done , " Wrong wrapper count \n " ) ;
ok = true ;
done :
TALLOC_FREE ( wrap_ev ) ;
TALLOC_FREE ( ev ) ;
if ( sock [ 0 ] ! = - 1 ) {
close ( sock [ 0 ] ) ;
}
if ( sock [ 1 ] ! = - 1 ) {
close ( sock [ 1 ] ) ;
}
return ok ;
pop_use :
tevent_context_pop_use ( wrap_ev ) ;
goto done ;
}
2018-06-16 17:55:44 +03:00
static void test_free_wrapper_signal_handler ( struct tevent_context * ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
void * private_data )
{
struct torture_context * tctx =
talloc_get_type_abort ( private_data ,
struct torture_context ) ;
torture_comment ( tctx , " signal handler \n " ) ;
talloc_free ( se ) ;
/*
* signal handlers have highest priority in tevent , so this signal
* handler will always be started before the other handlers
* below . Freeing the ( wrapper ) event context here tests that the
2023-04-13 14:18:05 +03:00
* wrapper implementation correctly handles the wrapper ev going away
2018-06-16 17:55:44 +03:00
* with pending events .
*/
talloc_free ( ev ) ;
return ;
}
static void test_free_wrapper_fd_handler ( struct tevent_context * ev ,
struct tevent_fd * fde ,
unsigned short fd_flags ,
void * private_data )
{
/*
* This should never be called as
* test_free_wrapper_signal_handler ( )
* already destroyed the wrapper tevent_context .
*/
abort ( ) ;
}
static void test_free_wrapper_immediate_handler ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_data )
{
/*
* This should never be called as
* test_free_wrapper_signal_handler ( )
* already destroyed the wrapper tevent_context .
*/
abort ( ) ;
}
static void test_free_wrapper_timer_handler ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval tv ,
void * private_data )
{
/*
* This should never be called as
* test_free_wrapper_signal_handler ( )
* already destroyed the wrapper tevent_context .
*/
abort ( ) ;
}
static bool test_free_wrapper ( struct torture_context * tctx ,
const void * test_data )
{
struct test_wrapper_state * state = NULL ;
int sock [ 2 ] = { - 1 , - 1 } ;
uint8_t c = 0 ;
const char * backend = ( const char * ) test_data ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct tevent_context * ev = NULL ;
struct tevent_context * wrap_ev = NULL ;
struct tevent_fd * fde = NULL ;
struct tevent_timer * te = NULL ;
struct tevent_signal * se = NULL ;
struct tevent_immediate * im = NULL ;
int ret ;
bool ok = false ;
2023-01-10 14:57:58 +03:00
ev = test_tevent_context_init_byname ( frame , backend ) ;
2018-06-16 17:55:44 +03:00
if ( ev = = NULL ) {
torture_skip ( tctx , talloc_asprintf ( tctx ,
" event backend '%s' not supported \n " ,
backend ) ) ;
return true ;
}
torture_comment ( tctx , " tevent backend '%s' \n " , backend ) ;
wrap_ev = tevent_context_wrapper_create (
ev , ev , & test_wrapper_ops , & state , struct test_wrapper_state ) ;
torture_assert_not_null_goto ( tctx , wrap_ev , ok , done ,
" tevent_context_wrapper_create failed \n " ) ;
* state = ( struct test_wrapper_state ) {
. tctx = tctx ,
} ;
ret = socketpair ( AF_UNIX , SOCK_STREAM , 0 , sock ) ;
torture_assert_goto ( tctx , ret = = 0 , ok , done , " socketpair failed \n " ) ;
fde = tevent_add_fd ( wrap_ev , frame ,
sock [ 1 ] ,
TEVENT_FD_READ ,
test_free_wrapper_fd_handler ,
NULL ) ;
torture_assert_not_null_goto ( tctx , fde , ok , done ,
" tevent_add_fd failed \n " ) ;
te = tevent_add_timer ( wrap_ev , frame ,
timeval_current_ofs ( 0 , 0 ) ,
test_free_wrapper_timer_handler , NULL ) ;
torture_assert_not_null_goto ( tctx , te , ok , done ,
" tevent_add_timer failed \n " ) ;
im = tevent_create_immediate ( frame ) ;
torture_assert_not_null_goto ( tctx , im , ok , done ,
" tevent_create_immediate failed \n " ) ;
se = tevent_add_signal ( wrap_ev , frame ,
SIGUSR1 ,
0 ,
test_free_wrapper_signal_handler ,
tctx ) ;
torture_assert_not_null_goto ( tctx , se , ok , done ,
" tevent_add_signal failed \n " ) ;
do_write ( sock [ 0 ] , & c , 1 ) ;
kill ( getpid ( ) , SIGUSR1 ) ;
tevent_schedule_immediate ( im ,
wrap_ev ,
test_free_wrapper_immediate_handler ,
NULL ) ;
ret = tevent_loop_wait ( ev ) ;
torture_assert_goto ( tctx , ret = = 0 , ok , done , " tevent_loop_wait failed \n " ) ;
ok = true ;
done :
TALLOC_FREE ( frame ) ;
if ( sock [ 0 ] ! = - 1 ) {
close ( sock [ 0 ] ) ;
}
if ( sock [ 1 ] ! = - 1 ) {
close ( sock [ 1 ] ) ;
}
return ok ;
}
2012-07-30 11:09:46 +04:00
# ifdef HAVE_PTHREAD
static pthread_mutex_t threaded_mutex = PTHREAD_MUTEX_INITIALIZER ;
static bool do_shutdown = false ;
static void test_event_threaded_lock ( void )
{
int ret ;
ret = pthread_mutex_lock ( & threaded_mutex ) ;
assert ( ret = = 0 ) ;
}
static void test_event_threaded_unlock ( void )
{
int ret ;
ret = pthread_mutex_unlock ( & threaded_mutex ) ;
assert ( ret = = 0 ) ;
}
static void test_event_threaded_trace ( enum tevent_trace_point point ,
void * private_data )
{
switch ( point ) {
case TEVENT_TRACE_BEFORE_WAIT :
test_event_threaded_unlock ( ) ;
break ;
case TEVENT_TRACE_AFTER_WAIT :
test_event_threaded_lock ( ) ;
break ;
2013-02-26 18:54:57 +04:00
case TEVENT_TRACE_BEFORE_LOOP_ONCE :
case TEVENT_TRACE_AFTER_LOOP_ONCE :
break ;
2012-07-30 11:09:46 +04:00
}
}
static void test_event_threaded_timer ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval current_time ,
void * private_data )
{
return ;
}
static void * test_event_poll_thread ( void * private_data )
{
struct tevent_context * ev = ( struct tevent_context * ) private_data ;
test_event_threaded_lock ( ) ;
while ( true ) {
int ret ;
ret = tevent_loop_once ( ev ) ;
assert ( ret = = 0 ) ;
if ( do_shutdown ) {
test_event_threaded_unlock ( ) ;
return NULL ;
}
}
}
static void test_event_threaded_read_handler ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data )
{
int * pfd = ( int * ) private_data ;
char c ;
ssize_t nread ;
if ( ( flags & TEVENT_FD_READ ) = = 0 ) {
return ;
}
do {
nread = read ( * pfd , & c , 1 ) ;
} while ( ( nread = = - 1 ) & & ( errno = = EINTR ) ) ;
assert ( nread = = 1 ) ;
}
static bool test_event_context_threaded ( struct torture_context * test ,
const void * test_data )
{
struct tevent_context * ev ;
struct tevent_timer * te ;
struct tevent_fd * fde ;
pthread_t poll_thread ;
int fds [ 2 ] ;
int ret ;
char c = 0 ;
2023-01-10 14:57:58 +03:00
ev = test_tevent_context_init_byname ( test , " poll_mt " ) ;
2012-07-30 11:09:46 +04:00
torture_assert ( test , ev ! = NULL , " poll_mt not supported " ) ;
tevent_set_trace_callback ( ev , test_event_threaded_trace , NULL ) ;
te = tevent_add_timer ( ev , ev , timeval_current_ofs ( 5 , 0 ) ,
test_event_threaded_timer , NULL ) ;
torture_assert ( test , te ! = NULL , " Could not add timer " ) ;
ret = pthread_create ( & poll_thread , NULL , test_event_poll_thread , ev ) ;
torture_assert ( test , ret = = 0 , " Could not create poll thread " ) ;
ret = pipe ( fds ) ;
torture_assert ( test , ret = = 0 , " Could not create pipe " ) ;
poll ( NULL , 0 , 100 ) ;
test_event_threaded_lock ( ) ;
fde = tevent_add_fd ( ev , ev , fds [ 0 ] , TEVENT_FD_READ ,
test_event_threaded_read_handler , & fds [ 0 ] ) ;
torture_assert ( test , fde ! = NULL , " Could not add fd event " ) ;
test_event_threaded_unlock ( ) ;
poll ( NULL , 0 , 100 ) ;
2016-04-05 19:51:13 +03:00
do_write ( fds [ 1 ] , & c , 1 ) ;
2012-07-30 11:09:46 +04:00
poll ( NULL , 0 , 100 ) ;
test_event_threaded_lock ( ) ;
do_shutdown = true ;
test_event_threaded_unlock ( ) ;
2016-04-05 19:51:13 +03:00
do_write ( fds [ 1 ] , & c , 1 ) ;
2012-07-30 11:09:46 +04:00
ret = pthread_join ( poll_thread , NULL ) ;
torture_assert ( test , ret = = 0 , " pthread_join failed " ) ;
return true ;
}
2015-07-24 18:50:31 +03:00
# define NUM_TEVENT_THREADS 100
/* Ugly, but needed for torture_comment... */
static struct torture_context * thread_test_ctx ;
static pthread_t thread_map [ NUM_TEVENT_THREADS ] ;
static unsigned thread_counter ;
/* Called in master thread context */
static void callback_nowait ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_ptr )
{
pthread_t * thread_id_ptr =
talloc_get_type_abort ( private_ptr , pthread_t ) ;
unsigned i ;
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
if ( pthread_equal ( * thread_id_ptr ,
thread_map [ i ] ) ) {
break ;
}
}
torture_comment ( thread_test_ctx ,
" Callback %u from thread %u \n " ,
thread_counter ,
i ) ;
thread_counter + + ;
}
/* Blast the master tevent_context with a callback, no waiting. */
static void * thread_fn_nowait ( void * private_ptr )
{
struct tevent_thread_proxy * master_tp =
talloc_get_type_abort ( private_ptr , struct tevent_thread_proxy ) ;
struct tevent_immediate * im ;
pthread_t * thread_id_ptr ;
im = tevent_create_immediate ( NULL ) ;
if ( im = = NULL ) {
return NULL ;
}
thread_id_ptr = talloc ( NULL , pthread_t ) ;
if ( thread_id_ptr = = NULL ) {
return NULL ;
}
* thread_id_ptr = pthread_self ( ) ;
tevent_thread_proxy_schedule ( master_tp ,
& im ,
callback_nowait ,
& thread_id_ptr ) ;
return NULL ;
}
static void timeout_fn ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval tv , void * p )
{
thread_counter = NUM_TEVENT_THREADS * 10 ;
}
static bool test_multi_tevent_threaded ( struct torture_context * test ,
const void * test_data )
{
unsigned i ;
struct tevent_context * master_ev ;
struct tevent_thread_proxy * tp ;
talloc_disable_null_tracking ( ) ;
/* Ugly global stuff. */
thread_test_ctx = test ;
thread_counter = 0 ;
2023-01-10 14:57:58 +03:00
master_ev = test_tevent_context_init ( NULL ) ;
2015-07-24 18:50:31 +03:00
if ( master_ev = = NULL ) {
return false ;
}
tp = tevent_thread_proxy_create ( master_ev ) ;
if ( tp = = NULL ) {
torture_fail ( test ,
talloc_asprintf ( test ,
" tevent_thread_proxy_create failed \n " ) ) ;
talloc_free ( master_ev ) ;
return false ;
}
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
int ret = pthread_create ( & thread_map [ i ] ,
NULL ,
thread_fn_nowait ,
tp ) ;
if ( ret ! = 0 ) {
torture_fail ( test ,
talloc_asprintf ( test ,
" Failed to create thread %i, %d \n " ,
i , ret ) ) ;
return false ;
}
}
/* Ensure we don't wait more than 10 seconds. */
tevent_add_timer ( master_ev ,
master_ev ,
timeval_current_ofs ( 10 , 0 ) ,
timeout_fn ,
NULL ) ;
while ( thread_counter < NUM_TEVENT_THREADS ) {
int ret = tevent_loop_once ( master_ev ) ;
torture_assert ( test , ret = = 0 , " tevent_loop_once failed " ) ;
}
torture_assert ( test , thread_counter = = NUM_TEVENT_THREADS ,
" thread_counter fail \n " ) ;
talloc_free ( master_ev ) ;
return true ;
}
2015-07-24 19:27:21 +03:00
struct reply_state {
struct tevent_thread_proxy * reply_tp ;
pthread_t thread_id ;
int * p_finished ;
} ;
static void thread_timeout_fn ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval tv , void * p )
{
int * p_finished = ( int * ) p ;
* p_finished = 2 ;
}
/* Called in child-thread context */
static void thread_callback ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_ptr )
{
struct reply_state * rsp =
talloc_get_type_abort ( private_ptr , struct reply_state ) ;
talloc_steal ( ev , rsp ) ;
* rsp - > p_finished = 1 ;
}
/* Called in master thread context */
static void master_callback ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_ptr )
{
struct reply_state * rsp =
talloc_get_type_abort ( private_ptr , struct reply_state ) ;
unsigned i ;
talloc_steal ( ev , rsp ) ;
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
if ( pthread_equal ( rsp - > thread_id ,
thread_map [ i ] ) ) {
break ;
}
}
torture_comment ( thread_test_ctx ,
" Callback %u from thread %u \n " ,
thread_counter ,
i ) ;
/* Now reply to the thread ! */
tevent_thread_proxy_schedule ( rsp - > reply_tp ,
& im ,
thread_callback ,
& rsp ) ;
thread_counter + + ;
}
static void * thread_fn_1 ( void * private_ptr )
{
struct tevent_thread_proxy * master_tp =
talloc_get_type_abort ( private_ptr , struct tevent_thread_proxy ) ;
struct tevent_thread_proxy * tp ;
struct tevent_immediate * im ;
struct tevent_context * ev ;
struct reply_state * rsp ;
int finished = 0 ;
int ret ;
ev = tevent_context_init ( NULL ) ;
if ( ev = = NULL ) {
return NULL ;
}
tp = tevent_thread_proxy_create ( ev ) ;
if ( tp = = NULL ) {
talloc_free ( ev ) ;
return NULL ;
}
im = tevent_create_immediate ( ev ) ;
if ( im = = NULL ) {
talloc_free ( ev ) ;
return NULL ;
}
rsp = talloc ( ev , struct reply_state ) ;
if ( rsp = = NULL ) {
talloc_free ( ev ) ;
return NULL ;
}
rsp - > thread_id = pthread_self ( ) ;
rsp - > reply_tp = tp ;
rsp - > p_finished = & finished ;
/* Introduce a little randomness into the mix.. */
usleep ( random ( ) % 7000 ) ;
tevent_thread_proxy_schedule ( master_tp ,
& im ,
master_callback ,
& rsp ) ;
/* Ensure we don't wait more than 10 seconds. */
tevent_add_timer ( ev ,
ev ,
timeval_current_ofs ( 10 , 0 ) ,
thread_timeout_fn ,
& finished ) ;
while ( finished = = 0 ) {
ret = tevent_loop_once ( ev ) ;
assert ( ret = = 0 ) ;
}
if ( finished > 1 ) {
/* Timeout ! */
abort ( ) ;
}
/*
* NB . We should talloc_free ( ev ) here , but if we do
* we currently get hit by helgrind Fix # 323432
* " When calling pthread_cond_destroy or pthread_mutex_destroy
* with initializers as argument Helgrind ( incorrectly ) reports errors . "
*
* http : //valgrind.10908.n7.nabble.com/Helgrind-3-9-0-false-positive-
* with - pthread - mutex - destroy - td47757 . html
*
* Helgrind doesn ' t understand that the request / reply
* messages provide synchronization between the lock / unlock
* in tevent_thread_proxy_schedule ( ) , and the pthread_destroy ( )
* when the struct tevent_thread_proxy object is talloc_free ' d .
*
* As a work - around for now return ev for the parent thread to free .
*/
return ev ;
}
static bool test_multi_tevent_threaded_1 ( struct torture_context * test ,
const void * test_data )
{
unsigned i ;
struct tevent_context * master_ev ;
struct tevent_thread_proxy * master_tp ;
int ret ;
talloc_disable_null_tracking ( ) ;
/* Ugly global stuff. */
thread_test_ctx = test ;
thread_counter = 0 ;
2023-01-10 14:57:58 +03:00
master_ev = test_tevent_context_init ( NULL ) ;
2015-07-24 19:27:21 +03:00
if ( master_ev = = NULL ) {
return false ;
}
master_tp = tevent_thread_proxy_create ( master_ev ) ;
if ( master_tp = = NULL ) {
torture_fail ( test ,
talloc_asprintf ( test ,
" tevent_thread_proxy_create failed \n " ) ) ;
talloc_free ( master_ev ) ;
return false ;
}
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
ret = pthread_create ( & thread_map [ i ] ,
NULL ,
thread_fn_1 ,
master_tp ) ;
if ( ret ! = 0 ) {
torture_fail ( test ,
talloc_asprintf ( test ,
" Failed to create thread %i, %d \n " ,
i , ret ) ) ;
return false ;
}
}
while ( thread_counter < NUM_TEVENT_THREADS ) {
ret = tevent_loop_once ( master_ev ) ;
torture_assert ( test , ret = = 0 , " tevent_loop_once failed " ) ;
}
/* Wait for all the threads to finish - join 'em. */
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
void * retval ;
ret = pthread_join ( thread_map [ i ] , & retval ) ;
torture_assert ( test , ret = = 0 , " pthread_join failed " ) ;
/* Free the child thread event context. */
talloc_free ( retval ) ;
}
talloc_free ( master_ev ) ;
return true ;
}
2016-08-08 13:53:08 +03:00
struct threaded_test_2 {
struct tevent_threaded_context * tctx ;
struct tevent_immediate * im ;
pthread_t thread_id ;
} ;
static void master_callback_2 ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_data ) ;
static void * thread_fn_2 ( void * private_data )
{
struct threaded_test_2 * state = private_data ;
state - > thread_id = pthread_self ( ) ;
usleep ( random ( ) % 7000 ) ;
tevent_threaded_schedule_immediate (
state - > tctx , state - > im , master_callback_2 , state ) ;
return NULL ;
}
static void master_callback_2 ( struct tevent_context * ev ,
struct tevent_immediate * im ,
void * private_data )
{
struct threaded_test_2 * state = private_data ;
int i ;
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
if ( pthread_equal ( state - > thread_id , thread_map [ i ] ) ) {
break ;
}
}
torture_comment ( thread_test_ctx ,
" Callback_2 %u from thread %u \n " ,
thread_counter ,
i ) ;
thread_counter + + ;
}
static bool test_multi_tevent_threaded_2 ( struct torture_context * test ,
const void * test_data )
{
unsigned i ;
struct tevent_context * ev ;
struct tevent_threaded_context * tctx ;
int ret ;
thread_test_ctx = test ;
thread_counter = 0 ;
2023-01-10 14:57:58 +03:00
ev = test_tevent_context_init ( test ) ;
2016-08-08 13:53:08 +03:00
torture_assert ( test , ev ! = NULL , " tevent_context_init failed " ) ;
2017-06-05 08:29:11 +03:00
/*
* tevent_re_initialise used to have a bug where it did not
* re - initialise the thread support after taking it
2018-11-28 01:10:17 +03:00
* down . Exercise that code path .
2017-06-05 08:29:11 +03:00
*/
ret = tevent_re_initialise ( ev ) ;
torture_assert ( test , ret = = 0 , " tevent_re_initialise failed " ) ;
2016-08-08 13:53:08 +03:00
tctx = tevent_threaded_context_create ( ev , ev ) ;
torture_assert ( test , tctx ! = NULL ,
" tevent_threaded_context_create failed " ) ;
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
struct threaded_test_2 * state ;
state = talloc ( ev , struct threaded_test_2 ) ;
torture_assert ( test , state ! = NULL , " talloc failed " ) ;
state - > tctx = tctx ;
state - > im = tevent_create_immediate ( state ) ;
torture_assert ( test , state - > im ! = NULL ,
" tevent_create_immediate failed " ) ;
ret = pthread_create ( & thread_map [ i ] , NULL , thread_fn_2 , state ) ;
torture_assert ( test , ret = = 0 , " pthread_create failed " ) ;
}
while ( thread_counter < NUM_TEVENT_THREADS ) {
ret = tevent_loop_once ( ev ) ;
torture_assert ( test , ret = = 0 , " tevent_loop_once failed " ) ;
}
/* Wait for all the threads to finish - join 'em. */
for ( i = 0 ; i < NUM_TEVENT_THREADS ; i + + ) {
void * retval ;
ret = pthread_join ( thread_map [ i ] , & retval ) ;
torture_assert ( test , ret = = 0 , " pthread_join failed " ) ;
/* Free the child thread event context. */
}
talloc_free ( tctx ) ;
talloc_free ( ev ) ;
return true ;
}
2022-07-25 15:13:34 +03:00
struct test_cached_pid_thread_state {
pid_t thread_cached_pid ;
pid_t thread_pid ;
} ;
static void * test_cached_pid_thread ( void * private_data )
{
struct test_cached_pid_thread_state * state =
( struct test_cached_pid_thread_state * ) private_data ;
state - > thread_cached_pid = tevent_cached_getpid ( ) ;
state - > thread_pid = getpid ( ) ;
return NULL ;
}
2012-07-30 11:09:46 +04:00
# endif
2022-07-25 15:13:34 +03:00
static bool test_cached_pid ( struct torture_context * test ,
const void * test_data )
{
pid_t parent_pid = getpid ( ) ;
pid_t child_pid ;
pid_t finished_pid ;
int child_status ;
torture_assert ( test , tevent_cached_getpid ( ) = = parent_pid , " tevent_cached_getpid() " ) ;
# ifdef HAVE_PTHREAD
{
struct test_cached_pid_thread_state state = { . thread_cached_pid = - 1 , } ;
pthread_t thread ;
void * retval = NULL ;
int ret ;
ret = pthread_create ( & thread , NULL , test_cached_pid_thread , & state ) ;
torture_assert ( test , ret = = 0 , " pthread_create failed " ) ;
ret = pthread_join ( thread , & retval ) ;
torture_assert ( test , ret = = 0 , " pthread_join failed " ) ;
torture_assert ( test , state . thread_pid = = parent_pid , " getpid() in thread " ) ;
torture_assert ( test , state . thread_cached_pid = = parent_pid , " tevent_cached_getpid() in thread " ) ;
}
# endif /* HAVE_PTHREAD */
child_pid = fork ( ) ;
if ( child_pid = = 0 ) {
/* child */
pid_t pid = getpid ( ) ;
pid_t cached_pid = tevent_cached_getpid ( ) ;
if ( parent_pid = = pid ) {
exit ( 1 ) ;
}
if ( pid ! = cached_pid ) {
exit ( 2 ) ;
}
exit ( 0 ) ;
}
torture_assert ( test , child_pid > 0 , " fork failed " ) ;
finished_pid = waitpid ( child_pid , & child_status , 0 ) ;
torture_assert ( test , finished_pid = = child_pid , " wrong child " ) ;
torture_assert ( test , child_status = = 0 , " child_status " ) ;
return true ;
}
2006-06-17 04:17:50 +04:00
struct torture_suite * torture_local_event ( TALLOC_CTX * mem_ctx )
2006-05-24 18:19:34 +04:00
{
2010-12-11 05:26:31 +03:00
struct torture_suite * suite = torture_suite_create ( mem_ctx , " event " ) ;
2012-05-14 13:48:00 +04:00
const char * * list = tevent_backend_list ( suite ) ;
2007-01-05 12:35:49 +03:00
int i ;
2006-05-24 18:19:34 +04:00
2007-01-05 12:35:49 +03:00
for ( i = 0 ; list & & list [ i ] ; i + + ) {
2013-02-28 13:12:42 +04:00
struct torture_suite * backend_suite ;
backend_suite = torture_suite_create ( mem_ctx , list [ i ] ) ;
torture_suite_add_simple_tcase_const ( backend_suite ,
" context " ,
2007-01-05 12:35:49 +03:00
test_event_context ,
( const void * ) list [ i ] ) ;
2023-01-27 14:12:45 +03:00
torture_suite_add_simple_tcase_const ( backend_suite ,
" fd_speed1 " ,
test_fd_speed1 ,
( const void * ) list [ i ] ) ;
torture_suite_add_simple_tcase_const ( backend_suite ,
" fd_speed2 " ,
test_fd_speed2 ,
( const void * ) list [ i ] ) ;
2013-02-23 00:48:44 +04:00
torture_suite_add_simple_tcase_const ( backend_suite ,
" fd1 " ,
test_event_fd1 ,
( const void * ) list [ i ] ) ;
2013-02-27 19:43:44 +04:00
torture_suite_add_simple_tcase_const ( backend_suite ,
" fd2 " ,
test_event_fd2 ,
( const void * ) list [ i ] ) ;
2018-06-16 15:12:01 +03:00
torture_suite_add_simple_tcase_const ( backend_suite ,
" wrapper " ,
test_wrapper ,
( const void * ) list [ i ] ) ;
2018-06-16 17:55:44 +03:00
torture_suite_add_simple_tcase_const ( backend_suite ,
" free_wrapper " ,
test_free_wrapper ,
( const void * ) list [ i ] ) ;
2013-02-28 13:12:42 +04:00
torture_suite_add_suite ( suite , backend_suite ) ;
2007-01-05 12:35:49 +03:00
}
2006-05-24 18:19:34 +04:00
2012-07-30 11:09:46 +04:00
# ifdef HAVE_PTHREAD
2013-02-28 13:12:42 +04:00
torture_suite_add_simple_tcase_const ( suite , " threaded_poll_mt " ,
2012-07-30 11:09:46 +04:00
test_event_context_threaded ,
NULL ) ;
2015-07-24 18:50:31 +03:00
torture_suite_add_simple_tcase_const ( suite , " multi_tevent_threaded " ,
test_multi_tevent_threaded ,
NULL ) ;
2015-07-24 19:27:21 +03:00
torture_suite_add_simple_tcase_const ( suite , " multi_tevent_threaded_1 " ,
test_multi_tevent_threaded_1 ,
NULL ) ;
2016-08-08 13:53:08 +03:00
torture_suite_add_simple_tcase_const ( suite , " multi_tevent_threaded_2 " ,
test_multi_tevent_threaded_2 ,
NULL ) ;
2012-07-30 11:09:46 +04:00
# endif
2022-07-25 15:13:34 +03:00
torture_suite_add_simple_tcase_const ( suite , " tevent_cached_getpid " ,
test_cached_pid ,
NULL ) ;
2006-06-17 04:17:50 +04:00
return suite ;
2006-05-24 18:19:34 +04:00
}