/* The type ps_user_t, for per-user info. Copyright (C) 1995 Free Software Foundation, Inc. Written by Miles Bader <miles@gnu.ai.mit.edu> This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include <hurd.h> #include <stdio.h> #include <stdlib.h> #include <assert.h> #include "ps.h" #include "common.h" /* ---------------------------------------------------------------- */ /* Create a ps_user_t for the user referred to by UID, returning it in U. If a memory allocation error occurs, ENOMEM is returned, otherwise 0. */ error_t ps_user_create(uid_t uid, ps_user_t *u) { *u = NEW(struct ps_user); if (*u == NULL) return ENOMEM; (*u)->uid = uid; (*u)->passwd_state = PS_USER_PASSWD_PENDING; return 0; } /* Free U and any resources it consumes. */ void ps_user_free(ps_user_t u) { if (u->passwd_state == PS_USER_PASSWD_OK) free(u->storage); free(u); } /* ---------------------------------------------------------------- */ /* Returns the password file entry (struct passwd, from <pwd.h>) for the user referred to by U, or NULL if it can't be gotten. */ struct passwd *ps_user_passwd(ps_user_t u) { if (u->passwd_state == PS_USER_PASSWD_OK) return &u->passwd; else if (u->passwd_state == PS_USER_PASSWD_ERROR) return NULL; else { struct passwd *pw = getpwuid(u->uid); if (pw != NULL) { int needed = 0; #define COUNT(field) if (pw->field != NULL) (needed += strlen(pw->field) + 1) COUNT(pw_name); COUNT(pw_passwd); COUNT(pw_gecos); COUNT(pw_dir); COUNT(pw_shell); u->storage = malloc(needed); if (u->storage != NULL) { char *p = u->storage; /* Copy each string field into storage allocated in the u structure and point the fields at that instead of the static storage that pw currently points to. */ #define COPY(field) \ if (pw->field != NULL) \ strcpy(p, pw->field), (pw->field = p), (p += strlen(p) + 1) COPY(pw_name); COPY(pw_passwd); COPY(pw_gecos); COPY(pw_dir); COPY(pw_shell); u->passwd = *pw; u->passwd_state = PS_USER_PASSWD_OK; return &u->passwd; } } } u->passwd_state = PS_USER_PASSWD_ERROR; return NULL; } /* Returns the user name for the user referred to by U, or NULL if it can't be gotten. */ char *ps_user_name(ps_user_t u) { struct passwd *pw = ps_user_passwd(u); if (pw) return pw->pw_name; else return NULL; }