17c478bd9Sstevel@tonic-gate /*
2*9525b14bSRao Shoaib  * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC")
37c478bd9Sstevel@tonic-gate  * Portions Copyright (c) 1996,1998 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 #include "port_before.h"
197c478bd9Sstevel@tonic-gate 
207c478bd9Sstevel@tonic-gate #include <syslog.h>
217c478bd9Sstevel@tonic-gate #include <sys/types.h>
227c478bd9Sstevel@tonic-gate #include <sys/socket.h>
237c478bd9Sstevel@tonic-gate 
247c478bd9Sstevel@tonic-gate #ifdef IRS_LCL_SV_DB
257c478bd9Sstevel@tonic-gate #include <db.h>
267c478bd9Sstevel@tonic-gate #endif
277c478bd9Sstevel@tonic-gate #include <errno.h>
287c478bd9Sstevel@tonic-gate #include <fcntl.h>
297c478bd9Sstevel@tonic-gate #include <limits.h>
307c478bd9Sstevel@tonic-gate #include <stdio.h>
317c478bd9Sstevel@tonic-gate #include <string.h>
327c478bd9Sstevel@tonic-gate #include <stdlib.h>
337c478bd9Sstevel@tonic-gate #include <syslog.h>
347c478bd9Sstevel@tonic-gate 
357c478bd9Sstevel@tonic-gate #include <irs.h>
367c478bd9Sstevel@tonic-gate #include <irp.h>
377c478bd9Sstevel@tonic-gate #include <isc/irpmarshall.h>
387c478bd9Sstevel@tonic-gate #include <isc/memcluster.h>
397c478bd9Sstevel@tonic-gate 
407c478bd9Sstevel@tonic-gate #include "irs_p.h"
417c478bd9Sstevel@tonic-gate #include "lcl_p.h"
427c478bd9Sstevel@tonic-gate #include "irp_p.h"
437c478bd9Sstevel@tonic-gate 
447c478bd9Sstevel@tonic-gate #include "port_after.h"
457c478bd9Sstevel@tonic-gate 
467c478bd9Sstevel@tonic-gate /* Types */
477c478bd9Sstevel@tonic-gate 
487c478bd9Sstevel@tonic-gate struct pvt {
497c478bd9Sstevel@tonic-gate 	struct irp_p	       *girpdata;
507c478bd9Sstevel@tonic-gate 	int			warned;
517c478bd9Sstevel@tonic-gate 	struct servent		service;
527c478bd9Sstevel@tonic-gate };
537c478bd9Sstevel@tonic-gate 
547c478bd9Sstevel@tonic-gate /* Forward */
557c478bd9Sstevel@tonic-gate 
567c478bd9Sstevel@tonic-gate static void			sv_close(struct irs_sv*);
577c478bd9Sstevel@tonic-gate static struct servent *		sv_next(struct irs_sv *);
587c478bd9Sstevel@tonic-gate static struct servent *		sv_byname(struct irs_sv *, const char *,
597c478bd9Sstevel@tonic-gate 					  const char *);
607c478bd9Sstevel@tonic-gate static struct servent *		sv_byport(struct irs_sv *, int, const char *);
617c478bd9Sstevel@tonic-gate static void			sv_rewind(struct irs_sv *);
627c478bd9Sstevel@tonic-gate static void			sv_minimize(struct irs_sv *);
637c478bd9Sstevel@tonic-gate 
647c478bd9Sstevel@tonic-gate static void			free_service(struct servent *sv);
657c478bd9Sstevel@tonic-gate 
667c478bd9Sstevel@tonic-gate 
677c478bd9Sstevel@tonic-gate 
687c478bd9Sstevel@tonic-gate /* Public */
697c478bd9Sstevel@tonic-gate 
70*9525b14bSRao Shoaib /*%
717c478bd9Sstevel@tonic-gate  * struct irs_sv * irs_irp_sv(struct irs_acc *this)
727c478bd9Sstevel@tonic-gate  *
737c478bd9Sstevel@tonic-gate  */
747c478bd9Sstevel@tonic-gate 
757c478bd9Sstevel@tonic-gate struct irs_sv *
irs_irp_sv(struct irs_acc * this)767c478bd9Sstevel@tonic-gate irs_irp_sv(struct irs_acc *this) {
777c478bd9Sstevel@tonic-gate 	struct irs_sv *sv;
787c478bd9Sstevel@tonic-gate 	struct pvt *pvt;
797c478bd9Sstevel@tonic-gate 
807c478bd9Sstevel@tonic-gate 	if ((sv = memget(sizeof *sv)) == NULL) {
817c478bd9Sstevel@tonic-gate 		errno = ENOMEM;
827c478bd9Sstevel@tonic-gate 		return (NULL);
837c478bd9Sstevel@tonic-gate 	}
847c478bd9Sstevel@tonic-gate 	memset(sv, 0x0, sizeof *sv);
857c478bd9Sstevel@tonic-gate 
867c478bd9Sstevel@tonic-gate 	if ((pvt = memget(sizeof *pvt)) == NULL) {
877c478bd9Sstevel@tonic-gate 		memput(sv, sizeof *sv);
887c478bd9Sstevel@tonic-gate 		errno = ENOMEM;
897c478bd9Sstevel@tonic-gate 		return (NULL);
907c478bd9Sstevel@tonic-gate 	}
917c478bd9Sstevel@tonic-gate 	memset(pvt, 0, sizeof *pvt);
927c478bd9Sstevel@tonic-gate 	pvt->girpdata = this->private;
937c478bd9Sstevel@tonic-gate 
947c478bd9Sstevel@tonic-gate 	sv->private = pvt;
957c478bd9Sstevel@tonic-gate 	sv->close = sv_close;
967c478bd9Sstevel@tonic-gate 	sv->next = sv_next;
977c478bd9Sstevel@tonic-gate 	sv->byname = sv_byname;
987c478bd9Sstevel@tonic-gate 	sv->byport = sv_byport;
997c478bd9Sstevel@tonic-gate 	sv->rewind = sv_rewind;
1007c478bd9Sstevel@tonic-gate 	sv->minimize = sv_minimize;
1017c478bd9Sstevel@tonic-gate 
1027c478bd9Sstevel@tonic-gate 	return (sv);
1037c478bd9Sstevel@tonic-gate }
1047c478bd9Sstevel@tonic-gate 
1057c478bd9Sstevel@tonic-gate /* Methods */
1067c478bd9Sstevel@tonic-gate 
107*9525b14bSRao Shoaib /*%
1087c478bd9Sstevel@tonic-gate  * void sv_close(struct irs_sv *this)
1097c478bd9Sstevel@tonic-gate  *
1107c478bd9Sstevel@tonic-gate  */
1117c478bd9Sstevel@tonic-gate 
1127c478bd9Sstevel@tonic-gate static void
sv_close(struct irs_sv * this)1137c478bd9Sstevel@tonic-gate sv_close(struct irs_sv *this) {
1147c478bd9Sstevel@tonic-gate 	struct pvt *pvt = (struct pvt *)this->private;
1157c478bd9Sstevel@tonic-gate 
1167c478bd9Sstevel@tonic-gate 	sv_minimize(this);
1177c478bd9Sstevel@tonic-gate 
1187c478bd9Sstevel@tonic-gate 	free_service(&pvt->service);
1197c478bd9Sstevel@tonic-gate 
1207c478bd9Sstevel@tonic-gate 	memput(pvt, sizeof *pvt);
1217c478bd9Sstevel@tonic-gate 	memput(this, sizeof *this);
1227c478bd9Sstevel@tonic-gate }
1237c478bd9Sstevel@tonic-gate 
124*9525b14bSRao Shoaib /*%
1257c478bd9Sstevel@tonic-gate  *	Fills the cache if necessary and returns the next item from it.
1267c478bd9Sstevel@tonic-gate  *
1277c478bd9Sstevel@tonic-gate  */
1287c478bd9Sstevel@tonic-gate 
1297c478bd9Sstevel@tonic-gate static struct servent *
sv_next(struct irs_sv * this)1307c478bd9Sstevel@tonic-gate sv_next(struct irs_sv *this) {
1317c478bd9Sstevel@tonic-gate 	struct pvt *pvt = (struct pvt *)this->private;
1327c478bd9Sstevel@tonic-gate 	struct servent *sv = &pvt->service;
1337c478bd9Sstevel@tonic-gate 	char *body;
1347c478bd9Sstevel@tonic-gate 	size_t bodylen;
1357c478bd9Sstevel@tonic-gate 	int code;
1367c478bd9Sstevel@tonic-gate 	char text[256];
1377c478bd9Sstevel@tonic-gate 
1387c478bd9Sstevel@tonic-gate 	if (irs_irp_connection_setup(pvt->girpdata, &pvt->warned) != 0) {
1397c478bd9Sstevel@tonic-gate 		return (NULL);
1407c478bd9Sstevel@tonic-gate 	}
1417c478bd9Sstevel@tonic-gate 
1427c478bd9Sstevel@tonic-gate 	if (irs_irp_send_command(pvt->girpdata, "getservent") != 0) {
1437c478bd9Sstevel@tonic-gate 		return (NULL);
1447c478bd9Sstevel@tonic-gate 	}
1457c478bd9Sstevel@tonic-gate 
1467c478bd9Sstevel@tonic-gate 	if (irs_irp_get_full_response(pvt->girpdata, &code,
1477c478bd9Sstevel@tonic-gate 				      text, sizeof text,
1487c478bd9Sstevel@tonic-gate 				      &body, &bodylen) != 0) {
1497c478bd9Sstevel@tonic-gate 		return (NULL);
1507c478bd9Sstevel@tonic-gate 	}
1517c478bd9Sstevel@tonic-gate 
1527c478bd9Sstevel@tonic-gate 	if (code == IRPD_GETSERVICE_OK) {
1537c478bd9Sstevel@tonic-gate 		free_service(sv);
1547c478bd9Sstevel@tonic-gate 		if (irp_unmarshall_sv(sv, body) != 0) {
1557c478bd9Sstevel@tonic-gate 			sv = NULL;
1567c478bd9Sstevel@tonic-gate 		}
1577c478bd9Sstevel@tonic-gate 	} else {
1587c478bd9Sstevel@tonic-gate 		sv = NULL;
1597c478bd9Sstevel@tonic-gate 	}
1607c478bd9Sstevel@tonic-gate 
1617c478bd9Sstevel@tonic-gate 	if (body != NULL) {
1627c478bd9Sstevel@tonic-gate 		memput(body, bodylen);
1637c478bd9Sstevel@tonic-gate 	}
1647c478bd9Sstevel@tonic-gate 
1657c478bd9Sstevel@tonic-gate 	return (sv);
1667c478bd9Sstevel@tonic-gate }
1677c478bd9Sstevel@tonic-gate 
168*9525b14bSRao Shoaib /*%
1697c478bd9Sstevel@tonic-gate  * struct servent * sv_byname(struct irs_sv *this, const char *name,
1707c478bd9Sstevel@tonic-gate  *				const char *proto)
1717c478bd9Sstevel@tonic-gate  *
1727c478bd9Sstevel@tonic-gate  */
1737c478bd9Sstevel@tonic-gate 
1747c478bd9Sstevel@tonic-gate static struct servent *
sv_byname(struct irs_sv * this,const char * name,const char * proto)1757c478bd9Sstevel@tonic-gate sv_byname(struct irs_sv *this, const char *name, const char *proto) {
1767c478bd9Sstevel@tonic-gate 	struct pvt *pvt = (struct pvt *)this->private;
1777c478bd9Sstevel@tonic-gate 	struct servent *sv = &pvt->service;
1787c478bd9Sstevel@tonic-gate 	char *body;
1797c478bd9Sstevel@tonic-gate 	char text[256];
1807c478bd9Sstevel@tonic-gate 	size_t bodylen;
1817c478bd9Sstevel@tonic-gate 	int code;
1827c478bd9Sstevel@tonic-gate 
1837c478bd9Sstevel@tonic-gate 	if (sv->s_name != NULL &&
1847c478bd9Sstevel@tonic-gate 	    strcmp(name, sv->s_name) == 0 &&
1857c478bd9Sstevel@tonic-gate 	    strcasecmp(proto, sv->s_proto) == 0) {
1867c478bd9Sstevel@tonic-gate 		return (sv);
1877c478bd9Sstevel@tonic-gate 	}
1887c478bd9Sstevel@tonic-gate 
1897c478bd9Sstevel@tonic-gate 	if (irs_irp_connection_setup(pvt->girpdata, &pvt->warned) != 0) {
1907c478bd9Sstevel@tonic-gate 		return (NULL);
1917c478bd9Sstevel@tonic-gate 	}
1927c478bd9Sstevel@tonic-gate 
1937c478bd9Sstevel@tonic-gate 	if (irs_irp_send_command(pvt->girpdata, "getservbyname %s %s",
1947c478bd9Sstevel@tonic-gate 				 name, proto) != 0)
1957c478bd9Sstevel@tonic-gate 		return (NULL);
1967c478bd9Sstevel@tonic-gate 
1977c478bd9Sstevel@tonic-gate 	if (irs_irp_get_full_response(pvt->girpdata, &code,
1987c478bd9Sstevel@tonic-gate 				      text, sizeof text,
1997c478bd9Sstevel@tonic-gate 				      &body, &bodylen) != 0) {
2007c478bd9Sstevel@tonic-gate 		return (NULL);
2017c478bd9Sstevel@tonic-gate 	}
2027c478bd9Sstevel@tonic-gate 
2037c478bd9Sstevel@tonic-gate 	if (code == IRPD_GETSERVICE_OK) {
2047c478bd9Sstevel@tonic-gate 		free_service(sv);
2057c478bd9Sstevel@tonic-gate 		if (irp_unmarshall_sv(sv, body) != 0) {
2067c478bd9Sstevel@tonic-gate 			sv = NULL;
2077c478bd9Sstevel@tonic-gate 		}
2087c478bd9Sstevel@tonic-gate 	} else {
2097c478bd9Sstevel@tonic-gate 		sv = NULL;
2107c478bd9Sstevel@tonic-gate 	}
2117c478bd9Sstevel@tonic-gate 
2127c478bd9Sstevel@tonic-gate 	if (body != NULL) {
2137c478bd9Sstevel@tonic-gate 		memput(body, bodylen);
2147c478bd9Sstevel@tonic-gate 	}
2157c478bd9Sstevel@tonic-gate 
2167c478bd9Sstevel@tonic-gate 	return (sv);
2177c478bd9Sstevel@tonic-gate }
2187c478bd9Sstevel@tonic-gate 
219*9525b14bSRao Shoaib /*%
2207c478bd9Sstevel@tonic-gate  * struct servent * sv_byport(struct irs_sv *this, int port,
2217c478bd9Sstevel@tonic-gate  *				const char *proto)
2227c478bd9Sstevel@tonic-gate  *
2237c478bd9Sstevel@tonic-gate  */
2247c478bd9Sstevel@tonic-gate 
2257c478bd9Sstevel@tonic-gate static struct servent *
sv_byport(struct irs_sv * this,int port,const char * proto)2267c478bd9Sstevel@tonic-gate sv_byport(struct irs_sv *this, int port, const char *proto) {
2277c478bd9Sstevel@tonic-gate 	struct pvt *pvt = (struct pvt *)this->private;
2287c478bd9Sstevel@tonic-gate 	struct servent *sv = &pvt->service;
2297c478bd9Sstevel@tonic-gate 	char *body;
2307c478bd9Sstevel@tonic-gate 	size_t bodylen;
2317c478bd9Sstevel@tonic-gate 	char text[256];
2327c478bd9Sstevel@tonic-gate 	int code;
2337c478bd9Sstevel@tonic-gate 
2347c478bd9Sstevel@tonic-gate 	if (sv->s_name != NULL &&
2357c478bd9Sstevel@tonic-gate 	    port == sv->s_port &&
2367c478bd9Sstevel@tonic-gate 	    strcasecmp(proto, sv->s_proto) == 0) {
2377c478bd9Sstevel@tonic-gate 		return (sv);
2387c478bd9Sstevel@tonic-gate 	}
2397c478bd9Sstevel@tonic-gate 
2407c478bd9Sstevel@tonic-gate 	if (irs_irp_connection_setup(pvt->girpdata, &pvt->warned) != 0) {
2417c478bd9Sstevel@tonic-gate 		return (NULL);
2427c478bd9Sstevel@tonic-gate 	}
2437c478bd9Sstevel@tonic-gate 
2447c478bd9Sstevel@tonic-gate 	if (irs_irp_send_command(pvt->girpdata, "getservbyport %d %s",
2457c478bd9Sstevel@tonic-gate 				 ntohs((short)port), proto) != 0) {
2467c478bd9Sstevel@tonic-gate 		return (NULL);
2477c478bd9Sstevel@tonic-gate 	}
2487c478bd9Sstevel@tonic-gate 
2497c478bd9Sstevel@tonic-gate 	if (irs_irp_get_full_response(pvt->girpdata, &code,
2507c478bd9Sstevel@tonic-gate 				      text, sizeof text,
2517c478bd9Sstevel@tonic-gate 				      &body, &bodylen) != 0) {
2527c478bd9Sstevel@tonic-gate 		return (NULL);
2537c478bd9Sstevel@tonic-gate 	}
2547c478bd9Sstevel@tonic-gate 
2557c478bd9Sstevel@tonic-gate 	if (code == IRPD_GETSERVICE_OK) {
2567c478bd9Sstevel@tonic-gate 		free_service(sv);
2577c478bd9Sstevel@tonic-gate 		if (irp_unmarshall_sv(sv, body) != 0) {
2587c478bd9Sstevel@tonic-gate 			sv = NULL;
2597c478bd9Sstevel@tonic-gate 		}
2607c478bd9Sstevel@tonic-gate 	} else {
2617c478bd9Sstevel@tonic-gate 		sv = NULL;
2627c478bd9Sstevel@tonic-gate 	}
2637c478bd9Sstevel@tonic-gate 
2647c478bd9Sstevel@tonic-gate 	if (body != NULL) {
2657c478bd9Sstevel@tonic-gate 		memput(body, bodylen);
2667c478bd9Sstevel@tonic-gate 	}
2677c478bd9Sstevel@tonic-gate 
2687c478bd9Sstevel@tonic-gate 	return (sv);
2697c478bd9Sstevel@tonic-gate }
2707c478bd9Sstevel@tonic-gate 
271*9525b14bSRao Shoaib /*%
2727c478bd9Sstevel@tonic-gate  * void sv_rewind(struct irs_sv *this)
2737c478bd9Sstevel@tonic-gate  *
2747c478bd9Sstevel@tonic-gate  */
2757c478bd9Sstevel@tonic-gate 
2767c478bd9Sstevel@tonic-gate static void
sv_rewind(struct irs_sv * this)2777c478bd9Sstevel@tonic-gate sv_rewind(struct irs_sv *this) {
2787c478bd9Sstevel@tonic-gate 	struct pvt *pvt = (struct pvt *)this->private;
2797c478bd9Sstevel@tonic-gate 	char text[256];
2807c478bd9Sstevel@tonic-gate 	int code;
2817c478bd9Sstevel@tonic-gate 
2827c478bd9Sstevel@tonic-gate 	if (irs_irp_connection_setup(pvt->girpdata, &pvt->warned) != 0) {
2837c478bd9Sstevel@tonic-gate 		return;
2847c478bd9Sstevel@tonic-gate 	}
2857c478bd9Sstevel@tonic-gate 
2867c478bd9Sstevel@tonic-gate 	if (irs_irp_send_command(pvt->girpdata, "setservent") != 0) {
2877c478bd9Sstevel@tonic-gate 		return;
2887c478bd9Sstevel@tonic-gate 	}
2897c478bd9Sstevel@tonic-gate 
2907c478bd9Sstevel@tonic-gate 	code = irs_irp_read_response(pvt->girpdata, text, sizeof text);
2917c478bd9Sstevel@tonic-gate 	if (code != IRPD_GETSERVICE_SETOK) {
2927c478bd9Sstevel@tonic-gate 		if (irp_log_errors) {
2937c478bd9Sstevel@tonic-gate 			syslog(LOG_WARNING, "setservent failed: %s", text);
2947c478bd9Sstevel@tonic-gate 		}
2957c478bd9Sstevel@tonic-gate 	}
2967c478bd9Sstevel@tonic-gate 
2977c478bd9Sstevel@tonic-gate 	return;
2987c478bd9Sstevel@tonic-gate }
2997c478bd9Sstevel@tonic-gate 
300*9525b14bSRao Shoaib /*%
3017c478bd9Sstevel@tonic-gate  * void sv_minimize(struct irs_sv *this)
3027c478bd9Sstevel@tonic-gate  *
3037c478bd9Sstevel@tonic-gate  */
3047c478bd9Sstevel@tonic-gate 
3057c478bd9Sstevel@tonic-gate static void
sv_minimize(struct irs_sv * this)3067c478bd9Sstevel@tonic-gate sv_minimize(struct irs_sv *this) {
3077c478bd9Sstevel@tonic-gate 	struct pvt *pvt = (struct pvt *)this->private;
3087c478bd9Sstevel@tonic-gate 
3097c478bd9Sstevel@tonic-gate 	irs_irp_disconnect(pvt->girpdata);
3107c478bd9Sstevel@tonic-gate }
3117c478bd9Sstevel@tonic-gate 
3127c478bd9Sstevel@tonic-gate 
3137c478bd9Sstevel@tonic-gate 
3147c478bd9Sstevel@tonic-gate 
3157c478bd9Sstevel@tonic-gate 
3167c478bd9Sstevel@tonic-gate 
3177c478bd9Sstevel@tonic-gate static void
free_service(struct servent * sv)3187c478bd9Sstevel@tonic-gate free_service(struct servent *sv) {
3197c478bd9Sstevel@tonic-gate 	char **p;
3207c478bd9Sstevel@tonic-gate 
3217c478bd9Sstevel@tonic-gate 	if (sv == NULL) {
3227c478bd9Sstevel@tonic-gate 		return;
3237c478bd9Sstevel@tonic-gate 	}
3247c478bd9Sstevel@tonic-gate 
3257c478bd9Sstevel@tonic-gate 	if (sv->s_name != NULL) {
3267c478bd9Sstevel@tonic-gate 		free(sv->s_name);
3277c478bd9Sstevel@tonic-gate 	}
3287c478bd9Sstevel@tonic-gate 
3297c478bd9Sstevel@tonic-gate 	for (p = sv->s_aliases ; p != NULL && *p != NULL ; p++) {
3307c478bd9Sstevel@tonic-gate 		free(*p);
3317c478bd9Sstevel@tonic-gate 	}
3327c478bd9Sstevel@tonic-gate 
3337c478bd9Sstevel@tonic-gate 	if (sv->s_proto != NULL) {
3347c478bd9Sstevel@tonic-gate 		free(sv->s_proto);
3357c478bd9Sstevel@tonic-gate 	}
3367c478bd9Sstevel@tonic-gate }
3377c478bd9Sstevel@tonic-gate 
3387c478bd9Sstevel@tonic-gate 
339*9525b14bSRao Shoaib 
340*9525b14bSRao Shoaib /*! \file */
341