Blame src/plugins/kdb/db2/libdb2/mpool/mpool.c

Packit fd8b60
/*-
Packit fd8b60
 * Copyright (c) 1990, 1993, 1994
Packit fd8b60
 *	The Regents of the University of California.  All rights reserved.
Packit fd8b60
 *
Packit fd8b60
 * Redistribution and use in source and binary forms, with or without
Packit fd8b60
 * modification, are permitted provided that the following conditions
Packit fd8b60
 * are met:
Packit fd8b60
 * 1. Redistributions of source code must retain the above copyright
Packit fd8b60
 *    notice, this list of conditions and the following disclaimer.
Packit fd8b60
 * 2. Redistributions in binary form must reproduce the above copyright
Packit fd8b60
 *    notice, this list of conditions and the following disclaimer in the
Packit fd8b60
 *    documentation and/or other materials provided with the distribution.
Packit fd8b60
 * 3. All advertising materials mentioning features or use of this software
Packit fd8b60
 *    must display the following acknowledgement:
Packit fd8b60
 *	This product includes software developed by the University of
Packit fd8b60
 *	California, Berkeley and its contributors.
Packit fd8b60
 * 4. Neither the name of the University nor the names of its contributors
Packit fd8b60
 *    may be used to endorse or promote products derived from this software
Packit fd8b60
 *    without specific prior written permission.
Packit fd8b60
 *
Packit fd8b60
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
Packit fd8b60
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
Packit fd8b60
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
Packit fd8b60
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
Packit fd8b60
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
Packit fd8b60
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
Packit fd8b60
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
Packit fd8b60
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
Packit fd8b60
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
Packit fd8b60
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
Packit fd8b60
 * SUCH DAMAGE.
Packit fd8b60
 */
Packit fd8b60
Packit fd8b60
#if defined(LIBC_SCCS) && !defined(lint)
Packit fd8b60
static char sccsid[] = "@(#)mpool.c	8.7 (Berkeley) 11/2/95";
Packit fd8b60
#endif /* LIBC_SCCS and not lint */
Packit fd8b60
Packit fd8b60
#include <sys/param.h>
Packit fd8b60
#include <sys/stat.h>
Packit fd8b60
Packit fd8b60
#include <errno.h>
Packit fd8b60
#include <stdio.h>
Packit fd8b60
#include <stdlib.h>
Packit fd8b60
#include <string.h>
Packit fd8b60
#include <unistd.h>
Packit fd8b60
Packit fd8b60
#include "db-int.h"
Packit fd8b60
#include "mpool.h"
Packit fd8b60
Packit fd8b60
static BKT *mpool_bkt __P((MPOOL *));
Packit fd8b60
static BKT *mpool_look __P((MPOOL *, db_pgno_t));
Packit fd8b60
static int  mpool_write __P((MPOOL *, BKT *));
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_open --
Packit fd8b60
 *	Initialize a memory pool.
Packit fd8b60
 */
Packit fd8b60
MPOOL *
Packit fd8b60
mpool_open(key, fd, pagesize, maxcache)
Packit fd8b60
	void *key;
Packit fd8b60
	int fd;
Packit fd8b60
	db_pgno_t pagesize, maxcache;
