[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.158.2.2 and 1.188

version 1.158.2.2, 2009/05/04 08:13:48 version 1.188, 2011/01/17 07:36:58
Line 1 
Line 1 
 /*      $NetBSD$        */  /*      $NetBSD$        */
   
 /*-  /*-
  * Copyright (c) 1997, 1999, 2000, 2002, 2007, 2008 The NetBSD Foundation, Inc.   * Copyright (c) 1997, 1999, 2000, 2002, 2007, 2008, 2010
    *     The NetBSD Foundation, Inc.
  * All rights reserved.   * All rights reserved.
  *   *
  * This code is derived from software contributed to The NetBSD Foundation   * This code is derived from software contributed to The NetBSD Foundation
Line 53  __KERNEL_RCSID(0, "$NetBSD$");
Line 54  __KERNEL_RCSID(0, "$NetBSD$");
 #include <sys/cpu.h>  #include <sys/cpu.h>
 #include <sys/atomic.h>  #include <sys/atomic.h>
   
 #include <uvm/uvm.h>  #include <uvm/uvm_extern.h>
   #ifdef DIAGNOSTIC
   #include <uvm/uvm_km.h> /* uvm_km_va_drain */
   #endif
   
 /*  /*
  * Pool resource management utility.   * Pool resource management utility.
Line 69  __KERNEL_RCSID(0, "$NetBSD$");
Line 73  __KERNEL_RCSID(0, "$NetBSD$");
  */   */
   
 /* List of all pools */  /* List of all pools */
 TAILQ_HEAD(,pool) pool_head = TAILQ_HEAD_INITIALIZER(pool_head);  static TAILQ_HEAD(, pool) pool_head = TAILQ_HEAD_INITIALIZER(pool_head);
   
 /* Private pool for page header structures */  /* Private pool for page header structures */
 #define PHPOOL_MAX      8  #define PHPOOL_MAX      8
Line 104  static struct pool *drainpp;
Line 108  static struct pool *drainpp;
 static kmutex_t pool_head_lock;  static kmutex_t pool_head_lock;
 static kcondvar_t pool_busy;  static kcondvar_t pool_busy;
   
   /* This lock protects initialization of a potentially shared pool allocator */
   static kmutex_t pool_allocator_lock;
   
 typedef uint32_t pool_item_bitmap_t;  typedef uint32_t pool_item_bitmap_t;
 #define BITMAP_SIZE     (CHAR_BIT * sizeof(pool_item_bitmap_t))  #define BITMAP_SIZE     (CHAR_BIT * sizeof(pool_item_bitmap_t))
 #define BITMAP_MASK     (BITMAP_SIZE - 1)  #define BITMAP_MASK     (BITMAP_SIZE - 1)
Line 188  static bool pool_cache_get_slow(pool_cac
Line 195  static bool pool_cache_get_slow(pool_cac
                                     void **, paddr_t *, int);                                      void **, paddr_t *, int);
 static void     pool_cache_cpu_init1(struct cpu_info *, pool_cache_t);  static void     pool_cache_cpu_init1(struct cpu_info *, pool_cache_t);
 static void     pool_cache_invalidate_groups(pool_cache_t, pcg_t *);  static void     pool_cache_invalidate_groups(pool_cache_t, pcg_t *);
   static void     pool_cache_invalidate_cpu(pool_cache_t, u_int);
 static void     pool_cache_xcall(pool_cache_t);  static void     pool_cache_xcall(pool_cache_t);
   
 static int      pool_catchup(struct pool *);  static int      pool_catchup(struct pool *);
Line 230  int pool_logsize = POOL_LOGSIZE;
Line 238  int pool_logsize = POOL_LOGSIZE;
 static inline void  static inline void
 pr_log(struct pool *pp, void *v, int action, const char *file, long line)  pr_log(struct pool *pp, void *v, int action, const char *file, long line)
 {  {
         int n = pp->pr_curlogentry;          int n;
         struct pool_log *pl;          struct pool_log *pl;
   
         if ((pp->pr_roflags & PR_LOGGING) == 0)          if ((pp->pr_roflags & PR_LOGGING) == 0)
                 return;                  return;
   
           if (pp->pr_log == NULL) {
                   if (kmem_map != NULL)
                           pp->pr_log = malloc(
                                   pool_logsize * sizeof(struct pool_log),
                                   M_TEMP, M_NOWAIT | M_ZERO);
                   if (pp->pr_log == NULL)
                           return;
                   pp->pr_curlogentry = 0;
                   pp->pr_logsize = pool_logsize;
           }
   
         /*          /*
          * Fill in the current entry. Wrap around and overwrite           * Fill in the current entry. Wrap around and overwrite
          * the oldest entry if necessary.           * the oldest entry if necessary.
          */           */
           n = pp->pr_curlogentry;
         pl = &pp->pr_log[n];          pl = &pp->pr_log[n];
         pl->pl_file = file;          pl->pl_file = file;
         pl->pl_line = line;          pl->pl_line = line;
Line 257  pr_printlog(struct pool *pp, struct pool
Line 277  pr_printlog(struct pool *pp, struct pool
         int i = pp->pr_logsize;          int i = pp->pr_logsize;
         int n = pp->pr_curlogentry;          int n = pp->pr_curlogentry;
   
         if ((pp->pr_roflags & PR_LOGGING) == 0)          if (pp->pr_log == NULL)
                 return;                  return;
   
         /*          /*
Line 545  pool_reclaim_register(struct pool *pp)
Line 565  pool_reclaim_register(struct pool *pp)
         callback_register(&vm_map_to_kernel(map)->vmk_reclaim_callback,          callback_register(&vm_map_to_kernel(map)->vmk_reclaim_callback,
             &pp->pr_reclaimerentry, pp, pool_reclaim_callback);              &pp->pr_reclaimerentry, pp, pool_reclaim_callback);
         splx(s);          splx(s);
   
   #ifdef DIAGNOSTIC
           /* Diagnostic drain attempt. */
           uvm_km_va_drain(map, 0);
   #endif
 }  }
   
 static void  static void
Line 587  void
Line 612  void
 pool_subsystem_init(void)  pool_subsystem_init(void)
 {  {
         struct pool_allocator *pa;          struct pool_allocator *pa;
         __link_set_decl(pools, struct link_pool_init);  
         struct link_pool_init * const *pi;  
   
         mutex_init(&pool_head_lock, MUTEX_DEFAULT, IPL_NONE);          mutex_init(&pool_head_lock, MUTEX_DEFAULT, IPL_NONE);
           mutex_init(&pool_allocator_lock, MUTEX_DEFAULT, IPL_NONE);
         cv_init(&pool_busy, "poolbusy");          cv_init(&pool_busy, "poolbusy");
   
         __link_set_foreach(pi, pools)  
                 pool_init((*pi)->pp, (*pi)->size, (*pi)->align,  
                     (*pi)->align_offset, (*pi)->flags, (*pi)->wchan,  
                     (*pi)->palloc, (*pi)->ipl);  
   
         while ((pa = SLIST_FIRST(&pa_deferinitq)) != NULL) {          while ((pa = SLIST_FIRST(&pa_deferinitq)) != NULL) {
                 KASSERT(pa->pa_backingmapptr != NULL);                  KASSERT(pa->pa_backingmapptr != NULL);
                 KASSERT(*pa->pa_backingmapptr != NULL);                  KASSERT(*pa->pa_backingmapptr != NULL);
Line 656  pool_init(struct pool *pp, size_t size, 
Line 675  pool_init(struct pool *pp, size_t size, 
                         palloc = &pool_allocator_nointr_fullpage;                          palloc = &pool_allocator_nointr_fullpage;
         }          }
 #endif /* POOL_SUBPAGE */  #endif /* POOL_SUBPAGE */
         if ((palloc->pa_flags & PA_INITIALIZED) == 0) {          if (!cold)
                   mutex_enter(&pool_allocator_lock);
           if (palloc->pa_refcnt++ == 0) {
                 if (palloc->pa_pagesz == 0)                  if (palloc->pa_pagesz == 0)
                         palloc->pa_pagesz = PAGE_SIZE;                          palloc->pa_pagesz = PAGE_SIZE;
   
Line 669  pool_init(struct pool *pp, size_t size, 
Line 690  pool_init(struct pool *pp, size_t size, 
                 if (palloc->pa_backingmapptr != NULL) {                  if (palloc->pa_backingmapptr != NULL) {
                         pa_reclaim_register(palloc);                          pa_reclaim_register(palloc);
                 }                  }
                 palloc->pa_flags |= PA_INITIALIZED;  
         }          }
           if (!cold)
                   mutex_exit(&pool_allocator_lock);
   
         if (align == 0)          if (align == 0)
                 align = ALIGN(1);                  align = ALIGN(1);
Line 793  pool_init(struct pool *pp, size_t size, 
Line 815  pool_init(struct pool *pp, size_t size, 
         pp->pr_nidle = 0;          pp->pr_nidle = 0;
         pp->pr_refcnt = 0;          pp->pr_refcnt = 0;
   
 #ifdef POOL_DIAGNOSTIC          pp->pr_log = NULL;
         if (flags & PR_LOGGING) {  
                 if (kmem_map == NULL ||  
                     (pp->pr_log = malloc(pool_logsize * sizeof(struct pool_log),  
                      M_TEMP, M_NOWAIT)) == NULL)  
                         pp->pr_roflags &= ~PR_LOGGING;  
                 pp->pr_curlogentry = 0;  
                 pp->pr_logsize = pool_logsize;  
         }  
 #endif  
   
         pp->pr_entered_file = NULL;          pp->pr_entered_file = NULL;
         pp->pr_entered_line = 0;          pp->pr_entered_line = 0;
Line 851  pool_init(struct pool *pp, size_t size, 
Line 864  pool_init(struct pool *pp, size_t size, 
         }          }
   
         /* Insert into the list of all pools. */          /* Insert into the list of all pools. */
         if (__predict_true(!cold))          if (!cold)
                 mutex_enter(&pool_head_lock);                  mutex_enter(&pool_head_lock);
         TAILQ_FOREACH(pp1, &pool_head, pr_poollist) {          TAILQ_FOREACH(pp1, &pool_head, pr_poollist) {
                 if (strcmp(pp1->pr_wchan, pp->pr_wchan) > 0)                  if (strcmp(pp1->pr_wchan, pp->pr_wchan) > 0)
Line 861  pool_init(struct pool *pp, size_t size, 
Line 874  pool_init(struct pool *pp, size_t size, 
                 TAILQ_INSERT_TAIL(&pool_head, pp, pr_poollist);                  TAILQ_INSERT_TAIL(&pool_head, pp, pr_poollist);
         else          else
                 TAILQ_INSERT_BEFORE(pp1, pp, pr_poollist);                  TAILQ_INSERT_BEFORE(pp1, pp, pr_poollist);
         if (__predict_true(!cold))          if (!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))          if (!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))          if (!cold)
                 mutex_exit(&palloc->pa_lock);                  mutex_exit(&palloc->pa_lock);
   
         pool_reclaim_register(pp);          pool_reclaim_register(pp);
Line 898  pool_destroy(struct pool *pp)
Line 911  pool_destroy(struct pool *pp)
         TAILQ_REMOVE(&pp->pr_alloc->pa_list, pp, pr_alloc_list);          TAILQ_REMOVE(&pp->pr_alloc->pa_list, pp, pr_alloc_list);
         mutex_exit(&pp->pr_alloc->pa_lock);          mutex_exit(&pp->pr_alloc->pa_lock);
   
           mutex_enter(&pool_allocator_lock);
           if (--pp->pr_alloc->pa_refcnt == 0)
                   mutex_destroy(&pp->pr_alloc->pa_lock);
           mutex_exit(&pool_allocator_lock);
   
         mutex_enter(&pp->pr_lock);          mutex_enter(&pp->pr_lock);
   
         KASSERT(pp->pr_cache == NULL);          KASSERT(pp->pr_cache == NULL);
Line 923  pool_destroy(struct pool *pp)
Line 941  pool_destroy(struct pool *pp)
         pr_pagelist_free(pp, &pq);          pr_pagelist_free(pp, &pq);
   
 #ifdef POOL_DIAGNOSTIC  #ifdef POOL_DIAGNOSTIC
         if ((pp->pr_roflags & PR_LOGGING) != 0)          if (pp->pr_log != NULL) {
                 free(pp->pr_log, M_TEMP);                  free(pp->pr_log, M_TEMP);
                   pp->pr_log = NULL;
           }
 #endif  #endif
   
         cv_destroy(&pp->pr_cv);          cv_destroy(&pp->pr_cv);
Line 972  pool_get(struct pool *pp, int flags)
Line 992  pool_get(struct pool *pp, int flags)
         void *v;          void *v;
   
 #ifdef DIAGNOSTIC  #ifdef DIAGNOSTIC
         if (__predict_false(pp->pr_itemsperpage == 0))          if (pp->pr_itemsperpage == 0)
                 panic("pool_get: pool %p: pr_itemsperpage is zero, "                  panic("pool_get: pool '%s': pr_itemsperpage is zero, "
                     "pool not initialized?", pp);                      "pool not initialized?", pp->pr_wchan);
         if (__predict_false(curlwp == NULL && doing_shutdown == 0 &&          if ((cpu_intr_p() || cpu_softintr_p()) && pp->pr_ipl == IPL_NONE &&
                             (flags & PR_WAITOK) != 0))              !cold && panicstr == NULL)
                 panic("pool_get: %s: must have NOWAIT", pp->pr_wchan);                  panic("pool '%s' is IPL_NONE, but called from "
                       "interrupt context\n", pp->pr_wchan);
 #endif /* DIAGNOSTIC */  #endif
 #ifdef LOCKDEBUG  
         if (flags & PR_WAITOK) {          if (flags & PR_WAITOK) {
                 ASSERT_SLEEPABLE();                  ASSERT_SLEEPABLE();
         }          }
 #endif  
   
         mutex_enter(&pp->pr_lock);          mutex_enter(&pp->pr_lock);
         pr_enter(pp, file, line);          pr_enter(pp, file, line);
Line 1592  pool_sethardlimit(struct pool *pp, int n
Line 1610  pool_sethardlimit(struct pool *pp, int n
   
 /*  /*
  * Release all complete pages that have not been used recently.   * Release all complete pages that have not been used recently.
    *
    * Might be called from interrupt context.
  */   */
 int  int
 #ifdef POOL_DIAGNOSTIC  #ifdef POOL_DIAGNOSTIC
Line 1606  pool_reclaim(struct pool *pp)
Line 1626  pool_reclaim(struct pool *pp)
         bool klock;          bool klock;
         int rv;          int rv;
   
           if (cpu_intr_p() || cpu_softintr_p()) {
                   KASSERT(pp->pr_ipl != IPL_NONE);
           }
   
         if (pp->pr_drain_hook != NULL) {          if (pp->pr_drain_hook != NULL) {
                 /*                  /*
                  * The drain hook must be called with the pool unlocked.                   * The drain hook must be called with the pool unlocked.
Line 1724  pool_drain_start(struct pool **ppp, uint
Line 1748  pool_drain_start(struct pool **ppp, uint
         }          }
 }  }
   
 void  bool
 pool_drain_end(struct pool *pp, uint64_t where)  pool_drain_end(struct pool *pp, uint64_t where)
 {  {
           bool reclaimed;
   
         if (pp == NULL)          if (pp == NULL)
                 return;                  return false;
   
         KASSERT(pp->pr_refcnt > 0);          KASSERT(pp->pr_refcnt > 0);
   
Line 1738  pool_drain_end(struct pool *pp, uint64_t
Line 1763  pool_drain_end(struct pool *pp, uint64_t
                 xc_wait(where);                  xc_wait(where);
   
         /* Drain the cache (if any) and pool.. */          /* Drain the cache (if any) and pool.. */
         pool_reclaim(pp);          reclaimed = pool_reclaim(pp);
   
         /* Finally, unlock the pool. */          /* Finally, unlock the pool. */
         mutex_enter(&pool_head_lock);          mutex_enter(&pool_head_lock);
         pp->pr_refcnt--;          pp->pr_refcnt--;
         cv_broadcast(&pool_busy);          cv_broadcast(&pool_busy);
         mutex_exit(&pool_head_lock);          mutex_exit(&pool_head_lock);
   
           return reclaimed;
 }  }
   
 /*  /*
Line 1893  pool_print1(struct pool *pp, const char 
Line 1920  pool_print1(struct pool *pp, const char 
         if (pc != NULL) {          if (pc != NULL) {
                 cpuhit = 0;                  cpuhit = 0;
                 cpumiss = 0;                  cpumiss = 0;
                 for (i = 0; i < MAXCPUS; i++) {                  for (i = 0; i < __arraycount(pc->pc_cpus); i++) {
                         if ((cc = pc->pc_cpus[i]) == NULL)                          if ((cc = pc->pc_cpus[i]) == NULL)
                                 continue;                                  continue;
                         cpuhit += cc->cc_hits;                          cpuhit += cc->cc_hits;
Line 2129  void
Line 2156  void
 pool_cache_destroy(pool_cache_t pc)  pool_cache_destroy(pool_cache_t pc)
 {  {
         struct pool *pp = &pc->pc_pool;          struct pool *pp = &pc->pc_pool;
         pool_cache_cpu_t *cc;          u_int i;
         pcg_t *pcg;  
         int i;  
   
         /* Remove it from the global list. */          /* Remove it from the global list. */
         mutex_enter(&pool_head_lock);          mutex_enter(&pool_head_lock);
Line 2149  pool_cache_destroy(pool_cache_t pc)
Line 2174  pool_cache_destroy(pool_cache_t pc)
         mutex_exit(&pp->pr_lock);          mutex_exit(&pp->pr_lock);
   
         /* Destroy per-CPU data */          /* Destroy per-CPU data */
         for (i = 0; i < MAXCPUS; i++) {          for (i = 0; i < __arraycount(pc->pc_cpus); i++)
                 if ((cc = pc->pc_cpus[i]) == NULL)                  pool_cache_invalidate_cpu(pc, i);
                         continue;  
                 if ((pcg = cc->cc_current) != &pcg_dummy) {  
                         pcg->pcg_next = NULL;  
                         pool_cache_invalidate_groups(pc, pcg);  
                 }  
                 if ((pcg = cc->cc_previous) != &pcg_dummy) {  
                         pcg->pcg_next = NULL;  
                         pool_cache_invalidate_groups(pc, pcg);  
                 }  
                 if (cc != &pc->pc_cpu0)  
                         pool_put(&cache_cpu_pool, cc);  
         }  
   
         /* Finally, destroy it. */          /* Finally, destroy it. */
         mutex_destroy(&pc->pc_lock);          mutex_destroy(&pc->pc_lock);
Line 2183  pool_cache_cpu_init1(struct cpu_info *ci
Line 2196  pool_cache_cpu_init1(struct cpu_info *ci
   
         index = ci->ci_index;          index = ci->ci_index;
   
         KASSERT(index < MAXCPUS);          KASSERT(index < __arraycount(pc->pc_cpus));
   
         if ((cc = pc->pc_cpus[index]) != NULL) {          if ((cc = pc->pc_cpus[index]) != NULL) {
                 KASSERT(cc->cc_cpuindex == index);                  KASSERT(cc->cc_cpuindex == index);
Line 2309  pool_cache_invalidate_groups(pool_cache_
Line 2322  pool_cache_invalidate_groups(pool_cache_
  *   *
  *      Invalidate a pool cache (destruct and release all of the   *      Invalidate a pool cache (destruct and release all of the
  *      cached objects).  Does not reclaim objects from the pool.   *      cached objects).  Does not reclaim objects from the pool.
    *
    *      Note: For pool caches that provide constructed objects, there
    *      is an assumption that another level of synchronization is occurring
    *      between the input to the constructor and the cache invalidation.
  */   */
 void  void
 pool_cache_invalidate(pool_cache_t pc)  pool_cache_invalidate(pool_cache_t pc)
 {  {
         pcg_t *full, *empty, *part;          pcg_t *full, *empty, *part;
   #if 0
           uint64_t where;
   
           if (ncpu < 2 || !mp_online) {
                   /*
                    * We might be called early enough in the boot process
                    * for the CPU data structures to not be fully initialized.
                    * In this case, simply gather the local CPU's cache now
                    * since it will be the only one running.
                    */
                   pool_cache_xcall(pc);
           } else {
                   /*
                    * Gather all of the CPU-specific caches into the
                    * global cache.
                    */
                   where = xc_broadcast(0, (xcfunc_t)pool_cache_xcall, pc, NULL);
                   xc_wait(where);
           }
   #endif
         mutex_enter(&pc->pc_lock);          mutex_enter(&pc->pc_lock);
         full = pc->pc_fullgroups;          full = pc->pc_fullgroups;
         empty = pc->pc_emptygroups;          empty = pc->pc_emptygroups;
Line 2332  pool_cache_invalidate(pool_cache_t pc)
Line 2368  pool_cache_invalidate(pool_cache_t pc)
         pool_cache_invalidate_groups(pc, part);          pool_cache_invalidate_groups(pc, part);
 }  }
   
   /*
    * pool_cache_invalidate_cpu:
    *
    *      Invalidate all CPU-bound cached objects in pool cache, the CPU being
    *      identified by its associated index.
    *      It is caller's responsibility to ensure that no operation is
    *      taking place on this pool cache while doing this invalidation.
    *      WARNING: as no inter-CPU locking is enforced, trying to invalidate
    *      pool cached objects from a CPU different from the one currently running
    *      may result in an undefined behaviour.
    */
   static void
   pool_cache_invalidate_cpu(pool_cache_t pc, u_int index)
   {
   
           pool_cache_cpu_t *cc;
           pcg_t *pcg;
   
           if ((cc = pc->pc_cpus[index]) == NULL)
                   return;
   
           if ((pcg = cc->cc_current) != &pcg_dummy) {
                   pcg->pcg_next = NULL;
                   pool_cache_invalidate_groups(pc, pcg);
           }
           if ((pcg = cc->cc_previous) != &pcg_dummy) {
                   pcg->pcg_next = NULL;
                   pool_cache_invalidate_groups(pc, pcg);
           }
           if (cc != &pc->pc_cpu0)
                   pool_put(&cache_cpu_pool, cc);
   
   }
   
 void  void
 pool_cache_set_drain_hook(pool_cache_t pc, void (*fn)(void *, int), void *arg)  pool_cache_set_drain_hook(pool_cache_t pc, void (*fn)(void *, int), void *arg)
 {  {
Line 2465  pool_cache_get_paddr(pool_cache_t pc, in
Line 2535  pool_cache_get_paddr(pool_cache_t pc, in
         void *object;          void *object;
         int s;          int s;
   
 #ifdef LOCKDEBUG          KASSERTMSG((!cpu_intr_p() && !cpu_softintr_p()) ||
               (pc->pc_pool.pr_ipl != IPL_NONE || cold || panicstr != NULL),
               ("pool '%s' is IPL_NONE, but called from interrupt context\n",
               pc->pc_pool.pr_wchan));
   
         if (flags & PR_WAITOK) {          if (flags & PR_WAITOK) {
                 ASSERT_SLEEPABLE();                  ASSERT_SLEEPABLE();
         }          }
 #endif  
   
         /* Lock out interrupts and disable preemption. */          /* Lock out interrupts and disable preemption. */
         s = splvm();          s = splvm();
Line 2985  found:
Line 3058  found:
                                         goto print;                                          goto print;
                                 }                                  }
                         }                          }
                         for (i = 0; i < MAXCPUS; i++) {                          for (i = 0; i < __arraycount(pc->pc_cpus); i++) {
                                 pool_cache_cpu_t *cc;                                  pool_cache_cpu_t *cc;
   
                                 if ((cc = pc->pc_cpus[i]) == NULL) {                                  if ((cc = pc->pc_cpus[i]) == NULL) {

Legend:
Removed from v.1.158.2.2  
changed lines
  Added in v.1.188

CVSweb <webmaster@jp.NetBSD.org>