1 /* passwd.cc: getpwnam () and friends
3 Copyright 1996, 1997, 1998, 2001, 2002, 2003, 2007, 2008, 2009 Red Hat, Inc.
5 This file is part of Cygwin.
7 This software is a copyrighted work licensed under the terms of the
8 Cygwin license. Please consult the file "CYGWIN_LICENSE" for
22 #include "shared_info.h"
24 /* Read /etc/passwd only once for better performance. This is done
25 on the first call that needs information from it. */
28 static pwdgrp pr (passwd_buf);
30 /* Parse /etc/passwd line into passwd structure. */
32 pwdgrp::parse_passwd ()
34 # define res (*passwd_buf)[curr_lines]
35 res.pw_name = next_str (':');
36 res.pw_passwd = next_str (':');
37 if (!next_num (res.pw_uid))
39 if (!next_num (res.pw_gid))
41 res.pw_comment = NULL;
42 res.pw_gecos = next_str (':');
43 res.pw_dir = next_str (':');
44 res.pw_shell = next_str (':');
49 /* Read in /etc/passwd and save contents in the password cache.
50 This sets pr to loaded or emulated so functions in this file can
51 tell that /etc/passwd has been read in or will be emulated. */
53 pwdgrp::read_passwd ()
55 load (L"\\etc\\passwd");
57 char strbuf[128] = "";
58 bool searchentry = true;
61 static char NO_COPY pretty_ls[] = "????????:*:-1:-1:";
64 cygsid tu = cygheap->user.sid ();
66 if (!user_shared->cb || myself->uid == ILLEGAL_UID)
67 searchentry = !internal_getpwsid (tu);
69 && (!(pw = internal_getpwnam (cygheap->user.name ()))
71 || (myself->uid != ILLEGAL_UID
72 && myself->uid != (__uid32_t) pw->pw_uid
73 && !internal_getpwuid (myself->uid))))
75 static char linebuf[1024]; // must be static and
76 // should not be NO_COPY
77 snprintf (linebuf, sizeof (linebuf), "%s:*:%lu:%lu:,%s:%s:/bin/sh",
78 cygheap->user.name (),
79 (!user_shared->cb || myself->uid == ILLEGAL_UID)
80 ? UNKNOWN_UID : myself->uid,
81 !user_shared->cb ? UNKNOWN_GID : myself->gid,
82 strbuf, getenv ("HOME") ?: "");
83 debug_printf ("Completing /etc/passwd: %s", linebuf);
89 internal_getpwsid (cygpsid &sid)
92 char *ptr1, *ptr2, *endptr;
93 char sid_string[128] = {0,','};
97 if (sid.string (sid_string + 2))
99 endptr = strchr (sid_string + 2, 0) - 1;
100 for (int i = 0; i < pr.curr_lines; i++)
101 if ((pw = passwd_buf + i)->pw_dir > pw->pw_gecos + 8)
102 for (ptr1 = endptr, ptr2 = pw->pw_dir - 2;
103 *ptr1 == *ptr2; ptr2--)
111 internal_getpwuid (__uid32_t uid, bool check)
115 for (int i = 0; i < pr.curr_lines; i++)
116 if (uid == (__uid32_t) passwd_buf[i].pw_uid)
117 return passwd_buf + i;
122 internal_getpwnam (const char *name, bool check)
126 for (int i = 0; i < pr.curr_lines; i++)
127 /* on Windows NT user names are case-insensitive */
128 if (strcasematch (name, passwd_buf[i].pw_name))
129 return passwd_buf + i;
134 extern "C" struct passwd *
135 getpwuid32 (__uid32_t uid)
137 struct passwd *temppw = internal_getpwuid (uid, true);
138 pthread_testcancel ();
142 extern "C" struct passwd *
143 getpwuid (__uid16_t uid)
145 return getpwuid32 (uid16touid32 (uid));
149 getpwuid_r32 (__uid32_t uid, struct passwd *pwd, char *buffer, size_t bufsize, struct passwd **result)
156 struct passwd *temppw = internal_getpwuid (uid, true);
157 pthread_testcancel ();
161 /* check needed buffer size. */
162 size_t needsize = strlen (temppw->pw_name) + strlen (temppw->pw_passwd)
163 + strlen (temppw->pw_gecos) + strlen (temppw->pw_dir)
164 + strlen (temppw->pw_shell) + 5;
165 if (needsize > bufsize)
168 /* make a copy of temppw */
170 pwd->pw_uid = temppw->pw_uid;
171 pwd->pw_gid = temppw->pw_gid;
172 buffer = stpcpy (pwd->pw_name = buffer, temppw->pw_name);
173 buffer = stpcpy (pwd->pw_passwd = buffer + 1, temppw->pw_passwd);
174 buffer = stpcpy (pwd->pw_gecos = buffer + 1, temppw->pw_gecos);
175 buffer = stpcpy (pwd->pw_dir = buffer + 1, temppw->pw_dir);
176 stpcpy (pwd->pw_shell = buffer + 1, temppw->pw_shell);
177 pwd->pw_comment = NULL;
182 getpwuid_r (__uid16_t uid, struct passwd *pwd, char *buffer, size_t bufsize, struct passwd **result)
184 return getpwuid_r32 (uid16touid32 (uid), pwd, buffer, bufsize, result);
187 extern "C" struct passwd *
188 getpwnam (const char *name)
190 struct passwd *temppw = internal_getpwnam (name, true);
191 pthread_testcancel ();
196 /* the max size buffer we can expect to
197 * use is returned via sysconf with _SC_GETPW_R_SIZE_MAX.
198 * This may need updating! - Rob Collins April 2001.
201 getpwnam_r (const char *nam, struct passwd *pwd, char *buffer, size_t bufsize, struct passwd **result)
205 if (!pwd || !buffer || !nam)
208 struct passwd *temppw = internal_getpwnam (nam, true);
209 pthread_testcancel ();
214 /* check needed buffer size. */
215 size_t needsize = strlen (temppw->pw_name) + strlen (temppw->pw_passwd)
216 + strlen (temppw->pw_gecos) + strlen (temppw->pw_dir)
217 + strlen (temppw->pw_shell) + 5;
218 if (needsize > bufsize)
221 /* make a copy of temppw */
223 pwd->pw_uid = temppw->pw_uid;
224 pwd->pw_gid = temppw->pw_gid;
225 buffer = stpcpy (pwd->pw_name = buffer, temppw->pw_name);
226 buffer = stpcpy (pwd->pw_passwd = buffer + 1, temppw->pw_passwd);
227 buffer = stpcpy (pwd->pw_gecos = buffer + 1, temppw->pw_gecos);
228 buffer = stpcpy (pwd->pw_dir = buffer + 1, temppw->pw_dir);
229 stpcpy (pwd->pw_shell = buffer + 1, temppw->pw_shell);
230 pwd->pw_comment = NULL;
234 extern "C" struct passwd *
237 if (_my_tls.locals.pw_pos == 0)
239 if (_my_tls.locals.pw_pos < pr.curr_lines)
240 return passwd_buf + _my_tls.locals.pw_pos++;
245 extern "C" struct passwd *
246 getpwduid (__uid16_t)
254 _my_tls.locals.pw_pos = 0;
260 _my_tls.locals.pw_pos = 0;
270 getpass (const char * prompt)
272 char *pass = _my_tls.locals.pass;
273 struct termios ti, newti;
275 cygheap_fdget fhstdin (0);
281 fhstdin->tcgetattr (&ti);
283 newti.c_lflag &= ~ECHO;
284 fhstdin->tcsetattr (TCSANOW, &newti);
285 fputs (prompt, stderr);
286 fgets (pass, _PASSWORD_LEN, stdin);
287 fprintf (stderr, "\n");
288 for (int i=0; pass[i]; i++)
289 if (pass[i] == '\r' || pass[i] == '\n')
291 fhstdin->tcsetattr (TCSANOW, &ti);