[BACK]Return to subr_pool.c CVS log [TXT][DIR] Up to [cvs.NetBSD.org] / src / sys / kern

Please note that diffs are not public domain; they are subject to the copyright notices on the relevant files.

Diff for /src/sys/kern/subr_pool.c between version 1.101.2.11 and 1.144

version 1.101.2.11, 2008/02/27 08:36:56 version 1.144, 2007/12/22 04:31:34
Line 52  __KERNEL_RCSID(0, "$NetBSD$");
Line 52  __KERNEL_RCSID(0, "$NetBSD$");
 #include <sys/errno.h>  #include <sys/errno.h>
 #include <sys/kernel.h>  #include <sys/kernel.h>
 #include <sys/malloc.h>  #include <sys/malloc.h>
   #include <sys/lock.h>
 #include <sys/pool.h>  #include <sys/pool.h>
 #include <sys/syslog.h>  #include <sys/syslog.h>
 #include <sys/debug.h>  #include <sys/debug.h>
 #include <sys/lockdebug.h>  #include <sys/lockdebug.h>
 #include <sys/xcall.h>  #include <sys/xcall.h>
 #include <sys/cpu.h>  #include <sys/cpu.h>
 #include <sys/atomic.h>  
   
 #include <uvm/uvm.h>  #include <uvm/uvm.h>
   
Line 76  __KERNEL_RCSID(0, "$NetBSD$");
Line 76  __KERNEL_RCSID(0, "$NetBSD$");
  */   */
   
 /* List of all pools */  /* List of all pools */
 TAILQ_HEAD(,pool) pool_head = TAILQ_HEAD_INITIALIZER(pool_head);  LIST_HEAD(,pool) pool_head = LIST_HEAD_INITIALIZER(pool_head);
   
   /* List of all caches. */
   LIST_HEAD(,pool_cache) pool_cache_head =
       LIST_HEAD_INITIALIZER(pool_cache_head);
   
 /* Private pool for page header structures */  /* Private pool for page header structures */
 #define PHPOOL_MAX      8  #define PHPOOL_MAX      8
