17c478bd9Sstevel@tonic-gate /*
2*9525b14bSRao Shoaib * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC")
37c478bd9Sstevel@tonic-gate * Copyright (c) 1996,1999 by Internet Software Consortium.
47c478bd9Sstevel@tonic-gate *
57c478bd9Sstevel@tonic-gate * Permission to use, copy, modify, and distribute this software for any
67c478bd9Sstevel@tonic-gate * purpose with or without fee is hereby granted, provided that the above
77c478bd9Sstevel@tonic-gate * copyright notice and this permission notice appear in all copies.
87c478bd9Sstevel@tonic-gate *
9*9525b14bSRao Shoaib * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES
10*9525b14bSRao Shoaib * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11*9525b14bSRao Shoaib * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR
12*9525b14bSRao Shoaib * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13*9525b14bSRao Shoaib * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14*9525b14bSRao Shoaib * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
15*9525b14bSRao Shoaib * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
167c478bd9Sstevel@tonic-gate */
177c478bd9Sstevel@tonic-gate
187c478bd9Sstevel@tonic-gate #if defined(LIBC_SCCS) && !defined(lint)
19*9525b14bSRao Shoaib static const char rcsid[] = "$Id: dns_sv.c,v 1.5 2005/04/27 04:56:23 sra Exp $";
207c478bd9Sstevel@tonic-gate #endif
217c478bd9Sstevel@tonic-gate
227c478bd9Sstevel@tonic-gate /* Imports */
237c478bd9Sstevel@tonic-gate
247c478bd9Sstevel@tonic-gate #include "port_before.h"
257c478bd9Sstevel@tonic-gate
267c478bd9Sstevel@tonic-gate #include <sys/types.h>
277c478bd9Sstevel@tonic-gate #include <netinet/in.h>
287c478bd9Sstevel@tonic-gate
297c478bd9Sstevel@tonic-gate #include <stdio.h>
307c478bd9Sstevel@tonic-gate #include <string.h>
317c478bd9Sstevel@tonic-gate #include <netdb.h>
327c478bd9Sstevel@tonic-gate #include <ctype.h>
337c478bd9Sstevel@tonic-gate #include <stdlib.h>
347c478bd9Sstevel@tonic-gate #include <errno.h>
357c478bd9Sstevel@tonic-gate
367c478bd9Sstevel@tonic-gate #include <sys/types.h>
377c478bd9Sstevel@tonic-gate #include <netinet/in.h>
387c478bd9Sstevel@tonic-gate #include <arpa/nameser.h>
397c478bd9Sstevel@tonic-gate #include <resolv.h>
407c478bd9Sstevel@tonic-gate
417c478bd9Sstevel@tonic-gate #include <isc/memcluster.h>
427c478bd9Sstevel@tonic-gate #include <irs.h>
437c478bd9Sstevel@tonic-gate
447c478bd9Sstevel@tonic-gate #include "port_after.h"
457c478bd9Sstevel@tonic-gate
467c478bd9Sstevel@tonic-gate #include "irs_p.h"
477c478bd9Sstevel@tonic-gate #include "hesiod.h"
487c478bd9Sstevel@tonic-gate #include "dns_p.h"
497c478bd9Sstevel@tonic-gate
507c478bd9Sstevel@tonic-gate /* Definitions */
517c478bd9Sstevel@tonic-gate
527c478bd9Sstevel@tonic-gate struct pvt {
537c478bd9Sstevel@tonic-gate struct dns_p * dns;
547c478bd9Sstevel@tonic-gate struct servent serv;
557c478bd9Sstevel@tonic-gate char * svbuf;
567c478bd9Sstevel@tonic-gate struct __res_state * res;
577c478bd9Sstevel@tonic-gate void (*free_res)(void *);
587c478bd9Sstevel@tonic-gate };
597c478bd9Sstevel@tonic-gate
607c478bd9Sstevel@tonic-gate /* Forward. */
617c478bd9Sstevel@tonic-gate
627c478bd9Sstevel@tonic-gate static void sv_close(struct irs_sv *);
637c478bd9Sstevel@tonic-gate static struct servent * sv_byname(struct irs_sv *,
647c478bd9Sstevel@tonic-gate const char *, const char *);
657c478bd9Sstevel@tonic-gate static struct servent * sv_byport(struct irs_sv *, int, const char *);
667c478bd9Sstevel@tonic-gate static struct servent * sv_next(struct irs_sv *);
677c478bd9Sstevel@tonic-gate static void sv_rewind(struct irs_sv *);
687c478bd9Sstevel@tonic-gate static void sv_minimize(struct irs_sv *);
697c478bd9Sstevel@tonic-gate #ifdef SV_RES_SETGET
707c478bd9Sstevel@tonic-gate static struct __res_state * sv_res_get(struct irs_sv *);
717c478bd9Sstevel@tonic-gate static void sv_res_set(struct irs_sv *,
727c478bd9Sstevel@tonic-gate struct __res_state *,
737c478bd9Sstevel@tonic-gate void (*)(void *));
747c478bd9Sstevel@tonic-gate #endif
757c478bd9Sstevel@tonic-gate
767c478bd9Sstevel@tonic-gate static struct servent * parse_hes_list(struct irs_sv *,
777c478bd9Sstevel@tonic-gate char **, const char *);
787c478bd9Sstevel@tonic-gate
797c478bd9Sstevel@tonic-gate /* Public */
807c478bd9Sstevel@tonic-gate
817c478bd9Sstevel@tonic-gate struct irs_sv *
irs_dns_sv(struct irs_acc * this)827c478bd9Sstevel@tonic-gate irs_dns_sv(struct irs_acc *this) {
837c478bd9Sstevel@tonic-gate struct dns_p *dns = (struct dns_p *)this->private;
847c478bd9Sstevel@tonic-gate struct irs_sv *sv;
857c478bd9Sstevel@tonic-gate struct pvt *pvt;
867c478bd9Sstevel@tonic-gate
877c478bd9Sstevel@tonic-gate if (!dns || !dns->hes_ctx) {
887c478bd9Sstevel@tonic-gate errno = ENODEV;
897c478bd9Sstevel@tonic-gate return (NULL);
907c478bd9Sstevel@tonic-gate }
917c478bd9Sstevel@tonic-gate if (!(pvt = memget(sizeof *pvt))) {
927c478bd9Sstevel@tonic-gate errno = ENOMEM;
937c478bd9Sstevel@tonic-gate return (NULL);
947c478bd9Sstevel@tonic-gate }
957c478bd9Sstevel@tonic-gate memset(pvt, 0, sizeof *pvt);
967c478bd9Sstevel@tonic-gate pvt->dns = dns;
977c478bd9Sstevel@tonic-gate if (!(sv = memget(sizeof *sv))) {
987c478bd9Sstevel@tonic-gate memput(pvt, sizeof *pvt);
997c478bd9Sstevel@tonic-gate errno = ENOMEM;
1007c478bd9Sstevel@tonic-gate return (NULL);
1017c478bd9Sstevel@tonic-gate }
1027c478bd9Sstevel@tonic-gate memset(sv, 0x5e, sizeof *sv);
1037c478bd9Sstevel@tonic-gate sv->private = pvt;
1047c478bd9Sstevel@tonic-gate sv->byname = sv_byname;
1057c478bd9Sstevel@tonic-gate sv->byport = sv_byport;
1067c478bd9Sstevel@tonic-gate sv->next = sv_next;
1077c478bd9Sstevel@tonic-gate sv->rewind = sv_rewind;
1087c478bd9Sstevel@tonic-gate sv->close = sv_close;
1097c478bd9Sstevel@tonic-gate sv->minimize = sv_minimize;
1107c478bd9Sstevel@tonic-gate #ifdef SV_RES_SETGET
1117c478bd9Sstevel@tonic-gate sv->res_get = sv_res_get;
1127c478bd9Sstevel@tonic-gate sv->res_set = sv_res_set;
1137c478bd9Sstevel@tonic-gate #else
114*9525b14bSRao Shoaib sv->res_get = NULL; /*%< sv_res_get; */
115*9525b14bSRao Shoaib sv->res_set = NULL; /*%< sv_res_set; */
1167c478bd9Sstevel@tonic-gate #endif
1177c478bd9Sstevel@tonic-gate return (sv);
1187c478bd9Sstevel@tonic-gate }
1197c478bd9Sstevel@tonic-gate
1207c478bd9Sstevel@tonic-gate /* Methods */
1217c478bd9Sstevel@tonic-gate
1227c478bd9Sstevel@tonic-gate static void
sv_close(struct irs_sv * this)1237c478bd9Sstevel@tonic-gate sv_close(struct irs_sv *this) {
1247c478bd9Sstevel@tonic-gate struct pvt *pvt = (struct pvt *)this->private;
1257c478bd9Sstevel@tonic-gate
1267c478bd9Sstevel@tonic-gate if (pvt->serv.s_aliases)
1277c478bd9Sstevel@tonic-gate free(pvt->serv.s_aliases);
1287c478bd9Sstevel@tonic-gate if (pvt->svbuf)
1297c478bd9Sstevel@tonic-gate free(pvt->svbuf);
1307c478bd9Sstevel@tonic-gate
1317c478bd9Sstevel@tonic-gate if (pvt->res && pvt->free_res)
1327c478bd9Sstevel@tonic-gate (*pvt->free_res)(pvt->res);
1337c478bd9Sstevel@tonic-gate memput(pvt, sizeof *pvt);
1347c478bd9Sstevel@tonic-gate memput(this, sizeof *this);
1357c478bd9Sstevel@tonic-gate }
1367c478bd9Sstevel@tonic-gate
1377c478bd9Sstevel@tonic-gate static struct servent *
sv_byname(struct irs_sv * this,const char * name,const char * proto)1387c478bd9Sstevel@tonic-gate sv_byname(struct irs_sv *this, const char *name, const char *proto) {
1397c478bd9Sstevel@tonic-gate struct pvt *pvt = (struct pvt *)this->private;
1407c478bd9Sstevel@tonic-gate struct dns_p *dns = pvt->dns;
1417c478bd9Sstevel@tonic-gate struct servent *s;
1427c478bd9Sstevel@tonic-gate char **hes_list;
1437c478bd9Sstevel@tonic-gate
1447c478bd9Sstevel@tonic-gate if (!(hes_list = hesiod_resolve(dns->hes_ctx, name, "service")))
1457c478bd9Sstevel@tonic-gate return (NULL);
1467c478bd9Sstevel@tonic-gate
1477c478bd9Sstevel@tonic-gate s = parse_hes_list(this, hes_list, proto);
1487c478bd9Sstevel@tonic-gate hesiod_free_list(dns->hes_ctx, hes_list);
1497c478bd9Sstevel@tonic-gate return (s);
1507c478bd9Sstevel@tonic-gate }
1517c478bd9Sstevel@tonic-gate
1527c478bd9Sstevel@tonic-gate static struct servent *
sv_byport(struct irs_sv * this,int port,const char * proto)1537c478bd9Sstevel@tonic-gate sv_byport(struct irs_sv *this, int port, const char *proto) {
1547c478bd9Sstevel@tonic-gate struct pvt *pvt = (struct pvt *)this->private;
1557c478bd9Sstevel@tonic-gate struct dns_p *dns = pvt->dns;
1567c478bd9Sstevel@tonic-gate struct servent *s;
1577c478bd9Sstevel@tonic-gate char portstr[16];
1587c478bd9Sstevel@tonic-gate char **hes_list;
1597c478bd9Sstevel@tonic-gate
1607c478bd9Sstevel@tonic-gate sprintf(portstr, "%d", ntohs(port));
1617c478bd9Sstevel@tonic-gate if (!(hes_list = hesiod_resolve(dns->hes_ctx, portstr, "port")))
1627c478bd9Sstevel@tonic-gate return (NULL);
1637c478bd9Sstevel@tonic-gate
1647c478bd9Sstevel@tonic-gate s = parse_hes_list(this, hes_list, proto);
1657c478bd9Sstevel@tonic-gate hesiod_free_list(dns->hes_ctx, hes_list);
1667c478bd9Sstevel@tonic-gate return (s);
1677c478bd9Sstevel@tonic-gate }
1687c478bd9Sstevel@tonic-gate
1697c478bd9Sstevel@tonic-gate static struct servent *
sv_next(struct irs_sv * this)1707c478bd9Sstevel@tonic-gate sv_next(struct irs_sv *this) {
1717c478bd9Sstevel@tonic-gate UNUSED(this);
1727c478bd9Sstevel@tonic-gate errno = ENODEV;
1737c478bd9Sstevel@tonic-gate return (NULL);
1747c478bd9Sstevel@tonic-gate }
1757c478bd9Sstevel@tonic-gate
1767c478bd9Sstevel@tonic-gate static void
sv_rewind(struct irs_sv * this)1777c478bd9Sstevel@tonic-gate sv_rewind(struct irs_sv *this) {
1787c478bd9Sstevel@tonic-gate UNUSED(this);
1797c478bd9Sstevel@tonic-gate /* NOOP */
1807c478bd9Sstevel@tonic-gate }
1817c478bd9Sstevel@tonic-gate
1827c478bd9Sstevel@tonic-gate /* Private */
1837c478bd9Sstevel@tonic-gate
1847c478bd9Sstevel@tonic-gate static struct servent *
parse_hes_list(struct irs_sv * this,char ** hes_list,const char * proto)1857c478bd9Sstevel@tonic-gate parse_hes_list(struct irs_sv *this, char **hes_list, const char *proto) {
1867c478bd9Sstevel@tonic-gate struct pvt *pvt = (struct pvt *)this->private;
1877c478bd9Sstevel@tonic-gate char *p, *cp, **cpp, **new;
1887c478bd9Sstevel@tonic-gate int proto_len;
1897c478bd9Sstevel@tonic-gate int num = 0;
1907c478bd9Sstevel@tonic-gate int max = 0;
1917c478bd9Sstevel@tonic-gate
1927c478bd9Sstevel@tonic-gate for (cpp = hes_list; *cpp; cpp++) {
1937c478bd9Sstevel@tonic-gate cp = *cpp;
1947c478bd9Sstevel@tonic-gate
1957c478bd9Sstevel@tonic-gate /* Strip away comments, if any. */
1967c478bd9Sstevel@tonic-gate if ((p = strchr(cp, '#')))
1977c478bd9Sstevel@tonic-gate *p = 0;
1987c478bd9Sstevel@tonic-gate
1997c478bd9Sstevel@tonic-gate /* Check to make sure the protocol matches. */
2007c478bd9Sstevel@tonic-gate p = cp;
2017c478bd9Sstevel@tonic-gate while (*p && !isspace((unsigned char)*p))
2027c478bd9Sstevel@tonic-gate p++;
2037c478bd9Sstevel@tonic-gate if (!*p)
2047c478bd9Sstevel@tonic-gate continue;
2057c478bd9Sstevel@tonic-gate if (proto) {
2067c478bd9Sstevel@tonic-gate proto_len = strlen(proto);
2077c478bd9Sstevel@tonic-gate if (strncasecmp(++p, proto, proto_len) != 0)
2087c478bd9Sstevel@tonic-gate continue;
2097c478bd9Sstevel@tonic-gate if (p[proto_len] && !isspace(p[proto_len]&0xff))
2107c478bd9Sstevel@tonic-gate continue;
2117c478bd9Sstevel@tonic-gate }
2127c478bd9Sstevel@tonic-gate /* OK, we've got a live one. Let's parse it for real. */
2137c478bd9Sstevel@tonic-gate if (pvt->svbuf)
2147c478bd9Sstevel@tonic-gate free(pvt->svbuf);
2157c478bd9Sstevel@tonic-gate pvt->svbuf = strdup(cp);
2167c478bd9Sstevel@tonic-gate
2177c478bd9Sstevel@tonic-gate p = pvt->svbuf;
2187c478bd9Sstevel@tonic-gate pvt->serv.s_name = p;
2197c478bd9Sstevel@tonic-gate while (*p && !isspace(*p&0xff))
2207c478bd9Sstevel@tonic-gate p++;
2217c478bd9Sstevel@tonic-gate if (!*p)
2227c478bd9Sstevel@tonic-gate continue;
2237c478bd9Sstevel@tonic-gate *p++ = '\0';
2247c478bd9Sstevel@tonic-gate
2257c478bd9Sstevel@tonic-gate pvt->serv.s_proto = p;
2267c478bd9Sstevel@tonic-gate while (*p && !isspace(*p&0xff))
2277c478bd9Sstevel@tonic-gate p++;
2287c478bd9Sstevel@tonic-gate if (!*p)
2297c478bd9Sstevel@tonic-gate continue;
2307c478bd9Sstevel@tonic-gate *p++ = '\0';
2317c478bd9Sstevel@tonic-gate
2327c478bd9Sstevel@tonic-gate pvt->serv.s_port = htons((u_short) atoi(p));
2337c478bd9Sstevel@tonic-gate while (*p && !isspace(*p&0xff))
2347c478bd9Sstevel@tonic-gate p++;
2357c478bd9Sstevel@tonic-gate if (*p)
2367c478bd9Sstevel@tonic-gate *p++ = '\0';
2377c478bd9Sstevel@tonic-gate
2387c478bd9Sstevel@tonic-gate while (*p) {
2397c478bd9Sstevel@tonic-gate if ((num + 1) >= max || !pvt->serv.s_aliases) {
2407c478bd9Sstevel@tonic-gate max += 10;
2417c478bd9Sstevel@tonic-gate new = realloc(pvt->serv.s_aliases,
2427c478bd9Sstevel@tonic-gate max * sizeof(char *));
2437c478bd9Sstevel@tonic-gate if (!new) {
2447c478bd9Sstevel@tonic-gate errno = ENOMEM;
2457c478bd9Sstevel@tonic-gate goto cleanup;
2467c478bd9Sstevel@tonic-gate }
2477c478bd9Sstevel@tonic-gate pvt->serv.s_aliases = new;
2487c478bd9Sstevel@tonic-gate }
2497c478bd9Sstevel@tonic-gate pvt->serv.s_aliases[num++] = p;
2507c478bd9Sstevel@tonic-gate while (*p && !isspace(*p&0xff))
2517c478bd9Sstevel@tonic-gate p++;
2527c478bd9Sstevel@tonic-gate if (*p)
2537c478bd9Sstevel@tonic-gate *p++ = '\0';
2547c478bd9Sstevel@tonic-gate }
2557c478bd9Sstevel@tonic-gate if (!pvt->serv.s_aliases)
2567c478bd9Sstevel@tonic-gate pvt->serv.s_aliases = malloc(sizeof(char *));
2577c478bd9Sstevel@tonic-gate if (!pvt->serv.s_aliases)
2587c478bd9Sstevel@tonic-gate goto cleanup;
2597c478bd9Sstevel@tonic-gate pvt->serv.s_aliases[num] = NULL;
2607c478bd9Sstevel@tonic-gate return (&pvt->serv);
2617c478bd9Sstevel@tonic-gate }
2627c478bd9Sstevel@tonic-gate
2637c478bd9Sstevel@tonic-gate cleanup:
2647c478bd9Sstevel@tonic-gate if (pvt->serv.s_aliases) {
2657c478bd9Sstevel@tonic-gate free(pvt->serv.s_aliases);
2667c478bd9Sstevel@tonic-gate pvt->serv.s_aliases = NULL;
2677c478bd9Sstevel@tonic-gate }
2687c478bd9Sstevel@tonic-gate if (pvt->svbuf) {
2697c478bd9Sstevel@tonic-gate free(pvt->svbuf);
2707c478bd9Sstevel@tonic-gate pvt->svbuf = NULL;
2717c478bd9Sstevel@tonic-gate }
2727c478bd9Sstevel@tonic-gate return (NULL);
2737c478bd9Sstevel@tonic-gate }
2747c478bd9Sstevel@tonic-gate
2757c478bd9Sstevel@tonic-gate static void
sv_minimize(struct irs_sv * this)2767c478bd9Sstevel@tonic-gate sv_minimize(struct irs_sv *this) {
2777c478bd9Sstevel@tonic-gate UNUSED(this);
2787c478bd9Sstevel@tonic-gate /* NOOP */
2797c478bd9Sstevel@tonic-gate }
2807c478bd9Sstevel@tonic-gate
2817c478bd9Sstevel@tonic-gate #ifdef SV_RES_SETGET
2827c478bd9Sstevel@tonic-gate static struct __res_state *
sv_res_get(struct irs_sv * this)2837c478bd9Sstevel@tonic-gate sv_res_get(struct irs_sv *this) {
2847c478bd9Sstevel@tonic-gate struct pvt *pvt = (struct pvt *)this->private;
2857c478bd9Sstevel@tonic-gate struct dns_p *dns = pvt->dns;
2867c478bd9Sstevel@tonic-gate
2877c478bd9Sstevel@tonic-gate return (__hesiod_res_get(dns->hes_ctx));
2887c478bd9Sstevel@tonic-gate }
2897c478bd9Sstevel@tonic-gate
2907c478bd9Sstevel@tonic-gate static void
sv_res_set(struct irs_sv * this,struct __res_state * res,void (* free_res)(void *))2917c478bd9Sstevel@tonic-gate sv_res_set(struct irs_sv *this, struct __res_state * res,
2927c478bd9Sstevel@tonic-gate void (*free_res)(void *)) {
2937c478bd9Sstevel@tonic-gate struct pvt *pvt = (struct pvt *)this->private;
2947c478bd9Sstevel@tonic-gate struct dns_p *dns = pvt->dns;
2957c478bd9Sstevel@tonic-gate
2967c478bd9Sstevel@tonic-gate __hesiod_res_set(dns->hes_ctx, res, free_res);
2977c478bd9Sstevel@tonic-gate }
2987c478bd9Sstevel@tonic-gate #endif
299*9525b14bSRao Shoaib
300*9525b14bSRao Shoaib /*! \file */
301