The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
proto_detail_file.c
Go to the documentation of this file.
1/*
2 * This program 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
5 * (at 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: de853b9393cc7bc0bf45ac4b6e46ceb3b8ca495c $
19 * @file proto_detail_file.c
20 * @brief Detail handler for files
21 *
22 * @copyright 2017 The FreeRADIUS server project.
23 * @copyright 2017 Alan DeKok (aland@deployingradius.com)
24 */
25
26#include <freeradius-devel/io/application.h>
27#include <freeradius-devel/io/listen.h>
28#include <freeradius-devel/io/schedule.h>
29
30#include <freeradius-devel/server/main_config.h>
31#include <freeradius-devel/server/protocol.h>
32
33#include <freeradius-devel/util/syserror.h>
34#include <freeradius-devel/util/misc.h>
35
36#include "proto_detail.h"
37
38#include <netdb.h>
39
40#include <fcntl.h>
41#include <sys/stat.h>
42
43#ifdef HAVE_GLOB_H
44#include <glob.h>
45#else
46#error proto_detail_file requires <glob.h>
47#endif
48
49DIAG_OFF(unused-macros)
50#if 0
51/*
52 * When we want detailed debugging here, without detailed server
53 * debugging.
54 */
55#define MPRINT DEBUG
56#else
57#define MPRINT DEBUG3
58#endif
59DIAG_ON(unused-macros)
60
61/*
62 * For talloc names, ".name = detail_file", and dl.c prepends "proto_", and appends "_t".
63 */
65
66/*
67 * @todo - this should really now be a different data structure
68 */
70
71static void work_init(proto_detail_file_thread_t *thread, bool triggered_by_delete);
72static void mod_vnode_delete(fr_event_list_t *el, int fd, UNUSED int fflags, void *ctx);
73
76
77 { FR_CONF_OFFSET("filename_work", proto_detail_file_t, filename_work ) },
78
79 { FR_CONF_OFFSET("poll_interval", proto_detail_file_t, poll_interval), .dflt = "5" },
80
81 { FR_CONF_OFFSET("immediate", proto_detail_file_t, immediate) },
82
84};
85
86
87/*
88 * All of the decoding is done by proto_detail and proto_detail_work
89 */
90static int mod_decode(void const *instance, request_t *request, uint8_t *const data, size_t data_len)
91{
93
94 return inst->parent->work_io->decode(inst->parent->work_io_instance, request, data, data_len);
95}
96
97#if 0
98static ssize_t mod_write(UNUSED fr_listen_t *li, UNUSED void *packet_ctx, UNUSED fr_time_t request_time,
99 UNUSED uint8_t *buffer, UNUSED size_t buffer_len, UNUSED size_t written)
100{
101#if 1
102 fr_assert(0);
103
104 return -1;
105#else
106 proto_detail_file_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_detail_file_thread_t);
107
108 return thread->listen->app_io->write(thread->listen, packet_ctx, request_time, buffer, buffer_len, written);
109#endif
110}
111#endif
112
114{
115 proto_detail_file_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_detail_file_thread_t);
116
117 bool has_worker = false;
118
119 pthread_mutex_lock(&thread->worker_mutex);
120 has_worker = (thread->num_workers != 0);
121 pthread_mutex_unlock(&thread->worker_mutex);
122
123 if (has_worker) return;
124
125 FR_TIMER_DISARM(thread->ev);
126
127 work_init(thread, false);
128}
129
130/** Open a detail listener
131 *
132 */
133static int mod_open(fr_listen_t *li)
134{
136 proto_detail_file_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_detail_file_thread_t);
137
138 if (inst->poll_interval == 0) {
139 int oflag;
140
141#ifdef O_EVTONLY
142 oflag = O_EVTONLY;
143#else
144 oflag = O_RDONLY;
145#endif
146 li->fd = thread->fd = open(inst->directory, oflag);
147 if (thread->fd < 0) {
148 cf_log_err(inst->cs, "Failed opening %s: %s", inst->directory, fr_syserror(errno));
149 return -1;
150 }
151 } else {
152 li->fd = thread->fd = -1;
153 li->non_socket_listener = true;
154 li->needs_full_setup = true;
155 }
156 thread->inst = inst;
157 thread->name = talloc_typed_asprintf(thread, "detail_file which will read files matching %s", inst->filename);
158 thread->vnode_fd = -1;
159 pthread_mutex_init(&thread->worker_mutex, NULL);
160
161 li->no_write_callback = true;
162
163 return 0;
164}
165
166/*
167 * The "detail.work" file doesn't exist. Let's see if we can rename one.
168 */
170{
171 proto_detail_file_t const *inst = thread->inst;
172 unsigned int i;
173 int found;
174 time_t chtime;
175 char const *filename;
176 glob_t files;
177 struct stat st;
178
179 DEBUG3("proto_detail (%s): polling for detail files in %s",
180 thread->name, inst->directory);
181
182 memset(&files, 0, sizeof(files));
183 if (glob(inst->filename, 0, NULL, &files) != 0) {
184 noop:
185 DEBUG3("proto_detail (%s): no matching files for %s",
186 thread->name, inst->filename);
187 globfree(&files);
188 return -1;
189 }
190
191 /*
192 * Loop over the glob'd files, looking for the
193 * oldest one.
194 */
195 chtime = 0;
196 found = -1;
197 for (i = 0; i < files.gl_pathc; i++) {
198 if (stat(files.gl_pathv[i], &st) < 0) continue;
199
200 if ((i == 0) || (st.st_ctime < chtime)) {
201 chtime = st.st_ctime;
202 found = i;
203 }
204 }
205
206 /*
207 * No matching files, reset the timer and continue.
208 */
209 if (found < 0) goto noop;
210
211 /*
212 * Rename detail to detail.work
213 */
214 filename = files.gl_pathv[found];
215
216 DEBUG("proto_detail (%s): Renaming %s -> %s", thread->name, filename, inst->filename_work);
217 if (rename(filename, inst->filename_work) < 0) {
218 ERROR("detail (%s): Failed renaming %s to %s: %s",
219 thread->name, filename, inst->filename_work, fr_syserror(errno));
220 goto noop;
221 }
222
223 globfree(&files); /* Shouldn't be using anything in files now */
224
225 /*
226 * The file should now exist, return the open'd FD.
227 */
228 return open(inst->filename_work, inst->mode);
229}
230
231/*
232 * Start polling again after a timeout.
233 */
234static void work_retry_timer(UNUSED fr_timer_list_t *tl, UNUSED fr_time_t now, void *uctx)
235{
236 proto_detail_file_thread_t *thread = talloc_get_type_abort(uctx, proto_detail_file_thread_t);
237
238 work_init(thread, false);
239}
240
241/*
242 * The "detail.work" file exists, and is open in the 'fd'.
243 */
244static int work_exists(proto_detail_file_thread_t *thread, int fd)
245{
246 proto_detail_file_t const *inst = thread->inst;
247 bool opened = false;
249 fr_listen_t *li = NULL;
250 struct stat st;
251
253
254 DEBUG3("proto_detail (%s): Trying to lock %s", thread->name, inst->filename_work);
255
256 /*
257 * "detail.work" exists, try to lock it.
258 */
259 if (rad_lockfd_nonblock(fd, 0) < 0) {
260 fr_time_delta_t delay;
261
262 DEBUG3("proto_detail (%s): Failed locking %s: %s",
263 thread->name, inst->filename_work, fr_syserror(errno));
264
265 close(fd);
266
267 delay = thread->lock_interval;
268
269 /*
270 * Set the next interval, and ensure that we
271 * don't do massive busy-polling.
272 */
278 }
279
280 DEBUG3("proto_detail (%s): Waiting %.6fs for lock on file %s",
281 thread->name, fr_time_delta_unwrap(delay) / (double)NSEC, inst->filename_work);
282
283 if (fr_timer_in(thread, thread->el->tl, &thread->ev, delay,
284 false, work_retry_timer, thread) < 0) {
285 ERROR("Failed inserting retry timer for %s", inst->filename_work);
286 }
287 return 0;
288 }
289
290 DEBUG3("proto_detail (%s): Obtained lock and starting to process file %s",
291 thread->name, inst->filename_work);
292
293 /*
294 * Ignore empty files.
295 */
296 if (fstat(fd, &st) < 0) {
297 ERROR("Failed opening %s: %s", inst->filename_work,
298 fr_syserror(errno));
299 unlink(inst->filename_work);
300 close(fd);
301 return 1;
302 }
303
304 if (!st.st_size) {
305 DEBUG3("proto_detail (%s): %s file is empty, ignoring it.",
306 thread->name, inst->filename_work);
307 unlink(inst->filename_work);
308 close(fd);
309 return 1;
310 }
311
312 /*
313 * This listener is allocated in a thread-specific
314 * context, so it doesn't need a destructor,
315 */
316 MEM(li = talloc_zero(NULL, fr_listen_t));
317
318 /*
319 * Create a new listener, and insert it into the
320 * scheduler. Shamelessly copied from proto_detail.c
321 * mod_open(), with changes.
322 *
323 * This listener is parented from the worker. So that
324 * when the worker goes away, so does the listener.
325 */
326 li->cs = inst->parent->work_io_conf;
327 li->app_io = inst->parent->work_io;
328
329 li->app = inst->parent->self;
330 li->app_instance = inst->parent;
331 li->server_cs = inst->parent->server_cs;
332
333 /*
334 * The worker may be in a different thread, so avoid
335 * talloc threading issues by using a NULL TALLOC_CTX.
336 */
337 MEM(li->thread_instance = work = talloc_zero(li, proto_detail_work_thread_t));
338
339 li->app_io_instance = inst->parent->work_io_instance;
340 work->inst = li->app_io_instance;
341 work->file_parent = thread;
342 work->ev = NULL;
343
344 li->fd = work->fd = dup(fd);
345 if (work->fd < 0) {
346 DEBUG("proto_detail (%s): Failed opening %s: %s",
347 thread->name, inst->filename_work, fr_syserror(errno));
348
349 close(fd);
350 talloc_free(li);
351 return -1;
352 }
353
354 /*
355 * Don't do anything until the file has been deleted.
356 *
357 * @todo - ensure that proto_detail_work is done the file...
358 * maybe by creating a new instance?
359 */
360 if (fr_event_filter_insert(thread, NULL, thread->el, fd, FR_EVENT_FILTER_VNODE,
361 &funcs, NULL, thread) < 0) {
362 PERROR("Failed adding work socket to event loop");
363 close(li->fd);
364 close(fd);
365 talloc_free(li);
366 return -1;
367 }
368
369 /*
370 * Remember this for later.
371 */
372 thread->vnode_fd = fd;
373
374 /*
375 * For us, this is the worker listener.
376 * For the worker, this is it's own parent
377 */
378 thread->listen = li;
379
380 work->filename_work = talloc_strdup(work, inst->filename_work);
381
382 /*
383 * Set configurable parameters for message ring buffer.
384 */
385 li->default_message_size = inst->parent->max_packet_size;
386 li->num_messages = inst->parent->num_messages;
387
388 /*
389 * Open the detail.work file.
390 */
391 if (li->app_io->open(li) < 0) {
392 ERROR("Failed opening %s", li->app_io->common.name);
393 goto error;
394 }
395 opened = true;
396
398 li->name = li->app_io->get_name(li);
399
400 if (!fr_schedule_listen_add(inst->parent->sc, li)) {
401 error:
402 if (fr_event_fd_delete(thread->el, thread->vnode_fd, FR_EVENT_FILTER_VNODE) < 0) {
403 PERROR("Failed removing DELETE callback when opening work file");
404 }
405 close(thread->vnode_fd);
406 thread->vnode_fd = -1;
407
408 if (opened) {
409 (void) li->app_io->close(li);
410 thread->listen = NULL;
411 li = NULL;
412 } else {
413 close(li->fd);
414 }
415
416 talloc_free(li);
417 return -1;
418 }
419
420 /*
421 * Tell the worker to clean itself up.
422 */
423 work->listen = li;
424
425 pthread_mutex_lock(&thread->worker_mutex);
426 thread->num_workers++;
427 pthread_mutex_unlock(&thread->worker_mutex);
428
429 return 0;
430}
431
432
433static void mod_vnode_delete(fr_event_list_t *el, int fd, UNUSED int fflags, void *ctx)
434{
435 proto_detail_file_thread_t *thread = talloc_get_type_abort(ctx, proto_detail_file_thread_t);
436 proto_detail_file_t const *inst = thread->inst;
437
438 DEBUG("proto_detail (%s): Deleted %s", thread->name, inst->filename_work);
439
440 /*
441 * Silently ignore notifications from the directory. We
442 * didn't ask for them, but libkqueue delivers them to
443 * us.
444 */
445 if (fd == thread->fd) return;
446
447 if (fd != thread->vnode_fd) {
448 ERROR("Received DELETE for FD %d, when we were expecting one on FD %d - ignoring it",
449 fd, thread->vnode_fd);
450 return;
451 }
452
454 PERROR("Failed removing DELETE callback after deletion");
455 }
456 close(fd);
457 thread->vnode_fd = -1;
458
459 /*
460 * Re-initialize the state machine.
461 *
462 * Note that a "delete" may be the result of an atomic
463 * "move", which both deletes the old file, and creates
464 * the new one.
465 */
466 work_init(thread, true);
467}
468
469
470/** Start processing a new work file
471 *
472 * @param[in] thread the thread instance.
473 * @param[in] triggered_by_delete true if this was triggered by a vnode_delete.
474 * When a new file is moved over a workfile
475 * vnode delete can serve as an indication
476 * that new data is available.
477 * It can also mean however, that the file
478 * has just been deleted, so we shouldn't
479 * treat this failure to open the new file
480 * as a fatal error.
481 */
482static void work_init(proto_detail_file_thread_t *thread, bool triggered_by_delete)
483{
484 proto_detail_file_t const *inst = thread->inst;
485 int fd, rcode;
486 bool has_worker;
487
488 pthread_mutex_lock(&thread->worker_mutex);
489 has_worker = (thread->num_workers != 0);
490 pthread_mutex_unlock(&thread->worker_mutex);
491
492 /*
493 * The worker is still processing the file, poll until
494 * it's done.
495 */
496 if (has_worker) {
497 DEBUG3("proto_detail (%s): worker %s is still alive, waiting for it to finish.",
498 thread->name, inst->filename_work);
499 goto delay;
500 }
501
502 fr_assert(thread->vnode_fd < 0);
503
504 /*
505 * See if there is a "detail.work" file. If not, try to
506 * rename an existing file to "detail.work".
507 */
508 DEBUG3("Trying to open %s", inst->filename_work);
509 fd = open(inst->filename_work, inst->mode);
510
511 /*
512 * If the work file didn't exist, try to rename detail* ->
513 * detail.work, and return the newly opened file.
514 */
515 if (fd < 0) {
516 if (errno != ENOENT) {
517 DEBUG("proto_detail (%s): Failed opening %s: %s",
518 thread->name, inst->filename_work,
519 fr_syserror(errno));
520 goto delay;
521 }
522
523retry:
524 fd = work_rename(thread);
525 }
526
527 /*
528 * The work file still doesn't exist. Go set up timers,
529 * or wait for an event which signals us that something
530 * in the directory changed.
531 */
532 if (fd < 0) {
533 /*
534 * Wait for the directory to change before
535 * looking for another "detail" file.
536 */
537 if (!inst->poll_interval) return;
538
539 /*
540 * File really was deleted, and didn't have
541 * anything moved over the top.
542 */
543 if (inst->immediate && triggered_by_delete) return;
544
545delay:
546 /*
547 * If we're processing one file and exiting, then the input file must exist.
548 */
549 if (inst->immediate && inst->parent->exit_when_done) {
550 ERROR("Input file does not exist");
551 fr_exit(EXIT_FAILURE);
552 }
553
554 /*
555 * Check every N seconds.
556 */
557 DEBUG3("Waiting %d.000000s for new files in %s", inst->poll_interval, thread->name);
558
559 if (fr_timer_in(thread, thread->el->tl, &thread->ev,
560 fr_time_delta_from_sec(inst->poll_interval),
561 false, work_retry_timer, thread) < 0) {
562 ERROR("Failed inserting poll timer for %s", inst->filename_work);
563 }
564 return;
565 }
566
568
569 /*
570 * It exists, go process it!
571 *
572 * We will get back to the main loop when the
573 * "detail.work" file is deleted.
574 */
575 rcode = work_exists(thread, fd);
576 if (rcode < 0) goto delay;
577
578 /*
579 * The file was empty, so we try to get another one.
580 */
581 if (rcode == 1) goto retry;
582
583 /*
584 * Otherwise the child is successfully processing the
585 * file.
586 */
587}
588
589
590/** Set the event list for a new IO instance
591 *
592 * @param[in] li the listener
593 * @param[in] el the event list
594 * @param[in] nr context from the network side
595 */
597{
599 proto_detail_file_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_detail_file_thread_t);
600
601 thread->el = el;
602
603 if (inst->immediate) {
604 work_init(thread, false);
605 return;
606 }
607
608 /*
609 * Delay for a bit, before reading the detail files.
610 * This gives the server time to call
611 * rad_suid_down_permanent(), and for /proc/PID to
612 * therefore change permissions, so that libkqueue can
613 * read it.
614 */
615 if (fr_timer_in(thread, thread->el->tl, &thread->ev,
616 fr_time_delta_from_sec(1), false, work_retry_timer, thread) < 0) {
617 ERROR("Failed inserting poll timer for %s", inst->filename_work);
618 }
619}
620
621
622static char const *mod_name(fr_listen_t *li)
623{
624 proto_detail_file_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_detail_file_thread_t);
625
626 return thread->name;
627}
628
629
631static pthread_mutex_t detail_file_mutex = PTHREAD_MUTEX_INITIALIZER;
632
633/** Compare two thread instances based on node pointer
634 *
635 * @param[in] one First thread specific xlat expansion instance.
636 * @param[in] two Second thread specific xlat expansion instance.
637 * @return CMP(one, two)
638 */
639static int8_t _detail_file_cmp(void const *one, void const *two)
640{
641 proto_detail_file_t const *a = one, *b = two;
642
643 return CMP(strcmp(a->filename, b->filename), 0);
644}
645
646/*
647 * Check for multiple readers on the same set of files.
648 */
649static int mod_instantiate(module_inst_ctx_t const *mctx)
650{
651 proto_detail_file_t *inst = talloc_get_type_abort(mctx->mi->data, proto_detail_file_t);
652 CONF_SECTION *conf = mctx->mi->conf;
653
654 module_instance_t const *mi;
655 char *p;
656
657#ifdef __linux__
658 /*
659 * The kqueue API takes an FD, but inotify requires a filename.
660 * libkqueue uses /proc/PID/fd/# to look up the FD -> filename mapping.
661 *
662 * However, if you start the server as "root", and then swap to "radiusd",
663 * /proc/PID will be owned by "root" for security reasons. The only way
664 * to make /proc/PID owned by "radiusd" is to set the DUMPABLE flag.
665 *
666 * Instead of making the poor sysadmin figure this out,
667 * we check for this situation, and give them a
668 * descriptive message telling them what to do.
669 */
671 main_config->uid_is_set &&
672 main_config->server_uid != 0) {
673 cf_log_err(conf, "Cannot start detail file reader due to Linux limitations.");
674 cf_log_err(conf, "Please set 'allow_core_dumps = true' in the main configuration file.");
675 return -1;
676 }
677#endif
678
679 /*
680 * Find the module_instance_t holding our instance data
681 * so we can find out what the parent of our instance
682 * was.
683 */
684 mi = mctx->mi;
685
686#ifndef __linux__
687 /*
688 * Linux inotify works. So we allow poll_interval==0
689 */
690 FR_INTEGER_BOUND_CHECK("poll_interval", inst->poll_interval, >=, 1);
691#endif
692 FR_INTEGER_BOUND_CHECK("poll_interval", inst->poll_interval, <=, 3600);
693
694 inst->parent = talloc_get_type_abort(mi->parent->data, proto_detail_t);
695 inst->cs = conf;
696
697 inst->directory = p = talloc_strdup(inst, inst->filename);
698
699 p = strrchr(p, '/');
700 if (!p) {
701 cf_log_err(conf, "Filename must contain '/'");
702 return -1;
703 }
704
705 *p = '\0';
706
707 if (!inst->filename_work) {
708 inst->filename_work = talloc_typed_asprintf(inst, "%s/detail.work", inst->directory);
709 }
710
711 /*
712 * We need this for the lock.
713 */
714 inst->mode = O_RDWR;
715
716 if (inst->parent->exit_when_done && !inst->immediate) {
717 cf_log_warn(conf, "Ignoring 'exit_when_done' due to 'immediate' flag not being set");
718 inst->parent->exit_when_done = false;
719 }
720
721 pthread_mutex_lock(&detail_file_mutex);
722 if (!detail_file_tree) {
724 if (!detail_file_tree) {
725 pthread_mutex_unlock(&detail_file_mutex);
726 cf_log_err(conf, "Failed initializing detail_file");
727 return -1;
728 }
729 }
730
732 proto_detail_file_t const *old;
733
735 fr_assert(old);
736
737 pthread_mutex_unlock(&detail_file_mutex);
738 cf_log_err(conf, "Duplicate detail listener %s", inst->filename);
739 cf_log_err(conf, "Original detail listener is in virtual server %s", cf_section_name2(old->parent->server_cs));
740 return -1;
741 }
742 pthread_mutex_unlock(&detail_file_mutex);
743
744 return 0;
745}
746
747
748static int mod_close(fr_listen_t *li)
749{
751 proto_detail_file_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_detail_file_thread_t);
752
753 if (thread->nr) {
754 if (unlikely(fr_network_listen_delete(thread->nr, inst->parent->listen) < 0)) {
755 PERROR("Failed removing listener from network on detach");
756 }
757 }
758
759 /*
760 * @todo - have our OWN event loop for timers, and a
761 * "copy timer from -> to, which means we only have to
762 * delete our child event loop from the parent on close.
763 */
764 if (thread->fd >= 0) close(thread->fd);
765
766 if (thread->vnode_fd >= 0) {
767 if (thread->nr) {
768 if (unlikely(fr_network_listen_delete(thread->nr, inst->parent->listen) < 0)) {
769 PERROR("Failed removing listener from network on detach");
770 }
771 } else {
772 if (fr_event_fd_delete(thread->el, thread->vnode_fd, FR_EVENT_FILTER_VNODE) < 0) {
773 PERROR("Failed removing DELETE callback on detach");
774 }
775 }
776 close(thread->vnode_fd);
777 thread->vnode_fd = -1;
778 }
779
780 pthread_mutex_destroy(&thread->worker_mutex);
781
782 return 0;
783}
784
785
786/** Private interface for use by proto_detail_file
787 *
788 */
791 .common = {
792 .magic = MODULE_MAGIC_INIT,
793 .name = "detail_file",
795 .inst_size = sizeof(proto_detail_file_t),
796 .thread_inst_size = sizeof(proto_detail_file_thread_t),
797 .instantiate = mod_instantiate,
798 },
799 .default_message_size = 65536,
800 .default_reply_size = 32,
801
802 .open = mod_open,
803 .close = mod_close,
804 .vnode = mod_vnode_extend,
805 .decode = mod_decode,
806// .write = mod_write,
807 .event_list_set = mod_event_list_set,
808 .get_name = mod_name,
809};
static int const char char buffer[256]
Definition acutest.h:576
fr_io_close_t close
Close the transport.
Definition app_io.h:60
fr_io_open_t open
Open a new socket for listening, or accept/connect a new connection.
Definition app_io.h:43
module_t common
Common fields to all loadable modules.
Definition app_io.h:34
fr_io_data_write_t write
Write from a data buffer to a socket.
Definition app_io.h:48
fr_io_name_t get_name
get the socket name
Definition app_io.h:70
Public structure describing an I/O path for a protocol.
Definition app_io.h:33
#define DIAG_ON(_x)
Definition build.h:487
#define CMP(_a, _b)
Same as CMP_PREFER_SMALLER use when you don't really care about ordering, you just want an ordering.
Definition build.h:113
#define unlikely(_x)
Definition build.h:407
#define UNUSED
Definition build.h:336
#define DIAG_OFF(_x)
Definition build.h:486
#define CONF_PARSER_TERMINATOR
Definition cf_parse.h:669
#define FR_INTEGER_BOUND_CHECK(_name, _var, _op, _bound)
Definition cf_parse.h:529
#define FR_CONF_OFFSET(_name, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:280
#define FR_CONF_OFFSET_FLAGS(_name, _flags, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:268
@ CONF_FLAG_REQUIRED
Error out if no matching CONF_PAIR is found, and no dflt value is set.
Definition cf_parse.h:429
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:606
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:106
char const * cf_section_name2(CONF_SECTION const *cs)
Return the second identifier of a CONF_SECTION.
Definition cf_util.c:1352
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:345
#define cf_log_warn(_cf, _fmt,...)
Definition cf_util.h:346
#define fr_exit(_x)
Exit, producing a log message in debug builds.
Definition debug.h:220
#define MEM(x)
Definition debug.h:36
#define ERROR(fmt,...)
Definition dhcpclient.c:40
#define DEBUG(fmt,...)
Definition dhcpclient.c:38
#define MODULE_MAGIC_INIT
Stop people using different module/library/server versions together.
Definition dl_module.h:63
@ FR_EVENT_FILTER_VNODE
Filter for vnode subfilters.
Definition event.h:84
#define fr_event_filter_insert(...)
Definition event.h:234
fr_event_fd_cb_t delete
The file was deleted.
Definition event.h:196
Callbacks for the FR_EVENT_FILTER_VNODE filter.
Definition event.h:195
talloc_free(hp)
size_t num_messages
for the message ring buffer
Definition listen.h:57
CONF_SECTION * cs
of this listener
Definition listen.h:41
bool non_socket_listener
special internal listener that does not use sockets.
Definition listen.h:47
char const * name
printable name for this socket - set by open
Definition listen.h:29
void const * app_instance
Definition listen.h:39
size_t default_message_size
copied from app_io, but may be changed
Definition listen.h:56
fr_app_t const * app
Definition listen.h:38
void const * app_io_instance
I/O path configuration context.
Definition listen.h:33
CONF_SECTION * server_cs
CONF_SECTION of the server.
Definition listen.h:42
void * thread_instance
thread / socket context
Definition listen.h:34
bool no_write_callback
sometimes we don't need to do writes
Definition listen.h:46
int fd
file descriptor for this socket - set by open
Definition listen.h:28
bool needs_full_setup
Set to true to avoid the short cut when adding the listener.
Definition listen.h:48
fr_app_io_t const * app_io
I/O path functions.
Definition listen.h:32
int fr_network_listen_delete(fr_network_t *nr, fr_listen_t *li)
Delete a socket from a network.
Definition network.c:266
#define PERROR(_fmt,...)
Definition log.h:228
#define DEBUG3(_fmt,...)
Definition log.h:266
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:1203
Stores all information relating to an event list.
Definition event.c:377
main_config_t const * main_config
Main server configuration.
Definition main_config.c:56
bool allow_core_dumps
Whether the server is allowed to drop a core when receiving a fatal signal.
static ssize_t mod_write(fr_listen_t *li, void *packet_ctx, fr_time_t request_time, uint8_t *buffer, size_t buffer_len, size_t written)
Definition master.c:2615
unsigned int uint32_t
long int ssize_t
unsigned char uint8_t
int rad_lockfd_nonblock(int fd, int lock_len)
Definition misc.c:129
module_instance_t * mi
Instance of the module being instantiated.
Definition module_ctx.h:51
Temporary structure to hold arguments for instantiation calls.
Definition module_ctx.h:50
Detail master protocol handler.
proto_detail_t * parent
The module that spawned us!
proto_detail_work_thread_t * file_parent
thread instance of the directory reader that spawned us
fr_time_delta_t lock_interval
interval between trying the locks.
fr_listen_t * listen
talloc_parent() is slow
char const * filename
file name, usually with wildcards
CONF_SECTION * server_cs
server CS for this listener
char const * name
debug name for printing
int num_workers
number of workers
fr_timer_t * ev
for detail file timers.
fr_event_list_t * el
for various timers
pthread_mutex_t worker_mutex
for the workers
char const * filename_work
work file name
proto_detail_work_t const * inst
instance data
int fd
file descriptor
int vnode_fd
file descriptor for vnode_delete
fr_network_t * nr
for Linux-specific callbacks
static void mod_vnode_extend(fr_listen_t *li, UNUSED uint32_t fflags)
static const conf_parser_t file_listen_config[]
static void work_retry_timer(UNUSED fr_timer_list_t *tl, UNUSED fr_time_t now, void *uctx)
fr_app_io_t proto_detail_file
Private interface for use by proto_detail_file.
struct proto_detail_work_s proto_detail_file_t
static int8_t _detail_file_cmp(void const *one, void const *two)
Compare two thread instances based on node pointer.
static pthread_mutex_t detail_file_mutex
static int mod_decode(void const *instance, request_t *request, uint8_t *const data, size_t data_len)
static int work_rename(proto_detail_file_thread_t *thread)
static int mod_open(fr_listen_t *li)
Open a detail listener.
static fr_rb_tree_t * detail_file_tree
static void mod_event_list_set(fr_listen_t *li, fr_event_list_t *el, UNUSED void *nr)
Set the event list for a new IO instance.
static char const * mod_name(fr_listen_t *li)
static int mod_close(fr_listen_t *li)
static int mod_instantiate(module_inst_ctx_t const *mctx)
static void work_init(proto_detail_file_thread_t *thread, bool triggered_by_delete)
Start processing a new work file.
static void mod_vnode_delete(fr_event_list_t *el, int fd, UNUSED int fflags, void *ctx)
static int work_exists(proto_detail_file_thread_t *thread, int fd)
#define fr_assert(_expr)
Definition rad_assert.h:37
static rs_t * conf
Definition radsniff.c:52
void * fr_rb_find(fr_rb_tree_t const *tree, void const *data)
Find an element in the tree, returning the data, not the node.
Definition rb.c:577
bool fr_rb_insert(fr_rb_tree_t *tree, void const *data)
Insert data into a tree.
Definition rb.c:626
#define fr_rb_inline_talloc_alloc(_ctx, _type, _field, _data_cmp, _data_free)
Allocs a red black that verifies elements are of a specific talloc type.
Definition rb.h:244
The main red black tree structure.
Definition rb.h:71
fr_network_t * fr_schedule_listen_add(fr_schedule_t *sc, fr_listen_t *li)
Add a fr_listen_t to a scheduler.
Definition schedule.c:730
CONF_SECTION * conf
Module's instance configuration.
Definition module.h:351
void * data
Module's instance data.
Definition module.h:293
module_instance_t const * parent
Parent module's instance (if any).
Definition module.h:359
conf_parser_t const * config
How to convert a CONF_SECTION to a module instance.
Definition module.h:206
Module instance data.
Definition module.h:287
eap_aka_sim_process_conf_t * inst
char const * fr_syserror(int num)
Guaranteed to be thread-safe version of strerror.
Definition syserror.c:243
char * talloc_typed_asprintf(TALLOC_CTX *ctx, char const *fmt,...)
Call talloc vasprintf, setting the type on the new chunk correctly.
Definition talloc.c:546
#define talloc_get_type_abort_const
Definition talloc.h:117
#define talloc_strdup(_ctx, _str)
Definition talloc.h:149
static fr_time_delta_t fr_time_delta_from_msec(int64_t msec)
Definition time.h:575
static fr_time_delta_t fr_time_delta_add(fr_time_delta_t a, fr_time_delta_t b)
Definition time.h:255
static int64_t fr_time_delta_unwrap(fr_time_delta_t time)
Definition time.h:154
static fr_time_delta_t fr_time_delta_from_sec(int64_t sec)
Definition time.h:590
#define fr_time_delta_wrap(_time)
Definition time.h:152
#define NSEC
Definition time.h:379
static fr_time_delta_t fr_time_delta_div(fr_time_delta_t a, fr_time_delta_t b)
Definition time.h:267
#define fr_time_delta_gt(_a, _b)
Definition time.h:283
A time delta, a difference in time measured in nanoseconds.
Definition time.h:80
"server local" time.
Definition time.h:69
An event timer list.
Definition timer.c:49
#define fr_timer_in(...)
Definition timer.h:87
#define FR_TIMER_DISARM(_ev)
Definition timer.h:91
static fr_event_list_t * el
static fr_slen_t data
Definition value.h:1340