2005-02-15 13:36:59 +03:00
/*
Unix SMB / CIFS implementation .
main select loop and event handling
wrapper for http : //liboop.org/
Copyright ( C ) Stefan Metzmacher 2005
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 ,
2005-02-15 13:36:59 +03: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/>.
2005-02-15 13:36:59 +03:00
*/
2008-06-14 19:23:31 +04:00
# include "events.h"
# include "events_internal.h"
2005-02-15 13:36:59 +03:00
# include <oop.h>
2005-12-08 11:31:59 +03:00
/*
! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! !
2010-02-21 08:31:01 +03:00
NOTE : this code compiles fine , but is completely * UNTESTED *
and is only committed as an example
2005-12-08 11:31:59 +03:00
! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! !
*/
2008-12-29 22:24:57 +03:00
static int oop_event_context_destructor ( struct tevent_context * ev )
2005-02-15 13:36:59 +03:00
{
oop_source_sys * oop_sys = ev - > additional_data ;
oop_sys_delete ( oop_sys ) ;
return 0 ;
}
/*
create a oop_event_context structure .
*/
2008-12-29 22:24:57 +03:00
static int oop_event_context_init ( struct tevent_context * ev , void * private_data )
2005-02-15 13:36:59 +03:00
{
oop_source_sys * oop_sys = private_data ;
if ( ! oop_sys ) {
oop_sys = oop_sys_new ( ) ;
if ( ! oop_sys ) {
return - 1 ;
}
talloc_set_destructor ( ev , oop_event_context_destructor ) ;
}
ev - > additional_data = oop_sys ;
return 0 ;
}
static void * oop_event_fd_handler ( oop_source * oop , int fd , oop_event oop_type , void * ptr )
{
2008-12-29 22:24:57 +03:00
struct tevent_fd * fde = ptr ;
2005-02-15 13:36:59 +03:00
if ( fd ! = fde - > fd ) return OOP_ERROR ;
switch ( oop_type ) {
case OOP_READ :
fde - > handler ( fde - > event_ctx , fde , EVENT_FD_READ , fde - > private_data ) ;
return OOP_CONTINUE ;
case OOP_WRITE :
fde - > handler ( fde - > event_ctx , fde , EVENT_FD_WRITE , fde - > private_data ) ;
return OOP_CONTINUE ;
case OOP_EXCEPTION :
return OOP_ERROR ;
case OOP_NUM_EVENTS :
return OOP_ERROR ;
}
return OOP_ERROR ;
}
/*
destroy an fd_event
*/
2008-12-29 22:24:57 +03:00
static int oop_event_fd_destructor ( struct tevent_fd * fde )
2005-02-15 13:36:59 +03:00
{
2008-12-29 22:24:57 +03:00
struct tevent_context * ev = fde - > event_ctx ;
2005-02-15 13:36:59 +03:00
oop_source_sys * oop_sys = ev - > additional_data ;
oop_source * oop = oop_sys_source ( oop_sys ) ;
if ( fde - > flags & EVENT_FD_READ )
oop - > cancel_fd ( oop , fde - > fd , OOP_READ ) ;
if ( fde - > flags & EVENT_FD_WRITE )
oop - > cancel_fd ( oop , fde - > fd , OOP_WRITE ) ;
2007-05-14 04:57:48 +04:00
if ( fde - > flags & EVENT_FD_AUTOCLOSE ) {
close ( fde - > fd ) ;
fde - > fd = - 1 ;
}
2005-02-15 13:36:59 +03:00
return 0 ;
}
/*
add a fd based event
return NULL on failure ( memory allocation error )
*/
2008-12-29 22:24:57 +03:00
static struct tevent_fd * oop_event_add_fd ( struct tevent_context * ev , TALLOC_CTX * mem_ctx ,
2005-02-15 13:36:59 +03:00
int fd , uint16_t flags ,
event_fd_handler_t handler ,
void * private_data )
{
2008-12-29 22:24:57 +03:00
struct tevent_fd * fde ;
2005-02-15 13:36:59 +03:00
oop_source_sys * oop_sys = ev - > additional_data ;
oop_source * oop = oop_sys_source ( oop_sys ) ;
2008-12-29 22:24:57 +03:00
fde = talloc ( mem_ctx ? mem_ctx : ev , struct tevent_fd ) ;
2005-02-15 13:36:59 +03:00
if ( ! fde ) return NULL ;
fde - > event_ctx = ev ;
fde - > fd = fd ;
fde - > flags = flags ;
fde - > handler = handler ;
fde - > private_data = private_data ;
2005-12-09 19:43:19 +03:00
fde - > additional_flags = 0 ;
2005-02-15 13:36:59 +03:00
fde - > additional_data = NULL ;
if ( fde - > flags & EVENT_FD_READ )
oop - > on_fd ( oop , fde - > fd , OOP_READ , oop_event_fd_handler , fde ) ;
if ( fde - > flags & EVENT_FD_WRITE )
oop - > on_fd ( oop , fde - > fd , OOP_WRITE , oop_event_fd_handler , fde ) ;
talloc_set_destructor ( fde , oop_event_fd_destructor ) ;
return fde ;
}
/*
return the fd event flags
*/
2008-12-29 22:24:57 +03:00
static uint16_t oop_event_get_fd_flags ( struct tevent_fd * fde )
2005-02-15 13:36:59 +03:00
{
return fde - > flags ;
}
/*
set the fd event flags
*/
2008-12-29 22:24:57 +03:00
static void oop_event_set_fd_flags ( struct tevent_fd * fde , uint16_t flags )
2005-02-15 13:36:59 +03:00
{
oop_source_sys * oop_sys ;
oop_source * oop ;
oop_sys = fde - > event_ctx - > additional_data ;
oop = oop_sys_source ( oop_sys ) ;
if ( ( fde - > flags & EVENT_FD_READ ) & & ( ! ( flags & EVENT_FD_READ ) ) )
oop - > cancel_fd ( oop , fde - > fd , OOP_READ ) ;
if ( ( ! ( fde - > flags & EVENT_FD_READ ) ) & & ( flags & EVENT_FD_READ ) )
oop - > on_fd ( oop , fde - > fd , OOP_READ , oop_event_fd_handler , fde ) ;
if ( ( fde - > flags & EVENT_FD_WRITE ) & & ( ! ( flags & EVENT_FD_WRITE ) ) )
oop - > cancel_fd ( oop , fde - > fd , OOP_WRITE ) ;
if ( ( ! ( fde - > flags & EVENT_FD_WRITE ) ) & & ( flags & EVENT_FD_WRITE ) )
oop - > on_fd ( oop , fde - > fd , OOP_WRITE , oop_event_fd_handler , fde ) ;
fde - > flags = flags ;
}
2008-12-29 22:24:57 +03:00
static int oop_event_timed_destructor ( struct tevent_timer * te ) ;
2006-05-24 11:34:11 +04:00
2008-12-29 22:24:57 +03:00
static int oop_event_timed_deny_destructor ( struct tevent_timer * te )
2005-09-27 16:54:08 +04:00
{
return - 1 ;
}
2005-02-15 13:36:59 +03:00
static void * oop_event_timed_handler ( oop_source * oop , struct timeval t , void * ptr )
{
2008-12-29 22:24:57 +03:00
struct tevent_timer * te = ptr ;
2005-02-15 13:36:59 +03:00
2005-09-27 16:54:08 +04:00
/* deny the handler to free the event */
talloc_set_destructor ( te , oop_event_timed_deny_destructor ) ;
2005-02-15 13:36:59 +03:00
te - > handler ( te - > event_ctx , te , t , te - > private_data ) ;
2005-09-27 16:54:08 +04:00
talloc_set_destructor ( te , oop_event_timed_destructor ) ;
talloc_free ( te ) ;
2005-02-15 13:36:59 +03:00
return OOP_CONTINUE ;
}
/*
destroy a timed event
*/
2008-12-29 22:24:57 +03:00
static int oop_event_timed_destructor ( struct tevent_timer * te )
2005-02-15 13:36:59 +03:00
{
2008-12-29 22:24:57 +03:00
struct tevent_context * ev = te - > event_ctx ;
2005-02-15 13:36:59 +03:00
oop_source_sys * oop_sys = ev - > additional_data ;
oop_source * oop = oop_sys_source ( oop_sys ) ;
oop - > cancel_time ( oop , te - > next_event , oop_event_timed_handler , te ) ;
return 0 ;
}
/*
add a timed event
return NULL on failure ( memory allocation error )
*/
2008-12-29 22:24:57 +03:00
static struct tevent_timer * oop_event_add_timed ( struct tevent_context * ev , TALLOC_CTX * mem_ctx ,
2005-02-15 13:36:59 +03:00
struct timeval next_event ,
event_timed_handler_t handler ,
void * private_data )
{
oop_source_sys * oop_sys = ev - > additional_data ;
oop_source * oop = oop_sys_source ( oop_sys ) ;
2008-12-29 22:24:57 +03:00
struct tevent_timer * te ;
2005-02-15 13:36:59 +03:00
2008-12-29 22:24:57 +03:00
te = talloc ( mem_ctx ? mem_ctx : ev , struct tevent_timer ) ;
2005-02-15 13:36:59 +03:00
if ( te = = NULL ) return NULL ;
te - > event_ctx = ev ;
te - > next_event = next_event ;
te - > handler = handler ;
te - > private_data = private_data ;
te - > additional_data = NULL ;
2005-09-27 16:54:08 +04:00
oop - > on_time ( oop , te - > next_event , oop_event_timed_handler , te ) ;
2005-02-15 13:36:59 +03:00
talloc_set_destructor ( te , oop_event_timed_destructor ) ;
return te ;
}
/*
do a single event loop using the events defined in ev
*/
2008-12-29 22:24:57 +03:00
static int oop_event_loop_once ( struct tevent_context * ev )
2005-02-15 13:36:59 +03:00
{
void * oop_ret ;
oop_source_sys * oop_sys = ev - > additional_data ;
oop_ret = oop_sys_run_once ( oop_sys ) ;
if ( oop_ret = = OOP_CONTINUE ) {
return 0 ;
}
return - 1 ;
}
/*
return on failure or ( with 0 ) if all fd events are removed
*/
2008-12-29 22:24:57 +03:00
static int oop_event_loop_wait ( struct tevent_context * ev )
2005-02-15 13:36:59 +03:00
{
void * oop_ret ;
oop_source_sys * oop_sys = ev - > additional_data ;
oop_ret = oop_sys_run ( oop_sys ) ;
if ( oop_ret = = OOP_CONTINUE ) {
return 0 ;
}
return - 1 ;
}
2005-12-09 19:43:19 +03:00
static const struct event_ops event_oop_ops = {
2005-02-15 13:36:59 +03:00
. context_init = oop_event_context_init ,
. add_fd = oop_event_add_fd ,
. get_fd_flags = oop_event_get_fd_flags ,
. set_fd_flags = oop_event_set_fd_flags ,
2008-12-24 15:52:57 +03:00
. add_timer = oop_event_add_timed ,
2007-01-21 11:23:14 +03:00
. add_signal = common_event_add_signal ,
2005-02-15 13:36:59 +03:00
. loop_once = oop_event_loop_once ,
. loop_wait = oop_event_loop_wait ,
} ;
2005-12-09 19:43:19 +03:00
const struct event_ops * event_liboop_get_ops ( void )
2005-02-15 13:36:59 +03:00
{
2005-12-09 19:43:19 +03:00
return & event_oop_ops ;
2005-02-15 13:36:59 +03:00
}