1tevent_thread(3) tevent tevent_thread(3)
2
3
4
6 tevent_thread - Chapter 6: Tevent with threads
7
8
10 In order to use tevent with threads, you must first understand how to
11 use the talloc library in threaded programs. For more information about
12 working with talloc, please visit talloc website where tutorial and
13 documentation are located.
14
15 If a tevent context structure is talloced from a NULL, thread-safe
16 talloc context, then it can be safe to use in a threaded program. The
17 function talloc_disable_null_tracking() must be called from the initial
18 program thread before any talloc calls are made to ensure talloc is
19 thread-safe.
20
21 Each thread must create it's own tevent context structure as follows
22 tevent_context_init(NULL) and no talloc memory contexts can be shared
23 between threads.
24
25 Separate threads using tevent in this way can communicate by writing
26 data into file descriptors that are being monitored by a tevent context
27 on another thread. For example (simplified with no error handling):
28
29 Main thread:
30
31 main()
32 {
33 talloc_disable_null_tracking();
34
35 struct tevent_context *master_ev = tevent_context_init(NULL);
36 void *mem_ctx = talloc_new(master_ev);
37
38 // Create file descriptor to monitor.
39 int pipefds[2];
40
41 pipe(pipefds);
42
43 struct tevent_fd *fde = tevent_add_fd(master_ev,
44 mem_ctx,
45 pipefds[0], // read side of pipe
46 TEVENT_FD_READ,
47 pipe_read_handler, // callback function
48 private_data_pointer);
49
50 // Create sub thread, pass pipefds[1] write side of pipe to it.
51 // The above code not shown here..
52
53 // Process events.
54 tevent_loop_wait(master_ev);
55
56 // Cleanup if loop exits.
57 talloc_free(master_ev);
58 }
59
60 When the subthread writes to pipefds[1], the function
61 pipe_read_handler() will be called in the main thread.
62
63 sophisticated use
64 A popular way to use an event library within threaded programs is to
65 allow a sub-thread to asynchronously schedule a tevent_immediate
66 function call from the event loop of another thread. This can be built
67 out of the basic functions and isolation mechanisms of tevent, but
68 tevent also comes with some utility functions that make this easier, so
69 long as you understand the limitations that using threads with talloc
70 and tevent impose.
71
72 To allow a tevent context to receive an asynchronous tevent_immediate
73 function callback from another thread, create a struct
74 tevent_thread_proxy * by calling
75
76 struct tevent_thread_proxy *tevent_thread_proxy_create(
77 struct tevent_context *dest_ev_ctx);
78
79
80 This function allocates the internal data structures to allow
81 asynchronous callbacks as a talloc child of the struct tevent_context
82 *, and returns a struct tevent_thread_proxy * that can be passed to
83 another thread.
84
85 When you have finished receiving asynchronous callbacks, simply
86 talloc_free the struct tevent_thread_proxy *, or talloc_free the struct
87 tevent_context *, which will deallocate the resources used.
88
89 To schedule an asynchronous tevent_immediate function call from one
90 thread on the tevent loop of another thread, use
91
92 void tevent_thread_proxy_schedule(struct tevent_thread_proxy *tp,
93 struct tevent_immediate **pp_im,
94 tevent_immediate_handler_t handler,
95 void **pp_private_data);
96
97
98 This function causes the function handler() to be invoked as a
99 tevent_immediate callback from the event loop of the thread that
100 created the struct tevent_thread_proxy * (so the owning struct
101 tevent_context * should be long-lived and not in the process of being
102 torn down).
103
104 The struct tevent_thread_proxy object being used here is a child of the
105 event context of the target thread. So external synchronization
106 mechanisms must be used to ensure that the target object is still in
107 use at the time of the tevent_thread_proxy_schedule() call. In the
108 example below, the request/response nature of the communication ensures
109 this.
110
111 The struct tevent_immediate **pp_im passed into this function should be
112 a struct tevent_immediate * allocated on a talloc context local to this
113 thread, and will be reparented via talloc_move to be owned by struct
114 tevent_thread_proxy *tp. *pp_im will be set to NULL on successful
115 scheduling of the tevent_immediate call.
116
117 handler() will be called as a normal tevent_immediate callback from the
118 struct tevent_context * of the destination event loop that created the
119 struct tevent_thread_proxy *
120
121 Returning from this functions does not mean that the handler has been
122 invoked, merely that it has been scheduled to be called in the
123 destination event loop.
124
125 Because the calling thread does not wait for the callback to be
126 scheduled and run on the destination thread, this is a fire-and-forget
127 call. If you wish confirmation of the handler() being successfully
128 invoked, you must ensure it replies to the caller in some way.
129
130 Because of asynchronous nature of this call, the nature of the
131 parameter passed to the destination thread has some restructions. If
132 you don't need parameters, merely pass NULL as the value of void
133 **pp_private_data.
134
135 If you wish to pass a pointer to data between the threads, it MUST be a
136 pointer to a talloced pointer, which is not part of a talloc-pool, and
137 it must not have a destructor attached. The ownership of the memory
138 pointed to will be passed from the calling thread to the tevent
139 library, and if the receiving thread does not talloc-reparent it to its
140 own contexts, it will be freed once the handler is called.
141
142 On success, *pp_private will be NULL to signify the talloc memory
143 ownership has been moved.
144
145 In practice for message passing between threads in event loops these
146 restrictions are not very onerous.
147
148 The easiest way to to a request-reply pair between tevent loops on
149 different threads is to pass the parameter block of memory back and
150 forth using a reply tevent_thread_proxy_schedule() call.
151
152 Here is an example (without error checking for simplicity):
153
154 ------------------------------------------------
155 // Master thread.
156
157 main()
158 {
159 // Make talloc thread-safe.
160
161 talloc_disable_null_tracking();
162
163 // Create the master event context.
164
165 struct tevent_context *master_ev = tevent_context_init(NULL);
166
167 // Create the master thread proxy to allow it to receive
168 // async callbacks from other threads.
169
170 struct tevent_thread_proxy *master_tp =
171 tevent_thread_proxy_create(master_ev);
172
173 // Create sub-threads, passing master_tp in
174 // some way to them.
175 // This code not shown..
176
177 // Process events.
178 // Function master_callback() below
179 // will be invoked on this thread on
180 // master_ev event context.
181
182 tevent_loop_wait(master_ev);
183
184 // Cleanup if loop exits.
185
186 talloc_free(master_ev);
187 }
188
189 // Data passed between threads.
190 struct reply_state {
191 struct tevent_thread_proxy *reply_tp;
192 pthread_t thread_id;
193 bool *p_finished;
194 };
195
196 // Callback Called in child thread context.
197
198 static void thread_callback(struct tevent_context *ev,
199 struct tevent_immediate *im,
200 void *private_ptr)
201 {
202 // Move the ownership of what private_ptr
203 // points to from the tevent library back to this thread.
204
205 struct reply_state *rsp =
206 talloc_get_type_abort(private_ptr, struct reply_state);
207
208 talloc_steal(ev, rsp);
209
210 *rsp->p_finished = true;
211
212 // im will be talloc_freed on return from this call.
213 // but rsp will not.
214 }
215
216 // Callback Called in master thread context.
217
218 static void master_callback(struct tevent_context *ev,
219 struct tevent_immediate *im,
220 void *private_ptr)
221 {
222 // Move the ownership of what private_ptr
223 // points to from the tevent library to this thread.
224
225 struct reply_state *rsp =
226 talloc_get_type_abort(private_ptr, struct reply_state);
227
228 talloc_steal(ev, rsp);
229
230 printf("Callback from thread %s0, thread_id_to_string(rsp->thread_id));
231
232 /* Now reply to the thread ! */
233 tevent_thread_proxy_schedule(rsp->reply_tp,
234 &im,
235 thread_callback,
236 &rsp);
237
238 // Note - rsp and im are now NULL as the tevent library
239 // owns the memory.
240 }
241
242 // Child thread.
243
244 static void *thread_fn(void *private_ptr)
245 {
246 struct tevent_thread_proxy *master_tp =
247 talloc_get_type_abort(private_ptr, struct tevent_thread_proxy);
248 bool finished = false;
249 int ret;
250
251 // Create our own event context.
252
253 struct tevent_context *ev = tevent_context_init(NULL);
254
255 // Create the local thread proxy to allow us to receive
256 // async callbacks from other threads.
257
258 struct tevent_thread_proxy *local_tp =
259 tevent_thread_proxy_create(master_ev);
260
261 // Setup the data to send.
262
263 struct reply_state *rsp = talloc(ev, struct reply_state);
264
265 rsp->reply_tp = local_tp;
266 rsp->thread_id = pthread_self();
267 rsp->p_finished = &finished;
268
269 // Create the immediate event to use.
270
271 struct tevent_immediate *im = tevent_create_immediate(ev);
272
273 // Call the master thread.
274
275 tevent_thread_proxy_schedule(master_tp,
276 &im,
277 master_callback,
278 &rsp);
279
280 // Note - rsp and im are now NULL as the tevent library
281 // owns the memory.
282
283 // Wait for the reply.
284
285 while (!finished) {
286 tevent_loop_once(ev);
287 }
288
289 // Cleanup.
290
291 talloc_free(ev);
292 return NULL;
293 }
294
295 Note this doesn't have to be a master-subthread communication. Any
296 thread that has access to the struct tevent_thread_proxy * pointer of
297 another thread that has called tevent_thread_proxy_create() can send
298 an async tevent_immediate request.
299
300 But remember the caveat that external synchronization must be used to
301 ensure the target struct tevent_thread_proxy * object exists at the
302 time of the tevent_thread_proxy_schedule() call or unreproducible
303 crashes will result.
304
305
306
307Version 0.9.8 Fri Jun 10 2022 tevent_thread(3)