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