2022-03-14 11:16:58 +01:00
/*
2005-02-15 10:36:59 +00:00
Unix SMB / CIFS implementation .
generalised event loop handling
2009-05-16 04:14:21 +02:00
INTERNAL STRUCTS . THERE ARE NO API GUARANTEES .
2022-03-14 11:16:58 +01:00
External users should only ever have to include this header when
2009-05-16 04:14:21 +02:00
implementing new tevent backends .
2005-02-15 10:36:59 +00:00
2009-02-16 08:52:06 +01:00
Copyright ( C ) Stefan Metzmacher 2005 - 2009
* * 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 10:36:59 +00:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2009-02-16 08:52:06 +01: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 10:36:59 +00:00
*/
2009-02-28 15:44:30 -05:00
struct tevent_req {
/**
* @ brief What to do on completion
*
* This is used for the user of an async request , fn is called when
* the request completes , either successfully or with an error .
*/
struct {
/**
* @ brief Completion function
* Completion function , to be filled by the API user
*/
tevent_req_fn fn ;
/**
* @ brief Private data for the completion function
*/
void * private_data ;
2023-04-24 15:04:06 +02:00
/**
* @ brief The completion function name , for flow tracing .
*/
const char * fn_name ;
2009-02-28 15:44:30 -05:00
} async ;
/**
* @ brief Private state pointer for the actual implementation
*
* The implementation doing the work for the async request needs to
* keep around current data like for example a fd event . The user of
* an async request should not touch this .
*/
void * data ;
/**
* @ brief A function to overwrite the default print function
*
2009-03-10 13:54:57 +01:00
* The implementation doing the work may want to implement a
2009-02-28 15:44:30 -05:00
* custom function to print the text representation of the async
* request .
*/
tevent_req_print_fn private_print ;
2009-08-15 09:46:23 +02:00
/**
* @ brief A function to cancel the request
*
* The implementation might want to set a function
* that is called when the tevent_req_cancel ( ) function
* was called .
*/
2023-05-23 06:38:27 +02:00
struct {
tevent_req_cancel_fn fn ;
const char * fn_name ;
} private_cancel ;
2009-08-15 09:46:23 +02:00
2013-09-27 02:29:57 +02:00
/**
* @ brief A function to cleanup the request
*
* The implementation might want to set a function
* that is called before the tevent_req_done ( ) and tevent_req_error ( )
* trigger the callers callback function .
*/
struct {
tevent_req_cleanup_fn fn ;
2023-05-23 06:39:06 +02:00
const char * fn_name ;
2013-09-27 02:29:57 +02:00
enum tevent_req_state state ;
} private_cleanup ;
2009-02-28 15:44:30 -05:00
/**
* @ brief Internal state of the request
*
* Callers should only access this via functions and never directly .
*/
struct {
/**
* @ brief The talloc type of the data pointer
*
* This is filled by the tevent_req_create ( ) macro .
*
* This for debugging only .
*/
const char * private_type ;
/**
* @ brief The location where the request was created
*
* This uses the __location__ macro via the tevent_req_create ( )
* macro .
*
* This for debugging only .
*/
2009-03-17 20:13:34 +01:00
const char * create_location ;
/**
* @ brief The location where the request was finished
*
* This uses the __location__ macro via the tevent_req_done ( ) ,
* tevent_req_error ( ) or tevent_req_nomem ( ) macro .
*
* This for debugging only .
*/
const char * finish_location ;
2009-02-28 15:44:30 -05:00
2009-08-15 09:46:23 +02:00
/**
* @ brief The location where the request was canceled
*
* This uses the __location__ macro via the
* tevent_req_cancel ( ) macro .
*
* This for debugging only .
*/
const char * cancel_location ;
2009-02-28 15:44:30 -05:00
/**
* @ brief The external state - will be queried by the caller
*
* While the async request is being processed , state will remain in
* TEVENT_REQ_IN_PROGRESS . A request is finished if
* req - > state > = TEVENT_REQ_DONE .
*/
enum tevent_req_state state ;
/**
* @ brief status code when finished
*
* This status can be queried in the async completion function . It
* will be set to 0 when everything went fine .
*/
uint64_t error ;
/**
2009-03-17 10:18:34 +01:00
* @ brief the immediate event used by tevent_req_post
2009-02-28 15:44:30 -05:00
*
*/
2009-03-17 10:18:34 +01:00
struct tevent_immediate * trigger ;
2009-02-28 15:44:30 -05:00
2011-07-08 15:54:51 +02:00
/**
* @ brief An event context which will be used to
* defer the _tevent_req_notify_callback ( ) .
*/
struct tevent_context * defer_callback_ev ;
2009-02-28 15:44:30 -05:00
/**
2009-11-26 17:18:00 +01:00
* @ brief the timer event if tevent_req_set_endtime was used
2009-02-28 15:44:30 -05:00
*
*/
struct tevent_timer * timer ;
2018-05-02 14:01:56 +02:00
/**
* @ brief The place where profiling data is kept
*/
struct tevent_req_profile * profile ;
2022-06-16 16:23:22 +02:00
size_t call_depth ;
2009-02-28 15:44:30 -05:00
} internal ;
} ;
2018-05-02 14:01:56 +02:00
struct tevent_req_profile {
struct tevent_req_profile * prev , * next ;
struct tevent_req_profile * parent ;
const char * req_name ;
pid_t pid ;
const char * start_location ;
struct timeval start_time ;
const char * stop_location ;
struct timeval stop_time ;
enum tevent_req_state state ;
uint64_t user_error ;
struct tevent_req_profile * subprofiles ;
} ;
2008-12-24 13:52:57 +01:00
struct tevent_fd {
struct tevent_fd * prev , * next ;
struct tevent_context * event_ctx ;
2014-07-22 16:51:38 +02:00
struct tevent_wrapper_glue * wrapper ;
2014-07-22 14:45:33 +02:00
bool busy ;
bool destroyed ;
2005-02-15 10:36:59 +00:00
int fd ;
2009-01-03 11:27:00 +01:00
uint16_t flags ; /* see TEVENT_FD_* flags */
2008-12-24 13:52:57 +01:00
tevent_fd_handler_t handler ;
2009-01-03 11:27:00 +01:00
tevent_fd_close_fn_t close_fn ;
2005-02-15 10:36:59 +00:00
/* this is private for the specific handler */
void * private_data ;
2009-01-02 13:26:32 +01:00
/* this is for debugging only! */
const char * handler_name ;
const char * location ;
2005-02-15 10:36:59 +00:00
/* this is private for the events_ops implementation */
2011-03-02 15:20:46 +01:00
uint64_t additional_flags ;
2005-02-15 10:36:59 +00:00
void * additional_data ;
2021-06-01 13:57:45 +02:00
/* custom tag that can be set by caller */
uint64_t tag ;
2022-11-09 22:48:10 +01:00
struct tevent_fd_mpx {
struct tevent_fd_mpx * prev , * next ;
struct tevent_fd * fde ;
struct tevent_fd * primary ;
struct tevent_fd_mpx * list ;
uint16_t total_flags ;
bool has_mpx ;
} mpx ;
2005-02-15 10:36:59 +00:00
} ;
2008-12-24 13:52:57 +01:00
struct tevent_timer {
struct tevent_timer * prev , * next ;
struct tevent_context * event_ctx ;
2014-07-22 16:51:38 +02:00
struct tevent_wrapper_glue * wrapper ;
2014-07-22 13:08:42 +02:00
bool busy ;
bool destroyed ;
2005-02-15 10:36:59 +00:00
struct timeval next_event ;
2008-12-24 13:52:57 +01:00
tevent_timer_handler_t handler ;
2005-02-15 10:36:59 +00:00
/* this is private for the specific handler */
void * private_data ;
2009-01-02 13:26:32 +01:00
/* this is for debugging only! */
const char * handler_name ;
const char * location ;
2005-02-15 10:36:59 +00:00
/* this is private for the events_ops implementation */
void * additional_data ;
2021-06-01 13:57:45 +02:00
/* custom tag that can be set by caller */
uint64_t tag ;
2005-02-15 10:36:59 +00:00
} ;
2009-03-13 15:47:33 +01:00
struct tevent_immediate {
struct tevent_immediate * prev , * next ;
struct tevent_context * event_ctx ;
2014-07-22 16:51:38 +02:00
struct tevent_wrapper_glue * wrapper ;
2014-07-22 13:08:42 +02:00
bool busy ;
bool destroyed ;
2021-06-01 14:10:05 +02:00
struct tevent_context * detach_ev_ctx ;
2009-03-13 15:47:33 +01:00
tevent_immediate_handler_t handler ;
/* this is private for the specific handler */
void * private_data ;
/* this is for debugging only! */
const char * handler_name ;
const char * create_location ;
const char * schedule_location ;
/* this is private for the events_ops implementation */
void ( * cancel_fn ) ( struct tevent_immediate * im ) ;
void * additional_data ;
2021-06-01 13:57:45 +02:00
/* custom tag that can be set by caller */
uint64_t tag ;
2009-03-13 15:47:33 +01:00
} ;
2008-12-24 13:52:57 +01:00
struct tevent_signal {
struct tevent_signal * prev , * next ;
struct tevent_context * event_ctx ;
2014-07-22 16:51:38 +02:00
struct tevent_wrapper_glue * wrapper ;
2014-07-22 13:01:01 +02:00
bool busy ;
bool destroyed ;
2007-01-21 08:23:14 +00:00
int signum ;
2007-01-21 10:32:39 +00:00
int sa_flags ;
2009-01-02 13:26:32 +01:00
tevent_signal_handler_t handler ;
/* this is private for the specific handler */
void * private_data ;
/* this is for debugging only! */
const char * handler_name ;
const char * location ;
/* this is private for the events_ops implementation */
void * additional_data ;
2021-06-01 13:57:45 +02:00
/* custom tag that can be set by caller */
uint64_t tag ;
2007-01-21 08:23:14 +00:00
} ;
2016-08-08 11:26:37 +02:00
struct tevent_threaded_context {
struct tevent_threaded_context * next , * prev ;
2016-09-07 20:25:36 +02:00
# ifdef HAVE_PTHREAD
pthread_mutex_t event_ctx_mutex ;
# endif
2016-08-08 11:26:37 +02:00
struct tevent_context * event_ctx ;
} ;
2009-01-02 13:39:56 +01:00
struct tevent_debug_ops {
2022-11-11 15:05:53 +01:00
enum tevent_debug_level max_level ;
2009-01-02 13:39:56 +01:00
void ( * debug ) ( void * context , enum tevent_debug_level level ,
2008-06-14 11:23:31 -04:00
const char * fmt , va_list ap ) PRINTF_ATTRIBUTE ( 3 , 0 ) ;
void * context ;
} ;
2009-01-02 13:39:56 +01:00
void tevent_debug ( struct tevent_context * ev , enum tevent_debug_level level ,
const char * fmt , . . . ) PRINTF_ATTRIBUTE ( 3 , 4 ) ;
2022-11-11 15:05:53 +01:00
# define TEVENT_DEBUG(__ev, __level, __fmt, ...) do { \
if ( unlikely ( ( __ev ) ! = NULL & & \
( __level ) < = ( __ev ) - > debug_ops . max_level ) ) \
{ \
tevent_debug ( ( __ev ) , ( __level ) , ( __fmt ) , __VA_ARGS__ ) ; \
} \
} while ( 0 )
2008-06-14 11:23:31 -04:00
2014-07-22 15:10:00 +02:00
void tevent_abort ( struct tevent_context * ev , const char * reason ) ;
2018-04-17 16:33:47 +02:00
void tevent_common_check_double_free ( TALLOC_CTX * ptr , const char * reason ) ;
2008-12-24 13:52:57 +01:00
struct tevent_context {
2005-02-15 10:36:59 +00:00
/* the specific events implementation */
2008-12-24 13:52:57 +01:00
const struct tevent_ops * ops ;
2007-01-05 09:35:49 +00:00
2016-08-08 08:56:23 +02:00
/*
* The following three pointers are queried on every loop_once
* in the order in which they appear here . Not measured , but
* hopefully putting them at the top together with " ops "
* should make tevent a * bit * more cache - friendly than before .
*/
/* list of signal events - used by common code */
struct tevent_signal * signal_events ;
/* List of threaded job indicators */
struct tevent_threaded_context * threaded_contexts ;
/* list of immediate events - used by common code */
struct tevent_immediate * immediate_events ;
2009-01-05 17:36:50 +01:00
/* list of fd events - used by common code */
struct tevent_fd * fd_events ;
2007-01-05 09:35:49 +00:00
/* list of timed events - used by common code */
2008-12-24 13:52:57 +01:00
struct tevent_timer * timer_events ;
2007-01-05 09:35:49 +00:00
2016-08-08 11:26:37 +02:00
/* List of scheduled immediates */
pthread_mutex_t scheduled_mutex ;
struct tevent_immediate * scheduled_immediates ;
2005-02-15 10:36:59 +00:00
/* this is private for the events_ops implementation */
void * additional_data ;
2007-01-21 08:23:14 +00:00
/* pipe hack used with signal handlers */
2016-08-12 16:32:33 +02:00
struct tevent_fd * wakeup_fde ;
2016-09-07 19:17:21 +02:00
int wakeup_fd ; /* fd to write into */
2016-08-12 16:32:33 +02:00
# ifndef HAVE_EVENT_FD
2016-09-07 19:17:21 +02:00
int wakeup_read_fd ;
2016-08-12 16:32:33 +02:00
# endif
2008-06-14 11:23:31 -04:00
/* debugging operations */
2009-01-02 13:39:56 +01:00
struct tevent_debug_ops debug_ops ;
2009-03-12 09:51:33 +01:00
/* info about the nesting status */
struct {
bool allowed ;
uint32_t level ;
2009-03-12 10:23:30 +01:00
tevent_nesting_hook hook_fn ;
void * hook_private ;
2009-03-12 09:51:33 +01:00
} nesting ;
2012-06-05 16:00:07 +10:00
struct {
2021-06-01 14:10:05 +02:00
struct {
tevent_trace_callback_t callback ;
void * private_data ;
} point ;
struct {
tevent_trace_fd_callback_t callback ;
void * private_data ;
} fde ;
struct {
tevent_trace_signal_callback_t callback ;
void * private_data ;
} se ;
struct {
tevent_trace_timer_callback_t callback ;
void * private_data ;
} te ;
struct {
tevent_trace_immediate_callback_t callback ;
void * private_data ;
} im ;
2022-03-14 11:22:15 +01:00
struct {
tevent_trace_queue_callback_t callback ;
void * private_data ;
} qe ;
2012-06-05 16:00:07 +10:00
} tracing ;
2013-02-22 12:45:39 +01:00
2014-07-22 16:51:38 +02:00
struct {
/*
* This is used on the main event context
*/
struct tevent_wrapper_glue * list ;
/*
* This is used on the wrapper event context
*/
struct tevent_wrapper_glue * glue ;
} wrapper ;
2013-02-22 12:45:39 +01:00
/*
* an optimization pointer into timer_events
* used by used by common code via
* tevent_common_add_timer_v2 ( )
*/
struct tevent_timer * last_zero_timer ;
2016-08-08 11:26:37 +02:00
# ifdef HAVE_PTHREAD
struct tevent_context * prev , * next ;
# endif
2005-02-15 10:36:59 +00:00
} ;
2009-01-05 19:23:23 +01:00
int tevent_common_context_destructor ( struct tevent_context * ev ) ;
2009-03-16 14:15:07 +01:00
int tevent_common_loop_wait ( struct tevent_context * ev ,
const char * location ) ;
2009-01-05 19:23:23 +01:00
2022-11-11 22:25:34 +01:00
struct tevent_common_fd_buf {
char buf [ 128 ] ;
} ;
const char * tevent_common_fd_str ( struct tevent_common_fd_buf * buf ,
const char * description ,
const struct tevent_fd * fde ) ;
2009-01-05 17:36:50 +01:00
int tevent_common_fd_destructor ( struct tevent_fd * fde ) ;
struct tevent_fd * tevent_common_add_fd ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
int fd ,
uint16_t flags ,
tevent_fd_handler_t handler ,
void * private_data ,
const char * handler_name ,
const char * location ) ;
2009-01-03 11:27:00 +01:00
void tevent_common_fd_set_close_fn ( struct tevent_fd * fde ,
tevent_fd_close_fn_t close_fn ) ;
2009-01-03 11:18:14 +01:00
uint16_t tevent_common_fd_get_flags ( struct tevent_fd * fde ) ;
void tevent_common_fd_set_flags ( struct tevent_fd * fde , uint16_t flags ) ;
2014-07-22 14:45:33 +02:00
int tevent_common_invoke_fd_handler ( struct tevent_fd * fde , uint16_t flags ,
bool * removed ) ;
2009-01-03 11:18:14 +01:00
2009-01-02 13:26:32 +01:00
struct tevent_timer * tevent_common_add_timer ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
struct timeval next_event ,
tevent_timer_handler_t handler ,
void * private_data ,
const char * handler_name ,
const char * location ) ;
2013-02-22 12:45:39 +01:00
struct tevent_timer * tevent_common_add_timer_v2 ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
struct timeval next_event ,
tevent_timer_handler_t handler ,
void * private_data ,
const char * handler_name ,
const char * location ) ;
2009-01-02 13:26:32 +01:00
struct timeval tevent_common_loop_timer_delay ( struct tevent_context * ) ;
2014-07-22 13:08:42 +02:00
int tevent_common_invoke_timer_handler ( struct tevent_timer * te ,
struct timeval current_time ,
bool * removed ) ;
2009-01-02 13:26:32 +01:00
2009-03-13 15:47:33 +01:00
void tevent_common_schedule_immediate ( struct tevent_immediate * im ,
struct tevent_context * ev ,
tevent_immediate_handler_t handler ,
void * private_data ,
const char * handler_name ,
const char * location ) ;
2014-07-22 13:08:42 +02:00
int tevent_common_invoke_immediate_handler ( struct tevent_immediate * im ,
bool * removed ) ;
2009-03-13 15:47:33 +01:00
bool tevent_common_loop_immediate ( struct tevent_context * ev ) ;
2016-08-08 11:26:37 +02:00
void tevent_common_threaded_activate_immediate ( struct tevent_context * ev ) ;
2016-07-29 08:53:59 +02:00
bool tevent_common_have_events ( struct tevent_context * ev ) ;
int tevent_common_wakeup_init ( struct tevent_context * ev ) ;
2016-09-07 19:47:55 +02:00
int tevent_common_wakeup_fd ( int fd ) ;
2016-07-29 08:53:59 +02:00
int tevent_common_wakeup ( struct tevent_context * ev ) ;
2009-03-13 15:47:33 +01:00
2009-01-02 13:26:32 +01:00
struct tevent_signal * tevent_common_add_signal ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
int signum ,
int sa_flags ,
tevent_signal_handler_t handler ,
void * private_data ,
const char * handler_name ,
const char * location ) ;
int tevent_common_check_signal ( struct tevent_context * ev ) ;
2010-02-09 17:02:20 +08:00
void tevent_cleanup_pending_signal_handlers ( struct tevent_signal * se ) ;
2014-07-22 13:01:01 +02:00
int tevent_common_invoke_signal_handler ( struct tevent_signal * se ,
int signum , int count , void * siginfo ,
bool * removed ) ;
2008-06-14 11:23:31 -04:00
2014-07-22 16:51:38 +02:00
struct tevent_context * tevent_wrapper_main_ev ( struct tevent_context * ev ) ;
struct tevent_wrapper_ops ;
struct tevent_wrapper_glue {
struct tevent_wrapper_glue * prev , * next ;
struct tevent_context * wrap_ev ;
struct tevent_context * main_ev ;
bool busy ;
bool destroyed ;
const struct tevent_wrapper_ops * ops ;
void * private_state ;
} ;
void tevent_wrapper_push_use_internal ( struct tevent_context * ev ,
struct tevent_wrapper_glue * wrapper ) ;
void tevent_wrapper_pop_use_internal ( const struct tevent_context * __ev_ptr ,
struct tevent_wrapper_glue * wrapper ) ;
2009-01-02 13:35:32 +01:00
bool tevent_standard_init ( void ) ;
2011-02-09 15:28:10 +01:00
bool tevent_poll_init ( void ) ;
2018-06-18 19:49:52 +02:00
bool tevent_poll_event_add_fd_internal ( struct tevent_context * ev ,
2013-02-11 11:20:28 -08:00
struct tevent_fd * fde ) ;
2012-08-13 16:06:01 +02:00
bool tevent_poll_mt_init ( void ) ;
2009-01-02 13:39:26 +01:00
# ifdef HAVE_EPOLL
2009-01-02 13:35:32 +01:00
bool tevent_epoll_init ( void ) ;
2013-12-11 18:58:41 +01:00
void tevent_epoll_set_panic_fallback ( struct tevent_context * ev ,
2013-02-11 10:43:39 -08:00
bool ( * panic_fallback ) ( struct tevent_context * ev ,
bool replay ) ) ;
2008-06-14 11:23:31 -04:00
# endif
2012-06-05 16:00:07 +10:00
2023-05-02 21:57:16 +02:00
static inline void tevent_thread_call_depth_notify (
enum tevent_thread_call_depth_cmd cmd ,
struct tevent_req * req ,
size_t depth ,
const char * fname )
{
if ( tevent_thread_call_depth_state_g . cb ! = NULL ) {
tevent_thread_call_depth_state_g . cb (
tevent_thread_call_depth_state_g . cb_private ,
cmd ,
req ,
depth ,
fname ) ;
}
}
2013-02-11 11:20:28 -08:00
2012-06-05 16:00:07 +10:00
void tevent_trace_point_callback ( struct tevent_context * ev ,
enum tevent_trace_point ) ;
2021-06-01 14:10:05 +02:00
void tevent_trace_fd_callback ( struct tevent_context * ev ,
struct tevent_fd * fde ,
enum tevent_event_trace_point ) ;
void tevent_trace_signal_callback ( struct tevent_context * ev ,
struct tevent_signal * se ,
enum tevent_event_trace_point ) ;
void tevent_trace_timer_callback ( struct tevent_context * ev ,
struct tevent_timer * te ,
enum tevent_event_trace_point ) ;
void tevent_trace_immediate_callback ( struct tevent_context * ev ,
struct tevent_immediate * im ,
enum tevent_event_trace_point ) ;
2022-03-14 11:22:15 +01:00
void tevent_trace_queue_callback ( struct tevent_context * ev ,
struct tevent_queue_entry * qe ,
enum tevent_event_trace_point ) ;
2023-08-31 18:09:28 +02:00
# include "tevent_dlinklist.h"
2022-11-09 22:48:10 +01:00
static inline void tevent_common_fd_mpx_reinit ( struct tevent_fd * fde )
{
fde - > mpx = ( struct tevent_fd_mpx ) { . fde = fde , } ;
}
2023-08-31 18:09:28 +02:00
static inline void tevent_common_fd_disarm ( struct tevent_fd * fde )
{
if ( fde - > event_ctx ! = NULL ) {
tevent_trace_fd_callback ( fde - > event_ctx , fde ,
TEVENT_EVENT_TRACE_DETACH ) ;
DLIST_REMOVE ( fde - > event_ctx - > fd_events , fde ) ;
fde - > event_ctx = NULL ;
}
2022-11-09 22:48:10 +01:00
tevent_common_fd_mpx_reinit ( fde ) ;
2023-08-31 18:09:28 +02:00
fde - > wrapper = NULL ;
}
2022-11-09 22:48:10 +01:00
/*
* tevent_common_fd_mpx_primary ( ) returns the fde that is responsible
* for the low level state .
*
* By default ( when there ' s no multiplexing ) it just returns ' any_fde ' .
*
* Note it always returns a valid pointer .
*/
static inline
struct tevent_fd * tevent_common_fd_mpx_primary ( struct tevent_fd * any_fde )
{
struct tevent_fd * primary = NULL ;
if ( any_fde - > mpx . primary ! = NULL ) {
primary = any_fde - > mpx . primary ;
} else {
primary = any_fde ;
}
return primary ;
}
/*
* tevent_common_fd_mpx_update_flags ( ) needs to be called
* if update_fde - > flags has changed . It is needed in
* order to let tevent_common_fd_mpx_flags ( ) return a valid
* result .
*/
static inline
void tevent_common_fd_mpx_update_flags ( struct tevent_fd * update_fde )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( update_fde ) ;
struct tevent_fd_mpx * mpx = NULL ;
uint16_t new_total_flags = 0 ;
if ( ! primary - > mpx . has_mpx ) {
primary - > mpx . total_flags = primary - > flags ;
return ;
}
for ( mpx = primary - > mpx . list ; mpx ! = NULL ; mpx = mpx - > next ) {
struct tevent_fd * mpx_fde = mpx - > fde ;
/* we don't care that mpx_fde might be == primary */
new_total_flags | = mpx_fde - > flags ;
}
primary - > mpx . total_flags = new_total_flags ;
}
/*
* tevent_common_fd_mpx_flags ( ) return the effective flags
* ( TEVEND_FD_ * ) of the primary fde and all multiplexed fdes .
*
* Valid after tevent_common_fd_mpx_update_flags ( ) was called
*/
static inline
uint16_t tevent_common_fd_mpx_flags ( struct tevent_fd * any_fde )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( any_fde ) ;
return primary - > mpx . total_flags ;
}
/*
* tevent_common_fd_mpx_clear_writeable ( ) clears TEVENT_FD_WRITE
* from all fdes belonging together .
*/
static inline
void tevent_common_fd_mpx_clear_writeable ( struct tevent_fd * any_fde )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( any_fde ) ;
struct tevent_fd_mpx * mpx = NULL ;
primary - > flags & = ~ TEVENT_FD_WRITE ;
for ( mpx = primary - > mpx . list ; mpx ! = NULL ; mpx = mpx - > next ) {
struct tevent_fd * mpx_fde = mpx - > fde ;
/* we don't care that mpx_fde might be == primary */
mpx_fde - > flags & = ~ TEVENT_FD_WRITE ;
}
primary - > mpx . total_flags & = ~ TEVENT_FD_WRITE ;
}
/*
* tevent_common_fd_mpx_additional_flags ( ) modifies
* fde - > additional_flags for all fdes belonging together .
*/
static inline
void tevent_common_fd_mpx_additional_flags ( struct tevent_fd * any_fde ,
uint64_t clear_flags ,
uint64_t add_flags )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( any_fde ) ;
struct tevent_fd_mpx * mpx = NULL ;
primary - > additional_flags & = ~ clear_flags ;
primary - > additional_flags | = add_flags ;
for ( mpx = primary - > mpx . list ; mpx ! = NULL ; mpx = mpx - > next ) {
struct tevent_fd * mpx_fde = mpx - > fde ;
/* we don't care that mpx_fde might be == primary */
mpx_fde - > additional_flags & = ~ clear_flags ;
mpx_fde - > additional_flags | = add_flags ;
}
}
/*
* tevent_common_fd_mpx_disarm_all ( ) detaches
* all fdes currently belonging together from each other
* and also from the tevent_context , which means their
* handler will never be called again .
*/
static inline
void tevent_common_fd_mpx_disarm_all ( struct tevent_fd * any_fde )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( any_fde ) ;
struct tevent_fd_mpx * mpx = NULL , * next = NULL ;
for ( mpx = primary - > mpx . list ; mpx ! = NULL ; mpx = next ) {
struct tevent_fd * mpx_fde = mpx - > fde ;
next = mpx - > next ;
DLIST_REMOVE ( primary - > mpx . list , mpx ) ;
if ( mpx_fde = = primary ) {
/* primary is handled below */
continue ;
}
tevent_common_fd_disarm ( mpx_fde ) ;
}
tevent_common_fd_disarm ( primary ) ;
}
/*
* tevent_common_fd_mpx_select ( ) selects the handler that
* should be called for the given low level event .
*
* Note it ' s important to pass the primary fde !
*/
static inline
struct tevent_fd * tevent_common_fd_mpx_select ( struct tevent_fd * primary ,
uint16_t flags ,
bool got_error )
{
struct tevent_fd_mpx * mpx = NULL ;
struct tevent_fd * selected = NULL ;
/* optimize for the single event case. */
if ( ! primary - > mpx . has_mpx ) {
/*
* If we got an error , we won ' t report it if
* the caller only asked for TEVENT_FD_WRITE .
*/
2011-07-13 09:46:26 +02:00
if ( got_error & &
! ( primary - > flags & ( TEVENT_FD_READ | TEVENT_FD_ERROR ) ) )
{
2022-11-09 22:48:10 +01:00
return NULL ;
}
if ( flags & primary - > flags ) {
return primary ;
}
return NULL ;
}
for ( mpx = primary - > mpx . list ; mpx ! = NULL ; mpx = mpx - > next ) {
struct tevent_fd * mpx_fde = mpx - > fde ;
/*
* If we got an error , we won ' t report it if
* the caller only asked for TEVENT_FD_WRITE .
*/
2011-07-13 09:46:26 +02:00
if ( got_error & &
! ( mpx_fde - > flags & ( TEVENT_FD_READ | TEVENT_FD_ERROR ) ) )
{
2022-11-09 22:48:10 +01:00
continue ;
}
if ( flags & mpx_fde - > flags ) {
selected = mpx_fde ;
break ;
}
}
if ( selected = = NULL ) {
return NULL ;
}
/*
* Maintain fairness and demote the just selected fde
*/
DLIST_DEMOTE_SHORT ( primary - > mpx . list , & selected - > mpx ) ;
return selected ;
}
/*
* tevent_common_fd_mpx_add ( ) searches for an existing ( active ) fde
* for the same low level fd and adds the given ' add_fde '
* as multiplexed to the found fde .
*
* If another fde was found it is returned .
* NULL is returned to indicate no match
*/
static inline
struct tevent_fd * tevent_common_fd_mpx_add ( struct tevent_fd * add_fde )
{
struct tevent_context * ev = add_fde - > event_ctx ;
struct tevent_fd * add_primary = tevent_common_fd_mpx_primary ( add_fde ) ;
uint16_t add_flags = tevent_common_fd_mpx_flags ( add_primary ) ;
struct tevent_fd * mpx_fde = NULL ;
struct tevent_fd * mpx_primary = NULL ;
struct tevent_fd_mpx * tmp = NULL ;
struct tevent_fd_mpx * next = NULL ;
/* Find the existing fde that caused the EEXIST error. */
for ( mpx_fde = ev - > fd_events ; mpx_fde ; mpx_fde = mpx_fde - > next ) {
mpx_primary = tevent_common_fd_mpx_primary ( mpx_fde ) ;
if ( mpx_primary - > fd ! = add_primary - > fd ) {
mpx_primary = NULL ;
continue ;
}
if ( mpx_primary = = add_primary ) {
mpx_primary = NULL ;
continue ;
}
if ( add_flags ! = 0 & &
tevent_common_fd_mpx_flags ( mpx_primary ) = = 0 )
{
/*
* only active events should match
*/
mpx_primary = NULL ;
continue ;
}
break ;
}
if ( mpx_primary = = NULL ) {
tevent_debug ( ev , TEVENT_DEBUG_FATAL ,
" can't find multiplex fde for fd[%d] " ,
add_fde - > fd ) ;
return NULL ;
}
/*
* If add_primary is not in it ' s own list
* we add it in order to simplify the loop below .
*/
if ( add_primary - > mpx . prev = = NULL & & add_primary - > mpx . next = = NULL ) {
DLIST_ADD_END ( add_primary - > mpx . list , & add_primary - > mpx ) ;
}
/*
* Add the new mpx_primary to its own list before others ,
* if it is not already added .
*/
if ( mpx_primary - > mpx . prev = = NULL & & mpx_primary - > mpx . next = = NULL ) {
DLIST_ADD_END ( mpx_primary - > mpx . list , & mpx_primary - > mpx ) ;
}
/*
* Now we clear all entries and move them to the
* new primary
*/
for ( tmp = add_primary - > mpx . list ; tmp ! = NULL ; tmp = next ) {
struct tevent_fd * tmp_fde = tmp - > fde ;
next = tmp - > next ;
DLIST_REMOVE ( add_primary - > mpx . list , tmp ) ;
tevent_common_fd_mpx_reinit ( tmp_fde ) ;
DLIST_ADD_END ( mpx_primary - > mpx . list , tmp ) ;
tmp - > primary = mpx_primary ;
tmp - > has_mpx = true ;
}
mpx_primary - > mpx . has_mpx = true ;
return mpx_primary ;
}
/*
* tevent_common_fd_mpx_update ( ) calls tevent_common_fd_mpx_update_flags ( )
* and compares tevent_common_fd_mpx_flags ( ) before and after .
*
* When there ' s a low level update needed the primary fde ,
* otherwise NULL is returned .
*/
static inline
struct tevent_fd * tevent_common_fd_mpx_update ( struct tevent_fd * update_fde )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( update_fde ) ;
uint16_t old_total_flags ;
uint16_t new_total_flags ;
old_total_flags = primary - > mpx . total_flags ;
tevent_common_fd_mpx_update_flags ( primary ) ;
new_total_flags = primary - > mpx . total_flags ;
if ( old_total_flags = = new_total_flags ) {
/* No update needed */
return NULL ;
}
return primary ;
}
/*
* tevent_common_fd_mpx_remove ( ) removes remove_fde from its possible primary ,
* if remove_fde is a primary itself , a new primary is selected .
*
* The remaining primary or NULL is returned .
*/
static inline
struct tevent_fd * tevent_common_fd_mpx_remove ( struct tevent_fd * remove_fde )
{
struct tevent_fd * primary = tevent_common_fd_mpx_primary ( remove_fde ) ;
struct tevent_fd_mpx * mpx = NULL , * next = NULL ;
struct tevent_fd * new_primary = NULL ;
DLIST_REMOVE ( primary - > mpx . list , & remove_fde - > mpx ) ;
if ( primary ! = remove_fde ) {
tevent_common_fd_mpx_reinit ( remove_fde ) ;
return primary ;
}
for ( mpx = primary - > mpx . list ; mpx ! = NULL ; mpx = next ) {
struct tevent_fd * mpx_fde = mpx - > fde ;
next = mpx - > next ;
DLIST_REMOVE ( primary - > mpx . list , & mpx_fde - > mpx ) ;
tevent_common_fd_mpx_reinit ( mpx_fde ) ;
mpx - > primary = new_primary ;
if ( new_primary = = NULL ) {
/*
* Select the first one as the new primary and add
* itself as the first mpx - fde to the mpx list
*/
new_primary = mpx_fde ;
DLIST_ADD ( new_primary - > mpx . list , & mpx_fde - > mpx ) ;
continue ;
}
new_primary - > mpx . has_mpx = true ;
mpx - > has_mpx = true ;
DLIST_ADD_END ( new_primary - > mpx . list , & mpx_fde - > mpx ) ;
}
/* primary == remove_fde */
tevent_common_fd_mpx_reinit ( primary ) ;
return new_primary ;
}