diff options
author | Theo de Raadt <deraadt@cvs.openbsd.org> | 2008-06-24 14:27:27 +0000 |
---|---|---|
committer | Theo de Raadt <deraadt@cvs.openbsd.org> | 2008-06-24 14:27:27 +0000 |
commit | 3c00af7e9ff7ae6d3fd059414ddfacb8b93b6254 (patch) | |
tree | 6351c162a7187ca965cd914033181351c0ea1b76 /lib/libc/gen/getpwent.c | |
parent | 53ad4afa4c64d3779defb74205750e18bc153077 (diff) |
implement getpwnam_r() and getpwuid_r() -- very nearly a rewrite of the
entire file. much help from kurt, and tested by many
Diffstat (limited to 'lib/libc/gen/getpwent.c')
-rw-r--r-- | lib/libc/gen/getpwent.c | 872 |
1 files changed, 403 insertions, 469 deletions
diff --git a/lib/libc/gen/getpwent.c b/lib/libc/gen/getpwent.c index 5a2e9b1d8ed..38c0b9425f4 100644 --- a/lib/libc/gen/getpwent.c +++ b/lib/libc/gen/getpwent.c @@ -1,5 +1,6 @@ -/* $OpenBSD: getpwent.c,v 1.34 2005/08/08 08:05:34 espie Exp $ */ +/* $OpenBSD: getpwent.c,v 1.35 2008/06/24 14:27:24 deraadt Exp $ */ /* + * Copyright (c) 2008 Theo de Raadt * Copyright (c) 1988, 1993 * The Regents of the University of California. All rights reserved. * Portions Copyright (c) 1994, 1995, 1996, Jason Downs. All rights reserved. @@ -48,53 +49,69 @@ #include <rpcsvc/ypclnt.h> #include "ypinternal.h" #endif +#include "thread_private.h" + +_THREAD_PRIVATE_KEY(pw); -static struct passwd _pw_passwd; /* password structure */ static DB *_pw_db; /* password database */ + +/* Following are used only by setpwent(), getpwent(), and endpwent() */ +static struct passwd _pw_passwd; /* password structure */ +static char _pw_string[1024]; /* string pointed to by _pw_passwd */ static int _pw_keynum; /* key counter */ static int _pw_stayopen; /* keep fd's open */ static int _pw_flags; /* password flags */ -static int __hashpw(DBT *); -static int __initdb(void); + +static int __hashpw(DBT *, char *buf, size_t buflen, struct passwd *, int *); +static int __initdb(); +static struct passwd *_pwhashbyname(const char *name, char *buf, + size_t buflen, struct passwd *pw, int *); +static struct passwd *_pwhashbyuid(uid_t uid, char *buf, + size_t buflen, struct passwd *pw, int *); #ifdef YP -enum _ypmode { YPMODE_NONE, YPMODE_FULL, YPMODE_USER, YPMODE_NETGRP }; -static enum _ypmode __ypmode; +static char *__ypdomain; -static char *__ypcurrent, *__ypdomain; +struct _ypexclude { + const char *name; + struct _ypexclude *next; +}; + +/* Following are used only by setpwent(), getpwent(), and endpwent() */ +enum _ypmode { YPMODE_NONE, YPMODE_FULL, YPMODE_USER, YPMODE_NETGRP }; +static enum _ypmode __ypmode; +static char *__ypcurrent; static int __ypcurrentlen; -static struct passwd *__ypproto = (struct passwd *)NULL; -static int __ypflags; +static int __yp_pw_flags; +static struct passwd *__ypproto; static char __ypline[1024]; -static long __yppbuf[1024 / sizeof(long)]; -static int __yp_override_passwd = 0; +static int __getpwent_has_yppw = -1; +static struct _ypexclude *__ypexhead; -static int __has_yppw(void); +static int __has_yppw(); static int __has_ypmaster(void); +static int __ypexclude_add(struct _ypexclude **, const char *); +static int __ypexclude_is(struct _ypexclude **, const char *); +static void __ypexclude_free(struct _ypexclude **); +static void __ypproto_set(struct passwd *, long *, int, int *); +static int __ypparse(struct passwd *pw, char *s, int); -static int __ypexclude_add(const char *); -static int __ypexclude_is(const char *); -static void __ypexclude_free(void); -static void __ypproto_set(void); -static int __ypparse(struct passwd *pw, char *s); +#define LOOKUP_BYNAME 0 +#define LOOKUP_BYUID 1 +static struct passwd *__yppwlookup(int, char *, uid_t, struct passwd *, + char *, size_t, int *); /* macro for deciding which YP maps to use. */ #define PASSWD_BYNAME \ - __has_ypmaster() ? "master.passwd.byname" : "passwd.byname" + (__has_ypmaster() ? "master.passwd.byname" : "passwd.byname") #define PASSWD_BYUID \ - __has_ypmaster() ? "master.passwd.byuid" : "passwd.byuid" - -struct _ypexclude { - const char *name; - struct _ypexclude *next; -}; -static struct _ypexclude *__ypexclude = (struct _ypexclude *)NULL; + (__has_ypmaster() ? "master.passwd.byuid" : "passwd.byuid") /* * Using DB for this just wastes too damn much memory. */ static int -__ypexclude_add(const char *name) +__ypexclude_add(struct _ypexclude **headp, const char *name) { struct _ypexclude *new; @@ -105,24 +122,22 @@ __ypexclude_add(const char *name) if (new == NULL) return (1); new->name = strdup(name); - if (new->name == (char *)NULL) { + if (new->name == NULL) { free(new); return (1); } - new->next = __ypexclude; - __ypexclude = new; - + new->next = *headp; + *headp = new; return (0); } static int -__ypexclude_is(const char *name) +__ypexclude_is(struct _ypexclude **headp, const char *name) { struct _ypexclude *curr; - for (curr = __ypexclude; curr != (struct _ypexclude *)NULL; - curr = curr->next) { + for (curr = *headp; curr; curr = curr->next) { if (strcmp(curr->name, name) == 0) return (1); /* excluded */ } @@ -130,28 +145,25 @@ __ypexclude_is(const char *name) } static void -__ypexclude_free(void) +__ypexclude_free(struct _ypexclude **headp) { struct _ypexclude *curr, *next; - for (curr = __ypexclude; curr != (struct _ypexclude *)NULL; - curr = next) { + for (curr = *headp; curr; curr = next) { next = curr->next; - free((void *)curr->name); free(curr); } - __ypexclude = (struct _ypexclude *)NULL; + *headp = NULL; } static void -__ypproto_set(void) +__ypproto_set(struct passwd *pw, long *buf, int flags, int *yp_pw_flagsp) { char *ptr; - struct passwd *pw = &_pw_passwd; /* make this the new prototype */ - ptr = (char *)__yppbuf; + ptr = (char *)buf; /* first allocate the struct. */ __ypproto = (struct passwd *)ptr; @@ -164,7 +176,7 @@ __ypproto_set(void) __ypproto->pw_name = ptr; ptr += (strlen(pw->pw_name) + 1); } else - __ypproto->pw_name = (char *)NULL; + __ypproto->pw_name = NULL; /* password */ if (pw->pw_passwd && (pw->pw_passwd)[0]) { @@ -173,7 +185,7 @@ __ypproto_set(void) __ypproto->pw_passwd = ptr; ptr += (strlen(pw->pw_passwd) + 1); } else - __ypproto->pw_passwd = (char *)NULL; + __ypproto->pw_passwd = NULL; /* uid */ __ypproto->pw_uid = pw->pw_uid; @@ -194,7 +206,7 @@ __ypproto_set(void) __ypproto->pw_gecos = ptr; ptr += (strlen(pw->pw_gecos) + 1); } else - __ypproto->pw_gecos = (char *)NULL; + __ypproto->pw_gecos = NULL; /* dir */ if (pw->pw_dir && (pw->pw_dir)[0]) { @@ -203,7 +215,7 @@ __ypproto_set(void) __ypproto->pw_dir = ptr; ptr += (strlen(pw->pw_dir) + 1); } else - __ypproto->pw_dir = (char *)NULL; + __ypproto->pw_dir = NULL; /* shell */ if (pw->pw_shell && (pw->pw_shell)[0]) { @@ -212,17 +224,17 @@ __ypproto_set(void) __ypproto->pw_shell = ptr; ptr += (strlen(pw->pw_shell) + 1); } else - __ypproto->pw_shell = (char *)NULL; + __ypproto->pw_shell = NULL; /* expire (ignored anyway) */ __ypproto->pw_expire = pw->pw_expire; /* flags */ - __ypflags = _pw_flags; + *yp_pw_flagsp = flags; } static int -__ypparse(struct passwd *pw, char *s) +__ypparse(struct passwd *pw, char *s, int yp_pw_flags) { char *bp, *cp, *endp; u_long ul; @@ -240,16 +252,16 @@ __ypparse(struct passwd *pw, char *s) pw->pw_name = strsep(&bp, ":\n"); pw->pw_passwd = strsep(&bp, ":\n"); if (!(cp = strsep(&bp, ":\n"))) - return 1; + return (1); ul = strtoul(cp, &endp, 10); if (endp == cp || *endp != '\0' || ul >= UID_MAX) - return 1; + return (1); pw->pw_uid = (uid_t)ul; if (!(cp = strsep(&bp, ":\n"))) - return 1; + return (1); ul = strtoul(cp, &endp, 10); if (endp == cp || *endp != '\0' || ul >= GID_MAX) - return 1; + return (1); pw->pw_gid = (gid_t)ul; if (count == 9) { long l; @@ -257,16 +269,16 @@ __ypparse(struct passwd *pw, char *s) /* If the ypserv gave us all the fields, use them. */ pw->pw_class = strsep(&bp, ":\n"); if (!(cp = strsep(&bp, ":\n"))) - return 1; + return (1); l = strtol(cp, &endp, 10); if (endp == cp || *endp != '\0' || l >= INT_MAX || l <= INT_MIN) - return 1; + return (1); pw->pw_change = (time_t)l; if (!(cp = strsep(&bp, ":\n"))) - return 1; + return (1); l = strtol(cp, &endp, 10); if (endp == cp || *endp != '\0' || l >= INT_MAX || l <= INT_MIN) - return 1; + return (1); pw->pw_expire = (time_t)l; } else { /* ..else it is a normal ypserv. */ @@ -279,50 +291,48 @@ __ypparse(struct passwd *pw, char *s) pw->pw_shell = strsep(&bp, ":\n"); /* now let the prototype override, if set. */ - if (__ypproto != (struct passwd *)NULL) { - if (__yp_override_passwd && __ypproto->pw_passwd != (char *)NULL) - pw->pw_passwd = __ypproto->pw_passwd; - if (!(__ypflags & _PASSWORD_NOUID)) + if (__ypproto) { + if (!(yp_pw_flags & _PASSWORD_NOUID)) pw->pw_uid = __ypproto->pw_uid; - if (!(__ypflags & _PASSWORD_NOGID)) + if (!(yp_pw_flags & _PASSWORD_NOGID)) pw->pw_gid = __ypproto->pw_gid; - if (__ypproto->pw_gecos != (char *)NULL) + if (__ypproto->pw_gecos) pw->pw_gecos = __ypproto->pw_gecos; - if (__ypproto->pw_dir != (char *)NULL) + if (__ypproto->pw_dir) pw->pw_dir = __ypproto->pw_dir; - if (__ypproto->pw_shell != (char *)NULL) + if (__ypproto->pw_shell) pw->pw_shell = __ypproto->pw_shell; } - return 0; + return (0); } #endif -#ifdef YP -static int __getpwent_has_yppw = -1; -#endif - struct passwd * getpwent(void) { - DBT key; - char bf[sizeof(_pw_keynum) + 1]; #ifdef YP - static char *name = (char *)NULL; - const char *user, *host, *dom; + static char *name = NULL; + char *map; #endif + char bf[1 + sizeof(_pw_keynum)], pwbuf[1024]; + struct passwd *pw = NULL; + DBT key; + _THREAD_PRIVATE_MUTEX_LOCK(pw); if (!_pw_db && !__initdb()) - return ((struct passwd *)NULL); + goto done; #ifdef YP + map = PASSWD_BYNAME; + if (__getpwent_has_yppw == -1) __getpwent_has_yppw = __has_yppw(); again: if (__getpwent_has_yppw && (__ypmode != YPMODE_NONE)) { + const char *user, *host, *dom; + int keylen, datalen, r, s; char *key, *data; - int keylen, datalen; - int r, s; if (!__ypdomain) { if (_yp_check(&__ypdomain) == 0) { @@ -333,12 +343,12 @@ again: switch (__ypmode) { case YPMODE_FULL: if (__ypcurrent) { - r = yp_next(__ypdomain, (PASSWD_BYNAME), + r = yp_next(__ypdomain, map, __ypcurrent, __ypcurrentlen, &key, &keylen, &data, &datalen); free(__ypcurrent); + __ypcurrent = NULL; if (r != 0) { - __ypcurrent = NULL; __ypmode = YPMODE_NONE; if (data) free(data); @@ -347,23 +357,21 @@ again: } __ypcurrent = key; __ypcurrentlen = keylen; - bcopy(data, __ypline, datalen); - free(data); - data = NULL; } else { - r = yp_first(__ypdomain, (PASSWD_BYNAME), + r = yp_first(__ypdomain, map, &__ypcurrent, &__ypcurrentlen, &data, &datalen); - if (r != 0) { + if (r != 0 || + __ypcurrentlen > sizeof(__ypline)) { __ypmode = YPMODE_NONE; if (data) free(data); goto again; } - bcopy(data, __ypline, datalen); - free(data); - data = NULL; } + bcopy(data, __ypline, datalen); + free(data); + data = NULL; break; case YPMODE_NETGRP: s = getnetgrent(&host, &user, &dom); @@ -373,12 +381,12 @@ again: goto again; } if (user && *user) { - r = yp_match(__ypdomain, (PASSWD_BYNAME), - user, strlen(user), - &data, &datalen); + r = yp_match(__ypdomain, map, + user, strlen(user), &data, &datalen); } else goto again; - if (r != 0) { + if (r != 0 || + __ypcurrentlen > sizeof(__ypline)) { /* * if the netgroup is invalid, keep looking * as there may be valid users later on. @@ -389,24 +397,24 @@ again: } bcopy(data, __ypline, datalen); free(data); - data = (char *)NULL; + data = NULL; break; case YPMODE_USER: - if (name != (char *)NULL) { - r = yp_match(__ypdomain, (PASSWD_BYNAME), - name, strlen(name), - &data, &datalen); + if (name) { + r = yp_match(__ypdomain, map, + name, strlen(name), &data, &datalen); __ypmode = YPMODE_NONE; free(name); - name = (char *)NULL; - if (r != 0) { + name = NULL; + if (r != 0 || + __ypcurrentlen > sizeof(__ypline)) { if (data) free(data); goto again; } bcopy(data, __ypline, datalen); free(data); - data = (char *)NULL; + data = NULL; } else { /* XXX */ __ypmode = YPMODE_NONE; goto again; @@ -418,19 +426,23 @@ again: } __ypline[datalen] = '\0'; - if (__ypparse(&_pw_passwd, __ypline)) + if (__ypparse(&_pw_passwd, __ypline, __yp_pw_flags)) goto again; - return &_pw_passwd; + pw = &_pw_passwd; + goto done; } #endif ++_pw_keynum; bf[0] = _PW_KEYBYNUM; - bcopy((char *)&_pw_keynum, bf + 1, sizeof(_pw_keynum)); + bcopy((char *)&_pw_keynum, &bf[1], sizeof(_pw_keynum)); key.data = (u_char *)bf; - key.size = sizeof(_pw_keynum) + 1; - if (__hashpw(&key)) { + key.size = 1 + sizeof(_pw_keynum); + if (__hashpw(&key, pwbuf, sizeof pwbuf, &_pw_passwd, &_pw_flags)) { #ifdef YP + static long __yppbuf[1024 / sizeof(long)]; + const char *user, *host, *dom; + /* if we don't have YP at all, don't bother. */ if (__getpwent_has_yppw) { if (_pw_passwd.pw_name[0] == '+') { @@ -449,8 +461,8 @@ again: break; } - /* save the prototype */ - __ypproto_set(); + __ypproto_set(&_pw_passwd, __yppbuf, + _pw_flags, &__yp_pw_flags); goto again; } else if (_pw_passwd.pw_name[0] == '-') { /* an attempted exclusion */ @@ -461,25 +473,30 @@ again: setnetgrent(_pw_passwd.pw_name + 2); while (getnetgrent(&host, &user, &dom)) { if (user && *user) - __ypexclude_add(user); + __ypexclude_add(&__ypexhead, + user); } endnetgrent(); break; default: - __ypexclude_add(_pw_passwd.pw_name + 1); + __ypexclude_add(&__ypexhead, + _pw_passwd.pw_name + 1); break; } goto again; } } #endif - return &_pw_passwd; + pw = &_pw_passwd; + goto done; } - return (struct passwd *)NULL; + +done: + _THREAD_PRIVATE_MUTEX_UNLOCK(pw); + return (pw); } #ifdef YP - /* * See if the YP token is in the database. Only works if pwd_mkdb knows * about the token. @@ -487,10 +504,9 @@ again: static int __has_yppw(void) { - DBT key, data; - DBT pkey, pdata; + DBT key, data, pkey, pdata; + char bf[1 + _PW_NAME_LEN]; int len; - char bf[_PW_NAME_LEN + 1]; key.data = (u_char *)_PW_YPTOKEN; key.size = strlen(_PW_YPTOKEN); @@ -498,9 +514,9 @@ __has_yppw(void) /* Pre-token database support. */ bf[0] = _PW_KEYBYNAME; len = strlen("+"); - bcopy("+", bf + 1, MIN(len, _PW_NAME_LEN)); + bcopy("+", &bf[1], MIN(len, _PW_NAME_LEN)); pkey.data = (u_char *)bf; - pkey.size = MIN(len, _PW_NAME_LEN) + 1; + pkey.size = 1 + MIN(len, _PW_NAME_LEN); if ((_pw_db->get)(_pw_db, &key, &data, 0) && (_pw_db->get)(_pw_db, &pkey, &pdata, 0)) @@ -509,8 +525,7 @@ __has_yppw(void) } /* - * See if there's a FreeBSD-style master.passwd map set. From the FreeBSD - * libc code. + * See if there's a master.passwd map. */ static int __has_ypmaster(void) @@ -551,370 +566,300 @@ __has_ypmaster(void) checked = 0; return (checked); } - free (result); + free(result); + if (key) + free(key); saved_uid = uid; saved_euid = euid; checked = 1; return (checked); } -#endif -struct passwd * -getpwnam(const char *name) +static struct passwd * +__yppwlookup(int lookup, char *name, uid_t uid, struct passwd *pw, + char *buf, size_t buflen, int *flagsp) { + char bf[1 + _PW_NAME_LEN], *ypcurrent = NULL, *map; + int yp_pw_flags = 0, ypcurrentlen, r, s = -1, pw_keynum; + static long yppbuf[1024 / sizeof(long)]; + struct _ypexclude *ypexhead = NULL; + const char *host, *user, *dom; DBT key; - int len, rval; - char bf[_PW_NAME_LEN + 1]; - if (!_pw_db && !__initdb()) - return ((struct passwd *)NULL); + if (lookup == LOOKUP_BYNAME) { + map = PASSWD_BYNAME; + name = strdup(name); + } else { + map = PASSWD_BYUID; + asprintf(&name, "%u", uid); + } -#ifdef YP - /* - * If YP is active, we must sequence through the passwd file - * in sequence. - */ - if (__has_yppw()) { - int r; - int s = -1; - const char *host, *user, *dom; - - for (_pw_keynum=1; _pw_keynum; _pw_keynum++) { - bf[0] = _PW_KEYBYNUM; - bcopy((char *)&_pw_keynum, bf + 1, sizeof(_pw_keynum)); - key.data = (u_char *)bf; - key.size = sizeof(_pw_keynum) + 1; - if (__hashpw(&key) == 0) - break; - switch (_pw_passwd.pw_name[0]) { - case '+': - if (!__ypdomain) { - if (_yp_check(&__ypdomain) == 0) { - continue; - } - } - /* save the prototype */ - __ypproto_set(); + for (pw_keynum = 1; pw_keynum; pw_keynum++) { + bf[0] = _PW_KEYBYNUM; + bcopy((char *)&pw_keynum, &bf[1], sizeof(pw_keynum)); + key.data = (u_char *)bf; + key.size = 1 + sizeof(pw_keynum); + if (__hashpw(&key, buf, buflen, pw, flagsp) == 0) + break; + switch (pw->pw_name[0]) { + case '+': + if (!__ypdomain) { + if (_yp_check(&__ypdomain) == 0) + continue; + } + __ypproto_set(pw, yppbuf, *flagsp, &yp_pw_flags); - switch (_pw_passwd.pw_name[1]) { - case '\0': - if (__ypcurrent) { - free(__ypcurrent); - __ypcurrent = NULL; - } - r = yp_match(__ypdomain, - (PASSWD_BYNAME), - name, strlen(name), - &__ypcurrent, &__ypcurrentlen); - if (r != 0) { - if (__ypcurrent) - free(__ypcurrent); - __ypcurrent = NULL; - continue; - } - break; - case '@': -pwnam_netgrp: - if (__ypcurrent) { - free(__ypcurrent); - __ypcurrent = NULL; - } - if (s == -1) /* first time */ - setnetgrent(_pw_passwd.pw_name + 2); - s = getnetgrent(&host, &user, &dom); - if (s == 0) { /* end of group */ - endnetgrent(); - s = -1; - continue; - } else { - if (user && *user) { - r = yp_match(__ypdomain, - (PASSWD_BYNAME), - user, strlen(user), - &__ypcurrent, - &__ypcurrentlen); - } else - goto pwnam_netgrp; - if (r != 0) { - if (__ypcurrent) - free(__ypcurrent); - __ypcurrent = NULL; - /* - * just because this - * user is bad, doesn't - * mean they all are. - */ - goto pwnam_netgrp; - } - } - break; - default: - if (__ypcurrent) { - free(__ypcurrent); - __ypcurrent = NULL; - } - user = _pw_passwd.pw_name + 1; - r = yp_match(__ypdomain, - (PASSWD_BYNAME), - user, strlen(user), - &__ypcurrent, - &__ypcurrentlen); - if (r != 0) { - if (__ypcurrent) - free(__ypcurrent); - __ypcurrent = NULL; - continue; - } - break; + switch (pw->pw_name[1]) { + case '\0': + if (ypcurrent) { + free(ypcurrent); + ypcurrent = NULL; } - bcopy(__ypcurrent, __ypline, __ypcurrentlen); - __ypline[__ypcurrentlen] = '\0'; - if (__ypparse(&_pw_passwd, __ypline) - || __ypexclude_is(_pw_passwd.pw_name)) { - if (s == 1) /* inside netgrp */ - goto pwnam_netgrp; + r = yp_match(__ypdomain, map, + name, strlen(name), + &ypcurrent, &ypcurrentlen); + if (r != 0 || ypcurrentlen > buflen) { + if (ypcurrent) + free(ypcurrent); + ypcurrent = NULL; continue; } break; - case '-': - /* attempted exclusion */ - switch (_pw_passwd.pw_name[1]) { - case '\0': - break; - case '@': - setnetgrent(_pw_passwd.pw_name + 2); - while (getnetgrent(&host, &user, &dom)) { - if (user && *user) - __ypexclude_add(user); - } + case '@': +pwnam_netgrp: + if (ypcurrent) { + free(ypcurrent); + ypcurrent = NULL; + } + if (s == -1) /* first time */ + setnetgrent(pw->pw_name + 2); + s = getnetgrent(&host, &user, &dom); + if (s == 0) { /* end of group */ endnetgrent(); - break; - default: - __ypexclude_add(_pw_passwd.pw_name + 1); - break; + s = -1; + continue; + } else { + if (user && *user) { + r = yp_match(__ypdomain, map, + user, strlen(user), + &ypcurrent, &ypcurrentlen); + } else + goto pwnam_netgrp; + if (r != 0 || ypcurrentlen > buflen) { + if (ypcurrent) + free(ypcurrent); + ypcurrent = NULL; + /* + * just because this + * user is bad, doesn't + * mean they all are. + */ + goto pwnam_netgrp; + } } break; + default: + if (ypcurrent) { + free(ypcurrent); + ypcurrent = NULL; + } + user = pw->pw_name + 1; + r = yp_match(__ypdomain, map, + user, strlen(user), + &ypcurrent, &ypcurrentlen); + if (r != 0 || ypcurrentlen > buflen) { + if (ypcurrent) + free(ypcurrent); + ypcurrent = NULL; + continue; + } + break; + } + bcopy(ypcurrent, buf, ypcurrentlen); + buf[ypcurrentlen] = '\0'; + if (__ypparse(pw, buf, yp_pw_flags) || + __ypexclude_is(&ypexhead, pw->pw_name)) { + if (s == 1) /* inside netgrp */ + goto pwnam_netgrp; + continue; } - if (strcmp(_pw_passwd.pw_name, name) == 0) { - if (!_pw_stayopen) { - (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; + break; + case '-': + /* attempted exclusion */ + switch (pw->pw_name[1]) { + case '\0': + break; + case '@': + setnetgrent(pw->pw_name + 2); + while (getnetgrent(&host, &user, &dom)) { + if (user && *user) + __ypexclude_add(&ypexhead, user); } - __ypexclude_free(); - __ypproto = (struct passwd *)NULL; - return &_pw_passwd; + endnetgrent(); + break; + default: + __ypexclude_add(&ypexhead, pw->pw_name + 1); + break; } - if (s == 1) /* inside netgrp */ - goto pwnam_netgrp; - continue; - } - if (!_pw_stayopen) { - (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; + break; } - __ypexclude_free(); - __ypproto = (struct passwd *)NULL; - return (struct passwd *)NULL; + if ((lookup == LOOKUP_BYUID && pw->pw_uid == uid) || + (lookup == LOOKUP_BYNAME && strcmp(pw->pw_name, name) == 0)) + goto done; + if (s == 1) /* inside netgrp */ + goto pwnam_netgrp; + continue; } + pw = NULL; +done: + __ypexclude_free(&ypexhead); + __ypproto = NULL; + if (ypcurrent) + free(ypcurrent); + ypcurrent = NULL; + free(name); + return (pw); +} #endif /* YP */ - bf[0] = _PW_KEYBYNAME; +static struct passwd * +_pwhashbyname(const char *name, char *buf, size_t buflen, struct passwd *pw, + int *flagsp) +{ + char bf[1 + _PW_NAME_LEN]; + int len, r; + DBT key; + len = strlen(name); if (len > _PW_NAME_LEN) - rval = 0; - else { - bcopy(name, bf + 1, MIN(len, _PW_NAME_LEN)); - key.data = (u_char *)bf; - key.size = MIN(len, _PW_NAME_LEN) + 1; - rval = __hashpw(&key); - } + return (NULL); + bf[0] = _PW_KEYBYNAME; + bcopy(name, &bf[1], MIN(len, _PW_NAME_LEN)); + key.data = (u_char *)bf; + key.size = 1 + MIN(len, _PW_NAME_LEN); + r = __hashpw(&key, buf, buflen, pw, flagsp); + if (r) + return (pw); + return (NULL); +} + +static struct passwd * +_pwhashbyuid(uid_t uid, char *buf, size_t buflen, struct passwd *pw, + int *flagsp) +{ + char bf[1 + sizeof(int)]; + DBT key; + int r; + + bf[0] = _PW_KEYBYUID; + bcopy(&uid, &bf[1], sizeof(uid)); + key.data = (u_char *)bf; + key.size = 1 + sizeof(uid); + r = __hashpw(&key, buf, buflen, pw, flagsp); + if (r) + return (pw); + return (NULL); +} + +int +getpwnam_r(const char *name, struct passwd *pw, char *buf, size_t buflen, + struct passwd **pwretp) +{ + struct passwd *pwret = NULL; + int flags = 0, *flagsp; + DB *savedb; + + _THREAD_PRIVATE_MUTEX_LOCK(pw); + savedb = _pw_db; + if (!_pw_db && !__initdb()) + goto fail; + + if (pw == &_pw_passwd) + flagsp = &_pw_flags; + else + flagsp = &flags; - if (!_pw_stayopen) { +#ifdef YP + if (__has_yppw()) + pwret = __yppwlookup(LOOKUP_BYNAME, (char *)name, 0, pw, + buf, buflen, flagsp); +#endif /* YP */ + if (!pwret) + pwret = _pwhashbyname(name, buf, buflen, pw, flagsp); + + if (savedb != _pw_db || !_pw_stayopen) { (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; + _pw_db = NULL; } - return (rval ? &_pw_passwd : (struct passwd *)NULL); +fail: + if (pwretp) + *pwretp = pwret; + _THREAD_PRIVATE_MUTEX_UNLOCK(pw); + return (pwret ? 0 : 1); } struct passwd * -getpwuid(uid_t uid) +getpwnam(const char *name) { - DBT key; - char bf[sizeof(_pw_keynum) + 1]; - uid_t keyuid; - int rval; + struct passwd *pw = NULL; + if (getpwnam_r(name, &_pw_passwd, _pw_string, sizeof _pw_string, &pw)) + pw = NULL; + return (pw); +} + +int +getpwuid_r(uid_t uid, struct passwd *pw, char *buf, size_t buflen, + struct passwd **pwretp) +{ + struct passwd *pwret = NULL; + int flags = 0, *flagsp; + DB *savedb; + + _THREAD_PRIVATE_MUTEX_LOCK(pw); + savedb = _pw_db; if (!_pw_db && !__initdb()) - return ((struct passwd *)NULL); + goto fail; -#ifdef YP - /* - * If YP is active, we must sequence through the passwd file - * in sequence. - */ - if (__has_yppw()) { - char uidbuf[20]; - int r; - int s = -1; - const char *host, *user, *dom; - - snprintf(uidbuf, sizeof uidbuf, "%u", uid); - for (_pw_keynum=1; _pw_keynum; _pw_keynum++) { - bf[0] = _PW_KEYBYNUM; - bcopy((char *)&_pw_keynum, bf + 1, sizeof(_pw_keynum)); - key.data = (u_char *)bf; - key.size = sizeof(_pw_keynum) + 1; - if (__hashpw(&key) == 0) - break; - switch (_pw_passwd.pw_name[0]) { - case '+': - if (!__ypdomain) { - if (_yp_check(&__ypdomain) == 0) { - continue; - } - } - /* save the prototype */ - __ypproto_set(); + if (pw == &_pw_passwd) + flagsp = &_pw_flags; + else + flagsp = &flags; - switch (_pw_passwd.pw_name[1]) { - case '\0': - if (__ypcurrent) { - free(__ypcurrent); - __ypcurrent = NULL; - } - r = yp_match(__ypdomain, (PASSWD_BYUID), - uidbuf, strlen(uidbuf), - &__ypcurrent, &__ypcurrentlen); - if (r != 0) { - if (__ypcurrent) - free(__ypcurrent); - __ypcurrent = NULL; - continue; - } - break; - case '@': -pwuid_netgrp: - if (__ypcurrent) { - free(__ypcurrent); - __ypcurrent = NULL; - } - if (s == -1) /* first time */ - setnetgrent(_pw_passwd.pw_name + 2); - s = getnetgrent(&host, &user, &dom); - if (s == 0) { /* end of group */ - endnetgrent(); - s = -1; - continue; - } else { - if (user && *user) { - r = yp_match(__ypdomain, - (PASSWD_BYNAME), - user, strlen(user), - &__ypcurrent, - &__ypcurrentlen); - } else - goto pwuid_netgrp; - if (r != 0) { - if (__ypcurrent) - free(__ypcurrent); - __ypcurrent = NULL; - /* - * just because this - * user is bad, doesn't - * mean they all are. - */ - goto pwuid_netgrp; - } - } - break; - default: - if (__ypcurrent) { - free(__ypcurrent); - __ypcurrent = NULL; - } - user = _pw_passwd.pw_name + 1; - r = yp_match(__ypdomain, - (PASSWD_BYNAME), - user, strlen(user), - &__ypcurrent, - &__ypcurrentlen); - if (r != 0) { - if (__ypcurrent) - free(__ypcurrent); - __ypcurrent = NULL; - continue; - } - break; - } - bcopy(__ypcurrent, __ypline, __ypcurrentlen); - __ypline[__ypcurrentlen] = '\0'; - if (__ypparse(&_pw_passwd, __ypline) - || __ypexclude_is(_pw_passwd.pw_name)) { - if (s == 1) /* inside netgroup */ - goto pwuid_netgrp; - continue; - } - break; - case '-': - /* attempted exclusion */ - switch (_pw_passwd.pw_name[1]) { - case '\0': - break; - case '@': - setnetgrent(_pw_passwd.pw_name + 2); - while (getnetgrent(&host, &user, &dom)) { - if (user && *user) - __ypexclude_add(user); - } - endnetgrent(); - break; - default: - __ypexclude_add(_pw_passwd.pw_name + 1); - break; - } - break; - } - if (_pw_passwd.pw_uid == uid) { - if (!_pw_stayopen) { - (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; - } - __ypexclude_free(); - __ypproto = NULL; - return &_pw_passwd; - } - if (s == 1) /* inside netgroup */ - goto pwuid_netgrp; - continue; - } - if (!_pw_stayopen) { - (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; - } - __ypexclude_free(); - __ypproto = (struct passwd *)NULL; - return (struct passwd *)NULL; - } +#ifdef YP + if (__has_yppw()) + pwret = __yppwlookup(LOOKUP_BYUID, NULL, uid, pw, + buf, buflen, flagsp); #endif /* YP */ + if (!pwret) + pwret = _pwhashbyuid(uid, buf, buflen, pw, flagsp); - bf[0] = _PW_KEYBYUID; - keyuid = uid; - bcopy(&keyuid, bf + 1, sizeof(keyuid)); - key.data = (u_char *)bf; - key.size = sizeof(keyuid) + 1; - rval = __hashpw(&key); - - if (!_pw_stayopen) { + if (savedb != _pw_db || !_pw_stayopen) { (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; + _pw_db = NULL; } - return (rval ? &_pw_passwd : (struct passwd *)NULL); +fail: + if (pwretp) + *pwretp = pwret; + _THREAD_PRIVATE_MUTEX_UNLOCK(pw); + return (pwret ? 0 : 1); +} + +struct passwd * +getpwuid(uid_t uid) +{ + struct passwd *pw = NULL; + + if (getpwuid_r(uid, &_pw_passwd, _pw_string, sizeof _pw_string, &pw)) + pw = NULL; + return (pw); } int setpassent(int stayopen) { + _THREAD_PRIVATE_MUTEX_LOCK(pw); _pw_keynum = 0; _pw_stayopen = stayopen; #ifdef YP @@ -922,9 +867,10 @@ setpassent(int stayopen) if (__ypcurrent) free(__ypcurrent); __ypcurrent = NULL; - __ypexclude_free(); - __ypproto = (struct passwd *)NULL; + __ypexclude_free(&__ypexhead); + __ypproto = NULL; #endif + _THREAD_PRIVATE_MUTEX_UNLOCK(pw); return (1); } @@ -937,19 +883,21 @@ setpwent(void) void endpwent(void) { + _THREAD_PRIVATE_MUTEX_LOCK(pw); _pw_keynum = 0; if (_pw_db) { (void)(_pw_db->close)(_pw_db); - _pw_db = (DB *)NULL; + _pw_db = NULL; } #ifdef YP __ypmode = YPMODE_NONE; if (__ypcurrent) free(__ypcurrent); __ypcurrent = NULL; - __ypexclude_free(); - __ypproto = (struct passwd *)NULL; + __ypexclude_free(&__ypexhead); + __ypproto = NULL; #endif + _THREAD_PRIVATE_MUTEX_UNLOCK(pw); } static int @@ -971,54 +919,40 @@ __initdb(void) } static int -__hashpw(DBT *key) +__hashpw(DBT *key, char *buf, size_t buflen, struct passwd *pw, + int *flagsp) { char *p, *t; - static u_int max; - static char *line; DBT data; if ((_pw_db->get)(_pw_db, key, &data, 0)) return (0); p = (char *)data.data; - if (data.size > max) { - char *nline; - - max = data.size + 256; - nline = realloc(line, max); - if (nline == NULL) { - if (line) - free(line); - line = NULL; - max = 0; - return 0; - } - line = nline; - } + if (data.size > buflen) + return (0); - t = line; + t = buf; #define EXPAND(e) e = t; while ((*t++ = *p++)); - EXPAND(_pw_passwd.pw_name); - EXPAND(_pw_passwd.pw_passwd); - bcopy(p, (char *)&_pw_passwd.pw_uid, sizeof(int)); + EXPAND(pw->pw_name); + EXPAND(pw->pw_passwd); + bcopy(p, (char *)&pw->pw_uid, sizeof(int)); p += sizeof(int); - bcopy(p, (char *)&_pw_passwd.pw_gid, sizeof(int)); + bcopy(p, (char *)&pw->pw_gid, sizeof(int)); p += sizeof(int); - bcopy(p, (char *)&_pw_passwd.pw_change, sizeof(time_t)); + bcopy(p, (char *)&pw->pw_change, sizeof(time_t)); p += sizeof(time_t); - EXPAND(_pw_passwd.pw_class); - EXPAND(_pw_passwd.pw_gecos); - EXPAND(_pw_passwd.pw_dir); - EXPAND(_pw_passwd.pw_shell); - bcopy(p, (char *)&_pw_passwd.pw_expire, sizeof(time_t)); + EXPAND(pw->pw_class); + EXPAND(pw->pw_gecos); + EXPAND(pw->pw_dir); + EXPAND(pw->pw_shell); + bcopy(p, (char *)&pw->pw_expire, sizeof(time_t)); p += sizeof(time_t); /* See if there's any data left. If so, read in flags. */ if (data.size > (p - (char *)data.data)) { - bcopy(p, (char *)&_pw_flags, sizeof(int)); + bcopy(p, (char *)flagsp, sizeof(int)); p += sizeof(int); } else - _pw_flags = _PASSWORD_NOUID|_PASSWORD_NOGID; /* default */ - + *flagsp = _PASSWORD_NOUID|_PASSWORD_NOGID; /* default */ return (1); } |