runtime/mempool.c - systemtap

Data types defined

Functions defined

Macros defined

Source code

/*  -*- linux-c -*-
* Preallocated memory pools
* Copyright (C) 2008-2009 Red Hat Inc.
*
* This file is part of systemtap, and is free software.  You can
* redistribute it and/or modify it under the terms of the GNU General
* Public License (GPL); either version 2, or (at your option) any
* later version.
*/

#ifndef _STP_MEMPOOL_C_
#define _STP_MEMPOOL_C_

/* An opaque struct identifying the memory pool. */
typedef struct {
    struct list_head free_list;
    unsigned num;
    unsigned size;
    spinlock_t lock;
} _stp_mempool_t;

/* for internal use only */
struct _stp_mem_buffer {
    struct list_head list;
    _stp_mempool_t *pool;
    void *buf;
};

/* Delete a memory pool */
static void _stp_mempool_destroy(_stp_mempool_t *pool)
{
    struct list_head *p, *tmp;
    if (pool) {
        list_for_each_safe(p, tmp, &pool->free_list) {
            list_del(p);
            _stp_kfree(p);
        }
        _stp_kfree(pool);
    }
}

/* Create a new memory pool */
static _stp_mempool_t *_stp_mempool_init(size_t size, size_t num)
{
    int i, alloc_size;
    struct _stp_mem_buffer *m;

    _stp_mempool_t *pool = (_stp_mempool_t *)_stp_kmalloc(sizeof(_stp_mempool_t));
    if (unlikely(pool == NULL)) {
        errk("Memory allocation failed.\n");
        return NULL;
    }

    INIT_LIST_HEAD(&pool->free_list);
    spin_lock_init(&pool->lock);

    alloc_size = size + sizeof(struct _stp_mem_buffer) - sizeof(void *);

    for (i = 0; i < num; i++) {
        m = (struct _stp_mem_buffer *)_stp_kmalloc(alloc_size);
        if (unlikely(m == NULL))
            goto err;
        m->pool = pool;
        list_add((struct list_head *)m, &pool->free_list);
    }
    pool->num = num;
    pool->size = alloc_size;
    return pool;

err:
    _stp_mempool_destroy(pool);
    return NULL;
}

/* allocate a buffer from a memory pool */
static void *_stp_mempool_alloc(_stp_mempool_t *pool)
{
    unsigned long flags;
    struct _stp_mem_buffer *ptr = NULL;
        /* PR14804: tolerate accidental early call, before pool is
         actually initialized. */
        if (pool == NULL)
                return NULL;
    spin_lock_irqsave(&pool->lock, flags);
    if (likely(!list_empty(&pool->free_list))) {
        ptr = (struct _stp_mem_buffer *)pool->free_list.next;
        list_del_init(&ptr->list);
        spin_unlock_irqrestore(&pool->lock, flags);
        return &ptr->buf;
    }
    spin_unlock_irqrestore(&pool->lock, flags);
    return NULL;
}

/* return a buffer to its memory pool */
static void _stp_mempool_free(void *buf)
{
    unsigned long flags;
    struct _stp_mem_buffer *m = container_of(buf, struct _stp_mem_buffer, buf);
    spin_lock_irqsave(&m->pool->lock, flags);
    list_add(&m->list, &m->pool->free_list);
    spin_unlock_irqrestore(&m->pool->lock, flags);
}
#endif /* _STP_MEMPOOL_C_ */