2003-08-13 05:53:07 +04:00
/*
Unix SMB / CIFS implementation .
2005-02-03 05:35:52 +03:00
generalised event loop handling
Copyright ( C ) Andrew Tridgell 2005
2009-02-13 17:37:35 +03:00
Copyright ( C ) Stefan Metzmacher 2005 - 2009
Copyright ( C ) Volker Lendecke 2008
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 ,
2003-08-13 05:53:07 +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/>.
2003-08-13 05:53:07 +04:00
*/
2008-12-16 21:57:09 +03:00
# ifndef __TEVENT_H__
# define __TEVENT_H__
2006-03-18 18:42:57 +03:00
2007-12-04 15:19:21 +03:00
# include <stdint.h>
2007-08-31 03:15:12 +04:00
# include <talloc.h>
2009-01-06 00:10:25 +03:00
# include <sys/time.h>
2009-02-25 12:54:19 +03:00
# include <stdbool.h>
2007-02-12 15:12:12 +03:00
2008-12-29 22:24:57 +03:00
struct tevent_context ;
2008-12-24 15:52:57 +03:00
struct tevent_ops ;
2008-12-29 22:24:57 +03:00
struct tevent_fd ;
2008-12-24 15:52:57 +03:00
struct tevent_timer ;
2009-03-13 17:47:33 +03:00
struct tevent_immediate ;
2008-12-24 15:52:57 +03:00
struct tevent_signal ;
2010-01-14 16:41:49 +03:00
/**
* @ defgroup tevent The tevent API
*
* The tevent low - level API
*
2010-01-15 18:38:24 +03:00
* This API provides the public interface to manage events in the tevent
* mainloop . Functions are provided for managing low - level events such
* as timer events , fd events and signal handling .
2010-01-14 16:41:49 +03:00
*
* @ {
*/
2005-02-03 05:35:52 +03:00
/* event handler types */
2010-01-15 18:38:24 +03:00
/**
* Called when a file descriptor monitored by tevent has
* data to be read or written on it .
*/
2009-01-03 12:42:10 +03:00
typedef void ( * tevent_fd_handler_t ) ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data ) ;
2010-01-15 18:38:24 +03:00
/**
* Called when tevent is ceasing the monitoring of a file descriptor .
*/
2009-01-03 13:27:00 +03:00
typedef void ( * tevent_fd_close_fn_t ) ( struct tevent_context * ev ,
struct tevent_fd * fde ,
int fd ,
void * private_data ) ;
2010-01-15 18:38:24 +03:00
/**
* Called when a tevent timer has fired .
*/
2009-01-03 12:42:10 +03:00
typedef void ( * tevent_timer_handler_t ) ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval current_time ,
void * private_data ) ;
2010-01-15 18:38:24 +03:00
/**
* Called when a tevent immediate event is invoked .
*/
2009-03-13 17:47:33 +03:00
typedef void ( * tevent_immediate_handler_t ) ( struct tevent_context * ctx ,
struct tevent_immediate * im ,
void * private_data ) ;
2010-01-15 18:38:24 +03:00
/**
* Called after tevent detects the specified signal .
*/
2009-01-03 12:42:10 +03:00
typedef void ( * tevent_signal_handler_t ) ( struct tevent_context * ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
void * private_data ) ;
2008-12-24 15:52:57 +03:00
2010-01-14 16:41:49 +03:00
/**
* @ brief Create a event_context structure .
*
* This must be the first events call , and all subsequent calls pass this
* event_context as the first element . Event handlers also receive this as
* their first argument .
*
* @ param [ in ] mem_ctx The memory context to use .
*
* @ return An allocated tevent context , NULL on error .
*
* @ see tevent_context_init ( )
*/
2008-12-24 15:52:57 +03:00
struct tevent_context * tevent_context_init ( TALLOC_CTX * mem_ctx ) ;
2010-01-14 16:41:49 +03:00
/**
2011-08-11 14:39:57 +04:00
* @ brief Create a event_context structure and select a specific backend .
2010-01-14 16:41:49 +03:00
*
* This must be the first events call , and all subsequent calls pass this
* event_context as the first element . Event handlers also receive this as
* their first argument .
*
* @ param [ in ] mem_ctx The memory context to use .
*
2011-08-11 14:39:57 +04:00
* @ param [ in ] name The name of the backend to use .
2010-01-14 16:41:49 +03:00
*
* @ return An allocated tevent context , NULL on error .
*/
2008-12-24 15:52:57 +03:00
struct tevent_context * tevent_context_init_byname ( TALLOC_CTX * mem_ctx , const char * name ) ;
2010-01-14 16:41:49 +03:00
2012-05-11 17:19:55 +04:00
/**
* @ brief Create a custom event context
*
* @ param [ in ] mem_ctx The memory context to use .
* @ param [ in ] ops The function pointer table of the backend .
* @ param [ in ] additional_data The additional / private data to this instance
*
* @ return An allocated tevent context , NULL on error .
*
*/
struct tevent_context * tevent_context_init_ops ( TALLOC_CTX * mem_ctx ,
const struct tevent_ops * ops ,
void * additional_data ) ;
2010-01-14 16:41:49 +03:00
/**
* @ brief List available backends .
*
* @ param [ in ] mem_ctx The memory context to use .
*
* @ return A string vector with a terminating NULL element , NULL
* on error .
*/
2008-12-24 15:52:57 +03:00
const char * * tevent_backend_list ( TALLOC_CTX * mem_ctx ) ;
2010-01-14 16:41:49 +03:00
/**
2011-05-09 09:13:00 +04:00
* @ brief Set the default tevent backend .
2010-01-14 16:41:49 +03:00
*
* @ param [ in ] backend The name of the backend to set .
*/
2008-12-24 15:52:57 +03:00
void tevent_set_default_backend ( const char * backend ) ;
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
2010-01-14 16:41:49 +03:00
/**
* @ brief Add a file descriptor based event .
*
* @ param [ in ] ev The event context to work on .
*
* @ param [ in ] mem_ctx The talloc memory context to use .
*
* @ param [ in ] fd The file descriptor to base the event on .
*
2010-01-15 18:38:24 +03:00
* @ param [ in ] flags # TEVENT_FD_READ or # TEVENT_FD_WRITE
2010-01-14 16:41:49 +03:00
*
* @ param [ in ] handler The callback handler for the event .
*
* @ param [ in ] private_data The private data passed to the callback handler .
*
* @ return The file descriptor based event , NULL on error .
*
2010-01-15 18:38:24 +03:00
* @ note To cancel the monitoring of a file descriptor , call talloc_free ( )
* on the object returned by this function .
2010-01-14 16:41:49 +03:00
*/
struct tevent_fd * tevent_add_fd ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
int fd ,
uint16_t flags ,
tevent_fd_handler_t handler ,
2010-01-15 18:38:24 +03:00
void * private_data ) ;
2010-01-14 16:41:49 +03:00
# else
2009-01-02 13:44:17 +03:00
struct tevent_fd * _tevent_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 ) ;
# define tevent_add_fd(ev, mem_ctx, fd, flags, handler, private_data) \
_tevent_add_fd ( ev , mem_ctx , fd , flags , handler , private_data , \
# handler, __location__)
2010-01-14 16:41:49 +03:00
# endif
2009-01-02 13:44:17 +03:00
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
/**
* @ brief Add a timed event
*
* @ param [ in ] ev The event context to work on .
*
* @ param [ in ] mem_ctx The talloc memory context to use .
*
* @ param [ in ] next_event Timeval specifying the absolute time to fire this
* event . This is not an offset .
*
* @ param [ in ] handler The callback handler for the event .
*
* @ param [ in ] private_data The private data passed to the callback handler .
*
* @ return The newly - created timer event , or NULL on error .
*
* @ note To cancel a timer event before it fires , call talloc_free ( ) on the
* event returned from this function . This event is automatically
* talloc_free ( ) - ed after its event handler files , if it hasn ' t been freed yet .
*
* @ note Unlike some mainloops , tevent timers are one - time events . To set up
* a recurring event , it is necessary to call tevent_add_timer ( ) again during
* the handler processing .
*
* @ note Due to the internal mainloop processing , a timer set to run
* immediately will do so after any other pending timers fire , but before
* any further file descriptor or signal handling events fire . Callers should
* not rely on this behavior !
*/
struct tevent_timer * tevent_add_timer ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
struct timeval next_event ,
tevent_timer_handler_t handler ,
void * private_data ) ;
# else
2009-01-02 13:44:17 +03:00
struct tevent_timer * _tevent_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 ) ;
# define tevent_add_timer(ev, mem_ctx, next_event, handler, private_data) \
_tevent_add_timer ( ev , mem_ctx , next_event , handler , private_data , \
2009-01-05 12:53:12 +03:00
# handler, __location__)
2010-01-15 18:38:24 +03:00
# endif
2009-01-02 13:44:17 +03:00
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
/**
* Initialize an immediate event object
*
* This object can be used to trigger an event to occur immediately after
* returning from the current event ( before any other event occurs )
*
* @ param [ in ] mem_ctx The talloc memory context to use as the parent
*
* @ return An empty tevent_immediate object . Use tevent_schedule_immediate
* to populate and use it .
*
* @ note Available as of tevent 0.9 .8
*/
struct tevent_immediate * tevent_create_immediate ( TALLOC_CTX * mem_ctx ) ;
# else
2009-03-13 17:47:33 +03:00
struct tevent_immediate * _tevent_create_immediate ( TALLOC_CTX * mem_ctx ,
const char * location ) ;
# define tevent_create_immediate(mem_ctx) \
_tevent_create_immediate ( mem_ctx , __location__ )
2010-01-15 18:38:24 +03:00
# endif
# ifdef DOXYGEN
2009-03-13 17:47:33 +03:00
2010-01-15 18:38:24 +03:00
/**
* Schedule an event for immediate execution . This event will occur
* immediately after returning from the current event ( before any other
* event occurs )
*
* @ param [ in ] im The tevent_immediate object to populate and use
* @ param [ in ] ctx The tevent_context to run this event
* @ param [ in ] handler The event handler to run when this event fires
* @ param [ in ] private_data Data to pass to the event handler
*/
void tevent_schedule_immediate ( struct tevent_immediate * im ,
struct tevent_context * ctx ,
tevent_immediate_handler_t handler ,
void * private_data ) ;
# else
2009-03-13 17:47:33 +03:00
void _tevent_schedule_immediate ( struct tevent_immediate * im ,
struct tevent_context * ctx ,
tevent_immediate_handler_t handler ,
void * private_data ,
const char * handler_name ,
const char * location ) ;
# define tevent_schedule_immediate(im, ctx, handler, private_data) \
_tevent_schedule_immediate ( im , ctx , handler , private_data , \
# handler, __location__);
2010-01-15 18:38:24 +03:00
# endif
2009-03-13 17:47:33 +03:00
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
/**
* @ brief Add a tevent signal handler
*
* tevent_add_signal ( ) creates a new event for handling a signal the next
* time through the mainloop . It implements a very simple traditional signal
* handler whose only purpose is to add the handler event into the mainloop .
*
* @ param [ in ] ev The event context to work on .
*
* @ param [ in ] mem_ctx The talloc memory context to use .
*
* @ param [ in ] signum The signal to trap
*
* @ param [ in ] handler The callback handler for the signal .
*
* @ param [ in ] sa_flags sigaction flags for this signal handler .
*
* @ param [ in ] private_data The private data passed to the callback handler .
*
* @ return The newly - created signal handler event , or NULL on error .
*
* @ note To cancel a signal handler , call talloc_free ( ) on the event returned
* from this function .
*/
struct tevent_signal * tevent_add_signal ( struct tevent_context * ev ,
TALLOC_CTX * mem_ctx ,
int signum ,
int sa_flags ,
tevent_signal_handler_t handler ,
void * private_data ) ;
# else
2009-01-02 13:44:17 +03:00
struct tevent_signal * _tevent_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 ) ;
# define tevent_add_signal(ev, mem_ctx, signum, sa_flags, handler, private_data) \
_tevent_add_signal ( ev , mem_ctx , signum , sa_flags , handler , private_data , \
# handler, __location__)
2010-01-15 18:38:24 +03:00
# endif
2008-12-24 15:52:57 +03:00
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
/**
* @ brief Pass a single time through the mainloop
*
* This will process any appropriate signal , immediate , fd and timer events
*
* @ param [ in ] ev The event context to process
*
* @ return Zero on success , nonzero if an internal error occurred
*/
int tevent_loop_once ( struct tevent_context * ev ) ;
# else
2009-03-12 11:33:26 +03:00
int _tevent_loop_once ( struct tevent_context * ev , const char * location ) ;
# define tevent_loop_once(ev) \
2010-01-15 18:38:24 +03:00
_tevent_loop_once ( ev , __location__ )
# endif
2009-03-12 11:33:26 +03:00
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
/**
* @ brief Run the mainloop
*
* The mainloop will run until there are no events remaining to be processed
*
* @ param [ in ] ev The event context to process
*
* @ return Zero if all events have been processed . Nonzero if an internal
* error occurred .
*/
int tevent_loop_wait ( struct tevent_context * ev ) ;
# else
2009-03-12 11:33:26 +03:00
int _tevent_loop_wait ( struct tevent_context * ev , const char * location ) ;
# define tevent_loop_wait(ev) \
2010-01-15 18:38:24 +03:00
_tevent_loop_wait ( ev , __location__ )
# endif
2008-12-24 15:52:57 +03:00
2010-01-15 18:38:24 +03:00
/**
* Assign a function to run when a tevent_fd is freed
*
* This function is a destructor for the tevent_fd . It does not automatically
* close the file descriptor . If this is the desired behavior , then it must be
* performed by the close_fn .
*
* @ param [ in ] fde File descriptor event on which to set the destructor
* @ param [ in ] close_fn Destructor to execute when fde is freed
*/
2009-01-03 13:27:00 +03:00
void tevent_fd_set_close_fn ( struct tevent_fd * fde ,
tevent_fd_close_fn_t close_fn ) ;
2010-01-15 18:38:24 +03:00
/**
* Automatically close the file descriptor when the tevent_fd is freed
*
* This function calls close ( fd ) internally .
*
* @ param [ in ] fde File descriptor event to auto - close
*/
2009-01-03 14:39:11 +03:00
void tevent_fd_set_auto_close ( struct tevent_fd * fde ) ;
2010-01-15 18:38:24 +03:00
/**
* Return the flags set on this file descriptor event
*
* @ param [ in ] fde File descriptor event to query
*
* @ return The flags set on the event . See # TEVENT_FD_READ and
* # TEVENT_FD_WRITE
*/
2008-12-24 15:52:57 +03:00
uint16_t tevent_fd_get_flags ( struct tevent_fd * fde ) ;
2010-01-15 18:38:24 +03:00
/**
* Set flags on a file descriptor event
*
* @ param [ in ] fde File descriptor event to set
* @ param [ in ] flags Flags to set on the event . See # TEVENT_FD_READ and
* # TEVENT_FD_WRITE
*/
2008-12-24 15:52:57 +03:00
void tevent_fd_set_flags ( struct tevent_fd * fde , uint16_t flags ) ;
2010-01-15 18:38:24 +03:00
/**
* Query whether tevent supports signal handling
*
* @ param [ in ] ev An initialized tevent context
*
* @ return True if this platform and tevent context support signal handling
*/
2009-03-12 10:48:59 +03:00
bool tevent_signal_support ( struct tevent_context * ev ) ;
2009-03-12 11:22:41 +03:00
void tevent_set_abort_fn ( void ( * abort_fn ) ( const char * reason ) ) ;
2008-12-24 15:52:57 +03:00
/* bits for file descriptor event flags */
2010-01-15 18:38:24 +03:00
/**
* Monitor a file descriptor for write availability
*/
2008-12-24 15:52:57 +03:00
# define TEVENT_FD_READ 1
2010-01-15 18:38:24 +03:00
/**
* Monitor a file descriptor for data to be read
*/
2008-12-24 15:52:57 +03:00
# define TEVENT_FD_WRITE 2
2010-01-15 18:38:24 +03:00
/**
* Convenience function for declaring a tevent_fd writable
*/
2008-12-24 15:52:57 +03:00
# define TEVENT_FD_WRITEABLE(fde) \
tevent_fd_set_flags ( fde , tevent_fd_get_flags ( fde ) | TEVENT_FD_WRITE )
2010-01-15 18:38:24 +03:00
/**
* Convenience function for declaring a tevent_fd readable
*/
2008-12-24 15:52:57 +03:00
# define TEVENT_FD_READABLE(fde) \
tevent_fd_set_flags ( fde , tevent_fd_get_flags ( fde ) | TEVENT_FD_READ )
2010-01-15 18:38:24 +03:00
/**
* Convenience function for declaring a tevent_fd non - writable
*/
2008-12-24 15:52:57 +03:00
# define TEVENT_FD_NOT_WRITEABLE(fde) \
tevent_fd_set_flags ( fde , tevent_fd_get_flags ( fde ) & ~ TEVENT_FD_WRITE )
2010-01-15 18:38:24 +03:00
/**
* Convenience function for declaring a tevent_fd non - readable
*/
2008-12-24 15:52:57 +03:00
# define TEVENT_FD_NOT_READABLE(fde) \
tevent_fd_set_flags ( fde , tevent_fd_get_flags ( fde ) & ~ TEVENT_FD_READ )
2010-01-14 16:41:49 +03:00
/**
* Debug level of tevent
*/
2009-01-02 15:39:56 +03:00
enum tevent_debug_level {
TEVENT_DEBUG_FATAL ,
TEVENT_DEBUG_ERROR ,
TEVENT_DEBUG_WARNING ,
TEVENT_DEBUG_TRACE
} ;
2010-02-08 23:10:55 +03:00
/**
* @ brief The tevent debug callbac .
*
* @ param [ in ] context The memory context to use .
*
* @ param [ in ] level The debug level .
*
* @ param [ in ] fmt The format string .
*
* @ param [ in ] ap The arguments for the format string .
*/
typedef void ( * tevent_debug_fn ) ( void * context ,
enum tevent_debug_level level ,
const char * fmt ,
va_list ap ) PRINTF_ATTRIBUTE ( 3 , 0 ) ;
2010-01-15 18:38:24 +03:00
/**
* Set destination for tevent debug messages
*
2010-02-08 23:10:55 +03:00
* @ param [ in ] ev Event context to debug
* @ param [ in ] debug Function to handle output printing
* @ param [ in ] context The context to pass to the debug function .
2010-01-15 18:38:24 +03:00
*
* @ return Always returns 0 as of version 0.9 .8
*
* @ note Default is to emit no debug messages
*/
2009-01-02 15:39:56 +03:00
int tevent_set_debug ( struct tevent_context * ev ,
2010-02-08 23:10:55 +03:00
tevent_debug_fn debug ,
2009-01-02 15:39:56 +03:00
void * context ) ;
2010-01-15 18:38:24 +03:00
/**
* Designate stderr for debug message output
*
* @ param [ in ] ev Event context to debug
*
* @ note This function will only output TEVENT_DEBUG_FATAL , TEVENT_DEBUG_ERROR
* and TEVENT_DEBUG_WARNING messages . For TEVENT_DEBUG_TRACE , please define a
* function for tevent_set_debug ( )
*/
2009-01-02 15:39:56 +03:00
int tevent_set_debug_stderr ( struct tevent_context * ev ) ;
2012-06-05 10:00:07 +04:00
enum tevent_trace_point {
/**
* Corresponds to a trace point just before waiting
*/
TEVENT_TRACE_BEFORE_WAIT ,
/**
* Corresponds to a trace point just after waiting
*/
TEVENT_TRACE_AFTER_WAIT ,
2013-02-26 18:54:57 +04:00
# define TEVENT_HAS_LOOP_ONCE_TRACE_POINTS 1
/**
* Corresponds to a trace point just before calling
* the loop_once ( ) backend function .
*/
TEVENT_TRACE_BEFORE_LOOP_ONCE ,
/**
* Corresponds to a trace point right after the
* loop_once ( ) backend function has returned .
*/
TEVENT_TRACE_AFTER_LOOP_ONCE ,
2012-06-05 10:00:07 +04:00
} ;
typedef void ( * tevent_trace_callback_t ) ( enum tevent_trace_point ,
void * private_data ) ;
/**
* Register a callback to be called at certain trace points
*
* @ param [ in ] ev Event context
* @ param [ in ] cb Trace callback
* @ param [ in ] private_data Data to be passed to callback
*
* @ note The callback will be called at trace points defined by
* tevent_trace_point . Call with NULL to reset .
*/
void tevent_set_trace_callback ( struct tevent_context * ev ,
tevent_trace_callback_t cb ,
void * private_data ) ;
/**
* Retrieve the current trace callback
*
* @ param [ in ] ev Event context
* @ param [ out ] cb Registered trace callback
* @ param [ out ] private_data Registered data to be passed to callback
*
* @ note This can be used to allow one component that wants to
* register a callback to respect the callback that another component
* has already registered .
*/
void tevent_get_trace_callback ( struct tevent_context * ev ,
tevent_trace_callback_t * cb ,
void * private_data ) ;
2010-01-14 16:41:49 +03:00
/**
* @ }
*/
/**
* @ defgroup tevent_request The tevent request functions .
* @ ingroup tevent
*
2010-12-26 01:45:30 +03:00
* A tevent_req represents an asynchronous computation .
*
* The tevent_req group of API calls is the recommended way of
* programming async computations within tevent . In particular the
* file descriptor ( tevent_add_fd ) and timer ( tevent_add_timed ) events
* are considered too low - level to be used in larger computations . To
* read and write from and to sockets , Samba provides two calls on top
* of tevent_add_fd : read_packet_send / recv and writev_send / recv . These
* requests are much easier to compose than the low - level event
* handlers called from tevent_add_fd .
*
2010-12-26 17:00:53 +03:00
* A lot of the simplicity tevent_req has brought to the notoriously
* hairy async programming came via a set of conventions that every
* async computation programmed should follow . One central piece of
* these conventions is the naming of routines and variables .
*
* Every async computation needs a name ( sensibly called " computation "
* down from here ) . From this name quite a few naming conventions are
* derived .
*
* Every computation that requires local state needs a
* @ code
* struct computation_state {
* int local_var ;
* } ;
* @ endcode
* Even if no local variables are required , such a state struct should
* be created containing a dummy variable . Quite a few helper
* functions and macros ( for example tevent_req_create ( ) ) assume such
* a state struct .
*
2010-12-26 01:45:30 +03:00
* An async computation is started by a computation_send
* function . When it is finished , its result can be received by a
2010-12-26 17:00:53 +03:00
* computation_recv function . For an example how to set up an async
* computation , see the code example in the documentation for
* tevent_req_create ( ) and tevent_req_post ( ) . The prototypes for _send
* and _recv functions should follow some conventions :
*
* @ code
* struct tevent_req * computation_send ( TALLOC_CTX * mem_ctx ,
* struct tevent_req * ev ,
* . . . further args ) ;
* int computation_recv ( struct tevent_req * req , . . . further output args ) ;
* @ endcode
*
* The " int " result of computation_recv ( ) depends on the result the
* sync version of the function would have , " int " is just an example
* here .
*
* Another important piece of the conventions is that the program flow
* is interrupted as little as possible . Because a blocking
* sub - computation requires that the flow needs to continue in a
* separate function that is the logical sequel of some computation ,
* it should lexically follow sending off the blocking
* sub - computation . Setting the callback function via
* tevent_req_set_callback ( ) requires referencing a function lexically
* below the call to tevent_req_set_callback ( ) , forward declarations
* are required . A lot of the async computations thus begin with a
* sequence of declarations such as
*
* @ code
* static void computation_step1_done ( struct tevent_req * subreq ) ;
* static void computation_step2_done ( struct tevent_req * subreq ) ;
* static void computation_step3_done ( struct tevent_req * subreq ) ;
* @ endcode
*
* It really helps readability a lot to do these forward declarations ,
* because the lexically sequential program flow makes the async
* computations almost as clear to read as a normal , sync program
* flow .
2010-01-14 16:41:49 +03:00
*
2010-12-26 01:09:19 +03:00
* It is up to the user of the async computation to talloc_free it
* after it has finished . If an async computation should be aborted ,
* the tevent_req structure can be talloc_free ' ed . After it has
* finished , it should talloc_free ' ed by the API user .
2010-01-14 16:41:49 +03:00
*
* @ {
*/
2009-02-13 17:37:35 +03:00
/**
2010-12-26 01:45:30 +03:00
* An async request moves from TEVENT_REQ_INIT to
* TEVENT_REQ_IN_PROGRESS . All other states are valid after a request
* has finished .
2009-02-13 17:37:35 +03:00
*/
enum tevent_req_state {
/**
2010-12-26 01:45:30 +03:00
* We are creating the request
2009-02-13 17:37:35 +03:00
*/
TEVENT_REQ_INIT ,
/**
2010-12-26 01:45:30 +03:00
* We are waiting the request to complete
2009-02-13 17:37:35 +03:00
*/
TEVENT_REQ_IN_PROGRESS ,
/**
2010-12-26 01:45:30 +03:00
* The request is finished successfully
2009-02-13 17:37:35 +03:00
*/
TEVENT_REQ_DONE ,
/**
2010-12-26 01:45:30 +03:00
* A user error has occurred . The user error has been
* indicated by tevent_req_error ( ) , it can be retrieved via
* tevent_req_is_error ( ) .
2009-02-13 17:37:35 +03:00
*/
TEVENT_REQ_USER_ERROR ,
/**
2010-12-26 01:45:30 +03:00
* Request timed out after the timeout set by tevent_req_set_endtime .
2009-02-13 17:37:35 +03:00
*/
TEVENT_REQ_TIMED_OUT ,
/**
2010-12-26 01:45:30 +03:00
* An internal allocation has failed , or tevent_req_nomem has
* been given a NULL pointer as the first argument .
2009-02-13 17:37:35 +03:00
*/
2009-03-10 15:54:57 +03:00
TEVENT_REQ_NO_MEMORY ,
/**
2010-12-26 01:45:30 +03:00
* The request has been received by the caller . No further
* action is valid .
2009-03-10 15:54:57 +03:00
*/
TEVENT_REQ_RECEIVED
2009-02-13 17:37:35 +03:00
} ;
/**
* @ brief An async request
*/
2009-02-28 23:44:30 +03:00
struct tevent_req ;
2009-02-13 17:37:35 +03:00
2010-05-03 22:29:54 +04:00
/**
* @ brief A tevent request callback function .
*
* @ param [ in ] req The tevent async request which executed this callback .
*/
typedef void ( * tevent_req_fn ) ( struct tevent_req * req ) ;
2009-02-13 17:37:35 +03:00
2010-05-03 22:29:54 +04:00
/**
* @ brief Set an async request callback .
*
2010-12-26 17:00:53 +03:00
* See the documentation of tevent_req_post ( ) for an example how this
* is supposed to be used .
*
2010-05-03 22:29:54 +04:00
* @ param [ in ] req The async request to set the callback .
*
* @ param [ in ] fn The callback function to set .
*
* @ param [ in ] pvt A pointer to private data to pass to the async request
* callback .
*/
2009-02-28 23:44:30 +03:00
void tevent_req_set_callback ( struct tevent_req * req , tevent_req_fn fn , void * pvt ) ;
2009-02-25 16:29:31 +03:00
2010-05-03 22:29:54 +04:00
# ifdef DOXYGEN
/**
2010-12-26 01:09:19 +03:00
* @ brief Get the private data cast to the given type for a callback from
2010-05-03 22:29:54 +04:00
* a tevent request structure .
*
2010-12-26 01:09:19 +03:00
* @ code
* static void computation_done ( struct tevent_req * subreq ) {
* struct tevent_req * req = tevent_req_callback_data ( subreq , struct tevent_req ) ;
* struct computation_state * state = tevent_req_data ( req , struct computation_state ) ;
* . . . . more things , eventually maybe call tevent_req_done ( req ) ;
* }
* @ endcode
*
2010-05-03 22:29:54 +04:00
* @ param [ in ] req The structure to get the callback data from .
*
* @ param [ in ] type The type of the private callback data to get .
*
* @ return The type casted private data set NULL if not set .
*/
void * tevent_req_callback_data ( struct tevent_req * req , # type ) ;
# else
2010-01-14 16:41:49 +03:00
void * _tevent_req_callback_data ( struct tevent_req * req ) ;
2009-02-28 23:44:30 +03:00
# define tevent_req_callback_data(_req, _type) \
talloc_get_type_abort ( _tevent_req_callback_data ( _req ) , _type )
2010-05-03 22:29:54 +04:00
# endif
2010-01-14 16:41:49 +03:00
2010-05-03 22:29:54 +04:00
# ifdef DOXYGEN
/**
* @ brief Get the private data for a callback from a tevent request structure .
*
* @ param [ in ] req The structure to get the callback data from .
*
* @ param [ in ] req The structure to get the data from .
*
* @ return The private data or NULL if not set .
*/
void * tevent_req_callback_data_void ( struct tevent_req * req ) ;
# else
2009-03-10 16:02:30 +03:00
# define tevent_req_callback_data_void(_req) \
_tevent_req_callback_data ( _req )
2010-05-03 22:29:54 +04:00
# endif
2010-01-14 16:41:49 +03:00
2010-05-03 22:29:54 +04:00
# ifdef DOXYGEN
/**
* @ brief Get the private data from a tevent request structure .
*
2010-12-26 17:00:53 +03:00
* When the tevent_req has been created by tevent_req_create , the
* result of tevent_req_data ( ) is the state variable created by
* tevent_req_create ( ) as a child of the req .
*
2010-05-03 22:29:54 +04:00
* @ param [ in ] req The structure to get the private data from .
*
2010-12-26 01:09:19 +03:00
* @ param [ in ] type The type of the private data
*
2010-05-03 22:29:54 +04:00
* @ return The private data or NULL if not set .
*/
2010-12-26 01:09:19 +03:00
void * tevent_req_data ( struct tevent_req * req , # type ) ;
2010-05-03 22:29:54 +04:00
# else
2010-01-14 16:41:49 +03:00
void * _tevent_req_data ( struct tevent_req * req ) ;
2009-02-28 23:44:30 +03:00
# define tevent_req_data(_req, _type) \
talloc_get_type_abort ( _tevent_req_data ( _req ) , _type )
2010-05-03 22:29:54 +04:00
# endif
2009-02-28 23:44:30 +03:00
2010-05-03 23:04:54 +04:00
/**
* @ brief The print function which can be set for a tevent async request .
*
* @ param [ in ] req The tevent async request .
*
* @ param [ in ] ctx A talloc memory context which can be uses to allocate
* memory .
*
* @ return An allocated string buffer to print .
*
* Example :
* @ code
* static char * my_print ( struct tevent_req * req , TALLOC_CTX * mem_ctx )
* {
* struct my_data * data = tevent_req_data ( req , struct my_data ) ;
* char * result ;
*
* result = tevent_req_default_print ( mem_ctx , req ) ;
* if ( result = = NULL ) {
* return NULL ;
* }
*
* return talloc_asprintf_append_buffer ( result , " foo=%d, bar=%d " ,
* data - > foo , data - > bar ) ;
* }
* @ endcode
*/
typedef char * ( * tevent_req_print_fn ) ( struct tevent_req * req , TALLOC_CTX * ctx ) ;
2009-02-28 23:44:30 +03:00
2010-01-14 16:41:49 +03:00
/**
* @ brief This function sets a print function for the given request .
*
* This function can be used to setup a print function for the given request .
* This will be triggered if the tevent_req_print ( ) function was
* called on the given request .
*
* @ param [ in ] req The request to use .
*
* @ param [ in ] fn A pointer to the print function
*
* @ note This function should only be used for debugging .
*/
2009-02-28 23:44:30 +03:00
void tevent_req_set_print_fn ( struct tevent_req * req , tevent_req_print_fn fn ) ;
2009-02-13 17:37:35 +03:00
2010-01-14 16:41:49 +03:00
/**
* @ brief The default print function for creating debug messages .
*
* The function should not be used by users of the async API ,
* but custom print function can use it and append custom text
* to the string .
*
* @ param [ in ] req The request to be printed .
*
* @ param [ in ] mem_ctx The memory context for the result .
*
* @ return Text representation of request .
*
*/
2009-02-25 16:29:31 +03:00
char * tevent_req_default_print ( struct tevent_req * req , TALLOC_CTX * mem_ctx ) ;
2010-01-14 16:41:49 +03:00
/**
* @ brief Print an tevent_req structure in debug messages .
*
* This function should be used by callers of the async API .
*
* @ param [ in ] mem_ctx The memory context for the result .
*
* @ param [ in ] req The request to be printed .
*
* @ return Text representation of request .
*/
2009-02-14 19:39:55 +03:00
char * tevent_req_print ( TALLOC_CTX * mem_ctx , struct tevent_req * req ) ;
2010-05-03 23:04:54 +04:00
/**
* @ brief A typedef for a cancel function for a tevent request .
*
* @ param [ in ] req The tevent request calling this function .
*
* @ return True if the request could be canceled , false if not .
*/
typedef bool ( * tevent_req_cancel_fn ) ( struct tevent_req * req ) ;
2009-08-15 11:46:23 +04:00
2010-01-14 16:41:49 +03:00
/**
* @ brief This function sets a cancel function for the given tevent request .
*
* This function can be used to setup a cancel function for the given request .
* This will be triggered if the tevent_req_cancel ( ) function was
* called on the given request .
*
* @ param [ in ] req The request to use .
*
* @ param [ in ] fn A pointer to the cancel function .
*/
2009-08-15 11:46:23 +04:00
void tevent_req_set_cancel_fn ( struct tevent_req * req , tevent_req_cancel_fn fn ) ;
2010-01-14 16:41:49 +03:00
# ifdef DOXYGEN
/**
* @ brief Try to cancel the given tevent request .
*
* This function can be used to cancel the given request .
*
* It is only possible to cancel a request when the implementation
* has registered a cancel function via the tevent_req_set_cancel_fn ( ) .
*
* @ param [ in ] req The request to use .
*
* @ return This function returns true is the request is cancelable ,
* othererwise false is returned .
*
* @ note Even if the function returns true , the caller need to wait
* for the function to complete normally .
* Only the _recv ( ) function of the given request indicates
* if the request was really canceled .
*/
bool tevent_req_cancel ( struct tevent_req * req ) ;
# else
2009-08-15 11:46:23 +04:00
bool _tevent_req_cancel ( struct tevent_req * req , const char * location ) ;
# define tevent_req_cancel(req) \
_tevent_req_cancel ( req , __location__ )
2010-01-14 16:41:49 +03:00
# endif
2009-08-15 11:46:23 +04:00
2010-01-15 18:38:24 +03:00
# ifdef DOXYGEN
2010-01-14 16:41:49 +03:00
/**
* @ brief Create an async tevent request .
*
2010-12-26 01:09:19 +03:00
* The new async request will be initialized in state TEVENT_REQ_IN_PROGRESS .
2010-01-14 16:41:49 +03:00
*
2010-12-26 01:09:19 +03:00
* @ code
* struct tevent_req * req ;
* struct computation_state * state ;
* req = tevent_req_create ( mem_ctx , & state , struct computation_state ) ;
* @ endcode
2010-01-14 16:41:49 +03:00
*
2013-07-08 17:11:25 +04:00
* Tevent_req_create ( ) allocates and zeros the state variable as a talloc
* child of its result . The state variable should be used as the talloc
* parent for all temporary variables that are allocated during the async
2010-12-26 17:00:53 +03:00
* computation . This way , when the user of the async computation frees
* the request , the state as a talloc child will be free ' d along with
* all the temporary variables hanging off the state .
*
2010-12-26 01:09:19 +03:00
* @ param [ in ] mem_ctx The memory context for the result .
* @ param [ in ] pstate Pointer to the private request state .
2010-01-14 16:41:49 +03:00
* @ param [ in ] type The name of the request .
*
* @ return A new async request . NULL on error .
*/
struct tevent_req * tevent_req_create ( TALLOC_CTX * mem_ctx ,
2010-12-26 01:09:19 +03:00
void * * pstate , # type ) ;
2010-01-14 16:41:49 +03:00
# else
2009-02-13 17:37:35 +03:00
struct tevent_req * _tevent_req_create ( TALLOC_CTX * mem_ctx ,
void * pstate ,
size_t state_size ,
const char * type ,
const char * location ) ;
# define tevent_req_create(_mem_ctx, _pstate, _type) \
_tevent_req_create ( ( _mem_ctx ) , ( _pstate ) , sizeof ( _type ) , \
# _type, __location__)
2010-01-14 16:41:49 +03:00
# endif
2009-02-13 17:37:35 +03:00
2010-05-03 22:29:54 +04:00
/**
* @ brief Set a timeout for an async request .
*
* @ param [ in ] req The request to set the timeout for .
*
* @ param [ in ] ev The event context to use for the timer .
*
* @ param [ in ] endtime The endtime of the request .
*
* @ return True if succeeded , false if not .
*/
2009-02-17 12:15:18 +03:00
bool tevent_req_set_endtime ( struct tevent_req * req ,
2009-02-13 17:37:35 +03:00
struct tevent_context * ev ,
struct timeval endtime ) ;
2010-05-03 22:29:54 +04:00
# ifdef DOXYGEN
/**
* @ brief Call the notify callback of the given tevent request manually .
*
* @ param [ in ] req The tevent request to call the notify function from .
*
* @ see tevent_req_set_callback ( )
*/
void tevent_req_notify_callback ( struct tevent_req * req ) ;
# else
2009-06-04 19:26:23 +04:00
void _tevent_req_notify_callback ( struct tevent_req * req , const char * location ) ;
# define tevent_req_notify_callback(req) \
_tevent_req_notify_callback ( req , __location__ )
2010-05-03 22:29:54 +04:00
# endif
2009-06-04 19:26:23 +04:00
2010-01-14 16:41:49 +03:00
# ifdef DOXYGEN
/**
* @ brief An async request has successfully finished .
*
* This function is to be used by implementors of async requests . When a
* request is successfully finished , this function calls the user ' s completion
* function .
*
* @ param [ in ] req The finished request .
*/
void tevent_req_done ( struct tevent_req * req ) ;
# else
2009-03-17 22:13:34 +03:00
void _tevent_req_done ( struct tevent_req * req ,
const char * location ) ;
# define tevent_req_done(req) \
_tevent_req_done ( req , __location__ )
2010-01-14 16:41:49 +03:00
# endif
2009-03-17 22:13:34 +03:00
2010-01-14 16:41:49 +03:00
# ifdef DOXYGEN
/**
* @ brief An async request has seen an error .
*
* This function is to be used by implementors of async requests . When a
* request can not successfully completed , the implementation should call this
* function with the appropriate status code .
*
* If error is 0 the function returns false and does nothing more .
*
* @ param [ in ] req The request with an error .
*
* @ param [ in ] error The error code .
*
* @ return On success true is returned , false if error is 0.
*
* @ code
* int error = first_function ( ) ;
* if ( tevent_req_error ( req , error ) ) {
* return ;
* }
*
* error = second_function ( ) ;
* if ( tevent_req_error ( req , error ) ) {
* return ;
* }
*
* tevent_req_done ( req ) ;
* return ;
* @ endcode
*/
bool tevent_req_error ( struct tevent_req * req ,
uint64_t error ) ;
# else
2009-03-17 22:13:34 +03:00
bool _tevent_req_error ( struct tevent_req * req ,
uint64_t error ,
const char * location ) ;
# define tevent_req_error(req, error) \
_tevent_req_error ( req , error , __location__ )
2010-01-14 16:41:49 +03:00
# endif
2009-03-17 22:13:34 +03:00
2010-01-14 16:41:49 +03:00
# ifdef DOXYGEN
/**
* @ brief Helper function for nomem check .
*
* Convenience helper to easily check alloc failure within a callback
* implementing the next step of an async request .
*
* @ param [ in ] p The pointer to be checked .
*
* @ param [ in ] req The request being processed .
*
* @ code
* p = talloc ( mem_ctx , bla ) ;
* if ( tevent_req_nomem ( p , req ) ) {
* return ;
* }
* @ endcode
*/
bool tevent_req_nomem ( const void * p ,
struct tevent_req * req ) ;
# else
2009-03-17 22:13:34 +03:00
bool _tevent_req_nomem ( const void * p ,
struct tevent_req * req ,
const char * location ) ;
# define tevent_req_nomem(p, req) \
_tevent_req_nomem ( p , req , __location__ )
2010-01-14 16:41:49 +03:00
# endif
2009-02-13 17:37:35 +03:00
2011-06-19 22:55:46 +04:00
# ifdef DOXYGEN
/**
* @ brief Indicate out of memory to a request
*
* @ param [ in ] req The request being processed .
*/
void tevent_req_oom ( struct tevent_req * req ) ;
# else
void _tevent_req_oom ( struct tevent_req * req ,
const char * location ) ;
# define tevent_req_oom(req) \
_tevent_req_oom ( req , __location__ )
# endif
2010-01-14 16:41:49 +03:00
/**
* @ brief Finish a request before the caller had the change to set the callback .
*
* An implementation of an async request might find that it can either finish
2010-12-26 01:09:19 +03:00
* the request without waiting for an external event , or it can not even start
2010-01-14 16:41:49 +03:00
* the engine . To present the illusion of a callback to the user of the API ,
* the implementation can call this helper function which triggers an
2011-07-09 13:01:10 +04:00
* immediate event . This way the caller can use the same calling
2010-01-14 16:41:49 +03:00
* conventions , independent of whether the request was actually deferred .
*
2010-12-26 01:45:30 +03:00
* @ code
* struct tevent_req * computation_send ( TALLOC_CTX * mem_ctx ,
* struct tevent_context * ev )
* {
* struct tevent_req * req , * subreq ;
* struct computation_state * state ;
* req = tevent_req_create ( mem_ctx , & state , struct computation_state ) ;
* if ( req = = NULL ) {
* return NULL ;
* }
* subreq = subcomputation_send ( state , ev ) ;
* if ( tevent_req_nomem ( subreq , req ) ) {
* return tevent_req_post ( req , ev ) ;
* }
2010-12-26 17:00:53 +03:00
* tevent_req_set_callback ( subreq , computation_done , req ) ;
2010-12-26 01:45:30 +03:00
* return req ;
* }
* @ endcode
*
2010-01-14 16:41:49 +03:00
* @ param [ in ] req The finished request .
*
2011-07-09 13:01:10 +04:00
* @ param [ in ] ev The tevent_context for the immediate event .
2010-01-14 16:41:49 +03:00
*
* @ return The given request will be returned .
*/
2009-02-13 17:37:35 +03:00
struct tevent_req * tevent_req_post ( struct tevent_req * req ,
struct tevent_context * ev ) ;
2011-07-08 17:54:51 +04:00
/**
* @ brief Finish multiple requests within one function
*
* Normally tevent_req_notify_callback ( ) and all wrappers
* ( e . g . tevent_req_done ( ) and tevent_req_error ( ) )
* need to be the last thing an event handler should call .
* This is because the callback is likely to destroy the
* context of the current function .
*
* If a function wants to notify more than one caller ,
* it is dangerous if it just triggers multiple callbacks
* in a row . With tevent_req_defer_callback ( ) it is possible
* to set an event context that will be used to defer the callback
* via an immediate event ( similar to tevent_req_post ( ) ) .
*
* @ code
* struct complete_state {
* struct tevent_context * ev ;
*
* struct tevent_req * * reqs ;
* } ;
*
* void complete ( struct complete_state * state )
* {
* size_t i , c = talloc_array_length ( state - > reqs ) ;
*
* for ( i = 0 ; i < c ; i + + ) {
* tevent_req_defer_callback ( state - > reqs [ i ] , state - > ev ) ;
* tevent_req_done ( state - > reqs [ i ] ) ;
* }
* }
* @ endcode
*
* @ param [ in ] req The finished request .
*
* @ param [ in ] ev The tevent_context for the immediate event .
*
* @ return The given request will be returned .
*/
void tevent_req_defer_callback ( struct tevent_req * req ,
struct tevent_context * ev ) ;
2010-01-14 16:41:49 +03:00
/**
* @ brief Check if the given request is still in progress .
*
* It is typically used by sync wrapper functions .
*
* @ param [ in ] req The request to poll .
*
* @ return The boolean form of " is in progress " .
*/
2009-02-13 17:37:35 +03:00
bool tevent_req_is_in_progress ( struct tevent_req * req ) ;
2010-01-14 16:41:49 +03:00
/**
* @ brief Actively poll for the given request to finish .
*
* This function is typically used by sync wrapper functions .
*
* @ param [ in ] req The request to poll .
*
* @ param [ in ] ev The tevent_context to be used .
*
* @ return On success true is returned . If a critical error has
* happened in the tevent loop layer false is returned .
* This is not the return value of the given request !
*
* @ note This should only be used if the given tevent context was created by the
* caller , to avoid event loop nesting .
*
* @ code
* req = tstream_writev_queue_send ( mem_ctx ,
* ev_ctx ,
* tstream ,
* send_queue ,
* iov , 2 ) ;
* ok = tevent_req_poll ( req , tctx - > ev ) ;
* rc = tstream_writev_queue_recv ( req , & sys_errno ) ;
* TALLOC_FREE ( req ) ;
* @ endcode
*/
2009-02-25 15:53:19 +03:00
bool tevent_req_poll ( struct tevent_req * req ,
struct tevent_context * ev ) ;
2010-05-03 22:29:54 +04:00
/**
2010-12-26 01:45:30 +03:00
* @ brief Get the tevent request state and the actual error set by
2010-12-26 01:09:19 +03:00
* tevent_req_error .
*
* @ code
* int computation_recv ( struct tevent_req * req , uint64_t * perr )
* {
* enum tevent_req_state state ;
* uint64_t err ;
* if ( tevent_req_is_error ( req , & state , & err ) ) {
* * perr = err ;
* return - 1 ;
* }
* return 0 ;
* }
* @ endcode
2010-05-03 22:29:54 +04:00
*
* @ param [ in ] req The tevent request to get the error from .
*
* @ param [ out ] state A pointer to store the tevent request error state .
*
* @ param [ out ] error A pointer to store the error set by tevent_req_error ( ) .
*
* @ return True if the function could set error and state , false
* otherwise .
*
* @ see tevent_req_error ( )
*/
2009-02-13 17:37:35 +03:00
bool tevent_req_is_error ( struct tevent_req * req ,
enum tevent_req_state * state ,
uint64_t * error ) ;
2010-01-14 16:41:49 +03:00
/**
* @ brief Use as the last action of a _recv ( ) function .
*
* This function destroys the attached private data .
*
* @ param [ in ] req The finished request .
*/
2009-03-10 15:54:57 +03:00
void tevent_req_received ( struct tevent_req * req ) ;
2010-05-03 22:29:54 +04:00
/**
* @ brief Create a tevent subrequest at a given time .
*
* The idea is that always the same syntax for tevent requests .
*
* @ param [ in ] mem_ctx The talloc memory context to use .
*
* @ param [ in ] ev The event handle to setup the request .
*
* @ param [ in ] wakeup_time The time to wakeup and execute the request .
*
* @ return The new subrequest , NULL on error .
*
* Example :
* @ code
2011-03-08 19:04:44 +03:00
* static void my_callback_wakeup_done ( tevent_req * subreq )
2010-05-03 22:29:54 +04:00
* {
* struct tevent_req * req = tevent_req_callback_data ( subreq ,
* struct tevent_req ) ;
* bool ok ;
*
* ok = tevent_wakeup_recv ( subreq ) ;
* TALLOC_FREE ( subreq ) ;
* if ( ! ok ) {
* tevent_req_error ( req , - 1 ) ;
* return ;
* }
* . . .
* }
* @ endcode
*
* @ code
* subreq = tevent_wakeup_send ( mem_ctx , ev , wakeup_time ) ;
* if ( tevent_req_nomem ( subreq , req ) ) {
* return false ;
* }
* tevent_set_callback ( subreq , my_callback_wakeup_done , req ) ;
* @ endcode
*
* @ see tevent_wakeup_recv ( )
*/
2009-02-17 12:22:33 +03:00
struct tevent_req * tevent_wakeup_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct timeval wakeup_time ) ;
2010-05-03 22:29:54 +04:00
/**
* @ brief Check if the wakeup has been correctly executed .
*
* This function needs to be called in the callback function set after calling
* tevent_wakeup_send ( ) .
*
* @ param [ in ] req The tevent request to check .
*
* @ return True on success , false otherwise .
*
* @ see tevent_wakeup_recv ( )
*/
2009-02-17 12:22:33 +03:00
bool tevent_wakeup_recv ( struct tevent_req * req ) ;
2009-02-13 17:37:35 +03:00
2010-01-14 16:41:49 +03:00
/* @} */
/**
* @ defgroup tevent_helpers The tevent helper functiions
* @ ingroup tevent
*
* @ todo description
*
* @ {
*/
2010-05-04 18:57:42 +04:00
/**
* @ brief Compare two timeval values .
*
* @ param [ in ] tv1 The first timeval value to compare .
*
* @ param [ in ] tv2 The second timeval value to compare .
*
* @ return 0 if they are equal .
* 1 if the first time is greater than the second .
* - 1 if the first time is smaller than the second .
*/
2009-02-17 01:34:15 +03:00
int tevent_timeval_compare ( const struct timeval * tv1 ,
const struct timeval * tv2 ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Get a zero timval value .
*
* @ return A zero timval value .
*/
2009-02-17 01:34:15 +03:00
struct timeval tevent_timeval_zero ( void ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Get a timeval value for the current time .
*
* @ return A timval value with the current time .
*/
2009-02-17 01:34:15 +03:00
struct timeval tevent_timeval_current ( void ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Get a timeval structure with the given values .
*
* @ param [ in ] secs The seconds to set .
*
2011-06-01 06:50:11 +04:00
* @ param [ in ] usecs The microseconds to set .
2010-05-04 18:57:42 +04:00
*
* @ return A timeval structure with the given values .
*/
2009-02-17 01:34:15 +03:00
struct timeval tevent_timeval_set ( uint32_t secs , uint32_t usecs ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Get the difference between two timeval values .
*
* @ param [ in ] tv1 The first timeval .
*
* @ param [ in ] tv2 The second timeval .
*
* @ return A timeval structure with the difference between the
* first and the second value .
*/
2009-02-17 01:34:15 +03:00
struct timeval tevent_timeval_until ( const struct timeval * tv1 ,
const struct timeval * tv2 ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Check if a given timeval structure is zero .
*
* @ param [ in ] tv The timeval to check if it is zero .
*
* @ return True if it is zero , false otherwise .
*/
2009-02-17 01:34:15 +03:00
bool tevent_timeval_is_zero ( const struct timeval * tv ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Add the given amount of time to a timeval structure .
*
* @ param [ in ] tv The timeval structure to add the time .
*
* @ param [ in ] secs The seconds to add to the timeval .
*
2011-06-01 06:50:11 +04:00
* @ param [ in ] usecs The microseconds to add to the timeval .
2010-05-04 18:57:42 +04:00
*
* @ return The timeval structure with the new time .
*/
2009-02-17 01:34:15 +03:00
struct timeval tevent_timeval_add ( const struct timeval * tv , uint32_t secs ,
uint32_t usecs ) ;
2010-05-04 18:57:42 +04:00
/**
* @ brief Get a timeval in the future with a specified offset from now .
*
* @ param [ in ] secs The seconds of the offset from now .
*
2011-06-01 06:50:11 +04:00
* @ param [ in ] usecs The microseconds of the offset from now .
2010-05-04 18:57:42 +04:00
*
* @ return A timval with the given offset in the future .
*/
2009-02-17 01:34:15 +03:00
struct timeval tevent_timeval_current_ofs ( uint32_t secs , uint32_t usecs ) ;
2010-01-14 16:41:49 +03:00
/* @} */
/**
* @ defgroup tevent_queue The tevent queue functions
* @ ingroup tevent
*
2010-05-10 12:27:42 +04:00
* A tevent_queue is used to queue up async requests that must be
* serialized . For example writing buffers into a socket must be
* serialized . Writing a large lump of data into a socket can require
* multiple write ( 2 ) or send ( 2 ) system calls . If more than one async
* request is outstanding to write large buffers into a socket , every
* request must individually be completed before the next one begins ,
* even if multiple syscalls are required .
*
* Take a look at @ ref tevent_queue_tutorial for more details .
2010-01-14 16:41:49 +03:00
* @ {
*/
2009-02-17 13:42:01 +03:00
struct tevent_queue ;
2011-07-28 12:45:22 +04:00
struct tevent_queue_entry ;
2009-02-17 13:42:01 +03:00
2010-05-04 19:18:22 +04:00
# ifdef DOXYGEN
/**
* @ brief Create and start a tevent queue .
*
* @ param [ in ] mem_ctx The talloc memory context to allocate the queue .
*
* @ param [ in ] name The name to use to identify the queue .
*
* @ return An allocated tevent queue on success , NULL on error .
*
2011-08-09 18:05:36 +04:00
* @ see tevent_queue_start ( )
* @ see tevent_queue_stop ( )
2010-05-04 19:18:22 +04:00
*/
struct tevent_queue * tevent_queue_create ( TALLOC_CTX * mem_ctx ,
const char * name ) ;
# else
2009-02-17 13:42:01 +03:00
struct tevent_queue * _tevent_queue_create ( TALLOC_CTX * mem_ctx ,
const char * name ,
const char * location ) ;
# define tevent_queue_create(_mem_ctx, _name) \
_tevent_queue_create ( ( _mem_ctx ) , ( _name ) , __location__ )
2010-05-04 19:18:22 +04:00
# endif
2009-02-17 13:42:01 +03:00
2010-05-04 19:18:22 +04:00
/**
* @ brief A callback trigger function run by the queue .
*
* @ param [ in ] req The tevent request the trigger function is executed on .
*
* @ param [ in ] private_data The private data pointer specified by
* tevent_queue_add ( ) .
*
* @ see tevent_queue_add ( )
2011-07-28 12:45:22 +04:00
* @ see tevent_queue_add_entry ( )
* @ see tevent_queue_add_optimize_empty ( )
2010-05-04 19:18:22 +04:00
*/
2009-02-17 13:42:01 +03:00
typedef void ( * tevent_queue_trigger_fn_t ) ( struct tevent_req * req ,
void * private_data ) ;
2010-05-04 19:18:22 +04:00
/**
* @ brief Add a tevent request to the queue .
*
* @ param [ in ] queue The queue to add the request .
*
* @ param [ in ] ev The event handle to use for the request .
*
* @ param [ in ] req The tevent request to add to the queue .
*
* @ param [ in ] trigger The function triggered by the queue when the request
2011-07-28 11:45:15 +04:00
* is called . Since tevent 0.9 .14 it ' s possible to
* pass NULL , in order to just add a " blocker " to the
* queue .
2010-05-04 19:18:22 +04:00
*
* @ param [ in ] private_data The private data passed to the trigger function .
*
* @ return True if the request has been successfully added , false
* otherwise .
*/
2009-02-17 13:42:01 +03:00
bool tevent_queue_add ( struct tevent_queue * queue ,
struct tevent_context * ev ,
struct tevent_req * req ,
tevent_queue_trigger_fn_t trigger ,
void * private_data ) ;
2010-05-04 19:18:22 +04:00
2011-07-28 12:45:22 +04:00
/**
* @ brief Add a tevent request to the queue .
*
* The request can be removed from the queue by calling talloc_free ( )
* ( or a similar function ) on the returned queue entry . This
* is the only difference to tevent_queue_add ( ) .
*
* @ param [ in ] queue The queue to add the request .
*
* @ param [ in ] ev The event handle to use for the request .
*
* @ param [ in ] req The tevent request to add to the queue .
*
* @ param [ in ] trigger The function triggered by the queue when the request
* is called . Since tevent 0.9 .14 it ' s possible to
* pass NULL , in order to just add a " blocker " to the
* queue .
*
* @ param [ in ] private_data The private data passed to the trigger function .
*
* @ return a pointer to the tevent_queue_entry if the request
* has been successfully added , NULL otherwise .
*
* @ see tevent_queue_add ( )
* @ see tevent_queue_add_optimize_empty ( )
*/
struct tevent_queue_entry * tevent_queue_add_entry (
struct tevent_queue * queue ,
struct tevent_context * ev ,
struct tevent_req * req ,
tevent_queue_trigger_fn_t trigger ,
void * private_data ) ;
/**
* @ brief Add a tevent request to the queue using a possible optimization .
*
* This tries to optimize for the empty queue case and may calls
* the trigger function directly . This is the only difference compared
* to tevent_queue_add_entry ( ) .
*
* The caller needs to be prepared that the trigger function has
* already called tevent_req_notify_callback ( ) , tevent_req_error ( ) ,
* tevent_req_done ( ) or a similar function .
*
* The request can be removed from the queue by calling talloc_free ( )
* ( or a similar function ) on the returned queue entry .
*
* @ param [ in ] queue The queue to add the request .
*
* @ param [ in ] ev The event handle to use for the request .
*
* @ param [ in ] req The tevent request to add to the queue .
*
* @ param [ in ] trigger The function triggered by the queue when the request
* is called . Since tevent 0.9 .14 it ' s possible to
* pass NULL , in order to just add a " blocker " to the
* queue .
*
* @ param [ in ] private_data The private data passed to the trigger function .
*
* @ return a pointer to the tevent_queue_entry if the request
* has been successfully added , NULL otherwise .
*
* @ see tevent_queue_add ( )
* @ see tevent_queue_add_entry ( )
*/
struct tevent_queue_entry * tevent_queue_add_optimize_empty (
struct tevent_queue * queue ,
struct tevent_context * ev ,
struct tevent_req * req ,
tevent_queue_trigger_fn_t trigger ,
void * private_data ) ;
2010-05-04 19:18:22 +04:00
/**
* @ brief Start a tevent queue .
*
* The queue is started by default .
*
* @ param [ in ] queue The queue to start .
*/
2009-03-17 12:42:55 +03:00
void tevent_queue_start ( struct tevent_queue * queue ) ;
2010-05-04 19:18:22 +04:00
/**
* @ brief Stop a tevent queue .
*
* The queue is started by default .
*
* @ param [ in ] queue The queue to stop .
*/
2009-02-17 13:42:01 +03:00
void tevent_queue_stop ( struct tevent_queue * queue ) ;
2010-05-04 19:18:22 +04:00
/**
* @ brief Get the length of the queue .
*
* @ param [ in ] queue The queue to get the length from .
*
* @ return The number of elements .
*/
2009-02-17 13:42:01 +03:00
size_t tevent_queue_length ( struct tevent_queue * queue ) ;
2011-07-28 11:46:34 +04:00
/**
* @ brief Is the tevent queue running .
*
* The queue is started by default .
*
* @ param [ in ] queue The queue .
*
* @ return Wether the queue is running or not . .
*/
bool tevent_queue_running ( struct tevent_queue * queue ) ;
2009-03-12 12:23:30 +03:00
typedef int ( * tevent_nesting_hook ) ( struct tevent_context * ev ,
void * private_data ,
uint32_t level ,
bool begin ,
void * stack_ptr ,
const char * location ) ;
2009-03-12 11:51:33 +03:00
# ifdef TEVENT_DEPRECATED
# ifndef _DEPRECATED_
# if (__GNUC__ >= 3) && (__GNUC_MINOR__ >= 1 )
# define _DEPRECATED_ __attribute__ ((deprecated))
# else
# define _DEPRECATED_
# endif
# endif
void tevent_loop_allow_nesting ( struct tevent_context * ev ) _DEPRECATED_ ;
2009-03-12 12:23:30 +03:00
void tevent_loop_set_nesting_hook ( struct tevent_context * ev ,
tevent_nesting_hook hook ,
void * private_data ) _DEPRECATED_ ;
2009-03-12 12:35:23 +03:00
int _tevent_loop_until ( struct tevent_context * ev ,
bool ( * finished ) ( void * private_data ) ,
void * private_data ,
const char * location ) _DEPRECATED_ ;
# define tevent_loop_until(ev, finished, private_data) \
_tevent_loop_until ( ev , finished , private_data , __location__ )
2009-03-12 11:51:33 +03:00
# endif
2010-03-26 13:13:27 +03:00
int tevent_re_initialise ( struct tevent_context * ev ) ;
2010-01-14 16:41:49 +03:00
/* @} */
2009-06-19 04:06:00 +04:00
/**
2010-01-14 16:41:49 +03:00
* @ defgroup tevent_ops The tevent operation functions
* @ ingroup tevent
*
2009-06-19 04:06:00 +04:00
* The following structure and registration functions are exclusively
* needed for people writing and pluggin a different event engine .
* There is nothing useful for normal tevent user in here .
2010-01-14 16:41:49 +03:00
* @ {
2009-06-19 04:06:00 +04:00
*/
struct tevent_ops {
/* context init */
int ( * context_init ) ( struct tevent_context * ev ) ;
/* fd_event functions */
struct tevent_fd * ( * 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 ) ;
void ( * set_fd_close_fn ) ( struct tevent_fd * fde ,
tevent_fd_close_fn_t close_fn ) ;
uint16_t ( * get_fd_flags ) ( struct tevent_fd * fde ) ;
void ( * set_fd_flags ) ( struct tevent_fd * fde , uint16_t flags ) ;
/* timed_event functions */
struct tevent_timer * ( * 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 ) ;
/* immediate event functions */
void ( * 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 ) ;
/* signal functions */
struct tevent_signal * ( * 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 ) ;
/* loop functions */
int ( * loop_once ) ( struct tevent_context * ev , const char * location ) ;
int ( * loop_wait ) ( struct tevent_context * ev , const char * location ) ;
} ;
bool tevent_register_backend ( const char * name , const struct tevent_ops * ops ) ;
2010-01-14 16:41:49 +03:00
/* @} */
2009-06-19 04:06:00 +04:00
/**
2010-01-14 16:41:49 +03:00
* @ defgroup tevent_compat The tevent compatibility functions
* @ ingroup tevent
*
2009-06-19 04:06:00 +04:00
* The following definitions are usueful only for compatibility with the
* implementation originally developed within the samba4 code and will be
* soon removed . Please NEVER use in new code .
2010-01-14 16:41:49 +03:00
*
* @ todo Ignore it ?
*
* @ {
2009-06-19 04:06:00 +04:00
*/
2008-12-24 15:52:57 +03:00
# ifdef TEVENT_COMPAT_DEFINES
2003-08-13 05:53:07 +04:00
2008-12-29 22:24:57 +03:00
# define event_context tevent_context
2008-12-24 15:52:57 +03:00
# define event_ops tevent_ops
2008-12-29 22:24:57 +03:00
# define fd_event tevent_fd
2008-12-24 15:52:57 +03:00
# define timed_event tevent_timer
# define signal_event tevent_signal
2003-08-13 05:53:07 +04:00
2008-12-24 15:52:57 +03:00
# define event_fd_handler_t tevent_fd_handler_t
# define event_timed_handler_t tevent_timer_handler_t
# define event_signal_handler_t tevent_signal_handler_t
2003-08-13 05:53:07 +04:00
2008-12-24 15:52:57 +03:00
# define event_context_init(mem_ctx) \
tevent_context_init ( mem_ctx )
2007-01-21 11:23:14 +03:00
2008-12-24 15:52:57 +03:00
# define event_context_init_byname(mem_ctx, name) \
tevent_context_init_byname ( mem_ctx , name )
2007-01-05 12:35:49 +03:00
2008-12-24 15:52:57 +03:00
# define event_backend_list(mem_ctx) \
tevent_backend_list ( mem_ctx )
2003-08-13 05:53:07 +04:00
2008-12-24 15:52:57 +03:00
# define event_set_default_backend(backend) \
tevent_set_default_backend ( backend )
2004-07-23 10:40:49 +04:00
2008-12-24 15:52:57 +03:00
# define event_add_fd(ev, mem_ctx, fd, flags, handler, private_data) \
tevent_add_fd ( ev , mem_ctx , fd , flags , handler , private_data )
2005-08-28 06:37:14 +04:00
2008-12-24 15:52:57 +03:00
# define event_add_timed(ev, mem_ctx, next_event, handler, private_data) \
tevent_add_timer ( ev , mem_ctx , next_event , handler , private_data )
# define event_add_signal(ev, mem_ctx, signum, sa_flags, handler, private_data) \
tevent_add_signal ( ev , mem_ctx , signum , sa_flags , handler , private_data )
# define event_loop_once(ev) \
tevent_loop_once ( ev )
# define event_loop_wait(ev) \
tevent_loop_wait ( ev )
# define event_get_fd_flags(fde) \
tevent_fd_get_flags ( fde )
# define event_set_fd_flags(fde, flags) \
tevent_fd_set_flags ( fde , flags )
# define EVENT_FD_READ TEVENT_FD_READ
# define EVENT_FD_WRITE TEVENT_FD_WRITE
2005-01-23 14:49:15 +03:00
2005-02-03 05:35:52 +03:00
# define EVENT_FD_WRITEABLE(fde) \
2008-12-24 15:52:57 +03:00
TEVENT_FD_WRITEABLE ( fde )
2005-02-03 05:35:52 +03:00
# define EVENT_FD_READABLE(fde) \
2008-12-24 15:52:57 +03:00
TEVENT_FD_READABLE ( fde )
2003-08-13 05:53:07 +04:00
2005-02-03 05:35:52 +03:00
# define EVENT_FD_NOT_WRITEABLE(fde) \
2008-12-24 15:52:57 +03:00
TEVENT_FD_NOT_WRITEABLE ( fde )
2005-02-03 05:35:52 +03:00
# define EVENT_FD_NOT_READABLE(fde) \
2008-12-24 15:52:57 +03:00
TEVENT_FD_NOT_READABLE ( fde )
2009-01-02 15:39:56 +03:00
# define ev_debug_level tevent_debug_level
# define EV_DEBUG_FATAL TEVENT_DEBUG_FATAL
# define EV_DEBUG_ERROR TEVENT_DEBUG_ERROR
# define EV_DEBUG_WARNING TEVENT_DEBUG_WARNING
# define EV_DEBUG_TRACE TEVENT_DEBUG_TRACE
# define ev_set_debug(ev, debug, context) \
tevent_set_debug ( ev , debug , context )
# define ev_set_debug_stderr(_ev) tevent_set_debug_stderr(ev)
2008-12-24 15:52:57 +03:00
# endif /* TEVENT_COMPAT_DEFINES */
2006-03-18 18:42:57 +03:00
2010-01-14 16:41:49 +03:00
/* @} */
2008-12-16 21:57:09 +03:00
# endif /* __TEVENT_H__ */