4afaaefa05
factor of 10x improvement on SMP system due to reduced lock contention. This may put me in the ballpark of what is needed. We can still further improve things on NUMA systems by creating an additional L3 cache per memory node instead of the current global pool. With luck this won't be needed. I should also take another look at the locking now that everything is working. There's a good chance I can tighten it up a little bit and improve things a little more. kmem_lock: time (sec) slabs objs hash kmem_lock: tot/max/calc tot/max/calc size/depth kmem_lock: 0.000999926 6/6/1 192/192/32 32768/0 kmem_lock: 0.000999926 4/4/2 128/128/64 32768/0 kmem_lock: 0.000999926 4/4/4 128/128/128 32768/0 kmem_lock: 0.000999926 4/4/8 128/128/256 32768/0 kmem_lock: 0.000999926 4/4/16 128/128/512 32768/0 kmem_lock: 0.000999926 4/4/32 128/128/1024 32768/0 kmem_lock: 0.000999926 4/4/64 128/128/2048 32768/0 kmem_lock: 0.000999926 8/8/128 256/256/4096 32768/0 kmem_lock: 0.003999704 24/23/256 768/736/8192 32768/1 kmem_lock: 0.012999038 44/41/512 1408/1312/16384 32768/1 kmem_lock: 0.051996153 96/93/1024 3072/2976/32768 32768/2 kmem_lock: 0.181986536 187/184/2048 5984/5888/65536 32768/3 kmem_lock: 0.655951469 342/339/4096 10944/10848/131072 32768/4 git-svn-id: https://outreach.scidac.gov/svn/spl/trunk@136 7e1ea52c-4ff2-0310-8f11-9dd32ca42a1c
1046 lines
28 KiB
C
1046 lines
28 KiB
C
/*
|
|
* This file is part of the SPL: Solaris Porting Layer.
|
|
*
|
|
* Copyright (c) 2008 Lawrence Livermore National Security, LLC.
|
|
* Produced at Lawrence Livermore National Laboratory
|
|
* Written by:
|
|
* Brian Behlendorf <behlendorf1@llnl.gov>,
|
|
* Herb Wartens <wartens2@llnl.gov>,
|
|
* Jim Garlick <garlick@llnl.gov>
|
|
* UCRL-CODE-235197
|
|
*
|
|
* This 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 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This 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.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <sys/kmem.h>
|
|
|
|
#ifdef DEBUG_SUBSYSTEM
|
|
#undef DEBUG_SUBSYSTEM
|
|
#endif
|
|
|
|
#define DEBUG_SUBSYSTEM S_KMEM
|
|
|
|
/*
|
|
* Memory allocation interfaces and debugging for basic kmem_*
|
|
* and vmem_* style memory allocation. When DEBUG_KMEM is enable
|
|
* all allocations will be tracked when they are allocated and
|
|
* freed. When the SPL module is unload a list of all leaked
|
|
* addresses and where they were allocated will be dumped to the
|
|
* console. Enabling this feature has a significant impant on
|
|
* performance but it makes finding memory leaks staight forward.
|
|
*/
|
|
#ifdef DEBUG_KMEM
|
|
/* Shim layer memory accounting */
|
|
atomic64_t kmem_alloc_used;
|
|
unsigned long kmem_alloc_max = 0;
|
|
atomic64_t vmem_alloc_used;
|
|
unsigned long vmem_alloc_max = 0;
|
|
int kmem_warning_flag = 1;
|
|
atomic64_t kmem_cache_alloc_failed;
|
|
|
|
spinlock_t kmem_lock;
|
|
struct hlist_head kmem_table[KMEM_TABLE_SIZE];
|
|
struct list_head kmem_list;
|
|
|
|
spinlock_t vmem_lock;
|
|
struct hlist_head vmem_table[VMEM_TABLE_SIZE];
|
|
struct list_head vmem_list;
|
|
|
|
EXPORT_SYMBOL(kmem_alloc_used);
|
|
EXPORT_SYMBOL(kmem_alloc_max);
|
|
EXPORT_SYMBOL(vmem_alloc_used);
|
|
EXPORT_SYMBOL(vmem_alloc_max);
|
|
EXPORT_SYMBOL(kmem_warning_flag);
|
|
|
|
EXPORT_SYMBOL(kmem_lock);
|
|
EXPORT_SYMBOL(kmem_table);
|
|
EXPORT_SYMBOL(kmem_list);
|
|
|
|
EXPORT_SYMBOL(vmem_lock);
|
|
EXPORT_SYMBOL(vmem_table);
|
|
EXPORT_SYMBOL(vmem_list);
|
|
|
|
int kmem_set_warning(int flag) { return (kmem_warning_flag = !!flag); }
|
|
#else
|
|
int kmem_set_warning(int flag) { return 0; }
|
|
#endif
|
|
EXPORT_SYMBOL(kmem_set_warning);
|
|
|
|
/*
|
|
* Slab allocation interfaces
|
|
*
|
|
* While the Linux slab implementation was inspired by the Solaris
|
|
* implemenation I cannot use it to emulate the Solaris APIs. I
|
|
* require two features which are not provided by the Linux slab.
|
|
*
|
|
* 1) Constructors AND destructors. Recent versions of the Linux
|
|
* kernel have removed support for destructors. This is a deal
|
|
* breaker for the SPL which contains particularly expensive
|
|
* initializers for mutex's, condition variables, etc. We also
|
|
* require a minimal level of cleaner for these data types unlike
|
|
* may Linux data type which do need to be explicitly destroyed.
|
|
*
|
|
* 2) Virtual address backed slab. Callers of the Solaris slab
|
|
* expect it to work well for both small are very large allocations.
|
|
* Because of memory fragmentation the Linux slab which is backed
|
|
* by kmalloc'ed memory performs very badly when confronted with
|
|
* large numbers of large allocations. Basing the slab on the
|
|
* virtual address space removes the need for contigeous pages
|
|
* and greatly improve performance for large allocations.
|
|
*
|
|
* For these reasons, the SPL has its own slab implementation with
|
|
* the needed features. It is not as highly optimized as either the
|
|
* Solaris or Linux slabs, but it should get me most of what is
|
|
* needed until it can be optimized or obsoleted by another approach.
|
|
*
|
|
* One serious concern I do have about this method is the relatively
|
|
* small virtual address space on 32bit arches. This will seriously
|
|
* constrain the size of the slab caches and their performance.
|
|
*
|
|
* XXX: Implement SPL proc interface to export full per cache stats.
|
|
*
|
|
* XXX: Implement work requests to keep an eye on each cache and
|
|
* shrink them via spl_slab_reclaim() when they are wasting lots
|
|
* of space. Currently this process is driven by the reapers.
|
|
*
|
|
* XXX: Implement proper small cache object support by embedding
|
|
* the spl_kmem_slab_t, spl_kmem_obj_t's, and objects in the
|
|
* allocated for a particular slab.
|
|
*
|
|
* XXX: Implement a resizable used object hash. Currently the hash
|
|
* is statically sized for thousands of objects but it should
|
|
* grow based on observed worst case slab depth.
|
|
*
|
|
* XXX: Improve the partial slab list by carefully maintaining a
|
|
* strict ordering of fullest to emptiest slabs based on
|
|
* the slab reference count. This gaurentees the when freeing
|
|
* slabs back to the system we need only linearly traverse the
|
|
* last N slabs in the list to discover all the freeable slabs.
|
|
*
|
|
* XXX: NUMA awareness for optionally allocating memory close to a
|
|
* particular core. This can be adventageous if you know the slab
|
|
* object will be short lived and primarily accessed from one core.
|
|
*
|
|
* XXX: Slab coloring may also yield performance improvements and would
|
|
* be desirable to implement.
|
|
*
|
|
* XXX: Proper hardware cache alignment would be good too.
|
|
*/
|
|
|
|
/* Ensure the __kmem_cache_create/__kmem_cache_destroy macros are
|
|
* removed here to prevent a recursive substitution, we want to call
|
|
* the native linux version.
|
|
*/
|
|
#undef kmem_cache_t
|
|
#undef kmem_cache_create
|
|
#undef kmem_cache_destroy
|
|
#undef kmem_cache_alloc
|
|
#undef kmem_cache_free
|
|
|
|
static struct list_head spl_kmem_cache_list; /* List of caches */
|
|
static struct rw_semaphore spl_kmem_cache_sem; /* Cache list lock */
|
|
static kmem_cache_t *spl_slab_cache; /* Cache for slab structs */
|
|
static kmem_cache_t *spl_obj_cache; /* Cache for obj structs */
|
|
|
|
static int spl_cache_flush(spl_kmem_cache_t *skc,
|
|
spl_kmem_magazine_t *skm, int flush);
|
|
|
|
#ifdef HAVE_SET_SHRINKER
|
|
static struct shrinker *spl_kmem_cache_shrinker;
|
|
#else
|
|
static int spl_kmem_cache_generic_shrinker(int nr_to_scan,
|
|
unsigned int gfp_mask);
|
|
static struct shrinker spl_kmem_cache_shrinker = {
|
|
.shrink = spl_kmem_cache_generic_shrinker,
|
|
.seeks = KMC_DEFAULT_SEEKS,
|
|
};
|
|
#endif
|
|
|
|
static spl_kmem_slab_t *
|
|
spl_slab_alloc(spl_kmem_cache_t *skc, int flags) {
|
|
spl_kmem_slab_t *sks;
|
|
spl_kmem_obj_t *sko, *n;
|
|
int i;
|
|
ENTRY;
|
|
|
|
sks = kmem_cache_alloc(spl_slab_cache, flags);
|
|
if (sks == NULL)
|
|
RETURN(sks);
|
|
|
|
sks->sks_magic = SKS_MAGIC;
|
|
sks->sks_objs = SPL_KMEM_CACHE_OBJ_PER_SLAB;
|
|
sks->sks_age = jiffies;
|
|
sks->sks_cache = skc;
|
|
INIT_LIST_HEAD(&sks->sks_list);
|
|
INIT_LIST_HEAD(&sks->sks_free_list);
|
|
sks->sks_ref = 0;
|
|
|
|
for (i = 0; i < sks->sks_objs; i++) {
|
|
sko = kmem_cache_alloc(spl_obj_cache, flags);
|
|
if (sko == NULL) {
|
|
out_alloc:
|
|
/* Unable to fully construct slab, objects,
|
|
* and object data buffers unwind everything.
|
|
*/
|
|
list_for_each_entry_safe(sko, n, &sks->sks_free_list,
|
|
sko_list) {
|
|
ASSERT(sko->sko_magic == SKO_MAGIC);
|
|
vmem_free(sko->sko_addr, skc->skc_obj_size);
|
|
list_del(&sko->sko_list);
|
|
kmem_cache_free(spl_obj_cache, sko);
|
|
}
|
|
|
|
kmem_cache_free(spl_slab_cache, sks);
|
|
GOTO(out, sks = NULL);
|
|
}
|
|
|
|
sko->sko_addr = vmem_alloc(skc->skc_obj_size, flags);
|
|
if (sko->sko_addr == NULL) {
|
|
kmem_cache_free(spl_obj_cache, sko);
|
|
GOTO(out_alloc, sks = NULL);
|
|
}
|
|
|
|
sko->sko_magic = SKO_MAGIC;
|
|
sko->sko_flags = 0;
|
|
sko->sko_slab = sks;
|
|
INIT_LIST_HEAD(&sko->sko_list);
|
|
INIT_HLIST_NODE(&sko->sko_hlist);
|
|
list_add(&sko->sko_list, &sks->sks_free_list);
|
|
}
|
|
out:
|
|
RETURN(sks);
|
|
}
|
|
|
|
/* Removes slab from complete or partial list, so it must
|
|
* be called with the 'skc->skc_lock' held.
|
|
* */
|
|
static void
|
|
spl_slab_free(spl_kmem_slab_t *sks) {
|
|
spl_kmem_cache_t *skc;
|
|
spl_kmem_obj_t *sko, *n;
|
|
int i = 0;
|
|
ENTRY;
|
|
|
|
ASSERT(sks->sks_magic == SKS_MAGIC);
|
|
ASSERT(sks->sks_ref == 0);
|
|
skc = sks->sks_cache;
|
|
skc->skc_obj_total -= sks->sks_objs;
|
|
skc->skc_slab_total--;
|
|
|
|
ASSERT(spin_is_locked(&skc->skc_lock));
|
|
|
|
list_for_each_entry_safe(sko, n, &sks->sks_free_list, sko_list) {
|
|
ASSERT(sko->sko_magic == SKO_MAGIC);
|
|
|
|
/* Run destructors for being freed */
|
|
if (skc->skc_dtor)
|
|
skc->skc_dtor(sko->sko_addr, skc->skc_private);
|
|
|
|
vmem_free(sko->sko_addr, skc->skc_obj_size);
|
|
list_del(&sko->sko_list);
|
|
kmem_cache_free(spl_obj_cache, sko);
|
|
i++;
|
|
}
|
|
|
|
ASSERT(sks->sks_objs == i);
|
|
list_del(&sks->sks_list);
|
|
kmem_cache_free(spl_slab_cache, sks);
|
|
|
|
EXIT;
|
|
}
|
|
|
|
static int
|
|
__spl_slab_reclaim(spl_kmem_cache_t *skc)
|
|
{
|
|
spl_kmem_slab_t *sks, *m;
|
|
int rc = 0;
|
|
ENTRY;
|
|
|
|
ASSERT(spin_is_locked(&skc->skc_lock));
|
|
/*
|
|
* Free empty slabs which have not been touched in skc_delay
|
|
* seconds. This delay time is important to avoid thrashing.
|
|
* Empty slabs will be at the end of the skc_partial_list.
|
|
*/
|
|
list_for_each_entry_safe_reverse(sks, m, &skc->skc_partial_list,
|
|
sks_list) {
|
|
if (sks->sks_ref > 0)
|
|
break;
|
|
|
|
if (time_after(jiffies, sks->sks_age + skc->skc_delay * HZ)) {
|
|
spl_slab_free(sks);
|
|
rc++;
|
|
}
|
|
}
|
|
|
|
/* Returns number of slabs reclaimed */
|
|
RETURN(rc);
|
|
}
|
|
|
|
static int
|
|
spl_slab_reclaim(spl_kmem_cache_t *skc)
|
|
{
|
|
int rc;
|
|
ENTRY;
|
|
|
|
spin_lock(&skc->skc_lock);
|
|
rc = __spl_slab_reclaim(skc);
|
|
spin_unlock(&skc->skc_lock);
|
|
|
|
RETURN(rc);
|
|
}
|
|
|
|
static int
|
|
spl_magazine_size(spl_kmem_cache_t *skc)
|
|
{
|
|
int size;
|
|
ENTRY;
|
|
|
|
/* Guesses for reasonable magazine sizes, they
|
|
* should really adapt based on observed usage. */
|
|
if (skc->skc_obj_size > (PAGE_SIZE * 256))
|
|
size = 1;
|
|
else if (skc->skc_obj_size > (PAGE_SIZE * 32))
|
|
size = 4;
|
|
else if (skc->skc_obj_size > (PAGE_SIZE))
|
|
size = 16;
|
|
else if (skc->skc_obj_size > (PAGE_SIZE / 4))
|
|
size = 32;
|
|
else if (skc->skc_obj_size > (PAGE_SIZE / 16))
|
|
size = 64;
|
|
else
|
|
size = 128;
|
|
|
|
RETURN(size);
|
|
}
|
|
|
|
static spl_kmem_magazine_t *
|
|
spl_magazine_alloc(spl_kmem_cache_t *skc, int node)
|
|
{
|
|
spl_kmem_magazine_t *skm;
|
|
int size = sizeof(spl_kmem_magazine_t) +
|
|
sizeof(void *) * skc->skc_mag_size;
|
|
ENTRY;
|
|
|
|
skm = kmalloc_node(size, GFP_KERNEL, node);
|
|
if (skm) {
|
|
skm->skm_magic = SKM_MAGIC;
|
|
skm->skm_avail = 0;
|
|
skm->skm_size = skc->skc_mag_size;
|
|
skm->skm_refill = skc->skc_mag_refill;
|
|
skm->skm_age = jiffies;
|
|
}
|
|
|
|
RETURN(skm);
|
|
}
|
|
|
|
static void
|
|
spl_magazine_free(spl_kmem_magazine_t *skm)
|
|
{
|
|
ENTRY;
|
|
ASSERT(skm->skm_magic == SKM_MAGIC);
|
|
ASSERT(skm->skm_avail == 0);
|
|
kfree(skm);
|
|
EXIT;
|
|
}
|
|
|
|
static int
|
|
spl_magazine_create(spl_kmem_cache_t *skc)
|
|
{
|
|
int i;
|
|
ENTRY;
|
|
|
|
skc->skc_mag_size = spl_magazine_size(skc);
|
|
skc->skc_mag_refill = (skc->skc_mag_size + 1) / 2;
|
|
|
|
for_each_online_cpu(i) {
|
|
skc->skc_mag[i] = spl_magazine_alloc(skc, cpu_to_node(i));
|
|
if (!skc->skc_mag[i]) {
|
|
for (i--; i >= 0; i--)
|
|
spl_magazine_free(skc->skc_mag[i]);
|
|
|
|
RETURN(-ENOMEM);
|
|
}
|
|
}
|
|
|
|
RETURN(0);
|
|
}
|
|
|
|
static void
|
|
spl_magazine_destroy(spl_kmem_cache_t *skc)
|
|
{
|
|
spl_kmem_magazine_t *skm;
|
|
int i;
|
|
ENTRY;
|
|
|
|
for_each_online_cpu(i) {
|
|
skm = skc->skc_mag[i];
|
|
(void)spl_cache_flush(skc, skm, skm->skm_avail);
|
|
spl_magazine_free(skm);
|
|
}
|
|
|
|
EXIT;
|
|
}
|
|
|
|
spl_kmem_cache_t *
|
|
spl_kmem_cache_create(char *name, size_t size, size_t align,
|
|
spl_kmem_ctor_t ctor,
|
|
spl_kmem_dtor_t dtor,
|
|
spl_kmem_reclaim_t reclaim,
|
|
void *priv, void *vmp, int flags)
|
|
{
|
|
spl_kmem_cache_t *skc;
|
|
int i, rc, kmem_flags = KM_SLEEP;
|
|
ENTRY;
|
|
|
|
/* We may be called when there is a non-zero preempt_count or
|
|
* interrupts are disabled is which case we must not sleep.
|
|
*/
|
|
if (current_thread_info()->preempt_count || irqs_disabled())
|
|
kmem_flags = KM_NOSLEEP;
|
|
|
|
/* Allocate new cache memory and initialize. */
|
|
skc = (spl_kmem_cache_t *)kmem_alloc(sizeof(*skc), kmem_flags);
|
|
if (skc == NULL)
|
|
RETURN(NULL);
|
|
|
|
skc->skc_magic = SKC_MAGIC;
|
|
skc->skc_name_size = strlen(name) + 1;
|
|
skc->skc_name = (char *)kmem_alloc(skc->skc_name_size, kmem_flags);
|
|
if (skc->skc_name == NULL) {
|
|
kmem_free(skc, sizeof(*skc));
|
|
RETURN(NULL);
|
|
}
|
|
strncpy(skc->skc_name, name, skc->skc_name_size);
|
|
|
|
skc->skc_ctor = ctor;
|
|
skc->skc_dtor = dtor;
|
|
skc->skc_reclaim = reclaim;
|
|
skc->skc_private = priv;
|
|
skc->skc_vmp = vmp;
|
|
skc->skc_flags = flags;
|
|
skc->skc_obj_size = size;
|
|
skc->skc_chunk_size = 0; /* XXX: Needed only when implementing */
|
|
skc->skc_slab_size = 0; /* small slab object optimizations */
|
|
skc->skc_max_chunks = 0; /* which are yet supported. */
|
|
skc->skc_delay = SPL_KMEM_CACHE_DELAY;
|
|
|
|
skc->skc_hash_bits = SPL_KMEM_CACHE_HASH_BITS;
|
|
skc->skc_hash_size = SPL_KMEM_CACHE_HASH_SIZE;
|
|
skc->skc_hash_elts = SPL_KMEM_CACHE_HASH_ELTS;
|
|
skc->skc_hash = (struct hlist_head *)
|
|
kmem_alloc(skc->skc_hash_size, kmem_flags);
|
|
if (skc->skc_hash == NULL) {
|
|
kmem_free(skc->skc_name, skc->skc_name_size);
|
|
kmem_free(skc, sizeof(*skc));
|
|
RETURN(NULL);
|
|
}
|
|
|
|
for (i = 0; i < skc->skc_hash_elts; i++)
|
|
INIT_HLIST_HEAD(&skc->skc_hash[i]);
|
|
|
|
INIT_LIST_HEAD(&skc->skc_list);
|
|
INIT_LIST_HEAD(&skc->skc_complete_list);
|
|
INIT_LIST_HEAD(&skc->skc_partial_list);
|
|
spin_lock_init(&skc->skc_lock);
|
|
skc->skc_slab_fail = 0;
|
|
skc->skc_slab_create = 0;
|
|
skc->skc_slab_destroy = 0;
|
|
skc->skc_slab_total = 0;
|
|
skc->skc_slab_alloc = 0;
|
|
skc->skc_slab_max = 0;
|
|
skc->skc_obj_total = 0;
|
|
skc->skc_obj_alloc = 0;
|
|
skc->skc_obj_max = 0;
|
|
skc->skc_hash_depth = 0;
|
|
skc->skc_hash_count = 0;
|
|
|
|
rc = spl_magazine_create(skc);
|
|
if (rc) {
|
|
kmem_free(skc->skc_hash, skc->skc_hash_size);
|
|
kmem_free(skc->skc_name, skc->skc_name_size);
|
|
kmem_free(skc, sizeof(*skc));
|
|
RETURN(NULL);
|
|
}
|
|
|
|
down_write(&spl_kmem_cache_sem);
|
|
list_add_tail(&skc->skc_list, &spl_kmem_cache_list);
|
|
up_write(&spl_kmem_cache_sem);
|
|
|
|
RETURN(skc);
|
|
}
|
|
EXPORT_SYMBOL(spl_kmem_cache_create);
|
|
|
|
/* The caller must ensure there are no racing calls to
|
|
* spl_kmem_cache_alloc() for this spl_kmem_cache_t.
|
|
*/
|
|
void
|
|
spl_kmem_cache_destroy(spl_kmem_cache_t *skc)
|
|
{
|
|
spl_kmem_slab_t *sks, *m;
|
|
ENTRY;
|
|
|
|
down_write(&spl_kmem_cache_sem);
|
|
list_del_init(&skc->skc_list);
|
|
up_write(&spl_kmem_cache_sem);
|
|
|
|
spl_magazine_destroy(skc);
|
|
spin_lock(&skc->skc_lock);
|
|
|
|
/* Validate there are no objects in use and free all the
|
|
* spl_kmem_slab_t, spl_kmem_obj_t, and object buffers. */
|
|
ASSERT(list_empty(&skc->skc_complete_list));
|
|
ASSERTF(skc->skc_hash_count == 0, "skc->skc_hash_count=%d\n",
|
|
skc->skc_hash_count);
|
|
|
|
list_for_each_entry_safe(sks, m, &skc->skc_partial_list, sks_list)
|
|
spl_slab_free(sks);
|
|
|
|
kmem_free(skc->skc_hash, skc->skc_hash_size);
|
|
kmem_free(skc->skc_name, skc->skc_name_size);
|
|
spin_unlock(&skc->skc_lock);
|
|
kmem_free(skc, sizeof(*skc));
|
|
|
|
EXIT;
|
|
}
|
|
EXPORT_SYMBOL(spl_kmem_cache_destroy);
|
|
|
|
/* The kernel provided hash_ptr() function behaves exceptionally badly
|
|
* when all the addresses are page aligned which is likely the case
|
|
* here. To avoid this issue shift off the low order non-random bits.
|
|
*/
|
|
static unsigned long
|
|
spl_hash_ptr(void *ptr, unsigned int bits)
|
|
{
|
|
return hash_long((unsigned long)ptr >> PAGE_SHIFT, bits);
|
|
}
|
|
|
|
static spl_kmem_obj_t *
|
|
spl_hash_obj(spl_kmem_cache_t *skc, void *obj)
|
|
{
|
|
struct hlist_node *node;
|
|
spl_kmem_obj_t *sko = NULL;
|
|
unsigned long key = spl_hash_ptr(obj, skc->skc_hash_bits);
|
|
int i = 0;
|
|
|
|
ASSERT(spin_is_locked(&skc->skc_lock));
|
|
|
|
hlist_for_each_entry(sko, node, &skc->skc_hash[key], sko_hlist) {
|
|
|
|
if (unlikely((++i) > skc->skc_hash_depth))
|
|
skc->skc_hash_depth = i;
|
|
|
|
if (sko->sko_addr == obj) {
|
|
ASSERT(sko->sko_magic == SKO_MAGIC);
|
|
RETURN(sko);
|
|
}
|
|
}
|
|
|
|
RETURN(NULL);
|
|
}
|
|
|
|
static void *
|
|
spl_cache_obj(spl_kmem_cache_t *skc, spl_kmem_slab_t *sks)
|
|
{
|
|
spl_kmem_obj_t *sko;
|
|
unsigned long key;
|
|
|
|
ASSERT(spin_is_locked(&skc->skc_lock));
|
|
|
|
sko = list_entry((&sks->sks_free_list)->next,spl_kmem_obj_t,sko_list);
|
|
ASSERT(sko->sko_magic == SKO_MAGIC);
|
|
ASSERT(sko->sko_addr != NULL);
|
|
|
|
/* Remove from sks_free_list and add to used hash */
|
|
list_del_init(&sko->sko_list);
|
|
key = spl_hash_ptr(sko->sko_addr, skc->skc_hash_bits);
|
|
hlist_add_head(&sko->sko_hlist, &skc->skc_hash[key]);
|
|
|
|
sks->sks_age = jiffies;
|
|
sks->sks_ref++;
|
|
skc->skc_obj_alloc++;
|
|
skc->skc_hash_count++;
|
|
|
|
/* Track max obj usage statistics */
|
|
if (skc->skc_obj_alloc > skc->skc_obj_max)
|
|
skc->skc_obj_max = skc->skc_obj_alloc;
|
|
|
|
/* Track max slab usage statistics */
|
|
if (sks->sks_ref == 1) {
|
|
skc->skc_slab_alloc++;
|
|
|
|
if (skc->skc_slab_alloc > skc->skc_slab_max)
|
|
skc->skc_slab_max = skc->skc_slab_alloc;
|
|
}
|
|
|
|
return sko->sko_addr;
|
|
}
|
|
|
|
/* No available objects create a new slab. Since this is an
|
|
* expensive operation we do it without holding the spinlock
|
|
* and only briefly aquire it when we link in the fully
|
|
* allocated and constructed slab.
|
|
*/
|
|
static spl_kmem_slab_t *
|
|
spl_cache_grow(spl_kmem_cache_t *skc, int flags)
|
|
{
|
|
spl_kmem_slab_t *sks;
|
|
spl_kmem_obj_t *sko;
|
|
ENTRY;
|
|
|
|
if (flags & __GFP_WAIT) {
|
|
flags |= __GFP_NOFAIL;
|
|
might_sleep();
|
|
local_irq_enable();
|
|
}
|
|
|
|
sks = spl_slab_alloc(skc, flags);
|
|
if (sks == NULL) {
|
|
if (flags & __GFP_WAIT)
|
|
local_irq_disable();
|
|
|
|
RETURN(NULL);
|
|
}
|
|
|
|
/* Run all the constructors now that the slab is fully allocated */
|
|
list_for_each_entry(sko, &sks->sks_free_list, sko_list) {
|
|
ASSERT(sko->sko_magic == SKO_MAGIC);
|
|
|
|
if (skc->skc_ctor)
|
|
skc->skc_ctor(sko->sko_addr, skc->skc_private, flags);
|
|
}
|
|
|
|
if (flags & __GFP_WAIT)
|
|
local_irq_disable();
|
|
|
|
/* Link the new empty slab in to the end of skc_partial_list */
|
|
spin_lock(&skc->skc_lock);
|
|
skc->skc_slab_total++;
|
|
skc->skc_obj_total += sks->sks_objs;
|
|
list_add_tail(&sks->sks_list, &skc->skc_partial_list);
|
|
spin_unlock(&skc->skc_lock);
|
|
|
|
RETURN(sks);
|
|
}
|
|
|
|
static int
|
|
spl_cache_refill(spl_kmem_cache_t *skc, spl_kmem_magazine_t *skm, int flags)
|
|
{
|
|
spl_kmem_slab_t *sks;
|
|
int refill = skm->skm_refill;
|
|
ENTRY;
|
|
|
|
/* XXX: Check for refill bouncing by age perhaps */
|
|
|
|
spin_lock(&skc->skc_lock);
|
|
while (refill > 0) {
|
|
/* No slabs available we must grow the cache */
|
|
if (list_empty(&skc->skc_partial_list)) {
|
|
spin_unlock(&skc->skc_lock);
|
|
sks = spl_cache_grow(skc, flags);
|
|
if (!sks)
|
|
GOTO(out, refill);
|
|
|
|
/* Rescheduled to different CPU skm is not local */
|
|
if (skm != skc->skc_mag[smp_processor_id()])
|
|
GOTO(out, refill);
|
|
|
|
spin_lock(&skc->skc_lock);
|
|
continue;
|
|
}
|
|
|
|
/* Grab the next available slab */
|
|
sks = list_entry((&skc->skc_partial_list)->next,
|
|
spl_kmem_slab_t, sks_list);
|
|
ASSERT(sks->sks_magic == SKS_MAGIC);
|
|
ASSERT(sks->sks_ref < sks->sks_objs);
|
|
ASSERT(!list_empty(&sks->sks_free_list));
|
|
|
|
/* Consume as many objects as needed to refill the requested
|
|
* cache. We must be careful to lock here because our local
|
|
* magazine may not be local anymore due to spl_cache_grow. */
|
|
while ((sks->sks_ref < sks->sks_objs) && (refill-- > 0))
|
|
skm->skm_objs[skm->skm_avail++]=spl_cache_obj(skc,sks);
|
|
|
|
/* Move slab to skc_complete_list when full */
|
|
if (sks->sks_ref == sks->sks_objs) {
|
|
list_del(&sks->sks_list);
|
|
list_add(&sks->sks_list, &skc->skc_complete_list);
|
|
}
|
|
}
|
|
|
|
spin_unlock(&skc->skc_lock);
|
|
out:
|
|
/* Returns the number of entries added to cache */
|
|
RETURN(skm->skm_refill - refill);
|
|
}
|
|
|
|
static void
|
|
spl_cache_shrink(spl_kmem_cache_t *skc, void *obj)
|
|
{
|
|
spl_kmem_slab_t *sks = NULL;
|
|
spl_kmem_obj_t *sko = NULL;
|
|
ENTRY;
|
|
|
|
ASSERT(spin_is_locked(&skc->skc_lock));
|
|
|
|
sko = spl_hash_obj(skc, obj);
|
|
ASSERTF(sko, "Obj %p missing from in-use hash (%d) for cache %s\n",
|
|
obj, skc->skc_hash_count, skc->skc_name);
|
|
|
|
sks = sko->sko_slab;
|
|
ASSERTF(sks, "Obj %p/%p linked to invalid slab for cache %s\n",
|
|
obj, sko, skc->skc_name);
|
|
|
|
ASSERT(sks->sks_cache == skc);
|
|
hlist_del_init(&sko->sko_hlist);
|
|
list_add(&sko->sko_list, &sks->sks_free_list);
|
|
|
|
sks->sks_age = jiffies;
|
|
sks->sks_ref--;
|
|
skc->skc_obj_alloc--;
|
|
skc->skc_hash_count--;
|
|
|
|
/* Move slab to skc_partial_list when no longer full. Slabs
|
|
* are added to the head to keep the partial list is quasi-full
|
|
* sorted order. Fuller at the head, emptier at the tail. */
|
|
if (sks->sks_ref == (sks->sks_objs - 1)) {
|
|
list_del(&sks->sks_list);
|
|
list_add(&sks->sks_list, &skc->skc_partial_list);
|
|
}
|
|
|
|
/* Move emply slabs to the end of the partial list so
|
|
* they can be easily found and freed during reclamation. */
|
|
if (sks->sks_ref == 0) {
|
|
list_del(&sks->sks_list);
|
|
list_add_tail(&sks->sks_list, &skc->skc_partial_list);
|
|
skc->skc_slab_alloc--;
|
|
}
|
|
|
|
EXIT;
|
|
}
|
|
|
|
static int
|
|
spl_cache_flush(spl_kmem_cache_t *skc, spl_kmem_magazine_t *skm, int flush)
|
|
{
|
|
int i, count = MIN(flush, skm->skm_avail);
|
|
ENTRY;
|
|
|
|
|
|
spin_lock(&skc->skc_lock);
|
|
for (i = 0; i < count; i++)
|
|
spl_cache_shrink(skc, skm->skm_objs[i]);
|
|
|
|
__spl_slab_reclaim(skc);
|
|
skm->skm_avail -= count;
|
|
memmove(skm->skm_objs, &(skm->skm_objs[count]),
|
|
sizeof(void *) * skm->skm_avail);
|
|
|
|
spin_unlock(&skc->skc_lock);
|
|
|
|
RETURN(count);
|
|
}
|
|
|
|
void *
|
|
spl_kmem_cache_alloc(spl_kmem_cache_t *skc, int flags)
|
|
{
|
|
spl_kmem_magazine_t *skm;
|
|
unsigned long irq_flags;
|
|
void *obj = NULL;
|
|
ENTRY;
|
|
|
|
ASSERT(flags & KM_SLEEP);
|
|
local_irq_save(irq_flags);
|
|
|
|
restart:
|
|
/* Safe to update per-cpu structure without lock, but
|
|
* in the restart case we must be careful to reaquire
|
|
* the local magazine since this may have changed
|
|
* when we need to grow the cache. */
|
|
skm = skc->skc_mag[smp_processor_id()];
|
|
|
|
if (likely(skm->skm_avail)) {
|
|
/* Object available in CPU cache, use it */
|
|
obj = skm->skm_objs[--skm->skm_avail];
|
|
skm->skm_age = jiffies;
|
|
} else {
|
|
/* Per-CPU cache empty, directly allocate from
|
|
* the slab and refill the per-CPU cache. */
|
|
(void)spl_cache_refill(skc, skm, flags);
|
|
GOTO(restart, obj = NULL);
|
|
}
|
|
|
|
local_irq_restore(irq_flags);
|
|
|
|
/* Pre-emptively migrate object to CPU L1 cache */
|
|
prefetchw(obj);
|
|
|
|
RETURN(obj);
|
|
}
|
|
EXPORT_SYMBOL(spl_kmem_cache_alloc);
|
|
|
|
void
|
|
spl_kmem_cache_free(spl_kmem_cache_t *skc, void *obj)
|
|
{
|
|
spl_kmem_magazine_t *skm;
|
|
unsigned long flags;
|
|
ENTRY;
|
|
|
|
local_irq_save(flags);
|
|
|
|
/* Safe to update per-cpu structure without lock, but
|
|
* no remote memory allocation tracking is being performed
|
|
* it is entirely possible to allocate an object from one
|
|
* CPU cache and return it to another. */
|
|
skm = skc->skc_mag[smp_processor_id()];
|
|
|
|
/* Per-CPU cache full, flush it to make space */
|
|
if (unlikely(skm->skm_avail >= skm->skm_size))
|
|
(void)spl_cache_flush(skc, skm, skm->skm_refill);
|
|
|
|
/* Available space in cache, use it */
|
|
skm->skm_objs[skm->skm_avail++] = obj;
|
|
|
|
local_irq_restore(flags);
|
|
|
|
EXIT;
|
|
}
|
|
EXPORT_SYMBOL(spl_kmem_cache_free);
|
|
|
|
static int
|
|
spl_kmem_cache_generic_shrinker(int nr_to_scan, unsigned int gfp_mask)
|
|
{
|
|
spl_kmem_cache_t *skc;
|
|
|
|
/* Under linux a shrinker is not tightly coupled with a slab
|
|
* cache. In fact linux always systematically trys calling all
|
|
* registered shrinker callbacks until its target reclamation level
|
|
* is reached. Because of this we only register one shrinker
|
|
* function in the shim layer for all slab caches. And we always
|
|
* attempt to shrink all caches when this generic shrinker is called.
|
|
*/
|
|
down_read(&spl_kmem_cache_sem);
|
|
|
|
list_for_each_entry(skc, &spl_kmem_cache_list, skc_list)
|
|
spl_kmem_cache_reap_now(skc);
|
|
|
|
up_read(&spl_kmem_cache_sem);
|
|
|
|
/* XXX: Under linux we should return the remaining number of
|
|
* entries in the cache. We should do this as well.
|
|
*/
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
spl_kmem_cache_reap_now(spl_kmem_cache_t *skc)
|
|
{
|
|
spl_kmem_magazine_t *skm;
|
|
int i;
|
|
ENTRY;
|
|
ASSERT(skc && skc->skc_magic == SKC_MAGIC);
|
|
|
|
if (skc->skc_reclaim)
|
|
skc->skc_reclaim(skc->skc_private);
|
|
|
|
/* Ensure per-CPU caches which are idle gradually flush */
|
|
for_each_online_cpu(i) {
|
|
skm = skc->skc_mag[i];
|
|
|
|
if (time_after(jiffies, skm->skm_age + skc->skc_delay * HZ))
|
|
(void)spl_cache_flush(skc, skm, skm->skm_refill);
|
|
}
|
|
|
|
spl_slab_reclaim(skc);
|
|
|
|
EXIT;
|
|
}
|
|
EXPORT_SYMBOL(spl_kmem_cache_reap_now);
|
|
|
|
void
|
|
spl_kmem_reap(void)
|
|
{
|
|
spl_kmem_cache_generic_shrinker(KMC_REAP_CHUNK, GFP_KERNEL);
|
|
}
|
|
EXPORT_SYMBOL(spl_kmem_reap);
|
|
|
|
int
|
|
spl_kmem_init(void)
|
|
{
|
|
int rc = 0;
|
|
ENTRY;
|
|
|
|
init_rwsem(&spl_kmem_cache_sem);
|
|
INIT_LIST_HEAD(&spl_kmem_cache_list);
|
|
|
|
spl_slab_cache = NULL;
|
|
spl_obj_cache = NULL;
|
|
|
|
spl_slab_cache = __kmem_cache_create("spl_slab_cache",
|
|
sizeof(spl_kmem_slab_t),
|
|
0, 0, NULL, NULL);
|
|
if (spl_slab_cache == NULL)
|
|
GOTO(out_cache, rc = -ENOMEM);
|
|
|
|
spl_obj_cache = __kmem_cache_create("spl_obj_cache",
|
|
sizeof(spl_kmem_obj_t),
|
|
0, 0, NULL, NULL);
|
|
if (spl_obj_cache == NULL)
|
|
GOTO(out_cache, rc = -ENOMEM);
|
|
|
|
#ifdef HAVE_SET_SHRINKER
|
|
spl_kmem_cache_shrinker = set_shrinker(KMC_DEFAULT_SEEKS,
|
|
spl_kmem_cache_generic_shrinker);
|
|
if (spl_kmem_cache_shrinker == NULL)
|
|
GOTO(out_cache, rc = -ENOMEM);
|
|
#else
|
|
register_shrinker(&spl_kmem_cache_shrinker);
|
|
#endif
|
|
|
|
#ifdef DEBUG_KMEM
|
|
{ int i;
|
|
atomic64_set(&kmem_alloc_used, 0);
|
|
atomic64_set(&vmem_alloc_used, 0);
|
|
atomic64_set(&kmem_cache_alloc_failed, 0);
|
|
|
|
spin_lock_init(&kmem_lock);
|
|
INIT_LIST_HEAD(&kmem_list);
|
|
|
|
for (i = 0; i < KMEM_TABLE_SIZE; i++)
|
|
INIT_HLIST_HEAD(&kmem_table[i]);
|
|
|
|
spin_lock_init(&vmem_lock);
|
|
INIT_LIST_HEAD(&vmem_list);
|
|
|
|
for (i = 0; i < VMEM_TABLE_SIZE; i++)
|
|
INIT_HLIST_HEAD(&vmem_table[i]);
|
|
}
|
|
#endif
|
|
RETURN(rc);
|
|
|
|
out_cache:
|
|
if (spl_obj_cache)
|
|
(void)kmem_cache_destroy(spl_obj_cache);
|
|
|
|
if (spl_slab_cache)
|
|
(void)kmem_cache_destroy(spl_slab_cache);
|
|
|
|
RETURN(rc);
|
|
}
|
|
|
|
#ifdef DEBUG_KMEM
|
|
static char *
|
|
spl_sprintf_addr(kmem_debug_t *kd, char *str, int len, int min)
|
|
{
|
|
int size = ((len - 1) < kd->kd_size) ? (len - 1) : kd->kd_size;
|
|
int i, flag = 1;
|
|
|
|
ASSERT(str != NULL && len >= 17);
|
|
memset(str, 0, len);
|
|
|
|
/* Check for a fully printable string, and while we are at
|
|
* it place the printable characters in the passed buffer. */
|
|
for (i = 0; i < size; i++) {
|
|
str[i] = ((char *)(kd->kd_addr))[i];
|
|
if (isprint(str[i])) {
|
|
continue;
|
|
} else {
|
|
/* Minimum number of printable characters found
|
|
* to make it worthwhile to print this as ascii. */
|
|
if (i > min)
|
|
break;
|
|
|
|
flag = 0;
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
if (!flag) {
|
|
sprintf(str, "%02x%02x%02x%02x%02x%02x%02x%02x",
|
|
*((uint8_t *)kd->kd_addr),
|
|
*((uint8_t *)kd->kd_addr + 2),
|
|
*((uint8_t *)kd->kd_addr + 4),
|
|
*((uint8_t *)kd->kd_addr + 6),
|
|
*((uint8_t *)kd->kd_addr + 8),
|
|
*((uint8_t *)kd->kd_addr + 10),
|
|
*((uint8_t *)kd->kd_addr + 12),
|
|
*((uint8_t *)kd->kd_addr + 14));
|
|
}
|
|
|
|
return str;
|
|
}
|
|
#endif /* DEBUG_KMEM */
|
|
|
|
void
|
|
spl_kmem_fini(void)
|
|
{
|
|
#ifdef DEBUG_KMEM
|
|
unsigned long flags;
|
|
kmem_debug_t *kd;
|
|
char str[17];
|
|
|
|
/* Display all unreclaimed memory addresses, including the
|
|
* allocation size and the first few bytes of what's located
|
|
* at that address to aid in debugging. Performance is not
|
|
* a serious concern here since it is module unload time. */
|
|
if (atomic64_read(&kmem_alloc_used) != 0)
|
|
CWARN("kmem leaked %ld/%ld bytes\n",
|
|
atomic_read(&kmem_alloc_used), kmem_alloc_max);
|
|
|
|
spin_lock_irqsave(&kmem_lock, flags);
|
|
if (!list_empty(&kmem_list))
|
|
CDEBUG(D_WARNING, "%-16s %-5s %-16s %s:%s\n",
|
|
"address", "size", "data", "func", "line");
|
|
|
|
list_for_each_entry(kd, &kmem_list, kd_list)
|
|
CDEBUG(D_WARNING, "%p %-5d %-16s %s:%d\n",
|
|
kd->kd_addr, kd->kd_size,
|
|
spl_sprintf_addr(kd, str, 17, 8),
|
|
kd->kd_func, kd->kd_line);
|
|
|
|
spin_unlock_irqrestore(&kmem_lock, flags);
|
|
|
|
if (atomic64_read(&vmem_alloc_used) != 0)
|
|
CWARN("vmem leaked %ld/%ld bytes\n",
|
|
atomic_read(&vmem_alloc_used), vmem_alloc_max);
|
|
|
|
spin_lock_irqsave(&vmem_lock, flags);
|
|
if (!list_empty(&vmem_list))
|
|
CDEBUG(D_WARNING, "%-16s %-5s %-16s %s:%s\n",
|
|
"address", "size", "data", "func", "line");
|
|
|
|
list_for_each_entry(kd, &vmem_list, kd_list)
|
|
CDEBUG(D_WARNING, "%p %-5d %-16s %s:%d\n",
|
|
kd->kd_addr, kd->kd_size,
|
|
spl_sprintf_addr(kd, str, 17, 8),
|
|
kd->kd_func, kd->kd_line);
|
|
|
|
spin_unlock_irqrestore(&vmem_lock, flags);
|
|
#endif
|
|
ENTRY;
|
|
|
|
#ifdef HAVE_SET_SHRINKER
|
|
remove_shrinker(spl_kmem_cache_shrinker);
|
|
#else
|
|
unregister_shrinker(&spl_kmem_cache_shrinker);
|
|
#endif
|
|
|
|
(void)kmem_cache_destroy(spl_obj_cache);
|
|
(void)kmem_cache_destroy(spl_slab_cache);
|
|
|
|
EXIT;
|
|
}
|