809 lines
20 KiB
C
809 lines
20 KiB
C
|
#if !defined(lint) && !defined(SABER)
|
||
|
static char sccsid[] = "@(#)ns_init.c 4.38 (Berkeley) 3/21/91";
|
||
|
static char rcsid[] = "$Id:";
|
||
|
#endif /* not lint */
|
||
|
|
||
|
/*
|
||
|
* ++Copyright++ 1986, 1990
|
||
|
* -
|
||
|
* Copyright (c) 1986, 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.
|
||
|
* -
|
||
|
* --Copyright--
|
||
|
*/
|
||
|
|
||
|
#include <sys/param.h>
|
||
|
#include <sys/socket.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <netinet/in.h>
|
||
|
#include <arpa/nameser.h>
|
||
|
#include <arpa/inet.h>
|
||
|
#include <syslog.h>
|
||
|
#include <signal.h>
|
||
|
#include <resolv.h>
|
||
|
#include <stdio.h>
|
||
|
#include <errno.h>
|
||
|
#include <ctype.h>
|
||
|
|
||
|
#include "named.h"
|
||
|
|
||
|
#undef nsaddr
|
||
|
|
||
|
static void zoneinit __P((struct zoneinfo *)),
|
||
|
get_forwarders __P((FILE *)),
|
||
|
boot_read __P((char *)),
|
||
|
#ifdef DEBUG
|
||
|
content_zone __P((int)),
|
||
|
#endif
|
||
|
free_forwarders __P((void));
|
||
|
|
||
|
static struct zoneinfo *find_zone __P((char *, int, int));
|
||
|
|
||
|
/*
|
||
|
* Read boot file for configuration info.
|
||
|
*/
|
||
|
void
|
||
|
ns_init(bootfile)
|
||
|
char *bootfile;
|
||
|
{
|
||
|
register struct zoneinfo *zp;
|
||
|
static int loads = 0; /* number of times loaded */
|
||
|
|
||
|
dprintf(1, (ddt, "\nns_init(%s)\n", bootfile));
|
||
|
gettime(&tt);
|
||
|
|
||
|
if (loads == 0) {
|
||
|
if ((zones =
|
||
|
(struct zoneinfo *)calloc(64, sizeof(struct zoneinfo)))
|
||
|
== NULL) {
|
||
|
syslog(LOG_ERR,
|
||
|
"Not enough memory to allocate initial zones array");
|
||
|
exit(1);
|
||
|
}
|
||
|
nzones = 1; /* zone zero is cache data */
|
||
|
/* allocate cache hash table, formerly the root hash table. */
|
||
|
hashtab = savehash((struct hashbuf *)NULL);
|
||
|
|
||
|
/* allocate root-hints/file-cache hash table */
|
||
|
fcachetab = savehash((struct hashbuf *)NULL);
|
||
|
/* init zone data */
|
||
|
zones[0].z_type = Z_CACHE;
|
||
|
} else {
|
||
|
/* Mark previous zones as not yet found in boot file. */
|
||
|
for (zp = &zones[1]; zp < &zones[nzones]; zp++)
|
||
|
zp->z_flags &= ~Z_FOUND;
|
||
|
#ifdef LOCALDOM
|
||
|
if (localdomain) {
|
||
|
free(localdomain);
|
||
|
localdomain = NULL;
|
||
|
}
|
||
|
#endif
|
||
|
free_forwarders();
|
||
|
free_netlist(enettab);
|
||
|
#ifdef XFRNETS
|
||
|
free_netlist(&xfrnets);
|
||
|
#endif
|
||
|
#ifdef BOGUSNS
|
||
|
free_netlist(&boglist);
|
||
|
#endif
|
||
|
forward_only = 0;
|
||
|
}
|
||
|
|
||
|
dprintf(3, (ddt, "\n content of zones before loading \n"));
|
||
|
#ifdef DEBUG
|
||
|
if (debug >= 3) {
|
||
|
content_zone(nzones - 1);
|
||
|
}
|
||
|
#endif
|
||
|
boot_read(bootfile);
|
||
|
|
||
|
/* erase all old zones that were not found */
|
||
|
for (zp = &zones[1]; zp < &zones[nzones]; zp++) {
|
||
|
if (zp->z_type && (zp->z_flags & Z_FOUND) == 0) {
|
||
|
#ifdef CLEANCACHE
|
||
|
remove_zone(hashtab, zp - zones, 1);
|
||
|
#else
|
||
|
remove_zone(hashtab, zp - zones);
|
||
|
#endif
|
||
|
#ifdef SECURE_ZONES
|
||
|
free_netlist(&zp->secure_nets);
|
||
|
#endif
|
||
|
syslog(LOG_NOTICE, "Zone \"%s\" was removed", zp->z_origin);
|
||
|
free(zp->z_origin);
|
||
|
free(zp->z_source);
|
||
|
bzero((char *) zp, sizeof(*zp));
|
||
|
}
|
||
|
}
|
||
|
dprintf(2, (ddt,"\n content of zones after loading\n"));
|
||
|
|
||
|
#ifdef DEBUG
|
||
|
if (debug >= 2) {
|
||
|
content_zone(nzones-1);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
* Schedule calls to ns_maint().
|
||
|
*/
|
||
|
if (!needmaint)
|
||
|
sched_maint();
|
||
|
dprintf(1, (ddt, "exit ns_init()%s\n",
|
||
|
needmaint ? ", need maintenance immediately" : ""));
|
||
|
loads++;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Read the actual boot file.
|
||
|
* Set up to recurse.
|
||
|
*/
|
||
|
static void
|
||
|
boot_read(bootfile)
|
||
|
char *bootfile;
|
||
|
{
|
||
|
register struct zoneinfo *zp;
|
||
|
char buf[BUFSIZ], obuf[BUFSIZ], *source;
|
||
|
FILE *fp;
|
||
|
int type;
|
||
|
int class;
|
||
|
#ifdef GEN_AXFR
|
||
|
char *class_p;
|
||
|
#endif
|
||
|
struct stat f_time;
|
||
|
static int tmpnum = 0; /* unique number for tmp zone files */
|
||
|
#ifdef ALLOW_UPDATES
|
||
|
char *cp, *flag;
|
||
|
#endif
|
||
|
int slineno; /* Saved global line number. */
|
||
|
int i;
|
||
|
|
||
|
if ((fp = fopen(bootfile, "r")) == NULL) {
|
||
|
syslog(LOG_ERR, "%s: %m", bootfile);
|
||
|
exit(1);
|
||
|
}
|
||
|
|
||
|
slineno = lineno;
|
||
|
lineno = 0;
|
||
|
|
||
|
while (!feof(fp) && !ferror(fp)) {
|
||
|
if (!getword(buf, sizeof(buf), fp))
|
||
|
continue;
|
||
|
/* read named.boot keyword and process args */
|
||
|
if (strcasecmp(buf, "directory") == 0) {
|
||
|
(void) getword(buf, sizeof(buf), fp);
|
||
|
if (chdir(buf) < 0) {
|
||
|
syslog(LOG_CRIT, "directory %s: %m\n",
|
||
|
buf);
|
||
|
exit(1);
|
||
|
}
|
||
|
continue;
|
||
|
} else if (strcasecmp(buf, "sortlist") == 0) {
|
||
|
get_netlist(fp, enettab, ALLOW_NETS, buf);
|
||
|
continue;
|
||
|
} else if (strcasecmp(buf, "max-fetch") == 0) {
|
||
|
max_xfers_running = getnum(fp, bootfile, 0);
|
||
|
continue;
|
||
|
} else if (strcasecmp(buf, "forwarders") == 0) {
|
||
|
get_forwarders(fp);
|
||
|
continue;
|
||
|
} else if (strcasecmp(buf, "slave") == 0) {
|
||
|
forward_only++;
|
||
|
endline(fp);
|
||
|
continue;
|
||
|
#ifdef BOGUSNS
|
||
|
} else if (strcasecmp(buf, "bogusns") == 0) {
|
||
|
get_netlist(fp, &boglist, ALLOW_HOSTS, buf);
|
||
|
continue;
|
||
|
#endif
|
||
|
#ifdef XFRNETS
|
||
|
} else if ((strcasecmp(buf, "tcplist") == 0) ||
|
||
|
(strcasecmp(buf, "xfrnets") == 0)) {
|
||
|
get_netlist(fp, &xfrnets, ALLOW_NETS, buf);
|
||
|
continue;
|
||
|
#endif
|
||
|
#ifdef LOCALDOM
|
||
|
} else if (strcasecmp(buf, "domain") == 0) {
|
||
|
if (getword(buf, sizeof(buf), fp))
|
||
|
localdomain = savestr(buf);
|
||
|
endline(fp);
|
||
|
continue;
|
||
|
#endif
|
||
|
} else if (strcasecmp(buf, "include") == 0) {
|
||
|
if (getword(buf, sizeof(buf), fp))
|
||
|
boot_read(buf);
|
||
|
endline(fp);
|
||
|
continue;
|
||
|
} else if (strncasecmp(buf, "cache", 5) == 0) {
|
||
|
type = Z_CACHE;
|
||
|
class = C_IN;
|
||
|
#ifdef GEN_AXFR
|
||
|
if (class_p = strchr(buf, '/'))
|
||
|
class = get_class(class_p+1);
|
||
|
#endif
|
||
|
} else if (strncasecmp(buf, "primary", 7) == 0) {
|
||
|
type = Z_PRIMARY;
|
||
|
class = C_IN;
|
||
|
#ifdef GEN_AXFR
|
||
|
if (class_p = strchr(buf, '/'))
|
||
|
class = get_class(class_p+1);
|
||
|
#endif
|
||
|
} else if (strncasecmp(buf, "secondary", 9) == 0) {
|
||
|
type = Z_SECONDARY;
|
||
|
class = C_IN;
|
||
|
#ifdef GEN_AXFR
|
||
|
if (class_p = strchr(buf, '/'))
|
||
|
class = get_class(class_p+1);
|
||
|
#endif
|
||
|
#ifdef STUBS
|
||
|
} else if (strncasecmp(buf, "stub", 4) == 0) {
|
||
|
type = Z_STUB;
|
||
|
class = C_IN;
|
||
|
#ifdef GEN_AXFR
|
||
|
if (class_p = strchr(buf, '/'))
|
||
|
class = get_class(class_p+1);
|
||
|
#endif
|
||
|
#endif
|
||
|
} else {
|
||
|
syslog(LOG_ERR, "%s: line %d: unknown field '%s'\n",
|
||
|
bootfile, lineno+1, buf);
|
||
|
endline(fp);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* read zone origin
|
||
|
*/
|
||
|
if (!getword(obuf, sizeof(obuf), fp)) {
|
||
|
syslog(LOG_ERR, "%s: line %d: missing origin\n",
|
||
|
bootfile, lineno);
|
||
|
continue;
|
||
|
}
|
||
|
i = strlen(obuf);
|
||
|
if ((obuf[i-1] == '.') && (i != 1))
|
||
|
syslog(LOG_ERR, "%s: line %d: zone \"%s\" has trailing dot\n",
|
||
|
bootfile, lineno, obuf);
|
||
|
while ((--i >= 0) && (obuf[i] == '.'))
|
||
|
obuf[i] = '\0';
|
||
|
dprintf(1, (ddt, "zone origin %s", obuf[0]?obuf:"."));
|
||
|
/*
|
||
|
* read source file or host address
|
||
|
*/
|
||
|
if (!getword(buf, sizeof(buf), fp)) {
|
||
|
syslog(LOG_ERR, "%s: line %d: missing %s\n",
|
||
|
bootfile, lineno,
|
||
|
#ifdef STUBS
|
||
|
(type == Z_SECONDARY || type == Z_STUB)
|
||
|
#else
|
||
|
(type == Z_SECONDARY)
|
||
|
#endif
|
||
|
?"host address"
|
||
|
:"source file");
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
/* check for previous instance of this zone (reload) */
|
||
|
if (!(zp = find_zone(obuf, type, class))) {
|
||
|
if (type == Z_CACHE) {
|
||
|
zp = &zones[0];
|
||
|
zp->z_origin = "";
|
||
|
goto gotcache;
|
||
|
}
|
||
|
for (zp = &zones[1]; zp < &zones[nzones]; zp++)
|
||
|
if (zp->z_type == Z_NIL)
|
||
|
goto gotzone;
|
||
|
/*
|
||
|
* this code assumes that nzones never decreases
|
||
|
*/
|
||
|
if (nzones % 64 == 0) {
|
||
|
dprintf(1, (ddt,
|
||
|
"Reallocating zones structure\n"));
|
||
|
/*
|
||
|
* Realloc() not used since it might damage zones
|
||
|
* if an error occurs
|
||
|
*/
|
||
|
zp = (struct zoneinfo *)
|
||
|
malloc((64 + nzones)
|
||
|
* sizeof(struct zoneinfo));
|
||
|
if (zp == (struct zoneinfo *)0) {
|
||
|
syslog(LOG_ERR,
|
||
|
"no memory for more zones");
|
||
|
dprintf(1, (ddt,
|
||
|
"Out of memory for new zones\n"
|
||
|
)
|
||
|
);
|
||
|
endline(fp);
|
||
|
continue;
|
||
|
}
|
||
|
bcopy((char *)zones, (char *)zp,
|
||
|
nzones * sizeof(struct zoneinfo));
|
||
|
bzero((char *)&zp[nzones],
|
||
|
64 * sizeof(struct zoneinfo));
|
||
|
free(zones);
|
||
|
zones = zp;
|
||
|
}
|
||
|
zp = &zones[nzones++];
|
||
|
gotzone:
|
||
|
zp->z_origin = savestr(obuf);
|
||
|
gotcache:
|
||
|
zp->z_type = type;
|
||
|
zp->z_class = class;
|
||
|
}
|
||
|
zp->z_addrcnt = 0;
|
||
|
|
||
|
switch (type) {
|
||
|
case Z_CACHE:
|
||
|
source = savestr(buf);
|
||
|
dprintf(1, (ddt, ", source = %s\n", source));
|
||
|
zp->z_refresh = 0; /* by default, no dumping */
|
||
|
if (getword(buf, sizeof(buf), fp)) {
|
||
|
#ifdef notyet
|
||
|
zp->z_refresh = atoi(buf);
|
||
|
if (zp->z_refresh <= 0) {
|
||
|
syslog(LOG_ERR,
|
||
|
"%s: line %d: bad refresh time '%s', ignored\n",
|
||
|
bootfile, lineno, buf);
|
||
|
zp->z_refresh = 0;
|
||
|
} else if (cache_file == NULL)
|
||
|
cache_file = source;
|
||
|
#else
|
||
|
syslog(LOG_WARNING,
|
||
|
"%s: line %d: cache refresh ignored\n",
|
||
|
bootfile, lineno);
|
||
|
#endif
|
||
|
endline(fp);
|
||
|
}
|
||
|
/*
|
||
|
* If we've loaded this file, and the file has
|
||
|
* not been modified and contains no $include,
|
||
|
* then there's no need to reload.
|
||
|
*/
|
||
|
if (zp->z_source &&
|
||
|
!strcmp(source, zp->z_source) &&
|
||
|
!(zp->z_flags & Z_INCLUDE) &&
|
||
|
stat(zp->z_source, &f_time) != -1 &&
|
||
|
zp->z_ftime == f_time.st_mtime) {
|
||
|
dprintf(1, (ddt, "cache is up to date\n"));
|
||
|
if (source != cache_file)
|
||
|
free(source);
|
||
|
break; /* zone is already up to date */
|
||
|
}
|
||
|
|
||
|
/* file has changed, or hasn't been loaded yet */
|
||
|
if (zp->z_source) {
|
||
|
free(zp->z_source);
|
||
|
#ifdef CLEANCACHE
|
||
|
remove_zone(fcachetab, 0, 1);
|
||
|
#else
|
||
|
remove_zone(fcachetab, 0);
|
||
|
#endif
|
||
|
}
|
||
|
zp->z_source = source;
|
||
|
dprintf(1, (ddt, "reloading zone\n"));
|
||
|
(void) db_load(zp->z_source, zp->z_origin, zp, 0);
|
||
|
break;
|
||
|
|
||
|
case Z_PRIMARY:
|
||
|
source = savestr(buf);
|
||
|
#ifdef ALLOW_UPDATES
|
||
|
if (getword(buf, sizeof(buf), fp)) {
|
||
|
endline(fp);
|
||
|
flag = buf;
|
||
|
while (flag) {
|
||
|
cp = strchr(flag, ',');
|
||
|
if (cp)
|
||
|
*cp++ = 0;
|
||
|
if (strcasecmp(flag, "dynamic") == 0)
|
||
|
zp->z_flags |= Z_DYNAMIC;
|
||
|
else if (strcasecmp(flag, "addonly") == 0)
|
||
|
zp->z_flags |= Z_DYNADDONLY;
|
||
|
else {
|
||
|
syslog(LOG_ERR,
|
||
|
"%s: line %d: bad flag '%s'\n",
|
||
|
bootfile, lineno, flag);
|
||
|
}
|
||
|
flag = cp;
|
||
|
}
|
||
|
}
|
||
|
#else /*ALLOW_UPDATES*/
|
||
|
endline(fp);
|
||
|
#endif
|
||
|
|
||
|
dprintf(1, (ddt, ", source = %s\n", source));
|
||
|
/*
|
||
|
* If we've loaded this file, and the file has
|
||
|
* not been modified and contains no $include,
|
||
|
* then there's no need to reload.
|
||
|
*/
|
||
|
if (zp->z_source &&
|
||
|
!strcmp(source, zp->z_source) &&
|
||
|
!(zp->z_flags & Z_INCLUDE) &&
|
||
|
stat(zp->z_source, &f_time) != -1 &&
|
||
|
zp->z_ftime == f_time.st_mtime) {
|
||
|
dprintf(1, (ddt, "zone is up to date\n"));
|
||
|
free(source);
|
||
|
break; /* zone is already up to date */
|
||
|
}
|
||
|
if (zp->z_source) {
|
||
|
free(zp->z_source);
|
||
|
#ifdef CLEANCACHE
|
||
|
remove_zone(hashtab, zp - zones, 1);
|
||
|
#else
|
||
|
remove_zone(hashtab, zp - zones);
|
||
|
#endif
|
||
|
}
|
||
|
zp->z_source = source;
|
||
|
zp->z_flags &= ~Z_AUTH;
|
||
|
dprintf(1, (ddt, "reloading zone\n"));
|
||
|
if (db_load(zp->z_source, zp->z_origin, zp, 0) == 0)
|
||
|
zp->z_flags |= Z_AUTH;
|
||
|
#ifdef ALLOW_UPDATES
|
||
|
/* Guarantee calls to ns_maint() */
|
||
|
zp->z_refresh = maint_interval;
|
||
|
#else
|
||
|
zp->z_refresh = 0; /* no maintenance needed */
|
||
|
zp->z_time = 0;
|
||
|
#endif
|
||
|
break;
|
||
|
|
||
|
case Z_SECONDARY:
|
||
|
#ifdef STUBS
|
||
|
case Z_STUB:
|
||
|
#endif
|
||
|
source = NULL;
|
||
|
dprintf(1, (ddt, "\n\taddrs: "));
|
||
|
do {
|
||
|
if (!inet_aton(buf,
|
||
|
&zp->z_addr[zp->z_addrcnt])
|
||
|
) {
|
||
|
source = savestr(buf);
|
||
|
endline(fp);
|
||
|
break;
|
||
|
}
|
||
|
dprintf(1, (ddt, "%s, ", buf));
|
||
|
if ((int)++zp->z_addrcnt > NSMAX - 1) {
|
||
|
zp->z_addrcnt = NSMAX - 1;
|
||
|
dprintf(1, (ddt,
|
||
|
"\nns.h NSMAX reached\n"));
|
||
|
}
|
||
|
} while (getword(buf, sizeof(buf), fp));
|
||
|
dprintf(1, (ddt, "addrcnt = %d\n", zp->z_addrcnt));
|
||
|
if (!source) {
|
||
|
/*
|
||
|
* We will always transfer this zone again
|
||
|
* after a reload.
|
||
|
*/
|
||
|
sprintf(buf, "/%s/NsTmp%d.%d", _PATH_TMPDIR,
|
||
|
getpid(), tmpnum++);
|
||
|
source = savestr(buf);
|
||
|
zp->z_flags |= Z_TMP_FILE;
|
||
|
} else
|
||
|
zp->z_flags &= ~Z_TMP_FILE;
|
||
|
/*
|
||
|
* If we had a backup file name, and it was changed,
|
||
|
* free old zone and start over. If we don't have
|
||
|
* current zone contents, try again now in case
|
||
|
* we have a new server on the list.
|
||
|
*/
|
||
|
if (zp->z_source &&
|
||
|
(strcmp(source, zp->z_source) ||
|
||
|
(stat(zp->z_source, &f_time) == -1 ||
|
||
|
(zp->z_ftime != f_time.st_mtime)))) {
|
||
|
dprintf(1, (ddt, "backup file changed\n"));
|
||
|
free(zp->z_source);
|
||
|
zp->z_source = NULL;
|
||
|
zp->z_flags &= ~Z_AUTH;
|
||
|
zp->z_serial = 0; /* force xfer */
|
||
|
#ifdef CLEANCACHE
|
||
|
remove_zone(hashtab, zp - zones, 1);
|
||
|
#else
|
||
|
remove_zone(hashtab, zp - zones);
|
||
|
#endif
|
||
|
}
|
||
|
if (zp->z_source)
|
||
|
free(source);
|
||
|
else
|
||
|
zp->z_source = source;
|
||
|
if (!(zp->z_flags & Z_AUTH))
|
||
|
zoneinit(zp);
|
||
|
#ifdef FORCED_RELOAD
|
||
|
else {
|
||
|
/*
|
||
|
** Force secondary to try transfer right away
|
||
|
** after SIGHUP.
|
||
|
*/
|
||
|
if (reloading) {
|
||
|
zp->z_time = tt.tv_sec;
|
||
|
needmaint = 1;
|
||
|
}
|
||
|
}
|
||
|
#endif /* FORCED_RELOAD */
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
zp->z_flags |= Z_FOUND;
|
||
|
dprintf(1, (ddt, "zone[%d] type %d: '%s'",
|
||
|
zp-zones, type,
|
||
|
*(zp->z_origin) == '\0' ? "." : zp->z_origin));
|
||
|
if (zp->z_refresh && zp->z_time == 0)
|
||
|
zp->z_time = zp->z_refresh + tt.tv_sec;
|
||
|
if (zp->z_time <= tt.tv_sec)
|
||
|
needmaint = 1;
|
||
|
dprintf(1, (ddt, " z_time %d, z_refresh %d\n",
|
||
|
zp->z_time, zp->z_refresh));
|
||
|
}
|
||
|
(void) my_fclose(fp);
|
||
|
lineno = slineno;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zoneinit(zp)
|
||
|
register struct zoneinfo *zp;
|
||
|
{
|
||
|
struct stat sb;
|
||
|
|
||
|
/*
|
||
|
* Try to load zone from backup file,
|
||
|
* if one was specified and it exists.
|
||
|
* If not, or if the data are out of date,
|
||
|
* we will refresh the zone from a primary
|
||
|
* immediately.
|
||
|
*/
|
||
|
if (!zp->z_source)
|
||
|
return;
|
||
|
if (stat(zp->z_source, &sb) == -1 ||
|
||
|
db_load(zp->z_source, zp->z_origin, zp, 0) != 0) {
|
||
|
/*
|
||
|
* Set zone to be refreshed immediately.
|
||
|
*/
|
||
|
zp->z_refresh = INIT_REFRESH;
|
||
|
zp->z_retry = INIT_REFRESH;
|
||
|
zp->z_time = tt.tv_sec;
|
||
|
needmaint = 1;
|
||
|
} else {
|
||
|
zp->z_flags |= Z_AUTH;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef ALLOW_UPDATES
|
||
|
/*
|
||
|
* Look for the authoritative zone with the longest matching RHS of dname
|
||
|
* and return its zone # or zero if not found.
|
||
|
*/
|
||
|
int
|
||
|
findzone(dname, class)
|
||
|
char *dname;
|
||
|
int class;
|
||
|
{
|
||
|
char *dZoneName, *zoneName;
|
||
|
int dZoneNameLen, zoneNameLen;
|
||
|
int maxMatchLen = 0;
|
||
|
int maxMatchZoneNum = 0;
|
||
|
int zoneNum;
|
||
|
|
||
|
dprintf(4, (ddt, "findzone(dname=%s, class=%d)\n", dname, class));
|
||
|
#ifdef DEBUG
|
||
|
if (debug >= 5) {
|
||
|
fprintf(ddt, "zone dump:\n");
|
||
|
for (zoneNum = 1; zoneNum < nzones; zoneNum++)
|
||
|
printzoneinfo(zoneNum);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
dZoneName = strchr(dname, '.');
|
||
|
if (dZoneName == NULL)
|
||
|
dZoneName = ""; /* root */
|
||
|
else
|
||
|
dZoneName++; /* There is a '.' in dname, so use remainder of
|
||
|
string as the zone name */
|
||
|
dZoneNameLen = strlen(dZoneName);
|
||
|
for (zoneNum = 1; zoneNum < nzones; zoneNum++) {
|
||
|
zoneName = (zones[zoneNum]).z_origin;
|
||
|
zoneNameLen = strlen(zoneName);
|
||
|
/* The zone name may or may not end with a '.' */
|
||
|
if (zoneName[zoneNameLen - 1] == '.')
|
||
|
zoneNameLen--;
|
||
|
if (dZoneNameLen != zoneNameLen)
|
||
|
continue;
|
||
|
dprintf(5, (ddt, "about to strncasecmp('%s', '%s', %d)\n",
|
||
|
dZoneName, zoneName, dZoneNameLen));
|
||
|
if (strncasecmp(dZoneName, zoneName, dZoneNameLen) == 0) {
|
||
|
dprintf(5, (ddt, "match\n"));
|
||
|
/*
|
||
|
* See if this is as long a match as any so far.
|
||
|
* Check if "<=" instead of just "<" so that if
|
||
|
* root domain (whose name length is 0) matches,
|
||
|
* we use it's zone number instead of just 0
|
||
|
*/
|
||
|
if (maxMatchLen <= zoneNameLen) {
|
||
|
maxMatchZoneNum = zoneNum;
|
||
|
maxMatchLen = zoneNameLen;
|
||
|
}
|
||
|
} else {
|
||
|
dprintf(5, (ddt, "no match\n"));
|
||
|
}
|
||
|
}
|
||
|
dprintf(4, (ddt, "findzone: returning %d\n", maxMatchZoneNum));
|
||
|
return (maxMatchZoneNum);
|
||
|
}
|
||
|
#endif /* ALLOW_UPDATES */
|
||
|
|
||
|
static void
|
||
|
get_forwarders(fp)
|
||
|
FILE *fp;
|
||
|
{
|
||
|
char buf[BUFSIZ];
|
||
|
register struct fwdinfo *fip = NULL, *ftp = NULL;
|
||
|
|
||
|
#ifdef SLAVE_FORWARD
|
||
|
int forward_count = 0;
|
||
|
#endif
|
||
|
|
||
|
dprintf(1, (ddt, "forwarders "));
|
||
|
|
||
|
/* on mulitple forwarder lines, move to end of the list */
|
||
|
#ifdef SLAVE_FORWARD
|
||
|
if (fwdtab != NULL){
|
||
|
forward_count++;
|
||
|
for (fip = fwdtab; fip->next != NULL; fip = fip->next)
|
||
|
forward_count++;
|
||
|
}
|
||
|
#else
|
||
|
if (fwdtab != NULL) {
|
||
|
for (fip = fwdtab; fip->next != NULL; fip = fip->next) {
|
||
|
;
|
||
|
}
|
||
|
}
|
||
|
#endif /* SLAVE_FORWARD */
|
||
|
|
||
|
while (getword(buf, sizeof(buf), fp)) {
|
||
|
if (strlen(buf) == 0)
|
||
|
break;
|
||
|
dprintf(1, (ddt," %s",buf));
|
||
|
if (ftp == NULL)
|
||
|
ftp = (struct fwdinfo *)malloc(sizeof(struct fwdinfo));
|
||
|
if (inet_aton(buf, &ftp->fwdaddr.sin_addr)) {
|
||
|
ftp->fwdaddr.sin_port = ns_port;
|
||
|
ftp->fwdaddr.sin_family = AF_INET;
|
||
|
} else {
|
||
|
syslog(LOG_ERR, "'%s' (ignored, NOT dotted quad)",
|
||
|
buf);
|
||
|
dprintf(1, (ddt, " (ignored, NOT dotted quad)"));
|
||
|
continue;
|
||
|
}
|
||
|
#ifdef FWD_LOOP
|
||
|
if (aIsUs(ftp->fwdaddr.sin_addr)) {
|
||
|
syslog(LOG_ERR,
|
||
|
"Forwarder '%s' ignored, my address",
|
||
|
buf);
|
||
|
dprintf(1, (ddt, " (ignored, my address)"));
|
||
|
continue;
|
||
|
}
|
||
|
#endif /* FWD_LOOP */
|
||
|
ftp->next = NULL;
|
||
|
if (fwdtab == NULL)
|
||
|
fwdtab = ftp; /* First time only */
|
||
|
else
|
||
|
fip->next = ftp;
|
||
|
fip = ftp;
|
||
|
ftp = NULL;
|
||
|
#ifdef SLAVE_FORWARD
|
||
|
forward_count++;
|
||
|
#endif /* SLAVE_FORWARD */
|
||
|
}
|
||
|
if (ftp)
|
||
|
free((char *)ftp);
|
||
|
|
||
|
#ifdef SLAVE_FORWARD
|
||
|
/*
|
||
|
** Set the slave retry time to 60 seconds total divided
|
||
|
** between each forwarder
|
||
|
*/
|
||
|
if (forward_count != 0) {
|
||
|
slave_retry = (int) (60 / forward_count);
|
||
|
if(slave_retry <= 0)
|
||
|
slave_retry = 1;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
dprintf(1, (ddt, "\n"));
|
||
|
#ifdef DEBUG
|
||
|
if (debug > 2) {
|
||
|
for (ftp = fwdtab; ftp != NULL; ftp = ftp->next) {
|
||
|
fprintf(ddt,"ftp x%x [%s] next x%x\n",
|
||
|
ftp,
|
||
|
inet_ntoa(ftp->fwdaddr.sin_addr),
|
||
|
ftp->next);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
free_forwarders()
|
||
|
{
|
||
|
register struct fwdinfo *ftp, *fnext;
|
||
|
|
||
|
for (ftp = fwdtab; ftp != NULL; ftp = fnext) {
|
||
|
fnext = ftp->next;
|
||
|
free((char *)ftp);
|
||
|
}
|
||
|
fwdtab = NULL;
|
||
|
}
|
||
|
|
||
|
static struct zoneinfo *
|
||
|
find_zone(name, type, class)
|
||
|
char *name;
|
||
|
int type, class;
|
||
|
{
|
||
|
register struct zoneinfo *zp;
|
||
|
|
||
|
for (zp = &zones[1]; zp < &zones[nzones]; zp++) {
|
||
|
if (zp->z_type == type && zp->z_class == class &&
|
||
|
strcasecmp(name, zp->z_origin) == 0) {
|
||
|
dprintf(2, (ddt, ", old zone (%d)", zp - zones));
|
||
|
return (zp);
|
||
|
}
|
||
|
}
|
||
|
dprintf(2, (ddt, ", new zone"));
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#ifdef DEBUG
|
||
|
/* prints out the content of zones */
|
||
|
static void
|
||
|
content_zone(end)
|
||
|
int end;
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
for (i = 1; i <= end; i++) {
|
||
|
printzoneinfo(i);
|
||
|
}
|
||
|
}
|
||
|
#endif
|