Packit fd8b60
{
Packit fd8b60
	struct stat sb;
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	int entry;
Packit fd8b60
Packit fd8b60
	/*
Packit fd8b60
	 * Get information about the file.
Packit fd8b60
	 *
Packit fd8b60
	 * XXX
Packit fd8b60
	 * We don't currently handle pipes, although we should.
Packit fd8b60
	 */
Packit fd8b60
	if (fstat(fd, &sb))
Packit fd8b60
		return (NULL);
Packit fd8b60
	if (!S_ISREG(sb.st_mode)) {
Packit fd8b60
		errno = ESPIPE;
Packit fd8b60
		return (NULL);
Packit fd8b60
	}
Packit fd8b60
Packit fd8b60
	/* Allocate and initialize the MPOOL cookie. */
Packit fd8b60
	if ((mp = (MPOOL *)calloc(1, sizeof(MPOOL))) == NULL)
Packit fd8b60
		return (NULL);
Packit fd8b60
	TAILQ_INIT(&mp->lqh);
Packit fd8b60
	for (entry = 0; entry < HASHSIZE; ++entry)
Packit fd8b60
		TAILQ_INIT(&mp->hqh[entry]);
Packit fd8b60
	mp->maxcache = maxcache;
Packit fd8b60
	mp->npages = sb.st_size / pagesize;
Packit fd8b60
	mp->pagesize = pagesize;
Packit fd8b60
	mp->fd = fd;
Packit fd8b60
	return (mp);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_filter --
Packit fd8b60
 *	Initialize input/output filters.
Packit fd8b60
 */
Packit fd8b60
void
Packit fd8b60
mpool_filter(mp, pgin, pgout, pgcookie)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	void (*pgin) __P((void *, db_pgno_t, void *));
Packit fd8b60
	void (*pgout) __P((void *, db_pgno_t, void *));
Packit fd8b60
	void *pgcookie;
Packit fd8b60
{
Packit fd8b60
	mp->pgin = pgin;
Packit fd8b60
	mp->pgout = pgout;
Packit fd8b60
	mp->pgcookie = pgcookie;
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_new --
Packit fd8b60
 *	Get a new page of memory.
Packit fd8b60
 */
Packit fd8b60
void *
Packit fd8b60
mpool_new(mp, pgnoaddr, flags)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	db_pgno_t *pgnoaddr;
Packit fd8b60
	u_int flags;
Packit fd8b60
{
Packit fd8b60
	struct _hqh *head;
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
	if (mp->npages == MAX_PAGE_NUMBER) {
Packit fd8b60
		(void)fprintf(stderr, "mpool_new: page allocation overflow.\n");
Packit fd8b60
		abort();
Packit fd8b60
	}
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->pagenew;
Packit fd8b60
#endif
Packit fd8b60
	/*
Packit fd8b60
	 * Get a BKT from the cache.  Assign a new page number, attach
Packit fd8b60
	 * it to the head of the hash chain, the tail of the lru chain,
Packit fd8b60
	 * and return.
Packit fd8b60
	 */
Packit fd8b60
	if ((bp = mpool_bkt(mp)) == NULL)
Packit fd8b60
		return (NULL);
Packit fd8b60
	if (flags == MPOOL_PAGE_REQUEST) {
Packit fd8b60
		mp->npages++;
Packit fd8b60
		bp->pgno = *pgnoaddr;
Packit fd8b60
	} else
Packit fd8b60
		bp->pgno = *pgnoaddr = mp->npages++;
Packit fd8b60
Packit fd8b60
	bp->flags = MPOOL_PINNED | MPOOL_INUSE;
Packit fd8b60
Packit fd8b60
	head = &mp->hqh[HASHKEY(bp->pgno)];
Packit fd8b60
	TAILQ_INSERT_HEAD(head, bp, hq);
Packit fd8b60
	TAILQ_INSERT_TAIL(&mp->lqh, bp, q);
Packit fd8b60
	return (bp->page);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
int
Packit fd8b60
mpool_delete(mp, page)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	void *page;
Packit fd8b60
{
Packit fd8b60
	struct _hqh *head;
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
	bp = (void *)((char *)page - sizeof(BKT));
Packit fd8b60
Packit fd8b60
#ifdef DEBUG
Packit fd8b60
	if (!(bp->flags & MPOOL_PINNED)) {
Packit fd8b60
		(void)fprintf(stderr,
Packit fd8b60
		    "mpool_delete: page %d not pinned\n", bp->pgno);
Packit fd8b60
		abort();
Packit fd8b60
	}
Packit fd8b60
#endif
Packit fd8b60
Packit fd8b60
	/* Remove from the hash and lru queues. */
Packit fd8b60
	head = &mp->hqh[HASHKEY(bp->pgno)];
Packit fd8b60
	TAILQ_REMOVE(head, bp, hq);
Packit fd8b60
	TAILQ_REMOVE(&mp->lqh, bp, q);
Packit fd8b60
Packit fd8b60
	free(bp);
Packit fd8b60
	return (RET_SUCCESS);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_get
Packit fd8b60
 *	Get a page.
Packit fd8b60
 */
Packit fd8b60
void *
Packit fd8b60
mpool_get(mp, pgno, flags)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	db_pgno_t pgno;
Packit fd8b60
	u_int flags;				/* XXX not used? */
Packit fd8b60
{
Packit fd8b60
	struct _hqh *head;
Packit fd8b60
	BKT *bp;
Packit fd8b60
	off_t off;
Packit fd8b60
	int nr;
Packit fd8b60
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->pageget;
Packit fd8b60
#endif
Packit fd8b60
Packit fd8b60
	/* Check for a page that is cached. */
Packit fd8b60
	if ((bp = mpool_look(mp, pgno)) != NULL) {
Packit fd8b60
#ifdef DEBUG
Packit fd8b60
		if (!(flags & MPOOL_IGNOREPIN) && bp->flags & MPOOL_PINNED) {
Packit fd8b60
			(void)fprintf(stderr,
Packit fd8b60
			    "mpool_get: page %d already pinned\n", bp->pgno);
Packit fd8b60
			abort();
Packit fd8b60
		}
Packit fd8b60
#endif
Packit fd8b60
		/*
Packit fd8b60
		 * Move the page to the head of the hash chain and the tail
Packit fd8b60
		 * of the lru chain.
Packit fd8b60
		 */
Packit fd8b60
		head = &mp->hqh[HASHKEY(bp->pgno)];
Packit fd8b60
		TAILQ_REMOVE(head, bp, hq);
Packit fd8b60
		TAILQ_INSERT_HEAD(head, bp, hq);
Packit fd8b60
		TAILQ_REMOVE(&mp->lqh, bp, q);
Packit fd8b60
		TAILQ_INSERT_TAIL(&mp->lqh, bp, q);
Packit fd8b60
Packit fd8b60
		/* Return a pinned page. */
Packit fd8b60
		if (!(flags & MPOOL_IGNOREPIN))
Packit fd8b60
			bp->flags |= MPOOL_PINNED;
Packit fd8b60
		return (bp->page);
Packit fd8b60
	}
Packit fd8b60
Packit fd8b60
	/* Get a page from the cache. */
Packit fd8b60
	if ((bp = mpool_bkt(mp)) == NULL)
Packit fd8b60
		return (NULL);
Packit fd8b60
Packit fd8b60
	/* Read in the contents. */
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->pageread;
Packit fd8b60
#endif
Packit fd8b60
	off = mp->pagesize * pgno;
Packit fd8b60
	if (off / mp->pagesize != pgno) {
Packit fd8b60
	    /* Run past the end of the file, or at least the part we
Packit fd8b60
	       can address without large-file support?  */
Packit fd8b60
	    errno = E2BIG;
Packit fd8b60
	    return NULL;
Packit fd8b60
	}
Packit fd8b60
	if (lseek(mp->fd, off, SEEK_SET) != off)
Packit fd8b60
		return (NULL);
Packit fd8b60
Packit fd8b60
	if ((nr = read(mp->fd, bp->page, mp->pagesize)) !=
Packit fd8b60
	    (ssize_t)mp->pagesize) {
Packit fd8b60
		if (nr > 0) {
Packit fd8b60
			/* A partial read is definitely bad. */
Packit fd8b60
			errno = EINVAL;
Packit fd8b60
			return (NULL);
Packit fd8b60
		} else {
Packit fd8b60
			/*
Packit fd8b60
			 * A zero-length reads, means you need to create a
Packit fd8b60
			 * new page.
Packit fd8b60
			 */
Packit fd8b60
			memset(bp->page, 0, mp->pagesize);
Packit fd8b60
		}
Packit fd8b60
	}
Packit fd8b60
Packit fd8b60
	/* Set the page number, pin the page. */
Packit fd8b60
	bp->pgno = pgno;
Packit fd8b60
	if (!(flags & MPOOL_IGNOREPIN))
Packit fd8b60
		bp->flags = MPOOL_PINNED;
Packit fd8b60
	bp->flags |= MPOOL_INUSE;
Packit fd8b60
Packit fd8b60
	/*
Packit fd8b60
	 * Add the page to the head of the hash chain and the tail
Packit fd8b60
	 * of the lru chain.
Packit fd8b60
	 */
Packit fd8b60
	head = &mp->hqh[HASHKEY(bp->pgno)];
Packit fd8b60
	TAILQ_INSERT_HEAD(head, bp, hq);
Packit fd8b60
	TAILQ_INSERT_TAIL(&mp->lqh, bp, q);
Packit fd8b60
Packit fd8b60
	/* Run through the user's filter. */
Packit fd8b60
	if (mp->pgin != NULL)
Packit fd8b60
		(mp->pgin)(mp->pgcookie, bp->pgno, bp->page);
Packit fd8b60
Packit fd8b60
	return (bp->page);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_put
Packit fd8b60
 *	Return a page.
Packit fd8b60
 */
Packit fd8b60
int
Packit fd8b60
mpool_put(mp, page, flags)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	void *page;
Packit fd8b60
	u_int flags;
Packit fd8b60
{
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->pageput;
Packit fd8b60
#endif
Packit fd8b60
	bp = (void *)((char *)page - sizeof(BKT));
Packit fd8b60
#ifdef DEBUG
Packit fd8b60
	if (!(bp->flags & MPOOL_PINNED)) {
Packit fd8b60
		(void)fprintf(stderr,
Packit fd8b60
		    "mpool_put: page %d not pinned\n", bp->pgno);
Packit fd8b60
		abort();
Packit fd8b60
	}
Packit fd8b60
#endif
Packit fd8b60
	bp->flags &= ~MPOOL_PINNED;
Packit fd8b60
	if (flags & MPOOL_DIRTY)
Packit fd8b60
		bp->flags |= flags & MPOOL_DIRTY;
Packit fd8b60
	return (RET_SUCCESS);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_close
Packit fd8b60
 *	Close the buffer pool.
Packit fd8b60
 */
Packit fd8b60
int
Packit fd8b60
mpool_close(mp)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
{
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
	/* Free up any space allocated to the lru pages. */
Packit fd8b60
	while ((bp = mp->lqh.tqh_first) != NULL) {
Packit fd8b60
		TAILQ_REMOVE(&mp->lqh, mp->lqh.tqh_first, q);
Packit fd8b60
		free(bp);
Packit fd8b60
	}
Packit fd8b60
Packit fd8b60
	/* Free the MPOOL cookie. */
Packit fd8b60
	free(mp);
Packit fd8b60
	return (RET_SUCCESS);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_sync
Packit fd8b60
 *	Sync the pool to disk.
Packit fd8b60
 */
Packit fd8b60
int
Packit fd8b60
mpool_sync(mp)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
{
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
	/* Walk the lru chain, flushing any dirty pages to disk. */
Packit fd8b60
	for (bp = mp->lqh.tqh_first; bp != NULL; bp = bp->q.tqe_next)
Packit fd8b60
		if (bp->flags & MPOOL_DIRTY &&
Packit fd8b60
		    mpool_write(mp, bp) == RET_ERROR)
Packit fd8b60
			return (RET_ERROR);
Packit fd8b60
Packit fd8b60
	/* Sync the file descriptor. */
Packit fd8b60
	return (fsync(mp->fd) ? RET_ERROR : RET_SUCCESS);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_bkt
Packit fd8b60
 *	Get a page from the cache (or create one).
Packit fd8b60
 */
Packit fd8b60
static BKT *
Packit fd8b60
mpool_bkt(mp)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
{
Packit fd8b60
	struct _hqh *head;
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
	/* If under the max cached, always create a new page. */
Packit fd8b60
	if (mp->curcache < mp->maxcache)
Packit fd8b60
		goto new;
Packit fd8b60
Packit fd8b60
	/*
Packit fd8b60
	 * If the cache is max'd out, walk the lru list for a buffer we
Packit fd8b60
	 * can flush.  If we find one, write it (if necessary) and take it
Packit fd8b60
	 * off any lists.  If we don't find anything we grow the cache anyway.
Packit fd8b60
	 * The cache never shrinks.
Packit fd8b60
	 */
Packit fd8b60
	for (bp = mp->lqh.tqh_first; bp != NULL; bp = bp->q.tqe_next)
Packit fd8b60
		if (!(bp->flags & MPOOL_PINNED)) {
Packit fd8b60
			/* Flush if dirty. */
Packit fd8b60
			if (bp->flags & MPOOL_DIRTY &&
Packit fd8b60
			    mpool_write(mp, bp) == RET_ERROR)
Packit fd8b60
				return (NULL);
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
			++mp->pageflush;
Packit fd8b60
#endif
Packit fd8b60
			/* Remove from the hash and lru queues. */
Packit fd8b60
			head = &mp->hqh[HASHKEY(bp->pgno)];
Packit fd8b60
			TAILQ_REMOVE(head, bp, hq);
Packit fd8b60
			TAILQ_REMOVE(&mp->lqh, bp, q);
Packit fd8b60
#if defined(DEBUG) && !defined(DEBUG_IDX0SPLIT)
Packit fd8b60
			{ void *spage;
Packit fd8b60
				spage = bp->page;
Packit fd8b60
				memset(bp, 0xff, sizeof(BKT) + mp->pagesize);
Packit fd8b60
				bp->page = spage;
Packit fd8b60
			}
Packit fd8b60
#endif
Packit fd8b60
			bp->flags = 0;
Packit fd8b60
			return (bp);
Packit fd8b60
		}
Packit fd8b60
Packit fd8b60
new:	if ((bp = (BKT *)malloc(sizeof(BKT) + mp->pagesize)) == NULL)
Packit fd8b60
		return (NULL);
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->pagealloc;
Packit fd8b60
#endif
Packit fd8b60
#if defined(DEBUG) || defined(PURIFY) || 1
Packit fd8b60
	memset(bp, 0xff, sizeof(BKT) + mp->pagesize);
Packit fd8b60
#endif
Packit fd8b60
	bp->page = (char *)bp + sizeof(BKT);
Packit fd8b60
	bp->flags = 0;
Packit fd8b60
	++mp->curcache;
Packit fd8b60
	return (bp);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_write
Packit fd8b60
 *	Write a page to disk.
Packit fd8b60
 */
Packit fd8b60
static int
Packit fd8b60
mpool_write(mp, bp)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	BKT *bp;
Packit fd8b60
{
Packit fd8b60
	off_t off;
Packit fd8b60
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->pagewrite;
Packit fd8b60
#endif
Packit fd8b60
Packit fd8b60
	/* Run through the user's filter. */
Packit fd8b60
	if (mp->pgout)
Packit fd8b60
		(mp->pgout)(mp->pgcookie, bp->pgno, bp->page);
Packit fd8b60
Packit fd8b60
	off = mp->pagesize * bp->pgno;
Packit fd8b60
	if (off / mp->pagesize != bp->pgno) {
Packit fd8b60
	    /* Run past the end of the file, or at least the part we
Packit fd8b60
	       can address without large-file support?  */
Packit fd8b60
	    errno = E2BIG;
Packit fd8b60
	    return RET_ERROR;
Packit fd8b60
	}
Packit fd8b60
	if (lseek(mp->fd, off, SEEK_SET) != off)
Packit fd8b60
		return (RET_ERROR);
Packit fd8b60
	if (write(mp->fd, bp->page, mp->pagesize) !=
Packit fd8b60
	    (ssize_t)mp->pagesize)
Packit fd8b60
		return (RET_ERROR);
Packit fd8b60
Packit fd8b60
	/*
Packit fd8b60
	 * Re-run through the input filter since this page may soon be
Packit fd8b60
	 * accessed via the cache, and whatever the user's output filter
Packit fd8b60
	 * did may screw things up if we don't let the input filter
Packit fd8b60
	 * restore the in-core copy.
Packit fd8b60
	 */
Packit fd8b60
	if (mp->pgin)
Packit fd8b60
		(mp->pgin)(mp->pgcookie, bp->pgno, bp->page);
Packit fd8b60
	bp->flags &= ~MPOOL_DIRTY;
Packit fd8b60
	return (RET_SUCCESS);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
/*
Packit fd8b60
 * mpool_look
Packit fd8b60
 *	Lookup a page in the cache.
Packit fd8b60
 */
Packit fd8b60
static BKT *
Packit fd8b60
mpool_look(mp, pgno)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
	db_pgno_t pgno;
Packit fd8b60
{
Packit fd8b60
	struct _hqh *head;
Packit fd8b60
	BKT *bp;
Packit fd8b60
Packit fd8b60
	head = &mp->hqh[HASHKEY(pgno)];
Packit fd8b60
	for (bp = head->tqh_first; bp != NULL; bp = bp->hq.tqe_next)
Packit fd8b60
		if ((bp->pgno == pgno) && (bp->flags & MPOOL_INUSE)) {
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
			++mp->cachehit;
Packit fd8b60
#endif
Packit fd8b60
			return (bp);
Packit fd8b60
		}
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
	++mp->cachemiss;
Packit fd8b60
#endif
Packit fd8b60
	return (NULL);
Packit fd8b60
}
Packit fd8b60
Packit fd8b60
#ifdef STATISTICS
Packit fd8b60
/*
Packit fd8b60
 * mpool_stat
Packit fd8b60
 *	Print out cache statistics.
Packit fd8b60
 */
Packit fd8b60
void
Packit fd8b60
mpool_stat(mp)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
{
Packit fd8b60
	BKT *bp;
Packit fd8b60
	int cnt;
Packit fd8b60
	char *sep;
Packit fd8b60
Packit fd8b60
	(void)fprintf(stderr, "%lu pages in the file\n", mp->npages);
Packit fd8b60
	(void)fprintf(stderr,
Packit fd8b60
	    "page size %lu, cacheing %lu pages of %lu page max cache\n",
Packit fd8b60
	    mp->pagesize, mp->curcache, mp->maxcache);
Packit fd8b60
	(void)fprintf(stderr, "%lu page puts, %lu page gets, %lu page new\n",
Packit fd8b60
	    mp->pageput, mp->pageget, mp->pagenew);
Packit fd8b60
	(void)fprintf(stderr, "%lu page allocs, %lu page flushes\n",
Packit fd8b60
	    mp->pagealloc, mp->pageflush);
Packit fd8b60
	if (mp->cachehit + mp->cachemiss)
Packit fd8b60
		(void)fprintf(stderr,
Packit fd8b60
		    "%.0f%% cache hit rate (%lu hits, %lu misses)\n",
Packit fd8b60
		    ((double)mp->cachehit / (mp->cachehit + mp->cachemiss))
Packit fd8b60
		    * 100, mp->cachehit, mp->cachemiss);
Packit fd8b60
	(void)fprintf(stderr, "%lu page reads, %lu page writes\n",
Packit fd8b60
	    mp->pageread, mp->pagewrite);
Packit fd8b60
Packit fd8b60
	sep = "";
Packit fd8b60
	cnt = 0;
Packit fd8b60
	for (bp = mp->lqh.tqh_first; bp != NULL; bp = bp->q.tqe_next) {
Packit fd8b60
		(void)fprintf(stderr, "%s%d", sep, bp->pgno);
Packit fd8b60
		if (bp->flags & MPOOL_DIRTY)
Packit fd8b60
			(void)fprintf(stderr, "d");
Packit fd8b60
		if (bp->flags & MPOOL_PINNED)
Packit fd8b60
			(void)fprintf(stderr, "P");
Packit fd8b60
		if (++cnt == 10) {
Packit fd8b60
			sep = "\n";
Packit fd8b60
			cnt = 0;
Packit fd8b60
		} else
Packit fd8b60
			sep = ", ";
Packit fd8b60
Packit fd8b60
	}
Packit fd8b60
	(void)fprintf(stderr, "\n");
Packit fd8b60
}
Packit fd8b60
#else
Packit fd8b60
void
Packit fd8b60
mpool_stat(mp)
Packit fd8b60
	MPOOL *mp;
Packit fd8b60
{
Packit fd8b60
}
Packit fd8b60
#endif