/* * __getpwent_r.c - This file is part of the libc-8086/pwd package for ELKS, * Copyright (C) 1995, 1996 Nat Friedman <ndf@linux.mit.edu>. * Copyright (C) 2001 Erik Andersen <andersee@debian.org> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the Free * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * * March 7, 2001 -- Reworked to be reentrant by Erik Andersen */ #include <stdlib.h> #include <unistd.h> #include <string.h> #include <fcntl.h> #include "config.h" /* This isn't as flash as my previous version -- it doesn't dynamically scale down the gecos on too-long lines, but it also makes fewer syscalls, so it's probably nicer. Write me if you want the old version. Maybe I should include it as a build-time option... ? -Nat <ndf@linux.mit.edu> */ int __getpwent_r(struct passwd * passwd, char * line_buff, size_t buflen, int pwd_fd) { char *field_begin; char *endptr; char *gid_ptr=NULL; char *uid_ptr=NULL; int line_len; int i; /* We use the restart label to handle malformatted lines */ restart: /* Read the passwd line into the static buffer using a minimal of syscalls. */ if ((line_len = read(pwd_fd, line_buff, buflen)) <= 0) return -1; field_begin = strchr(line_buff, '\n'); if (field_begin != NULL) lseek(pwd_fd, (long) (1 + field_begin - (line_buff + line_len)), SEEK_CUR); else { /* The line is too long - skip it. :-\ */ do { if ((line_len = read(pwd_fd, line_buff, buflen)) <= 0) return -1; } while (!(field_begin = strchr(line_buff, '\n'))); lseek(pwd_fd, (long) (field_begin - line_buff) - line_len + 1, SEEK_CUR); goto restart; } if (*line_buff == '#' || *line_buff == ' ' || *line_buff == '\n' || *line_buff == '\t') goto restart; *field_begin = '\0'; /* We've read the line; now parse it. */ field_begin = line_buff; for (i = 0; i < 7; i++) { switch (i) { case 0: passwd->pw_name = field_begin; break; case 1: passwd->pw_passwd = field_begin; break; case 2: uid_ptr = field_begin; break; case 3: gid_ptr = field_begin; break; case 4: passwd->pw_gecos = field_begin; break; case 5: passwd->pw_dir = field_begin; break; case 6: passwd->pw_shell = field_begin; break; } if (i < 6) { field_begin = strchr(field_begin, ':'); if (field_begin == NULL) goto restart; *field_begin++ = '\0'; } } passwd->pw_gid = (gid_t) strtoul(gid_ptr, &endptr, 10); if (*endptr != '\0') goto restart; passwd->pw_uid = (uid_t) strtoul(uid_ptr, &endptr, 10); if (*endptr != '\0') goto restart; return 0; }