xref: /titanic_44/usr/src/cmd/ssh/sshd/auth-pam.c (revision 9d12795f87b63c2e39e87bff369182edd34677d3)
1 /*
2  * Copyright (c) 2000 Damien Miller.  All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
14  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
15  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
16  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
17  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
18  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
19  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
20  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
22  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23  */
24 /*
25  * Copyright (c) 2001, 2010, Oracle and/or its affiliates. All rights reserved.
26  */
27 
28 #include "includes.h"
29 
30 #ifdef USE_PAM
31 #include "xmalloc.h"
32 #include "log.h"
33 #include "auth.h"
34 #include "auth-options.h"
35 #include "auth-pam.h"
36 #include "buffer.h"
37 #include "servconf.h"
38 #include "canohost.h"
39 #include "compat.h"
40 #include "misc.h"
41 #include "sshlogin.h"
42 #include "ssh-gss.h"
43 
44 #include <security/pam_appl.h>
45 
46 extern char *__progname;
47 
48 extern u_int utmp_len;
49 extern ServerOptions options;
50 
51 extern Authmethod method_kbdint;
52 
53 RCSID("$Id: auth-pam.c,v 1.54 2002/07/28 20:24:08 stevesk Exp $");
54 
55 #define NEW_AUTHTOK_MSG \
56 	"Warning: Your password has expired, please change it now."
57 
58 /* PAM conversation for non-interactive userauth methods */
59 static int do_pam_conversation(int num_msg, const struct pam_message **msg,
60 	struct pam_response **resp, void *appdata_ptr);
61 
62 static void do_pam_cleanup_proc(void *context);
63 
64 static char *get_method_name(Authctxt *authctxt);
65 
66 /* PAM conversation for non-interactive userauth methods */
67 static struct pam_conv conv = {
68 	(int (*)())do_pam_conversation,
69 	NULL
70 };
71 static char *__pam_msg = NULL;
72 
73 static
74 char *
75 get_method_name(Authctxt *authctxt)
76 {
77 	if (!authctxt)
78 		return "(unknown)";
79 
80 	if (!compat20)
81 		return (authctxt->v1_auth_name) ? authctxt->v1_auth_name :
82 						  "(sshv1-unknown)";
83 
84 	if (!authctxt->method || !authctxt->method->name)
85 			return "(sshv2-unknown)";
86 
87 	return authctxt->method->name;
88 }
89 
90 char *
91 derive_pam_service_name(Authmethod *method)
92 {
93 	char *svcname = xmalloc(BUFSIZ);
94 
95 	/*
96 	 * If PamServiceName is set we use that for everything, including
97 	 * SSHv1
98 	 */
99 	if (options.pam_service_name != NULL) {
100 		(void) strlcpy(svcname, options.pam_service_name, BUFSIZ);
101 		return (svcname);
102 	}
103 
104 	if (compat20 && method) {
105 		char *method_name = method->name;
106 
107 		if (!method_name)
108 			fatal("Userauth method unknown while starting PAM");
109 
110 		/*
111 		 * For SSHv2 we use "sshd-<userauth name>
112 		 * The "sshd" prefix can be changed via the PAMServicePrefix
113 		 * sshd_config option.
114 		 */
115 		if (strcmp(method_name, "none") == 0) {
116 			snprintf(svcname, BUFSIZ, "%s-none",
117 			    options.pam_service_prefix);
118 		}
119 		if (strcmp(method_name, "password") == 0) {
120 			snprintf(svcname, BUFSIZ, "%s-password",
121 			    options.pam_service_prefix);
122 		}
123 		if (strcmp(method_name, "keyboard-interactive") == 0) {
124 			/* "keyboard-interactive" is too long, shorten it */
125 			snprintf(svcname, BUFSIZ, "%s-kbdint",
126 			    options.pam_service_prefix);
127 		}
128 		if (strcmp(method_name, "publickey") == 0) {
129 			/* "publickey" is too long, shorten it */
130 			snprintf(svcname, BUFSIZ, "%s-pubkey",
131 			    options.pam_service_prefix);
132 		}
133 		if (strcmp(method_name, "hostbased") == 0) {
134 			/* "hostbased" can't really be shortened... */
135 			snprintf(svcname, BUFSIZ, "%s-hostbased",
136 			    options.pam_service_prefix);
137 		}
138 		if (strncmp(method_name, "gss", 3) == 0) {
139 			/* "gss" is too short, elongate it */
140 			snprintf(svcname, BUFSIZ, "%s-gssapi",
141 			    options.pam_service_prefix);
142 		}
143 		return svcname;
144 	} else {
145 		/* SSHv1 doesn't get to be so cool */
146 		snprintf(svcname, BUFSIZ, "%s-v1",
147 		    options.pam_service_prefix);
148 	}
149 	return svcname;
150 }
151 
152 void
153 new_start_pam(Authctxt *authctxt, struct pam_conv *conv)
154 {
155 	int		retval;
156 	pam_handle_t	*pamh;
157 	const char	*rhost;
158 	char		*svc;
159 	char		*user = NULL;
160 	pam_stuff	*pam;
161 
162 	if (authctxt == NULL)
163 		fatal("Internal error during userauth");
164 
165 	if (compat20 && authctxt->method == NULL)
166 		fatal("Userauth method unknown while starting PAM");
167 
168 	/* PAM service selected here */
169 	svc = derive_pam_service_name(authctxt->method);
170 	debug2("Starting PAM service %s for method %s", svc,
171 		get_method_name(authctxt));
172 
173 	if (authctxt->user != NULL)
174 		user = authctxt->user;
175 
176 	/* Cleanup previous PAM state */
177 	if (authctxt->pam != NULL) {
178 		fatal_remove_cleanup(&do_pam_cleanup_proc, authctxt->pam);
179 		do_pam_cleanup_proc(authctxt->pam);
180 	}
181 
182 	pam = xmalloc(sizeof(pam_stuff));
183 	(void) memset(pam, 0, sizeof(pam_stuff));
184 
185 	/*
186 	 * pam->last_pam_retval has to be and is considered
187 	 * along with pam->state.
188 	 *
189 	 * pam->state = 0; -> no PAM auth, account, etc, work
190 	 * done yet.  (Set by memset() above.)
191 	 *
192 	 * pam->last_pam_retval = PAM_SUCCESS; -> meaningless at
193 	 * this point.
194 	 *
195 	 * See finish_userauth_do_pam() below.
196 	 */
197 	pam->authctxt = authctxt;
198 	pam->last_pam_retval = PAM_SUCCESS;
199 
200 	authctxt->pam = pam;
201 
202 	/* Free any previously stored text/error PAM prompts */
203 	if (__pam_msg) {
204 		xfree(__pam_msg);
205 		__pam_msg = NULL;
206 	}
207 
208 	if ((retval = pam_start(svc, user, conv, &pamh)) != PAM_SUCCESS) {
209 		fatal("PAM initialization failed during %s userauth",
210 			get_method_name(authctxt));
211 	}
212 
213 	free(svc);
214 
215 	fatal_add_cleanup((void (*)(void *)) &do_pam_cleanup_proc,
216 			  (void *) authctxt->pam);
217 
218 	rhost = get_remote_name_or_ip(utmp_len, options.verify_reverse_mapping);
219 	if ((retval = pam_set_item(pamh, PAM_RHOST, rhost)) != PAM_SUCCESS) {
220 		(void) pam_end(pamh, retval);
221 		fatal("Could not set PAM_RHOST item during %s userauth",
222 			get_method_name(authctxt));
223 	}
224 
225 	if ((retval = pam_set_item(pamh, PAM_TTY, "sshd")) != PAM_SUCCESS) {
226 		(void) pam_end(pamh, retval);
227 		fatal("Could not set PAM_TTY item during %s userauth",
228 			get_method_name(authctxt));
229 	}
230 
231 	if (authctxt->cuser != NULL)
232 		if ((retval = pam_set_item(pamh, PAM_AUSER, authctxt->cuser)) != PAM_SUCCESS) {
233 			(void) pam_end(pamh, retval);
234 			fatal("Could not set PAM_AUSER item during %s userauth",
235 				get_method_name(authctxt));
236 		}
237 
238 	authctxt->pam->h = pamh;
239 }
240 
241 /*
242  * To be called from userauth methods, directly (as in keyboard-interactive) or
243  * indirectly (from auth_pam_password() or from do_pam_non_initial_userauth().
244  *
245  * The caller is responsible for calling new_start_pam() first.
246  *
247  * PAM state is not cleaned up here on error.  This is left to subsequent calls
248  * to new_start_pam() or to the cleanup function upon authentication error.
249  */
250 int
251 finish_userauth_do_pam(Authctxt *authctxt)
252 {
253 	int retval;
254 	char *user, *method;
255 
256 	/* Various checks; fail gracefully */
257 	if (authctxt == NULL || authctxt->pam == NULL)
258 		return PAM_SYSTEM_ERR;	/* shouldn't happen */
259 
260 	if (compat20) {
261 		if (authctxt->method == NULL || authctxt->method->name == NULL)
262 			return PAM_SYSTEM_ERR;	/* shouldn't happen */
263 		method = authctxt->method->name;
264 	} else if ((method = authctxt->v1_auth_name) == NULL)
265 		return PAM_SYSTEM_ERR;	/* shouldn't happen */
266 
267 	if (AUTHPAM_DONE(authctxt))
268 		return PAM_SYSTEM_ERR;	/* shouldn't happen */
269 
270 	if (!(authctxt->pam->state & PAM_S_DONE_ACCT_MGMT)) {
271 		retval = pam_acct_mgmt(authctxt->pam->h, 0);
272 		authctxt->pam->last_pam_retval = retval;
273 		if (retval == PAM_NEW_AUTHTOK_REQD) {
274 			userauth_force_kbdint();
275 			return retval;
276 		}
277 		if (retval != PAM_SUCCESS)
278 			return retval;
279 		authctxt->pam->state |= PAM_S_DONE_ACCT_MGMT;
280 	}
281 
282 	/*
283 	 * Handle PAM_USER change, if any.
284 	 *
285 	 * We do this before pam_open_session() because we need the PAM_USER's
286 	 * UID for:
287 	 *
288 	 * a) PermitRootLogin checking
289 	 * b) to get at the lastlog entry before pam_open_session() updates it.
290 	 */
291 	retval = pam_get_item(authctxt->pam->h, PAM_USER, (void **) &user);
292 	if (retval != PAM_SUCCESS) {
293 		fatal("PAM failure: pam_get_item(PAM_USER) "
294 		      "returned %d: %.200s", retval,
295 		      PAM_STRERROR(authctxt->pam->h, retval));
296 	}
297 
298 	if (user == NULL || *user == '\0') {
299 		debug("PAM set NULL PAM_USER");
300 		return PAM_PERM_DENIED;
301 	}
302 
303 	if (strcmp(user, authctxt->user) != 0) {
304 		log("PAM changed the SSH username");
305 		pwfree(&authctxt->pw);
306 		authctxt->pw = getpwnamallow(user);
307 		authctxt->valid = (authctxt->pw != NULL);
308 		xfree(authctxt->user);
309 		authctxt->user = xstrdup(user);
310 	}
311 
312 	if (!authctxt->valid) {
313 		debug2("PAM set PAM_USER to unknown user");
314 		/*
315 		 * Return success, userauth_finish() will catch
316 		 * this and send back a failure message.
317 		 */
318 		return PAM_SUCCESS;
319 	}
320 
321 	/* Check PermitRootLogin semantics */
322 	if (authctxt->pw->pw_uid == 0 && !auth_root_allowed(method))
323 		return PAM_PERM_DENIED;
324 
325 	if (!(authctxt->pam->state & PAM_S_DONE_SETCRED)) {
326 		retval = pam_setcred(authctxt->pam->h,
327 				     PAM_ESTABLISH_CRED);
328 		authctxt->pam->last_pam_retval = retval;
329 		if (retval != PAM_SUCCESS)
330 			return retval;
331 		authctxt->pam->state |= PAM_S_DONE_SETCRED;
332 
333 #ifdef GSSAPI
334 		/*
335 		 * Store GSS-API delegated creds after pam_setcred(), which may
336 		 * have set the current credential store.
337 		 */
338 		ssh_gssapi_storecreds(NULL, authctxt);
339 #endif /* GSSAPI */
340 	}
341 
342 	/*
343 	 * On Solaris pam_unix_session.so updates the lastlog, but does
344 	 * not converse a PAM_TEXT_INFO message about it.  So we need to
345 	 * fetch the lastlog entry here and save it for use later.
346 	 */
347 	authctxt->last_login_time =
348 		get_last_login_time(authctxt->pw->pw_uid,
349 			authctxt->pw->pw_name,
350 			authctxt->last_login_host,
351 			sizeof(authctxt->last_login_host));
352 
353 	if (!(authctxt->pam->state & PAM_S_DONE_OPEN_SESSION)) {
354 		retval = pam_open_session(authctxt->pam->h, 0);
355 		authctxt->pam->last_pam_retval = retval;
356 		if (retval != PAM_SUCCESS)
357 			return retval;
358 		authctxt->pam->state |= PAM_S_DONE_OPEN_SESSION;
359 	}
360 
361 	/*
362 	 * All PAM work done successfully.
363 	 *
364 	 * PAM handle stays around so we can call pam_close_session() on
365 	 * it later.
366 	 */
367 	return PAM_SUCCESS;
368 }
369 
370 /*
371  * PAM conversation function for non-interactive userauth methods that
372  * really cannot do any prompting.  Password userauth and CHANGEREQ can
373  * always set the PAM_AUTHTOK and PAM_OLDAUTHTOK items to avoid
374  * conversation (and if they do and nonetheless some module tries to
375  * converse, then password userauth / CHANGEREQ MUST fail).
376  *
377  * Except, PAM_TEXT_INFO and PAM_ERROR_MSG prompts can be squirelled
378  * away and shown to the user later.
379  *
380  * Keyboard-interactive userauth has its own much more interesting
381  * conversation function.
382  *
383  */
384 static int
385 do_pam_conversation(int num_msg, const struct pam_message **msg,
386 	struct pam_response **resp, void *appdata_ptr)
387 {
388 	struct pam_response *reply;
389 	int count;
390 
391 	/* PAM will free this later */
392 	reply = xmalloc(num_msg * sizeof(*reply));
393 
394 	(void) memset(reply, 0, num_msg * sizeof(*reply));
395 
396 	for (count = 0; count < num_msg; count++) {
397 		/*
398 		 * We can't use stdio yet, queue messages for
399 		 * printing later
400 		 */
401 		switch(PAM_MSG_MEMBER(msg, count, msg_style)) {
402 		case PAM_PROMPT_ECHO_ON:
403 			xfree(reply);
404 			return PAM_CONV_ERR;
405 		case PAM_PROMPT_ECHO_OFF:
406 			xfree(reply);
407 			return PAM_CONV_ERR;
408 			break;
409 		case PAM_ERROR_MSG:
410 		case PAM_TEXT_INFO:
411 			if (PAM_MSG_MEMBER(msg, count, msg) != NULL) {
412 				message_cat(&__pam_msg,
413 				    PAM_MSG_MEMBER(msg, count, msg));
414 			}
415 			reply[count].resp = xstrdup("");
416 			reply[count].resp_retcode = PAM_SUCCESS;
417 			break;
418 		default:
419 			xfree(reply);
420 			return PAM_CONV_ERR;
421 		}
422 	}
423 
424 	*resp = reply;
425 
426 	return PAM_SUCCESS;
427 }
428 
429 /* Called at exit to cleanly shutdown PAM */
430 static void
431 do_pam_cleanup_proc(void *context)
432 {
433 	int pam_retval;
434 	pam_stuff *pam = (pam_stuff *) context;
435 
436 	if (pam == NULL)
437 		return;
438 
439 	if (pam->authctxt != NULL && pam->authctxt->pam == pam) {
440 		pam->authctxt->pam_retval = pam->last_pam_retval;
441 		pam->authctxt->pam = NULL;
442 		pam->authctxt = NULL;
443 	}
444 
445 	if (pam->h == NULL)
446 		return;
447 
448 	/*
449 	 * We're in fatal_cleanup() or not in userauth or without a
450 	 * channel -- can't converse now, too bad.
451 	 */
452 	pam_retval = pam_set_item(pam->h, PAM_CONV, NULL);
453 	if (pam_retval != PAM_SUCCESS) {
454 		log("Cannot remove PAM conv, close session or delete creds[%d]: %.200s",
455 			pam_retval, PAM_STRERROR(pam->h, pam_retval));
456 		goto cleanup;
457 	}
458 
459 	if (pam->state & PAM_S_DONE_OPEN_SESSION) {
460 		pam_retval = pam_close_session(pam->h, 0);
461 		if (pam_retval != PAM_SUCCESS)
462 			log("Cannot close PAM session[%d]: %.200s",
463 			    pam_retval, PAM_STRERROR(pam->h, pam_retval));
464 	}
465 
466 	if (pam->state & PAM_S_DONE_SETCRED) {
467 		pam_retval = pam_setcred(pam->h, PAM_DELETE_CRED);
468 		if (pam_retval != PAM_SUCCESS)
469 			debug("Cannot delete credentials[%d]: %.200s",
470 			    pam_retval, PAM_STRERROR(pam->h, pam_retval));
471 	}
472 
473 cleanup:
474 
475 	/* Use the previous PAM result, if not PAM_SUCCESS for pam_end() */
476 	if (pam->last_pam_retval != PAM_SUCCESS)
477 		pam_retval = pam_end(pam->h, pam->last_pam_retval);
478 	else if (pam_retval != PAM_SUCCESS)
479 		pam_retval = pam_end(pam->h, pam_retval);
480 	else
481 		pam_retval = pam_end(pam->h, PAM_ABORT);
482 
483 	if (pam_retval != PAM_SUCCESS)
484 		log("Cannot release PAM authentication[%d]: %.200s",
485 		    pam_retval, PAM_STRERROR(pam->h, pam_retval));
486 
487 	xfree(pam);
488 }
489 
490 /* Attempt password authentation using PAM */
491 int
492 auth_pam_password(Authctxt *authctxt, const char *password)
493 {
494 	int retval;
495 
496 	/* Ensure we have a fresh PAM handle / state */
497 	new_start_pam(authctxt, &conv);
498 
499 	retval = pam_set_item(authctxt->pam->h, PAM_AUTHTOK, password);
500 	if (retval != PAM_SUCCESS) {
501 		authctxt->pam->last_pam_retval = retval;
502 		return 1;
503 	}
504 
505 	retval = pam_authenticate(authctxt->pam->h,
506 			options.permit_empty_passwd ?  0 :
507 			PAM_DISALLOW_NULL_AUTHTOK);
508 
509 	if (retval != PAM_SUCCESS) {
510 		authctxt->pam->last_pam_retval = retval;
511 		return 0;
512 	}
513 
514 	if ((retval = finish_userauth_do_pam(authctxt)) != PAM_SUCCESS)
515 		return 0;
516 
517 	if (authctxt->method)
518 		authctxt->method->authenticated = 1;	/* SSHv2 */
519 
520 	return 1;
521 }
522 
523 int
524 do_pam_non_initial_userauth(Authctxt *authctxt)
525 {
526 	new_start_pam(authctxt, NULL);
527 	return (finish_userauth_do_pam(authctxt) == PAM_SUCCESS);
528 }
529 
530 /* Cleanly shutdown PAM */
531 void finish_pam(Authctxt *authctxt)
532 {
533 	fatal_remove_cleanup(&do_pam_cleanup_proc, authctxt->pam);
534 	do_pam_cleanup_proc(authctxt->pam);
535 }
536 
537 static
538 char **
539 find_env(char **env, char *var)
540 {
541 	char **p;
542 	int len;
543 
544 	if (strchr(var, '=') == NULL)
545 		len = strlen(var);
546 	else
547 		len = (strchr(var, '=') - var) + 1;
548 
549 	for ( p = env ; p != NULL && *p != NULL ; p++ ) {
550 		if (strncmp(*p, var, len) == 0)
551 			return (p);
552 	}
553 
554 	return (NULL);
555 }
556 
557 /* Return list of PAM environment strings */
558 char **
559 fetch_pam_environment(Authctxt *authctxt)
560 {
561 #ifdef HAVE_PAM_GETENVLIST
562 	char	**penv;
563 
564 	if (authctxt == NULL || authctxt->pam == NULL ||
565 	    authctxt->pam->h == NULL)
566 		return (NULL);
567 
568 	penv = pam_getenvlist(authctxt->pam->h);
569 
570 	return (penv);
571 #else /* HAVE_PAM_GETENVLIST */
572 	return(NULL);
573 #endif /* HAVE_PAM_GETENVLIST */
574 }
575 
576 void free_pam_environment(char **env)
577 {
578 	int i;
579 
580 	if (env != NULL) {
581 		for (i = 0; env[i] != NULL; i++)
582 			xfree(env[i]);
583 	}
584 
585 	xfree(env);
586 }
587 
588 /* Print any messages that have been generated during authentication */
589 /* or account checking to stderr */
590 void print_pam_messages(void)
591 {
592 	if (__pam_msg != NULL)
593 		(void) fputs(__pam_msg, stderr);
594 }
595 
596 /* Append a message to buffer */
597 void message_cat(char **p, const char *a)
598 {
599 	char *cp;
600 	size_t new_len;
601 
602 	new_len = strlen(a);
603 
604 	if (*p) {
605 		size_t len = strlen(*p);
606 
607 		*p = xrealloc(*p, new_len + len + 2);
608 		cp = *p + len;
609 	} else
610 		*p = cp = xmalloc(new_len + 2);
611 
612 	(void) memcpy(cp, a, new_len);
613 	cp[new_len] = '\n';
614 	cp[new_len + 1] = '\0';
615 }
616 
617 #endif /* USE_PAM */
618