xref: /freebsd/contrib/smbfs/lib/smb/ctx.c (revision 6780ab54325a71e7e70112b11657973edde8655e)
1 /*
2  * Copyright (c) 2000-2002, Boris Popov
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  * 3. All advertising materials mentioning features or use of this software
14  *    must display the following acknowledgement:
15  *    This product includes software developed by Boris Popov.
16  * 4. Neither the name of the author nor the names of any co-contributors
17  *    may be used to endorse or promote products derived from this software
18  *    without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30  * SUCH DAMAGE.
31  *
32  * $Id: ctx.c,v 1.24 2002/04/13 14:35:28 bp Exp $
33  */
34 #include <sys/param.h>
35 #include <sys/sysctl.h>
36 #include <sys/ioctl.h>
37 #include <sys/time.h>
38 #include <sys/mount.h>
39 #include <fcntl.h>
40 #include <ctype.h>
41 #include <errno.h>
42 #include <stdio.h>
43 #include <string.h>
44 #include <stdlib.h>
45 #include <pwd.h>
46 #include <grp.h>
47 #include <unistd.h>
48 #include <sys/iconv.h>
49 
50 #define NB_NEEDRESOLVER
51 
52 #include <netsmb/smb_lib.h>
53 #include <netsmb/netbios.h>
54 #include <netsmb/nb_lib.h>
55 #include <netsmb/smb_conn.h>
56 #include <cflib.h>
57 
58 /*
59  * Prescan command line for [-U user] argument
60  * and fill context with defaults
61  */
62 int
63 smb_ctx_init(struct smb_ctx *ctx, int argc, char *argv[],
64 	int minlevel, int maxlevel, int sharetype)
65 {
66 	int  opt, error = 0;
67 	uid_t euid;
68 	const char *arg, *cp;
69 	struct passwd *pwd;
70 
71 	bzero(ctx,sizeof(*ctx));
72 	error = nb_ctx_create(&ctx->ct_nb);
73 	if (error)
74 		return error;
75 	ctx->ct_fd = -1;
76 	ctx->ct_parsedlevel = SMBL_NONE;
77 	ctx->ct_minlevel = minlevel;
78 	ctx->ct_maxlevel = maxlevel;
79 
80 	ctx->ct_ssn.ioc_opt = SMBVOPT_CREATE;
81 	ctx->ct_ssn.ioc_timeout = 15;
82 	ctx->ct_ssn.ioc_retrycount = 4;
83 	ctx->ct_ssn.ioc_owner = SMBM_ANY_OWNER;
84 	ctx->ct_ssn.ioc_group = SMBM_ANY_GROUP;
85 	ctx->ct_ssn.ioc_mode = SMBM_EXEC;
86 	ctx->ct_ssn.ioc_rights = SMBM_DEFAULT;
87 
88 	ctx->ct_sh.ioc_opt = SMBVOPT_CREATE;
89 	ctx->ct_sh.ioc_owner = SMBM_ANY_OWNER;
90 	ctx->ct_sh.ioc_group = SMBM_ANY_GROUP;
91 	ctx->ct_sh.ioc_mode = SMBM_EXEC;
92 	ctx->ct_sh.ioc_rights = SMBM_DEFAULT;
93 	ctx->ct_sh.ioc_owner = SMBM_ANY_OWNER;
94 	ctx->ct_sh.ioc_group = SMBM_ANY_GROUP;
95 
96 	nb_ctx_setscope(ctx->ct_nb, "");
97 	euid = geteuid();
98 	if ((pwd = getpwuid(euid)) != NULL) {
99 		smb_ctx_setuser(ctx, pwd->pw_name);
100 		endpwent();
101 	} else if (euid == 0)
102 		smb_ctx_setuser(ctx, "root");
103 	else
104 		return 0;
105 	if (argv == NULL)
106 		return 0;
107 	for (opt = 1; opt < argc; opt++) {
108 		cp = argv[opt];
109 		if (strncmp(cp, "//", 2) != 0)
110 			continue;
111 		error = smb_ctx_parseunc(ctx, cp, sharetype, (const char**)&cp);
112 		if (error)
113 			return error;
114 		ctx->ct_uncnext = cp;
115 		break;
116 	}
117 	while (error == 0 && (opt = cf_getopt(argc, argv, ":E:L:U:")) != -1) {
118 		arg = cf_optarg;
119 		switch (opt) {
120 		    case 'E':
121 			error = smb_ctx_setcharset(ctx, arg);
122 			if (error)
123 				return error;
124 			break;
125 		    case 'L':
126 			error = nls_setlocale(optarg);
127 			if (error)
128 				break;
129 			break;
130 		    case 'U':
131 			error = smb_ctx_setuser(ctx, arg);
132 			break;
133 		}
134 	}
135 	cf_optind = cf_optreset = 1;
136 	return error;
137 }
138 
139 void
140 smb_ctx_done(struct smb_ctx *ctx)
141 {
142 	if (ctx->ct_ssn.ioc_server)
143 		nb_snbfree(ctx->ct_ssn.ioc_server);
144 	if (ctx->ct_ssn.ioc_local)
145 		nb_snbfree(ctx->ct_ssn.ioc_local);
146 	if (ctx->ct_srvaddr)
147 		free(ctx->ct_srvaddr);
148 	if (ctx->ct_nb)
149 		nb_ctx_done(ctx->ct_nb);
150 }
151 
152 static int
153 getsubstring(const char *p, u_char sep, char *dest, int maxlen, const char **next)
154 {
155 	int len;
156 
157 	maxlen--;
158 	for (len = 0; len < maxlen && *p != sep; p++, len++, dest++) {
159 		if (*p == 0)
160 			return EINVAL;
161 		*dest = *p;
162 	}
163 	*dest = 0;
164 	*next = *p ? p + 1 : p;
165 	return 0;
166 }
167 
168 /*
169  * Here we expect something like "[proto:]//[user@]host[/share][/path]"
170  */
171 int
172 smb_ctx_parseunc(struct smb_ctx *ctx, const char *unc, int sharetype,
173 	const char **next)
174 {
175 	const char *p = unc;
176 	char *p1;
177 	char tmp[1024];
178 	int error ;
179 
180 	ctx->ct_parsedlevel = SMBL_NONE;
181 	if (*p++ != '/' || *p++ != '/') {
182 		smb_error("UNC should start with '//'", 0);
183 		return EINVAL;
184 	}
185 	p1 = tmp;
186 	error = getsubstring(p, '@', p1, sizeof(tmp), &p);
187 	if (!error) {
188 		if (ctx->ct_maxlevel < SMBL_VC) {
189 			smb_error("no user name required", 0);
190 			return EINVAL;
191 		}
192 		if (*p1 == 0) {
193 			smb_error("empty user name", 0);
194 			return EINVAL;
195 		}
196 		error = smb_ctx_setuser(ctx, tmp);
197 		if (error)
198 			return error;
199 		ctx->ct_parsedlevel = SMBL_VC;
200 	}
201 	error = getsubstring(p, '/', p1, sizeof(tmp), &p);
202 	if (error) {
203 		error = getsubstring(p, '\0', p1, sizeof(tmp), &p);
204 		if (error) {
205 			smb_error("no server name found", 0);
206 			return error;
207 		}
208 	}
209 	if (*p1 == 0) {
210 		smb_error("empty server name", 0);
211 		return EINVAL;
212 	}
213 	error = smb_ctx_setserver(ctx, tmp);
214 	if (error)
215 		return error;
216 	if (sharetype == SMB_ST_NONE) {
217 		*next = p;
218 		return 0;
219 	}
220 	if (*p != 0 && ctx->ct_maxlevel < SMBL_SHARE) {
221 		smb_error("no share name required", 0);
222 		return EINVAL;
223 	}
224 	error = getsubstring(p, '/', p1, sizeof(tmp), &p);
225 	if (error) {
226 		error = getsubstring(p, '\0', p1, sizeof(tmp), &p);
227 		if (error) {
228 			smb_error("unexpected end of line", 0);
229 			return error;
230 		}
231 	}
232 	if (*p1 == 0 && ctx->ct_minlevel >= SMBL_SHARE) {
233 		smb_error("empty share name", 0);
234 		return EINVAL;
235 	}
236 	*next = p;
237 	if (*p1 == 0)
238 		return 0;
239 	error = smb_ctx_setshare(ctx, p1, sharetype);
240 	return error;
241 }
242 
243 int
244 smb_ctx_setcharset(struct smb_ctx *ctx, const char *arg)
245 {
246 	char *cp, *servercs, *localcs;
247 	int cslen = sizeof(ctx->ct_ssn.ioc_localcs);
248 	int scslen, lcslen, error;
249 
250 	cp = strchr(arg, ':');
251 	lcslen = cp ? (cp - arg) : 0;
252 	if (lcslen == 0 || lcslen >= cslen) {
253 		smb_error("invalid local charset specification (%s)", 0, arg);
254 		return EINVAL;
255 	}
256 	scslen = (size_t)strlen(++cp);
257 	if (scslen == 0 || scslen >= cslen) {
258 		smb_error("invalid server charset specification (%s)", 0, arg);
259 		return EINVAL;
260 	}
261 	localcs = memcpy(ctx->ct_ssn.ioc_localcs, arg, lcslen);
262 	localcs[lcslen] = 0;
263 	servercs = strcpy(ctx->ct_ssn.ioc_servercs, cp);
264 	error = nls_setrecode(localcs, servercs);
265 	if (error == 0)
266 		return 0;
267 	smb_error("can't initialize iconv support (%s:%s)",
268 	    error, localcs, servercs);
269 	localcs[0] = 0;
270 	servercs[0] = 0;
271 	return error;
272 }
273 
274 int
275 smb_ctx_setserver(struct smb_ctx *ctx, const char *name)
276 {
277 	if (strlen(name) >= SMB_MAXSRVNAMELEN) {
278 		smb_error("server name '%s' too long", 0, name);
279 		return ENAMETOOLONG;
280 	}
281 	nls_str_upper(ctx->ct_ssn.ioc_srvname, name);
282 	return 0;
283 }
284 
285 int
286 smb_ctx_setuser(struct smb_ctx *ctx, const char *name)
287 {
288 	if (strlen(name) >= SMB_MAXUSERNAMELEN) {
289 		smb_error("user name '%s' too long", 0, name);
290 		return ENAMETOOLONG;
291 	}
292 	nls_str_upper(ctx->ct_ssn.ioc_user, name);
293 	return 0;
294 }
295 
296 int
297 smb_ctx_setworkgroup(struct smb_ctx *ctx, const char *name)
298 {
299 	if (strlen(name) >= SMB_MAXUSERNAMELEN) {
300 		smb_error("workgroup name '%s' too long", 0, name);
301 		return ENAMETOOLONG;
302 	}
303 	nls_str_upper(ctx->ct_ssn.ioc_workgroup, name);
304 	return 0;
305 }
306 
307 int
308 smb_ctx_setpassword(struct smb_ctx *ctx, const char *passwd)
309 {
310 	if (passwd == NULL)
311 		return EINVAL;
312 	if (strlen(passwd) >= SMB_MAXPASSWORDLEN) {
313 		smb_error("password too long", 0);
314 		return ENAMETOOLONG;
315 	}
316 	if (strncmp(passwd, "$$1", 3) == 0)
317 		smb_simpledecrypt(ctx->ct_ssn.ioc_password, passwd);
318 	else
319 		strcpy(ctx->ct_ssn.ioc_password, passwd);
320 	strcpy(ctx->ct_sh.ioc_password, ctx->ct_ssn.ioc_password);
321 	return 0;
322 }
323 
324 int
325 smb_ctx_setshare(struct smb_ctx *ctx, const char *share, int stype)
326 {
327 	if (strlen(share) >= SMB_MAXSHARENAMELEN) {
328 		smb_error("share name '%s' too long", 0, share);
329 		return ENAMETOOLONG;
330 	}
331 	nls_str_upper(ctx->ct_sh.ioc_share, share);
332 	if (share[0] != 0)
333 		ctx->ct_parsedlevel = SMBL_SHARE;
334 	ctx->ct_sh.ioc_stype = stype;
335 	return 0;
336 }
337 
338 int
339 smb_ctx_setsrvaddr(struct smb_ctx *ctx, const char *addr)
340 {
341 	if (addr == NULL || addr[0] == 0)
342 		return EINVAL;
343 	if (ctx->ct_srvaddr)
344 		free(ctx->ct_srvaddr);
345 	if ((ctx->ct_srvaddr = strdup(addr)) == NULL)
346 		return ENOMEM;
347 	return 0;
348 }
349 
350 static int
351 smb_parse_owner(char *pair, uid_t *uid, gid_t *gid)
352 {
353 	struct group *gr;
354 	struct passwd *pw;
355 	char *cp;
356 
357 	cp = strchr(pair, ':');
358 	if (cp) {
359 		*cp++ = '\0';
360 		if (*cp) {
361 			gr = getgrnam(cp);
362 			if (gr) {
363 				*gid = gr->gr_gid;
364 			} else
365 				smb_error("Invalid group name %s, ignored",
366 				    0, cp);
367 		}
368 	}
369 	if (*pair) {
370 		pw = getpwnam(pair);
371 		if (pw) {
372 			*uid = pw->pw_uid;
373 		} else
374 			smb_error("Invalid user name %s, ignored", 0, pair);
375 	}
376 	endpwent();
377 	return 0;
378 }
379 
380 int
381 smb_ctx_opt(struct smb_ctx *ctx, int opt, const char *arg)
382 {
383 	int error = 0;
384 	char *p, *cp;
385 
386 	switch(opt) {
387 	    case 'U':
388 		break;
389 	    case 'I':
390 		error = smb_ctx_setsrvaddr(ctx, arg);
391 		break;
392 	    case 'M':
393 		ctx->ct_ssn.ioc_rights = strtol(arg, &cp, 8);
394 		if (*cp == '/') {
395 			ctx->ct_sh.ioc_rights = strtol(cp + 1, &cp, 8);
396 			ctx->ct_flags |= SMBCF_SRIGHTS;
397 		}
398 		break;
399 	    case 'N':
400 		ctx->ct_flags |= SMBCF_NOPWD;
401 		break;
402 	    case 'O':
403 		p = strdup(arg);
404 		cp = strchr(p, '/');
405 		if (cp) {
406 			*cp++ = '\0';
407 			error = smb_parse_owner(cp, &ctx->ct_sh.ioc_owner,
408 			    &ctx->ct_sh.ioc_group);
409 		}
410 		if (*p && error == 0) {
411 			error = smb_parse_owner(cp, &ctx->ct_ssn.ioc_owner,
412 			    &ctx->ct_ssn.ioc_group);
413 		}
414 		free(p);
415 		break;
416 	    case 'P':
417 /*		ctx->ct_ssn.ioc_opt |= SMBCOPT_PERMANENT;*/
418 		break;
419 	    case 'R':
420 		ctx->ct_ssn.ioc_retrycount = atoi(arg);
421 		break;
422 	    case 'T':
423 		ctx->ct_ssn.ioc_timeout = atoi(arg);
424 		break;
425 	    case 'W':
426 		error = smb_ctx_setworkgroup(ctx, arg);
427 		break;
428 	}
429 	return error;
430 }
431 
432 #if 0
433 static void
434 smb_hexdump(const u_char *buf, int len) {
435 	int ofs = 0;
436 
437 	while (len--) {
438 		if (ofs % 16 == 0)
439 			printf("\n%02X: ", ofs);
440 		printf("%02x ", *buf++);
441 		ofs++;
442 	}
443 	printf("\n");
444 }
445 #endif
446 
447 
448 static int
449 smb_addiconvtbl(const char *to, const char *from, const u_char *tbl)
450 {
451 	int error;
452 
453 	error = kiconv_add_xlat_table(to, from, tbl);
454 	if (error && error != EEXIST) {
455 		smb_error("can not setup kernel iconv table (%s:%s)", error,
456 		    from, to);
457 		return error;
458 	}
459 	return 0;
460 }
461 
462 /*
463  * Verify context before connect operation(s),
464  * lookup specified server and try to fill all forgotten fields.
465  */
466 int
467 smb_ctx_resolve(struct smb_ctx *ctx)
468 {
469 	struct smbioc_ossn *ssn = &ctx->ct_ssn;
470 	struct smbioc_oshare *sh = &ctx->ct_sh;
471 	struct nb_name nn;
472 	struct sockaddr *sap;
473 	struct sockaddr_nb *salocal, *saserver;
474 	char *cp;
475 	u_char cstbl[256];
476 	u_int i;
477 	int error = 0;
478 
479 	ctx->ct_flags &= ~SMBCF_RESOLVED;
480 	if (ssn->ioc_srvname[0] == 0) {
481 		smb_error("no server name specified", 0);
482 		return EINVAL;
483 	}
484 	if (ssn->ioc_user[0] == 0) {
485 		smb_error("no user name specified for server %s",
486 		    0, ssn->ioc_srvname);
487 		return EINVAL;
488 	}
489 	if (ctx->ct_minlevel >= SMBL_SHARE && sh->ioc_share[0] == 0) {
490 		smb_error("no share name specified for %s@%s",
491 		    0, ssn->ioc_user, ssn->ioc_srvname);
492 		return EINVAL;
493 	}
494 	error = nb_ctx_resolve(ctx->ct_nb);
495 	if (error)
496 		return error;
497 	if (ssn->ioc_localcs[0] == 0)
498 		strcpy(ssn->ioc_localcs, "default");	/* XXX: locale name ? */
499 	error = smb_addiconvtbl("tolower", ssn->ioc_localcs, nls_lower);
500 	if (error)
501 		return error;
502 	error = smb_addiconvtbl("toupper", ssn->ioc_localcs, nls_upper);
503 	if (error)
504 		return error;
505 	if (ssn->ioc_servercs[0] != 0) {
506 		for(i = 0; i < sizeof(cstbl); i++)
507 			cstbl[i] = i;
508 		nls_mem_toext(cstbl, cstbl, sizeof(cstbl));
509 		error = smb_addiconvtbl(ssn->ioc_servercs, ssn->ioc_localcs, cstbl);
510 		if (error)
511 			return error;
512 		for(i = 0; i < sizeof(cstbl); i++)
513 			cstbl[i] = i;
514 		nls_mem_toloc(cstbl, cstbl, sizeof(cstbl));
515 		error = smb_addiconvtbl(ssn->ioc_localcs, ssn->ioc_servercs, cstbl);
516 		if (error)
517 			return error;
518 	}
519 	if (ctx->ct_srvaddr) {
520 		error = nb_resolvehost_in(ctx->ct_srvaddr, &sap);
521 	} else {
522 		error = nbns_resolvename(ssn->ioc_srvname, ctx->ct_nb, &sap);
523 	}
524 	if (error) {
525 		smb_error("can't get server address", error);
526 		return error;
527 	}
528 	nn.nn_scope = ctx->ct_nb->nb_scope;
529 	nn.nn_type = NBT_SERVER;
530 	strcpy(nn.nn_name, ssn->ioc_srvname);
531 	error = nb_sockaddr(sap, &nn, &saserver);
532 	nb_snbfree(sap);
533 	if (error) {
534 		smb_error("can't allocate server address", error);
535 		return error;
536 	}
537 	ssn->ioc_server = (struct sockaddr*)saserver;
538 	if (ctx->ct_locname[0] == 0) {
539 		error = nb_getlocalname(ctx->ct_locname);
540 		if (error) {
541 			smb_error("can't get local name", error);
542 			return error;
543 		}
544 		nls_str_upper(ctx->ct_locname, ctx->ct_locname);
545 	}
546 	strcpy(nn.nn_name, ctx->ct_locname);
547 	nn.nn_type = NBT_WKSTA;
548 	nn.nn_scope = ctx->ct_nb->nb_scope;
549 	error = nb_sockaddr(NULL, &nn, &salocal);
550 	if (error) {
551 		nb_snbfree((struct sockaddr*)saserver);
552 		smb_error("can't allocate local address", error);
553 		return error;
554 	}
555 	ssn->ioc_local = (struct sockaddr*)salocal;
556 	ssn->ioc_lolen = salocal->snb_len;
557 	ssn->ioc_svlen = saserver->snb_len;
558 	if (ssn->ioc_password[0] == 0 && (ctx->ct_flags & SMBCF_NOPWD) == 0) {
559 		cp = getpass("Password:");
560 		error = smb_ctx_setpassword(ctx, cp);
561 		if (error)
562 			return error;
563 	}
564 	ctx->ct_flags |= SMBCF_RESOLVED;
565 	return 0;
566 }
567 
568 static int
569 smb_ctx_gethandle(struct smb_ctx *ctx)
570 {
571 	int fd, i;
572 	char buf[20];
573 
574 	/*
575 	 * First, try to open as cloned device
576 	 */
577 	fd = open("/dev/"NSMB_NAME, O_RDWR);
578 	if (fd >= 0) {
579 		ctx->ct_fd = fd;
580 		return 0;
581 	}
582 	/*
583 	 * well, no clone capabilities available - we have to scan
584 	 * all devices in order to get free one
585 	 */
586 	 for (i = 0; i < 1024; i++) {
587 	         snprintf(buf, sizeof(buf), "/dev/%s%d", NSMB_NAME, i);
588 		 fd = open(buf, O_RDWR);
589 		 if (fd >= 0) {
590 			ctx->ct_fd = fd;
591 			return 0;
592 		 }
593 	 }
594 	 /*
595 	  * This is a compatibility with old /dev/net/nsmb device
596 	  */
597 	 for (i = 0; i < 1024; i++) {
598 	         snprintf(buf, sizeof(buf), "/dev/net/%s%d", NSMB_NAME, i);
599 		 fd = open(buf, O_RDWR);
600 		 if (fd >= 0) {
601 			ctx->ct_fd = fd;
602 			return 0;
603 		 }
604 		 if (errno == ENOENT)
605 		         return ENOENT;
606 	 }
607 	 return ENOENT;
608 }
609 
610 int
611 smb_ctx_lookup(struct smb_ctx *ctx, int level, int flags)
612 {
613 	struct smbioc_lookup rq;
614 	int error;
615 
616 	if ((ctx->ct_flags & SMBCF_RESOLVED) == 0) {
617 		smb_error("smb_ctx_lookup() data is not resolved", 0);
618 		return EINVAL;
619 	}
620 	if (ctx->ct_fd != -1) {
621 		close(ctx->ct_fd);
622 		ctx->ct_fd = -1;
623 	}
624 	error = smb_ctx_gethandle(ctx);
625 	if (error) {
626 		smb_error("can't get handle to requester (no /dev/"NSMB_NAME"* device)", 0);
627 		return EINVAL;
628 	}
629 	bzero(&rq, sizeof(rq));
630 	bcopy(&ctx->ct_ssn, &rq.ioc_ssn, sizeof(struct smbioc_ossn));
631 	bcopy(&ctx->ct_sh, &rq.ioc_sh, sizeof(struct smbioc_oshare));
632 	rq.ioc_flags = flags;
633 	rq.ioc_level = level;
634 	if (ioctl(ctx->ct_fd, SMBIOC_LOOKUP, &rq) == -1) {
635 		error = errno;
636 		if (flags & SMBLK_CREATE)
637 			smb_error("unable to open connection", error);
638 		return error;
639 	}
640 	return 0;
641 }
642 
643 int
644 smb_ctx_login(struct smb_ctx *ctx)
645 {
646 	struct smbioc_ossn *ssn = &ctx->ct_ssn;
647 	struct smbioc_oshare *sh = &ctx->ct_sh;
648 	int error;
649 
650 	if ((ctx->ct_flags & SMBCF_RESOLVED) == 0) {
651 		smb_error("smb_ctx_resolve() should be called first", 0);
652 		return EINVAL;
653 	}
654 	if (ctx->ct_fd != -1) {
655 		close(ctx->ct_fd);
656 		ctx->ct_fd = -1;
657 	}
658 	error = smb_ctx_gethandle(ctx);
659 	if (error) {
660 		smb_error("can't get handle to requester", 0);
661 		return EINVAL;
662 	}
663 	if (ioctl(ctx->ct_fd, SMBIOC_OPENSESSION, ssn) == -1) {
664 		error = errno;
665 		smb_error("can't open session to server %s", error, ssn->ioc_srvname);
666 		return error;
667 	}
668 	if (sh->ioc_share[0] == 0)
669 		return 0;
670 	if (ioctl(ctx->ct_fd, SMBIOC_OPENSHARE, sh) == -1) {
671 		error = errno;
672 		smb_error("can't connect to share //%s/%s", error,
673 		    ssn->ioc_srvname, sh->ioc_share);
674 		return error;
675 	}
676 	return 0;
677 }
678 
679 int
680 smb_ctx_setflags(struct smb_ctx *ctx, int level, int mask, int flags)
681 {
682 	struct smbioc_flags fl;
683 
684 	if (ctx->ct_fd == -1)
685 		return EINVAL;
686 	fl.ioc_level = level;
687 	fl.ioc_mask = mask;
688 	fl.ioc_flags = flags;
689 	if (ioctl(ctx->ct_fd, SMBIOC_SETFLAGS, &fl) == -1)
690 		return errno;
691 	return 0;
692 }
693 
694 /*
695  * level values:
696  * 0 - default
697  * 1 - server
698  * 2 - server:user
699  * 3 - server:user:share
700  */
701 static int
702 smb_ctx_readrcsection(struct smb_ctx *ctx, const char *sname, int level)
703 {
704 	char *p;
705 	int error;
706 
707 	if (level >= 0) {
708 		rc_getstringptr(smb_rc, sname, "charsets", &p);
709 		if (p) {
710 			error = smb_ctx_setcharset(ctx, p);
711 			if (error)
712 				smb_error("charset specification in the section '%s' ignored", error, sname);
713 		}
714 	}
715 	if (level <= 1) {
716 		rc_getint(smb_rc, sname, "timeout", &ctx->ct_ssn.ioc_timeout);
717 		rc_getint(smb_rc, sname, "retry_count", &ctx->ct_ssn.ioc_retrycount);
718 	}
719 	if (level == 1) {
720 		rc_getstringptr(smb_rc, sname, "addr", &p);
721 		if (p) {
722 			error = smb_ctx_setsrvaddr(ctx, p);
723 			if (error) {
724 				smb_error("invalid address specified in the section %s", 0, sname);
725 				return error;
726 			}
727 		}
728 	}
729 	if (level >= 2) {
730 		rc_getstringptr(smb_rc, sname, "password", &p);
731 		if (p)
732 			smb_ctx_setpassword(ctx, p);
733 	}
734 	rc_getstringptr(smb_rc, sname, "workgroup", &p);
735 	if (p)
736 		smb_ctx_setworkgroup(ctx, p);
737 	return 0;
738 }
739 
740 /*
741  * read rc file as follows:
742  * 1. read [default] section
743  * 2. override with [server] section
744  * 3. override with [server:user:share] section
745  * Since abcence of rcfile is not fatal, silently ignore this fact.
746  * smb_rc file should be closed by caller.
747  */
748 int
749 smb_ctx_readrc(struct smb_ctx *ctx)
750 {
751 	char sname[SMB_MAXSRVNAMELEN + SMB_MAXUSERNAMELEN + SMB_MAXSHARENAMELEN + 4];
752 /*	char *p;*/
753 
754 	if (smb_open_rcfile() != 0)
755 		return 0;
756 
757 	if (ctx->ct_ssn.ioc_user[0] == 0 || ctx->ct_ssn.ioc_srvname[0] == 0)
758 		return 0;
759 
760 	smb_ctx_readrcsection(ctx, "default", 0);
761 	nb_ctx_readrcsection(smb_rc, ctx->ct_nb, "default", 0);
762 	smb_ctx_readrcsection(ctx, ctx->ct_ssn.ioc_srvname, 1);
763 	nb_ctx_readrcsection(smb_rc, ctx->ct_nb, ctx->ct_ssn.ioc_srvname, 1);
764 	/*
765 	 * SERVER:USER parameters
766 	 */
767 	snprintf(sname, sizeof(sname), "%s:%s", ctx->ct_ssn.ioc_srvname,
768 	    ctx->ct_ssn.ioc_user);
769 	smb_ctx_readrcsection(ctx, sname, 2);
770 
771 	if (ctx->ct_sh.ioc_share[0] != 0) {
772 		/*
773 		 * SERVER:USER:SHARE parameters
774 	         */
775 		snprintf(sname, sizeof(sname), "%s:%s:%s", ctx->ct_ssn.ioc_srvname,
776 		    ctx->ct_ssn.ioc_user, ctx->ct_sh.ioc_share);
777 		smb_ctx_readrcsection(ctx, sname, 3);
778 	}
779 	return 0;
780 }
781 
782