2011-02-12 00:22:29 -05:00
|
|
|
#include "pwf.h"
|
2012-02-01 23:51:19 -05:00
|
|
|
#include <pthread.h>
|
2011-02-12 00:22:29 -05:00
|
|
|
|
|
|
|
#define FIX(x) (pw->pw_##x = pw->pw_##x-line+buf)
|
|
|
|
|
|
|
|
static int getpw_r(const char *name, uid_t uid, struct passwd *pw, char *buf, size_t size, struct passwd **res)
|
|
|
|
{
|
|
|
|
FILE *f;
|
|
|
|
char *line = 0;
|
|
|
|
size_t len = 0;
|
|
|
|
int rv = 0;
|
2012-02-01 23:51:19 -05:00
|
|
|
int cs;
|
|
|
|
|
|
|
|
pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);
|
2011-02-12 00:22:29 -05:00
|
|
|
|
2012-09-29 18:14:46 -04:00
|
|
|
f = fopen("/etc/passwd", "rbe");
|
2012-02-01 23:51:19 -05:00
|
|
|
if (!f) {
|
|
|
|
rv = errno;
|
|
|
|
goto done;
|
|
|
|
}
|
2011-02-12 00:22:29 -05:00
|
|
|
|
|
|
|
*res = 0;
|
|
|
|
while (__getpwent_a(f, pw, &line, &len)) {
|
|
|
|
if (name && !strcmp(name, pw->pw_name)
|
|
|
|
|| !name && pw->pw_uid == uid) {
|
|
|
|
if (size < len) {
|
|
|
|
rv = ERANGE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
*res = pw;
|
|
|
|
memcpy(buf, line, len);
|
|
|
|
FIX(name);
|
|
|
|
FIX(passwd);
|
|
|
|
FIX(gecos);
|
|
|
|
FIX(dir);
|
|
|
|
FIX(shell);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
fclose(f);
|
2012-02-01 23:51:19 -05:00
|
|
|
done:
|
|
|
|
pthread_setcancelstate(cs, 0);
|
2011-02-12 00:22:29 -05:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
int getpwnam_r(const char *name, struct passwd *pw, char *buf, size_t size, struct passwd **res)
|
|
|
|
{
|
|
|
|
return getpw_r(name, 0, pw, buf, size, res);
|
|
|
|
}
|
|
|
|
|
|
|
|
int getpwuid_r(uid_t uid, struct passwd *pw, char *buf, size_t size, struct passwd **res)
|
|
|
|
{
|
|
|
|
return getpw_r(0, uid, pw, buf, size, res);
|
|
|
|
}
|