1ea022d16SRodney W. Grimes /* 2ea022d16SRodney W. Grimes * Copyright (c) 1988, 1992 The University of Utah and the Center 3ea022d16SRodney W. Grimes * for Software Science (CSS). 4ea022d16SRodney W. Grimes * Copyright (c) 1992, 1993 5ea022d16SRodney W. Grimes * The Regents of the University of California. All rights reserved. 6ea022d16SRodney W. Grimes * 7ea022d16SRodney W. Grimes * This code is derived from software contributed to Berkeley by 8ea022d16SRodney W. Grimes * the Center for Software Science of the University of Utah Computer 9ea022d16SRodney W. Grimes * Science Department. CSS requests users of this software to return 10ea022d16SRodney W. Grimes * to css-dist@cs.utah.edu any improvements that they make and grant 11ea022d16SRodney W. Grimes * CSS redistribution rights. 12ea022d16SRodney W. Grimes * 13ea022d16SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 14ea022d16SRodney W. Grimes * modification, are permitted provided that the following conditions 15ea022d16SRodney W. Grimes * are met: 16ea022d16SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 17ea022d16SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 18ea022d16SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 19ea022d16SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 20ea022d16SRodney W. Grimes * documentation and/or other materials provided with the distribution. 21ea022d16SRodney W. Grimes * 3. All advertising materials mentioning features or use of this software 22ea022d16SRodney W. Grimes * must display the following acknowledgement: 23ea022d16SRodney W. Grimes * This product includes software developed by the University of 24ea022d16SRodney W. Grimes * California, Berkeley and its contributors. 25ea022d16SRodney W. Grimes * 4. Neither the name of the University nor the names of its contributors 26ea022d16SRodney W. Grimes * may be used to endorse or promote products derived from this software 27ea022d16SRodney W. Grimes * without specific prior written permission. 28ea022d16SRodney W. Grimes * 29ea022d16SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 30ea022d16SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 31ea022d16SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 32ea022d16SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 33ea022d16SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 34ea022d16SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 35ea022d16SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 36ea022d16SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 37ea022d16SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 38ea022d16SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 39ea022d16SRodney W. Grimes * SUCH DAMAGE. 40ea022d16SRodney W. Grimes * 41ea022d16SRodney W. Grimes * @(#)rbootd.c 8.2 (Berkeley) 2/22/94 4291477cc4SWarner Losh * $Id: rbootd.c,v 1.5 1997/02/22 14:21:58 peter Exp $ 43ea022d16SRodney W. Grimes * 44ea022d16SRodney W. Grimes * Utah $Hdr: rbootd.c 3.1 92/07/06$ 45ea022d16SRodney W. Grimes * Author: Jeff Forys, University of Utah CSS 46ea022d16SRodney W. Grimes */ 47ea022d16SRodney W. Grimes 48ea022d16SRodney W. Grimes #ifndef lint 49ea022d16SRodney W. Grimes static char copyright[] = 50ea022d16SRodney W. Grimes "@(#) Copyright (c) 1992, 1993\n\ 51ea022d16SRodney W. Grimes The Regents of the University of California. All rights reserved.\n"; 52ea022d16SRodney W. Grimes #endif /* not lint */ 53ea022d16SRodney W. Grimes 54ea022d16SRodney W. Grimes #ifndef lint 55ea022d16SRodney W. Grimes static char sccsid[] = "@(#)rbootd.c 8.2 (Berkeley) 2/22/94"; 56ea022d16SRodney W. Grimes #endif /* not lint */ 57ea022d16SRodney W. Grimes 58ea022d16SRodney W. Grimes #include <sys/param.h> 59ea022d16SRodney W. Grimes #include <sys/ioctl.h> 60ea022d16SRodney W. Grimes #include <sys/time.h> 61ea022d16SRodney W. Grimes 62ea022d16SRodney W. Grimes #include <ctype.h> 63ea022d16SRodney W. Grimes #include <errno.h> 64ea022d16SRodney W. Grimes #include <fcntl.h> 65ea022d16SRodney W. Grimes #include <signal.h> 66ea022d16SRodney W. Grimes #include <stdio.h> 67ea022d16SRodney W. Grimes #include <stdlib.h> 68ea022d16SRodney W. Grimes #include <string.h> 69ea022d16SRodney W. Grimes #include <syslog.h> 70ea022d16SRodney W. Grimes #include <unistd.h> 71ea022d16SRodney W. Grimes #include "defs.h" 72ea022d16SRodney W. Grimes 73ea022d16SRodney W. Grimes 74ea022d16SRodney W. Grimes /* fd mask macros (backward compatibility with 4.2BSD) */ 75ea022d16SRodney W. Grimes #ifndef FD_SET 76ea022d16SRodney W. Grimes #ifdef notdef 77ea022d16SRodney W. Grimes typedef struct fd_set { /* this should already be in 4.2 */ 78ea022d16SRodney W. Grimes int fds_bits[1]; 79ea022d16SRodney W. Grimes } fd_set; 80ea022d16SRodney W. Grimes #endif 81ea022d16SRodney W. Grimes #define FD_ZERO(p) ((p)->fds_bits[0] = 0) 82ea022d16SRodney W. Grimes #define FD_SET(n, p) ((p)->fds_bits[0] |= (1 << (n))) 83ea022d16SRodney W. Grimes #define FD_CLR(n, p) ((p)->fds_bits[0] &= ~(1 << (n))) 84ea022d16SRodney W. Grimes #define FD_ISSET(n, p) ((p)->fds_bits[0] & (1 << (n))) 85ea022d16SRodney W. Grimes #endif 86ea022d16SRodney W. Grimes 87ea022d16SRodney W. Grimes int 88ea022d16SRodney W. Grimes main(argc, argv) 89ea022d16SRodney W. Grimes int argc; 90ea022d16SRodney W. Grimes char *argv[]; 91ea022d16SRodney W. Grimes { 92ea022d16SRodney W. Grimes int c, fd, omask, maxfds; 93ea022d16SRodney W. Grimes fd_set rset; 94ea022d16SRodney W. Grimes 95ea022d16SRodney W. Grimes /* 96ea022d16SRodney W. Grimes * Find what name we are running under. 97ea022d16SRodney W. Grimes */ 98ea022d16SRodney W. Grimes ProgName = (ProgName = rindex(argv[0],'/')) ? ++ProgName : *argv; 99ea022d16SRodney W. Grimes 100ea022d16SRodney W. Grimes /* 101ea022d16SRodney W. Grimes * Close any open file descriptors. 102ea022d16SRodney W. Grimes * Temporarily leave stdin & stdout open for `-d', 103ea022d16SRodney W. Grimes * and stderr open for any pre-syslog error messages. 104ea022d16SRodney W. Grimes */ 105ea022d16SRodney W. Grimes { 106ea022d16SRodney W. Grimes int i, nfds = getdtablesize(); 107ea022d16SRodney W. Grimes 108ea022d16SRodney W. Grimes for (i = 0; i < nfds; i++) 109ea022d16SRodney W. Grimes if (i != fileno(stdin) && i != fileno(stdout) && 110ea022d16SRodney W. Grimes i != fileno(stderr)) 111ea022d16SRodney W. Grimes (void) close(i); 112ea022d16SRodney W. Grimes } 113ea022d16SRodney W. Grimes 114ea022d16SRodney W. Grimes /* 115ea022d16SRodney W. Grimes * Parse any arguments. 116ea022d16SRodney W. Grimes */ 11791477cc4SWarner Losh while ((c = getopt(argc, argv, "adi:")) != -1) 118ea022d16SRodney W. Grimes switch(c) { 119ea022d16SRodney W. Grimes case 'a': 120ea022d16SRodney W. Grimes BootAny++; 121ea022d16SRodney W. Grimes break; 122ea022d16SRodney W. Grimes case 'd': 123ea022d16SRodney W. Grimes DebugFlg++; 124ea022d16SRodney W. Grimes break; 125ea022d16SRodney W. Grimes case 'i': 126ea022d16SRodney W. Grimes IntfName = optarg; 127ea022d16SRodney W. Grimes break; 128ea022d16SRodney W. Grimes } 129ea022d16SRodney W. Grimes for (; optind < argc; optind++) { 130ea022d16SRodney W. Grimes if (ConfigFile == NULL) 131ea022d16SRodney W. Grimes ConfigFile = argv[optind]; 132ea022d16SRodney W. Grimes else { 133ea022d16SRodney W. Grimes fprintf(stderr, 134ea022d16SRodney W. Grimes "%s: too many config files (`%s' ignored)\n", 135ea022d16SRodney W. Grimes ProgName, argv[optind]); 136ea022d16SRodney W. Grimes } 137ea022d16SRodney W. Grimes } 138ea022d16SRodney W. Grimes 139ea022d16SRodney W. Grimes if (ConfigFile == NULL) /* use default config file */ 140ea022d16SRodney W. Grimes ConfigFile = DfltConfig; 141ea022d16SRodney W. Grimes 142ea022d16SRodney W. Grimes if (DebugFlg) { 143ea022d16SRodney W. Grimes DbgFp = stdout; /* output to stdout */ 144ea022d16SRodney W. Grimes 145ea022d16SRodney W. Grimes (void) signal(SIGUSR1, SIG_IGN); /* dont muck w/DbgFp */ 146ea022d16SRodney W. Grimes (void) signal(SIGUSR2, SIG_IGN); 147ea022d16SRodney W. Grimes } else { 148ea022d16SRodney W. Grimes (void) fclose(stdin); /* dont need these */ 149ea022d16SRodney W. Grimes (void) fclose(stdout); 150ea022d16SRodney W. Grimes 151ea022d16SRodney W. Grimes /* 152ea022d16SRodney W. Grimes * Fork off a child to do the work & exit. 153ea022d16SRodney W. Grimes */ 154ea022d16SRodney W. Grimes switch(fork()) { 155ea022d16SRodney W. Grimes case -1: /* fork failed */ 156ea022d16SRodney W. Grimes fprintf(stderr, "%s: ", ProgName); 157ea022d16SRodney W. Grimes perror("fork"); 158ea022d16SRodney W. Grimes Exit(0); 159ea022d16SRodney W. Grimes case 0: /* this is the CHILD */ 160ea022d16SRodney W. Grimes break; 161ea022d16SRodney W. Grimes default: /* this is the PARENT */ 162ea022d16SRodney W. Grimes _exit(0); 163ea022d16SRodney W. Grimes } 164ea022d16SRodney W. Grimes 165ea022d16SRodney W. Grimes /* 166ea022d16SRodney W. Grimes * Try to disassociate from the current tty. 167ea022d16SRodney W. Grimes */ 168ea022d16SRodney W. Grimes { 169ea022d16SRodney W. Grimes char *devtty = "/dev/tty"; 170ea022d16SRodney W. Grimes int i; 171ea022d16SRodney W. Grimes 172ea022d16SRodney W. Grimes if ((i = open(devtty, O_RDWR)) < 0) { 173ea022d16SRodney W. Grimes /* probably already disassociated */ 174ea022d16SRodney W. Grimes if (setpgrp(0, 0) < 0) { 175ea022d16SRodney W. Grimes fprintf(stderr, "%s: ", ProgName); 176ea022d16SRodney W. Grimes perror("setpgrp"); 177ea022d16SRodney W. Grimes } 178ea022d16SRodney W. Grimes } else { 179ea022d16SRodney W. Grimes if (ioctl(i, (u_long)TIOCNOTTY, (char *)0) < 0){ 180ea022d16SRodney W. Grimes fprintf(stderr, "%s: ", ProgName); 181ea022d16SRodney W. Grimes perror("ioctl"); 182ea022d16SRodney W. Grimes } 183ea022d16SRodney W. Grimes (void) close(i); 184ea022d16SRodney W. Grimes } 185ea022d16SRodney W. Grimes } 186ea022d16SRodney W. Grimes 187ea022d16SRodney W. Grimes (void) signal(SIGUSR1, DebugOn); 188ea022d16SRodney W. Grimes (void) signal(SIGUSR2, DebugOff); 189ea022d16SRodney W. Grimes } 190ea022d16SRodney W. Grimes 191ea022d16SRodney W. Grimes (void) fclose(stderr); /* finished with it */ 192ea022d16SRodney W. Grimes 193ea022d16SRodney W. Grimes #ifdef SYSLOG4_2 194ea022d16SRodney W. Grimes openlog(ProgName, LOG_PID); 195ea022d16SRodney W. Grimes #else 196ea022d16SRodney W. Grimes openlog(ProgName, LOG_PID, LOG_DAEMON); 197ea022d16SRodney W. Grimes #endif 198ea022d16SRodney W. Grimes 199ea022d16SRodney W. Grimes /* 200ea022d16SRodney W. Grimes * If no interface was specified, get one now. 201ea022d16SRodney W. Grimes * 202ea022d16SRodney W. Grimes * This is convoluted because we want to get the default interface 203ea022d16SRodney W. Grimes * name for the syslog("restarted") message. If BpfGetIntfName() 204ea022d16SRodney W. Grimes * runs into an error, it will return a syslog-able error message 205ea022d16SRodney W. Grimes * (in `errmsg') which will be displayed here. 206ea022d16SRodney W. Grimes */ 207ea022d16SRodney W. Grimes if (IntfName == NULL) { 208ea022d16SRodney W. Grimes char *errmsg; 209ea022d16SRodney W. Grimes 210ea022d16SRodney W. Grimes if ((IntfName = BpfGetIntfName(&errmsg)) == NULL) { 211ea022d16SRodney W. Grimes syslog(LOG_NOTICE, "restarted (??)"); 212ea022d16SRodney W. Grimes syslog(LOG_ERR, errmsg); 213ea022d16SRodney W. Grimes Exit(0); 214ea022d16SRodney W. Grimes } 215ea022d16SRodney W. Grimes } 216ea022d16SRodney W. Grimes 217ea022d16SRodney W. Grimes syslog(LOG_NOTICE, "restarted (%s)", IntfName); 218ea022d16SRodney W. Grimes 219ea022d16SRodney W. Grimes (void) signal(SIGHUP, ReConfig); 220ea022d16SRodney W. Grimes (void) signal(SIGINT, Exit); 221ea022d16SRodney W. Grimes (void) signal(SIGTERM, Exit); 222ea022d16SRodney W. Grimes 223ea022d16SRodney W. Grimes /* 224ea022d16SRodney W. Grimes * Grab our host name and pid. 225ea022d16SRodney W. Grimes */ 226ea022d16SRodney W. Grimes if (gethostname(MyHost, MAXHOSTNAMELEN) < 0) { 227ea022d16SRodney W. Grimes syslog(LOG_ERR, "gethostname: %m"); 228ea022d16SRodney W. Grimes Exit(0); 229ea022d16SRodney W. Grimes } 230ea022d16SRodney W. Grimes MyHost[MAXHOSTNAMELEN] = '\0'; 231ea022d16SRodney W. Grimes 232ea022d16SRodney W. Grimes MyPid = getpid(); 233ea022d16SRodney W. Grimes 234ea022d16SRodney W. Grimes /* 235ea022d16SRodney W. Grimes * Write proc's pid to a file. 236ea022d16SRodney W. Grimes */ 237ea022d16SRodney W. Grimes { 238ea022d16SRodney W. Grimes FILE *fp; 239ea022d16SRodney W. Grimes 240ea022d16SRodney W. Grimes if ((fp = fopen(PidFile, "w")) != NULL) { 241ea022d16SRodney W. Grimes (void) fprintf(fp, "%d\n", MyPid); 242ea022d16SRodney W. Grimes (void) fclose(fp); 243ea022d16SRodney W. Grimes } else { 244ea022d16SRodney W. Grimes syslog(LOG_WARNING, "fopen: failed (%s)", PidFile); 245ea022d16SRodney W. Grimes } 246ea022d16SRodney W. Grimes } 247ea022d16SRodney W. Grimes 248ea022d16SRodney W. Grimes /* 249ea022d16SRodney W. Grimes * All boot files are relative to the boot directory, we might 250ea022d16SRodney W. Grimes * as well chdir() there to make life easier. 251ea022d16SRodney W. Grimes */ 252ea022d16SRodney W. Grimes if (chdir(BootDir) < 0) { 253ea022d16SRodney W. Grimes syslog(LOG_ERR, "chdir: %m (%s)", BootDir); 254ea022d16SRodney W. Grimes Exit(0); 255ea022d16SRodney W. Grimes } 256ea022d16SRodney W. Grimes 257ea022d16SRodney W. Grimes /* 258ea022d16SRodney W. Grimes * Initial configuration. 259ea022d16SRodney W. Grimes */ 260ea022d16SRodney W. Grimes omask = sigblock(sigmask(SIGHUP)); /* prevent reconfig's */ 261ea022d16SRodney W. Grimes if (GetBootFiles() == 0) /* get list of boot files */ 262ea022d16SRodney W. Grimes Exit(0); 263ea022d16SRodney W. Grimes if (ParseConfig() == 0) /* parse config file */ 264ea022d16SRodney W. Grimes Exit(0); 265ea022d16SRodney W. Grimes 266ea022d16SRodney W. Grimes /* 267ea022d16SRodney W. Grimes * Open and initialize a BPF device for the appropriate interface. 268ea022d16SRodney W. Grimes * If an error is encountered, a message is displayed and Exit() 269ea022d16SRodney W. Grimes * is called. 270ea022d16SRodney W. Grimes */ 271ea022d16SRodney W. Grimes fd = BpfOpen(); 272ea022d16SRodney W. Grimes 273ea022d16SRodney W. Grimes (void) sigsetmask(omask); /* allow reconfig's */ 274ea022d16SRodney W. Grimes 275ea022d16SRodney W. Grimes /* 276ea022d16SRodney W. Grimes * Main loop: receive a packet, determine where it came from, 277ea022d16SRodney W. Grimes * and if we service this host, call routine to handle request. 278ea022d16SRodney W. Grimes */ 279ea022d16SRodney W. Grimes maxfds = fd + 1; 280ea022d16SRodney W. Grimes FD_ZERO(&rset); 281ea022d16SRodney W. Grimes FD_SET(fd, &rset); 282ea022d16SRodney W. Grimes for (;;) { 283ea022d16SRodney W. Grimes struct timeval timeout; 284ea022d16SRodney W. Grimes fd_set r; 285ea022d16SRodney W. Grimes int nsel; 286ea022d16SRodney W. Grimes 287ea022d16SRodney W. Grimes r = rset; 288ea022d16SRodney W. Grimes 289ea022d16SRodney W. Grimes if (RmpConns == NULL) { /* timeout isnt necessary */ 290ea022d16SRodney W. Grimes nsel = select(maxfds, &r, (fd_set *)0, (fd_set *)0, 291ea022d16SRodney W. Grimes (struct timeval *)0); 292ea022d16SRodney W. Grimes } else { 293ea022d16SRodney W. Grimes timeout.tv_sec = RMP_TIMEOUT; 294ea022d16SRodney W. Grimes timeout.tv_usec = 0; 295ea022d16SRodney W. Grimes nsel = select(maxfds, &r, (fd_set *)0, (fd_set *)0, 296ea022d16SRodney W. Grimes &timeout); 297ea022d16SRodney W. Grimes } 298ea022d16SRodney W. Grimes 299ea022d16SRodney W. Grimes if (nsel < 0) { 300ea022d16SRodney W. Grimes if (errno == EINTR) 301ea022d16SRodney W. Grimes continue; 302ea022d16SRodney W. Grimes syslog(LOG_ERR, "select: %m"); 303ea022d16SRodney W. Grimes Exit(0); 304ea022d16SRodney W. Grimes } else if (nsel == 0) { /* timeout */ 305ea022d16SRodney W. Grimes DoTimeout(); /* clear stale conns */ 306ea022d16SRodney W. Grimes continue; 307ea022d16SRodney W. Grimes } 308ea022d16SRodney W. Grimes 309ea022d16SRodney W. Grimes if (FD_ISSET(fd, &r)) { 310ea022d16SRodney W. Grimes RMPCONN rconn; 311ea022d16SRodney W. Grimes CLIENT *client, *FindClient(); 312ea022d16SRodney W. Grimes int doread = 1; 313ea022d16SRodney W. Grimes 314ea022d16SRodney W. Grimes while (BpfRead(&rconn, doread)) { 315ea022d16SRodney W. Grimes doread = 0; 316ea022d16SRodney W. Grimes 317ea022d16SRodney W. Grimes if (DbgFp != NULL) /* display packet */ 318ea022d16SRodney W. Grimes DispPkt(&rconn,DIR_RCVD); 319ea022d16SRodney W. Grimes 320ea022d16SRodney W. Grimes omask = sigblock(sigmask(SIGHUP)); 321ea022d16SRodney W. Grimes 322ea022d16SRodney W. Grimes /* 323ea022d16SRodney W. Grimes * If we do not restrict service, set the 324ea022d16SRodney W. Grimes * client to NULL (ProcessPacket() handles 325ea022d16SRodney W. Grimes * this). Otherwise, check that we can 326ea022d16SRodney W. Grimes * service this host; if not, log a message 327ea022d16SRodney W. Grimes * and ignore the packet. 328ea022d16SRodney W. Grimes */ 329ea022d16SRodney W. Grimes if (BootAny) { 330ea022d16SRodney W. Grimes client = NULL; 331ea022d16SRodney W. Grimes } else if ((client=FindClient(&rconn))==NULL) { 332ea022d16SRodney W. Grimes syslog(LOG_INFO, 333ea022d16SRodney W. Grimes "%s: boot packet ignored", 334ea022d16SRodney W. Grimes EnetStr(&rconn)); 335ea022d16SRodney W. Grimes (void) sigsetmask(omask); 336ea022d16SRodney W. Grimes continue; 337ea022d16SRodney W. Grimes } 338ea022d16SRodney W. Grimes 339ea022d16SRodney W. Grimes ProcessPacket(&rconn,client); 340ea022d16SRodney W. Grimes 341ea022d16SRodney W. Grimes (void) sigsetmask(omask); 342ea022d16SRodney W. Grimes } 343ea022d16SRodney W. Grimes } 344ea022d16SRodney W. Grimes } 345ea022d16SRodney W. Grimes } 346ea022d16SRodney W. Grimes 347ea022d16SRodney W. Grimes /* 348ea022d16SRodney W. Grimes ** DoTimeout -- Free any connections that have timed out. 349ea022d16SRodney W. Grimes ** 350ea022d16SRodney W. Grimes ** Parameters: 351ea022d16SRodney W. Grimes ** None. 352ea022d16SRodney W. Grimes ** 353ea022d16SRodney W. Grimes ** Returns: 354ea022d16SRodney W. Grimes ** Nothing. 355ea022d16SRodney W. Grimes ** 356ea022d16SRodney W. Grimes ** Side Effects: 357ea022d16SRodney W. Grimes ** - Timed out connections in `RmpConns' will be freed. 358ea022d16SRodney W. Grimes */ 359ea022d16SRodney W. Grimes void 360ea022d16SRodney W. Grimes DoTimeout() 361ea022d16SRodney W. Grimes { 362ea022d16SRodney W. Grimes register RMPCONN *rtmp; 363ea022d16SRodney W. Grimes struct timeval now; 364ea022d16SRodney W. Grimes 365ea022d16SRodney W. Grimes (void) gettimeofday(&now, (struct timezone *)0); 366ea022d16SRodney W. Grimes 367ea022d16SRodney W. Grimes /* 368ea022d16SRodney W. Grimes * For each active connection, if RMP_TIMEOUT seconds have passed 369ea022d16SRodney W. Grimes * since the last packet was sent, delete the connection. 370ea022d16SRodney W. Grimes */ 371ea022d16SRodney W. Grimes for (rtmp = RmpConns; rtmp != NULL; rtmp = rtmp->next) 372ea022d16SRodney W. Grimes if ((rtmp->tstamp.tv_sec + RMP_TIMEOUT) < now.tv_sec) { 373ea022d16SRodney W. Grimes syslog(LOG_WARNING, "%s: connection timed out (%u)", 374ea022d16SRodney W. Grimes EnetStr(rtmp), rtmp->rmp.r_type); 375ea022d16SRodney W. Grimes RemoveConn(rtmp); 376ea022d16SRodney W. Grimes } 377ea022d16SRodney W. Grimes } 378ea022d16SRodney W. Grimes 379ea022d16SRodney W. Grimes /* 380ea022d16SRodney W. Grimes ** FindClient -- Find client associated with a packet. 381ea022d16SRodney W. Grimes ** 382ea022d16SRodney W. Grimes ** Parameters: 383ea022d16SRodney W. Grimes ** rconn - the new packet. 384ea022d16SRodney W. Grimes ** 385ea022d16SRodney W. Grimes ** Returns: 386ea022d16SRodney W. Grimes ** Pointer to client info if found, NULL otherwise. 387ea022d16SRodney W. Grimes ** 388ea022d16SRodney W. Grimes ** Side Effects: 389ea022d16SRodney W. Grimes ** None. 390ea022d16SRodney W. Grimes ** 391ea022d16SRodney W. Grimes ** Warnings: 392ea022d16SRodney W. Grimes ** - This routine must be called with SIGHUP blocked since 393ea022d16SRodney W. Grimes ** a reconfigure can invalidate the information returned. 394ea022d16SRodney W. Grimes */ 395ea022d16SRodney W. Grimes 396ea022d16SRodney W. Grimes CLIENT * 397ea022d16SRodney W. Grimes FindClient(rconn) 398ea022d16SRodney W. Grimes register RMPCONN *rconn; 399ea022d16SRodney W. Grimes { 400ea022d16SRodney W. Grimes register CLIENT *ctmp; 401ea022d16SRodney W. Grimes 402ea022d16SRodney W. Grimes for (ctmp = Clients; ctmp != NULL; ctmp = ctmp->next) 403ea022d16SRodney W. Grimes if (bcmp((char *)&rconn->rmp.hp_hdr.saddr[0], 404ea022d16SRodney W. Grimes (char *)&ctmp->addr[0], RMP_ADDRLEN) == 0) 405ea022d16SRodney W. Grimes break; 406ea022d16SRodney W. Grimes 407ea022d16SRodney W. Grimes return(ctmp); 408ea022d16SRodney W. Grimes } 409ea022d16SRodney W. Grimes 410ea022d16SRodney W. Grimes /* 411ea022d16SRodney W. Grimes ** Exit -- Log an error message and exit. 412ea022d16SRodney W. Grimes ** 413ea022d16SRodney W. Grimes ** Parameters: 414ea022d16SRodney W. Grimes ** sig - caught signal (or zero if not dying on a signal). 415ea022d16SRodney W. Grimes ** 416ea022d16SRodney W. Grimes ** Returns: 417ea022d16SRodney W. Grimes ** Does not return. 418ea022d16SRodney W. Grimes ** 419ea022d16SRodney W. Grimes ** Side Effects: 420ea022d16SRodney W. Grimes ** - This process ceases to exist. 421ea022d16SRodney W. Grimes */ 422ea022d16SRodney W. Grimes void 423ea022d16SRodney W. Grimes Exit(sig) 424ea022d16SRodney W. Grimes int sig; 425ea022d16SRodney W. Grimes { 426ea022d16SRodney W. Grimes if (sig > 0) 427ea022d16SRodney W. Grimes syslog(LOG_ERR, "going down on signal %d", sig); 428ea022d16SRodney W. Grimes else 429ea022d16SRodney W. Grimes syslog(LOG_ERR, "going down with fatal error"); 430ea022d16SRodney W. Grimes BpfClose(); 431ea022d16SRodney W. Grimes exit(1); 432ea022d16SRodney W. Grimes } 433ea022d16SRodney W. Grimes 434ea022d16SRodney W. Grimes /* 435ea022d16SRodney W. Grimes ** ReConfig -- Get new list of boot files and reread config files. 436ea022d16SRodney W. Grimes ** 437ea022d16SRodney W. Grimes ** Parameters: 438ea022d16SRodney W. Grimes ** None. 439ea022d16SRodney W. Grimes ** 440ea022d16SRodney W. Grimes ** Returns: 441ea022d16SRodney W. Grimes ** Nothing. 442ea022d16SRodney W. Grimes ** 443ea022d16SRodney W. Grimes ** Side Effects: 444ea022d16SRodney W. Grimes ** - All active connections are dropped. 445ea022d16SRodney W. Grimes ** - List of boot-able files is changed. 446ea022d16SRodney W. Grimes ** - List of clients is changed. 447ea022d16SRodney W. Grimes ** 448ea022d16SRodney W. Grimes ** Warnings: 449ea022d16SRodney W. Grimes ** - This routine must be called with SIGHUP blocked. 450ea022d16SRodney W. Grimes */ 451ea022d16SRodney W. Grimes void 452ea022d16SRodney W. Grimes ReConfig(signo) 453ea022d16SRodney W. Grimes int signo; 454ea022d16SRodney W. Grimes { 455ea022d16SRodney W. Grimes syslog(LOG_NOTICE, "reconfiguring boot server"); 456ea022d16SRodney W. Grimes 457ea022d16SRodney W. Grimes FreeConns(); 458ea022d16SRodney W. Grimes 459ea022d16SRodney W. Grimes if (GetBootFiles() == 0) 460ea022d16SRodney W. Grimes Exit(0); 461ea022d16SRodney W. Grimes 462ea022d16SRodney W. Grimes if (ParseConfig() == 0) 463ea022d16SRodney W. Grimes Exit(0); 464ea022d16SRodney W. Grimes } 465ea022d16SRodney W. Grimes 466ea022d16SRodney W. Grimes /* 467ea022d16SRodney W. Grimes ** DebugOff -- Turn off debugging. 468ea022d16SRodney W. Grimes ** 469ea022d16SRodney W. Grimes ** Parameters: 470ea022d16SRodney W. Grimes ** None. 471ea022d16SRodney W. Grimes ** 472ea022d16SRodney W. Grimes ** Returns: 473ea022d16SRodney W. Grimes ** Nothing. 474ea022d16SRodney W. Grimes ** 475ea022d16SRodney W. Grimes ** Side Effects: 476ea022d16SRodney W. Grimes ** - Debug file is closed. 477ea022d16SRodney W. Grimes */ 478ea022d16SRodney W. Grimes void 479ea022d16SRodney W. Grimes DebugOff(signo) 480ea022d16SRodney W. Grimes int signo; 481ea022d16SRodney W. Grimes { 482ea022d16SRodney W. Grimes if (DbgFp != NULL) 483ea022d16SRodney W. Grimes (void) fclose(DbgFp); 484ea022d16SRodney W. Grimes 485ea022d16SRodney W. Grimes DbgFp = NULL; 486ea022d16SRodney W. Grimes } 487ea022d16SRodney W. Grimes 488ea022d16SRodney W. Grimes /* 489ea022d16SRodney W. Grimes ** DebugOn -- Turn on debugging. 490ea022d16SRodney W. Grimes ** 491ea022d16SRodney W. Grimes ** Parameters: 492ea022d16SRodney W. Grimes ** None. 493ea022d16SRodney W. Grimes ** 494ea022d16SRodney W. Grimes ** Returns: 495ea022d16SRodney W. Grimes ** Nothing. 496ea022d16SRodney W. Grimes ** 497ea022d16SRodney W. Grimes ** Side Effects: 498ea022d16SRodney W. Grimes ** - Debug file is opened/truncated if not already opened, 499ea022d16SRodney W. Grimes ** otherwise do nothing. 500ea022d16SRodney W. Grimes */ 501ea022d16SRodney W. Grimes void 502ea022d16SRodney W. Grimes DebugOn(signo) 503ea022d16SRodney W. Grimes int signo; 504ea022d16SRodney W. Grimes { 505ea022d16SRodney W. Grimes if (DbgFp == NULL) { 506ea022d16SRodney W. Grimes if ((DbgFp = fopen(DbgFile, "w")) == NULL) 507ea022d16SRodney W. Grimes syslog(LOG_ERR, "can't open debug file (%s)", DbgFile); 508ea022d16SRodney W. Grimes } 509ea022d16SRodney W. Grimes } 510