/* $OpenBSD: config.c,v 1.30 2019/05/28 06:49:46 otto Exp $ */ /* * Copyright (c) 2003, 2004 Henning Brauer * * 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 THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR 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. */ #include #include #include #include #include #include #include #include #include #include "ntpd.h" struct ntp_addr *host_ip(const char *); int host_dns1(const char *, struct ntp_addr **, int); static u_int32_t maxid = 0; static u_int32_t constraint_maxid = 0; void host(const char *s, struct ntp_addr **hn) { struct ntp_addr *h; if (!strcmp(s, "*")) { if ((h = calloc(1, sizeof(*h))) == NULL) fatal(NULL); } else { if ((h = host_ip(s)) == NULL) return; } *hn = h; } struct ntp_addr * host_ip(const char *s) { struct addrinfo hints, *res; struct ntp_addr *h = NULL; memset(&hints, 0, sizeof(hints)); hints.ai_family = AF_UNSPEC; hints.ai_socktype = SOCK_DGRAM; /*dummy*/ hints.ai_flags = AI_NUMERICHOST; if (getaddrinfo(s, "0", &hints, &res) == 0) { if (res->ai_family == AF_INET || res->ai_family == AF_INET6) { if ((h = calloc(1, sizeof(*h))) == NULL) fatal(NULL); memcpy(&h->ss, res->ai_addr, res->ai_addrlen); } freeaddrinfo(res); } return (h); } void host_dns_free(struct ntp_addr *hn) { struct ntp_addr *h = hn, *tmp; while (h) { tmp = h; h = h->next; free(tmp); } } int host_dns1(const char *s, struct ntp_addr **hn, int notauth) { struct addrinfo hints, *res0, *res; int error, cnt = 0; struct ntp_addr *h, *hh = NULL; memset(&hints, 0, sizeof(hints)); hints.ai_family = AF_UNSPEC; hints.ai_socktype = SOCK_DGRAM; /* DUMMY */ /* ntpd MUST NOT use AI_ADDRCONFIG here */ error = getaddrinfo(s, NULL, &hints, &res0); if (error == EAI_AGAIN || error == EAI_NODATA || error == EAI_NONAME) return (0); if (error) { log_warnx("could not parse \"%s\": %s", s, gai_strerror(error)); return (-1); } for (res = res0; res && cnt < MAX_SERVERS_DNS; res = res->ai_next) { if (res->ai_family != AF_INET && res->ai_family != AF_INET6) continue; if ((h = calloc(1, sizeof(*h))) == NULL) fatal(NULL); memcpy(&h->ss, res->ai_addr, res->ai_addrlen); h->notauth = notauth; h->next = hh; hh = h; cnt++; } freeaddrinfo(res0); *hn = hh; return (cnt); } int host_dns(const char *s, struct ntp_addr **hn) { int error, save_opts; log_debug("trying to resolve %s", s); error = host_dns1(s, hn, 0); if (error <= 0) { log_debug("no luck, trying to resolve %s without checking", s); save_opts = _res.options; _res.options |= RES_USE_CD; error = host_dns1(s, hn, 1); _res.options = save_opts; } log_debug("resolve %s done: %d", s, error); return error; } struct ntp_peer * new_peer(void) { struct ntp_peer *p; if ((p = calloc(1, sizeof(struct ntp_peer))) == NULL) fatal("new_peer calloc"); p->id = ++maxid; return (p); } struct ntp_conf_sensor * new_sensor(char *device) { struct ntp_conf_sensor *s; if ((s = calloc(1, sizeof(struct ntp_conf_sensor))) == NULL) fatal("new_sensor calloc"); if ((s->device = strdup(device)) == NULL) fatal("new_sensor strdup"); return (s); } struct constraint * new_constraint(void) { struct constraint *p; if ((p = calloc(1, sizeof(struct constraint))) == NULL) fatal("new_constraint calloc"); p->id = ++constraint_maxid; p->fd = -1; return (p); }