mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-01 12:19:28 +00:00
15f7f56992
Obtained from: ftp.isc.org
2219 lines
53 KiB
C
2219 lines
53 KiB
C
#if !defined(lint) && !defined(SABER)
|
|
static char sccsid[] = "@(#)ns_main.c 4.55 (Berkeley) 7/1/91";
|
|
static char rcsid[] = "$Id: ns_main.c,v 8.67 1998/04/28 19:17:46 halley Exp $";
|
|
#endif /* not lint */
|
|
|
|
/*
|
|
* Copyright (c) 1986, 1989, 1990
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. 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.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the University of
|
|
* California, Berkeley and its contributors.
|
|
* 4. Neither the name of the University 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 REGENTS 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.
|
|
*/
|
|
|
|
/*
|
|
* Portions Copyright (c) 1993 by Digital Equipment Corporation.
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies, and that
|
|
* the name of Digital Equipment Corporation not be used in advertising or
|
|
* publicity pertaining to distribution of the document or software without
|
|
* specific, written prior permission.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND DIGITAL EQUIPMENT CORP. DISCLAIMS ALL
|
|
* WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL DIGITAL EQUIPMENT
|
|
* CORPORATION BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
|
|
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
* SOFTWARE.
|
|
*/
|
|
|
|
/*
|
|
* Portions Copyright (c) 1996, 1997 by Internet Software Consortium.
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
|
|
* ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
|
|
* CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
|
|
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
* SOFTWARE.
|
|
*/
|
|
|
|
#if !defined(lint) && !defined(SABER)
|
|
char copyright[] =
|
|
"@(#) Copyright (c) 1986, 1989, 1990 The Regents of the University of California.\n\
|
|
portions Copyright (c) 1993 Digital Equipment Corporation\n\
|
|
portions Copyright (c) 1995, 1996, 1997 Internet Software Consortium\n\
|
|
All rights reserved.\n";
|
|
#endif /* not lint */
|
|
|
|
/*
|
|
* Internet Name server (see RCF1035 & others).
|
|
*/
|
|
|
|
#include "port_before.h"
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/file.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/socket.h>
|
|
#ifdef SVR4 /* XXX */
|
|
# include <sys/sockio.h>
|
|
#else
|
|
# include <sys/mbuf.h>
|
|
#endif
|
|
|
|
#include <netinet/in.h>
|
|
#include <net/route.h>
|
|
#include <net/if.h>
|
|
#include <arpa/nameser.h>
|
|
#include <arpa/inet.h>
|
|
|
|
#include <ctype.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <grp.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <signal.h>
|
|
#include <netdb.h>
|
|
#include <pwd.h>
|
|
#include <resolv.h>
|
|
#include <string.h>
|
|
#include <syslog.h>
|
|
#include <time.h>
|
|
#include <unistd.h>
|
|
|
|
#include <isc/eventlib.h>
|
|
#include <isc/logging.h>
|
|
#include <isc/memcluster.h>
|
|
#include <isc/list.h>
|
|
|
|
#include "port_after.h"
|
|
|
|
#ifdef HAVE_GETRUSAGE /* XXX */
|
|
#include <sys/resource.h>
|
|
#endif
|
|
|
|
#define MAIN_PROGRAM
|
|
#include "named.h"
|
|
#undef MAIN_PROGRAM
|
|
|
|
/* list of interfaces */
|
|
static LIST(struct _interface) iflist;
|
|
static int iflist_initialized = 0;
|
|
|
|
|
|
static const int drbufsize = 8 * 1024, /* UDP rcv buf size */
|
|
dsbufsize = 16 * 1024, /* UDP snd buf size */
|
|
sbufsize = 16 * 1024; /* TCP snd buf size */
|
|
|
|
static u_int16_t nsid_state;
|
|
static int needs;
|
|
|
|
static struct qstream *sq_add(void);
|
|
static int opensocket_d(interface *),
|
|
opensocket_s(interface *);
|
|
static void sq_query(struct qstream *),
|
|
dq_remove(interface *),
|
|
ns_handle_needs(void);
|
|
static int sq_dowrite(struct qstream *);
|
|
static void use_desired_debug(void);
|
|
static void stream_write(evContext, void *, int, int);
|
|
|
|
static interface * if_find(struct in_addr, u_int16_t port);
|
|
|
|
static int sq_here(struct qstream *);
|
|
|
|
static void stream_accept(evContext, void *, int,
|
|
const void *, int,
|
|
const void *, int),
|
|
stream_getlen(evContext, void *, int, int),
|
|
stream_getmsg(evContext, void *, int, int),
|
|
datagram_read(evContext, void *, int, int),
|
|
dispatch_message(u_char *, int, int,
|
|
struct qstream *,
|
|
struct sockaddr_in, int,
|
|
interface *);
|
|
static void stream_send(evContext, void *, int,
|
|
const void *, int,
|
|
const void *, int);
|
|
static void init_signals(void);
|
|
static void set_signal_handler(int, SIG_FN (*)());
|
|
static int only_digits(const char *);
|
|
|
|
static void
|
|
usage() {
|
|
fprintf(stderr,
|
|
"Usage: named [-d #] [-q] [-r] [-f] [-p port] [[-b|-c] configfile]\n");
|
|
#ifdef CAN_CHANGE_ID
|
|
fprintf(stderr,
|
|
" [-u (username|uid)] [-g (groupname|gid)]\n");
|
|
#endif
|
|
#ifdef HAVE_CHROOT
|
|
fprintf(stderr,
|
|
" [-t directory]\n");
|
|
#endif
|
|
exit(1);
|
|
}
|
|
|
|
static char bad_p_option[] =
|
|
"-p remote/local obsolete; use 'listen-on' in config file to specify local";
|
|
|
|
static char bad_directory[] = "chdir failed for directory '%s': %s";
|
|
|
|
/*ARGSUSED*/
|
|
int
|
|
main(int argc, char *argv[], char *envp[]) {
|
|
int n, udpcnt;
|
|
char *arg;
|
|
struct qstream *sp;
|
|
interface *ifp;
|
|
const int on = 1;
|
|
int rfd, size, len, debug_option;
|
|
char **argp, *p;
|
|
int ch;
|
|
FILE *fp; /* file descriptor for pid file */
|
|
struct passwd *pw;
|
|
struct group *gr;
|
|
#ifdef HAVE_GETRUSAGE
|
|
struct rlimit rl;
|
|
#endif
|
|
|
|
user_id = getuid();
|
|
group_id = getgid();
|
|
|
|
ns_port = htons(NAMESERVER_PORT);
|
|
desired_debug = debug;
|
|
|
|
/* BSD has a better random number generator but it's not clear
|
|
* that we need it here.
|
|
*/
|
|
gettime(&tt);
|
|
srand(((unsigned)getpid()) + (unsigned)tt.tv_usec);
|
|
|
|
(void) umask(022);
|
|
|
|
while ((ch = getopt(argc, argv, "b:c:d:g:p:t:u:w:qrf")) != EOF) {
|
|
switch (ch) {
|
|
case 'b':
|
|
case 'c':
|
|
if (conffile != NULL)
|
|
freestr(conffile);
|
|
conffile = savestr(optarg, 1);
|
|
break;
|
|
|
|
case 'd':
|
|
desired_debug = atoi(optarg);
|
|
if (desired_debug <= 0)
|
|
desired_debug = 1;
|
|
break;
|
|
|
|
case 'p':
|
|
/* use nonstandard port number.
|
|
* usage: -p remote/local
|
|
* remote is the port number to which
|
|
* we send queries. local is the port
|
|
* on which we listen for queries.
|
|
* local defaults to same as remote.
|
|
*/
|
|
ns_port = htons((u_int16_t) atoi(optarg));
|
|
p = strchr(optarg, '/');
|
|
if (p) {
|
|
syslog(LOG_WARNING, bad_p_option);
|
|
fprintf(stderr, bad_p_option);
|
|
fputc('\n', stderr);
|
|
}
|
|
break;
|
|
|
|
case 'w':
|
|
if (chdir(optarg) < 0) {
|
|
syslog(LOG_CRIT, bad_directory, optarg,
|
|
strerror(errno));
|
|
fprintf(stderr, bad_directory, optarg,
|
|
strerror(errno));
|
|
fputc('\n', stderr);
|
|
exit(1);
|
|
}
|
|
break;
|
|
#ifdef QRYLOG
|
|
case 'q':
|
|
qrylog = 1;
|
|
break;
|
|
#endif
|
|
|
|
case 'r':
|
|
ns_setoption(OPTION_NORECURSE);
|
|
break;
|
|
|
|
case 'f':
|
|
foreground = 1;
|
|
break;
|
|
|
|
case 't':
|
|
chroot_dir = savestr(optarg, 1);
|
|
break;
|
|
|
|
#ifdef CAN_CHANGE_ID
|
|
case 'u':
|
|
user_name = savestr(optarg, 1);
|
|
if (only_digits(user_name))
|
|
user_id = atoi(user_name);
|
|
else {
|
|
pw = getpwnam(user_name);
|
|
if (pw == NULL) {
|
|
fprintf(stderr,
|
|
"user \"%s\" unknown\n",
|
|
user_name);
|
|
exit(1);
|
|
}
|
|
user_id = pw->pw_uid;
|
|
if (group_name == NULL) {
|
|
char name[256];
|
|
|
|
sprintf(name, "%lu",
|
|
(u_long)pw->pw_gid);
|
|
group_name = savestr(name, 1);
|
|
group_id = pw->pw_gid;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'g':
|
|
if (group_name != NULL)
|
|
freestr(group_name);
|
|
group_name = savestr(optarg, 1);
|
|
if (only_digits(group_name))
|
|
group_id = atoi(group_name);
|
|
else {
|
|
gr = getgrnam(group_name);
|
|
if (gr == NULL) {
|
|
fprintf(stderr,
|
|
"group \"%s\" unknown\n",
|
|
group_name);
|
|
exit(1);
|
|
}
|
|
group_id = gr->gr_gid;
|
|
}
|
|
break;
|
|
#endif /* CAN_CHANGE_ID */
|
|
|
|
case '?':
|
|
default:
|
|
usage();
|
|
}
|
|
}
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (argc) {
|
|
if (conffile != NULL)
|
|
freestr(conffile);
|
|
conffile = savestr(*argv, 1);
|
|
argc--, argv++;
|
|
}
|
|
if (argc)
|
|
usage();
|
|
|
|
if (conffile == NULL)
|
|
conffile = savestr(_PATH_CONF, 1);
|
|
|
|
/*
|
|
* Make sure we don't inherit any open descriptors
|
|
* other than those that daemon() can deal with.
|
|
*/
|
|
for (n = sysconf(_SC_OPEN_MAX) - 1; n >= 0; n--)
|
|
if (n != STDIN_FILENO &&
|
|
n != STDOUT_FILENO &&
|
|
n != STDERR_FILENO)
|
|
(void) close(n);
|
|
|
|
/*
|
|
* Chroot if desired.
|
|
*/
|
|
if (chroot_dir != NULL) {
|
|
#ifdef HAVE_CHROOT
|
|
if (chroot(chroot_dir) < 0) {
|
|
fprintf(stderr, "chroot %s failed: %s\n", chroot_dir,
|
|
strerror(errno));
|
|
exit(1);
|
|
}
|
|
if (chdir("/") < 0) {
|
|
fprintf(stderr, "chdir(\"/\") failed: %s\n",
|
|
strerror(errno));
|
|
exit(1);
|
|
}
|
|
#else
|
|
fprintf(stderr, "warning: chroot() not available\n");
|
|
freestr(chroot_dir);
|
|
chroot_dir = NULL;
|
|
#endif
|
|
}
|
|
|
|
/* Establish global event context. */
|
|
evCreate(&ev);
|
|
|
|
/*
|
|
* Set up logging.
|
|
*/
|
|
n = LOG_PID;
|
|
#ifdef LOG_NOWAIT
|
|
n |= LOG_NOWAIT;
|
|
#endif
|
|
#ifdef LOG_NDELAY
|
|
n |= LOG_NDELAY;
|
|
#endif
|
|
#if defined(LOG_CONS) && defined(USE_LOG_CONS)
|
|
n |= LOG_CONS;
|
|
#endif
|
|
#ifdef SYSLOG_42BSD
|
|
openlog("named", n);
|
|
#else
|
|
openlog("named", n, LOG_DAEMON);
|
|
#endif
|
|
|
|
init_logging();
|
|
set_assertion_failure_callback(ns_assertion_failed);
|
|
|
|
#ifdef DEBUG
|
|
use_desired_debug();
|
|
#endif
|
|
|
|
init_signals();
|
|
|
|
ns_notice(ns_log_default, "starting. %s", Version);
|
|
|
|
_res.options &= ~(RES_DEFNAMES | RES_DNSRCH | RES_RECURSE);
|
|
|
|
nsid_init();
|
|
|
|
/*
|
|
* Initialize and load database.
|
|
*/
|
|
gettime(&tt);
|
|
buildservicelist();
|
|
buildprotolist();
|
|
ns_init(conffile);
|
|
time(&boottime);
|
|
resettime = boottime;
|
|
|
|
/*
|
|
* Fork and go into background now that
|
|
* we've done any slow initialization
|
|
* and are ready to answer queries.
|
|
*/
|
|
|
|
if (foreground == 0) {
|
|
if (daemon(1, 0))
|
|
ns_panic(ns_log_default, 1, "daemon: %s",
|
|
strerror(errno));
|
|
update_pid_file();
|
|
}
|
|
|
|
/* Check that udp checksums are on. */
|
|
ns_udp();
|
|
|
|
/*
|
|
* We waited until now to log this because we wanted logging to
|
|
* be set up the way the user prefers.
|
|
*/
|
|
if (chroot_dir != NULL)
|
|
ns_info(ns_log_security, "chrooted to %s", chroot_dir);
|
|
|
|
#ifdef CAN_CHANGE_ID
|
|
/*
|
|
* Set user and group if desired.
|
|
*/
|
|
if (group_name != NULL) {
|
|
if (setgid(group_id) < 0)
|
|
ns_panic(ns_log_security, 1, "setgid(%s): %s",
|
|
group_name, strerror(errno));
|
|
ns_info(ns_log_security, "group = %s", group_name);
|
|
}
|
|
if (user_name != NULL) {
|
|
if (getuid() == 0 && initgroups(user_name, group_id) < 0)
|
|
ns_panic(ns_log_security, 1, "initgroups(%s, %d): %s",
|
|
user_name, (int)group_id, strerror(errno));
|
|
endgrent();
|
|
endpwent();
|
|
if (setuid(user_id) < 0)
|
|
ns_panic(ns_log_security, 1, "setuid(%s): %s",
|
|
user_name, strerror(errno));
|
|
ns_info(ns_log_security, "user = %s", user_name);
|
|
}
|
|
#endif /* CAN_CHANGE_ID */
|
|
|
|
ns_notice(ns_log_default, "Ready to answer queries.");
|
|
gettime(&tt);
|
|
prime_cache();
|
|
for (;;) {
|
|
evEvent event;
|
|
|
|
if (needs)
|
|
ns_handle_needs();
|
|
INSIST_ERR(evGetNext(ev, &event, EV_WAIT) != -1);
|
|
INSIST_ERR(evDispatch(ev, event) != -1);
|
|
}
|
|
/* NOTREACHED */
|
|
return (0);
|
|
}
|
|
|
|
#ifndef IP_OPT_BUF_SIZE
|
|
/* arbitrary size */
|
|
#define IP_OPT_BUF_SIZE 50
|
|
#endif
|
|
|
|
static void
|
|
stream_accept(evContext lev, void *uap, int rfd,
|
|
const void *lav, int lalen,
|
|
const void *rav, int ralen)
|
|
{
|
|
interface *ifp = uap;
|
|
struct qstream *sp;
|
|
struct iovec iov;
|
|
int n, len;
|
|
const int on = 1;
|
|
#ifdef IP_OPTIONS /* XXX */
|
|
u_char ip_opts[IP_OPT_BUF_SIZE];
|
|
#endif
|
|
const struct sockaddr_in *la, *ra;
|
|
|
|
la = (const struct sockaddr_in *)lav;
|
|
ra = (const struct sockaddr_in *)rav;
|
|
|
|
INSIST(ifp != NULL);
|
|
|
|
if (rfd < 0) {
|
|
switch (errno) {
|
|
case EINTR:
|
|
case EAGAIN:
|
|
#if (EWOULDBLOCK != EAGAIN)
|
|
case EWOULDBLOCK:
|
|
#endif
|
|
case ECONNABORTED:
|
|
#ifdef EPROTO
|
|
case EPROTO:
|
|
#endif
|
|
case EHOSTUNREACH:
|
|
case EHOSTDOWN:
|
|
case ENETUNREACH:
|
|
case ENETDOWN:
|
|
case ECONNREFUSED:
|
|
#ifdef ENONET
|
|
case ENONET:
|
|
#endif
|
|
/*
|
|
* These errors are expected and harmless, so
|
|
* we ignore them.
|
|
*/
|
|
return;
|
|
case EBADF:
|
|
case ENOTSOCK:
|
|
case EFAULT:
|
|
/*
|
|
* If one these happens, we're broken.
|
|
*/
|
|
ns_panic(ns_log_default, 1, "accept: %s",
|
|
strerror(errno));
|
|
case EMFILE:
|
|
/*
|
|
* If we're out of file descriptors, find the least
|
|
* busy fd and close it. Then we'll return to the
|
|
* eventlib which will call us right back.
|
|
*/
|
|
if (streamq) {
|
|
struct qstream *nextsp;
|
|
struct qstream *candidate = NULL;
|
|
time_t lasttime, maxctime = 0;
|
|
|
|
for (sp = streamq; sp; sp = nextsp) {
|
|
nextsp = sp->s_next;
|
|
if (sp->s_refcnt)
|
|
continue;
|
|
gettime(&tt);
|
|
lasttime = tt.tv_sec - sp->s_time;
|
|
if (lasttime >= VQEXPIRY)
|
|
sq_remove(sp);
|
|
else if (lasttime > maxctime) {
|
|
candidate = sp;
|
|
maxctime = lasttime;
|
|
}
|
|
}
|
|
if (candidate)
|
|
sq_remove(candidate);
|
|
return;
|
|
}
|
|
/* fall through */
|
|
default:
|
|
/*
|
|
* Either we got an error we didn't expect, or we
|
|
* got EMFILE and didn't have anything left to close.
|
|
* Log it and press on.
|
|
*/
|
|
ns_info(ns_log_default, "accept: %s", strerror(errno));
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Condition the socket. */
|
|
|
|
/* XXX clean up */
|
|
#if 0
|
|
if ((n = fcntl(rfd, F_GETFL, 0)) == -1) {
|
|
ns_info(ns_log_default, "fcntl(rfd, F_GETFL): %s",
|
|
strerror(errno));
|
|
(void) close(rfd);
|
|
return;
|
|
}
|
|
if (fcntl(rfd, F_SETFL, n|PORT_NONBLOCK) == -1) {
|
|
ns_info(ns_log_default, "fcntl(rfd, NONBLOCK): %s",
|
|
strerror(errno));
|
|
(void) close(rfd);
|
|
return;
|
|
}
|
|
#endif
|
|
#ifndef CANNOT_SET_SNDBUF
|
|
if (setsockopt(rfd, SOL_SOCKET, SO_SNDBUF,
|
|
(char*)&sbufsize, sizeof sbufsize) < 0) {
|
|
ns_info(ns_log_default, "setsockopt(rfd, SO_SNDBUF, %d): %s",
|
|
sbufsize, strerror(errno));
|
|
(void) close(rfd);
|
|
return;
|
|
}
|
|
#endif
|
|
if (setsockopt(rfd, SOL_SOCKET, SO_KEEPALIVE,
|
|
(char *)&on, sizeof on) < 0) {
|
|
ns_info(ns_log_default, "setsockopt(rfd, KEEPALIVE): %s",
|
|
strerror(errno));
|
|
(void) close(rfd);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* We don't like IP options. Turn them off if the connection came in
|
|
* with any. log this event since it usually indicates a security
|
|
* problem.
|
|
*/
|
|
#if defined(IP_OPTIONS) /* XXX */
|
|
len = sizeof ip_opts;
|
|
if (getsockopt(rfd, IPPROTO_IP, IP_OPTIONS,
|
|
(char *)ip_opts, &len) < 0) {
|
|
ns_info(ns_log_default, "getsockopt(rfd, IP_OPTIONS): %s",
|
|
strerror(errno));
|
|
(void) close(rfd);
|
|
return;
|
|
}
|
|
if (len != 0) {
|
|
nameserIncr(ra->sin_addr, nssRcvdOpts);
|
|
if (!haveComplained(ina_ulong(ra->sin_addr),
|
|
(u_long)"rcvd ip options")) {
|
|
ns_info(ns_log_default,
|
|
"rcvd IP_OPTIONS from %s (ignored)",
|
|
sin_ntoa(*ra));
|
|
}
|
|
if (setsockopt(rfd, IPPROTO_IP, IP_OPTIONS, NULL, 0) < 0) {
|
|
ns_info(ns_log_default, "setsockopt(!IP_OPTIONS): %s",
|
|
strerror(errno));
|
|
(void) close(rfd);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* Create and populate a qsp for this socket. */
|
|
if ((sp = sq_add()) == NULL) {
|
|
(void) close(rfd);
|
|
return;
|
|
}
|
|
sp->s_rfd = rfd; /* stream file descriptor */
|
|
gettime(&tt);
|
|
sp->s_time = tt.tv_sec; /* last transaction time */
|
|
sp->s_from = *ra; /* address to respond to */
|
|
sp->s_ifp = ifp;
|
|
INSIST(sizeof sp->s_temp >= INT16SZ);
|
|
iov = evConsIovec(sp->s_temp, INT16SZ);
|
|
INSIST_ERR(evRead(lev, rfd, &iov, 1, stream_getlen, sp, &sp->evID_r)
|
|
!= -1);
|
|
sp->flags |= STREAM_READ_EV;
|
|
#ifdef DEBUG
|
|
if (debug)
|
|
ns_info(ns_log_default, "IP/TCP connection from %s (fd %d)",
|
|
sin_ntoa(sp->s_from), rfd);
|
|
#endif
|
|
}
|
|
|
|
int
|
|
tcp_send(struct qinfo *qp) {
|
|
struct qstream *sp;
|
|
int on = 1;
|
|
|
|
ns_debug(ns_log_default, 1, "tcp_send");
|
|
if ((sp = sq_add()) == NULL) {
|
|
return(SERVFAIL);
|
|
}
|
|
if ((sp->s_rfd = socket(AF_INET, SOCK_STREAM, PF_UNSPEC)) == -1) {
|
|
sq_remove(sp);
|
|
return(SERVFAIL);
|
|
}
|
|
|
|
if (sq_openw(sp, qp->q_msglen + INT16SZ) == -1) {
|
|
sq_remove(sp);
|
|
return(SERVFAIL);
|
|
}
|
|
if (sq_write(sp, qp->q_msg, qp->q_msglen) == -1) {
|
|
sq_remove(sp);
|
|
return(SERVFAIL);
|
|
}
|
|
|
|
if (setsockopt(sp->s_rfd, SOL_SOCKET, SO_KEEPALIVE,
|
|
(char*)&on, sizeof(on)) < 0)
|
|
ns_info(ns_log_default,
|
|
"tcp_send: setsockopt(rfd, SO_KEEPALIVE): %s",
|
|
strerror(errno));
|
|
gettime(&tt);
|
|
sp->s_size = -1;
|
|
sp->s_time = tt.tv_sec; /* last transaction time */
|
|
sp->s_refcnt = 1;
|
|
sp->flags |= STREAM_DONE_CLOSE;
|
|
if (qp->q_fwd)
|
|
sp->s_from = qp->q_fwd->fwdaddr;
|
|
else
|
|
sp->s_from = qp->q_addr[qp->q_curaddr].ns_addr;
|
|
if (evConnect(ev, sp->s_rfd, &sp->s_from, sizeof(sp->s_from),
|
|
stream_send, sp, &sp->evID_c) == -1) {
|
|
sq_remove(sp);
|
|
return (SERVFAIL);
|
|
}
|
|
sp->flags |= STREAM_CONNECT_EV;
|
|
return(NOERROR);
|
|
}
|
|
|
|
static void
|
|
stream_send(evContext lev, void *uap, int fd, const void *la, int lalen,
|
|
const void *ra, int ralen) {
|
|
struct qstream *sp = uap;
|
|
|
|
ns_debug(ns_log_default, 1, "stream_send");
|
|
|
|
sp->flags &= ~STREAM_CONNECT_EV;
|
|
|
|
if (fd == -1) {
|
|
/* connect failed */
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
if (evSelectFD(ev, sp->s_rfd, EV_WRITE,
|
|
stream_write, sp, &sp->evID_w) < 0) {
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
sp->flags |= STREAM_WRITE_EV;
|
|
}
|
|
|
|
static void
|
|
stream_write(evContext ctx, void *uap, int fd, int evmask) {
|
|
struct qstream *sp = uap;
|
|
struct iovec iov;
|
|
|
|
ns_debug(ns_log_default, 1, "stream_write");
|
|
INSIST(evmask & EV_WRITE);
|
|
INSIST(fd == sp->s_rfd);
|
|
if (sq_dowrite(sp) < 0) {
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
if (sp->s_wbuf_free != sp->s_wbuf_send)
|
|
return;
|
|
|
|
if (sp->s_wbuf) {
|
|
memput(sp->s_wbuf, sp->s_wbuf_end - sp->s_wbuf);
|
|
sp->s_wbuf = NULL;
|
|
}
|
|
(void) evDeselectFD(ev, sp->evID_w);
|
|
sp->flags &= ~STREAM_WRITE_EV;
|
|
sp->s_refcnt = 0;
|
|
iov = evConsIovec(sp->s_temp, INT16SZ);
|
|
INSIST_ERR(evRead(ctx, fd, &iov, 1, stream_getlen, sp, &sp->evID_r) !=
|
|
-1);
|
|
sp->flags |= STREAM_READ_EV;
|
|
}
|
|
|
|
static void
|
|
stream_getlen(evContext lev, void *uap, int fd, int bytes) {
|
|
struct qstream *sp = uap;
|
|
struct iovec iov;
|
|
|
|
sp->flags &= ~STREAM_READ_EV;
|
|
if (bytes != INT16SZ) {
|
|
/*
|
|
* bytes == 0 is normal EOF; see if something unusual
|
|
* happened.
|
|
*/
|
|
if (bytes < 0) {
|
|
/*
|
|
* ECONNRESET happens frequently and is not worth
|
|
* logging.
|
|
*/
|
|
if (errno != ECONNRESET)
|
|
ns_info(ns_log_default,
|
|
"stream_getlen(%s): %s",
|
|
sin_ntoa(sp->s_from), strerror(errno));
|
|
} else if (bytes != 0)
|
|
ns_error(ns_log_default,
|
|
"stream_getlen(%s): unexpected byte count %d",
|
|
sin_ntoa(sp->s_from), bytes);
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Unpack the size, allocate memory for the query. This is
|
|
* tricky since in a low memory situation with possibly very
|
|
* large (64KB) queries, we want to make sure we can read at
|
|
* least the header since we need it to send back a SERVFAIL
|
|
* (owing to the out-of-memory condition).
|
|
*/
|
|
sp->s_size = ns_get16(sp->s_temp);
|
|
ns_debug(ns_log_default, 5, "stream message: %d bytes", sp->s_size);
|
|
|
|
if (!(sp->flags & STREAM_MALLOC)) {
|
|
sp->s_bufsize = 64*1024-1; /* maximum tcp message size */
|
|
sp->s_buf = (u_char *)memget(sp->s_bufsize);
|
|
if (sp->s_buf != NULL)
|
|
sp->flags |= STREAM_MALLOC;
|
|
else {
|
|
sp->s_buf = sp->s_temp;
|
|
sp->s_bufsize = HFIXEDSZ;
|
|
}
|
|
}
|
|
|
|
iov = evConsIovec(sp->s_buf, sp->s_size);
|
|
if (evRead(lev, sp->s_rfd, &iov, 1, stream_getmsg, sp, &sp->evID_r)
|
|
== -1)
|
|
ns_panic(ns_log_default, 1, "evRead(fd %d): %s",
|
|
(void *)sp->s_rfd, strerror(errno));
|
|
sp->flags |= STREAM_READ_EV;
|
|
}
|
|
|
|
static void
|
|
stream_getmsg(evContext lev, void *uap, int fd, int bytes) {
|
|
struct qstream *sp = uap;
|
|
int buflen, n;
|
|
|
|
sp->flags &= ~STREAM_READ_EV;
|
|
if (bytes == -1) {
|
|
ns_info(ns_log_default, "stream_getmsg(%s): %s",
|
|
sin_ntoa(sp->s_from), strerror(errno));
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
|
|
gettime(&tt);
|
|
sp->s_time = tt.tv_sec;
|
|
|
|
ns_debug(ns_log_default, 5, "sp %#x rfd %d size %d time %d next %#x",
|
|
sp, sp->s_rfd, sp->s_size, sp->s_time, sp->s_next);
|
|
ns_debug(ns_log_default, 5, "\tbufsize %d bytes %d", sp->s_bufsize,
|
|
bytes);
|
|
|
|
/*
|
|
* Do we have enough memory for the query? If not, and if we have a
|
|
* query id, then we will send a SERVFAIL error back to the client.
|
|
*/
|
|
if (bytes != sp->s_size) {
|
|
HEADER *hp = (HEADER *)sp->s_buf;
|
|
|
|
hp->qr = 1;
|
|
hp->ra = (NS_OPTION_P(OPTION_NORECURSE) == 0);
|
|
hp->ancount = htons(0);
|
|
hp->qdcount = htons(0);
|
|
hp->nscount = htons(0);
|
|
hp->arcount = htons(0);
|
|
hp->rcode = SERVFAIL;
|
|
writestream(sp, sp->s_buf, HFIXEDSZ);
|
|
sp->flags |= STREAM_DONE_CLOSE;
|
|
return;
|
|
}
|
|
|
|
nameserIncr(sp->s_from.sin_addr, nssRcvdTCP);
|
|
sq_query(sp);
|
|
dispatch_message(sp->s_buf, bytes, sp->s_bufsize, sp, sp->s_from, -1,
|
|
sp->s_ifp);
|
|
}
|
|
|
|
static void
|
|
datagram_read(evContext lev, void *uap, int fd, int evmask) {
|
|
interface *ifp = uap;
|
|
struct sockaddr_in from;
|
|
int from_len = sizeof from;
|
|
int n;
|
|
union {
|
|
HEADER h; /* Force alignment of 'buf'. */
|
|
u_char buf[PACKETSZ+1];
|
|
} u;
|
|
|
|
n = recvfrom(fd, (char *)u.buf, sizeof u.buf, 0,
|
|
(struct sockaddr *)&from, &from_len);
|
|
|
|
if (n < 0) {
|
|
switch (errno) {
|
|
case EINTR:
|
|
case EAGAIN:
|
|
#if (EWOULDBLOCK != EAGAIN)
|
|
case EWOULDBLOCK:
|
|
#endif
|
|
case EHOSTUNREACH:
|
|
case EHOSTDOWN:
|
|
case ENETUNREACH:
|
|
case ENETDOWN:
|
|
case ECONNREFUSED:
|
|
#ifdef ENONET
|
|
case ENONET:
|
|
#endif
|
|
/*
|
|
* These errors are expected and harmless, so we
|
|
* ignore them.
|
|
*/
|
|
return;
|
|
case EBADF:
|
|
case ENOTCONN:
|
|
case ENOTSOCK:
|
|
case EFAULT:
|
|
/*
|
|
* If one these happens, we're broken.
|
|
*/
|
|
ns_panic(ns_log_default, 1, "recvfrom: %s",
|
|
strerror(errno));
|
|
default:
|
|
/*
|
|
* An error we don't expect. Log it and press
|
|
* on.
|
|
*/
|
|
ns_info(ns_log_default, "recvfrom: %s",
|
|
strerror(errno));
|
|
return;
|
|
}
|
|
}
|
|
|
|
#ifndef BSD
|
|
/* Handle bogosity on systems that need it. */
|
|
if (n == 0)
|
|
return;
|
|
#endif
|
|
|
|
ns_debug(ns_log_default, 1, "datagram from %s, fd %d, len %d",
|
|
sin_ntoa(from), fd, n);
|
|
|
|
if (n > PACKETSZ) {
|
|
/*
|
|
* The message is too big. It's probably a response to
|
|
* one of our questions, so we truncate it and press on.
|
|
*/
|
|
n = trunc_adjust(u.buf, PACKETSZ, PACKETSZ);
|
|
ns_debug(ns_log_default, 1, "truncated oversize UDP packet");
|
|
}
|
|
|
|
gettime(&tt); /* Keep 'tt' current. */
|
|
dispatch_message(u.buf, n, PACKETSZ, NULL, from, fd, ifp);
|
|
}
|
|
|
|
static void
|
|
dispatch_message(u_char *msg, int msglen, int buflen, struct qstream *qsp,
|
|
struct sockaddr_in from, int dfd, interface *ifp)
|
|
{
|
|
HEADER *hp = (HEADER *)msg;
|
|
|
|
if (msglen < HFIXEDSZ) {
|
|
ns_debug(ns_log_default, 1, "dropping undersize message");
|
|
return;
|
|
}
|
|
if (hp->qr) {
|
|
ns_resp(msg, msglen, from, qsp);
|
|
if (qsp)
|
|
sq_done(qsp);
|
|
/* Now is a safe time for housekeeping. */
|
|
if (needs_prime_cache)
|
|
prime_cache();
|
|
} else if (ifp != NULL)
|
|
ns_req(msg, msglen, buflen, qsp, from, dfd);
|
|
else {
|
|
ns_notice(ns_log_security,
|
|
"refused query on non-query socket from %s",
|
|
sin_ntoa(from));
|
|
/* XXX Send refusal here. */
|
|
}
|
|
}
|
|
|
|
void
|
|
getnetconf(int periodic_scan) {
|
|
struct ifconf ifc;
|
|
struct ifreq ifreq;
|
|
struct in_addr ina;
|
|
interface *ifp;
|
|
char buf[32768], *cp, *cplim;
|
|
u_int32_t nm;
|
|
time_t my_generation = time(NULL);
|
|
int s, cpsize;
|
|
int found;
|
|
listen_info li;
|
|
u_int16_t port;
|
|
ip_match_element ime;
|
|
u_char *mask_ptr;
|
|
struct in_addr mask;
|
|
|
|
ns_debug(ns_log_default, 1, "getnetconf(generation %lu)",
|
|
(u_long)my_generation);
|
|
|
|
/* Get interface list from system. */
|
|
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
|
|
if (!periodic_scan)
|
|
ns_panic(ns_log_default, 1, "socket(SOCK_RAW): %s",
|
|
strerror(errno));
|
|
ns_error(ns_log_default, "socket(SOCK_RAW): %s",
|
|
strerror(errno));
|
|
return;
|
|
}
|
|
|
|
if (!iflist_initialized) {
|
|
INIT_LIST(iflist);
|
|
iflist_initialized = 1;
|
|
}
|
|
|
|
if (local_addresses != NULL)
|
|
free_ip_match_list(local_addresses);
|
|
local_addresses = new_ip_match_list();
|
|
if (local_networks != NULL)
|
|
free_ip_match_list(local_networks);
|
|
local_networks = new_ip_match_list();
|
|
|
|
ifc.ifc_len = sizeof buf;
|
|
ifc.ifc_buf = buf;
|
|
if (ioctl(s, SIOCGIFCONF, (char *)&ifc) < 0)
|
|
ns_panic(ns_log_default, 1, "get interface configuration: %s",
|
|
strerror(errno));
|
|
|
|
ns_debug(ns_log_default, 2, "getnetconf: SIOCGIFCONF: ifc_len = %d",
|
|
ifc.ifc_len);
|
|
|
|
/* Parse system's interface list and open some sockets. */
|
|
cplim = buf + ifc.ifc_len; /* skip over if's with big ifr_addr's */
|
|
for (cp = buf; cp < cplim; cp += cpsize) {
|
|
memcpy(&ifreq, cp, sizeof ifreq);
|
|
#if defined HAVE_SA_LEN
|
|
#ifdef FIX_ZERO_SA_LEN
|
|
if (ifreq.ifr_addr.sa_len == 0)
|
|
ifreq.ifr_addr.sa_len = 16;
|
|
#endif
|
|
#ifdef HAVE_MINIMUM_IFREQ
|
|
ns_debug(ns_log_default, 2, "%s sa_len = %d",
|
|
ifreq.ifr_name, (int)ifreq.ifr_addr.sa_len);
|
|
cpsize = sizeof ifreq;
|
|
if (ifreq.ifr_addr.sa_len > sizeof (struct sockaddr))
|
|
cpsize += (int)ifreq.ifr_addr.sa_len -
|
|
(int)(sizeof (struct sockaddr));
|
|
#else
|
|
cpsize = sizeof ifreq.ifr_name + ifreq.ifr_addr.sa_len;
|
|
#endif /* HAVE_MINIMUM_IFREQ */
|
|
#elif defined SIOCGIFCONF_ADDR
|
|
cpsize = sizeof ifreq;
|
|
#else
|
|
cpsize = sizeof ifreq.ifr_name;
|
|
if (ioctl(s, SIOCGIFADDR, (char *)&ifreq) < 0) {
|
|
ns_notice(ns_log_default,
|
|
"get interface addr (%s): %s",
|
|
ifreq.ifr_name, strerror(errno));
|
|
continue;
|
|
}
|
|
#endif
|
|
if (ifreq.ifr_addr.sa_family != AF_INET) {
|
|
ns_debug(ns_log_default, 2,
|
|
"getnetconf: %s AF %d != INET",
|
|
ifreq.ifr_name, ifreq.ifr_addr.sa_family);
|
|
continue;
|
|
}
|
|
ina = ina_get((u_char *)&((struct sockaddr_in *)
|
|
&ifreq.ifr_addr)->sin_addr);
|
|
ns_debug(ns_log_default, 1,
|
|
"getnetconf: considering %s [%s]",
|
|
ifreq.ifr_name, inet_ntoa(ina));
|
|
/*
|
|
* Don't test IFF_UP, packets may still be received at this
|
|
* address if any other interface is up.
|
|
*/
|
|
if (ina_hlong(ina) == INADDR_ANY) {
|
|
ns_debug(ns_log_default, 2,
|
|
"getnetconf: INADDR_ANY, ignoring.");
|
|
continue;
|
|
}
|
|
|
|
INSIST(server_options != NULL);
|
|
INSIST(server_options->listen_list != NULL);
|
|
|
|
found=0;
|
|
for (li = server_options->listen_list->first;
|
|
li != NULL;
|
|
li = li->next) {
|
|
if (ip_match_address(li->list, ina) > 0) {
|
|
found++;
|
|
/*
|
|
* Look for an already existing source
|
|
* interface address/port pair.
|
|
* This happens mostly when reinitializing.
|
|
* Also, if the machine has multiple point to
|
|
* point interfaces, then the local address
|
|
* may appear more than once.
|
|
*/
|
|
ifp = if_find(ina, li->port);
|
|
if (ifp != NULL) {
|
|
ns_debug(ns_log_default, 1,
|
|
"dup interface addr [%s].%u (%s)",
|
|
inet_ntoa(ina),
|
|
ntohs(li->port),
|
|
ifreq.ifr_name);
|
|
ifp->gen = my_generation;
|
|
continue;
|
|
}
|
|
|
|
ifp = (interface *)memget(sizeof *ifp);
|
|
if (!ifp)
|
|
ns_panic(ns_log_default, 1,
|
|
"memget(interface)", NULL);
|
|
memset(ifp, 0, sizeof *ifp);
|
|
APPEND(iflist, ifp, link);
|
|
ifp->addr = ina;
|
|
ifp->port = li->port;
|
|
ifp->gen = my_generation;
|
|
ifp->flags = 0;
|
|
ifp->dfd = -1;
|
|
ifp->sfd = -1;
|
|
if (opensocket_d(ifp) < 0 ||
|
|
opensocket_s(ifp) < 0) {
|
|
dq_remove(ifp);
|
|
found = 0;
|
|
break;
|
|
}
|
|
ns_info(ns_log_default,
|
|
"listening on [%s].%u (%s)",
|
|
inet_ntoa(ina), ntohs(li->port),
|
|
ifreq.ifr_name);
|
|
}
|
|
}
|
|
if (!found)
|
|
ns_debug(ns_log_default, 1,
|
|
"not listening on addr [%s] (%s)",
|
|
inet_ntoa(ina), ifreq.ifr_name);
|
|
|
|
/*
|
|
* Add this interface's address to the list of local
|
|
* addresses if we haven't added it already.
|
|
*/
|
|
if (ip_match_address(local_addresses, ina) < 0) {
|
|
ime = new_ip_match_pattern(ina, 32);
|
|
add_to_ip_match_list(local_addresses, ime);
|
|
}
|
|
|
|
/*
|
|
* Get interface flags.
|
|
*/
|
|
if (ioctl(s, SIOCGIFFLAGS, (char *)&ifreq) < 0) {
|
|
ns_notice(ns_log_default, "get interface flags: %s",
|
|
strerror(errno));
|
|
continue;
|
|
}
|
|
|
|
if ((ifreq.ifr_flags & IFF_POINTOPOINT)) {
|
|
/*
|
|
* The local network for a PPP link is just the
|
|
* two ends of the link, so for each endpoint we
|
|
* add a pattern that will only match the endpoint.
|
|
*/
|
|
if (ioctl(s, SIOCGIFDSTADDR, (char *)&ifreq) < 0) {
|
|
ns_notice(ns_log_default, "get dst addr: %s",
|
|
strerror(errno));
|
|
continue;
|
|
}
|
|
|
|
mask.s_addr = htonl(INADDR_BROADCAST);
|
|
|
|
/*
|
|
* Our end.
|
|
*
|
|
* Only add it if we haven't seen it before.
|
|
*/
|
|
if (ip_match_network(local_networks, ina, mask) < 0) {
|
|
ime = new_ip_match_pattern(ina, 32);
|
|
add_to_ip_match_list(local_networks, ime);
|
|
}
|
|
|
|
/*
|
|
* The other end.
|
|
*/
|
|
ina = ((struct sockaddr_in *)
|
|
&ifreq.ifr_addr)->sin_addr;
|
|
/*
|
|
* Only add it if we haven't seen it before.
|
|
*/
|
|
if (ip_match_network(local_networks, ina, mask) < 0) {
|
|
ime = new_ip_match_pattern(ina, 32);
|
|
add_to_ip_match_list(local_networks, ime);
|
|
}
|
|
} else {
|
|
/*
|
|
* Add this interface's network and netmask to the
|
|
* list of local networks.
|
|
*/
|
|
|
|
#ifdef SIOCGIFNETMASK /* XXX */
|
|
if (ioctl(s, SIOCGIFNETMASK, (char *)&ifreq) < 0) {
|
|
ns_notice(ns_log_default, "get netmask: %s",
|
|
strerror(errno));
|
|
continue;
|
|
}
|
|
/*
|
|
* Use ina_get because the ifreq structure might not
|
|
* be aligned.
|
|
*/
|
|
mask_ptr = (u_char *)
|
|
&((struct sockaddr_in *)&ifreq.ifr_addr)->sin_addr;
|
|
mask = ina_get(mask_ptr);
|
|
#else
|
|
mask = net_mask(ina);
|
|
#endif
|
|
|
|
ina.s_addr &= mask.s_addr; /* make network address */
|
|
|
|
/*
|
|
* Only add it if we haven't seen it before.
|
|
*/
|
|
if (ip_match_network(local_networks, ina, mask) < 0) {
|
|
ime = new_ip_match_mask(ina, mask);
|
|
add_to_ip_match_list(local_networks, ime);
|
|
}
|
|
}
|
|
}
|
|
close(s);
|
|
|
|
ns_debug(ns_log_default, 7, "local addresses:");
|
|
dprint_ip_match_list(ns_log_default, local_addresses, 2, "", "");
|
|
ns_debug(ns_log_default, 7, "local networks:");
|
|
dprint_ip_match_list(ns_log_default, local_networks, 2, "", "");
|
|
|
|
/*
|
|
* now go through the iflist and delete anything that
|
|
* does not have the current generation number. this is
|
|
* how we catch interfaces that go away or change their
|
|
* addresses. note that 0.0.0.0 is the wildcard element
|
|
* and should never be deleted by this code.
|
|
*/
|
|
dq_remove_gen(my_generation);
|
|
|
|
if (EMPTY(iflist))
|
|
ns_warning(ns_log_default, "not listening on any interfaces");
|
|
}
|
|
|
|
/* opensocket_d(ifp)
|
|
* Open datagram socket bound to interface address.
|
|
* Returns:
|
|
* 0 on success.
|
|
* -1 on failure.
|
|
*/
|
|
static int
|
|
opensocket_d(interface *ifp) {
|
|
struct sockaddr_in nsa;
|
|
const int on = 1;
|
|
int m, n;
|
|
int fd;
|
|
|
|
memset(&nsa, 0, sizeof nsa);
|
|
nsa.sin_family = AF_INET;
|
|
nsa.sin_addr = ifp->addr;
|
|
nsa.sin_port = ifp->port;
|
|
|
|
if ((ifp->dfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
|
|
ns_error(ns_log_default, "socket(SOCK_DGRAM): %s",
|
|
strerror(errno));
|
|
return (-1);
|
|
}
|
|
#ifdef F_DUPFD /* XXX */
|
|
/*
|
|
* Leave a space for stdio to work in.
|
|
*/
|
|
if ((fd = fcntl(ifp->dfd, F_DUPFD, 20)) != -1) {
|
|
close(ifp->dfd);
|
|
ifp->dfd = fd;
|
|
} else
|
|
ns_notice(ns_log_default, "fcntl(dfd, F_DUPFD, 20): %s",
|
|
strerror(errno));
|
|
#endif
|
|
ns_debug(ns_log_default, 1, "ifp->addr %s d_dfd %d",
|
|
sin_ntoa(nsa), ifp->dfd);
|
|
if (setsockopt(ifp->dfd, SOL_SOCKET, SO_REUSEADDR,
|
|
(char *)&on, sizeof(on)) != 0) {
|
|
ns_notice(ns_log_default, "setsockopt(REUSEADDR): %s",
|
|
strerror(errno));
|
|
/* XXX press on regardless, this is not too serious. */
|
|
}
|
|
#ifdef SO_RCVBUF /* XXX */
|
|
m = sizeof n;
|
|
if ((getsockopt(ifp->dfd, SOL_SOCKET, SO_RCVBUF, (char*)&n, &m) >= 0)
|
|
&& (m == sizeof n)
|
|
&& (n < drbufsize)) {
|
|
(void) setsockopt(ifp->dfd, SOL_SOCKET, SO_RCVBUF,
|
|
(char *)&drbufsize, sizeof drbufsize);
|
|
}
|
|
#endif /* SO_RCVBUF */
|
|
#ifndef CANNOT_SET_SNDBUF
|
|
if (setsockopt(ifp->dfd, SOL_SOCKET, SO_SNDBUF,
|
|
(char*)&dsbufsize, sizeof dsbufsize) < 0) {
|
|
ns_info(ns_log_default,
|
|
"setsockopt(dfd=%d, SO_SNDBUF, %d): %s",
|
|
ifp->dfd, dsbufsize, strerror(errno));
|
|
/* XXX press on regardless, this is not too serious. */
|
|
}
|
|
#endif
|
|
if (bind(ifp->dfd, (struct sockaddr *)&nsa, sizeof nsa)) {
|
|
ns_error(ns_log_default, "bind(dfd=%d, %s): %s",
|
|
ifp->dfd, sin_ntoa(nsa), strerror(errno));
|
|
return (-1);
|
|
}
|
|
if (evSelectFD(ev, ifp->dfd, EV_READ, datagram_read, ifp,
|
|
&ifp->evID_d) == -1) {
|
|
ns_error(ns_log_default, "evSelectFD(dfd=%d): %s",
|
|
ifp->dfd, strerror(errno));
|
|
return (-1);
|
|
}
|
|
ifp->flags |= INTERFACE_FILE_VALID;
|
|
return (0);
|
|
}
|
|
|
|
/* opensocket_s(ifp)
|
|
* Open stream (listener) socket bound to interface address.
|
|
* Returns:
|
|
* 0 on success.
|
|
* -1 on failure.
|
|
*/
|
|
static int
|
|
opensocket_s(interface *ifp) {
|
|
struct sockaddr_in nsa;
|
|
const int on = 1;
|
|
int n;
|
|
int fd;
|
|
|
|
memset(&nsa, 0, sizeof nsa);
|
|
nsa.sin_family = AF_INET;
|
|
nsa.sin_addr = ifp->addr;
|
|
nsa.sin_port = ifp->port;
|
|
|
|
/*
|
|
* Open stream (listener) port.
|
|
*/
|
|
n = 0;
|
|
again:
|
|
if ((ifp->sfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
|
|
ns_error(ns_log_default, "socket(SOCK_STREAM): %s",
|
|
strerror(errno));
|
|
return (-1);
|
|
}
|
|
#ifdef F_DUPFD /* XXX */
|
|
/*
|
|
* Leave a space for stdio to work in.
|
|
*/
|
|
if ((fd = fcntl(ifp->sfd, F_DUPFD, 20)) != -1) {
|
|
close(ifp->sfd);
|
|
ifp->sfd = fd;
|
|
} else
|
|
ns_notice(ns_log_default, "fcntl(sfd, F_DUPFD, 20): %s",
|
|
strerror(errno));
|
|
#endif
|
|
if (setsockopt(ifp->sfd, SOL_SOCKET, SO_REUSEADDR,
|
|
(char *)&on, sizeof on) != 0) {
|
|
ns_notice(ns_log_default, "setsockopt(REUSEADDR): %s",
|
|
strerror(errno));
|
|
/* Consider that your first warning of trouble to come. */
|
|
}
|
|
if (bind(ifp->sfd, (struct sockaddr *)&nsa, sizeof nsa) < 0) {
|
|
if (errno != EADDRINUSE || ++n > 4) {
|
|
if (errno == EADDRINUSE)
|
|
ns_error(ns_log_default,
|
|
"There may be a name server already running on %s",
|
|
sin_ntoa(nsa));
|
|
else
|
|
ns_error(ns_log_default,
|
|
"bind(sfd=%d, %s): %s", ifp->sfd,
|
|
sin_ntoa(nsa), strerror(errno));
|
|
return (-1);
|
|
}
|
|
|
|
/* Retry opening the socket a few times */
|
|
close(ifp->sfd);
|
|
ifp->sfd = -1;
|
|
sleep(30);
|
|
goto again;
|
|
}
|
|
if (evListen(ev, ifp->sfd, 5/*XXX*/, stream_accept, ifp, &ifp->evID_s)
|
|
== -1) {
|
|
ns_error(ns_log_default, "evListen(sfd=%d): %s",
|
|
ifp->sfd, strerror(errno));
|
|
return (-1);
|
|
}
|
|
ifp->flags |= INTERFACE_CONN_VALID;
|
|
return (0);
|
|
}
|
|
|
|
/* opensocket_f()
|
|
* Open datagram socket bound to no particular interface; use for ns_forw
|
|
* and sysquery.
|
|
*/
|
|
void
|
|
opensocket_f() {
|
|
static struct sockaddr_in prev_qsrc;
|
|
static int been_here;
|
|
static interface *prev_ifp;
|
|
struct sockaddr_in nsa;
|
|
const int on = 1;
|
|
int n, need_close;
|
|
interface *ifp;
|
|
|
|
need_close = 0;
|
|
if (been_here) {
|
|
if (prev_ifp != NULL)
|
|
prev_ifp->flags &= ~INTERFACE_FORWARDING;
|
|
else if (server_options->query_source.sin_port == htons(0) ||
|
|
prev_qsrc.sin_addr.s_addr !=
|
|
server_options->query_source.sin_addr.s_addr ||
|
|
prev_qsrc.sin_port !=
|
|
server_options->query_source.sin_port)
|
|
need_close = 1;
|
|
} else
|
|
ds = -1;
|
|
|
|
been_here = 1;
|
|
INSIST(server_options != NULL);
|
|
|
|
if (need_close) {
|
|
evDeselectFD(ev, ds_evID);
|
|
close(ds);
|
|
ds = -1;
|
|
}
|
|
|
|
/*
|
|
* If we're already listening on the query_source address and port,
|
|
* we don't need to open another socket. We mark the interface, so
|
|
* we'll notice we're in trouble if it goes away.
|
|
*/
|
|
ifp = if_find(server_options->query_source.sin_addr,
|
|
server_options->query_source.sin_port);
|
|
if (ifp != NULL) {
|
|
ifp->flags |= INTERFACE_FORWARDING;
|
|
prev_ifp = ifp;
|
|
ds = ifp->dfd;
|
|
ns_info(ns_log_default, "forwarding source address is %s",
|
|
sin_ntoa(server_options->query_source));
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* If we're already using the correct query source, we're done.
|
|
*/
|
|
if (ds >= 0)
|
|
return;
|
|
|
|
prev_qsrc = server_options->query_source;
|
|
prev_ifp = NULL;
|
|
|
|
if ((ds = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
|
|
ns_panic(ns_log_default, 1, "socket(SOCK_DGRAM): %s",
|
|
strerror(errno));
|
|
if (setsockopt(ds, SOL_SOCKET, SO_REUSEADDR,
|
|
(char *)&on, sizeof on) != 0) {
|
|
ns_notice(ns_log_default, "setsockopt(REUSEADDR): %s",
|
|
strerror(errno));
|
|
/* XXX press on regardless, this is not too serious. */
|
|
}
|
|
if (bind(ds, (struct sockaddr *)&server_options->query_source,
|
|
sizeof server_options->query_source) < 0)
|
|
ns_panic(ns_log_default, 1, "opensocket_f: bind(%s): %s",
|
|
sin_ntoa(server_options->query_source),
|
|
strerror(errno));
|
|
|
|
n = sizeof nsa;
|
|
if (getsockname(ds, (struct sockaddr *)&nsa, &n) < 0)
|
|
ns_panic(ns_log_default, 1, "opensocket_f: getsockaddr: %s",
|
|
strerror(errno));
|
|
|
|
ns_debug(ns_log_default, 1, "fwd ds %d addr %s", ds, sin_ntoa(nsa));
|
|
ns_info(ns_log_default, "Forwarding source address is %s",
|
|
sin_ntoa(nsa));
|
|
|
|
if (evSelectFD(ev, ds, EV_READ, datagram_read, NULL, &ds_evID) == -1)
|
|
ns_panic(ns_log_default, 1, "evSelectFD(fd %d): %s",
|
|
(void *)ds, strerror(errno));
|
|
/* XXX: should probably use a different FileFunc that only accepts
|
|
* responses, since requests on this socket make no sense.
|
|
*/
|
|
}
|
|
|
|
static void
|
|
setdebug(int new_debug) {
|
|
#ifdef DEBUG
|
|
int old_debug;
|
|
|
|
if (!new_debug)
|
|
ns_debug(ns_log_default, 1, "Debug off");
|
|
old_debug = debug;
|
|
debug = new_debug;
|
|
log_option(log_ctx, LOG_OPTION_DEBUG, debug);
|
|
log_option(log_ctx, LOG_OPTION_LEVEL, debug);
|
|
evSetDebug(ev, debug, log_get_stream(eventlib_channel));
|
|
if (debug) {
|
|
if (!old_debug)
|
|
open_special_channels();
|
|
ns_debug(ns_log_default, 1, "Debug level %d", debug);
|
|
if (!old_debug) {
|
|
ns_debug(ns_log_default, 1, "Version = %s", Version);
|
|
ns_debug(ns_log_default, 1, "conffile = %s", conffile);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static SIG_FN
|
|
onhup(int sig) {
|
|
ns_need(MAIN_NEED_RELOAD);
|
|
}
|
|
|
|
static SIG_FN
|
|
onintr(int sig) {
|
|
ns_need(MAIN_NEED_EXIT);
|
|
}
|
|
|
|
static SIG_FN
|
|
setdumpflg(int sig) {
|
|
ns_need(MAIN_NEED_DUMP);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
static SIG_FN
|
|
setIncrDbgFlg(int sig) {
|
|
desired_debug++;
|
|
ns_need(MAIN_NEED_DEBUG);
|
|
}
|
|
|
|
static SIG_FN
|
|
setNoDbgFlg(int sig) {
|
|
desired_debug = 0;
|
|
ns_need(MAIN_NEED_DEBUG);
|
|
}
|
|
#endif /*DEBUG*/
|
|
|
|
#if defined(QRYLOG) && defined(SIGWINCH)
|
|
static SIG_FN
|
|
setQrylogFlg(int sig) {
|
|
ns_need(MAIN_NEED_QRYLOG);
|
|
}
|
|
#endif /*QRYLOG && SIGWINCH*/
|
|
|
|
static SIG_FN
|
|
setstatsflg(int sig) {
|
|
ns_need(MAIN_NEED_STATSDUMP);
|
|
}
|
|
|
|
static SIG_FN
|
|
discard_pipe(int sig) {
|
|
#ifdef SIGPIPE_ONE_SHOT
|
|
int saved_errno = errno;
|
|
set_signal_handler(SIGPIPE, discard_pipe);
|
|
errno = saved_errno;
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
** Routines for managing stream queue
|
|
*/
|
|
|
|
static struct qstream *
|
|
sq_add() {
|
|
struct qstream *sqp;
|
|
|
|
if (!(sqp = (struct qstream *)memget(sizeof *sqp))) {
|
|
ns_error(ns_log_default, "sq_add: memget: %s",
|
|
strerror(errno));
|
|
return (NULL);
|
|
}
|
|
memset(sqp, 0, sizeof *sqp);
|
|
ns_debug(ns_log_default, 3, "sq_add(%#lx)", (u_long)sqp);
|
|
|
|
sqp->flags = 0;
|
|
/* XXX should init other fields too? */
|
|
sqp->s_next = streamq;
|
|
streamq = sqp;
|
|
return (sqp);
|
|
}
|
|
|
|
/* sq_remove(qp)
|
|
* remove stream queue structure `qp'.
|
|
* no current queries may refer to this stream when it is removed.
|
|
* side effects:
|
|
* memory is deallocated. sockets are closed. lists are relinked.
|
|
*/
|
|
void
|
|
sq_remove(struct qstream *qp) {
|
|
struct qstream *qsp;
|
|
|
|
ns_debug(ns_log_default, 2, "sq_remove(%#lx, %d) rfcnt=%d",
|
|
(u_long)qp, qp->s_rfd, qp->s_refcnt);
|
|
|
|
if (qp->s_wbuf != NULL) {
|
|
memput(qp->s_wbuf, qp->s_wbuf_end - qp->s_wbuf);
|
|
qp->s_wbuf = NULL;
|
|
}
|
|
if (qp->flags & STREAM_MALLOC)
|
|
memput(qp->s_buf, qp->s_bufsize);
|
|
if (qp->flags & STREAM_READ_EV)
|
|
INSIST_ERR(evCancelRW(ev, qp->evID_r) != -1);
|
|
if (qp->flags & STREAM_WRITE_EV)
|
|
INSIST_ERR(evDeselectFD(ev, qp->evID_w) != -1);
|
|
if (qp->flags & STREAM_CONNECT_EV)
|
|
INSIST_ERR(evCancelConn(ev, qp->evID_c) != -1);
|
|
if (qp->flags & STREAM_AXFR)
|
|
ns_freexfr(qp);
|
|
(void) close(qp->s_rfd);
|
|
if (qp == streamq)
|
|
streamq = qp->s_next;
|
|
else {
|
|
for (qsp = streamq;
|
|
qsp && (qsp->s_next != qp);
|
|
qsp = qsp->s_next)
|
|
(void)NULL;
|
|
if (qsp)
|
|
qsp->s_next = qp->s_next;
|
|
}
|
|
memput(qp, sizeof *qp);
|
|
}
|
|
|
|
/* void
|
|
* sq_flush(allbut)
|
|
* call sq_remove() on all open streams except `allbut'
|
|
* side effects:
|
|
* global list `streamq' modified
|
|
* idiocy:
|
|
* is N^2 due to the scan inside of sq_remove()
|
|
*/
|
|
void
|
|
sq_flush(struct qstream *allbut) {
|
|
struct qstream *sp, *spnext;
|
|
|
|
for (sp = streamq; sp != NULL; sp = spnext) {
|
|
spnext = sp->s_next;
|
|
if (sp != allbut)
|
|
sq_remove(sp);
|
|
}
|
|
}
|
|
|
|
/* int
|
|
* sq_openw(qs, buflen)
|
|
* add a write buffer to a stream
|
|
* return:
|
|
* 0 = success
|
|
* -1 = failure (check errno)
|
|
*/
|
|
int
|
|
sq_openw(struct qstream *qs, int buflen) {
|
|
#ifdef SO_LINGER /* XXX */
|
|
static const struct linger ll = { 1, 120 };
|
|
#endif
|
|
|
|
INSIST(qs->s_wbuf == NULL);
|
|
qs->s_wbuf = (u_char *)memget(buflen);
|
|
if (qs->s_wbuf == NULL)
|
|
return (-1);
|
|
qs->s_wbuf_send = qs->s_wbuf;
|
|
qs->s_wbuf_free = qs->s_wbuf;
|
|
qs->s_wbuf_end = qs->s_wbuf + buflen;
|
|
#ifdef SO_LINGER /* XXX */
|
|
/* kernels that map pages for IO end up failing if the pipe is full
|
|
* at exit and we take away the final buffer. this is really a kernel
|
|
* bug but it's harmless on systems that are not broken, so...
|
|
*/
|
|
setsockopt(qs->s_rfd, SOL_SOCKET, SO_LINGER, (char *)&ll, sizeof ll);
|
|
#endif
|
|
return (0);
|
|
}
|
|
|
|
/* static void
|
|
* sq_dowrite(qs)
|
|
* try to submit data to the system, remove it from our queue.
|
|
*/
|
|
static int
|
|
sq_dowrite(struct qstream *qs) {
|
|
if (qs->s_wbuf_free > qs->s_wbuf_send) {
|
|
int n = write(qs->s_rfd, qs->s_wbuf_send,
|
|
qs->s_wbuf_free - qs->s_wbuf_send);
|
|
if (n < 0) {
|
|
if (errno != EINTR && errno != EAGAIN
|
|
#if (EWOULDBLOCK != EAGAIN)
|
|
&& errno != EWOULDBLOCK
|
|
#endif
|
|
)
|
|
return (-1);
|
|
return (0);
|
|
}
|
|
qs->s_wbuf_send += n;
|
|
if (qs->s_wbuf_free > qs->s_wbuf_send) {
|
|
/* XXX: need some kind of delay here during which the
|
|
* socket will be deselected so we don't spin.
|
|
*/
|
|
n = qs->s_wbuf_free - qs->s_wbuf_send;
|
|
memmove(qs->s_wbuf, qs->s_wbuf_send, n);
|
|
qs->s_wbuf_send = qs->s_wbuf;
|
|
qs->s_wbuf_free = qs->s_wbuf + n;
|
|
}
|
|
}
|
|
if (qs->s_wbuf_free == qs->s_wbuf_send)
|
|
qs->s_wbuf_free = qs->s_wbuf_send = qs->s_wbuf;
|
|
return (0);
|
|
}
|
|
|
|
/* void
|
|
* sq_flushw(qs)
|
|
* called when the socket becomes writable and we want to flush our
|
|
* buffers and the system's socket buffers. use as a closure with
|
|
* sq_writeh().
|
|
*/
|
|
void
|
|
sq_flushw(struct qstream *qs) {
|
|
if (qs->s_wbuf_free == qs->s_wbuf_send) {
|
|
sq_writeh(qs, NULL);
|
|
sq_done(qs);
|
|
}
|
|
}
|
|
|
|
/* static void
|
|
* sq_writable(ctx, uap, fd, evmask)
|
|
* glue between eventlib closures and qstream closures
|
|
*/
|
|
static void
|
|
sq_writable(evContext ctx, void *uap, int fd, int evmask) {
|
|
struct qstream *qs = uap;
|
|
|
|
INSIST(evmask & EV_WRITE);
|
|
INSIST(fd == qs->s_rfd);
|
|
if (sq_dowrite(qs) < 0) {
|
|
sq_remove(qs);
|
|
return;
|
|
}
|
|
if (qs->s_wbuf_closure
|
|
&& qs->s_wbuf_end - qs->s_wbuf_free >= HFIXEDSZ+2) /* XXX guess */
|
|
(*qs->s_wbuf_closure)(qs);
|
|
if (sq_dowrite(qs) < 0) {
|
|
sq_remove(qs);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* int
|
|
* sq_writeh(qs, closure)
|
|
* register a closure to be called when a stream becomes writable
|
|
* return:
|
|
* 0 = success
|
|
* -1 = failure (check errno)
|
|
*/
|
|
int
|
|
sq_writeh(struct qstream *qs, sq_closure c) {
|
|
if (c) {
|
|
if (!qs->s_wbuf_closure) {
|
|
if (evSelectFD(ev, qs->s_rfd, EV_WRITE,
|
|
sq_writable, qs, &qs->evID_w) < 0) {
|
|
return (-1);
|
|
}
|
|
qs->flags |= STREAM_WRITE_EV;
|
|
}
|
|
} else {
|
|
(void) evDeselectFD(ev, qs->evID_w);
|
|
qs->flags &= ~STREAM_WRITE_EV;
|
|
}
|
|
qs->s_wbuf_closure = c;
|
|
return (0);
|
|
}
|
|
|
|
/* int
|
|
* sq_write(qs, buf, len)
|
|
* queue a message onto the stream, prepended by a two byte length field
|
|
* return:
|
|
* 0 = success
|
|
* -1 = failure (check errno; E2BIG means we can't handle this right now)
|
|
*/
|
|
int
|
|
sq_write(struct qstream *qs, const u_char *buf, int len) {
|
|
INSIST(qs->s_wbuf != NULL);
|
|
if (NS_INT16SZ + len > qs->s_wbuf_end - qs->s_wbuf_free) {
|
|
if (sq_dowrite(qs) < 0)
|
|
return (-1);
|
|
if (NS_INT16SZ + len > qs->s_wbuf_end - qs->s_wbuf_free) {
|
|
errno = E2BIG;
|
|
return (-1);
|
|
}
|
|
}
|
|
__putshort(len, qs->s_wbuf_free);
|
|
qs->s_wbuf_free += NS_INT16SZ;
|
|
memcpy(qs->s_wbuf_free, buf, len);
|
|
qs->s_wbuf_free += len;
|
|
return (0);
|
|
}
|
|
|
|
/* int
|
|
* sq_here(sp)
|
|
* determine whether stream 'sp' is still on the streamq
|
|
* return:
|
|
* boolean: is it here?
|
|
*/
|
|
static int
|
|
sq_here(struct qstream *sp) {
|
|
struct qstream *t;
|
|
|
|
for (t = streamq; t != NULL; t = t->s_next)
|
|
if (t == sp)
|
|
return (1);
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* Initiate query on stream;
|
|
* mark as referenced and stop selecting for input.
|
|
*/
|
|
static void
|
|
sq_query(struct qstream *sp) {
|
|
sp->s_refcnt++;
|
|
}
|
|
|
|
/*
|
|
* Note that the current request on a stream has completed,
|
|
* and that we should continue looking for requests on the stream.
|
|
*/
|
|
void
|
|
sq_done(struct qstream *sp) {
|
|
struct iovec iov;
|
|
|
|
if (sp->s_wbuf != NULL) {
|
|
memput(sp->s_wbuf, sp->s_wbuf_end - sp->s_wbuf);
|
|
sp->s_wbuf = NULL;
|
|
}
|
|
if (sp->flags & STREAM_AXFR)
|
|
ns_freexfr(sp);
|
|
sp->s_refcnt = 0;
|
|
sp->s_time = tt.tv_sec;
|
|
if (sp->flags & STREAM_DONE_CLOSE) {
|
|
/* XXX */
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
iov = evConsIovec(sp->s_temp, INT16SZ);
|
|
if (evRead(ev, sp->s_rfd, &iov, 1, stream_getlen, sp, &sp->evID_r) ==
|
|
-1)
|
|
ns_panic(ns_log_default, 1, "evRead(fd %d): %s",
|
|
(void *)sp->s_rfd, strerror(errno));
|
|
sp->flags |= STREAM_READ_EV;
|
|
}
|
|
|
|
/* void
|
|
* dq_remove_gen(gen)
|
|
* close/deallocate all the udp sockets (except 0.0.0.0) which are
|
|
* not from the current generation.
|
|
* side effects:
|
|
* global list `iflist' is modified.
|
|
*/
|
|
void
|
|
dq_remove_gen(time_t gen) {
|
|
interface *this, *next;
|
|
|
|
for (this = HEAD(iflist); this != NULL; this = next) {
|
|
next = NEXT(this, link);
|
|
if (this->gen != gen && ina_hlong(this->addr) != INADDR_ANY)
|
|
dq_remove(this);
|
|
}
|
|
}
|
|
|
|
/* void
|
|
* dq_remove_all()
|
|
* close/deallocate all interfaces.
|
|
* side effects:
|
|
* global list `iflist' is modified.
|
|
*/
|
|
void
|
|
dq_remove_all() {
|
|
interface *this, *next;
|
|
|
|
for (this = HEAD(iflist); this != NULL; this = next) {
|
|
next = NEXT(this, link);
|
|
/*
|
|
* Clear the forwarding flag so we don't panic the server.
|
|
*/
|
|
this->flags &= ~INTERFACE_FORWARDING;
|
|
dq_remove(this);
|
|
}
|
|
}
|
|
|
|
/* void
|
|
* dq_remove(interface *this)
|
|
* close/deallocate an interface's sockets. called on errors
|
|
* or if the interface disappears.
|
|
* side effects:
|
|
* global list `iflist' is modified.
|
|
*/
|
|
static void
|
|
dq_remove(interface *this) {
|
|
ns_notice(ns_log_default, "deleting interface [%s].%u",
|
|
inet_ntoa(this->addr), ntohs(this->port));
|
|
|
|
if ((this->flags & INTERFACE_FORWARDING) != 0)
|
|
ns_panic(ns_log_default, 0,
|
|
"forwarding interface [%s].%u gone",
|
|
inet_ntoa(this->addr),
|
|
ntohs(this->port));
|
|
|
|
/* Deallocate fields. */
|
|
if ((this->flags & INTERFACE_FILE_VALID) != 0)
|
|
(void) evDeselectFD(ev, this->evID_d);
|
|
if (this->dfd >= 0)
|
|
(void) close(this->dfd);
|
|
if ((this->flags & INTERFACE_CONN_VALID) != 0)
|
|
(void) evCancelConn(ev, this->evID_s);
|
|
if (this->sfd >= 0)
|
|
(void) close(this->sfd);
|
|
|
|
UNLINK(iflist, this, link);
|
|
memput(this, sizeof *this);
|
|
}
|
|
|
|
/* struct in_addr
|
|
* net_mask(ina)
|
|
* makes a classful assumption in a classless world, and returns it.
|
|
*/
|
|
struct in_addr
|
|
net_mask(struct in_addr ina) {
|
|
u_long hl = ina_hlong(ina);
|
|
struct in_addr ret;
|
|
|
|
if (IN_CLASSA(hl))
|
|
hl = IN_CLASSA_NET;
|
|
else if (IN_CLASSB(hl))
|
|
hl = IN_CLASSB_NET;
|
|
else if (IN_CLASSC(hl))
|
|
hl = IN_CLASSC_NET;
|
|
else
|
|
hl = INADDR_BROADCAST;
|
|
ina_ulong(ret) = htonl(hl);
|
|
return (ret);
|
|
}
|
|
|
|
/* aIsUs(addr)
|
|
* scan our list of interface addresses for "addr".
|
|
* returns:
|
|
* 0: address isn't one of our interfaces
|
|
* >0: address is one of our interfaces, or INADDR_ANY
|
|
*/
|
|
int
|
|
aIsUs(struct in_addr addr) {
|
|
interface *ifp;
|
|
|
|
if (ina_hlong(addr) == INADDR_ANY || if_find(addr, 0) != NULL)
|
|
return (1);
|
|
return (0);
|
|
}
|
|
|
|
/* interface *
|
|
* if_find(addr, port)
|
|
* scan our list of interface addresses for "addr" and port.
|
|
* port == 0 means match any port
|
|
* returns:
|
|
* pointer to interface with this address/port, or NULL if there isn't
|
|
* one.
|
|
*/
|
|
static interface *
|
|
if_find(struct in_addr addr, u_int16_t port) {
|
|
interface *ifp;
|
|
|
|
for (ifp = HEAD(iflist); ifp != NULL; ifp = NEXT(ifp, link))
|
|
if (ina_equal(addr, ifp->addr))
|
|
if (port == 0 || ifp->port == port)
|
|
break;
|
|
return (ifp);
|
|
}
|
|
|
|
/*
|
|
* These are here in case we ever want to get more clever, like perhaps
|
|
* using a bitmap to keep track of outstanding queries and a random
|
|
* allocation scheme to make it a little harder to predict them. Note
|
|
* that the resolver will need the same protection so the cleverness
|
|
* should be put there rather than here; this is just an interface layer.
|
|
*/
|
|
|
|
void
|
|
nsid_init() {
|
|
nsid_state = res_randomid();
|
|
}
|
|
|
|
u_int16_t
|
|
nsid_next() {
|
|
if (nsid_state == 65535)
|
|
nsid_state = 0;
|
|
else
|
|
nsid_state++;
|
|
return (nsid_state);
|
|
}
|
|
|
|
static void
|
|
deallocate_everything(void) {
|
|
FILE *f;
|
|
|
|
f = write_open(server_options->memstats_filename);
|
|
|
|
ns_freestats();
|
|
qflush();
|
|
sq_flush(NULL);
|
|
free_addinfo();
|
|
ns_shutdown();
|
|
dq_remove_all();
|
|
if (local_addresses != NULL)
|
|
free_ip_match_list(local_addresses);
|
|
if (local_networks != NULL)
|
|
free_ip_match_list(local_networks);
|
|
destroyservicelist();
|
|
destroyprotolist();
|
|
shutdown_logging();
|
|
evDestroy(ev);
|
|
if (conffile != NULL)
|
|
freestr(conffile);
|
|
if (user_name != NULL)
|
|
freestr(user_name);
|
|
if (group_name != NULL)
|
|
freestr(group_name);
|
|
if (chroot_dir != NULL)
|
|
freestr(chroot_dir);
|
|
if (f != NULL) {
|
|
memstats(f);
|
|
(void)fclose(f);
|
|
}
|
|
}
|
|
|
|
static void
|
|
ns_exit(void) {
|
|
ns_info(ns_log_default, "named shutting down");
|
|
#ifdef BIND_UPDATE
|
|
dynamic_about_to_exit();
|
|
#endif
|
|
if (server_options && server_options->pid_filename)
|
|
(void)unlink(server_options->pid_filename);
|
|
ns_logstats(ev, NULL, evNowTime(), evConsTime(0, 0));
|
|
if (NS_OPTION_P(OPTION_DEALLOC_ON_EXIT))
|
|
deallocate_everything();
|
|
exit(0);
|
|
}
|
|
|
|
static void
|
|
use_desired_debug(void) {
|
|
#ifdef DEBUG
|
|
sigset_t set;
|
|
int bad;
|
|
|
|
/* protect against race conditions by blocking debugging signals */
|
|
|
|
if (sigemptyset(&set) < 0) {
|
|
ns_error(ns_log_os,
|
|
"sigemptyset failed in use_desired_debug: %s",
|
|
strerror(errno));
|
|
return;
|
|
}
|
|
if (sigaddset(&set, SIGUSR1) < 0) {
|
|
ns_error(ns_log_os,
|
|
"sigaddset SIGUSR1 failed in use_desired_debug: %s",
|
|
strerror(errno));
|
|
return;
|
|
}
|
|
if (sigaddset(&set, SIGUSR2) < 0) {
|
|
ns_error(ns_log_os,
|
|
"sigaddset SIGUSR2 failed in use_desired_debug: %s",
|
|
strerror(errno));
|
|
return;
|
|
}
|
|
if (sigprocmask(SIG_BLOCK, &set, NULL) < 0) {
|
|
ns_error(ns_log_os,
|
|
"sigprocmask to block USR1 and USR2 failed: %s",
|
|
strerror(errno));
|
|
return;
|
|
}
|
|
setdebug(desired_debug);
|
|
if (sigprocmask(SIG_UNBLOCK, &set, NULL) < 0)
|
|
ns_error(ns_log_os,
|
|
"sigprocmask to unblock USR1 and USR2 failed: %s",
|
|
strerror(errno));
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
toggle_qrylog(void) {
|
|
qrylog = !qrylog;
|
|
ns_notice(ns_log_default, "query log %s\n", qrylog ?"on" :"off");
|
|
}
|
|
|
|
#ifdef BIND_NOTIFY
|
|
static void
|
|
do_notify_after_load(void) {
|
|
evDo(ev, (const void *)notify_after_load);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* This is a functional interface to the global needs and options.
|
|
*/
|
|
|
|
static const struct need_handler {
|
|
int need;
|
|
void (*handler)(void);
|
|
} need_handlers[] = {
|
|
{ MAIN_NEED_RELOAD, ns_reload },
|
|
{ MAIN_NEED_MAINT, ns_maint },
|
|
{ MAIN_NEED_ENDXFER, endxfer },
|
|
{ MAIN_NEED_ZONELOAD, loadxfer },
|
|
{ MAIN_NEED_DUMP, doadump },
|
|
{ MAIN_NEED_STATSDUMP, ns_stats },
|
|
{ MAIN_NEED_EXIT, ns_exit },
|
|
{ MAIN_NEED_QRYLOG, toggle_qrylog },
|
|
{ MAIN_NEED_DEBUG, use_desired_debug },
|
|
#ifdef BIND_NOTIFY
|
|
{ MAIN_NEED_NOTIFY, do_notify_after_load },
|
|
#endif
|
|
{ 0, NULL }
|
|
};
|
|
|
|
void
|
|
ns_setoption(int option) {
|
|
ns_warning(ns_log_default, "used obsolete ns_setoption(%d)", option);
|
|
}
|
|
|
|
void
|
|
ns_need(int need) {
|
|
needs |= need;
|
|
}
|
|
|
|
int
|
|
ns_need_p(int need) {
|
|
return ((needs & need) != 0);
|
|
}
|
|
|
|
static void
|
|
ns_handle_needs() {
|
|
const struct need_handler *nhp;
|
|
|
|
for (nhp = need_handlers; nhp->need && nhp->handler; nhp++) {
|
|
if ((needs & nhp->need) != 0) {
|
|
/*
|
|
* Turn off flag first, handler might turn it back on.
|
|
*/
|
|
needs &= ~nhp->need;
|
|
(*nhp->handler)();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
writestream(struct qstream *sp, const u_char *msg, int msglen) {
|
|
if (sq_openw(sp, msglen + INT16SZ) == -1) {
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
if (sq_write(sp, msg, msglen) == -1) {
|
|
sq_remove(sp);
|
|
return;
|
|
}
|
|
sq_writeh(sp, sq_flushw);
|
|
}
|
|
|
|
static void
|
|
set_signal_handler(int sig, SIG_FN (*handler)()) {
|
|
struct sigaction sa;
|
|
|
|
memset(&sa, 0, sizeof sa);
|
|
sa.sa_handler = handler;
|
|
if (sigemptyset(&sa.sa_mask) < 0) {
|
|
ns_error(ns_log_os,
|
|
"sigemptyset failed in set_signal_handler(%d): %s",
|
|
sig, strerror(errno));
|
|
return;
|
|
}
|
|
if (sigaction(sig, &sa, NULL) < 0)
|
|
ns_error(ns_log_os,
|
|
"sigaction failed in set_signal_handler(%d): %s",
|
|
sig, strerror(errno));
|
|
}
|
|
|
|
static void
|
|
init_signals() {
|
|
set_signal_handler(SIGINT, setdumpflg);
|
|
set_signal_handler(SIGILL, setstatsflg);
|
|
#ifdef DEBUG
|
|
set_signal_handler(SIGUSR1, setIncrDbgFlg);
|
|
set_signal_handler(SIGUSR2, setNoDbgFlg);
|
|
#endif
|
|
set_signal_handler(SIGHUP, onhup);
|
|
#if defined(SIGWINCH) && defined(QRYLOG) /* XXX */
|
|
set_signal_handler(SIGWINCH, setQrylogFlg);
|
|
#endif
|
|
set_signal_handler(SIGCHLD, reapchild);
|
|
set_signal_handler(SIGPIPE, discard_pipe);
|
|
set_signal_handler(SIGTERM, onintr);
|
|
#if defined(SIGXFSZ) /* XXX */
|
|
/* Wierd DEC Hesiodism, harmless. */
|
|
set_signal_handler(SIGXFSZ, onhup);
|
|
#endif
|
|
}
|
|
|
|
static int
|
|
only_digits(const char *s) {
|
|
if (*s == '\0')
|
|
return (0);
|
|
while (*s != '\0') {
|
|
if (!isdigit(*s))
|
|
return (0);
|
|
s++;
|
|
}
|
|
return (1);
|
|
}
|