xref: /lighttpd1.4/src/stat_cache.c (revision dbdab5db)
1 #include "first.h"
2 
3 #include "log.h"
4 #include "stat_cache.h"
5 #include "fdevent.h"
6 #include "etag.h"
7 
8 #include <sys/types.h>
9 #include <sys/stat.h>
10 
11 #include <stdlib.h>
12 #include <string.h>
13 #include <errno.h>
14 #include <unistd.h>
15 #include <stdio.h>
16 #include <fcntl.h>
17 #include <assert.h>
18 
19 #ifdef HAVE_ATTR_ATTRIBUTES_H
20 # include <attr/attributes.h>
21 #endif
22 
23 #ifdef HAVE_SYS_EXTATTR_H
24 # include <sys/extattr.h>
25 #endif
26 
27 #ifdef HAVE_FAM_H
28 # include <fam.h>
29 #endif
30 
31 #include "sys-mmap.h"
32 
33 /* NetBSD 1.3.x needs it */
34 #ifndef MAP_FAILED
35 # define MAP_FAILED -1
36 #endif
37 
38 #ifndef O_LARGEFILE
39 # define O_LARGEFILE 0
40 #endif
41 
42 #ifndef HAVE_LSTAT
43 # define lstat stat
44 #endif
45 
46 #if 0
47 /* enables debug code for testing if all nodes in the stat-cache as accessable */
48 #define DEBUG_STAT_CACHE
49 #endif
50 
51 /*
52  * stat-cache
53  *
54  * we cache the stat() calls in our own storage
55  * the directories are cached in FAM
56  *
57  * if we get a change-event from FAM, we increment the version in the FAM->dir mapping
58  *
59  * if the stat()-cache is queried we check if the version id for the directory is the
60  * same and return immediatly.
61  *
62  *
63  * What we need:
64  *
65  * - for each stat-cache entry we need a fast indirect lookup on the directory name
66  * - for each FAMRequest we have to find the version in the directory cache (index as userdata)
67  *
68  * stat <<-> directory <-> FAMRequest
69  *
70  * if file is deleted, directory is dirty, file is rechecked ...
71  * if directory is deleted, directory mapping is removed
72  *
73  * */
74 
75 #ifdef HAVE_FAM_H
76 typedef struct {
77 	FAMRequest *req;
78 
79 	buffer *name;
80 
81 	int version;
82 } fam_dir_entry;
83 #endif
84 
85 /* the directory name is too long to always compare on it
86  * - we need a hash
87  * - the hash-key is used as sorting criteria for a tree
88  * - a splay-tree is used as we can use the caching effect of it
89  */
90 
91 /* we want to cleanup the stat-cache every few seconds, let's say 10
92  *
93  * - remove entries which are outdated since 30s
94  * - remove entries which are fresh but havn't been used since 60s
95  * - if we don't have a stat-cache entry for a directory, release it from the monitor
96  */
97 
98 #ifdef DEBUG_STAT_CACHE
99 typedef struct {
100 	int *ptr;
101 
102 	size_t used;
103 	size_t size;
104 } fake_keys;
105 
106 static fake_keys ctrl;
107 #endif
108 
109 stat_cache *stat_cache_init(void) {
110 	stat_cache *sc = NULL;
111 
112 	sc = calloc(1, sizeof(*sc));
113 	force_assert(NULL != sc);
114 
115 	sc->dir_name = buffer_init();
116 	sc->hash_key = buffer_init();
117 
118 #ifdef HAVE_FAM_H
119 	sc->fam_fcce_ndx = -1;
120 #endif
121 
122 #ifdef DEBUG_STAT_CACHE
123 	ctrl.size = 0;
124 #endif
125 
126 	return sc;
127 }
128 
129 static stat_cache_entry * stat_cache_entry_init(void) {
130 	stat_cache_entry *sce = NULL;
131 
132 	sce = calloc(1, sizeof(*sce));
133 	force_assert(NULL != sce);
134 
135 	sce->name = buffer_init();
136 	sce->etag = buffer_init();
137 	sce->content_type = buffer_init();
138 
139 	return sce;
140 }
141 
142 static void stat_cache_entry_free(void *data) {
143 	stat_cache_entry *sce = data;
144 	if (!sce) return;
145 
146 	buffer_free(sce->etag);
147 	buffer_free(sce->name);
148 	buffer_free(sce->content_type);
149 
150 	free(sce);
151 }
152 
153 #ifdef HAVE_FAM_H
154 static fam_dir_entry * fam_dir_entry_init(void) {
155 	fam_dir_entry *fam_dir = NULL;
156 
157 	fam_dir = calloc(1, sizeof(*fam_dir));
158 	force_assert(NULL != fam_dir);
159 
160 	fam_dir->name = buffer_init();
161 
162 	return fam_dir;
163 }
164 
165 static void fam_dir_entry_free(FAMConnection *fc, void *data) {
166 	fam_dir_entry *fam_dir = data;
167 
168 	if (!fam_dir) return;
169 
170 	FAMCancelMonitor(fc, fam_dir->req);
171 
172 	buffer_free(fam_dir->name);
173 	free(fam_dir->req);
174 
175 	free(fam_dir);
176 }
177 #endif
178 
179 void stat_cache_free(stat_cache *sc) {
180 	while (sc->files) {
181 		int osize;
182 		splay_tree *node = sc->files;
183 
184 		osize = sc->files->size;
185 
186 		stat_cache_entry_free(node->data);
187 		sc->files = splaytree_delete(sc->files, node->key);
188 
189 		force_assert(osize - 1 == splaytree_size(sc->files));
190 	}
191 
192 	buffer_free(sc->dir_name);
193 	buffer_free(sc->hash_key);
194 
195 #ifdef HAVE_FAM_H
196 	while (sc->dirs) {
197 		int osize;
198 		splay_tree *node = sc->dirs;
199 
200 		osize = sc->dirs->size;
201 
202 		fam_dir_entry_free(&sc->fam, node->data);
203 		sc->dirs = splaytree_delete(sc->dirs, node->key);
204 
205 		if (osize == 1) {
206 			force_assert(NULL == sc->dirs);
207 		} else {
208 			force_assert(osize == (sc->dirs->size + 1));
209 		}
210 	}
211 
212 	if (-1 != sc->fam_fcce_ndx) {
213 		/* fd events already gone */
214 		sc->fam_fcce_ndx = -1;
215 
216 		FAMClose(&sc->fam);
217 	}
218 #endif
219 	free(sc);
220 }
221 
222 #if defined(HAVE_XATTR)
223 static int stat_cache_attr_get(buffer *buf, char *name, char *xattrname) {
224 	int attrlen;
225 	int ret;
226 
227 	buffer_string_prepare_copy(buf, 1023);
228 	attrlen = buf->size - 1;
229 	if(0 == (ret = attr_get(name, xattrname, buf->ptr, &attrlen, 0))) {
230 		buffer_commit(buf, attrlen);
231 	}
232 	return ret;
233 }
234 #elif defined(HAVE_EXTATTR)
235 static int stat_cache_attr_get(buffer *buf, char *name, char *xattrname) {
236 	ssize_t attrlen;
237 
238 	buffer_string_prepare_copy(buf, 1023);
239 
240 	if (-1 != (attrlen = extattr_get_file(name, EXTATTR_NAMESPACE_USER, xattrname, buf->ptr, buf->size - 1))) {
241 		buf->used = attrlen + 1;
242 		buf->ptr[attrlen] = '\0';
243 		return 0;
244 	}
245 	return -1;
246 }
247 #endif
248 
249 /* the famous DJB hash function for strings */
250 static uint32_t hashme(buffer *str) {
251 	uint32_t hash = 5381;
252 	const char *s;
253 	for (s = str->ptr; *s; s++) {
254 		hash = ((hash << 5) + hash) + *s;
255 	}
256 
257 	hash &= ~(((uint32_t)1) << 31); /* strip the highest bit */
258 
259 	return hash;
260 }
261 
262 #ifdef HAVE_FAM_H
263 handler_t stat_cache_handle_fdevent(server *srv, void *_fce, int revent) {
264 	size_t i;
265 	stat_cache *sc = srv->stat_cache;
266 	size_t events;
267 
268 	UNUSED(_fce);
269 	/* */
270 
271 	if (revent & FDEVENT_IN) {
272 		events = FAMPending(&sc->fam);
273 
274 		for (i = 0; i < events; i++) {
275 			FAMEvent fe;
276 			fam_dir_entry *fam_dir;
277 			splay_tree *node;
278 			int ndx, j;
279 
280 			FAMNextEvent(&sc->fam, &fe);
281 
282 			/* handle event */
283 
284 			switch(fe.code) {
285 			case FAMChanged:
286 			case FAMDeleted:
287 			case FAMMoved:
288 				/* if the filename is a directory remove the entry */
289 
290 				fam_dir = fe.userdata;
291 				fam_dir->version++;
292 
293 				/* file/dir is still here */
294 				if (fe.code == FAMChanged) break;
295 
296 				/* we have 2 versions, follow and no-follow-symlink */
297 
298 				for (j = 0; j < 2; j++) {
299 					buffer_copy_string(sc->hash_key, fe.filename);
300 					buffer_append_int(sc->hash_key, j);
301 
302 					ndx = hashme(sc->hash_key);
303 
304 					sc->dirs = splaytree_splay(sc->dirs, ndx);
305 					node = sc->dirs;
306 
307 					if (node && (node->key == ndx)) {
308 						int osize = splaytree_size(sc->dirs);
309 
310 						fam_dir_entry_free(&sc->fam, node->data);
311 						sc->dirs = splaytree_delete(sc->dirs, ndx);
312 
313 						force_assert(osize - 1 == splaytree_size(sc->dirs));
314 					}
315 				}
316 				break;
317 			default:
318 				break;
319 			}
320 		}
321 	}
322 
323 	if (revent & FDEVENT_HUP) {
324 		/* fam closed the connection */
325 		fdevent_event_del(srv->ev, &(sc->fam_fcce_ndx), FAMCONNECTION_GETFD(&sc->fam));
326 		fdevent_unregister(srv->ev, FAMCONNECTION_GETFD(&sc->fam));
327 
328 		FAMClose(&sc->fam);
329 	}
330 
331 	return HANDLER_GO_ON;
332 }
333 
334 static int buffer_copy_dirname(buffer *dst, buffer *file) {
335 	size_t i;
336 
337 	if (buffer_string_is_empty(file)) return -1;
338 
339 	for (i = buffer_string_length(file); i > 0; i--) {
340 		if (file->ptr[i] == '/') {
341 			buffer_copy_string_len(dst, file->ptr, i);
342 			return 0;
343 		}
344 	}
345 
346 	return -1;
347 }
348 #endif
349 
350 #ifdef HAVE_LSTAT
351 static int stat_cache_lstat(server *srv, buffer *dname, struct stat *lst) {
352 	if (lstat(dname->ptr, lst) == 0) {
353 		return S_ISLNK(lst->st_mode) ? 0 : 1;
354 	}
355 	else {
356 		log_error_write(srv, __FILE__, __LINE__, "sbs",
357 				"lstat failed for:",
358 				dname, strerror(errno));
359 	};
360 	return -1;
361 }
362 #endif
363 
364 /***
365  *
366  *
367  *
368  * returns:
369  *  - HANDLER_FINISHED on cache-miss (don't forget to reopen the file)
370  *  - HANDLER_ERROR on stat() failed -> see errno for problem
371  */
372 
373 handler_t stat_cache_get_entry(server *srv, connection *con, buffer *name, stat_cache_entry **ret_sce) {
374 #ifdef HAVE_FAM_H
375 	fam_dir_entry *fam_dir = NULL;
376 	int dir_ndx = -1;
377 #endif
378 	stat_cache_entry *sce = NULL;
379 	stat_cache *sc;
380 	struct stat st;
381 	size_t k;
382 	int fd;
383 	struct stat lst;
384 #ifdef DEBUG_STAT_CACHE
385 	size_t i;
386 #endif
387 
388 	int file_ndx;
389 
390 	*ret_sce = NULL;
391 
392 	/*
393 	 * check if the directory for this file has changed
394 	 */
395 
396 	sc = srv->stat_cache;
397 
398 	buffer_copy_buffer(sc->hash_key, name);
399 	buffer_append_int(sc->hash_key, con->conf.follow_symlink);
400 
401 	file_ndx = hashme(sc->hash_key);
402 	sc->files = splaytree_splay(sc->files, file_ndx);
403 
404 #ifdef DEBUG_STAT_CACHE
405 	for (i = 0; i < ctrl.used; i++) {
406 		if (ctrl.ptr[i] == file_ndx) break;
407 	}
408 #endif
409 
410 	if (sc->files && (sc->files->key == file_ndx)) {
411 #ifdef DEBUG_STAT_CACHE
412 		/* it was in the cache */
413 		force_assert(i < ctrl.used);
414 #endif
415 
416 		/* we have seen this file already and
417 		 * don't stat() it again in the same second */
418 
419 		sce = sc->files->data;
420 
421 		/* check if the name is the same, we might have a collision */
422 
423 		if (buffer_is_equal(name, sce->name)) {
424 			if (srv->srvconf.stat_cache_engine == STAT_CACHE_ENGINE_SIMPLE) {
425 				if (sce->stat_ts == srv->cur_ts) {
426 					*ret_sce = sce;
427 					return HANDLER_GO_ON;
428 				}
429 			}
430 		} else {
431 			/* collision, forget about the entry */
432 			sce = NULL;
433 		}
434 	} else {
435 #ifdef DEBUG_STAT_CACHE
436 		if (i != ctrl.used) {
437 			log_error_write(srv, __FILE__, __LINE__, "xSB",
438 				file_ndx, "was already inserted but not found in cache, ", name);
439 		}
440 		force_assert(i == ctrl.used);
441 #endif
442 	}
443 
444 #ifdef HAVE_FAM_H
445 	/* dir-check */
446 	if (srv->srvconf.stat_cache_engine == STAT_CACHE_ENGINE_FAM) {
447 		if (0 != buffer_copy_dirname(sc->dir_name, name)) {
448 			log_error_write(srv, __FILE__, __LINE__, "sb",
449 				"no '/' found in filename:", name);
450 			return HANDLER_ERROR;
451 		}
452 
453 		buffer_copy_buffer(sc->hash_key, sc->dir_name);
454 		buffer_append_int(sc->hash_key, con->conf.follow_symlink);
455 
456 		dir_ndx = hashme(sc->hash_key);
457 
458 		sc->dirs = splaytree_splay(sc->dirs, dir_ndx);
459 
460 		if ((NULL != sc->dirs) && (sc->dirs->key == dir_ndx)) {
461 			fam_dir = sc->dirs->data;
462 
463 			/* check whether we got a collision */
464 			if (buffer_is_equal(sc->dir_name, fam_dir->name)) {
465 				/* test whether a found file cache entry is still ok */
466 				if ((NULL != sce) && (fam_dir->version == sce->dir_version)) {
467 					/* the stat()-cache entry is still ok */
468 
469 					*ret_sce = sce;
470 					return HANDLER_GO_ON;
471 				}
472 			} else {
473 				/* hash collision, forget about the entry */
474 				fam_dir = NULL;
475 			}
476 		}
477 	}
478 #endif
479 
480 	/*
481 	 * *lol*
482 	 * - open() + fstat() on a named-pipe results in a (intended) hang.
483 	 * - stat() if regular file + open() to see if we can read from it is better
484 	 *
485 	 * */
486 	if (-1 == stat(name->ptr, &st)) {
487 		return HANDLER_ERROR;
488 	}
489 
490 
491 	if (S_ISREG(st.st_mode)) {
492 		/* fix broken stat/open for symlinks to reg files with appended slash on freebsd,osx */
493 		if (name->ptr[buffer_string_length(name) - 1] == '/') {
494 			errno = ENOTDIR;
495 			return HANDLER_ERROR;
496 		}
497 
498 		/* try to open the file to check if we can read it */
499 		if (-1 == (fd = open(name->ptr, O_RDONLY))) {
500 			return HANDLER_ERROR;
501 		}
502 		close(fd);
503 	}
504 
505 	if (NULL == sce) {
506 
507 		sce = stat_cache_entry_init();
508 		buffer_copy_buffer(sce->name, name);
509 
510 		/* already splayed file_ndx */
511 		if ((NULL != sc->files) && (sc->files->key == file_ndx)) {
512 			/* hash collision: replace old entry */
513 			stat_cache_entry_free(sc->files->data);
514 			sc->files->data = sce;
515 		} else {
516 			int osize = splaytree_size(sc->files);
517 
518 			sc->files = splaytree_insert(sc->files, file_ndx, sce);
519 			force_assert(osize + 1 == splaytree_size(sc->files));
520 
521 #ifdef DEBUG_STAT_CACHE
522 			if (ctrl.size == 0) {
523 				ctrl.size = 16;
524 				ctrl.used = 0;
525 				ctrl.ptr = malloc(ctrl.size * sizeof(*ctrl.ptr));
526 				force_assert(NULL != ctrl.ptr);
527 			} else if (ctrl.size == ctrl.used) {
528 				ctrl.size += 16;
529 				ctrl.ptr = realloc(ctrl.ptr, ctrl.size * sizeof(*ctrl.ptr));
530 				force_assert(NULL != ctrl.ptr);
531 			}
532 
533 			ctrl.ptr[ctrl.used++] = file_ndx;
534 #endif
535 		}
536 		force_assert(sc->files);
537 		force_assert(sc->files->data == sce);
538 	}
539 
540 	sce->st = st;
541 	sce->stat_ts = srv->cur_ts;
542 
543 	/* catch the obvious symlinks
544 	 *
545 	 * this is not a secure check as we still have a race-condition between
546 	 * the stat() and the open. We can only solve this by
547 	 * 1. open() the file
548 	 * 2. fstat() the fd
549 	 *
550 	 * and keeping the file open for the rest of the time. But this can
551 	 * only be done at network level.
552 	 *
553 	 * per default it is not a symlink
554 	 * */
555 #ifdef HAVE_LSTAT
556 	sce->is_symlink = 0;
557 
558 	/* we want to only check for symlinks if we should block symlinks.
559 	 */
560 	if (!con->conf.follow_symlink) {
561 		if (stat_cache_lstat(srv, name, &lst)  == 0) {
562 #ifdef DEBUG_STAT_CACHE
563 				log_error_write(srv, __FILE__, __LINE__, "sb",
564 						"found symlink", name);
565 #endif
566 				sce->is_symlink = 1;
567 		}
568 
569 		/*
570 		 * we assume "/" can not be symlink, so
571 		 * skip the symlink stuff if our path is /
572 		 **/
573 		else if (buffer_string_length(name) > 1) {
574 			buffer *dname;
575 			char *s_cur;
576 
577 			dname = buffer_init();
578 			buffer_copy_buffer(dname, name);
579 
580 			while ((s_cur = strrchr(dname->ptr, '/'))) {
581 				buffer_string_set_length(dname, s_cur - dname->ptr);
582 				if (dname->ptr == s_cur) {
583 #ifdef DEBUG_STAT_CACHE
584 					log_error_write(srv, __FILE__, __LINE__, "s", "reached /");
585 #endif
586 					break;
587 				}
588 #ifdef DEBUG_STAT_CACHE
589 				log_error_write(srv, __FILE__, __LINE__, "sbs",
590 						"checking if", dname, "is a symlink");
591 #endif
592 				if (stat_cache_lstat(srv, dname, &lst)  == 0) {
593 					sce->is_symlink = 1;
594 #ifdef DEBUG_STAT_CACHE
595 					log_error_write(srv, __FILE__, __LINE__, "sb",
596 							"found symlink", dname);
597 #endif
598 					break;
599 				};
600 			};
601 			buffer_free(dname);
602 		};
603 	};
604 #endif
605 
606 	if (S_ISREG(st.st_mode)) {
607 		/* determine mimetype */
608 		buffer_reset(sce->content_type);
609 #if defined(HAVE_XATTR) || defined(HAVE_EXTATTR)
610 		if (con->conf.use_xattr) {
611 			stat_cache_attr_get(sce->content_type, name->ptr, srv->srvconf.xattr_name->ptr);
612 		}
613 #endif
614 		/* xattr did not set a content-type. ask the config */
615 		if (buffer_string_is_empty(sce->content_type)) {
616 			size_t namelen = buffer_string_length(name);
617 
618 			for (k = 0; k < con->conf.mimetypes->used; k++) {
619 				data_string *ds = (data_string *)con->conf.mimetypes->data[k];
620 				buffer *type = ds->key;
621 				size_t typelen = buffer_string_length(type);
622 
623 				if (buffer_is_empty(type)) continue;
624 
625 				/* check if the right side is the same */
626 				if (typelen > namelen) continue;
627 
628 				if (0 == strncasecmp(name->ptr + namelen - typelen, type->ptr, typelen)) {
629 					buffer_copy_buffer(sce->content_type, ds->value);
630 					break;
631 				}
632 			}
633 		}
634 		etag_create(sce->etag, &(sce->st), con->etag_flags);
635 	} else if (S_ISDIR(st.st_mode)) {
636 		etag_create(sce->etag, &(sce->st), con->etag_flags);
637 	}
638 
639 #ifdef HAVE_FAM_H
640 	if (srv->srvconf.stat_cache_engine == STAT_CACHE_ENGINE_FAM) {
641 		/* is this directory already registered ? */
642 		if (NULL == fam_dir) {
643 			fam_dir = fam_dir_entry_init();
644 
645 			buffer_copy_buffer(fam_dir->name, sc->dir_name);
646 
647 			fam_dir->version = 1;
648 
649 			fam_dir->req = calloc(1, sizeof(FAMRequest));
650 			force_assert(NULL != fam_dir);
651 
652 			if (0 != FAMMonitorDirectory(&sc->fam, fam_dir->name->ptr,
653 						     fam_dir->req, fam_dir)) {
654 
655 				log_error_write(srv, __FILE__, __LINE__, "sbsbs",
656 						"monitoring dir failed:",
657 						fam_dir->name,
658 						"file:", name,
659 						FamErrlist[FAMErrno]);
660 
661 				fam_dir_entry_free(&sc->fam, fam_dir);
662 				fam_dir = NULL;
663 			} else {
664 				int osize = splaytree_size(sc->dirs);
665 
666 				/* already splayed dir_ndx */
667 				if ((NULL != sc->dirs) && (sc->dirs->key == dir_ndx)) {
668 					/* hash collision: replace old entry */
669 					fam_dir_entry_free(&sc->fam, sc->dirs->data);
670 					sc->dirs->data = fam_dir;
671 				} else {
672 					sc->dirs = splaytree_insert(sc->dirs, dir_ndx, fam_dir);
673 					force_assert(osize == (splaytree_size(sc->dirs) - 1));
674 				}
675 
676 				force_assert(sc->dirs);
677 				force_assert(sc->dirs->data == fam_dir);
678 			}
679 		}
680 
681 		/* bind the fam_fc to the stat() cache entry */
682 
683 		if (fam_dir) {
684 			sce->dir_version = fam_dir->version;
685 		}
686 	}
687 #endif
688 
689 	*ret_sce = sce;
690 
691 	return HANDLER_GO_ON;
692 }
693 
694 int stat_cache_open_rdonly_fstat (server *srv, connection *con, buffer *name, struct stat *st) {
695 	/*(Note: O_NOFOLLOW affects only the final path segment, the target file,
696 	 * not any intermediate symlinks along the path)*/
697 	#ifndef O_BINARY
698 	#define O_BINARY 0
699 	#endif
700 	#ifndef O_LARGEFILE
701 	#define O_LARGEFILE 0
702 	#endif
703 	#ifndef O_NOCTTY
704 	#define O_NOCTTY 0
705 	#endif
706 	#ifndef O_NONBLOCK
707 	#define O_NONBLOCK 0
708 	#endif
709 	#ifndef O_NOFOLLOW
710 	#define O_NOFOLLOW 0
711 	#endif
712 	const int oflags = O_BINARY | O_LARGEFILE | O_NOCTTY | O_NONBLOCK
713 			 | (con->conf.follow_symlink ? 0 : O_NOFOLLOW);
714 	const int fd = open(name->ptr, O_RDONLY | oflags);
715 	if (fd >= 0) {
716 		if (0 == fstat(fd, st)) {
717 			return fd;
718 		} else {
719 			close(fd);
720 		}
721 	}
722 	UNUSED(srv); /*(might log_error_write(srv, ...) in the future)*/
723 	return -1;
724 }
725 
726 /**
727  * remove stat() from cache which havn't been stat()ed for
728  * more than 10 seconds
729  *
730  *
731  * walk though the stat-cache, collect the ids which are too old
732  * and remove them in a second loop
733  */
734 
735 static int stat_cache_tag_old_entries(server *srv, splay_tree *t, int *keys, size_t *ndx) {
736 	stat_cache_entry *sce;
737 
738 	if (!t) return 0;
739 
740 	stat_cache_tag_old_entries(srv, t->left, keys, ndx);
741 	stat_cache_tag_old_entries(srv, t->right, keys, ndx);
742 
743 	sce = t->data;
744 
745 	if (srv->cur_ts - sce->stat_ts > 2) {
746 		keys[(*ndx)++] = t->key;
747 	}
748 
749 	return 0;
750 }
751 
752 int stat_cache_trigger_cleanup(server *srv) {
753 	stat_cache *sc;
754 	size_t max_ndx = 0, i;
755 	int *keys;
756 
757 	sc = srv->stat_cache;
758 
759 	if (!sc->files) return 0;
760 
761 	keys = calloc(1, sizeof(int) * sc->files->size);
762 	force_assert(NULL != keys);
763 
764 	stat_cache_tag_old_entries(srv, sc->files, keys, &max_ndx);
765 
766 	for (i = 0; i < max_ndx; i++) {
767 		int ndx = keys[i];
768 		splay_tree *node;
769 
770 		sc->files = splaytree_splay(sc->files, ndx);
771 
772 		node = sc->files;
773 
774 		if (node && (node->key == ndx)) {
775 #ifdef DEBUG_STAT_CACHE
776 			size_t j;
777 			int osize = splaytree_size(sc->files);
778 			stat_cache_entry *sce = node->data;
779 #endif
780 			stat_cache_entry_free(node->data);
781 			sc->files = splaytree_delete(sc->files, ndx);
782 
783 #ifdef DEBUG_STAT_CACHE
784 			for (j = 0; j < ctrl.used; j++) {
785 				if (ctrl.ptr[j] == ndx) {
786 					ctrl.ptr[j] = ctrl.ptr[--ctrl.used];
787 					break;
788 				}
789 			}
790 
791 			force_assert(osize - 1 == splaytree_size(sc->files));
792 #endif
793 		}
794 	}
795 
796 	free(keys);
797 
798 	return 0;
799 }
800