struct passwd pwbuf;
int old_errno = errno;
- int buflen = sysconf(_SC_GETPW_R_SIZE_MAX);
+ size_t buflen = sysconf(_SC_GETPW_R_SIZE_MAX);
char *buf = malloc(buflen);
if (buf == NULL) {
*errnop = ENOMEM;
nip = startp;
fct.ptr = fct_start;
do {
+ morebuf:
if (fct.l == _nss_nonlocal_getpwuid_r)
status = NSS_STATUS_NOTFOUND;
else
status = DL_CALL_FCT(fct.l, (uid, &pwbuf, buf, buflen, errnop));
- if (status == NSS_STATUS_TRYAGAIN && *errnop == ERANGE)
- break;
+ if (status == NSS_STATUS_TRYAGAIN && *errnop == ERANGE) {
+ free(buf);
+ buflen *= 2;
+ buf = malloc(buflen);
+ if (buf == NULL) {
+ *errnop = ENOMEM;
+ errno = old_errno;
+ return NSS_STATUS_TRYAGAIN;
+ }
+ goto morebuf;
+ }
} while (__nss_next(&nip, fct_name, &fct.ptr, status, 0) == 0);
if (status == NSS_STATUS_SUCCESS) {
struct passwd pwbuf;
int old_errno = errno;
- int buflen = sysconf(_SC_GETPW_R_SIZE_MAX);
+ size_t buflen = sysconf(_SC_GETPW_R_SIZE_MAX);
char *buf = malloc(buflen);
if (buf == NULL) {
*errnop = ENOMEM;
nip = startp;
fct.ptr = fct_start;
do {
+ morebuf:
if (fct.l == _nss_nonlocal_getpwnam_r)
status = NSS_STATUS_NOTFOUND;
else
status = DL_CALL_FCT(fct.l, (user, &pwbuf, buf, buflen, errnop));
- if (status == NSS_STATUS_TRYAGAIN && *errnop == ERANGE)
- break;
+ if (status == NSS_STATUS_TRYAGAIN && *errnop == ERANGE) {
+ free(buf);
+ buflen *= 2;
+ buf = malloc(buflen);
+ if (buf == NULL) {
+ *errnop = ENOMEM;
+ errno = old_errno;
+ return NSS_STATUS_TRYAGAIN;
+ }
+ goto morebuf;
+ }
} while (__nss_next(&nip, fct_name, &fct.ptr, status, 0) == 0);
if (status == NSS_STATUS_SUCCESS)
if (status != NSS_STATUS_SUCCESS)
return status;
+ if (strcmp(name, pwd->pw_name) != 0) {
+ syslog(LOG_ERR, "nss_nonlocal: discarding user %s from lookup for user %s\n", pwd->pw_name, name);
+ return NSS_STATUS_NOTFOUND;
+ }
+
status = check_nonlocal_uid(name, pwd->pw_uid, errnop);
if (status != NSS_STATUS_SUCCESS)
return status;