summaryrefslogtreecommitdiff
path: root/usr/src/lib/nsswitch/nis/common/getpwnam.c
diff options
context:
space:
mode:
Diffstat (limited to 'usr/src/lib/nsswitch/nis/common/getpwnam.c')
-rw-r--r--usr/src/lib/nsswitch/nis/common/getpwnam.c77
1 files changed, 73 insertions, 4 deletions
diff --git a/usr/src/lib/nsswitch/nis/common/getpwnam.c b/usr/src/lib/nsswitch/nis/common/getpwnam.c
index a23ee8af5c..ad55f9e5ca 100644
--- a/usr/src/lib/nsswitch/nis/common/getpwnam.c
+++ b/usr/src/lib/nsswitch/nis/common/getpwnam.c
@@ -19,14 +19,12 @@
* CDDL HEADER END
*/
/*
- * Copyright 2006 Sun Microsystems, Inc. All rights reserved.
+ * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*
* nis/getpwnam.c -- "nis" backend for nsswitch "passwd" database
*/
-#pragma ident "%Z%%M% %I% %E% SMI"
-
#include <pwd.h>
#include "nis_common.h"
@@ -49,10 +47,81 @@ getbyuid(be, a)
nss_XbyY_args_t *argp = (nss_XbyY_args_t *)a;
char uidstr[12]; /* More than enough */
- (void) snprintf(uidstr, 12, "%ld", argp->key.uid);
+ if (argp->key.uid > MAXUID)
+ return (NSS_NOTFOUND);
+ (void) snprintf(uidstr, 12, "%u", argp->key.uid);
return (_nss_nis_lookup(be, argp, 0, "passwd.byuid", uidstr, 0));
}
+/*
+ * Validates passwd entry replacing uid/gid > MAXUID by ID_NOBODY.
+ */
+int
+validate_passwd_ids(char **linepp, int *linelenp, int allocbuf)
+{
+ char *linep, *limit, *uidp, *gidp, *newline;
+ uid_t uid;
+ gid_t gid;
+ ulong_t uidl, gidl;
+ int olduidlen, oldgidlen, idlen;
+ int linelen = *linelenp, newlinelen;
+
+ linep = *linepp;
+ limit = linep + linelen;
+
+ /* +/- entries valid for compat source only */
+ if (linelen == 0 || *linep == '+' || *linep == '-')
+ return (NSS_STR_PARSE_SUCCESS);
+
+ while (linep < limit && *linep++ != ':') /* skip username */
+ continue;
+ while (linep < limit && *linep++ != ':') /* skip password */
+ continue;
+ if (linep == limit)
+ return (NSS_STR_PARSE_PARSE);
+
+ uidp = linep;
+ uidl = strtoul(uidp, (char **)&linep, 10); /* grab uid */
+ olduidlen = linep - uidp;
+ if (++linep >= limit || olduidlen == 0)
+ return (NSS_STR_PARSE_PARSE);
+
+ gidp = linep;
+ gidl = strtoul(gidp, (char **)&linep, 10); /* grab gid */
+ oldgidlen = linep - gidp;
+ if (linep >= limit || oldgidlen == 0)
+ return (NSS_STR_PARSE_PARSE);
+
+ if (uidl <= MAXUID && gidl <= MAXUID)
+ return (NSS_STR_PARSE_SUCCESS);
+ uid = (uidl > MAXUID) ? UID_NOBODY : (uid_t)uidl;
+ gid = (gidl > MAXUID) ? GID_NOBODY : (gid_t)gidl;
+
+ /* Check if we have enough space in the buffer */
+ idlen = snprintf(NULL, 0, "%u:%u", uid, gid);
+ newlinelen = linelen + idlen - olduidlen - oldgidlen - 1;
+ if (newlinelen > linelen) {
+ /* need a larger buffer */
+ if (!allocbuf || (newline = malloc(newlinelen + 1)) == NULL)
+ return (NSS_STR_PARSE_ERANGE);
+ /* Replace ephemeral ids by ID_NOBODY in the new buffer */
+ *(uidp - 1) = '\0';
+ (void) snprintf(newline, newlinelen + 1, "%s:%u:%u%s",
+ *linepp, uid, gid, linep);
+ free(*linepp);
+ *linepp = newline;
+ *linelenp = newlinelen;
+ return (NSS_STR_PARSE_SUCCESS);
+ }
+
+ /* Replace ephemeral ids by ID_NOBODY in the same buffer */
+ (void) bcopy(linep, uidp + idlen, limit - linep + 1);
+ (void) snprintf(uidp, idlen + 1, "%u:%u", uid, gid);
+ *(uidp + idlen) = ':'; /* restore : that was overwritten by snprintf */
+ *linelenp = newlinelen;
+ return (NSS_STR_PARSE_SUCCESS);
+}
+
static nis_backend_op_t passwd_ops[] = {
_nss_nis_destr,
_nss_nis_endent,