Blame lib/scheduler.c

Packit Service 5956c7
/*
Packit Service 5956c7
 * Soft:        Keepalived is a failover program for the LVS project
Packit Service 5956c7
 *              <www.linuxvirtualserver.org>. It monitor & manipulate
Packit Service 5956c7
 *              a loadbalanced server pool using multi-layer checks.
Packit Service 5956c7
 *
Packit Service 5956c7
 * Part:        Scheduling framework. This code is highly inspired from
Packit Service 5956c7
 *              the thread management routine (thread.c) present in the
Packit Service 5956c7
 *              very nice zebra project (http://www.zebra.org).
Packit Service 5956c7
 *
Packit Service 5956c7
 * Author:      Alexandre Cassen, <acassen@linux-vs.org>
Packit Service 5956c7
 *
Packit Service 5956c7
 *              This program is distributed in the hope that it will be useful,
Packit Service 5956c7
 *              but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 5956c7
 *              MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
Packit Service 5956c7
 *              See the GNU General Public License for more details.
Packit Service 5956c7
 *
Packit Service 5956c7
 *              This program is free software; you can redistribute it and/or
Packit Service 5956c7
 *              modify it under the terms of the GNU General Public License
Packit Service 5956c7
 *              as published by the Free Software Foundation; either version
Packit Service 5956c7
 *              2 of the License, or (at your option) any later version.
Packit Service 5956c7
 *
Packit Service 5956c7
 * Copyright (C) 2001-2017 Alexandre Cassen, <acassen@gmail.com>
Packit Service 5956c7
 */
