xref: /minix3/external/bsd/top/dist/username.c (revision b89261ba018da33f0bd8cd05f5a1fe9e7a9c837b)
1*b89261baSDavid van Moolenbroek /*
2*b89261baSDavid van Moolenbroek  * Copyright (c) 1984 through 2008, William LeFebvre
3*b89261baSDavid van Moolenbroek  * All rights reserved.
4*b89261baSDavid van Moolenbroek  *
5*b89261baSDavid van Moolenbroek  * Redistribution and use in source and binary forms, with or without
6*b89261baSDavid van Moolenbroek  * modification, are permitted provided that the following conditions are met:
7*b89261baSDavid van Moolenbroek  *
8*b89261baSDavid van Moolenbroek  *     * Redistributions of source code must retain the above copyright
9*b89261baSDavid van Moolenbroek  * notice, this list of conditions and the following disclaimer.
10*b89261baSDavid van Moolenbroek  *
11*b89261baSDavid van Moolenbroek  *     * Redistributions in binary form must reproduce the above
12*b89261baSDavid van Moolenbroek  * copyright notice, this list of conditions and the following disclaimer
13*b89261baSDavid van Moolenbroek  * in the documentation and/or other materials provided with the
14*b89261baSDavid van Moolenbroek  * distribution.
15*b89261baSDavid van Moolenbroek  *
16*b89261baSDavid van Moolenbroek  *     * Neither the name of William LeFebvre nor the names of other
17*b89261baSDavid van Moolenbroek  * contributors may be used to endorse or promote products derived from
18*b89261baSDavid van Moolenbroek  * this software without specific prior written permission.
19*b89261baSDavid van Moolenbroek  *
20*b89261baSDavid van Moolenbroek  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21*b89261baSDavid van Moolenbroek  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22*b89261baSDavid van Moolenbroek  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23*b89261baSDavid van Moolenbroek  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24*b89261baSDavid van Moolenbroek  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25*b89261baSDavid van Moolenbroek  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26*b89261baSDavid van Moolenbroek  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27*b89261baSDavid van Moolenbroek  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28*b89261baSDavid van Moolenbroek  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29*b89261baSDavid van Moolenbroek  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30*b89261baSDavid van Moolenbroek  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31*b89261baSDavid van Moolenbroek  */
32*b89261baSDavid van Moolenbroek 
33*b89261baSDavid van Moolenbroek /*
34*b89261baSDavid van Moolenbroek  *  Top users/processes display for Unix
35*b89261baSDavid van Moolenbroek  *  Version 3
36*b89261baSDavid van Moolenbroek  */
37*b89261baSDavid van Moolenbroek 
38*b89261baSDavid van Moolenbroek /*
39*b89261baSDavid van Moolenbroek  *  Username translation code for top.
40*b89261baSDavid van Moolenbroek  *
41*b89261baSDavid van Moolenbroek  *  These routines handle uid to username mapping.  They use a hash table to
42*b89261baSDavid van Moolenbroek  *  reduce reading overhead.  Entries are refreshed every EXPIRETIME seconds.
43*b89261baSDavid van Moolenbroek  *
44*b89261baSDavid van Moolenbroek  *  The old ad-hoc hash functions have been replaced with something a little
45*b89261baSDavid van Moolenbroek  *  more formal and (hopefully) more robust (found in hash.c)
46*b89261baSDavid van Moolenbroek  */
47*b89261baSDavid van Moolenbroek 
48*b89261baSDavid van Moolenbroek #include "os.h"
49*b89261baSDavid van Moolenbroek 
50*b89261baSDavid van Moolenbroek #include <pwd.h>
51*b89261baSDavid van Moolenbroek 
52*b89261baSDavid van Moolenbroek #include "top.h"
53*b89261baSDavid van Moolenbroek #include "utils.h"
54*b89261baSDavid van Moolenbroek #include "hash.h"
55*b89261baSDavid van Moolenbroek #include "username.h"
56*b89261baSDavid van Moolenbroek 
57*b89261baSDavid van Moolenbroek #define EXPIRETIME (60 * 5)
58*b89261baSDavid van Moolenbroek 
59*b89261baSDavid van Moolenbroek /* we need some sort of idea how long usernames can be */
60*b89261baSDavid van Moolenbroek #ifndef MAXLOGNAME
61*b89261baSDavid van Moolenbroek #ifdef _POSIX_LOGIN_NAME_MAX
62*b89261baSDavid van Moolenbroek #define MAXLOGNAME _POSIX_LOGIN_NAME_MAX
63*b89261baSDavid van Moolenbroek #else
64*b89261baSDavid van Moolenbroek #define MAXLOGNAME 9
65*b89261baSDavid van Moolenbroek #endif
66*b89261baSDavid van Moolenbroek #endif
67*b89261baSDavid van Moolenbroek 
68*b89261baSDavid van Moolenbroek struct hash_data {
69*b89261baSDavid van Moolenbroek     int    uid;
70*b89261baSDavid van Moolenbroek     char   name[MAXLOGNAME];  /* big enough? */
71*b89261baSDavid van Moolenbroek     time_t expire;
72*b89261baSDavid van Moolenbroek };
73*b89261baSDavid van Moolenbroek 
74*b89261baSDavid van Moolenbroek hash_table *userhash;
75*b89261baSDavid van Moolenbroek 
76*b89261baSDavid van Moolenbroek 
77*b89261baSDavid van Moolenbroek void
init_username(void)78*b89261baSDavid van Moolenbroek init_username(void)
79*b89261baSDavid van Moolenbroek 
80*b89261baSDavid van Moolenbroek {
81*b89261baSDavid van Moolenbroek     userhash = hash_create(211);
82*b89261baSDavid van Moolenbroek }
83*b89261baSDavid van Moolenbroek 
84*b89261baSDavid van Moolenbroek char *
username(int xuid)85*b89261baSDavid van Moolenbroek username(int xuid)
86*b89261baSDavid van Moolenbroek 
87*b89261baSDavid van Moolenbroek {
88*b89261baSDavid van Moolenbroek     struct hash_data *data;
89*b89261baSDavid van Moolenbroek     struct passwd *pw;
90*b89261baSDavid van Moolenbroek     time_t now;
91*b89261baSDavid van Moolenbroek 
92*b89261baSDavid van Moolenbroek     /* what time is it? */
93*b89261baSDavid van Moolenbroek     now = time(NULL);
94*b89261baSDavid van Moolenbroek 
95*b89261baSDavid van Moolenbroek     /* get whatever is in the cache */
96*b89261baSDavid van Moolenbroek     data = hash_lookup_uint(userhash, (unsigned int)xuid);
97*b89261baSDavid van Moolenbroek 
98*b89261baSDavid van Moolenbroek     /* if we had a cache miss, then create space for a new entry */
99*b89261baSDavid van Moolenbroek     if (data == NULL)
100*b89261baSDavid van Moolenbroek     {
101*b89261baSDavid van Moolenbroek 	/* make space */
102*b89261baSDavid van Moolenbroek 	data = emalloc(sizeof(struct hash_data));
103*b89261baSDavid van Moolenbroek 
104*b89261baSDavid van Moolenbroek 	/* fill in some data, including an already expired time */
105*b89261baSDavid van Moolenbroek 	data->uid = xuid;
106*b89261baSDavid van Moolenbroek 	data->expire = (time_t)0;
107*b89261baSDavid van Moolenbroek 
108*b89261baSDavid van Moolenbroek 	/* add it to the hash: the rest gets filled in later */
109*b89261baSDavid van Moolenbroek 	hash_add_uint(userhash, xuid, data);
110*b89261baSDavid van Moolenbroek     }
111*b89261baSDavid van Moolenbroek 
112*b89261baSDavid van Moolenbroek     /* Now data points to the correct hash entry for "xuid".  If this is
113*b89261baSDavid van Moolenbroek        a new entry, then expire is 0 and the next test will be true. */
114*b89261baSDavid van Moolenbroek     if (data->expire <= now)
115*b89261baSDavid van Moolenbroek     {
116*b89261baSDavid van Moolenbroek 	if ((pw = getpwuid(xuid)) != NULL)
117*b89261baSDavid van Moolenbroek 	{
118*b89261baSDavid van Moolenbroek 	    strncpy(data->name, pw->pw_name, MAXLOGNAME-1);
119*b89261baSDavid van Moolenbroek 	    data->expire = now + EXPIRETIME;
120*b89261baSDavid van Moolenbroek 	    dprintf("username: updating %d with %s, expires %d\n",
121*b89261baSDavid van Moolenbroek 		    data->uid, data->name, data->expire);
122*b89261baSDavid van Moolenbroek 	}
123*b89261baSDavid van Moolenbroek 	else
124*b89261baSDavid van Moolenbroek 	{
125*b89261baSDavid van Moolenbroek 	    /* username doesnt exist ... so invent one */
126*b89261baSDavid van Moolenbroek 	    snprintf(data->name, sizeof(data->name), "%d", xuid);
127*b89261baSDavid van Moolenbroek 	    data->expire = now + EXPIRETIME;
128*b89261baSDavid van Moolenbroek 	    dprintf("username: updating %d with %s, expires %d\n",
129*b89261baSDavid van Moolenbroek 		    data->uid, data->name, data->expire);
130*b89261baSDavid van Moolenbroek 	}
131*b89261baSDavid van Moolenbroek     }
132*b89261baSDavid van Moolenbroek 
133*b89261baSDavid van Moolenbroek     /* return what we have */
134*b89261baSDavid van Moolenbroek     return data->name;
135*b89261baSDavid van Moolenbroek }
136*b89261baSDavid van Moolenbroek 
137*b89261baSDavid van Moolenbroek int
userid(char * xusername)138*b89261baSDavid van Moolenbroek userid(char *xusername)
139*b89261baSDavid van Moolenbroek 
140*b89261baSDavid van Moolenbroek {
141*b89261baSDavid van Moolenbroek     struct passwd *pwd;
142*b89261baSDavid van Moolenbroek 
143*b89261baSDavid van Moolenbroek     if ((pwd = getpwnam(xusername)) == NULL)
144*b89261baSDavid van Moolenbroek     {
145*b89261baSDavid van Moolenbroek 	return(-1);
146*b89261baSDavid van Moolenbroek     }
147*b89261baSDavid van Moolenbroek 
148*b89261baSDavid van Moolenbroek     /* return our result */
149*b89261baSDavid van Moolenbroek     return(pwd->pw_uid);
150*b89261baSDavid van Moolenbroek }
151*b89261baSDavid van Moolenbroek 
152