dmake: do not set MAKEFLAGS=k
[unleashed/tickless.git] / bin / pax / cache.c
blob8a2cd5379528312ea30ad58096e4867f51a61c42
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 $ */
4 /*-
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
14 * are met:
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
34 * SUCH DAMAGE.
37 #include <sys/types.h>
38 #include <sys/stat.h>
39 #include <grp.h>
40 #include <pwd.h>
41 #include <stdio.h>
42 #include <stdlib.h>
43 #include <string.h>
45 #include "pax.h"
46 #include "extern.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.
70 typedef struct uidc {
71 int valid; /* is this a valid or a miss entry */
72 char name[UNMLEN]; /* uid name */
73 uid_t uid; /* cached uid */
74 } UIDC;
76 typedef struct gidc {
77 int valid; /* is this a valid or a miss entry */
78 char name[GNMLEN]; /* gid name */
79 gid_t gid; /* cached gid */
80 } GIDC;
84 * routines that control user, group, uid and gid caches (for the archive
85 * member print routine).
86 * IMPORTANT:
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 */
96 * usrtb_start
97 * creates an empty usrtb
98 * Return:
99 * 0 if ok, -1 otherwise
103 usrtb_start(void)
105 static int fail = 0;
107 if (usrtb != NULL)
108 return(0);
109 if (fail)
110 return(-1);
111 if ((usrtb = calloc(UNM_SZ, sizeof(UIDC *))) == NULL) {
112 ++fail;
113 paxwarn(1, "Unable to allocate memory for user name cache table");
114 return(-1);
116 return(0);
120 * grptb_start
121 * creates an empty grptb
122 * Return:
123 * 0 if ok, -1 otherwise
127 grptb_start(void)
129 static int fail = 0;
131 if (grptb != NULL)
132 return(0);
133 if (fail)
134 return(-1);
135 if ((grptb = calloc(GNM_SZ, sizeof(GIDC *))) == NULL) {
136 ++fail;
137 paxwarn(1,"Unable to allocate memory for group name cache table");
138 return(-1);
140 return(0);
144 * uid_name()
145 * caches the uid for a given user name. We use a simple hash table.
146 * Return
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)
153 struct passwd *pw;
154 UIDC *ptr;
155 int namelen;
158 * return -1 for mangled names
160 if (((namelen = strlen(name)) == 0) || (name[0] == '\0'))
161 return(-1);
162 if ((usrtb == NULL) && (usrtb_start() < 0))
163 return(-1);
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)
172 return(-1);
173 *uid = ptr->uid;
174 return(0);
177 if (!pwopn) {
178 setpassent(1);
179 ++pwopn;
182 if (ptr == NULL)
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
190 if (ptr == NULL) {
191 if ((pw = getpwnam(name)) == NULL)
192 return(-1);
193 *uid = pw->pw_uid;
194 return(0);
196 (void)strlcpy(ptr->name, name, sizeof(ptr->name));
197 if ((pw = getpwnam(name)) == NULL) {
198 ptr->valid = INVALID;
199 return(-1);
201 ptr->valid = VALID;
202 *uid = ptr->uid = pw->pw_uid;
203 return(0);
207 * gid_name()
208 * caches the gid for a given group name. We use a simple hash table.
209 * Return
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)
216 struct group *gr;
217 GIDC *ptr;
218 int namelen;
221 * return -1 for mangled names
223 if (((namelen = strlen(name)) == 0) || (name[0] == '\0'))
224 return(-1);
225 if ((grptb == NULL) && (grptb_start() < 0))
226 return(-1);
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)
235 return(-1);
236 *gid = ptr->gid;
237 return(0);
240 if (!gropn) {
241 setgroupent(1);
242 ++gropn;
244 if (ptr == NULL)
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
252 if (ptr == NULL) {
253 if ((gr = getgrnam(name)) == NULL)
254 return(-1);
255 *gid = gr->gr_gid;
256 return(0);
259 (void)strlcpy(ptr->name, name, sizeof(ptr->name));
260 if ((gr = getgrnam(name)) == NULL) {
261 ptr->valid = INVALID;
262 return(-1);
264 ptr->valid = VALID;
265 *gid = ptr->gid = gr->gr_gid;
266 return(0);