xref: /titanic_51/usr/src/lib/libsmbfs/smb/getaddr.c (revision ae3d7f90695ef456a6da4f7bdccd448ebe0b99e1)
1613a2f6bSGordon Ross /*
2613a2f6bSGordon Ross  * CDDL HEADER START
3613a2f6bSGordon Ross  *
4613a2f6bSGordon Ross  * The contents of this file are subject to the terms of the
5613a2f6bSGordon Ross  * Common Development and Distribution License (the "License").
6613a2f6bSGordon Ross  * You may not use this file except in compliance with the License.
7613a2f6bSGordon Ross  *
8613a2f6bSGordon Ross  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9613a2f6bSGordon Ross  * or http://www.opensolaris.org/os/licensing.
10613a2f6bSGordon Ross  * See the License for the specific language governing permissions
11613a2f6bSGordon Ross  * and limitations under the License.
12613a2f6bSGordon Ross  *
13613a2f6bSGordon Ross  * When distributing Covered Code, include this CDDL HEADER in each
14613a2f6bSGordon Ross  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15613a2f6bSGordon Ross  * If applicable, add the following below this CDDL HEADER, with the
16613a2f6bSGordon Ross  * fields enclosed by brackets "[]" replaced with your own identifying
17613a2f6bSGordon Ross  * information: Portions Copyright [yyyy] [name of copyright owner]
18613a2f6bSGordon Ross  *
19613a2f6bSGordon Ross  * CDDL HEADER END
20613a2f6bSGordon Ross  */
21613a2f6bSGordon Ross 
22613a2f6bSGordon Ross /*
23*ae3d7f90SGordon Ross  * Copyright (c) 2009, 2010, Oracle and/or its affiliates. All rights reserved.
24613a2f6bSGordon Ross  */
25613a2f6bSGordon Ross 
26613a2f6bSGordon Ross /*
27613a2f6bSGordon Ross  * Functions to get list of addresses (TCP and/or NetBIOS)
28613a2f6bSGordon Ross  */
29613a2f6bSGordon Ross 
30613a2f6bSGordon Ross #include <errno.h>
31613a2f6bSGordon Ross #include <stdio.h>
32613a2f6bSGordon Ross #include <stdlib.h>
33613a2f6bSGordon Ross #include <string.h>
34613a2f6bSGordon Ross #include <strings.h>
35613a2f6bSGordon Ross #include <unistd.h>
36613a2f6bSGordon Ross #include <netdb.h>
37613a2f6bSGordon Ross #include <libintl.h>
38613a2f6bSGordon Ross #include <xti.h>
39613a2f6bSGordon Ross #include <assert.h>
40613a2f6bSGordon Ross 
41613a2f6bSGordon Ross #include <sys/types.h>
42613a2f6bSGordon Ross #include <sys/time.h>
43613a2f6bSGordon Ross #include <sys/byteorder.h>
44613a2f6bSGordon Ross #include <sys/socket.h>
45613a2f6bSGordon Ross #include <sys/fcntl.h>
46613a2f6bSGordon Ross 
47613a2f6bSGordon Ross #include <netinet/in.h>
48613a2f6bSGordon Ross #include <netinet/tcp.h>
49613a2f6bSGordon Ross #include <arpa/inet.h>
50613a2f6bSGordon Ross 
51613a2f6bSGordon Ross #include <netsmb/smb.h>
52613a2f6bSGordon Ross #include <netsmb/smb_lib.h>
53613a2f6bSGordon Ross #include <netsmb/netbios.h>
54613a2f6bSGordon Ross #include <netsmb/nb_lib.h>
55613a2f6bSGordon Ross #include <netsmb/smb_dev.h>
56613a2f6bSGordon Ross 
57613a2f6bSGordon Ross #include "charsets.h"
58613a2f6bSGordon Ross #include "private.h"
59613a2f6bSGordon Ross 
60613a2f6bSGordon Ross void
61613a2f6bSGordon Ross dump_addrinfo(struct addrinfo *ai)
62613a2f6bSGordon Ross {
63613a2f6bSGordon Ross 	int i;
64613a2f6bSGordon Ross 
65613a2f6bSGordon Ross 	if (ai == NULL) {
66613a2f6bSGordon Ross 		printf("ai==NULL\n");
67613a2f6bSGordon Ross 		return;
68613a2f6bSGordon Ross 	}
69613a2f6bSGordon Ross 
70613a2f6bSGordon Ross 	for (i = 0; ai; i++, ai = ai->ai_next) {
71613a2f6bSGordon Ross 		printf("ai[%d]: af=%d, len=%d", i,
72613a2f6bSGordon Ross 		    ai->ai_family, ai->ai_addrlen);
73613a2f6bSGordon Ross 		dump_sockaddr(ai->ai_addr);
74613a2f6bSGordon Ross 		if (ai->ai_canonname) {
75613a2f6bSGordon Ross 			printf("ai[%d]: cname=\"%s\"\n",
76613a2f6bSGordon Ross 			    i, ai->ai_canonname);
77613a2f6bSGordon Ross 		}
78613a2f6bSGordon Ross 	}
79613a2f6bSGordon Ross }
80613a2f6bSGordon Ross 
81613a2f6bSGordon Ross void
82613a2f6bSGordon Ross dump_sockaddr(struct sockaddr *sa)
83613a2f6bSGordon Ross {
84613a2f6bSGordon Ross 	char paddrbuf[INET6_ADDRSTRLEN];
85613a2f6bSGordon Ross 	struct sockaddr_in *sin;
86613a2f6bSGordon Ross 	struct sockaddr_in6 *sin6;
87613a2f6bSGordon Ross 	int af = sa->sa_family;
88613a2f6bSGordon Ross 	const char *ip;
89613a2f6bSGordon Ross 
90613a2f6bSGordon Ross 	printf(" saf=%d,", af);
91613a2f6bSGordon Ross 	switch (af) {
92613a2f6bSGordon Ross 	case AF_NETBIOS: /* see nbns_rq.c */
93613a2f6bSGordon Ross 	case AF_INET:
94613a2f6bSGordon Ross 		sin = (void *)sa;
95613a2f6bSGordon Ross 		ip = inet_ntop(AF_INET, &sin->sin_addr,
96613a2f6bSGordon Ross 		    paddrbuf, sizeof (paddrbuf));
97613a2f6bSGordon Ross 		break;
98613a2f6bSGordon Ross 	case AF_INET6:
99613a2f6bSGordon Ross 		sin6 = (void *)sa;
100613a2f6bSGordon Ross 		ip = inet_ntop(AF_INET6, &sin6->sin6_addr,
101613a2f6bSGordon Ross 		    paddrbuf, sizeof (paddrbuf));
102613a2f6bSGordon Ross 		break;
103613a2f6bSGordon Ross 	default:
104613a2f6bSGordon Ross 		ip = "?";
105613a2f6bSGordon Ross 		break;
106613a2f6bSGordon Ross 	}
107613a2f6bSGordon Ross 	printf(" IP=%s\n", ip);
108613a2f6bSGordon Ross }
109613a2f6bSGordon Ross 
110613a2f6bSGordon Ross 
111613a2f6bSGordon Ross /*
112613a2f6bSGordon Ross  * SMB client name resolution - normal, and/or NetBIOS.
113613a2f6bSGordon Ross  * Returns an EAI_xxx error number like getaddrinfo(3)
114613a2f6bSGordon Ross  */
115613a2f6bSGordon Ross int
116613a2f6bSGordon Ross smb_ctx_getaddr(struct smb_ctx *ctx)
117613a2f6bSGordon Ross {
118613a2f6bSGordon Ross 	struct nb_ctx	*nbc = ctx->ct_nb;
119613a2f6bSGordon Ross 	struct addrinfo hints, *res;
120613a2f6bSGordon Ross 	char *srvaddr_str;
121613a2f6bSGordon Ross 	int gaierr, gaierr2;
122613a2f6bSGordon Ross 
123613a2f6bSGordon Ross 	if (ctx->ct_fullserver == NULL || ctx->ct_fullserver[0] == '\0')
124613a2f6bSGordon Ross 		return (EAI_NONAME);
125613a2f6bSGordon Ross 
126613a2f6bSGordon Ross 	if (ctx->ct_addrinfo != NULL) {
127613a2f6bSGordon Ross 		freeaddrinfo(ctx->ct_addrinfo);
128613a2f6bSGordon Ross 		ctx->ct_addrinfo = NULL;
129613a2f6bSGordon Ross 	}
130613a2f6bSGordon Ross 
131613a2f6bSGordon Ross 	/*
132613a2f6bSGordon Ross 	 * If the user specified an address, use it,
133613a2f6bSGordon Ross 	 * and don't do NetBIOS lookup.
134613a2f6bSGordon Ross 	 */
135613a2f6bSGordon Ross 	if (ctx->ct_srvaddr_s) {
136613a2f6bSGordon Ross 		srvaddr_str = ctx->ct_srvaddr_s;
137613a2f6bSGordon Ross 		nbc->nb_flags &= ~NBCF_NS_ENABLE;
138613a2f6bSGordon Ross 	} else
139613a2f6bSGordon Ross 		srvaddr_str = ctx->ct_fullserver;
140613a2f6bSGordon Ross 
141613a2f6bSGordon Ross 	/*
142613a2f6bSGordon Ross 	 * Default the server name we'll use in the
143613a2f6bSGordon Ross 	 * protocol (i.e. NTLM, tree connect).
144613a2f6bSGordon Ross 	 */
145613a2f6bSGordon Ross 	strlcpy(ctx->ct_srvname, ctx->ct_fullserver,
146613a2f6bSGordon Ross 	    sizeof (ctx->ct_srvname));
147613a2f6bSGordon Ross 
148613a2f6bSGordon Ross 	/*
149613a2f6bSGordon Ross 	 * Try to lookup the host address using the
150613a2f6bSGordon Ross 	 * normal name-to-IP address mechanisms.
151613a2f6bSGordon Ross 	 * If that fails, we MAY try NetBIOS.
152613a2f6bSGordon Ross 	 */
153613a2f6bSGordon Ross 	memset(&hints, 0, sizeof (hints));
154613a2f6bSGordon Ross 	hints.ai_flags = AI_CANONNAME;
155613a2f6bSGordon Ross 	hints.ai_family = PF_UNSPEC;
156613a2f6bSGordon Ross 	hints.ai_socktype = SOCK_STREAM;
157613a2f6bSGordon Ross 	gaierr = getaddrinfo(srvaddr_str, NULL, &hints, &res);
158613a2f6bSGordon Ross 	if (gaierr == 0) {
159613a2f6bSGordon Ross 		ctx->ct_addrinfo = res;
160613a2f6bSGordon Ross 		return (0);
161613a2f6bSGordon Ross 	}
162613a2f6bSGordon Ross 
163613a2f6bSGordon Ross 	/*
164613a2f6bSGordon Ross 	 * If regular IP name lookup failed, try NetBIOS,
165613a2f6bSGordon Ross 	 * but only if given a valid NetBIOS name and if
166613a2f6bSGordon Ross 	 * NetBIOS name lookup is enabled.
167613a2f6bSGordon Ross 	 */
168613a2f6bSGordon Ross 	if (nbc->nb_flags & NBCF_NS_ENABLE) {
169613a2f6bSGordon Ross 		gaierr2 = nbns_getaddrinfo(ctx->ct_fullserver, nbc, &res);
170613a2f6bSGordon Ross 		if (gaierr2 == 0) {
171613a2f6bSGordon Ross 			if (res->ai_canonname)
172613a2f6bSGordon Ross 				strlcpy(ctx->ct_srvname,
173613a2f6bSGordon Ross 				    res->ai_canonname,
174613a2f6bSGordon Ross 				    sizeof (ctx->ct_srvname));
175613a2f6bSGordon Ross 			ctx->ct_addrinfo = res;
176613a2f6bSGordon Ross 			return (0);
177613a2f6bSGordon Ross 		}
178613a2f6bSGordon Ross 	}
179613a2f6bSGordon Ross 
180613a2f6bSGordon Ross 	/*
181613a2f6bSGordon Ross 	 * Return the original error from getaddrinfo
182613a2f6bSGordon Ross 	 */
183613a2f6bSGordon Ross 	if (smb_verbose) {
184613a2f6bSGordon Ross 		smb_error(dgettext(TEXT_DOMAIN,
185613a2f6bSGordon Ross 		    "getaddrinfo: %s: %s"), 0,
186613a2f6bSGordon Ross 		    ctx->ct_fullserver,
187613a2f6bSGordon Ross 		    gai_strerror(gaierr));
188613a2f6bSGordon Ross 	}
189613a2f6bSGordon Ross 	return (gaierr);
190613a2f6bSGordon Ross }
191