Packit Service 5956c7
Packit Service 5956c7
#include "config.h"
Packit Service 5956c7
Packit Service 5956c7
/* SNMP should be included first: it redefines "FREE" */
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
#include <net-snmp/net-snmp-config.h>
Packit Service 5956c7
#include <net-snmp/net-snmp-includes.h>
Packit Service 5956c7
#include <net-snmp/agent/net-snmp-agent-includes.h>
Packit Service 5956c7
#undef FREE
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
#ifndef _DEBUG_
Packit Service 5956c7
#define NDEBUG
Packit Service 5956c7
#endif
Packit Service 5956c7
#include <assert.h>
Packit Service 5956c7
#include <errno.h>
Packit Service 5956c7
#include <sys/wait.h>
Packit Service 5956c7
#include <sys/timerfd.h>
Packit Service 5956c7
#include <sys/epoll.h>
Packit Service 5956c7
#include <unistd.h>
Packit Service 5956c7
#ifdef HAVE_SIGNALFD
Packit Service 5956c7
#include <sys/signalfd.h>
Packit Service 5956c7
#endif
Packit Service 5956c7
#include <sys/utsname.h>
Packit Service 5956c7
#include <linux/version.h>
Packit Service 5956c7
Packit Service 5956c7
#include "scheduler.h"
Packit Service 5956c7
#include "memory.h"
Packit Service 5956c7
#include "rbtree.h"
Packit Service 5956c7
#include "utils.h"
Packit Service 5956c7
#include "signals.h"
Packit Service 5956c7
#include "logger.h"
Packit Service 5956c7
#include "bitops.h"
Packit Service 5956c7
#include "git-commit.h"
Packit Service 5956c7
#include "timer.h"
Packit Service 5956c7
#if !HAVE_EPOLL_CREATE1 || !defined TFD_NONBLOCK
Packit Service 5956c7
#include "old_socket.h"
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
Packit Service 5956c7
#ifdef THREAD_DUMP
Packit Service 5956c7
typedef struct _func_det {
Packit Service 5956c7
	const char *name;
Packit Service 5956c7
	int (*func)(thread_t *);
Packit Service 5956c7
	rb_node_t n;
Packit Service 5956c7
} func_det_t;
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* global vars */
Packit Service 5956c7
thread_master_t *master = NULL;
Packit Service 5956c7
#ifndef _DEBUG_
Packit Service 5956c7
prog_type_t prog_type;		/* Parent/VRRP/Checker process */
Packit Service 5956c7
#endif
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
bool snmp_running;		/* True if this process is running SNMP */
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* local variables */
Packit Service 5956c7
static bool shutting_down;
Packit Service 5956c7
static int sav_argc;
Packit Service 5956c7
static char **sav_argv;
Packit Service 5956c7
#ifdef _EPOLL_DEBUG_
Packit Service 5956c7
bool do_epoll_debug;
Packit Service 5956c7
#endif
Packit Service 5956c7
#ifdef _EPOLL_THREAD_DUMP_
Packit Service 5956c7
bool do_epoll_thread_dump;
Packit Service 5956c7
#endif
Packit Service 5956c7
#ifdef THREAD_DUMP
Packit Service 5956c7
static rb_root_t funcs = RB_ROOT;
Packit Service 5956c7
#endif
Packit Service 5956c7
#ifdef _VRRP_FD_DEBUG_
Packit Service 5956c7
static void (*extra_threads_debug)(void);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
Packit Service 5956c7
/* Function that returns prog_name if pid is a known child */
Packit Service 5956c7
static char const * (*child_finder_name)(pid_t);
Packit Service 5956c7
Packit Service 5956c7
#ifdef THREAD_DUMP
Packit Service 5956c7
static const char *
Packit Service 5956c7
get_thread_type_str(thread_type_t id)
Packit Service 5956c7
{
Packit Service 5956c7
	if (id == THREAD_READ) return "READ";
Packit Service 5956c7
	if (id == THREAD_WRITE) return "WRITE";
Packit Service 5956c7
	if (id == THREAD_TIMER) return "TIMER";
Packit Service 5956c7
	if (id == THREAD_TIMER_SHUTDOWN) return "TIMER_SHUTDOWN";
Packit Service 5956c7
	if (id == THREAD_EVENT) return "EVENT";
Packit Service 5956c7
	if (id == THREAD_CHILD) return "CHILD";
Packit Service 5956c7
	if (id == THREAD_READY) return "READY";
Packit Service 5956c7
	if (id == THREAD_UNUSED) return "UNUSED";
Packit Service 5956c7
	if (id == THREAD_WRITE_TIMEOUT) return "WRITE_TIMEOUT";
Packit Service 5956c7
	if (id == THREAD_READ_TIMEOUT) return "READ_TIMEOUT";
Packit Service 5956c7
	if (id == THREAD_CHILD_TIMEOUT) return "CHILD_TIMEOUT";
Packit Service 5956c7
	if (id == THREAD_CHILD_TERMINATED) return "CHILD_TERMINATED";
Packit Service 5956c7
	if (id == THREAD_TERMINATE_START) return "TERMINATE_START";
Packit Service 5956c7
	if (id == THREAD_TERMINATE) return "TERMINATE";
Packit Service 5956c7
	if (id == THREAD_READY_FD) return "READY_FD";
Packit Service 5956c7
	if (id == THREAD_READ_ERROR) return "READ_ERROR";
Packit Service 5956c7
	if (id == THREAD_WRITE_ERROR) return "WRITE_ERROR";
Packit Service 5956c7
#ifdef USE_SIGNAL_THREADS
Packit Service 5956c7
	if (id == THREAD_SIGNAL) return "SIGNAL";
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	return "unknown";
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static inline int
Packit Service 5956c7
function_cmp(const func_det_t *func1, const func_det_t *func2)
Packit Service 5956c7
{
Packit Service 5956c7
	if (func1->func < func2->func)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
	if (func1->func > func2->func)
Packit Service 5956c7
		return 1;
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static const char *
Packit Service 5956c7
get_function_name(int (*func)(thread_t *))
Packit Service 5956c7
{
Packit Service 5956c7
	func_det_t func_det = { .func = func };
Packit Service 5956c7
	func_det_t *match;
Packit Service 5956c7
	static char address[19];
Packit Service 5956c7
Packit Service 5956c7
	if (!RB_EMPTY_ROOT(&funcs)) {
Packit Service 5956c7
		match = rb_search(&funcs, &func_det, n, function_cmp);
Packit Service 5956c7
		if (match)
Packit Service 5956c7
			return match->name;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	snprintf(address, sizeof address, "%p", func);
Packit Service 5956c7
	return address;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
const char *
Packit Service 5956c7
get_signal_function_name(void (*func)(void *, int))
Packit Service 5956c7
{
Packit Service 5956c7
	/* The cast should really be (int (*)(thread_t *))func, but gcc 8.1 produces
Packit Service 5956c7
	 * a warning with -Wcast-function-type, that the cast is to an incompatible
Packit Service 5956c7
	 * function type. Since we don't actually call the function, but merely use
Packit Service 5956c7
	 * it to compare function addresses, what we cast it do doesn't really matter */
Packit Service 5956c7
	return get_function_name((void *)func);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
register_thread_address(const char *func_name, int (*func)(thread_t *))
Packit Service 5956c7
{
Packit Service 5956c7
	func_det_t *func_det;
Packit Service 5956c7
Packit Service 5956c7
	func_det = (func_det_t *) MALLOC(sizeof(func_det_t));
Packit Service 5956c7
	if (!func_det)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	func_det->name = func_name;
Packit Service 5956c7
	func_det->func = func;
Packit Service 5956c7
Packit Service 5956c7
	rb_insert_sort(&funcs, func_det, n, function_cmp);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
register_signal_handler_address(const char *func_name, void (*func)(void *, int))
Packit Service 5956c7
{
Packit Service 5956c7
	/* See comment in get_signal_function_name() above */
Packit Service 5956c7
	register_thread_address(func_name, (void *)func);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
deregister_thread_addresses(void)
Packit Service 5956c7
{
Packit Service 5956c7
	func_det_t *func_det, *func_det_tmp;
Packit Service 5956c7
Packit Service 5956c7
	if (RB_EMPTY_ROOT(&funcs))
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	rb_for_each_entry_safe(func_det, func_det_tmp, &funcs, n) {
Packit Service 5956c7
		rb_erase(&func_det->n, &funcs);
Packit Service 5956c7
		FREE(func_det);
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
#ifdef _VRRP_FD_DEBUG_
Packit Service 5956c7
void
Packit Service 5956c7
set_extra_threads_debug(void (*func)(void))
Packit Service 5956c7
{
Packit Service 5956c7
	extra_threads_debug = func;
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* Move ready thread into ready queue */
Packit Service 5956c7
static int
Packit Service 5956c7
thread_move_ready(thread_master_t *m, rb_root_cached_t *root, thread_t *thread, int type)
Packit Service 5956c7
{
Packit Service 5956c7
	rb_erase_cached(&thread->n, root);
Packit Service 5956c7
	INIT_LIST_HEAD(&thread->next);
Packit Service 5956c7
	list_add_tail(&thread->next, &m->ready);
Packit Service 5956c7
	if (thread->type != THREAD_TIMER_SHUTDOWN)
Packit Service 5956c7
		thread->type = type;
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Move ready thread into ready queue */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_rb_move_ready(thread_master_t *m, rb_root_cached_t *root, int type)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread, *thread_tmp;
Packit Service 5956c7
Packit Service 5956c7
	rb_for_each_entry_safe_cached(thread, thread_tmp, root, n) {
Packit Service 5956c7
		if (thread->sands.tv_sec == TIMER_DISABLED || timercmp(&time_now, &thread->sands, <))
Packit Service 5956c7
			break;
Packit Service 5956c7
Packit Service 5956c7
		if (type == THREAD_READ_TIMEOUT)
Packit Service 5956c7
			thread->event->read = NULL;
Packit Service 5956c7
		else if (type == THREAD_WRITE_TIMEOUT)
Packit Service 5956c7
			thread->event->write = NULL;
Packit Service 5956c7
		thread_move_ready(m, root, thread, type);
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Update timer value */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_update_timer(rb_root_cached_t *root, timeval_t *timer_min)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *first;
Packit Service 5956c7
Packit Service 5956c7
	if (!root->rb_root.rb_node)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	first = rb_entry(rb_first_cached(root), thread_t, n);
Packit Service 5956c7
	if (!first)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	if (first->sands.tv_sec == TIMER_DISABLED)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	if (!timerisset(timer_min) ||
Packit Service 5956c7
	    timercmp(&first->sands, timer_min, <=))
Packit Service 5956c7
		*timer_min = first->sands;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Compute the wait timer. Take care of timeouted fd */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_set_timer(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	timeval_t timer_wait;
Packit Service 5956c7
	struct itimerspec its;
Packit Service 5956c7
Packit Service 5956c7
	/* Prepare timer */
Packit Service 5956c7
	timerclear(&timer_wait);
Packit Service 5956c7
	thread_update_timer(&m->timer, &timer_wait);
Packit Service 5956c7
	thread_update_timer(&m->write, &timer_wait);
Packit Service 5956c7
	thread_update_timer(&m->read, &timer_wait);
Packit Service 5956c7
	thread_update_timer(&m->child, &timer_wait);
Packit Service 5956c7
Packit Service 5956c7
	if (timerisset(&timer_wait)) {
Packit Service 5956c7
		/* Re-read the current time to get the maximum accuracy */
Packit Service 5956c7
		set_time_now();
Packit Service 5956c7
Packit Service 5956c7
		/* Take care about monotonic clock */
Packit Service 5956c7
		timersub(&timer_wait, &time_now, &timer_wait);
Packit Service 5956c7
Packit Service 5956c7
		if (timer_wait.tv_sec < 0) {
Packit Service 5956c7
			/* This will disable the timerfd */
Packit Service 5956c7
			timerclear(&timer_wait);
Packit Service 5956c7
		}
Packit Service 5956c7
	} else {
Packit Service 5956c7
		/* set timer to a VERY long time */
Packit Service 5956c7
		timer_wait.tv_sec = LONG_MAX;
Packit Service 5956c7
		timer_wait.tv_usec = 0;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	its.it_value.tv_sec = timer_wait.tv_sec;
Packit Service 5956c7
	if (!timerisset(&timer_wait)) {
Packit Service 5956c7
		/* We could try to avoid doing the epoll_wait since
Packit Service 5956c7
		 * testing shows it takes about 4 microseconds
Packit Service 5956c7
		 * for the timer to expire. */
Packit Service 5956c7
		its.it_value.tv_nsec = 1;
Packit Service 5956c7
	}
Packit Service 5956c7
	else
Packit Service 5956c7
		its.it_value.tv_nsec = timer_wait.tv_usec * 1000;
Packit Service 5956c7
Packit Service 5956c7
	/* We don't want periodic timer expiry */
Packit Service 5956c7
	its.it_interval.tv_sec = its.it_interval.tv_nsec = 0;
Packit Service 5956c7
Packit Service 5956c7
	timerfd_settime(m->timer_fd, 0, &its, NULL);
Packit Service 5956c7
Packit Service 5956c7
#ifdef _EPOLL_DEBUG_
Packit Service 5956c7
	if (do_epoll_debug)
Packit Service 5956c7
		log_message(LOG_INFO, "Setting timer_fd %lu.%9.9ld", its.it_value.tv_sec, its.it_value.tv_nsec);
Packit Service 5956c7
#endif
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static int
Packit Service 5956c7
thread_timerfd_handler(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_master_t *m = thread->master;
Packit Service 5956c7
	uint64_t expired;
Packit Service 5956c7
	ssize_t len;
Packit Service 5956c7
Packit Service 5956c7
	len = read(m->timer_fd, &expired, sizeof(expired));
Packit Service 5956c7
	if (len < 0)
Packit Service 5956c7
		log_message(LOG_ERR, "scheduler: Error reading on timerfd fd:%d (%m)", m->timer_fd);
Packit Service 5956c7
Packit Service 5956c7
	/* Read, Write, Timer, Child thread. */
Packit Service 5956c7
	thread_rb_move_ready(m, &m->read, THREAD_READ_TIMEOUT);
Packit Service 5956c7
	thread_rb_move_ready(m, &m->write, THREAD_WRITE_TIMEOUT);
Packit Service 5956c7
	thread_rb_move_ready(m, &m->timer, THREAD_READY);
Packit Service 5956c7
	thread_rb_move_ready(m, &m->child, THREAD_CHILD_TIMEOUT);
Packit Service 5956c7
Packit Service 5956c7
	/* Register next timerfd thread */
Packit Service 5956c7
	m->timer_thread = thread_add_read(m, thread_timerfd_handler, NULL, m->timer_fd, TIMER_NEVER);
Packit Service 5956c7
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Child PID cmp helper */
Packit Service 5956c7
static inline int
Packit Service 5956c7
thread_child_pid_cmp(thread_t *t1, thread_t *t2)
Packit Service 5956c7
{
Packit Service 5956c7
	return t1->u.c.pid - t2->u.c.pid;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
set_child_finder_name(char const * (*func)(pid_t))
Packit Service 5956c7
{
Packit Service 5956c7
	child_finder_name = func;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
save_cmd_line_options(int argc, char **argv)
Packit Service 5956c7
{
Packit Service 5956c7
	sav_argc = argc;
Packit Service 5956c7
	sav_argv = argv;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
#ifndef _DEBUG_
Packit Service 5956c7
static const char *
Packit Service 5956c7
get_end(const char *str, size_t max_len)
Packit Service 5956c7
{
Packit Service 5956c7
	size_t len = strlen(str);
Packit Service 5956c7
	const char *end;
Packit Service 5956c7
Packit Service 5956c7
	if (len <= max_len)
Packit Service 5956c7
		return str + len;
Packit Service 5956c7
Packit Service 5956c7
	end = str + max_len;
Packit Service 5956c7
	if (*end == ' ')
Packit Service 5956c7
		return end;
Packit Service 5956c7
Packit Service 5956c7
	while (end > str && *--end != ' ');
Packit Service 5956c7
	if (end > str)
Packit Service 5956c7
		return end;
Packit Service 5956c7
Packit Service 5956c7
	return str + max_len;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static void
Packit Service 5956c7
log_options(const char *option, const char *option_str, unsigned indent)
Packit Service 5956c7
{
Packit Service 5956c7
	const char *p = option_str;
Packit Service 5956c7
	size_t opt_len = strlen(option);
Packit Service 5956c7
	const char *end;
Packit Service 5956c7
	bool first_line = true;
Packit Service 5956c7
Packit Service 5956c7
	while (*p) {
Packit Service 5956c7
		/* Skip leading spaces */
Packit Service 5956c7
		while (*p == ' ')
Packit Service 5956c7
			p++;
Packit Service 5956c7
Packit Service 5956c7
		end = get_end(p, 100 - opt_len);
Packit Service 5956c7
		if (first_line) {
Packit Service 5956c7
			log_message(LOG_INFO, "%*s%s: %.*s", indent, "", option, (int)(end - p), p);
Packit Service 5956c7
			first_line = false;
Packit Service 5956c7
		}
Packit Service 5956c7
		else
Packit Service 5956c7
			log_message(LOG_INFO, "%*s%.*s", (int)(indent + opt_len + 2), "", (int)(end - p), p);
Packit Service 5956c7
		p = end;
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
log_command_line(unsigned indent)
Packit Service 5956c7
{
Packit Service 5956c7
	size_t len = 0;
Packit Service 5956c7
	char *log_str;
Packit Service 5956c7
	char *p;
Packit Service 5956c7
	int i;
Packit Service 5956c7
Packit Service 5956c7
	if (!sav_argv)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	for (i = 0; i < sav_argc; i++)
Packit Service 5956c7
		len += strlen(sav_argv[i]) + 3;	/* Add opening and closing 's, and following space or '\0' */
Packit Service 5956c7
Packit Service 5956c7
	log_str = MALLOC(len);
Packit Service 5956c7
Packit Service 5956c7
	for (i = 0, p = log_str; i < sav_argc; i++)
Packit Service 5956c7
		p += sprintf(p, "%s'%s'", i ? " " : "", sav_argv[i]);
Packit Service 5956c7
Packit Service 5956c7
	log_options("Command line", log_str, indent);
Packit Service 5956c7
Packit Service 5956c7
	FREE(log_str);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* report_child_status returns true if the exit is a hard error, so unable to continue */
Packit Service 5956c7
bool
Packit Service 5956c7
report_child_status(int status, pid_t pid, char const *prog_name)
Packit Service 5956c7
{
Packit Service 5956c7
	char const *prog_id = NULL;
Packit Service 5956c7
	char pid_buf[12];	/* "pid 4194303" + '\0' - see definition of PID_MAX_LIMIT in include/linux/threads.h */
Packit Service 5956c7
	int exit_status ;
Packit Service 5956c7
Packit Service 5956c7
	if (prog_name)
Packit Service 5956c7
		prog_id = prog_name;
Packit Service 5956c7
	else if (child_finder_name)
Packit Service 5956c7
		prog_id = child_finder_name(pid);
Packit Service 5956c7
Packit Service 5956c7
	if (!prog_id) {
Packit Service 5956c7
		snprintf(pid_buf, sizeof(pid_buf), "pid %d", pid);
Packit Service 5956c7
		prog_id = pid_buf;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	if (WIFEXITED(status)) {
Packit Service 5956c7
		exit_status = WEXITSTATUS(status);
Packit Service 5956c7
Packit Service 5956c7
		/* Handle exit codes of vrrp or checker child */
Packit Service 5956c7
		if (exit_status == KEEPALIVED_EXIT_FATAL ||
Packit Service 5956c7
		    exit_status == KEEPALIVED_EXIT_CONFIG) {
Packit Service 5956c7
			log_message(LOG_INFO, "%s exited with permanent error %s. Terminating", prog_id, exit_status == KEEPALIVED_EXIT_CONFIG ? "CONFIG" : "FATAL" );
Packit Service 5956c7
			return true;
Packit Service 5956c7
		}
Packit Service 5956c7
Packit Service 5956c7
		if (exit_status != EXIT_SUCCESS)
Packit Service 5956c7
			log_message(LOG_INFO, "%s exited with status %d", prog_id, exit_status);
Packit Service 5956c7
	} else if (WIFSIGNALED(status)) {
Packit Service 5956c7
		if (WTERMSIG(status) == SIGSEGV) {
Packit Service 5956c7
			struct utsname uname_buf;
Packit Service 5956c7
Packit Service 5956c7
			log_message(LOG_INFO, "%s exited due to segmentation fault (SIGSEGV).", prog_id);
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "Please report a bug at https://github.com/acassen/keepalived/issues");
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "and include this log from when keepalived started, a description");
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "of what happened before the crash, your configuration file and the details below.");
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "Also provide the output of keepalived -v, what Linux distro and version");
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "you are running on, and whether keepalived is being run in a container or VM.");
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "A failure to provide all this information may mean the crash cannot be investigated.");
Packit Service 5956c7
			log_message(LOG_INFO, "  %s", "If you are able to provide a stack backtrace with gdb that would really help.");
Packit Service 5956c7
			log_message(LOG_INFO, "  Source version %s %s%s", PACKAGE_VERSION,
Packit Service 5956c7
#ifdef GIT_COMMIT
Packit Service 5956c7
									   ", git commit ", GIT_COMMIT
Packit Service 5956c7
#else
Packit Service 5956c7
									   "", ""
Packit Service 5956c7
#endif
Packit Service 5956c7
				   );
Packit Service 5956c7
			log_message(LOG_INFO, "  Built with kernel headers for Linux %d.%d.%d",
Packit Service 5956c7
						(LINUX_VERSION_CODE >> 16) & 0xff,
Packit Service 5956c7
						(LINUX_VERSION_CODE >>  8) & 0xff,
Packit Service 5956c7
						(LINUX_VERSION_CODE      ) & 0xff);
Packit Service 5956c7
			uname(&uname_buf);
Packit Service 5956c7
			log_message(LOG_INFO, "  Running on %s %s %s", uname_buf.sysname, uname_buf.release, uname_buf.version);
Packit Service 5956c7
			log_command_line(2);
Packit Service 5956c7
			log_options("configure options", KEEPALIVED_CONFIGURE_OPTIONS, 2);
Packit Service 5956c7
			log_options("Config options", CONFIGURATION_OPTIONS, 2);
Packit Service 5956c7
			log_options("System options", SYSTEM_OPTIONS, 2);
Packit Service 5956c7
Packit Service 5956c7
//			if (__test_bit(DONT_RESPAWN_BIT, &debug))
Packit Service 5956c7
//				segv_termination = true;
Packit Service 5956c7
		}
Packit Service 5956c7
		else
Packit Service 5956c7
			log_message(LOG_INFO, "%s exited due to signal %d", prog_id, WTERMSIG(status));
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	return false;
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* epoll related */
Packit Service 5956c7
static int
Packit Service 5956c7
thread_events_resize(thread_master_t *m, int delta)
Packit Service 5956c7
{
Packit Service 5956c7
	unsigned int new_size;
Packit Service 5956c7
Packit Service 5956c7
	m->epoll_count += delta;
Packit Service 5956c7
	if (m->epoll_count < m->epoll_size)
Packit Service 5956c7
		return 0;
Packit Service 5956c7
Packit Service 5956c7
	new_size = ((m->epoll_count / THREAD_EPOLL_REALLOC_THRESH) + 1);
Packit Service 5956c7
	new_size *= THREAD_EPOLL_REALLOC_THRESH;
Packit Service 5956c7
Packit Service 5956c7
	if (m->epoll_events)
Packit Service 5956c7
		FREE(m->epoll_events);
Packit Service 5956c7
	m->epoll_events = MALLOC(new_size * sizeof(struct epoll_event));
Packit Service 5956c7
	if (!m->epoll_events) {
Packit Service 5956c7
		m->epoll_size = 0;
Packit Service 5956c7
		return -1;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	m->epoll_size = new_size;
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static inline int
Packit Service 5956c7
thread_event_cmp(const thread_event_t *event1, const thread_event_t *event2)
Packit Service 5956c7
{
Packit Service 5956c7
	if (event1->fd < event2->fd)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
	if (event1->fd > event2->fd)
Packit Service 5956c7
		return 1;
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static thread_event_t *
Packit Service 5956c7
thread_event_new(thread_master_t *m, int fd)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event;
Packit Service 5956c7
Packit Service 5956c7
	event = (thread_event_t *) MALLOC(sizeof(thread_event_t));
Packit Service 5956c7
	if (!event)
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
Packit Service 5956c7
	if (thread_events_resize(m, 1) < 0) {
Packit Service 5956c7
		FREE(event);
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	event->fd = fd;
Packit Service 5956c7
Packit Service 5956c7
	rb_insert_sort(&m->io_events, event, n, thread_event_cmp);
Packit Service 5956c7
Packit Service 5956c7
	return event;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static thread_event_t *
Packit Service 5956c7
thread_event_get(thread_master_t *m, int fd)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t event = { .fd = fd };
Packit Service 5956c7
Packit Service 5956c7
	return rb_search(&m->io_events, &event, n, thread_event_cmp);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static int
Packit Service 5956c7
thread_event_set(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event = thread->event;
Packit Service 5956c7
	thread_master_t *m = thread->master;
Packit Service 5956c7
	struct epoll_event ev;
Packit Service 5956c7
	int op;
Packit Service 5956c7
Packit Service 5956c7
	memset(&ev, 0, sizeof(struct epoll_event));
Packit Service 5956c7
	ev.data.ptr = event;
Packit Service 5956c7
	if (__test_bit(THREAD_FL_READ_BIT, &event->flags))
Packit Service 5956c7
		ev.events |= EPOLLIN;
Packit Service 5956c7
Packit Service 5956c7
	if (__test_bit(THREAD_FL_WRITE_BIT, &event->flags))
Packit Service 5956c7
		ev.events |= EPOLLOUT;
Packit Service 5956c7
Packit Service 5956c7
	if (__test_bit(THREAD_FL_EPOLL_BIT, &event->flags))
Packit Service 5956c7
		op = EPOLL_CTL_MOD;
Packit Service 5956c7
	else
Packit Service 5956c7
		op = EPOLL_CTL_ADD;
Packit Service 5956c7
Packit Service 5956c7
	if (epoll_ctl(m->epoll_fd, op, event->fd, &ev) < 0) {
Packit Service 5956c7
		log_message(LOG_INFO, "scheduler: Error performing control on EPOLL instance (%m)");
Packit Service 5956c7
		return -1;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	__set_bit(THREAD_FL_EPOLL_BIT, &event->flags);
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static int
Packit Service 5956c7
thread_event_cancel(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event = thread->event;
Packit Service 5956c7
	thread_master_t *m = thread->master;
Packit Service 5956c7
Packit Service 5956c7
	if (!event) {
Packit Service 5956c7
		log_message(LOG_INFO, "scheduler: Error performing epoll_ctl DEL op no event linked?!");
Packit Service 5956c7
		return -1;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	if (m->epoll_fd != -1 &&
Packit Service 5956c7
	    epoll_ctl(m->epoll_fd, EPOLL_CTL_DEL, event->fd, NULL) < 0 &&
Packit Service 5956c7
	    errno != EBADF) {
Packit Service 5956c7
		log_message(LOG_INFO, "scheduler: Error performing epoll_ctl DEL op for fd:%d (%m)", event->fd);
Packit Service 5956c7
		return -1;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	rb_erase(&event->n, &m->io_events);
Packit Service 5956c7
	if (event == m->current_event)
Packit Service 5956c7
		m->current_event = NULL;
Packit Service 5956c7
	FREE(thread->event);
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static int
Packit Service 5956c7
thread_event_del(thread_t *thread, unsigned flag)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event = thread->event;
Packit Service 5956c7
Packit Service 5956c7
	if (!__test_bit(flag, &event->flags))
Packit Service 5956c7
		return 0;
Packit Service 5956c7
Packit Service 5956c7
	if (flag == THREAD_FL_EPOLL_READ_BIT) {
Packit Service 5956c7
		__clear_bit(THREAD_FL_READ_BIT, &event->flags);
Packit Service 5956c7
		if (!__test_bit(THREAD_FL_EPOLL_WRITE_BIT, &event->flags))
Packit Service 5956c7
			return thread_event_cancel(thread);
Packit Service 5956c7
Packit Service 5956c7
		event->read = NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
	else if (flag == THREAD_FL_EPOLL_WRITE_BIT) {
Packit Service 5956c7
		__clear_bit(THREAD_FL_WRITE_BIT, &event->flags);
Packit Service 5956c7
		if (!__test_bit(THREAD_FL_EPOLL_READ_BIT, &event->flags))
Packit Service 5956c7
			return thread_event_cancel(thread);
Packit Service 5956c7
Packit Service 5956c7
		event->write = NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	if (thread_event_set(thread) < 0)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
Packit Service 5956c7
	__clear_bit(flag, &event->flags);
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Make thread master. */
Packit Service 5956c7
thread_master_t *
Packit Service 5956c7
thread_make_master(void)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_master_t *new;
Packit Service 5956c7
Packit Service 5956c7
	new = (thread_master_t *) MALLOC(sizeof (thread_master_t));
Packit Service 5956c7
Packit Service 5956c7
#if HAVE_EPOLL_CREATE1
Packit Service 5956c7
	new->epoll_fd = epoll_create1(EPOLL_CLOEXEC);
Packit Service 5956c7
#else
Packit Service 5956c7
	new->epoll_fd = epoll_create(1);
Packit Service 5956c7
#endif
Packit Service 5956c7
	if (new->epoll_fd < 0) {
Packit Service 5956c7
		log_message(LOG_INFO, "scheduler: Error creating EPOLL instance (%m)");
Packit Service 5956c7
		FREE(new);
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
#if !HAVE_EPOLL_CREATE1
Packit Service 5956c7
	if (set_sock_flags(new->epoll_fd, F_SETFD, FD_CLOEXEC))
Packit Service 5956c7
		log_message(LOG_INFO, "Unable to set CLOEXEC on epoll_fd - %s (%d)", strerror(errno), errno);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	new->read = RB_ROOT_CACHED;
Packit Service 5956c7
	new->write = RB_ROOT_CACHED;
Packit Service 5956c7
	new->timer = RB_ROOT_CACHED;
Packit Service 5956c7
	new->child = RB_ROOT_CACHED;
Packit Service 5956c7
	new->io_events = RB_ROOT;
Packit Service 5956c7
	new->child_pid = RB_ROOT;
Packit Service 5956c7
	INIT_LIST_HEAD(&new->event);
Packit Service 5956c7
#ifdef USE_SIGNAL_THREADS
Packit Service 5956c7
	INIT_LIST_HEAD(&new->signal);
Packit Service 5956c7
#endif
Packit Service 5956c7
	INIT_LIST_HEAD(&new->ready);
Packit Service 5956c7
	INIT_LIST_HEAD(&new->unuse);
Packit Service 5956c7
Packit Service 5956c7
Packit Service 5956c7
	/* Register timerfd thread */
Packit Service 5956c7
	new->timer_fd = timerfd_create(CLOCK_MONOTONIC,
Packit Service 5956c7
#ifdef TFD_NONBLOCK				/* Since Linux 2.6.27 */
Packit Service 5956c7
						        TFD_NONBLOCK | TFD_CLOEXEC
Packit Service 5956c7
#else
Packit Service 5956c7
							0
Packit Service 5956c7
#endif
Packit Service 5956c7
										  );
Packit Service 5956c7
	if (new->timer_fd < 0) {
Packit Service 5956c7
		log_message(LOG_ERR, "scheduler: Cant create timerfd (%m)");
Packit Service 5956c7
		FREE(new);
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
#ifndef TFD_NONBLOCK
Packit Service 5956c7
	if (set_sock_flags(new->timer_fd, F_SETFL, O_NONBLOCK))
Packit Service 5956c7
		log_message(LOG_INFO, "Unable to set NONBLOCK on timer_fd - %s (%d)", strerror(errno), errno);
Packit Service 5956c7
Packit Service 5956c7
	if (set_sock_flags(new->timer_fd, F_SETFD, FD_CLOEXEC))
Packit Service 5956c7
		log_message(LOG_INFO, "Unable to set CLOEXEC on timer_fd - %s (%d)", strerror(errno), errno);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	new->signal_fd = signal_handler_init();
Packit Service 5956c7
Packit Service 5956c7
	new->timer_thread = thread_add_read(new, thread_timerfd_handler, NULL, new->timer_fd, TIMER_NEVER);
Packit Service 5956c7
Packit Service 5956c7
	add_signal_read_thread(new);
Packit Service 5956c7
Packit Service 5956c7
	return new;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
#ifdef THREAD_DUMP
Packit Service 5956c7
static char *
Packit Service 5956c7
timer_delay(timeval_t sands)
Packit Service 5956c7
{
Packit Service 5956c7
	static char str[42];
Packit Service 5956c7
Packit Service 5956c7
	if (sands.tv_sec == TIMER_DISABLED)
Packit Service 5956c7
		return "NEVER";
Packit Service 5956c7
	if (sands.tv_sec == 0 && sands.tv_usec == 0)
Packit Service 5956c7
		return "UNSET";
Packit Service 5956c7
Packit Service 5956c7
	if (timercmp(&sands, &time_now, >=)) {
Packit Service 5956c7
		sands = timer_sub_now(sands);
Packit Service 5956c7
		snprintf(str, sizeof str, "%lu.%6.6ld", sands.tv_sec, sands.tv_usec);
Packit Service 5956c7
	} else {
Packit Service 5956c7
		timersub(&time_now, &sands, &sands);
Packit Service 5956c7
		snprintf(str, sizeof str, "-%lu.%6.6ld", sands.tv_sec, sands.tv_usec);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	return str;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Dump rbtree */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_rb_dump(rb_root_cached_t *root, const char *tree, FILE *fp)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
	int i = 1;
Packit Service 5956c7
Packit Service 5956c7
	conf_write(fp, "----[ Begin rb_dump %s ]----", tree);
Packit Service 5956c7
Packit Service 5956c7
	rb_for_each_entry_cached(thread, root, n)
Packit Service 5956c7
		conf_write(fp, "#%.2d Thread type %s, event_fd %d, val/fd/pid %d, timer: %s, func %s(), id %ld", i++, get_thread_type_str(thread->type), thread->event ? thread->event->fd: -2, thread->u.val, timer_delay(thread->sands), get_function_name(thread->func), thread->id);
Packit Service 5956c7
Packit Service 5956c7
	conf_write(fp, "----[ End rb_dump ]----");
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static void
Packit Service 5956c7
thread_list_dump(list_head_t *l, const char *list, FILE *fp)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
	int i = 1;
Packit Service 5956c7
Packit Service 5956c7
	conf_write(fp, "----[ Begin list_dump %s ]----", list);
Packit Service 5956c7
Packit Service 5956c7
	list_for_each_entry(thread, l, next)
Packit Service 5956c7
		conf_write(fp, "#%.2d Thread:%p type %s func %s() id %ld",
Packit Service 5956c7
				i++, thread, get_thread_type_str(thread->type), get_function_name(thread->func), thread->id);
Packit Service 5956c7
Packit Service 5956c7
	conf_write(fp, "----[ End list_dump ]----");
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static void
Packit Service 5956c7
event_rb_dump(rb_root_t *root, const char *tree, FILE *fp)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event;
Packit Service 5956c7
	int i = 1;
Packit Service 5956c7
Packit Service 5956c7
	conf_write(fp, "----[ Begin rb_dump %s ]----", tree);
Packit Service 5956c7
	rb_for_each_entry(event, root, n)
Packit Service 5956c7
		conf_write(fp, "#%.2d event %p fd %d, flags: 0x%lx, read %p, write %p", i++, event, event->fd, event->flags, event->read, event->write);
Packit Service 5956c7
	conf_write(fp, "----[ End rb_dump ]----");
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
dump_thread_data(thread_master_t *m, FILE *fp)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_rb_dump(&m->read, "read", fp);
Packit Service 5956c7
	thread_rb_dump(&m->write, "write", fp);
Packit Service 5956c7
	thread_rb_dump(&m->child, "child", fp);
Packit Service 5956c7
	thread_rb_dump(&m->timer, "timer", fp);
Packit Service 5956c7
	thread_list_dump(&m->event, "event", fp);
Packit Service 5956c7
	thread_list_dump(&m->ready, "ready", fp);
Packit Service 5956c7
#ifdef USE_SIGNAL_THREADS
Packit Service 5956c7
	thread_list_dump(&m->signal, "signal", fp);
Packit Service 5956c7
#endif
Packit Service 5956c7
	thread_list_dump(&m->unuse, "unuse", fp);
Packit Service 5956c7
	event_rb_dump(&m->io_events, "io_events", fp);
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* declare thread_timer_cmp() for rbtree compares */
Packit Service 5956c7
RB_TIMER_CMP(thread);
Packit Service 5956c7
Packit Service 5956c7
/* Free all unused thread. */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_clean_unuse(thread_master_t * m)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread, *thread_tmp;
Packit Service 5956c7
	list_head_t *l = &m->unuse;
Packit Service 5956c7
Packit Service 5956c7
	list_for_each_entry_safe(thread, thread_tmp, l, next) {
Packit Service 5956c7
		list_head_del(&thread->next);
Packit Service 5956c7
Packit Service 5956c7
		/* free the thread */
Packit Service 5956c7
		FREE(thread);
Packit Service 5956c7
		m->alloc--;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	INIT_LIST_HEAD(l);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Move thread to unuse list. */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_add_unuse(thread_master_t *m, thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
	thread->type = THREAD_UNUSED;
Packit Service 5956c7
	thread->event = NULL;
Packit Service 5956c7
	INIT_LIST_HEAD(&thread->next);
Packit Service 5956c7
	list_add_tail(&thread->next, &m->unuse);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Move list element to unuse queue */
Packit Service 5956c7
static void
Packit Service 5956c7
thread_destroy_list(thread_master_t *m, list_head_t *l)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread, *thread_tmp;
Packit Service 5956c7
Packit Service 5956c7
	list_for_each_entry_safe(thread, thread_tmp, l, next) {
Packit Service 5956c7
		if (thread->event) {
Packit Service 5956c7
			thread_del_read(thread);
Packit Service 5956c7
			thread_del_write(thread);
Packit Service 5956c7
		}
Packit Service 5956c7
		list_head_del(&thread->next);
Packit Service 5956c7
		thread_add_unuse(m, thread);
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static void
Packit Service 5956c7
thread_destroy_rb(thread_master_t *m, rb_root_cached_t *root)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread, *thread_tmp;
Packit Service 5956c7
Packit Service 5956c7
	rb_for_each_entry_safe_cached(thread, thread_tmp, root, n) {
Packit Service 5956c7
		rb_erase_cached(&thread->n, root);
Packit Service 5956c7
Packit Service 5956c7
		/* Do we have a thread_event, and does it need deleting? */
Packit Service 5956c7
		if (thread->type == THREAD_READ)
Packit Service 5956c7
			thread_del_read(thread);
Packit Service 5956c7
		else if (thread->type == THREAD_WRITE)
Packit Service 5956c7
			thread_del_write(thread);
Packit Service 5956c7
Packit Service 5956c7
		thread_add_unuse(m, thread);
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Cleanup master */
Packit Service 5956c7
void
Packit Service 5956c7
thread_cleanup_master(thread_master_t * m)
Packit Service 5956c7
{
Packit Service 5956c7
	/* Unuse current thread lists */
Packit Service 5956c7
	thread_destroy_rb(m, &m->read);
Packit Service 5956c7
	thread_destroy_rb(m, &m->write);
Packit Service 5956c7
	thread_destroy_rb(m, &m->timer);
Packit Service 5956c7
	thread_destroy_rb(m, &m->child);
Packit Service 5956c7
	thread_destroy_list(m, &m->event);
Packit Service 5956c7
#ifdef USE_SIGNAL_THREADS
Packit Service 5956c7
	thread_destroy_list(m, &m->signal);
Packit Service 5956c7
#endif
Packit Service 5956c7
	thread_destroy_list(m, &m->ready);
Packit Service 5956c7
	m->child_pid = RB_ROOT;
Packit Service 5956c7
Packit Service 5956c7
	/* Clean garbage */
Packit Service 5956c7
	thread_clean_unuse(m);
Packit Service 5956c7
Packit Service 5956c7
	FREE(m->epoll_events);
Packit Service 5956c7
	m->epoll_size = 0;
Packit Service 5956c7
	m->epoll_count = 0;
Packit Service 5956c7
Packit Service 5956c7
	m->timer_thread = NULL;
Packit Service 5956c7
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
	m->snmp_timer_thread = NULL;
Packit Service 5956c7
	FD_ZERO(&m->snmp_fdset);
Packit Service 5956c7
	m->snmp_fdsetsize = 0;
Packit Service 5956c7
#endif
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Stop thread scheduler. */
Packit Service 5956c7
void
Packit Service 5956c7
thread_destroy_master(thread_master_t * m)
Packit Service 5956c7
{
Packit Service 5956c7
	if (m->epoll_fd != -1) {
Packit Service 5956c7
		close(m->epoll_fd);
Packit Service 5956c7
		m->epoll_fd = -1;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	if (m->timer_fd != -1)
Packit Service 5956c7
		close(m->timer_fd);
Packit Service 5956c7
Packit Service 5956c7
	if (m->signal_fd != -1)
Packit Service 5956c7
		signal_handler_destroy();
Packit Service 5956c7
Packit Service 5956c7
	thread_cleanup_master(m);
Packit Service 5956c7
Packit Service 5956c7
	FREE(m);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Delete top of the list and return it. */
Packit Service 5956c7
static thread_t *
Packit Service 5956c7
thread_trim_head(list_head_t *l)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	if (list_empty(l))
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
Packit Service 5956c7
	thread = list_first_entry(l, thread_t, next);
Packit Service 5956c7
	list_del_init(&thread->next);
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Make unique thread id for non pthread version of thread manager. */
Packit Service 5956c7
static inline unsigned long
Packit Service 5956c7
thread_get_id(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	return m->id++;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Make new thread. */
Packit Service 5956c7
static thread_t *
Packit Service 5956c7
thread_new(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *new;
Packit Service 5956c7
Packit Service 5956c7
	/* If one thread is already allocated return it */
Packit Service 5956c7
	new = thread_trim_head(&m->unuse);
Packit Service 5956c7
	if (!new) {
Packit Service 5956c7
		new = (thread_t *)MALLOC(sizeof(thread_t));
Packit Service 5956c7
		m->alloc++;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	INIT_LIST_HEAD(&new->next);
Packit Service 5956c7
	new->id = thread_get_id(m);
Packit Service 5956c7
	return new;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Add new read thread. */
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_read_sands(thread_master_t *m, int (*func) (thread_t *), void *arg, int fd, timeval_t *sands)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event;
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	/* I feel lucky ! :D */
Packit Service 5956c7
	if (m->current_event && m->current_event->fd == fd)
Packit Service 5956c7
		event = m->current_event;
Packit Service 5956c7
	else
Packit Service 5956c7
		event = thread_event_get(m, fd);
Packit Service 5956c7
Packit Service 5956c7
	if (!event) {
Packit Service 5956c7
		if (!(event = thread_event_new(m, fd))) {
Packit Service 5956c7
			log_message(LOG_INFO, "scheduler: Cant allocate read event for fd [%d](%m)", fd);
Packit Service 5956c7
			return NULL;
Packit Service 5956c7
		}
Packit Service 5956c7
	}
Packit Service 5956c7
	else if (__test_bit(THREAD_FL_READ_BIT, &event->flags) && event->read) {
Packit Service 5956c7
		log_message(LOG_INFO, "scheduler: There is already read event %p (read %p) registered on fd [%d]", event, event->read, fd);
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = THREAD_READ;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = arg;
Packit Service 5956c7
	thread->u.fd = fd;
Packit Service 5956c7
	thread->event = event;
Packit Service 5956c7
Packit Service 5956c7
	/* Set & flag event */
Packit Service 5956c7
	__set_bit(THREAD_FL_READ_BIT, &event->flags);
Packit Service 5956c7
	event->read = thread;
Packit Service 5956c7
	if (!__test_bit(THREAD_FL_EPOLL_READ_BIT, &event->flags)) {
Packit Service 5956c7
		if (thread_event_set(thread) < 0) {
Packit Service 5956c7
			log_message(LOG_INFO, "scheduler: Cant register read event for fd [%d](%m)", fd);
Packit Service 5956c7
			thread_add_unuse(m, thread);
Packit Service 5956c7
			return NULL;
Packit Service 5956c7
		}
Packit Service 5956c7
		__set_bit(THREAD_FL_EPOLL_READ_BIT, &event->flags);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	thread->sands = *sands;
Packit Service 5956c7
Packit Service 5956c7
	/* Sort the thread. */
Packit Service 5956c7
	rb_insert_sort_cached(&m->read, thread, n, thread_timer_cmp);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_read(thread_master_t *m, int (*func) (thread_t *), void *arg, int fd, unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	timeval_t sands;
Packit Service 5956c7
Packit Service 5956c7
	/* Compute read timeout value */
Packit Service 5956c7
	if (timer == TIMER_NEVER)
Packit Service 5956c7
		sands.tv_sec = TIMER_DISABLED;
Packit Service 5956c7
	else {
Packit Service 5956c7
		set_time_now();
Packit Service 5956c7
		sands = timer_add_long(time_now, timer);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	return thread_add_read_sands(m, func, arg, fd, &sands);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
int
Packit Service 5956c7
thread_del_read(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	if (!thread || !thread->event)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
Packit Service 5956c7
	if (thread_event_del(thread, THREAD_FL_EPOLL_READ_BIT) < 0)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
static void
Packit Service 5956c7
thread_del_read_fd(thread_master_t *m, int fd)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event;
Packit Service 5956c7
Packit Service 5956c7
	event = thread_event_get(m, fd);
Packit Service 5956c7
	if (!event || !event->read)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	thread_cancel(event->read);
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
static void
Packit Service 5956c7
thread_read_requeue(thread_master_t *m, int fd, const timeval_t *new_sands)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
	thread_event_t *event;
Packit Service 5956c7
Packit Service 5956c7
	event = thread_event_get(m, fd);
Packit Service 5956c7
	if (!event || !event->read)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	thread = event->read;
Packit Service 5956c7
Packit Service 5956c7
	if (thread->type != THREAD_READ) {
Packit Service 5956c7
		/* If the thread is not on the read list, don't touch it */
Packit Service 5956c7
		return;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	thread->sands = *new_sands;
Packit Service 5956c7
Packit Service 5956c7
	rb_move_cached(&thread->master->read, thread, n, thread_timer_cmp);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
thread_requeue_read(thread_master_t *m, int fd, const timeval_t *sands)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_read_requeue(m, fd, sands);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Add new write thread. */
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_write(thread_master_t *m, int (*func) (thread_t *), void *arg, int fd, unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_event_t *event;
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	/* I feel lucky ! :D */
Packit Service 5956c7
	if (m->current_event && m->current_event->fd == fd)
Packit Service 5956c7
		event = m->current_event;
Packit Service 5956c7
	else
Packit Service 5956c7
		event = thread_event_get(m, fd);
Packit Service 5956c7
Packit Service 5956c7
	if (!event) {
Packit Service 5956c7
		if (!(event = thread_event_new(m, fd))) {
Packit Service 5956c7
			log_message(LOG_INFO, "scheduler: Cant allocate write event for fd [%d](%m)", fd);
Packit Service 5956c7
			return NULL;
Packit Service 5956c7
		}
Packit Service 5956c7
	}
Packit Service 5956c7
	else if (__test_bit(THREAD_FL_WRITE_BIT, &event->flags) && event->write) {
Packit Service 5956c7
		log_message(LOG_INFO, "scheduler: There is already write event registered on fd [%d]", fd);
Packit Service 5956c7
		return NULL;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = THREAD_WRITE;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = arg;
Packit Service 5956c7
	thread->u.fd = fd;
Packit Service 5956c7
	thread->event = event;
Packit Service 5956c7
Packit Service 5956c7
	/* Set & flag event */
Packit Service 5956c7
	__set_bit(THREAD_FL_WRITE_BIT, &event->flags);
Packit Service 5956c7
	event->write = thread;
Packit Service 5956c7
	if (!__test_bit(THREAD_FL_EPOLL_WRITE_BIT, &event->flags)) {
Packit Service 5956c7
		if (thread_event_set(thread) < 0) {
Packit Service 5956c7
			log_message(LOG_INFO, "scheduler: Cant register write event for fd [%d](%m)" , fd);
Packit Service 5956c7
			thread_add_unuse(m, thread);
Packit Service 5956c7
			return NULL;
Packit Service 5956c7
		}
Packit Service 5956c7
		__set_bit(THREAD_FL_EPOLL_WRITE_BIT, &event->flags);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	/* Compute write timeout value */
Packit Service 5956c7
	if (timer == TIMER_NEVER)
Packit Service 5956c7
		thread->sands.tv_sec = TIMER_DISABLED;
Packit Service 5956c7
	else {
Packit Service 5956c7
		set_time_now();
Packit Service 5956c7
		thread->sands = timer_add_long(time_now, timer);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	/* Sort the thread. */
Packit Service 5956c7
	rb_insert_sort_cached(&m->write, thread, n, thread_timer_cmp);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
int
Packit Service 5956c7
thread_del_write(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	if (!thread || !thread->event)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
Packit Service 5956c7
	if (thread_event_del(thread, THREAD_FL_EPOLL_WRITE_BIT) < 0)
Packit Service 5956c7
		return -1;
Packit Service 5956c7
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
thread_close_fd(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	if (thread->u.fd == -1)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	if (thread->event)
Packit Service 5956c7
		thread_event_cancel(thread);
Packit Service 5956c7
Packit Service 5956c7
	close(thread->u.fd);
Packit Service 5956c7
	thread->u.fd = -1;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Add timer event thread. */
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_timer(thread_master_t *m, int (*func) (thread_t *), void *arg, unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = THREAD_TIMER;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = arg;
Packit Service 5956c7
Packit Service 5956c7
	/* Do we need jitter here? */
Packit Service 5956c7
	if (timer == TIMER_NEVER)
Packit Service 5956c7
		thread->sands.tv_sec = TIMER_DISABLED;
Packit Service 5956c7
	else {
Packit Service 5956c7
		set_time_now();
Packit Service 5956c7
		thread->sands = timer_add_long(time_now, timer);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	/* Sort by timeval. */
Packit Service 5956c7
	rb_insert_sort_cached(&m->timer, thread, n, thread_timer_cmp);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
timer_thread_update_timeout(thread_t *thread, unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	timeval_t sands;
Packit Service 5956c7
Packit Service 5956c7
	if (thread->type > THREAD_MAX_WAITING) {
Packit Service 5956c7
		/* It is probably on the ready list, so we'd better just let it run */
Packit Service 5956c7
		return;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	set_time_now();
Packit Service 5956c7
	sands = timer_add_long(time_now, timer);
Packit Service 5956c7
Packit Service 5956c7
	if (timercmp(&thread->sands, &sands, ==))
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	thread->sands = sands;
Packit Service 5956c7
Packit Service 5956c7
	rb_move_cached(&thread->master->timer, thread, n, thread_timer_cmp);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_timer_shutdown(thread_master_t *m, int(*func)(thread_t *), void *arg, unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread = thread_add_timer(m, func, arg, timer);
Packit Service 5956c7
Packit Service 5956c7
	thread->type = THREAD_TIMER_SHUTDOWN;
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Add a child thread. */
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_child(thread_master_t * m, int (*func) (thread_t *), void * arg, pid_t pid, unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = THREAD_CHILD;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = arg;
Packit Service 5956c7
	thread->u.c.pid = pid;
Packit Service 5956c7
	thread->u.c.status = 0;
Packit Service 5956c7
Packit Service 5956c7
	/* Compute child timeout value */
Packit Service 5956c7
	if (timer == TIMER_NEVER)
Packit Service 5956c7
		thread->sands.tv_sec = TIMER_DISABLED;
Packit Service 5956c7
	else {
Packit Service 5956c7
		set_time_now();
Packit Service 5956c7
		thread->sands = timer_add_long(time_now, timer);
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	/* Sort by timeval. */
Packit Service 5956c7
	rb_insert_sort_cached(&m->child, thread, n, thread_timer_cmp);
Packit Service 5956c7
Packit Service 5956c7
	/* Sort by PID */
Packit Service 5956c7
	rb_insert_sort(&m->child_pid, thread, rb_data, thread_child_pid_cmp);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
thread_children_reschedule(thread_master_t *m, int (*func)(thread_t *), unsigned long timer)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
// What is this used for ??
Packit Service 5956c7
	set_time_now();
Packit Service 5956c7
	rb_for_each_entry_cached(thread, &m->child, n) {
Packit Service 5956c7
		thread->func = func;
Packit Service 5956c7
		thread->sands = timer_add_long(time_now, timer);
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Add simple event thread. */
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_event(thread_master_t * m, int (*func) (thread_t *), void *arg, int val)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = THREAD_EVENT;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = arg;
Packit Service 5956c7
	thread->u.val = val;
Packit Service 5956c7
	INIT_LIST_HEAD(&thread->next);
Packit Service 5956c7
	list_add_tail(&thread->next, &m->event);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Add terminate event thread. */
Packit Service 5956c7
static thread_t *
Packit Service 5956c7
thread_add_generic_terminate_event(thread_master_t * m, thread_type_t type, int (*func)(thread_t *))
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = type;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = NULL;
Packit Service 5956c7
	thread->u.val = 0;
Packit Service 5956c7
	INIT_LIST_HEAD(&thread->next);
Packit Service 5956c7
	list_add_tail(&thread->next, &m->event);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_terminate_event(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	return thread_add_generic_terminate_event(m, THREAD_TERMINATE, NULL);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_start_terminate_event(thread_master_t *m, int(*func)(thread_t *))
Packit Service 5956c7
{
Packit Service 5956c7
	return thread_add_generic_terminate_event(m, THREAD_TERMINATE_START, func);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
#ifdef USE_SIGNAL_THREADS
Packit Service 5956c7
/* Add signal thread. */
Packit Service 5956c7
thread_t *
Packit Service 5956c7
thread_add_signal(thread_master_t *m, int (*func) (thread_t *), void *arg, int signum)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	thread = thread_new(m);
Packit Service 5956c7
	thread->type = THREAD_SIGNAL;
Packit Service 5956c7
	thread->master = m;
Packit Service 5956c7
	thread->func = func;
Packit Service 5956c7
	thread->arg = arg;
Packit Service 5956c7
	thread->u.val = signum;
Packit Service 5956c7
	INIT_LIST_HEAD(&thread->next);
Packit Service 5956c7
	list_add_tail(&thread->next, &m->signal);
Packit Service 5956c7
Packit Service 5956c7
	/* Update signalfd accordingly */
Packit Service 5956c7
	if (sigismember(&m->signal_mask, signum))
Packit Service 5956c7
		return thread;
Packit Service 5956c7
	sigaddset(&m->signal_mask, signum);
Packit Service 5956c7
	signalfd(m->signal_fd, &m->signal_mask, 0);
Packit Service 5956c7
Packit Service 5956c7
	return thread;
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* Cancel thread from scheduler. */
Packit Service 5956c7
void
Packit Service 5956c7
thread_cancel(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_master_t *m;
Packit Service 5956c7
Packit Service 5956c7
	if (!thread || thread->type == THREAD_UNUSED)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	m = thread->master;
Packit Service 5956c7
Packit Service 5956c7
	switch (thread->type) {
Packit Service 5956c7
	case THREAD_READ:
Packit Service 5956c7
		thread_event_del(thread, THREAD_FL_EPOLL_READ_BIT);
Packit Service 5956c7
		rb_erase_cached(&thread->n, &m->read);
Packit Service 5956c7
		break;
Packit Service 5956c7
	case THREAD_WRITE:
Packit Service 5956c7
		thread_event_del(thread, THREAD_FL_EPOLL_WRITE_BIT);
Packit Service 5956c7
		rb_erase_cached(&thread->n, &m->write);
Packit Service 5956c7
		break;
Packit Service 5956c7
	case THREAD_TIMER:
Packit Service 5956c7
		rb_erase_cached(&thread->n, &m->timer);
Packit Service 5956c7
		break;
Packit Service 5956c7
	case THREAD_CHILD:
Packit Service 5956c7
		/* Does this need to kill the child, or is that the
Packit Service 5956c7
		 * caller's job?
Packit Service 5956c7
		 * This function is currently unused, so leave it for now.
Packit Service 5956c7
		 */
Packit Service 5956c7
		rb_erase_cached(&thread->n, &m->child);
Packit Service 5956c7
		rb_erase(&thread->rb_data, &m->child_pid);
Packit Service 5956c7
		break;
Packit Service 5956c7
	case THREAD_READY_FD:
Packit Service 5956c7
	case THREAD_READ_TIMEOUT:
Packit Service 5956c7
	case THREAD_WRITE_TIMEOUT:
Packit Service 5956c7
		if (thread->event) {
Packit Service 5956c7
			rb_erase(&thread->event->n, &m->io_events);
Packit Service 5956c7
			FREE(thread->event);
Packit Service 5956c7
		}
Packit Service 5956c7
		/* ... falls through ... */
Packit Service 5956c7
	case THREAD_EVENT:
Packit Service 5956c7
	case THREAD_READY:
Packit Service 5956c7
#ifdef USE_SIGNAL_THREADS
Packit Service 5956c7
	case THREAD_SIGNAL:
Packit Service 5956c7
#endif
Packit Service 5956c7
	case THREAD_CHILD_TIMEOUT:
Packit Service 5956c7
	case THREAD_CHILD_TERMINATED:
Packit Service 5956c7
		list_head_del(&thread->next);
Packit Service 5956c7
		break;
Packit Service 5956c7
	default:
Packit Service 5956c7
		break;
Packit Service 5956c7
	}
Packit Service 5956c7
Packit Service 5956c7
	thread_add_unuse(m, thread);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
thread_cancel_read(thread_master_t *m, int fd)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread, *thread_tmp;
Packit Service 5956c7
Packit Service 5956c7
	rb_for_each_entry_safe_cached(thread, thread_tmp, &m->read, n) {
Packit Service 5956c7
		if (thread->u.fd == fd) {
Packit Service 5956c7
			if (thread->event->write) {
Packit Service 5956c7
				thread_cancel(thread->event->write);
Packit Service 5956c7
				thread->event->write = NULL;
Packit Service 5956c7
			}
Packit Service 5956c7
			thread_cancel(thread);
Packit Service 5956c7
			break;
Packit Service 5956c7
		}
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
#ifdef _INCLUDE_UNUSED_CODE_
Packit Service 5956c7
/* Delete all events which has argument value arg. */
Packit Service 5956c7
void
Packit Service 5956c7
thread_cancel_event(thread_master_t *m, void *arg)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t *thread, *thread_tmp;
Packit Service 5956c7
	list_head_t *l = &m->event;
Packit Service 5956c7
Packit Service 5956c7
// Why doesn't this use thread_cancel() above
Packit Service 5956c7
	list_for_each_entry_safe(thread, thread_tmp, l, next) {
Packit Service 5956c7
		if (thread->arg == arg) {
Packit Service 5956c7
			list_head_del(&thread->next);
Packit Service 5956c7
			thread_add_unuse(m, thread);
Packit Service 5956c7
		}
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
static int
Packit Service 5956c7
snmp_read_thread(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	fd_set snmp_fdset;
Packit Service 5956c7
Packit Service 5956c7
	FD_ZERO(&snmp_fdset);
Packit Service 5956c7
	FD_SET(thread->u.fd, &snmp_fdset);
Packit Service 5956c7
Packit Service 5956c7
	snmp_read(&snmp_fdset);
Packit Service 5956c7
	netsnmp_check_outstanding_agent_requests();
Packit Service 5956c7
Packit Service 5956c7
	thread_add_read(thread->master, snmp_read_thread, thread->arg, thread->u.fd, TIMER_NEVER);
Packit Service 5956c7
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
int
Packit Service 5956c7
snmp_timeout_thread(thread_t *thread)
Packit Service 5956c7
{
Packit Service 5956c7
	snmp_timeout();
Packit Service 5956c7
	run_alarms();
Packit Service 5956c7
	netsnmp_check_outstanding_agent_requests();
Packit Service 5956c7
Packit Service 5956c7
	thread->master->snmp_timer_thread = thread_add_timer(thread->master, snmp_timeout_thread, thread->arg, TIMER_NEVER);
Packit Service 5956c7
Packit Service 5956c7
	return 0;
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
// See https://vincent.bernat.im/en/blog/2012-snmp-event-loop
Packit Service 5956c7
static void
Packit Service 5956c7
snmp_epoll_info(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	fd_set snmp_fdset;
Packit Service 5956c7
	int fdsetsize = 0;
Packit Service 5956c7
	int max_fdsetsize;
Packit Service 5956c7
	struct timeval snmp_timer_wait = { .tv_sec = TIMER_DISABLED };
Packit Service 5956c7
	int snmpblock = true;
Packit Service 5956c7
	unsigned long *old_set, *new_set;	// Must be unsigned for ffsl() to work for us
Packit Service 5956c7
	unsigned long diff;
Packit Service 5956c7
	int i;
Packit Service 5956c7
	int fd;
Packit Service 5956c7
	int bit;
Packit Service 5956c7
Packit Service 5956c7
#if 0
Packit Service 5956c7
// TODO
Packit Service 5956c7
#if sizeof fd_mask  != sizeof diff
Packit Service 5956c7
#error "snmp_epoll_info sizeof(fd_mask) does not match old_set/new_set/diff"
Packit Service 5956c7
#endif
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	FD_ZERO(&snmp_fdset);
Packit Service 5956c7
Packit Service 5956c7
	/* When SNMP is enabled, we may have to select() on additional
Packit Service 5956c7
	 * FD. snmp_select_info() will add them to `readfd'. The trick
Packit Service 5956c7
	 * with this function is its last argument. We need to set it
Packit Service 5956c7
	 * true to set its own timer that we then compare against ours. */
Packit Service 5956c7
	snmp_select_info(&fdsetsize, &snmp_fdset, &snmp_timer_wait, &snmpblock);
Packit Service 5956c7
Packit Service 5956c7
	if (snmpblock)
Packit Service 5956c7
		snmp_timer_wait.tv_sec = TIMER_DISABLED;
Packit Service 5956c7
	timer_thread_update_timeout(m->snmp_timer_thread, timer_long(snmp_timer_wait));
Packit Service 5956c7
Packit Service 5956c7
	max_fdsetsize = m->snmp_fdsetsize > fdsetsize ? m->snmp_fdsetsize : fdsetsize;
Packit Service 5956c7
	if (!max_fdsetsize)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	for (i = 0, old_set = (unsigned long *)&m->snmp_fdset, new_set = (unsigned long *)&snmp_fdset; i <= max_fdsetsize / (int)sizeof(*new_set); i++, old_set++, new_set++) {
Packit Service 5956c7
		if (*old_set == *new_set)
Packit Service 5956c7
			continue;
Packit Service 5956c7
Packit Service 5956c7
		diff = *old_set ^ *new_set;
Packit Service 5956c7
		fd = i * sizeof(*old_set) * CHAR_BIT - 1;
Packit Service 5956c7
		do {
Packit Service 5956c7
			bit = ffsl(diff);
Packit Service 5956c7
			diff >>= bit;
Packit Service 5956c7
			fd += bit;
Packit Service 5956c7
			if (FD_ISSET(fd, &snmp_fdset)) {
Packit Service 5956c7
				/* Add the fd */
Packit Service 5956c7
				thread_add_read(m, snmp_read_thread, 0, fd, TIMER_NEVER);
Packit Service 5956c7
				FD_SET(fd, &m->snmp_fdset);
Packit Service 5956c7
			} else {
Packit Service 5956c7
				/* Remove the fd */
Packit Service 5956c7
				thread_del_read_fd(m, fd);
Packit Service 5956c7
				FD_CLR(fd, &m->snmp_fdset);
Packit Service 5956c7
			}
Packit Service 5956c7
		} while (diff);
Packit Service 5956c7
	}
Packit Service 5956c7
	m->snmp_fdsetsize = fdsetsize;
Packit Service 5956c7
}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
/* Fetch next ready thread. */
Packit Service 5956c7
static list_head_t *
Packit Service 5956c7
thread_fetch_next_queue(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	int sav_errno;
Packit Service 5956c7
	int last_epoll_errno = 0;
Packit Service 5956c7
	int ret;
Packit Service 5956c7
	int i;
Packit Service 5956c7
Packit Service 5956c7
	assert(m != NULL);
Packit Service 5956c7
Packit Service 5956c7
	/* If there is event process it first. */
Packit Service 5956c7
	if (m->event.next != &m->event)
Packit Service 5956c7
		return &m->event;
Packit Service 5956c7
Packit Service 5956c7
	/* If there are ready threads process them */
Packit Service 5956c7
	if (m->ready.next != &m->ready)
Packit Service 5956c7
		return &m->ready;
Packit Service 5956c7
Packit Service 5956c7
	do {
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
		if (snmp_running)
Packit Service 5956c7
			snmp_epoll_info(m);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
		/* Calculate and set wait timer. Take care of timeouted fd.  */
Packit Service 5956c7
		thread_set_timer(m);
Packit Service 5956c7
Packit Service 5956c7
#ifdef _VRRP_FD_DEBUG_
Packit Service 5956c7
		if (extra_threads_debug)
Packit Service 5956c7
			extra_threads_debug();
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
#ifdef _EPOLL_THREAD_DUMP_
Packit Service 5956c7
		if (do_epoll_thread_dump)
Packit Service 5956c7
			dump_thread_data(m, NULL);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
#ifdef _EPOLL_DEBUG_
Packit Service 5956c7
		if (do_epoll_debug)
Packit Service 5956c7
			log_message(LOG_INFO, "calling epoll_wait");
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
		/* Call epoll function. */
Packit Service 5956c7
		ret = epoll_wait(m->epoll_fd, m->epoll_events, m->epoll_count, -1);
Packit Service 5956c7
		sav_errno = errno;
Packit Service 5956c7
Packit Service 5956c7
#ifdef _EPOLL_DEBUG_
Packit Service 5956c7
		if (do_epoll_debug) {
Packit Service 5956c7
			if (ret == -1)
Packit Service 5956c7
				log_message(LOG_INFO, "epoll_wait returned %d, errno %d", ret, sav_errno);
Packit Service 5956c7
			else
Packit Service 5956c7
				log_message(LOG_INFO, "epoll_wait returned %d fds", ret);
Packit Service 5956c7
		}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
		if (ret < 0) {
Packit Service 5956c7
			if (sav_errno == EINTR)
Packit Service 5956c7
				continue;
Packit Service 5956c7
Packit Service 5956c7
			/* Real error. */
Packit Service 5956c7
			if (sav_errno != last_epoll_errno) {
Packit Service 5956c7
				/* Log the error first time only */
Packit Service 5956c7
				log_message(LOG_INFO, "scheduler: epoll_wait error: %s", strerror(sav_errno));
Packit Service 5956c7
				last_epoll_errno = sav_errno;
Packit Service 5956c7
			}
Packit Service 5956c7
Packit Service 5956c7
			/* Make sure we don't sit it a tight loop */
Packit Service 5956c7
			if (sav_errno == EBADF || sav_errno == EFAULT || sav_errno == EINVAL)
Packit Service 5956c7
				sleep(1);
Packit Service 5956c7
Packit Service 5956c7
			continue;
Packit Service 5956c7
		}
Packit Service 5956c7
Packit Service 5956c7
		/* Handle epoll events */
Packit Service 5956c7
		for (i = 0; i < ret; i++) {
Packit Service 5956c7
			struct epoll_event *ep_ev;
Packit Service 5956c7
			thread_event_t *ev;
Packit Service 5956c7
Packit Service 5956c7
			ep_ev = &m->epoll_events[i];
Packit Service 5956c7
			ev = ep_ev->data.ptr;
Packit Service 5956c7
Packit Service 5956c7
			/* Error */
Packit Service 5956c7
// TODO - no thread processing function handles THREAD_READ_ERROR/THREAD_WRITE_ERROR yet
Packit Service 5956c7
			if (ep_ev->events & (EPOLLHUP | EPOLLERR | EPOLLRDHUP)) {
Packit Service 5956c7
				if (ev->read) {
Packit Service 5956c7
					thread_move_ready(m, &m->read, ev->read, THREAD_READ_ERROR);
Packit Service 5956c7
					ev->read = NULL;
Packit Service 5956c7
				}
Packit Service 5956c7
Packit Service 5956c7
				if (ev->write) {
Packit Service 5956c7
					thread_move_ready(m, &m->write, ev->write, THREAD_WRITE_ERROR);
Packit Service 5956c7
					ev->write = NULL;
Packit Service 5956c7
				}
Packit Service 5956c7
Packit Service 5956c7
				continue;
Packit Service 5956c7
			}
Packit Service 5956c7
Packit Service 5956c7
			/* READ */
Packit Service 5956c7
			if (ep_ev->events & EPOLLIN) {
Packit Service 5956c7
				if (!ev->read) {
Packit Service 5956c7
					log_message(LOG_INFO, "scheduler: No read thread bound on fd:%d (fl:0x%.4X)"
Packit Service 5956c7
						      , ev->fd, ep_ev->events);
Packit Service 5956c7
					assert(0);
Packit Service 5956c7
				}
Packit Service 5956c7
				thread_move_ready(m, &m->read, ev->read, THREAD_READY_FD);
Packit Service 5956c7
				ev->read = NULL;
Packit Service 5956c7
			}
Packit Service 5956c7
Packit Service 5956c7
			/* WRITE */
Packit Service 5956c7
			if (ep_ev->events & EPOLLOUT) {
Packit Service 5956c7
				if (!ev->write) {
Packit Service 5956c7
					log_message(LOG_INFO, "scheduler: No write thread bound on fd:%d (fl:0x%.4X)"
Packit Service 5956c7
						      , ev->fd, ep_ev->events);
Packit Service 5956c7
					assert(0);
Packit Service 5956c7
				}
Packit Service 5956c7
				thread_move_ready(m, &m->write, ev->write, THREAD_READY_FD);
Packit Service 5956c7
				ev->write = NULL;
Packit Service 5956c7
			}
Packit Service 5956c7
Packit Service 5956c7
			if (ep_ev->events & EPOLLHUP) {
Packit Service 5956c7
				log_message(LOG_INFO, "Received EPOLLHUP for fd %d", ev->fd);
Packit Service 5956c7
			}
Packit Service 5956c7
Packit Service 5956c7
			if (ep_ev->events & EPOLLERR) {
Packit Service 5956c7
				log_message(LOG_INFO, "Received EPOLLERR for fd %d", ev->fd);
Packit Service 5956c7
			}
Packit Service 5956c7
		}
Packit Service 5956c7
Packit Service 5956c7
		/* Update current time */
Packit Service 5956c7
		set_time_now();
Packit Service 5956c7
Packit Service 5956c7
		/* If there is a ready thread, return it. */
Packit Service 5956c7
		if (m->ready.next != &m->ready)
Packit Service 5956c7
			return &m->ready;
Packit Service 5956c7
	} while (true);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Call thread ! */
Packit Service 5956c7
static inline void
Packit Service 5956c7
thread_call(thread_t * thread)
Packit Service 5956c7
{
Packit Service 5956c7
#ifdef _EPOLL_DEBUG_
Packit Service 5956c7
	if (do_epoll_debug)
Packit Service 5956c7
		log_message(LOG_INFO, "Calling thread function %s(), type %s, val/fd/pid %d, status %d id %lu", get_function_name(thread->func), get_thread_type_str(thread->type), thread->u.val, thread->u.c.status, thread->id);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	(*thread->func) (thread);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
process_threads(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_t* thread;
Packit Service 5956c7
	list_head_t *thread_list;
Packit Service 5956c7
	int thread_type;
Packit Service 5956c7
Packit Service 5956c7
	/*
Packit Service 5956c7
	 * Processing the master thread queues,
Packit Service 5956c7
	 * return and execute one ready thread.
Packit Service 5956c7
	 */
Packit Service 5956c7
	while ((thread_list = thread_fetch_next_queue(m))) {
Packit Service 5956c7
		/* Run until error, used for debuging only */
Packit Service 5956c7
#if defined _DEBUG_ && defined _MEM_CHECK_
Packit Service 5956c7
		if (__test_bit(MEM_ERR_DETECT_BIT, &debug)
Packit Service 5956c7
#ifdef _WITH_VRRP_
Packit Service 5956c7
		    && __test_bit(DONT_RELEASE_VRRP_BIT, &debug)
Packit Service 5956c7
#endif
Packit Service 5956c7
								) {
Packit Service 5956c7
			__clear_bit(MEM_ERR_DETECT_BIT, &debug);
Packit Service 5956c7
#ifdef _WITH_VRRP_
Packit Service 5956c7
			__clear_bit(DONT_RELEASE_VRRP_BIT, &debug);
Packit Service 5956c7
#endif
Packit Service 5956c7
			thread_add_terminate_event(master);
Packit Service 5956c7
		}
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
		/* If we are shutting down, only process relevant thread types.
Packit Service 5956c7
		 * We only want timer and signal fd, and don't want inotify, vrrp socket,
Packit Service 5956c7
		 * snmp_read, bfd_receiver, bfd pipe in vrrp/check, dbus pipe or netlink fds. */
Packit Service 5956c7
		thread = thread_trim_head(thread_list);
Packit Service 564e88
Packit Service 564e88
		if (thread && thread->type == THREAD_CHILD_TIMEOUT) {
Packit Service 564e88
			/* We remove the thread from the child_pid queue here so that
Packit Service 564e88
			 * if the termination arrives before we processed the timeout
Packit Service 564e88
			 * we can still handle the termination. */
Packit Service 564e88
			rb_erase(&thread->rb_data, &master->child_pid);
Packit Service 564e88
		}
Packit Service 564e88
Packit Service 5956c7
		if (!shutting_down ||
Packit Service 5956c7
		    (thread->type == THREAD_READY_FD &&
Packit Service 5956c7
		     (thread->u.fd == m->timer_fd || thread->u.fd == m->signal_fd)) ||
Packit Service 5956c7
		    thread->type == THREAD_CHILD ||
Packit Service 5956c7
		    thread->type == THREAD_CHILD_TIMEOUT ||
Packit Service 5956c7
		    thread->type == THREAD_CHILD_TERMINATED ||
Packit Service 5956c7
		    thread->type == THREAD_TIMER_SHUTDOWN ||
Packit Service 5956c7
		    thread->type == THREAD_TERMINATE) {
Packit Service 5956c7
			if (thread->func)
Packit Service 5956c7
				thread_call(thread);
Packit Service 5956c7
Packit Service 5956c7
			if (thread->type == THREAD_TERMINATE_START)
Packit Service 5956c7
				shutting_down = true;
Packit Service 5956c7
		}
Packit Service 5956c7
Packit Service 5956c7
		m->current_event = (thread->type == THREAD_READY_FD) ? thread->event : NULL;
Packit Service 5956c7
		thread_type = thread->type;
Packit Service 5956c7
		thread_add_unuse(master, thread);
Packit Service 5956c7
Packit Service 5956c7
		/* If we are shutting down, and the shutdown timer is not running and
Packit Service 5956c7
		 * all children have terminated, then we can terminate */
Packit Service 5956c7
		if (shutting_down && !m->shutdown_timer_running && !m->child.rb_root.rb_node)
Packit Service 5956c7
			break;
Packit Service 5956c7
Packit Service 5956c7
		/* If daemon hanging event is received stop processing */
Packit Service 5956c7
		if (thread_type == THREAD_TERMINATE)
Packit Service 5956c7
			break;
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
static void
Packit Service 5956c7
process_child_termination(pid_t pid, int status)
Packit Service 5956c7
{
Packit Service 5956c7
	thread_master_t * m = master;
Packit Service 5956c7
	thread_t th = { .u.c.pid = pid };
Packit Service 5956c7
	thread_t *thread;
Packit Service 5956c7
	bool permanent_vrrp_checker_error = false;
Packit Service 5956c7
Packit Service 5956c7
#ifndef _DEBUG_
Packit Service 5956c7
	if (prog_type == PROG_TYPE_PARENT)
Packit Service 5956c7
		permanent_vrrp_checker_error = report_child_status(status, pid, NULL);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	thread = rb_search(&master->child_pid, &th, rb_data, thread_child_pid_cmp);
Packit Service 5956c7
Packit Service 5956c7
#ifdef _EPOLL_DEBUG_
Packit Service 5956c7
	if (do_epoll_debug)
Packit Service 5956c7
		log_message(LOG_INFO, "Child %d terminated with status 0x%x, thread_id %lu", pid, status, thread ? thread->id : 0);
Packit Service 5956c7
#endif
Packit Service 5956c7
Packit Service 5956c7
	if (!thread)
Packit Service 5956c7
		return;
Packit Service 5956c7
Packit Service 5956c7
	rb_erase(&thread->rb_data, &master->child_pid);
Packit Service 5956c7
Packit Service 5956c7
	thread->u.c.status = status;
Packit Service 5956c7
Packit Service 5956c7
	if (permanent_vrrp_checker_error)
Packit Service 5956c7
	{
Packit Service 5956c7
		/* The child had a permanant error, so no point in respawning */
Packit Service 5956c7
		rb_erase_cached(&thread->n, &m->child);
Packit Service 5956c7
		thread_add_unuse(m, thread);
Packit Service 5956c7
Packit Service 5956c7
		thread_add_terminate_event(m);
Packit Service 5956c7
	}
Packit Service 564e88
	else if (thread->type == THREAD_CHILD_TIMEOUT) {
Packit Service 564e88
		/* The child had been timed out, but we have not processed the timeout
Packit Service 564e88
		 * and it is still on the thread->ready queue. Since we have now got
Packit Service 564e88
		 * the termination, just handle the termination instead. */
Packit Service 564e88
		thread->type = THREAD_CHILD_TERMINATED;
Packit Service 564e88
	}
Packit Service 5956c7
	else
Packit Service 5956c7
		thread_move_ready(m, &m->child, thread, THREAD_CHILD_TERMINATED);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Synchronous signal handler to reap child processes */
Packit Service 5956c7
void
Packit Service 5956c7
thread_child_handler(__attribute__((unused)) void *v, __attribute__((unused)) int unused)
Packit Service 5956c7
{
Packit Service 5956c7
	pid_t pid;
Packit Service 5956c7
	int status;
Packit Service 5956c7
Packit Service 5956c7
	while ((pid = waitpid(-1, &status, WNOHANG))) {
Packit Service 5956c7
		if (pid == -1) {
Packit Service 5956c7
			if (errno == ECHILD)
Packit Service 5956c7
				return;
Packit Service 5956c7
			DBG("waitpid error: %s", strerror(errno));
Packit Service 5956c7
			assert(0);
Packit Service 5956c7
		}
Packit Service 5956c7
		process_child_termination(pid, status);
Packit Service 5956c7
	}
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
void
Packit Service 5956c7
thread_add_base_threads(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
	m->timer_thread = thread_add_read(m, thread_timerfd_handler, NULL, m->timer_fd, TIMER_NEVER);
Packit Service 5956c7
	add_signal_read_thread(m);
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
	m->snmp_timer_thread = thread_add_timer(m, snmp_timeout_thread, 0, TIMER_NEVER);
Packit Service 5956c7
#endif
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
/* Our infinite scheduling loop */
Packit Service 5956c7
void
Packit Service 5956c7
launch_thread_scheduler(thread_master_t *m)
Packit Service 5956c7
{
Packit Service 5956c7
// TODO - do this somewhere better
Packit Service 5956c7
	signal_set(SIGCHLD, thread_child_handler, m);
Packit Service 5956c7
Packit Service 5956c7
	process_threads(m);
Packit Service 5956c7
}
Packit Service 5956c7
Packit Service 5956c7
#ifdef THREAD_DUMP
Packit Service 5956c7
void
Packit Service 5956c7
register_scheduler_addresses(void)
Packit Service 5956c7
{
Packit Service 5956c7
#ifdef _WITH_SNMP_
Packit Service 5956c7
	register_thread_address("snmp_timeout_thread", snmp_timeout_thread);
Packit Service 5956c7
	register_thread_address("snmp_read_thread", snmp_read_thread);
Packit Service 5956c7
#endif
Packit Service 5956c7
	register_thread_address("thread_timerfd_handler", thread_timerfd_handler);
Packit Service 5956c7
Packit Service 5956c7
	register_signal_handler_address("thread_child_handler", thread_child_handler);
Packit Service 5956c7
}
Packit Service 5956c7
#endif