The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
io.c
Go to the documentation of this file.
1/*
2 * This program is is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or (at
5 * your option) any later version.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 * You should have received a copy of the GNU General Public License
13 * along with this program; if not, write to the Free Software
14 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17/**
18 * $Id: 2a5d40500d4cc3f6affac0e226828565d033c22d $
19 * @file curl/io.c
20 * @brief Implement asynchronous callbacks for curl
21 *
22 * @copyright 2020 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
23 */
24#include <freeradius-devel/curl/base.h>
25#include <freeradius-devel/server/module.h>
26#include <freeradius-devel/unlang/interpret.h>
27#include <freeradius-devel/util/debug.h>
28#include <freeradius-devel/util/syserror.h>
29
30/*
31 * CURL headers do:
32 *
33 * #define curl_easy_setopt(handle,opt,param) curl_easy_setopt(handle,opt,param)
34 */
36DIAG_OFF(disabled-macro-expansion)
38#define SET_MOPTION(_mandle, _opt, _val)\
39do {\
40 if ((ret = curl_multi_setopt(mandle, _opt, _val)) != CURLM_OK) {\
41 option = STRINGIFY(_opt);\
42 goto error;\
43 }\
44} while (0)
45
46/** De-queue curl requests and wake up the requests that initiated them
47 *
48 * @param[in] mhandle containing the event loop and request counter.
49 * @param[in] mandle to dequeue curl easy handles/responses from.
50 */
51static inline void _fr_curl_io_demux(fr_curl_handle_t *mhandle, CURLM *mandle)
52{
53 struct CURLMsg *m;
54 int msg_queued = 0;
55
56 while ((m = curl_multi_info_read(mandle, &msg_queued))) {
57 switch (m->msg) {
58 case CURLMSG_DONE:
59 {
61 request_t *request = NULL;
62 CURL *candle = m->easy_handle;
63 CURLcode ret;
64
65 fr_assert(candle);
66
67 mhandle->transfers--;
68
69 ret = curl_easy_getinfo(candle, CURLINFO_PRIVATE, &randle);
70 /*
71 * In curl 7.61 we got bad request data returned
72 * here after cancellations were processed.
73 *
74 * For debug builds if the value is equal to
75 * 0xdeadc341, we know the request was cancelled.
76 *
77 * There is no good work around for this other than
78 * upgrading to a newer version of curl.
79 */
80 talloc_get_type_abort(randle, fr_curl_io_request_t);
81 if (!fr_cond_assert_msg(ret == CURLE_OK,
82 "Failed retrieving request data from CURL easy handle (candle)")) {
83 curl_multi_remove_handle(mandle, candle);
84 return;
85 }
86 request = randle->request;
87
88 REQUEST_VERIFY(request);
89
90 /*
91 * If the request failed, say why...
92 */
93 if (m->data.result != CURLE_OK) {
94 REDEBUG("curl request failed: %s (%i)",
95 curl_easy_strerror(m->data.result), m->data.result);
96 }
97 randle->result = m->data.result;
98
99 /*
100 * This needs to be done last, else m->data.result
101 * ends up being junk.
102 */
103 curl_multi_remove_handle(mandle, candle);
104
106 }
107 break;
108
109 default:
110#ifndef NDEBUG
111 DEBUG4("Got unknown msg (%i) when dequeueing curl responses", msg_queued);
112#endif
113 break;
114 }
115 }
116}
117
118/** libcurl's timer expired
119 *
120 * @param[in] tl the timer was inserted into.
121 * @param[in] now The current time according to the event loop.
122 * @param[in] uctx The rlm_fr_curl_thread_t specific to this thread.
123 */
125{
126 fr_curl_handle_t *mhandle = talloc_get_type_abort(uctx, fr_curl_handle_t);
127 CURLM *mandle = mhandle->mandle;
128 CURLMcode ret;
129 int running = 0;
130
131 DEBUG4("multi-handle %p - Timer fired", mandle);
132
133 ret = curl_multi_socket_action(mandle, CURL_SOCKET_TIMEOUT, 0, &running);
134 if (ret != CURLM_OK) {
135 ERROR("multi-handle %p - Servicing failed - %s (%i)", mandle, curl_multi_strerror(ret), ret);
136 return;
137 }
138
139 DEBUG3("multi-handle %p - Serviced by timer. %i request(s) in progress, %" PRIu64 " requests(s) to dequeue",
140 mandle, running, mhandle->transfers - (uint64_t)running);
141
142 _fr_curl_io_demux(mhandle, mandle);
143}
144
145/** Service an IO event on a file descriptor
146 *
147 * @param[in] mhandle containing the event loop and request counter.
148 * @param[in] fd the IO event occurred for.
149 * @param[in] event type.
150 */
151static inline void _fr_curl_io_service(fr_curl_handle_t *mhandle, int fd, int event)
152{
153 CURLMcode ret;
154 CURLM *mandle = mhandle->mandle;
155 int running = 0;
156
157 ret = curl_multi_socket_action(mandle, fd, event, &running);
158 if (ret != CURLM_OK) {
159 ERROR("multi-handle %p - Servicing failed - %s (%i)", mandle, curl_multi_strerror(ret), ret);
160 return;
161 }
162
163 if (DEBUG_ENABLED3) {
164 char const *event_str;
165
166 switch (event) {
167 case CURL_CSELECT_ERR:
168 event_str = "error";
169 break;
170
171 case CURL_CSELECT_OUT:
172 event_str = "socket-writable";
173 break;
174
175 case CURL_CSELECT_IN:
176 event_str = "socket-readable";
177 break;
178
179 case 0:
180 event_str = "closed"; /* Not really closed, more do your own eval! */
181 break;
182
183 default:
184 event_str = "<INVALID>";
185 break;
186 }
187
188 DEBUG3("multi-handle %p - Serviced on fd %i event (%s). "
189 "%i request(s) in progress, %" PRIu64 " requests(s) to dequeue",
190 mandle, fd, event_str, running, mhandle->transfers - (uint64_t)running);
191 }
192
193
194 _fr_curl_io_demux(mhandle, mandle);
195}
196
197/** File descriptor experienced an error
198 *
199 * @param[in] el fd was registered with.
200 * @param[in] fd that errored.
201 * @param[in] flags from kevent.
202 * @param[in] fd_errno from kevent.
203 * @param[in] uctx The rlm_fr_curl_thread_t specific to this thread.
204 */
205static void _fr_curl_io_service_errored(UNUSED fr_event_list_t *el, int fd, int flags, int fd_errno, void *uctx)
206{
207 fr_curl_handle_t *mhandle = talloc_get_type_abort(uctx, fr_curl_handle_t);
208
209 DEBUG4("multi-handle %p - fd %i errored: %s", mhandle->mandle, fd, fr_syserror(fd_errno));
210
211 /*
212 * The remote server closed the socket
213 *
214 * Instead of signalling curl with CURL_CSELECT_ERR
215 * which always results in spurious errors being
216 * printed, pass in '0', which causes libcurl to do
217 * its own evaluation on the socket state, and hopefully
218 * run the right code for socket closure.
219 */
220 if (flags & EV_EOF) {
221 _fr_curl_io_service(mhandle, fd, 0);
222 return;
223 }
224
225 _fr_curl_io_service(mhandle, fd, CURL_CSELECT_ERR);
226}
227
228/** File descriptor became writable
229 *
230 * @param[in] el fd was registered with.
231 * @param[in] fd that became writable.
232 * @param[in] flags from kevent.
233 * @param[in] uctx The rlm_fr_curl_thread_t specific to this thread.
234 */
235static void _fr_curl_io_service_writable(UNUSED fr_event_list_t *el, int fd, UNUSED int flags, void *uctx)
236{
237 fr_curl_handle_t *mhandle = talloc_get_type_abort(uctx, fr_curl_handle_t);
238
239 DEBUG4("multi-handle %p - fd %i now writable", mhandle->mandle, fd);
240
241 _fr_curl_io_service(mhandle, fd, CURL_CSELECT_OUT);
242}
243
244/** File descriptor became readable
245 *
246 * @param[in] el fd was registered with.
247 * @param[in] fd that became readable.
248 * @param[in] flags from kevent.
249 * @param[in] uctx The rlm_fr_curl_thread_t specific to this thread.
250 */
251static void _fr_curl_io_service_readable(UNUSED fr_event_list_t *el, int fd, UNUSED int flags, void *uctx)
252{
253 fr_curl_handle_t *mhandle = talloc_get_type_abort(uctx, fr_curl_handle_t);
254
255 DEBUG4("multi-handle %p - fd %i now readable", mhandle->mandle, fd);
256
257 _fr_curl_io_service(mhandle, fd, CURL_CSELECT_IN);
258}
259
260/** Callback called by libcurl to set/unset timers
261 *
262 * Each rlm_fr_curl_thread_t has a timer event which is controller by libcurl.
263 * This allows libcurl to honour timeouts set on requests to remote hosts,
264 * and means we don't need to set timeouts for individual I/O events.
265 *
266 * @param[in] mandle handle requesting the timer be set/unset.
267 * @param[in] timeout_ms If > 0, how long to wait before calling curl_multi_socket_action.
268 * If == 0, we call curl_multi_socket_action as soon as possible.
269 * If < 0, we delete the timer.
270 * @param[in] ctx The rlm_fr_curl_thread_t specific to this thread.
271 * @return
272 * - 0 on success.
273 * - -1 on error.
274 */
275static int _fr_curl_io_timer_modify(CURLM *mandle, long timeout_ms, void *ctx)
276{
277 fr_curl_handle_t *mhandle = talloc_get_type_abort(ctx, fr_curl_handle_t);
278
279 if (timeout_ms < 0) {
280 if (fr_timer_delete(&mhandle->ev) < 0) {
281 PERROR("Failed deleting multi-handle timer");
282 return -1;
283 }
284 DEBUG3("multi-handle %p - Timer removed", mandle);
285 return 0;
286 }
287
288 DEBUG3("multi-handle %p will need servicing in %li ms", mandle, timeout_ms);
289
290 /*
291 * https://curl.haxx.se/libcurl/c/CURLMOPT_TIMERFUNCTION.html
292 *
293 * WARNING: even if it feels tempting, avoid calling libcurl directly from within the
294 * callback itself when the timeout_ms value is zero, since it risks triggering an
295 * unpleasant recursive behavior that immediately calls another call to the callback
296 * with a zero timeout...
297 *
298 * Setting a timeout of zero when calling fr_timer_in should result in the event
299 * repeating at most twice during one iteration of the event loop.
300 *
301 * In a previous version of this code we called curl_multi_socket_action immediately
302 * if timeout_ms was 0. It was observed that this lead to this callback being called
303 * ~665 times per request which is why we no longer do that.
304 */
305 if (fr_timer_in(mhandle, mhandle->el->tl, &mhandle->ev,
306 fr_time_delta_from_msec(timeout_ms),
307 false, _fr_curl_io_timer_expired, mhandle) < 0) return -1;
308 return 0;
309}
310
311/** Called by libcurl to register a socket that it's intefr_curled in receiving IO events for
312 *
313 *
314 * @param[in] easy handle this fd relates to.
315 * @param[in] fd File descriptor curl wants to be notified about.
316 * @param[in] what Which events libcurl wants to be notified of, may be one of:
317 * - CURL_POLL_IN Wait for incoming data. For the socket
318 * to become readable.
319 * - CURL_POLL_OUT Wait for outgoing data. For the socket
320 * to become writable.
321 * - CURL_POLL_INOUT Wait for incoming and outgoing data.
322 * For the socket to become readable or writable.
323 * - CURL_POLL_REMOVE The specified socket/file descriptor is no
324 * longer used by libcurl.
325 * @param[in] ctx The fr_curl_handle_t specific to this thread.
326 * @param[in] fd_ctx Private data associated with the socket.
327 */
328static int _fr_curl_io_event_modify(UNUSED CURL *easy, curl_socket_t fd, int what, void *ctx, UNUSED void *fd_ctx)
329{
330 fr_curl_handle_t *mhandle = talloc_get_type_abort(ctx, fr_curl_handle_t);
331
332 switch (what) {
333 case CURL_POLL_IN:
334 if (fr_event_fd_insert(mhandle, NULL, mhandle->el, fd,
336 NULL,
338 mhandle) < 0) {
339 PERROR("multi-handle %p registration failed for read+error events on FD %i",
340 mhandle->mandle, fd);
341 return -1;
342 }
343 DEBUG4("multi-handle %p registered for read+error events on FD %i", mhandle->mandle, fd);
344 break;
345
346 case CURL_POLL_OUT:
347 if (fr_event_fd_insert(mhandle, NULL, mhandle->el, fd,
348 NULL,
351 mhandle) < 0) {
352 PERROR("multi-handle %p registration failed for write+error events on FD %i",
353 mhandle->mandle, fd);
354 return -1;
355 }
356 DEBUG4("multi-handle %p registered for write+error events on FD %i", mhandle->mandle, fd);
357 break;
358
359 case CURL_POLL_INOUT:
360 if (fr_event_fd_insert(mhandle, NULL, mhandle->el, fd,
364 mhandle) < 0) {
365 PERROR("multi-handle %p registration failed for read+write+error events on FD %i",
366 mhandle->mandle, fd);
367 return -1;
368 }
369 DEBUG4("multi-handle %p registered for read+write+error events on FD %i", mhandle->mandle, fd);
370 break;
371
372 case CURL_POLL_REMOVE:
373 if (fr_event_fd_delete(mhandle->el, fd, FR_EVENT_FILTER_IO) < 0) {
374 PERROR("multi-handle %p de-registration failed for FD %i", mhandle->mandle, fd);
375 return -1;
376 }
377 DEBUG4("multi-handle %p unregistered events for FD %i", mhandle->mandle, fd);
378 break;
379
380 default:
381 fr_assert(0);
382 return -1;
383 }
384
385 return CURLM_OK;
386}
387
388/** Callback to receive debugging data from libcurl
389 *
390 * @note Should only be set on a handle if RDEBUG_ENABLED3 is true.
391 *
392 * @param[in] candle Curl handle the debugging data pertains to.
393 * @param[in] type The type of debugging data we received.
394 * @param[in] data Buffer containing debug data (not \0 terminated). Despite the
395 * type being char *, this can be binary data depending on the
396 * curl_infotype.
397 * @param[in] len The length of the data in the buffer.
398 * @param[in] uctx The current request.
399 * @return
400 * - 0 (we always indicate success)
401 */
402static int curl_debug_log(UNUSED CURL *candle, curl_infotype type, char *data, size_t len, void *uctx)
403{
404 request_t *request = talloc_get_type_abort(uctx, request_t);
405 char const *p = data, *q, *end = p + len;
406 char const *verb;
407
408 switch (type) {
409 case CURLINFO_TEXT:
410 /*
411 * Curl debug output has trailing newlines, and could conceivably
412 * span multiple lines. Take care of both cases.
413 */
414 while (p < end) {
415 q = memchr(p, '\n', end - p);
416 if (!q) q = end;
417
418 RDEBUG3("libcurl - %pV", fr_box_strvalue_len(p, q ? q - p : p - end));
419 p = q + 1;
420 }
421
422 break;
423
424 case CURLINFO_HEADER_IN:
425 verb = "received";
426 print_header:
427 while (p < end) {
428 q = memchr(p, '\n', end - p);
429 q = q ? q + 1 : end;
430
431 if (RDEBUG_ENABLED4) {
432 RHEXDUMP4((uint8_t const *)p, q - p,
433 "%s header: %pV",
434 verb, fr_box_strvalue_len(p, q - p));
435 } else {
436 RDEBUG3("%s header: %pV",
437 verb, fr_box_strvalue_len(p, q - p));
438 }
439 p = q;
440 }
441 break;
442
443 case CURLINFO_HEADER_OUT:
444 verb = "sending";
445 goto print_header;
446
447 case CURLINFO_DATA_IN:
448 RHEXDUMP4((uint8_t const *)data, len, "received data[length %zu]", len);
449 break;
450
451 case CURLINFO_DATA_OUT:
452 RHEXDUMP4((uint8_t const *)data, len, "sending data[length %zu]", len);
453 break;
454
455 case CURLINFO_SSL_DATA_OUT:
456 RHEXDUMP4((uint8_t const *)data, len, "sending ssl-data[length %zu]", len);
457 break;
458
459 case CURLINFO_SSL_DATA_IN:
460 RHEXDUMP4((uint8_t const *)data, len, "received ssl-data[length %zu]", len);
461 break;
462
463 default:
464 RHEXDUMP3((uint8_t const *)data, len, "libcurl - debug data (unknown type %i)", (int)type);
465 break;
466 }
467
468 return 0;
469}
470
471/** Sends a request using libcurl
472 *
473 * Send the actual curl request to the server. The response will be handled by
474 * the numerous callbacks configured for the easy handle.
475 *
476 * @param[in] mhandle Thread-specific mhandle wrapper.
477 * @param[in] request Current request.
478 * @param[in] randle representing the request.
479 * @return
480 * - 0 on success.
481 * - -1 on failure.
482 */
484{
485 CURLcode ret;
486 CURLMcode mret;
487
488 REQUEST_VERIFY(request);
489
490 randle->request = request;
491
492 /*
493 * Set debugging functions so we can track the
494 * IO request's progress.
495 */
496 if (RDEBUG_ENABLED3) {
497 FR_CURL_REQUEST_SET_OPTION(CURLOPT_DEBUGFUNCTION, curl_debug_log);
498 FR_CURL_REQUEST_SET_OPTION(CURLOPT_DEBUGDATA, request);
499 FR_CURL_REQUEST_SET_OPTION(CURLOPT_VERBOSE, 1L);
500 }
501
502 /*
503 * Stick the current request in the curl handle's
504 * private data. This makes it simple to resume
505 * the request in the demux function later...
506 *
507 * Note: If you get 0xdeadc341 in the private data
508 * in the demux function, the result for the easy
509 * handle was erroneously delivered after a
510 * cancellation.
511 */
512 ret = curl_easy_setopt(randle->candle, CURLOPT_PRIVATE, randle);
513 if (ret != CURLE_OK) {
514 REDEBUG("Request failed: %i - %s", ret, curl_easy_strerror(ret));
515 return -1;
516 }
517
518 /*
519 * Increment here, else the debug output looks
520 * messed up is curl_multi_add_handle triggers
521 * event loop modifications calls immediately.
522 */
523 mhandle->transfers++;
524 mret = curl_multi_add_handle(mhandle->mandle, randle->candle);
525 if (mret != CURLM_OK) {
526 mhandle->transfers--;
527 REDEBUG("Request failed: %i - %s", mret, curl_multi_strerror(mret));
528 return -1;
529 }
530
531 return 0;
532
533error:
534 return -1;
535}
536
538{
539 curl_easy_cleanup(randle->candle);
540
541 return 0;
542}
543
544/** Allocate a new curl easy request and wrapper struct
545 *
546 */
548{
549 fr_curl_io_request_t *randle;
550
551 randle = talloc_zero(ctx, fr_curl_io_request_t);
552 if (unlikely(!randle)) return NULL;
553
554 randle->candle = curl_easy_init();
555 if (!randle->candle) {
556 talloc_free(randle);
557 return NULL;
558 }
559
560 talloc_set_destructor(randle, _fr_curl_io_request_free);
561
562 return randle;
563}
564
565/** Free the multi-handle
566 *
567 */
568static int _mhandle_free(fr_curl_handle_t *mhandle)
569{
570 curl_multi_cleanup(mhandle->mandle);
571
572 return 0;
573}
574
575/** Performs the libcurl initialisation of the thread
576 *
577 * @param[in] ctx to alloc handle in.
578 * @param[in] el to initial.
579 * @param[in] multiplex Run multiple requests over the same connection simultaneously.
580 * HTTP/2 only.
581 * @return
582 * - 0 on success.
583 * - -1 on error.
584 */
587#ifndef CURLPIPE_MULTIPLEX
588 UNUSED
589#endif
590 bool multiplex)
591{
592 CURLMcode ret;
593 CURLM *mandle;
594 fr_curl_handle_t *mhandle;
595 char const *option;
596
597 mandle = curl_multi_init();
598 if (!mandle) {
599 ERROR("Curl multi-handle instantiation failed");
600 return NULL;
601 }
602
603 /*
604 * Structure to store extra data.
605 *
606 * Passed to all curl I/O and timer callbacks.
607 *
608 * If uctx data is needed in the future, can be added here.
609 */
610 MEM(mhandle = talloc_zero(ctx, fr_curl_handle_t));
611 mhandle->el = el;
612 mhandle->mandle = mandle;
613 talloc_set_destructor(mhandle, _mhandle_free);
614
615 SET_MOPTION(mandle, CURLMOPT_TIMERFUNCTION, _fr_curl_io_timer_modify);
616 SET_MOPTION(mandle, CURLMOPT_TIMERDATA, mhandle);
617
618 SET_MOPTION(mandle, CURLMOPT_SOCKETFUNCTION, _fr_curl_io_event_modify);
619 SET_MOPTION(mandle, CURLMOPT_SOCKETDATA, mhandle);
620
621#ifdef CURLPIPE_MULTIPLEX
622 SET_MOPTION(mandle, CURLMOPT_PIPELINING, multiplex ? CURLPIPE_MULTIPLEX : CURLPIPE_NOTHING);
623#endif
624
625 return mhandle;
626
627error:
628 ERROR("Failed setting curl option %s: %s (%i)", option, curl_multi_strerror(ret), ret);
629
630 return NULL;
631}
#define DIAG_UNKNOWN_PRAGMAS
Definition build.h:458
#define DIAG_ON(_x)
Definition build.h:460
#define unlikely(_x)
Definition build.h:383
#define UNUSED
Definition build.h:317
#define DIAG_OFF(_x)
Definition build.h:459
fr_timer_t * ev
Multi-Handle timer.
Definition base.h:93
#define FR_CURL_REQUEST_SET_OPTION(_x, _y)
Definition base.h:67
fr_event_list_t * el
Event list servicing I/O events.
Definition base.h:92
CURLcode result
Result of executing the request.
Definition base.h:103
uint64_t transfers
How many transfers are current in progress.
Definition base.h:94
CURLM * mandle
The multi handle.
Definition base.h:95
request_t * request
Current request.
Definition base.h:104
CURL * candle
Request specific handle.
Definition base.h:102
Uctx data for timer and I/O functions.
Definition base.h:91
Structure representing an individual request being passed to curl for processing.
Definition base.h:101
#define fr_cond_assert_msg(_x, _fmt,...)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
Definition debug.h:156
#define MEM(x)
Definition debug.h:36
#define ERROR(fmt,...)
Definition dhcpclient.c:41
#define fr_event_fd_insert(...)
Definition event.h:248
@ FR_EVENT_FILTER_IO
Combined filter for read/write functions/.
Definition event.h:84
void unlang_interpret_mark_runnable(request_t *request)
Mark a request as resumable.
Definition interpret.c:1374
static int _fr_curl_io_event_modify(UNUSED CURL *easy, curl_socket_t fd, int what, void *ctx, UNUSED void *fd_ctx)
Called by libcurl to register a socket that it's intefr_curled in receiving IO events for.
Definition io.c:328
static int _fr_curl_io_request_free(fr_curl_io_request_t *randle)
Definition io.c:537
static void _fr_curl_io_service_writable(UNUSED fr_event_list_t *el, int fd, UNUSED int flags, void *uctx)
File descriptor became writable.
Definition io.c:235
#define SET_MOPTION(_mandle, _opt, _val)
Definition io.c:38
static void _fr_curl_io_demux(fr_curl_handle_t *mhandle, CURLM *mandle)
De-queue curl requests and wake up the requests that initiated them.
Definition io.c:51
static void _fr_curl_io_service_errored(UNUSED fr_event_list_t *el, int fd, int flags, int fd_errno, void *uctx)
File descriptor experienced an error.
Definition io.c:205
static int curl_debug_log(UNUSED CURL *candle, curl_infotype type, char *data, size_t len, void *uctx)
Callback to receive debugging data from libcurl.
Definition io.c:402
static void _fr_curl_io_service_readable(UNUSED fr_event_list_t *el, int fd, UNUSED int flags, void *uctx)
File descriptor became readable.
Definition io.c:251
int fr_curl_io_request_enqueue(fr_curl_handle_t *mhandle, request_t *request, fr_curl_io_request_t *randle)
Sends a request using libcurl.
Definition io.c:483
static int _fr_curl_io_timer_modify(CURLM *mandle, long timeout_ms, void *ctx)
Callback called by libcurl to set/unset timers.
Definition io.c:275
fr_curl_io_request_t * fr_curl_io_request_alloc(TALLOC_CTX *ctx)
Allocate a new curl easy request and wrapper struct.
Definition io.c:547
fr_curl_handle_t * fr_curl_io_init(TALLOC_CTX *ctx, fr_event_list_t *el, UNUSED bool multiplex)
Performs the libcurl initialisation of the thread.
Definition io.c:585
static void _fr_curl_io_timer_expired(UNUSED fr_timer_list_t *tl, UNUSED fr_time_t now, void *uctx)
libcurl's timer expired
Definition io.c:124
static void _fr_curl_io_service(fr_curl_handle_t *mhandle, int fd, int event)
Service an IO event on a file descriptor.
Definition io.c:151
static int _mhandle_free(fr_curl_handle_t *mhandle)
Free the multi-handle.
Definition io.c:568
#define PERROR(_fmt,...)
Definition log.h:228
#define DEBUG3(_fmt,...)
Definition log.h:266
#define RDEBUG_ENABLED3
True if request debug level 1-3 messages are enabled.
Definition log.h:335
#define RDEBUG3(fmt,...)
Definition log.h:343
#define RHEXDUMP4(_data, _len, _fmt,...)
Definition log.h:706
#define DEBUG4(_fmt,...)
Definition log.h:267
#define RHEXDUMP3(_data, _len, _fmt,...)
Definition log.h:705
#define DEBUG_ENABLED3
True if global debug level 1-3 messages are enabled.
Definition log.h:259
#define RDEBUG_ENABLED4
True if request debug level 1-4 messages are enabled.
Definition log.h:336
talloc_free(reap)
int fr_event_fd_delete(fr_event_list_t *el, int fd, fr_event_filter_t filter)
Remove a file descriptor from the event loop.
Definition event.c:1206
Stores all information relating to an event list.
Definition event.c:380
unsigned char uint8_t
#define fr_assert(_expr)
Definition rad_assert.h:38
#define REDEBUG(fmt,...)
Definition radclient.h:52
#define REQUEST_VERIFY(_x)
Definition request.h:276
fr_aka_sim_id_type_t type
char const * fr_syserror(int num)
Guaranteed to be thread-safe version of strerror.
Definition syserror.c:243
static fr_time_delta_t fr_time_delta_from_msec(int64_t msec)
Definition time.h:575
"server local" time.
Definition time.h:69
int fr_timer_delete(fr_timer_t **ev_p)
Delete a timer event and free its memory.
Definition timer.c:643
An event timer list.
Definition timer.c:53
#define fr_timer_in(...)
Definition timer.h:86
static fr_event_list_t * el
static fr_slen_t data
Definition value.h:1274
#define fr_box_strvalue_len(_val, _len)
Definition value.h:297