⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 mmap_cache.c

📁 用于移植到嵌入式linux系统的boa http服务器
💻 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 + -