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