]> git.ipfire.org Git - thirdparty/glibc.git/blame - nscd/initgrcache.c
nscd cleanups
[thirdparty/glibc.git] / nscd / initgrcache.c
CommitLineData
f7e7a396 1/* Cache handling for host lookup.
99231d9a 2 Copyright (C) 2004-2006, 2008, 2009, 2011 Free Software Foundation, Inc.
f7e7a396
UD
3 This file is part of the GNU C Library.
4 Contributed by Ulrich Drepper <drepper@redhat.com>, 2004.
5
43bc8ac6 6 This program is free software; you can redistribute it and/or modify
2e2efe65
RM
7 it under the terms of the GNU General Public License as published
8 by the Free Software Foundation; version 2 of the License, or
9 (at your option) any later version.
f7e7a396 10
43bc8ac6 11 This program is distributed in the hope that it will be useful,
f7e7a396 12 but WITHOUT ANY WARRANTY; without even the implied warranty of
43bc8ac6
UD
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
f7e7a396 15
43bc8ac6
UD
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software Foundation,
18 Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
f7e7a396
UD
19
20#include <assert.h>
21#include <errno.h>
22#include <grp.h>
23#include <libintl.h>
24#include <string.h>
25#include <time.h>
26#include <unistd.h>
27#include <sys/mman.h>
eac10791
UD
28
29#include "dbg_log.h"
30#include "nscd.h"
31#ifdef HAVE_SENDFILE
32# include <kernel-features.h>
33#endif
f7e7a396
UD
34
35#include "../nss/nsswitch.h"
36
37
38/* Type of the lookup function. */
39typedef enum nss_status (*initgroups_dyn_function) (const char *, gid_t,
40 long int *, long int *,
41 gid_t **, long int, int *);
42
43
44static const initgr_response_header notfound =
45{
46 .version = NSCD_VERSION,
47 .found = 0,
48 .ngrps = 0
49};
50
51
52#include "../grp/compat-initgroups.c"
53
54
a4c7ea7b 55static time_t
f7e7a396 56addinitgroupsX (struct database_dyn *db, int fd, request_header *req,
20e498bd 57 void *key, uid_t uid, struct hashentry *const he,
f7e7a396
UD
58 struct datahead *dh)
59{
60 /* Search for the entry matching the key. Please note that we don't
61 look again in the table whether the dataset is now available. We
62 simply insert it. It does not matter if it is in there twice. The
63 pruning function only will look at the timestamp. */
64
65
66 /* We allocate all data in one memory block: the iov vector,
67 the response header and the dataset itself. */
68 struct dataset
69 {
70 struct datahead head;
71 initgr_response_header resp;
72 char strdata[0];
73 } *dataset = NULL;
74
75 if (__builtin_expect (debug_level > 0, 0))
76 {
77 if (he == NULL)
78 dbg_log (_("Haven't found \"%s\" in group cache!"), (char *) key);
79 else
80 dbg_log (_("Reloading \"%s\" in group cache!"), (char *) key);
81 }
82
83 static service_user *group_database;
84 service_user *nip = NULL;
85 int no_more;
86
87 if (group_database != NULL)
88 {
89 nip = group_database;
90 no_more = 0;
91 }
92 else
93 no_more = __nss_database_lookup ("group", NULL,
94 "compat [NOTFOUND=return] files", &nip);
95
96 /* We always use sysconf even if NGROUPS_MAX is defined. That way, the
97 limit can be raised in the kernel configuration without having to
98 recompile libc. */
99 long int limit = __sysconf (_SC_NGROUPS_MAX);
100
101 long int size;
102 if (limit > 0)
103 /* We limit the size of the intially allocated array. */
104 size = MIN (limit, 64);
105 else
106 /* No fixed limit on groups. Pick a starting buffer size. */
107 size = 16;
108
109 long int start = 0;
110 bool all_tryagain = true;
cd248c3f 111 bool any_success = false;
f7e7a396 112
e8667ddc 113 /* This is temporary memory, we need not (and must not) call
f7e7a396
UD
114 mempool_alloc. */
115 // XXX This really should use alloca. need to change the backends.
116 gid_t *groups = (gid_t *) malloc (size * sizeof (gid_t));
117 if (__builtin_expect (groups == NULL, 0))
118 /* No more memory. */
119 goto out;
120
121 /* Nothing added yet. */
122 while (! no_more)
123 {
695c4370 124 long int prev_start = start;
f7e7a396
UD
125 enum nss_status status;
126 initgroups_dyn_function fct;
127 fct = __nss_lookup_function (nip, "initgroups_dyn");
128
129 if (fct == NULL)
130 {
131 status = compat_call (nip, key, -1, &start, &size, &groups,
132 limit, &errno);
133
134 if (nss_next_action (nip, NSS_STATUS_UNAVAIL) != NSS_ACTION_CONTINUE)
135 break;
136 }
137 else
138 status = DL_CALL_FCT (fct, (key, -1, &start, &size, &groups,
139 limit, &errno));
140
695c4370
UD
141 /* Remove duplicates. */
142 long int cnt = prev_start;
143 while (cnt < start)
144 {
145 long int inner;
146 for (inner = 0; inner < prev_start; ++inner)
147 if (groups[inner] == groups[cnt])
148 break;
149
150 if (inner < prev_start)
151 groups[cnt] = groups[--start];
152 else
153 ++cnt;
154 }
155
f7e7a396
UD
156 if (status != NSS_STATUS_TRYAGAIN)
157 all_tryagain = false;
158
159 /* This is really only for debugging. */
160 if (NSS_STATUS_TRYAGAIN > status || status > NSS_STATUS_RETURN)
161 __libc_fatal ("illegal status in internal_getgrouplist");
162
fbbc73b3
UD
163 any_success |= status == NSS_STATUS_SUCCESS;
164
f7e7a396
UD
165 if (status != NSS_STATUS_SUCCESS
166 && nss_next_action (nip, status) == NSS_ACTION_RETURN)
167 break;
168
169 if (nip->next == NULL)
170 no_more = -1;
171 else
172 nip = nip->next;
173 }
174
175 ssize_t total;
176 ssize_t written;
a4c7ea7b 177 time_t timeout;
f7e7a396 178 out:
a4c7ea7b 179 timeout = MAX_TIMEOUT_VALUE;
fbbc73b3 180 if (!any_success)
f7e7a396
UD
181 {
182 /* Nothing found. Create a negative result record. */
183 written = total = sizeof (notfound);
184
185 if (he != NULL && all_tryagain)
186 {
187 /* If we have an old record available but cannot find one now
188 because the service is not available we keep the old record
189 and make sure it does not get removed. */
190 if (reload_count != UINT_MAX && dh->nreloads == reload_count)
191 /* Do not reset the value if we never not reload the record. */
192 dh->nreloads = reload_count - 1;
a4c7ea7b
UD
193
194 /* Reload with the same time-to-live value. */
195 timeout = dh->timeout = time (NULL) + db->postimeout;
f7e7a396
UD
196 }
197 else
198 {
199 /* We have no data. This means we send the standard reply for this
200 case. */
201 if (fd != -1)
2c210d1e
UD
202 written = TEMP_FAILURE_RETRY (send (fd, &notfound, total,
203 MSG_NOSIGNAL));
f7e7a396 204
f7e7a396 205 /* If we cannot permanently store the result, so be it. */
99231d9a 206 if (__builtin_expect (db->negtimeout == 0, 0))
445b4a53
TK
207 {
208 /* Mark the old entry as obsolete. */
209 if (dh != NULL)
210 dh->usable = false;
211 }
99231d9a
UD
212 else if ((dataset = mempool_alloc (db, (sizeof (struct dataset)
213 + req->key_len), 1)) != NULL)
f7e7a396
UD
214 {
215 dataset->head.allocsize = sizeof (struct dataset) + req->key_len;
216 dataset->head.recsize = total;
217 dataset->head.notfound = true;
218 dataset->head.nreloads = 0;
219 dataset->head.usable = true;
220
221 /* Compute the timeout time. */
a4c7ea7b 222 timeout = dataset->head.timeout = time (NULL) + db->negtimeout;
f7e7a396
UD
223
224 /* This is the reply. */
225 memcpy (&dataset->resp, &notfound, total);
226
227 /* Copy the key data. */
228 char *key_copy = memcpy (dataset->strdata, key, req->key_len);
229
230 /* If necessary, we also propagate the data to disk. */
231 if (db->persistent)
232 {
233 // XXX async OK?
234 uintptr_t pval = (uintptr_t) dataset & ~pagesize_m1;
235 msync ((void *) pval,
236 ((uintptr_t) dataset & pagesize_m1)
237 + sizeof (struct dataset) + req->key_len, MS_ASYNC);
238 }
239
7e71e55f 240 (void) cache_add (req->type, key_copy, req->key_len,
528741cb 241 &dataset->head, true, db, uid, he == NULL);
f7e7a396 242
00ebd7ed
UD
243 pthread_rwlock_unlock (&db->lock);
244
f7e7a396
UD
245 /* Mark the old entry as obsolete. */
246 if (dh != NULL)
247 dh->usable = false;
248 }
f7e7a396
UD
249 }
250 }
251 else
252 {
253
5a337776
UD
254 written = total = (offsetof (struct dataset, strdata)
255 + start * sizeof (int32_t));
f7e7a396
UD
256
257 /* If we refill the cache, first assume the reconrd did not
258 change. Allocate memory on the cache since it is likely
259 discarded anyway. If it turns out to be necessary to have a
260 new record we can still allocate real memory. */
261 bool alloca_used = false;
262 dataset = NULL;
263
264 if (he == NULL)
20e498bd
UD
265 dataset = (struct dataset *) mempool_alloc (db, total + req->key_len,
266 1);
f7e7a396
UD
267
268 if (dataset == NULL)
269 {
270 /* We cannot permanently add the result in the moment. But
271 we can provide the result as is. Store the data in some
272 temporary memory. */
273 dataset = (struct dataset *) alloca (total + req->key_len);
274
275 /* We cannot add this record to the permanent database. */
276 alloca_used = true;
277 }
278
279 dataset->head.allocsize = total + req->key_len;
280 dataset->head.recsize = total - offsetof (struct dataset, resp);
281 dataset->head.notfound = false;
282 dataset->head.nreloads = he == NULL ? 0 : (dh->nreloads + 1);
283 dataset->head.usable = true;
284
285 /* Compute the timeout time. */
a4c7ea7b 286 timeout = dataset->head.timeout = time (NULL) + db->postimeout;
f7e7a396
UD
287
288 dataset->resp.version = NSCD_VERSION;
289 dataset->resp.found = 1;
290 dataset->resp.ngrps = start;
291
292 char *cp = dataset->strdata;
293
294 /* Copy the GID values. If the size of the types match this is
295 very simple. */
296 if (sizeof (gid_t) == sizeof (int32_t))
297 cp = mempcpy (cp, groups, start * sizeof (gid_t));
298 else
299 {
300 gid_t *gcp = (gid_t *) cp;
301
302 for (int i = 0; i < start; ++i)
303 *gcp++ = groups[i];
304
305 cp = (char *) gcp;
306 }
307
308 /* Finally the user name. */
309 memcpy (cp, key, req->key_len);
310
5a337776
UD
311 assert (cp == dataset->strdata + total - offsetof (struct dataset,
312 strdata));
313
f7e7a396
UD
314 /* Now we can determine whether on refill we have to create a new
315 record or not. */
316 if (he != NULL)
317 {
318 assert (fd == -1);
319
320 if (total + req->key_len == dh->allocsize
321 && total - offsetof (struct dataset, resp) == dh->recsize
322 && memcmp (&dataset->resp, dh->data,
323 dh->allocsize - offsetof (struct dataset, resp)) == 0)
324 {
325 /* The data has not changed. We will just bump the
326 timeout value. Note that the new record has been
327 allocated on the stack and need not be freed. */
328 dh->timeout = dataset->head.timeout;
329 ++dh->nreloads;
330 }
331 else
332 {
333 /* We have to create a new record. Just allocate
334 appropriate memory and copy it. */
335 struct dataset *newp
c52137d3 336 = (struct dataset *) mempool_alloc (db, total + req->key_len,
20e498bd 337 1);
f7e7a396
UD
338 if (newp != NULL)
339 {
340 /* Adjust pointer into the memory block. */
341 cp = (char *) newp + (cp - (char *) dataset);
342
343 dataset = memcpy (newp, dataset, total + req->key_len);
344 alloca_used = false;
345 }
346
347 /* Mark the old record as obsolete. */
348 dh->usable = false;
349 }
350 }
351 else
352 {
353 /* We write the dataset before inserting it to the database
354 since while inserting this thread might block and so would
355 unnecessarily let the receiver wait. */
356 assert (fd != -1);
357
eac10791 358#ifdef HAVE_SENDFILE
74158740 359 if (__builtin_expect (db->mmap_used, 1) && !alloca_used)
eac10791
UD
360 {
361 assert (db->wr_fd != -1);
362 assert ((char *) &dataset->resp > (char *) db->data);
ea547a1a 363 assert ((char *) dataset - (char *) db->head
eac10791
UD
364 + total
365 <= (sizeof (struct database_pers_head)
366 + db->head->module * sizeof (ref_t)
367 + db->head->data_size));
bd547139
UD
368 written = sendfileall (fd, db->wr_fd,
369 (char *) &dataset->resp
ea547a1a 370 - (char *) db->head, dataset->head.recsize);
eac10791
UD
371# ifndef __ASSUME_SENDFILE
372 if (written == -1 && errno == ENOSYS)
373 goto use_write;
374# endif
375 }
376 else
377# ifndef __ASSUME_SENDFILE
378 use_write:
379# endif
380#endif
ea547a1a 381 written = writeall (fd, &dataset->resp, dataset->head.recsize);
f7e7a396
UD
382 }
383
384
385 /* Add the record to the database. But only if it has not been
386 stored on the stack. */
387 if (! alloca_used)
388 {
389 /* If necessary, we also propagate the data to disk. */
390 if (db->persistent)
391 {
392 // XXX async OK?
393 uintptr_t pval = (uintptr_t) dataset & ~pagesize_m1;
394 msync ((void *) pval,
395 ((uintptr_t) dataset & pagesize_m1) + total +
396 req->key_len, MS_ASYNC);
397 }
398
7e71e55f 399 (void) cache_add (INITGROUPS, cp, req->key_len, &dataset->head, true,
528741cb 400 db, uid, he == NULL);
00ebd7ed
UD
401
402 pthread_rwlock_unlock (&db->lock);
f7e7a396
UD
403 }
404 }
405
406 free (groups);
407
408 if (__builtin_expect (written != total, 0) && debug_level > 0)
409 {
410 char buf[256];
411 dbg_log (_("short write in %s: %s"), __FUNCTION__,
412 strerror_r (errno, buf, sizeof (buf)));
413 }
a4c7ea7b
UD
414
415 return timeout;
f7e7a396
UD
416}
417
418
419void
420addinitgroups (struct database_dyn *db, int fd, request_header *req, void *key,
421 uid_t uid)
422{
423 addinitgroupsX (db, fd, req, key, uid, NULL, NULL);
424}
425
426
a4c7ea7b 427time_t
f7e7a396
UD
428readdinitgroups (struct database_dyn *db, struct hashentry *he,
429 struct datahead *dh)
430{
431 request_header req =
432 {
433 .type = INITGROUPS,
434 .key_len = he->len
435 };
436
a4c7ea7b 437 return addinitgroupsX (db, -1, &req, db->data + he->key, he->owner, he, dh);
f7e7a396 438}