/* * daemon/daemon.c - collection of workers that handles requests. * * Copyright (c) 2007, NLnet Labs. All rights reserved. * * This software is open source. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * Neither the name of the NLNET LABS nor the names of its contributors may * be used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ /** * \file * * The daemon consists of global settings and a number of workers. */ /** buffer size for network connections */ #define BUFSZ 65552 #include "config.h" #include "daemon/daemon.h" #include "daemon/worker.h" #include "util/log.h" #include "util/config_file.h" #include "services/listen_dnsport.h" /* @@@ TODO remove */ #include "pthread.h" #include struct daemon* daemon_init() { struct daemon* daemon = (struct daemon*)calloc(1, sizeof(struct daemon)); if(!daemon) return NULL; lock_basic_init(&daemon->lock); daemon->need_to_exit = 0; return daemon; } int daemon_open_shared_ports(struct daemon* daemon) { log_assert(daemon); if(daemon->cfg->port == daemon->listening_port) return 1; listening_ports_free(daemon->ports); if(!(daemon->ports=listening_ports_open(daemon->cfg))) return 0; daemon->listening_port = daemon->cfg->port; return 1; } /** * Allocate empty worker structures. With backptr and thread-number, * from 0..numthread initialised. Used as user arguments to new threads. * @param daemon: the daemon with (new) config settings. */ static void daemon_create_workers(struct daemon* daemon) { int i; log_assert(daemon && daemon->cfg); daemon->num = daemon->cfg->num_threads; daemon->workers = (struct worker**)calloc((size_t)daemon->num, sizeof(struct worker*)); for(i=0; inum; i++) { if(!(daemon->workers[i] = worker_create(daemon, i))) fatal_exit("malloc failure"); } } /** * Close all pipes except for the numbered thread. * @param daemon: daemon to close pipes in. * @param thr: thread number 0..num-1 of thread to skip. */ void close_other_pipes(struct daemon* daemon, int thr) { int i; for(i=0; inum; i++) if(i!=thr) { if(daemon->workers[i]->cmd_send_fd != -1) { close(daemon->workers[i]->cmd_send_fd); daemon->workers[i]->cmd_send_fd = -1; } if(daemon->workers[i]->cmd_recv_fd != -1) { close(daemon->workers[i]->cmd_recv_fd); daemon->workers[i]->cmd_recv_fd = -1; } } } /** * Function to start one thread. * @param arg: user argument. * @return: void* user return value could be used for thread_join results. */ static void* thread_start(void* arg) { struct worker* worker = (struct worker*)arg; int num = worker->thread_num; ub_thread_blocksigs(); #if !defined(HAVE_PTHREAD) && !defined(HAVE_SOLARIS_THREADS) /* close pipe ends used by main */ close(worker->cmd_send_fd); worker->cmd_send_fd = -1; close_other_pipes(worker->daemon, worker->thread_num); #endif /* no threads */ if(!worker_init(worker, worker->daemon->cfg, worker->daemon->ports, BUFSZ, 0)) fatal_exit("Could not initialize thread #%d", num); worker_work(worker); return NULL; } /** * Fork and init the other threads. Main thread returns for special handling. * @param daemon: the daemon with other threads to fork. */ static void daemon_start_others(struct daemon* daemon) { int i; log_assert(daemon); log_info("start others"); /* skip i=0, is this thread */ for(i=1; inum; i++) { ub_thread_create(&daemon->workers[i]->thr_id, thread_start, daemon->workers[i]); #if !defined(HAVE_PTHREAD) && !defined(HAVE_SOLARIS_THREADS) /* close pipe end of child */ close(daemon->workers[i]->cmd_recv_fd); daemon->workers[i]->cmd_recv_fd = -1; #endif /* no threads */ } } /** * Stop the other threads. * @param daemon: the daemon with other threads. */ static void daemon_stop_others(struct daemon* daemon) { int i; log_assert(daemon); log_info("stop others"); /* skip i=0, is this thread */ /* use i=0 buffer for sending cmds; because we are #0 */ for(i=1; inum; i++) { log_info("killing thread %d", i); worker_send_cmd(daemon->workers[i], daemon->workers[0]->front->udp_buff, worker_cmd_quit); } /** wait for them to quit */ for(i=1; inum; i++) { /* join it to make sure its dead */ log_info("join %d", i); ub_thread_join(daemon->workers[i]->thr_id); log_info("join success %d", i); } } void daemon_fork(struct daemon* daemon) { log_assert(daemon); log_info("daemon_fork"); /* first create all the worker structures, so we can pass * them to the newly created threads. */ daemon_create_workers(daemon); /* Now create the threads and init the workers. * By the way, this is thread #0 (the main thread). */ daemon_start_others(daemon); /* Special handling for the main thread. This is the thread * that handles signals. */ if(!worker_init(daemon->workers[0], daemon->cfg, daemon->ports, BUFSZ, 1)) fatal_exit("Could not initialize main thread"); /* see if other threads have started correctly? */ /* Start resolver service on main thread. */ log_info("start of service (%s).", PACKAGE_STRING); worker_work(daemon->workers[0]); /* we exited! a signal happened! Stop other threads */ daemon_stop_others(daemon); if(daemon->workers[0]->need_to_restart) daemon->need_to_exit = 0; else daemon->need_to_exit = 1; } void daemon_cleanup(struct daemon* daemon) { int i; log_assert(daemon); for(i=0; inum; i++) worker_delete(daemon->workers[i]); free(daemon->workers); daemon->workers = NULL; daemon->num = 0; daemon->cfg = NULL; } void daemon_delete(struct daemon* daemon) { if(!daemon) return; listening_ports_free(daemon->ports); lock_basic_destroy(&daemon->lock); free(daemon->cwd); free(daemon->pidfile); free(daemon); }