Line 122  struct pool_item_header {
Line 126  struct pool_item_header {
         SPLAY_ENTRY(pool_item_header)          SPLAY_ENTRY(pool_item_header)
                                 ph_node;        /* Off-page page headers */                                  ph_node;        /* Off-page page headers */
         void *                  ph_page;        /* this page's address */          void *                  ph_page;        /* this page's address */
         uint32_t                ph_time;        /* last referenced */          struct timeval          ph_time;        /* last referenced */
         uint16_t                ph_nmissing;    /* # of chunks in use */          uint16_t                ph_nmissing;    /* # of chunks in use */
         uint16_t                ph_off;         /* start offset in page */          uint16_t                ph_off;         /* start offset in page */
         union {          union {
Line 182  static struct pool pcg_large_pool;
Line 186  static struct pool pcg_large_pool;
 static struct pool cache_pool;  static struct pool cache_pool;
 static struct pool cache_cpu_pool;  static struct pool cache_cpu_pool;
   
 /* List of all caches. */  
 TAILQ_HEAD(,pool_cache) pool_cache_head =  
     TAILQ_HEAD_INITIALIZER(pool_cache_head);  
   
 int pool_cache_disable;  
   
   
 static pool_cache_cpu_t *pool_cache_put_slow(pool_cache_cpu_t *, int *,  static pool_cache_cpu_t *pool_cache_put_slow(pool_cache_cpu_t *, int *,
                                              void *, paddr_t);                                               void *, paddr_t);
 static pool_cache_cpu_t *pool_cache_get_slow(pool_cache_cpu_t *, int *,  static pool_cache_cpu_t *pool_cache_get_slow(pool_cache_cpu_t *, int *,
Line 629  void
Line 626  void
 pool_init(struct pool *pp, size_t size, u_int align, u_int ioff, int flags,  pool_init(struct pool *pp, size_t size, u_int align, u_int ioff, int flags,
     const char *wchan, struct pool_allocator *palloc, int ipl)      const char *wchan, struct pool_allocator *palloc, int ipl)
 {  {
   #ifdef DEBUG
         struct pool *pp1;          struct pool *pp1;
   #endif
         size_t trysize, phsize;          size_t trysize, phsize;
         int off, slack;          int off, slack;
   
Line 638  pool_init(struct pool *pp, size_t size, 
Line 637  pool_init(struct pool *pp, size_t size, 
          * Check that the pool hasn't already been initialised and           * Check that the pool hasn't already been initialised and
          * added to the list of all pools.           * added to the list of all pools.
          */           */
         TAILQ_FOREACH(pp1, &pool_head, pr_poollist) {          LIST_FOREACH(pp1, &pool_head, pr_poollist) {
                 if (pp == pp1)                  if (pp == pp1)
                         panic("pool_init: pool %s already initialised",                          panic("pool_init: pool %s already initialised",
                             wchan);                              wchan);
Line 864  pool_init(struct pool *pp, size_t size, 
Line 863  pool_init(struct pool *pp, size_t size, 
                     "pcglarge", &pool_allocator_meta, IPL_VM);                      "pcglarge", &pool_allocator_meta, IPL_VM);
         }          }
   
         /* Insert into the list of all pools. */          if (__predict_true(!cold)) {
         if (__predict_true(!cold))                  /* Insert into the list of all pools. */
                 mutex_enter(&pool_head_lock);                  mutex_enter(&pool_head_lock);
         TAILQ_FOREACH(pp1, &pool_head, pr_poollist) {                  LIST_INSERT_HEAD(&pool_head, pp, pr_poollist);
                 if (strcmp(pp1->pr_wchan, pp->pr_wchan) > 0)  
                         break;  
         }  
         if (pp1 == NULL)  
                 TAILQ_INSERT_TAIL(&pool_head, pp, pr_poollist);  
         else  
                 TAILQ_INSERT_BEFORE(pp1, pp, pr_poollist);  
         if (__predict_true(!cold))  
                 mutex_exit(&pool_head_lock);                  mutex_exit(&pool_head_lock);
   
                 /* Insert this into the list of pools using this allocator. */                  /* Insert this into the list of pools using this allocator. */
         if (__predict_true(!cold))  
                 mutex_enter(&palloc->pa_lock);                  mutex_enter(&palloc->pa_lock);
         TAILQ_INSERT_TAIL(&palloc->pa_list, pp, pr_alloc_list);                  TAILQ_INSERT_TAIL(&palloc->pa_list, pp, pr_alloc_list);
         if (__predict_true(!cold))  
                 mutex_exit(&palloc->pa_lock);                  mutex_exit(&palloc->pa_lock);
           } else {
                   LIST_INSERT_HEAD(&pool_head, pp, pr_poollist);
                   TAILQ_INSERT_TAIL(&palloc->pa_list, pp, pr_alloc_list);
           }
   
         pool_reclaim_register(pp);          pool_reclaim_register(pp);
 }  }
Line 901  pool_destroy(struct pool *pp)
Line 894  pool_destroy(struct pool *pp)
         mutex_enter(&pool_head_lock);          mutex_enter(&pool_head_lock);
         while (pp->pr_refcnt != 0)          while (pp->pr_refcnt != 0)
                 cv_wait(&pool_busy, &pool_head_lock);                  cv_wait(&pool_busy, &pool_head_lock);
         TAILQ_REMOVE(&pool_head, pp, pr_poollist);          LIST_REMOVE(pp, pr_poollist);
         if (drainpp == pp)          if (drainpp == pp)
                 drainpp = NULL;                  drainpp = NULL;
         mutex_exit(&pool_head_lock);          mutex_exit(&pool_head_lock);
Line 1300  pool_do_put(struct pool *pp, void *v, st
Line 1293  pool_do_put(struct pool *pp, void *v, st
                          * be idle for some period of time before it can                           * be idle for some period of time before it can
                          * be reclaimed by the pagedaemon.  This minimizes                           * be reclaimed by the pagedaemon.  This minimizes
                          * ping-pong'ing for memory.                           * ping-pong'ing for memory.
                          *  
                          * note for 64-bit time_t: truncating to 32-bit is not  
                          * a problem for our usage.  
                          */                           */
                         ph->ph_time = time_uptime;                          getmicrotime(&ph->ph_time);
                 }                  }
                 pool_update_curpage(pp);                  pool_update_curpage(pp);
         }          }
Line 1456  pool_prime_page(struct pool *pp, void *s
Line 1446  pool_prime_page(struct pool *pp, void *s
         LIST_INIT(&ph->ph_itemlist);          LIST_INIT(&ph->ph_itemlist);
         ph->ph_page = storage;          ph->ph_page = storage;
         ph->ph_nmissing = 0;          ph->ph_nmissing = 0;
         ph->ph_time = time_uptime;          getmicrotime(&ph->ph_time);
         if ((pp->pr_roflags & PR_PHINPAGE) == 0)          if ((pp->pr_roflags & PR_PHINPAGE) == 0)
                 SPLAY_INSERT(phtree, &pp->pr_phtree, ph);                  SPLAY_INSERT(phtree, &pp->pr_phtree, ph);
   
Line 1617  pool_reclaim(struct pool *pp)
Line 1607  pool_reclaim(struct pool *pp)
 {  {
         struct pool_item_header *ph, *phnext;          struct pool_item_header *ph, *phnext;
         struct pool_pagelist pq;          struct pool_pagelist pq;
         uint32_t curtime;          struct timeval curtime, diff;
         bool klock;          bool klock;
         int rv;          int rv;
   
Line 1654  pool_reclaim(struct pool *pp)
Line 1644  pool_reclaim(struct pool *pp)
   
         LIST_INIT(&pq);          LIST_INIT(&pq);
   
         curtime = time_uptime;          getmicrotime(&curtime);
   
         for (ph = LIST_FIRST(&pp->pr_emptypages); ph != NULL; ph = phnext) {          for (ph = LIST_FIRST(&pp->pr_emptypages); ph != NULL; ph = phnext) {
                 phnext = LIST_NEXT(ph, ph_pagelist);                  phnext = LIST_NEXT(ph, ph_pagelist);
Line 1664  pool_reclaim(struct pool *pp)
Line 1654  pool_reclaim(struct pool *pp)
                         break;                          break;
   
                 KASSERT(ph->ph_nmissing == 0);                  KASSERT(ph->ph_nmissing == 0);
                 if (curtime - ph->ph_time < pool_inactive_time                  timersub(&curtime, &ph->ph_time, &diff);
                   if (diff.tv_sec < pool_inactive_time
                     && !pa_starved_p(pp->pr_alloc))                      && !pa_starved_p(pp->pr_alloc))
                         continue;                          continue;
   
Line 1710  pool_drain_start(struct pool **ppp, uint
Line 1701  pool_drain_start(struct pool **ppp, uint
 {  {
         struct pool *pp;          struct pool *pp;
   
         KASSERT(!TAILQ_EMPTY(&pool_head));          KASSERT(!LIST_EMPTY(&pool_head));
   
         pp = NULL;          pp = NULL;
   
Line 1718  pool_drain_start(struct pool **ppp, uint
Line 1709  pool_drain_start(struct pool **ppp, uint
         mutex_enter(&pool_head_lock);          mutex_enter(&pool_head_lock);
         do {          do {
                 if (drainpp == NULL) {                  if (drainpp == NULL) {
                         drainpp = TAILQ_FIRST(&pool_head);                          drainpp = LIST_FIRST(&pool_head);
                 }                  }
                 if (drainpp != NULL) {                  if (drainpp != NULL) {
                         pp = drainpp;                          pp = drainpp;
                         drainpp = TAILQ_NEXT(pp, pr_poollist);                          drainpp = LIST_NEXT(pp, pr_poollist);
                 }                  }
                 /*                  /*
                  * Skip completely idle pools.  We depend on at least                   * Skip completely idle pools.  We depend on at least
Line 1778  pool_printall(const char *modif, void (*
Line 1769  pool_printall(const char *modif, void (*
 {  {
         struct pool *pp;          struct pool *pp;
   
         TAILQ_FOREACH(pp, &pool_head, pr_poollist) {          LIST_FOREACH(pp, &pool_head, pr_poollist) {
                 pool_printit(pp, modif, pr);                  pool_printit(pp, modif, pr);
         }          }
 }  }
Line 1805  pool_print_pagelist(struct pool *pp, str
Line 1796  pool_print_pagelist(struct pool *pp, str
 #endif  #endif
   
         LIST_FOREACH(ph, pl, ph_pagelist) {          LIST_FOREACH(ph, pl, ph_pagelist) {
                 (*pr)("\t\tpage %p, nmissing %d, time %" PRIu32 "\n",                  (*pr)("\t\tpage %p, nmissing %d, time %lu,%lu\n",
                     ph->ph_page, ph->ph_nmissing, ph->ph_time);                      ph->ph_page, ph->ph_nmissing,
                       (u_long)ph->ph_time.tv_sec,
                       (u_long)ph->ph_time.tv_usec);
 #ifdef DIAGNOSTIC  #ifdef DIAGNOSTIC
                 if (!(pp->pr_roflags & PR_NOTOUCH)) {                  if (!(pp->pr_roflags & PR_NOTOUCH)) {
                         LIST_FOREACH(pi, &ph->ph_itemlist, pi_list) {                          LIST_FOREACH(pi, &ph->ph_itemlist, pi_list) {
Line 2066  pool_cache_bootstrap(pool_cache_t pc, si
Line 2059  pool_cache_bootstrap(pool_cache_t pc, si
     void *arg)      void *arg)
 {  {
         CPU_INFO_ITERATOR cii;          CPU_INFO_ITERATOR cii;
         pool_cache_t pc1;  
         struct cpu_info *ci;          struct cpu_info *ci;
         struct pool *pp;          struct pool *pp;
   
Line 2123  pool_cache_bootstrap(pool_cache_t pc, si
Line 2115  pool_cache_bootstrap(pool_cache_t pc, si
                         pool_cache_cpu_init1(ci, pc);                          pool_cache_cpu_init1(ci, pc);
                 }                  }
         }          }
   
         /* Add to list of all pools. */          if (__predict_true(!cold)) {
         if (__predict_true(!cold))                  mutex_enter(&pp->pr_lock);
                   pp->pr_cache = pc;
                   mutex_exit(&pp->pr_lock);
                 mutex_enter(&pool_head_lock);                  mutex_enter(&pool_head_lock);
         TAILQ_FOREACH(pc1, &pool_cache_head, pc_cachelist) {                  LIST_INSERT_HEAD(&pool_cache_head, pc, pc_cachelist);
                 if (strcmp(pc1->pc_pool.pr_wchan, pc->pc_pool.pr_wchan) > 0)  
                         break;  
         }  
         if (pc1 == NULL)  
                 TAILQ_INSERT_TAIL(&pool_cache_head, pc, pc_cachelist);  
         else  
                 TAILQ_INSERT_BEFORE(pc1, pc, pc_cachelist);  
         if (__predict_true(!cold))  
                 mutex_exit(&pool_head_lock);                  mutex_exit(&pool_head_lock);
           } else {
         membar_sync();                  pp->pr_cache = pc;
         pp->pr_cache = pc;                  LIST_INSERT_HEAD(&pool_cache_head, pc, pc_cachelist);
           }
 }  }
   
 /*  /*
Line 2159  pool_cache_destroy(pool_cache_t pc)
Line 2146  pool_cache_destroy(pool_cache_t pc)
         mutex_enter(&pool_head_lock);          mutex_enter(&pool_head_lock);
         while (pc->pc_refcnt != 0)          while (pc->pc_refcnt != 0)
                 cv_wait(&pool_busy, &pool_head_lock);                  cv_wait(&pool_busy, &pool_head_lock);
         TAILQ_REMOVE(&pool_cache_head, pc, pc_cachelist);          LIST_REMOVE(pc, pc_cachelist);
         mutex_exit(&pool_head_lock);          mutex_exit(&pool_head_lock);
   
         /* First, invalidate the entire cache. */          /* First, invalidate the entire cache. */
Line 2250  pool_cache_cpu_init(struct cpu_info *ci)
Line 2237  pool_cache_cpu_init(struct cpu_info *ci)
         pool_cache_t pc;          pool_cache_t pc;
   
         mutex_enter(&pool_head_lock);          mutex_enter(&pool_head_lock);
         TAILQ_FOREACH(pc, &pool_cache_head, pc_cachelist) {          LIST_FOREACH(pc, &pool_cache_head, pc_cachelist) {
                 pc->pc_refcnt++;                  pc->pc_refcnt++;
                 mutex_exit(&pool_head_lock);                  mutex_exit(&pool_head_lock);
   
Line 2534  pool_cache_get_paddr(pool_cache_t pc, in
Line 2521  pool_cache_get_paddr(pool_cache_t pc, in
                         object = pcg->pcg_objects[--pcg->pcg_avail].pcgo_va;                          object = pcg->pcg_objects[--pcg->pcg_avail].pcgo_va;
                         if (pap != NULL)                          if (pap != NULL)
                                 *pap = pcg->pcg_objects[pcg->pcg_avail].pcgo_pa;                                  *pap = pcg->pcg_objects[pcg->pcg_avail].pcgo_pa;
 #if defined(DIAGNOSTIC)  
                         pcg->pcg_objects[pcg->pcg_avail].pcgo_va = NULL;                          pcg->pcg_objects[pcg->pcg_avail].pcgo_va = NULL;
 #endif /* defined(DIAGNOSTIC) */  
                         KASSERT(pcg->pcg_avail <= pcg->pcg_size);                          KASSERT(pcg->pcg_avail <= pcg->pcg_size);
                         KASSERT(object != NULL);                          KASSERT(object != NULL);
                         cc->cc_hits++;                          cc->cc_hits++;
Line 2607  pool_cache_put_slow(pool_cache_cpu_t *cc
Line 2592  pool_cache_put_slow(pool_cache_cpu_t *cc
                 /*                  /*
                  * If there's a empty group, release our full                   * If there's a empty group, release our full
                  * group back to the cache.  Install the empty                   * group back to the cache.  Install the empty
                  * group and return.                   * group as cc_current and return.
                  */                   */
                   if ((cur = cc->cc_current) != NULL) {
                           KASSERT(cur->pcg_avail == pcg->pcg_size);
                           cur->pcg_next = pc->pc_fullgroups;
                           pc->pc_fullgroups = cur;
                           pc->pc_nfull++;
                   }
                 KASSERT(pcg->pcg_avail == 0);                  KASSERT(pcg->pcg_avail == 0);
                   cc->cc_current = pcg;
                 pc->pc_emptygroups = pcg->pcg_next;                  pc->pc_emptygroups = pcg->pcg_next;
                 if (cc->cc_previous == NULL) {  
                         cc->cc_previous = pcg;  
                 } else {  
                         if ((cur = cc->cc_current) != NULL) {  
                                 KASSERT(cur->pcg_avail == pcg->pcg_size);  
                                 cur->pcg_next = pc->pc_fullgroups;  
                                 pc->pc_fullgroups = cur;  
                                 pc->pc_nfull++;  
                         }  
                         cc->cc_current = pcg;  
                 }  
                 pc->pc_hits++;                  pc->pc_hits++;
                 pc->pc_nempty--;                  pc->pc_nempty--;
                 mutex_exit(&pc->pc_lock);                  mutex_exit(&pc->pc_lock);
Line 2642  pool_cache_put_slow(pool_cache_cpu_t *cc
Line 2623  pool_cache_put_slow(pool_cache_cpu_t *cc
          * object away.           * object away.
          */           */
         nobj = pc->pc_pcgsize;          nobj = pc->pc_pcgsize;
         if (pool_cache_disable) {          if (nobj == PCG_NOBJECTS_LARGE) {
                 pcg = NULL;  
         } else if (nobj == PCG_NOBJECTS_LARGE) {  
                 pcg = pool_get(&pcg_large_pool, PR_NOWAIT);                  pcg = pool_get(&pcg_large_pool, PR_NOWAIT);
         } else {          } else {
                 pcg = pool_get(&pcg_normal_pool, PR_NOWAIT);                  pcg = pool_get(&pcg_normal_pool, PR_NOWAIT);
Line 3000  pool_whatis(uintptr_t addr, void (*pr)(c
Line 2979  pool_whatis(uintptr_t addr, void (*pr)(c
 {  {
         struct pool *pp;          struct pool *pp;
   
         TAILQ_FOREACH(pp, &pool_head, pr_poollist) {          LIST_FOREACH(pp, &pool_head, pr_poollist) {
                 struct pool_item_header *ph;                  struct pool_item_header *ph;
                 uintptr_t item;                  uintptr_t item;
                 bool allocated = true;                  bool allocated = true;

Legend:
Removed from v.1.101.2.11  
changed lines
  Added in v.1.144

CVSweb <webmaster@jp.NetBSD.org>