📄 mmap_cache.c
字号:
/* * Boa, an http server * Copyright (C) 1999-2005 Larry Doolittle <ldoolitt@boa.org> * Copyright (C) 2000-2004 Jon Nelson <jnelson@boa.org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 1, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * *//* $Id: mmap_cache.c,v 1.9.2.9 2005/02/22 14:11:29 jnelson Exp $*/#include "boa.h"static int mmap_list_entries_used = 0;static int mmap_list_total_requests = 0;static int mmap_list_hash_bounces = 0;#define MMAP_LIST_NEXT(i) (((i)+1)&MMAP_LIST_MASK)#define MMAP_LIST_HASH(dev,ino,size) ((ino)&MMAP_LIST_MASK)/* define local table variable */static struct mmap_entry mmap_list[MMAP_LIST_SIZE];struct mmap_entry *find_mmap(int data_fd, struct stat *s){ void *m; int i, start; mmap_list_total_requests++; i = start = MMAP_LIST_HASH(s->st_dev, s->st_ino, s->st_size); for (; mmap_list[i].use_count;) { if (mmap_list[i].dev == s->st_dev && mmap_list[i].ino == s->st_ino && mmap_list[i].len == s->st_size) { mmap_list[i].use_count++; DEBUG(DEBUG_MMAP_CACHE) { fprintf(stderr, "Old mmap_list entry %d use_count now %d (hash was %d)\n", i, mmap_list[i].use_count, start); } return mmap_list + i; } mmap_list_hash_bounces++; i = MMAP_LIST_NEXT(i); if (i == start) { /* didn't find an entry that matches our dev/inode/size. There might be an entry that matches later in the table, but that _should_ be rare. The worst case is that we needlessly mmap() a file that is already mmap'd, but we did that all the time before this code was written, so it shouldn't be _too_ bad. */ /* we've looped, and found neither a free one nor a * match. Thus, there is no room for a new entry. *//* WARN("mmap hash table is full. Consider enlarging."); */ return NULL; } } /* Enforce a size limit here */ /* Disallow more entries than MMAP_LIST_USE_MAX, despite * having found an available slot. */ if (mmap_list_entries_used > MMAP_LIST_USE_MAX) {/* WARN("Too many entries in mmap hash table."); */ return NULL; } m = mmap(0, s->st_size, PROT_READ, MAP_OPTIONS, data_fd, 0); if ((long) m == -1) { int saved_errno = errno; log_error_time(); fprintf(stderr, "Unable to mmap file: "); errno = saved_errno; perror("mmap"); return NULL; }#ifdef HAVE_MADVISE { int mret; mret = madvise(m, s->st_size, MADV_SEQUENTIAL); if (mret == -1) { int saved_errno = errno; log_error_time(); fprintf(stderr, "Unable to madvise file: "); errno = saved_errno; perror("madvise"); munmap(m, s->st_size); return NULL; } }#endif DEBUG(DEBUG_MMAP_CACHE) { fprintf(stderr, "New mmap_list entry %d (hash was %d)\n", i, start); } mmap_list_entries_used++; mmap_list[i].dev = s->st_dev; mmap_list[i].ino = s->st_ino; mmap_list[i].len = s->st_size; mmap_list[i].mmap = m; mmap_list[i].use_count = 1; return mmap_list + i;}void release_mmap(struct mmap_entry *e){ if (!e) return; if (!e->use_count) { DEBUG(DEBUG_MMAP_CACHE) { fprintf(stderr, "mmap_list(%p)->use_count already zero!\n", (void *) e); } return; } if (!--(e->use_count)) { munmap(e->mmap, e->len); mmap_list_entries_used--; }}#if 0static struct mmap_entry *find_named_mmap(char *fname){ int data_fd; struct stat statbuf; struct mmap_entry *e; data_fd = open(fname, O_RDONLY); if (data_fd == -1) { perror(fname); return NULL; } fstat(data_fd, &statbuf); if (S_ISDIR(statbuf.st_mode)) {#ifdef DEBUG fprintf(stderr, "%s is a directory\n", fname);#endif return NULL; } e = find_mmap(data_fd, &statbuf); close(data_fd); return e;}#endif/* int main(int argc, char *argv[]) { #define MAXTEST 2048 struct mmap_entry *mlist[MAXTEST]; char name[1024], *s; int i, tests=0; while (fgets(name,sizeof(name),stdin) && tests < MAXTEST) { if (name[0]=='-') { i=atoi(name+1); release_mmap(mlist[i]); mlist[i]=NULL; } else { if ((s=strchr(name,'\n'))) *s='\0'; mlist[tests] = find_named_mmap(name); if (mlist[tests]) tests++; else fprintf(stderr, "find_named_mmap(%s) failed\n",name); } } fprintf(stderr, "mmap_list entries_used=%d ",mmap_list_entries_used); fprintf(stderr, "total_requests=%d ",mmap_list_total_requests); fprintf(stderr, "hash_bounces=%d\n",mmap_list_hash_bounces); for (i=0; i<tests; i++) release_mmap(mlist[i]); fprintf(stderr, "mmap_list entries_used=%d ",mmap_list_entries_used); fprintf(stderr, "total_requests=%d ",mmap_list_total_requests); fprintf(stderr, "hash_bounces=%d\n",mmap_list_hash_bounces);*/
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -