xref: /onnv-gate/usr/src/cmd/krb5/kadmin/kdcmgr/klookup.c (revision 6815:664cfc3cd8e8)
15086Ssemery /*
25086Ssemery  * CDDL HEADER START
35086Ssemery  *
45086Ssemery  * The contents of this file are subject to the terms of the
55086Ssemery  * Common Development and Distribution License (the "License").
65086Ssemery  * You may not use this file except in compliance with the License.
75086Ssemery  *
85086Ssemery  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
95086Ssemery  * or http://www.opensolaris.org/os/licensing.
105086Ssemery  * See the License for the specific language governing permissions
115086Ssemery  * and limitations under the License.
125086Ssemery  *
135086Ssemery  * When distributing Covered Code, include this CDDL HEADER in each
145086Ssemery  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
155086Ssemery  * If applicable, add the following below this CDDL HEADER, with the
165086Ssemery  * fields enclosed by brackets "[]" replaced with your own identifying
175086Ssemery  * information: Portions Copyright [yyyy] [name of copyright owner]
185086Ssemery  *
195086Ssemery  * CDDL HEADER END
205086Ssemery  */
215086Ssemery 
225086Ssemery /*
236656Ssemery  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
245086Ssemery  * Use is subject to license terms.
255086Ssemery  */
265086Ssemery 
275086Ssemery #pragma ident	"%Z%%M%	%I%	%E% SMI"
285086Ssemery 
295086Ssemery #include <sys/types.h>
305086Ssemery #include <netinet/in.h>
315086Ssemery #include <arpa/nameser.h>
325086Ssemery #include <resolv.h>
335086Ssemery #include <netdb.h>
345086Ssemery #include <limits.h>
355086Ssemery #include <stdlib.h>
365086Ssemery #include <string.h>
375086Ssemery #include <unistd.h>
385086Ssemery #include <ctype.h>
395086Ssemery 
406656Ssemery /*
416656Ssemery  * Private resolver of target and type with arguments:
426656Ssemery  * klooukp [ target [ RR_type ] ]
436656Ssemery  *
446656Ssemery  * Utitilizes DNS lookups to discover domain and realm information.  This CLI
456656Ssemery  * is used primarily by kdcmgr(1M) and kclient(1M).
466656Ssemery  */
476656Ssemery 
485086Ssemery int
495086Ssemery /* ARGSUSED */
main(int argc,char ** argv)505086Ssemery main(int argc, char **argv)
515086Ssemery {
526656Ssemery 	unsigned char answer[NS_MAXMSG], *ansp = NULL, *end, a, b, c, d;
535086Ssemery 	int len = 0, anslen, hostlen, nq, na, type, class;
546656Ssemery 	int ttl, priority, weight, port, size;
556656Ssemery 	char name[NS_MAXDNAME], *cp, *typestr = NULL;
566656Ssemery 	char nbuf[INET6_ADDRSTRLEN];
575086Ssemery 	struct __res_state stat;
585086Ssemery 	int found = 0;
596656Ssemery 	int rr_type = T_A;
605086Ssemery 	HEADER *h;
615086Ssemery 
626656Ssemery 	if (argc > 3)
635086Ssemery 		exit(1);
645086Ssemery 
656656Ssemery 	if (argc == 1) {
666656Ssemery 		if (gethostname(name, MAXHOSTNAMELEN) != 0)
676656Ssemery 			exit(1);
686656Ssemery 	} else {
696656Ssemery 		(void) strncpy(name, (char *)argv[1], NS_MAXDNAME);
706656Ssemery 		if (argc == 3) {
716656Ssemery 			typestr = argv[2];
726656Ssemery 
736656Ssemery 			switch (*typestr) {
746656Ssemery 			case 'A':
756656Ssemery 				rr_type = T_A;
766656Ssemery 				break;
776656Ssemery 			case 'C':
786656Ssemery 				rr_type = T_CNAME;
796656Ssemery 				break;
806656Ssemery 			case 'I':
816656Ssemery 				rr_type = T_A;
826656Ssemery 				break;
836656Ssemery 			case 'P':
846656Ssemery 				rr_type = T_PTR;
856656Ssemery 				(void) sscanf(name, "%d.%d.%d.%d",
866656Ssemery 				    &a, &b, &c, &d);
876656Ssemery 				(void) sprintf(name, "%d.%d.%d.%d.in-addr.arpa",
886656Ssemery 				    d, c, b, a);
896656Ssemery 				break;
906656Ssemery 			case 'S':
916656Ssemery 				rr_type = T_SRV;
926656Ssemery 				break;
936656Ssemery 			default:
946656Ssemery 				exit(1);
956656Ssemery 			}
966656Ssemery 		}
976656Ssemery 	}
985086Ssemery 
995086Ssemery 	(void) memset(&stat, 0, sizeof (stat));
1005086Ssemery 
1015086Ssemery 	if (res_ninit(&stat) == -1)
1025086Ssemery 		exit(1);
1035086Ssemery 
1045086Ssemery 	anslen = sizeof (answer);
1056656Ssemery 	len = res_nsearch(&stat, name, C_IN, rr_type, answer, anslen);
1065086Ssemery 
1076656Ssemery 	if (len < sizeof (HEADER)) {
1086656Ssemery 		res_ndestroy(&stat);
1095086Ssemery 		exit(1);
1106656Ssemery 	}
1115086Ssemery 
1125086Ssemery 	ansp = answer;
1135086Ssemery 	end = ansp + anslen;
1145086Ssemery 
1155086Ssemery 	/* LINTED */
1165086Ssemery 	h = (HEADER *)answer;
1175086Ssemery 	nq = ntohs(h->qdcount);
1185086Ssemery 	na = ntohs(h->ancount);
1195086Ssemery 	ansp += HFIXEDSZ;
1205086Ssemery 
1216656Ssemery 	if (nq != 1 || na < 1) {
1226656Ssemery 		res_ndestroy(&stat);
1235086Ssemery 		exit(1);
1246656Ssemery 	}
1255086Ssemery 
1266656Ssemery 	hostlen = sizeof (name);
1276656Ssemery 	len = dn_expand(answer, end, ansp, name, hostlen);
1286656Ssemery 	if (len < 0) {
1296656Ssemery 		res_ndestroy(&stat);
1305086Ssemery 		exit(1);
1316656Ssemery 	}
1325086Ssemery 
1335086Ssemery 	ansp += len + QFIXEDSZ;
1345086Ssemery 
1356656Ssemery 	if (ansp > end) {
1366656Ssemery 		res_ndestroy(&stat);
1375086Ssemery 		exit(1);
1386656Ssemery 	}
1395086Ssemery 
1405086Ssemery 	while (na-- > 0 && ansp < end) {
1416656Ssemery 
1426656Ssemery 		len = dn_expand(answer, end, ansp, name, hostlen);
1435086Ssemery 
1445086Ssemery 		if (len < 0)
1455086Ssemery 			continue;
1466656Ssemery 		ansp += len;			/* name */
1476656Ssemery 		NS_GET16(type, ansp);		/* type */
1486656Ssemery 		NS_GET16(class, ansp);		/* class */
1496656Ssemery 		NS_GET32(ttl, ansp);		/* ttl */
1506656Ssemery 		NS_GET16(size, ansp);		/* size */
1516656Ssemery 
1526656Ssemery 		if ((ansp + size) > end) {
1536656Ssemery 			res_ndestroy(&stat);
1546656Ssemery 			exit(1);
1556656Ssemery 		}
1566656Ssemery 		if (type == T_SRV) {
1576656Ssemery 			NS_GET16(priority, ansp);
1586656Ssemery 			NS_GET16(weight, ansp);
1596656Ssemery 			NS_GET16(port, ansp);
1606656Ssemery 			len = dn_expand(answer, end, ansp, name, hostlen);
1616656Ssemery 			if (len < 0) {
1626656Ssemery 				res_ndestroy(&stat);
1636656Ssemery 				exit(1);
1646656Ssemery 			}
1656656Ssemery 			for (cp = name; *cp; cp++) {
1666656Ssemery 				*cp = tolower(*cp);
1676656Ssemery 			}
1686656Ssemery 			(void) printf("%s %d\n", name, port);
1696656Ssemery 		} else if (typestr && *typestr == 'I') {
1706656Ssemery 			(void) inet_ntop(AF_INET, (void *)ansp, nbuf,
1716656Ssemery 			    INET6_ADDRSTRLEN);
172*6815Ssemery 			len = size;
173*6815Ssemery 			(void) printf("%s\n", nbuf);
1746656Ssemery 		} else if (type == T_PTR) {
1756656Ssemery 			len = dn_expand(answer, end, ansp, name, hostlen);
1766656Ssemery 			if (len < 0) {
1776656Ssemery 				res_ndestroy(&stat);
1786656Ssemery 				exit(1);
1796656Ssemery 			}
1806656Ssemery 		}
1815086Ssemery 		ansp += len;
1826656Ssemery 		if (type == rr_type && class == C_IN) {
1835086Ssemery 			found = 1;
184*6815Ssemery 			if (type != T_SRV && !(typestr && *typestr == 'I'))
1856656Ssemery 				break;
1865086Ssemery 		}
1875086Ssemery 	}
1885086Ssemery 
1896656Ssemery 	if (found != 1) {
1906656Ssemery 		res_ndestroy(&stat);
1915086Ssemery 		exit(1);
1926656Ssemery 	}
1935086Ssemery 
1946656Ssemery 	for (cp = name; *cp; cp++) {
1955086Ssemery 		*cp = tolower(*cp);
1965086Ssemery 	}
1975086Ssemery 
198*6815Ssemery 	if (type != T_SRV && !(typestr && *typestr == 'I'))
1996656Ssemery 		(void) printf("%s\n", name);
2006656Ssemery 
2016656Ssemery 	res_ndestroy(&stat);
2025086Ssemery 
2035086Ssemery 	return (0);
2045086Ssemery }
205