1*b7579f77SDag-Erling Smørgrav /* 2*b7579f77SDag-Erling Smørgrav * daemon/worker.h - worker that handles a pending list of requests. 3*b7579f77SDag-Erling Smørgrav * 4*b7579f77SDag-Erling Smørgrav * Copyright (c) 2007, NLnet Labs. All rights reserved. 5*b7579f77SDag-Erling Smørgrav * 6*b7579f77SDag-Erling Smørgrav * This software is open source. 7*b7579f77SDag-Erling Smørgrav * 8*b7579f77SDag-Erling Smørgrav * Redistribution and use in source and binary forms, with or without 9*b7579f77SDag-Erling Smørgrav * modification, are permitted provided that the following conditions 10*b7579f77SDag-Erling Smørgrav * are met: 11*b7579f77SDag-Erling Smørgrav * 12*b7579f77SDag-Erling Smørgrav * Redistributions of source code must retain the above copyright notice, 13*b7579f77SDag-Erling Smørgrav * this list of conditions and the following disclaimer. 14*b7579f77SDag-Erling Smørgrav * 15*b7579f77SDag-Erling Smørgrav * Redistributions in binary form must reproduce the above copyright notice, 16*b7579f77SDag-Erling Smørgrav * this list of conditions and the following disclaimer in the documentation 17*b7579f77SDag-Erling Smørgrav * and/or other materials provided with the distribution. 18*b7579f77SDag-Erling Smørgrav * 19*b7579f77SDag-Erling Smørgrav * Neither the name of the NLNET LABS nor the names of its contributors may 20*b7579f77SDag-Erling Smørgrav * be used to endorse or promote products derived from this software without 21*b7579f77SDag-Erling Smørgrav * specific prior written permission. 22*b7579f77SDag-Erling Smørgrav * 23*b7579f77SDag-Erling Smørgrav * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 24*b7579f77SDag-Erling Smørgrav * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 25*b7579f77SDag-Erling Smørgrav * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 26*b7579f77SDag-Erling Smørgrav * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE 27*b7579f77SDag-Erling Smørgrav * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 28*b7579f77SDag-Erling Smørgrav * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 29*b7579f77SDag-Erling Smørgrav * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 30*b7579f77SDag-Erling Smørgrav * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 31*b7579f77SDag-Erling Smørgrav * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 32*b7579f77SDag-Erling Smørgrav * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 33*b7579f77SDag-Erling Smørgrav * POSSIBILITY OF SUCH DAMAGE. 34*b7579f77SDag-Erling Smørgrav */ 35*b7579f77SDag-Erling Smørgrav 36*b7579f77SDag-Erling Smørgrav /** 37*b7579f77SDag-Erling Smørgrav * \file 38*b7579f77SDag-Erling Smørgrav * 39*b7579f77SDag-Erling Smørgrav * This file describes the worker structure that holds a list of 40*b7579f77SDag-Erling Smørgrav * pending requests and handles them. 41*b7579f77SDag-Erling Smørgrav */ 42*b7579f77SDag-Erling Smørgrav 43*b7579f77SDag-Erling Smørgrav #ifndef DAEMON_WORKER_H 44*b7579f77SDag-Erling Smørgrav #define DAEMON_WORKER_H 45*b7579f77SDag-Erling Smørgrav 46*b7579f77SDag-Erling Smørgrav #include "util/netevent.h" 47*b7579f77SDag-Erling Smørgrav #include "util/locks.h" 48*b7579f77SDag-Erling Smørgrav #include "util/alloc.h" 49*b7579f77SDag-Erling Smørgrav #include "util/data/msgreply.h" 50*b7579f77SDag-Erling Smørgrav #include "util/data/msgparse.h" 51*b7579f77SDag-Erling Smørgrav #include "daemon/stats.h" 52*b7579f77SDag-Erling Smørgrav #include "util/module.h" 53*b7579f77SDag-Erling Smørgrav struct listen_dnsport; 54*b7579f77SDag-Erling Smørgrav struct outside_network; 55*b7579f77SDag-Erling Smørgrav struct config_file; 56*b7579f77SDag-Erling Smørgrav struct daemon; 57*b7579f77SDag-Erling Smørgrav struct listen_port; 58*b7579f77SDag-Erling Smørgrav struct ub_randstate; 59*b7579f77SDag-Erling Smørgrav struct regional; 60*b7579f77SDag-Erling Smørgrav struct tube; 61*b7579f77SDag-Erling Smørgrav struct daemon_remote; 62*b7579f77SDag-Erling Smørgrav 63*b7579f77SDag-Erling Smørgrav /** worker commands */ 64*b7579f77SDag-Erling Smørgrav enum worker_commands { 65*b7579f77SDag-Erling Smørgrav /** make the worker quit */ 66*b7579f77SDag-Erling Smørgrav worker_cmd_quit, 67*b7579f77SDag-Erling Smørgrav /** obtain statistics */ 68*b7579f77SDag-Erling Smørgrav worker_cmd_stats, 69*b7579f77SDag-Erling Smørgrav /** obtain statistics without statsclear */ 70*b7579f77SDag-Erling Smørgrav worker_cmd_stats_noreset, 71*b7579f77SDag-Erling Smørgrav /** execute remote control command */ 72*b7579f77SDag-Erling Smørgrav worker_cmd_remote 73*b7579f77SDag-Erling Smørgrav }; 74*b7579f77SDag-Erling Smørgrav 75*b7579f77SDag-Erling Smørgrav /** 76*b7579f77SDag-Erling Smørgrav * Structure holding working information for unbound. 77*b7579f77SDag-Erling Smørgrav * Holds globally visible information. 78*b7579f77SDag-Erling Smørgrav */ 79*b7579f77SDag-Erling Smørgrav struct worker { 80*b7579f77SDag-Erling Smørgrav /** the thread number (in daemon array). First in struct for debug. */ 81*b7579f77SDag-Erling Smørgrav int thread_num; 82*b7579f77SDag-Erling Smørgrav /** global shared daemon structure */ 83*b7579f77SDag-Erling Smørgrav struct daemon* daemon; 84*b7579f77SDag-Erling Smørgrav /** thread id */ 85*b7579f77SDag-Erling Smørgrav ub_thread_t thr_id; 86*b7579f77SDag-Erling Smørgrav /** pipe, for commands for this worker */ 87*b7579f77SDag-Erling Smørgrav struct tube* cmd; 88*b7579f77SDag-Erling Smørgrav /** the event base this worker works with */ 89*b7579f77SDag-Erling Smørgrav struct comm_base* base; 90*b7579f77SDag-Erling Smørgrav /** the frontside listening interface where request events come in */ 91*b7579f77SDag-Erling Smørgrav struct listen_dnsport* front; 92*b7579f77SDag-Erling Smørgrav /** the backside outside network interface to the auth servers */ 93*b7579f77SDag-Erling Smørgrav struct outside_network* back; 94*b7579f77SDag-Erling Smørgrav /** ports to be used by this worker. */ 95*b7579f77SDag-Erling Smørgrav int* ports; 96*b7579f77SDag-Erling Smørgrav /** number of ports for this worker */ 97*b7579f77SDag-Erling Smørgrav int numports; 98*b7579f77SDag-Erling Smørgrav /** the signal handler */ 99*b7579f77SDag-Erling Smørgrav struct comm_signal* comsig; 100*b7579f77SDag-Erling Smørgrav /** commpoint to listen to commands. */ 101*b7579f77SDag-Erling Smørgrav struct comm_point* cmd_com; 102*b7579f77SDag-Erling Smørgrav /** timer for statistics */ 103*b7579f77SDag-Erling Smørgrav struct comm_timer* stat_timer; 104*b7579f77SDag-Erling Smørgrav 105*b7579f77SDag-Erling Smørgrav /** random() table for this worker. */ 106*b7579f77SDag-Erling Smørgrav struct ub_randstate* rndstate; 107*b7579f77SDag-Erling Smørgrav /** do we need to restart or quit (on signal) */ 108*b7579f77SDag-Erling Smørgrav int need_to_exit; 109*b7579f77SDag-Erling Smørgrav /** allocation cache for this thread */ 110*b7579f77SDag-Erling Smørgrav struct alloc_cache alloc; 111*b7579f77SDag-Erling Smørgrav /** per thread statistics */ 112*b7579f77SDag-Erling Smørgrav struct server_stats stats; 113*b7579f77SDag-Erling Smørgrav /** thread scratch regional */ 114*b7579f77SDag-Erling Smørgrav struct regional* scratchpad; 115*b7579f77SDag-Erling Smørgrav 116*b7579f77SDag-Erling Smørgrav /** module environment passed to modules, changed for this thread */ 117*b7579f77SDag-Erling Smørgrav struct module_env env; 118*b7579f77SDag-Erling Smørgrav }; 119*b7579f77SDag-Erling Smørgrav 120*b7579f77SDag-Erling Smørgrav /** 121*b7579f77SDag-Erling Smørgrav * Create the worker structure. Bare bones version, zeroed struct, 122*b7579f77SDag-Erling Smørgrav * with backpointers only. Use worker_init on it later. 123*b7579f77SDag-Erling Smørgrav * @param daemon: the daemon that this worker thread is part of. 124*b7579f77SDag-Erling Smørgrav * @param id: the thread number from 0.. numthreads-1. 125*b7579f77SDag-Erling Smørgrav * @param ports: the ports it is allowed to use, array. 126*b7579f77SDag-Erling Smørgrav * @param n: the number of ports. 127*b7579f77SDag-Erling Smørgrav * @return: the new worker or NULL on alloc failure. 128*b7579f77SDag-Erling Smørgrav */ 129*b7579f77SDag-Erling Smørgrav struct worker* worker_create(struct daemon* daemon, int id, int* ports, int n); 130*b7579f77SDag-Erling Smørgrav 131*b7579f77SDag-Erling Smørgrav /** 132*b7579f77SDag-Erling Smørgrav * Initialize worker. 133*b7579f77SDag-Erling Smørgrav * Allocates event base, listens to ports 134*b7579f77SDag-Erling Smørgrav * @param worker: worker to initialize, created with worker_create. 135*b7579f77SDag-Erling Smørgrav * @param cfg: configuration settings. 136*b7579f77SDag-Erling Smørgrav * @param ports: list of shared query ports. 137*b7579f77SDag-Erling Smørgrav * @param do_sigs: if true, worker installs signal handlers. 138*b7579f77SDag-Erling Smørgrav * @return: false on error. 139*b7579f77SDag-Erling Smørgrav */ 140*b7579f77SDag-Erling Smørgrav int worker_init(struct worker* worker, struct config_file *cfg, 141*b7579f77SDag-Erling Smørgrav struct listen_port* ports, int do_sigs); 142*b7579f77SDag-Erling Smørgrav 143*b7579f77SDag-Erling Smørgrav /** 144*b7579f77SDag-Erling Smørgrav * Make worker work. 145*b7579f77SDag-Erling Smørgrav */ 146*b7579f77SDag-Erling Smørgrav void worker_work(struct worker* worker); 147*b7579f77SDag-Erling Smørgrav 148*b7579f77SDag-Erling Smørgrav /** 149*b7579f77SDag-Erling Smørgrav * Delete worker. 150*b7579f77SDag-Erling Smørgrav */ 151*b7579f77SDag-Erling Smørgrav void worker_delete(struct worker* worker); 152*b7579f77SDag-Erling Smørgrav 153*b7579f77SDag-Erling Smørgrav /** 154*b7579f77SDag-Erling Smørgrav * Send a command to a worker. Uses blocking writes. 155*b7579f77SDag-Erling Smørgrav * @param worker: worker to send command to. 156*b7579f77SDag-Erling Smørgrav * @param cmd: command to send. 157*b7579f77SDag-Erling Smørgrav */ 158*b7579f77SDag-Erling Smørgrav void worker_send_cmd(struct worker* worker, enum worker_commands cmd); 159*b7579f77SDag-Erling Smørgrav 160*b7579f77SDag-Erling Smørgrav /** 161*b7579f77SDag-Erling Smørgrav * Worker signal handler function. User argument is the worker itself. 162*b7579f77SDag-Erling Smørgrav * @param sig: signal number. 163*b7579f77SDag-Erling Smørgrav * @param arg: the worker (main worker) that handles signals. 164*b7579f77SDag-Erling Smørgrav */ 165*b7579f77SDag-Erling Smørgrav void worker_sighandler(int sig, void* arg); 166*b7579f77SDag-Erling Smørgrav 167*b7579f77SDag-Erling Smørgrav /** 168*b7579f77SDag-Erling Smørgrav * Worker service routine to send serviced queries to authoritative servers. 169*b7579f77SDag-Erling Smørgrav * @param qname: query name. (host order) 170*b7579f77SDag-Erling Smørgrav * @param qnamelen: length in bytes of qname, including trailing 0. 171*b7579f77SDag-Erling Smørgrav * @param qtype: query type. (host order) 172*b7579f77SDag-Erling Smørgrav * @param qclass: query class. (host order) 173*b7579f77SDag-Erling Smørgrav * @param flags: host order flags word, with opcode and CD bit. 174*b7579f77SDag-Erling Smørgrav * @param dnssec: if set, EDNS record will have DO bit set. 175*b7579f77SDag-Erling Smørgrav * @param want_dnssec: signatures needed. 176*b7579f77SDag-Erling Smørgrav * @param addr: where to. 177*b7579f77SDag-Erling Smørgrav * @param addrlen: length of addr. 178*b7579f77SDag-Erling Smørgrav * @param zone: wireformat dname of the zone. 179*b7579f77SDag-Erling Smørgrav * @param zonelen: length of zone name. 180*b7579f77SDag-Erling Smørgrav * @param q: wich query state to reactivate upon return. 181*b7579f77SDag-Erling Smørgrav * @return: false on failure (memory or socket related). no query was 182*b7579f77SDag-Erling Smørgrav * sent. 183*b7579f77SDag-Erling Smørgrav */ 184*b7579f77SDag-Erling Smørgrav struct outbound_entry* worker_send_query(uint8_t* qname, size_t qnamelen, 185*b7579f77SDag-Erling Smørgrav uint16_t qtype, uint16_t qclass, uint16_t flags, int dnssec, 186*b7579f77SDag-Erling Smørgrav int want_dnssec, struct sockaddr_storage* addr, socklen_t addrlen, 187*b7579f77SDag-Erling Smørgrav uint8_t* zone, size_t zonelen, struct module_qstate* q); 188*b7579f77SDag-Erling Smørgrav 189*b7579f77SDag-Erling Smørgrav /** 190*b7579f77SDag-Erling Smørgrav * process control messages from the main thread. Frees the control 191*b7579f77SDag-Erling Smørgrav * command message. 192*b7579f77SDag-Erling Smørgrav * @param tube: tube control message came on. 193*b7579f77SDag-Erling Smørgrav * @param msg: message contents. Is freed. 194*b7579f77SDag-Erling Smørgrav * @param len: length of message. 195*b7579f77SDag-Erling Smørgrav * @param error: if error (NETEVENT_*) happened. 196*b7579f77SDag-Erling Smørgrav * @param arg: user argument 197*b7579f77SDag-Erling Smørgrav */ 198*b7579f77SDag-Erling Smørgrav void worker_handle_control_cmd(struct tube* tube, uint8_t* msg, size_t len, 199*b7579f77SDag-Erling Smørgrav int error, void* arg); 200*b7579f77SDag-Erling Smørgrav 201*b7579f77SDag-Erling Smørgrav /** handles callbacks from listening event interface */ 202*b7579f77SDag-Erling Smørgrav int worker_handle_request(struct comm_point* c, void* arg, int error, 203*b7579f77SDag-Erling Smørgrav struct comm_reply* repinfo); 204*b7579f77SDag-Erling Smørgrav 205*b7579f77SDag-Erling Smørgrav /** process incoming replies from the network */ 206*b7579f77SDag-Erling Smørgrav int worker_handle_reply(struct comm_point* c, void* arg, int error, 207*b7579f77SDag-Erling Smørgrav struct comm_reply* reply_info); 208*b7579f77SDag-Erling Smørgrav 209*b7579f77SDag-Erling Smørgrav /** process incoming serviced query replies from the network */ 210*b7579f77SDag-Erling Smørgrav int worker_handle_service_reply(struct comm_point* c, void* arg, int error, 211*b7579f77SDag-Erling Smørgrav struct comm_reply* reply_info); 212*b7579f77SDag-Erling Smørgrav 213*b7579f77SDag-Erling Smørgrav /** cleanup the cache to remove all rrset IDs from it, arg is worker */ 214*b7579f77SDag-Erling Smørgrav void worker_alloc_cleanup(void* arg); 215*b7579f77SDag-Erling Smørgrav 216*b7579f77SDag-Erling Smørgrav /** 217*b7579f77SDag-Erling Smørgrav * Init worker stats - includes server_stats_init, outside network and mesh. 218*b7579f77SDag-Erling Smørgrav * @param worker: the worker to init 219*b7579f77SDag-Erling Smørgrav */ 220*b7579f77SDag-Erling Smørgrav void worker_stats_clear(struct worker* worker); 221*b7579f77SDag-Erling Smørgrav 222*b7579f77SDag-Erling Smørgrav /** statistics timer callback handler */ 223*b7579f77SDag-Erling Smørgrav void worker_stat_timer_cb(void* arg); 224*b7579f77SDag-Erling Smørgrav 225*b7579f77SDag-Erling Smørgrav /** probe timer callback handler */ 226*b7579f77SDag-Erling Smørgrav void worker_probe_timer_cb(void* arg); 227*b7579f77SDag-Erling Smørgrav 228*b7579f77SDag-Erling Smørgrav /** start accept callback handler */ 229*b7579f77SDag-Erling Smørgrav void worker_start_accept(void* arg); 230*b7579f77SDag-Erling Smørgrav 231*b7579f77SDag-Erling Smørgrav /** stop accept callback handler */ 232*b7579f77SDag-Erling Smørgrav void worker_stop_accept(void* arg); 233*b7579f77SDag-Erling Smørgrav 234*b7579f77SDag-Erling Smørgrav #endif /* DAEMON_WORKER_H */ 235