xref: /onnv-gate/usr/src/lib/libsmbfs/smb/newvc.c (revision 10367:56de61e9a99e)
110023SGordon.Ross@Sun.COM /*
210023SGordon.Ross@Sun.COM  * CDDL HEADER START
310023SGordon.Ross@Sun.COM  *
410023SGordon.Ross@Sun.COM  * The contents of this file are subject to the terms of the
510023SGordon.Ross@Sun.COM  * Common Development and Distribution License (the "License").
610023SGordon.Ross@Sun.COM  * You may not use this file except in compliance with the License.
710023SGordon.Ross@Sun.COM  *
810023SGordon.Ross@Sun.COM  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
910023SGordon.Ross@Sun.COM  * or http://www.opensolaris.org/os/licensing.
1010023SGordon.Ross@Sun.COM  * See the License for the specific language governing permissions
1110023SGordon.Ross@Sun.COM  * and limitations under the License.
1210023SGordon.Ross@Sun.COM  *
1310023SGordon.Ross@Sun.COM  * When distributing Covered Code, include this CDDL HEADER in each
1410023SGordon.Ross@Sun.COM  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
1510023SGordon.Ross@Sun.COM  * If applicable, add the following below this CDDL HEADER, with the
1610023SGordon.Ross@Sun.COM  * fields enclosed by brackets "[]" replaced with your own identifying
1710023SGordon.Ross@Sun.COM  * information: Portions Copyright [yyyy] [name of copyright owner]
1810023SGordon.Ross@Sun.COM  *
1910023SGordon.Ross@Sun.COM  * CDDL HEADER END
2010023SGordon.Ross@Sun.COM  */
2110023SGordon.Ross@Sun.COM 
2210023SGordon.Ross@Sun.COM /*
2310023SGordon.Ross@Sun.COM  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
2410023SGordon.Ross@Sun.COM  * Use is subject to license terms.
2510023SGordon.Ross@Sun.COM  */
2610023SGordon.Ross@Sun.COM 
2710023SGordon.Ross@Sun.COM /*
2810023SGordon.Ross@Sun.COM  * Create a new VC given a list of addresses.
2910023SGordon.Ross@Sun.COM  */
3010023SGordon.Ross@Sun.COM 
3110023SGordon.Ross@Sun.COM #include <errno.h>
3210023SGordon.Ross@Sun.COM #include <stdio.h>
3310023SGordon.Ross@Sun.COM #include <string.h>
3410023SGordon.Ross@Sun.COM #include <strings.h>
3510023SGordon.Ross@Sun.COM #include <stdlib.h>
3610023SGordon.Ross@Sun.COM #include <unistd.h>
3710023SGordon.Ross@Sun.COM #include <netdb.h>
3810023SGordon.Ross@Sun.COM #include <libintl.h>
3910023SGordon.Ross@Sun.COM #include <xti.h>
4010023SGordon.Ross@Sun.COM #include <assert.h>
4110023SGordon.Ross@Sun.COM 
4210023SGordon.Ross@Sun.COM #include <sys/types.h>
4310023SGordon.Ross@Sun.COM #include <sys/time.h>
4410023SGordon.Ross@Sun.COM #include <sys/byteorder.h>
4510023SGordon.Ross@Sun.COM #include <sys/socket.h>
4610023SGordon.Ross@Sun.COM #include <sys/fcntl.h>
4710023SGordon.Ross@Sun.COM 
4810023SGordon.Ross@Sun.COM #include <netinet/in.h>
4910023SGordon.Ross@Sun.COM #include <netinet/tcp.h>
5010023SGordon.Ross@Sun.COM #include <arpa/inet.h>
5110023SGordon.Ross@Sun.COM 
5210023SGordon.Ross@Sun.COM #include <netsmb/smb.h>
5310023SGordon.Ross@Sun.COM #include <netsmb/smb_lib.h>
5410023SGordon.Ross@Sun.COM #include <netsmb/netbios.h>
5510023SGordon.Ross@Sun.COM #include <netsmb/nb_lib.h>
5610023SGordon.Ross@Sun.COM #include <netsmb/smb_dev.h>
5710023SGordon.Ross@Sun.COM 
5810023SGordon.Ross@Sun.COM #include "charsets.h"
5910023SGordon.Ross@Sun.COM #include "private.h"
6010023SGordon.Ross@Sun.COM 
6110023SGordon.Ross@Sun.COM /*
6210023SGordon.Ross@Sun.COM  * Ask the IOD to create a VC with this IP address.
6310023SGordon.Ross@Sun.COM  */
6410023SGordon.Ross@Sun.COM static int
newvc(struct smb_ctx * ctx,struct addrinfo * ai)6510023SGordon.Ross@Sun.COM newvc(struct smb_ctx *ctx, struct addrinfo *ai)
6610023SGordon.Ross@Sun.COM {
6710023SGordon.Ross@Sun.COM 	smbioc_ossn_t *ssn = &ctx->ct_ssn;
6810023SGordon.Ross@Sun.COM 
6910023SGordon.Ross@Sun.COM 	/*
7010023SGordon.Ross@Sun.COM 	 * Copy the passed address into ssn_srvaddr,
7110023SGordon.Ross@Sun.COM 	 * but first sanity-check lengths.  Also,
7210023SGordon.Ross@Sun.COM 	 * zero it first to avoid trailing junk.
7310023SGordon.Ross@Sun.COM 	 */
7410023SGordon.Ross@Sun.COM 	if (ai->ai_addrlen > sizeof (ssn->ssn_srvaddr))
7510023SGordon.Ross@Sun.COM 		return (EINVAL);
7610023SGordon.Ross@Sun.COM 	bzero(&ssn->ssn_srvaddr, sizeof (ssn->ssn_srvaddr));
7710023SGordon.Ross@Sun.COM 	bcopy(ai->ai_addr, &ssn->ssn_srvaddr, ai->ai_addrlen);
7810023SGordon.Ross@Sun.COM 
7910023SGordon.Ross@Sun.COM 	return (smb_iod_cl_newvc(ctx));
8010023SGordon.Ross@Sun.COM }
8110023SGordon.Ross@Sun.COM 
8210023SGordon.Ross@Sun.COM /*
8310023SGordon.Ross@Sun.COM  * Setup a new VC via the IOD.
8410023SGordon.Ross@Sun.COM  * Similar to findvc.c
8510023SGordon.Ross@Sun.COM  */
8610023SGordon.Ross@Sun.COM int
smb_ctx_newvc(struct smb_ctx * ctx)8710023SGordon.Ross@Sun.COM smb_ctx_newvc(struct smb_ctx *ctx)
8810023SGordon.Ross@Sun.COM {
8910023SGordon.Ross@Sun.COM 	struct addrinfo *ai;
90*10367SGordon.Ross@Sun.COM 	int err;
9110023SGordon.Ross@Sun.COM 
9210023SGordon.Ross@Sun.COM 	/* Should already have the address list. */
9310023SGordon.Ross@Sun.COM 	if ((ctx->ct_flags & SMBCF_RESOLVED) == 0)
9410023SGordon.Ross@Sun.COM 		return (EINVAL);
9510023SGordon.Ross@Sun.COM 
96*10367SGordon.Ross@Sun.COM 	/*
97*10367SGordon.Ross@Sun.COM 	 * Get a door handle to the smbiod if we
98*10367SGordon.Ross@Sun.COM 	 * don't have one already.  This also
99*10367SGordon.Ross@Sun.COM 	 * starts the smbiod if necessary.
100*10367SGordon.Ross@Sun.COM 	 */
101*10367SGordon.Ross@Sun.COM 	if (ctx->ct_door_fd < 0) {
102*10367SGordon.Ross@Sun.COM 		err = smb_iod_start(ctx);
103*10367SGordon.Ross@Sun.COM 		if (err != 0)
104*10367SGordon.Ross@Sun.COM 			return (err);
105*10367SGordon.Ross@Sun.COM 	}
106*10367SGordon.Ross@Sun.COM 
107*10367SGordon.Ross@Sun.COM 	err = EPROTONOSUPPORT;  /* in case no AF match */
10810023SGordon.Ross@Sun.COM 	for (ai = ctx->ct_addrinfo; ai; ai = ai->ai_next) {
10910023SGordon.Ross@Sun.COM 
11010023SGordon.Ross@Sun.COM 		switch (ai->ai_family) {
11110023SGordon.Ross@Sun.COM 
11210023SGordon.Ross@Sun.COM 		case AF_INET:
11310023SGordon.Ross@Sun.COM 		case AF_INET6:
11410023SGordon.Ross@Sun.COM 		case AF_NETBIOS:
11510023SGordon.Ross@Sun.COM 			err = newvc(ctx, ai);
116*10367SGordon.Ross@Sun.COM 			if (err == 0)
117*10367SGordon.Ross@Sun.COM 				goto OK;
11810023SGordon.Ross@Sun.COM 			break;
11910023SGordon.Ross@Sun.COM 
12010023SGordon.Ross@Sun.COM 		default:
12110023SGordon.Ross@Sun.COM 			DPRINT("skipped family %d", ai->ai_family);
12210023SGordon.Ross@Sun.COM 			break;
12310023SGordon.Ross@Sun.COM 		}
12410023SGordon.Ross@Sun.COM 	}
12510023SGordon.Ross@Sun.COM 
126*10367SGordon.Ross@Sun.COM 	/*
127*10367SGordon.Ross@Sun.COM 	 * In the error case, the caller may try again
128*10367SGordon.Ross@Sun.COM 	 * with new auth. info, so keep the door open.
129*10367SGordon.Ross@Sun.COM 	 * Error return will close in smb_ctx_done.
130*10367SGordon.Ross@Sun.COM 	 */
13110023SGordon.Ross@Sun.COM 	return (err);
132*10367SGordon.Ross@Sun.COM 
133*10367SGordon.Ross@Sun.COM OK:
134*10367SGordon.Ross@Sun.COM 	/* Done with the door handle. */
135*10367SGordon.Ross@Sun.COM 	close(ctx->ct_door_fd);
136*10367SGordon.Ross@Sun.COM 	ctx->ct_door_fd = -1;
137*10367SGordon.Ross@Sun.COM 	return (0);
13810023SGordon.Ross@Sun.COM }
139