1 /* $OpenBSD: cache.c,v 1.23 2016/08/26 04:08:18 guenther Exp $ */
2 /* $NetBSD: cache.c,v 1.4 1995/03/21 09:07:10 cgd Exp $ */
5 * Copyright (c) 1992 Keith Muller.
6 * Copyright (c) 1992, 1993
7 * The Regents of the University of California. All rights reserved.
9 * This code is derived from software contributed to Berkeley by
10 * Keith Muller of the University of California, San Diego.
12 * Redistribution and use in source and binary forms, with or without
13 * modification, are permitted provided that the following conditions
15 * 1. Redistributions of source code must retain the above copyright
16 * notice, this list of conditions and the following disclaimer.
17 * 2. Redistributions in binary form must reproduce the above copyright
18 * notice, this list of conditions and the following disclaimer in the
19 * documentation and/or other materials provided with the distribution.
20 * 3. Neither the name of the University nor the names of its contributors
21 * may be used to endorse or promote products derived from this software
22 * without specific prior written permission.
24 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
25 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
26 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 #include <sys/types.h>
49 * Constants and data structures used to implement group and password file
50 * caches. Traditional passwd/group cache routines perform quite poorly with
51 * archives. The chances of hitting a valid lookup with an archive is quite a
52 * bit worse than with files already resident on the file system. These misses
53 * create a MAJOR performance cost. To address this problem, these routines
54 * cache both hits and misses.
56 * NOTE: name lengths must be as large as those stored in ANY PROTOCOL and
57 * as stored in the passwd and group files. CACHE SIZES MUST BE PRIME
59 #define UNMLEN 32 /* >= user name found in any protocol */
60 #define GNMLEN 32 /* >= group name found in any protocol */
61 #define UNM_SZ 317 /* size of uid_name() cache */
62 #define GNM_SZ 317 /* size of gid_name() cache */
63 #define VALID 1 /* entry and name are valid */
64 #define INVALID 2 /* entry valid, name NOT valid */
67 * Node structures used in the user, group, uid, and gid caches.
71 int valid
; /* is this a valid or a miss entry */
72 char name
[UNMLEN
]; /* uid name */
73 uid_t uid
; /* cached uid */
77 int valid
; /* is this a valid or a miss entry */
78 char name
[GNMLEN
]; /* gid name */
79 gid_t gid
; /* cached gid */
84 * routines that control user, group, uid and gid caches (for the archive
85 * member print routine).
87 * these routines cache BOTH hits and misses, a major performance improvement
90 static int pwopn
= 0; /* is password file open */
91 static int gropn
= 0; /* is group file open */
92 static UIDC
**usrtb
= NULL
; /* user name to uid cache */
93 static GIDC
**grptb
= NULL
; /* group name to gid cache */
97 * creates an empty usrtb
99 * 0 if ok, -1 otherwise
111 if ((usrtb
= calloc(UNM_SZ
, sizeof(UIDC
*))) == NULL
) {
113 paxwarn(1, "Unable to allocate memory for user name cache table");
121 * creates an empty grptb
123 * 0 if ok, -1 otherwise
135 if ((grptb
= calloc(GNM_SZ
, sizeof(GIDC
*))) == NULL
) {
137 paxwarn(1,"Unable to allocate memory for group name cache table");
145 * caches the uid for a given user name. We use a simple hash table.
147 * the uid (if any) for a user name, or a -1 if no match can be found
151 uid_name(char *name
, uid_t
*uid
)
158 * return -1 for mangled names
160 if (((namelen
= strlen(name
)) == 0) || (name
[0] == '\0'))
162 if ((usrtb
== NULL
) && (usrtb_start() < 0))
166 * look up in hash table, if found and valid return the uid,
167 * if found and invalid, return a -1
169 ptr
= usrtb
[st_hash(name
, namelen
, UNM_SZ
)];
170 if ((ptr
!= NULL
) && (ptr
->valid
> 0) && !strcmp(name
, ptr
->name
)) {
171 if (ptr
->valid
== INVALID
)
183 ptr
= usrtb
[st_hash(name
, namelen
, UNM_SZ
)] =
184 malloc(sizeof(UIDC
));
187 * no match, look it up, if no match store it as an invalid entry,
188 * or store the matching uid
191 if ((pw
= getpwnam(name
)) == NULL
)
196 (void)strlcpy(ptr
->name
, name
, sizeof(ptr
->name
));
197 if ((pw
= getpwnam(name
)) == NULL
) {
198 ptr
->valid
= INVALID
;
202 *uid
= ptr
->uid
= pw
->pw_uid
;
208 * caches the gid for a given group name. We use a simple hash table.
210 * the gid (if any) for a group name, or a -1 if no match can be found
214 gid_name(char *name
, gid_t
*gid
)
221 * return -1 for mangled names
223 if (((namelen
= strlen(name
)) == 0) || (name
[0] == '\0'))
225 if ((grptb
== NULL
) && (grptb_start() < 0))
229 * look up in hash table, if found and valid return the uid,
230 * if found and invalid, return a -1
232 ptr
= grptb
[st_hash(name
, namelen
, GNM_SZ
)];
233 if ((ptr
!= NULL
) && (ptr
->valid
> 0) && !strcmp(name
, ptr
->name
)) {
234 if (ptr
->valid
== INVALID
)
245 ptr
= grptb
[st_hash(name
, namelen
, GNM_SZ
)] =
246 malloc(sizeof(GIDC
));
249 * no match, look it up, if no match store it as an invalid entry,
250 * or store the matching gid
253 if ((gr
= getgrnam(name
)) == NULL
)
259 (void)strlcpy(ptr
->name
, name
, sizeof(ptr
->name
));
260 if ((gr
= getgrnam(name
)) == NULL
) {
261 ptr
->valid
= INVALID
;
265 *gid
= ptr
->gid
= gr
->gr_gid
;