tun2socks 0.0.1

High performance tun2socks
Documentation
/*
 ============================================================================
 Name        : hev-memory-allocator-slice.c
 Author      : Heiher <r@hev.cc>
 Copyright   : Copyright (c) 2017 everyone.
 Description : Memory allocator sliced
 ============================================================================
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include "lib/misc/hev-compiler.h"

#include "hev-memory-allocator-slice.h"

#define CACHED_SLICE_ALIGN CONFIG_MEMALLOC_SLICE_ALIGN
#define MAX_CACHED_SLICE_SIZE CONFIG_MEMALLOC_SLICE_MAX_SIZE
#define MAX_CACHED_SLICE_COUNT CONFIG_MEMALLOC_SLICE_MAX_COUNT
#define MAX_CACHED_SLICE_INDEX (MAX_CACHED_SLICE_SIZE / CACHED_SLICE_ALIGN)

typedef struct _HevMemorySlice HevMemorySlice;
typedef struct _HevMemoryLRUNode HevMemoryLRUNode;

struct _HevMemorySlice
{
    HevMemorySlice *next;
    int index;
};

struct _HevMemoryLRUNode
{
    HevMemoryLRUNode *prev;
    HevMemoryLRUNode *next;
};

struct _HevMemoryAllocatorSlice
{
    HevMemoryAllocator base;

    HevMemoryLRUNode *lru_head;
    HevMemoryLRUNode *lru_tail;
    HevMemoryLRUNode lru_nodes[MAX_CACHED_SLICE_INDEX];

    unsigned int cached_count;
    HevMemorySlice *cached_mslices[MAX_CACHED_SLICE_INDEX];
};

static void *_hev_memory_allocator_alloc (HevMemoryAllocator *self,
                                          size_t size);
static void *_hev_memory_allocator_realloc (HevMemoryAllocator *self, void *ptr,
                                            size_t size);
static void _hev_memory_allocator_free (HevMemoryAllocator *self, void *ptr);
static void _hev_memory_allocator_destroy (HevMemoryAllocator *self);
static void _hev_memory_allocator_lru_insert (HevMemoryAllocatorSlice *self,
                                              HevMemoryLRUNode *node);
static void _hev_memory_allocator_lru_remove (HevMemoryAllocatorSlice *self,
                                              HevMemoryLRUNode *node);

HevMemoryAllocator *
hev_memory_allocator_slice_new (void)
{
    HevMemoryAllocator *allocator = NULL;
    HevMemoryAllocatorSlice *self = NULL;

    allocator = malloc (sizeof (HevMemoryAllocatorSlice));
    if (!allocator)
        return NULL;

    allocator->ref_count = 1;
    allocator->alloc = _hev_memory_allocator_alloc;
    allocator->realloc = _hev_memory_allocator_realloc;
    allocator->free = _hev_memory_allocator_free;
    allocator->destroy = _hev_memory_allocator_destroy;

    self = (HevMemoryAllocatorSlice *)allocator;
    self->lru_head = NULL;
    self->lru_tail = NULL;
    self->cached_count = 0;
    memset (self->cached_mslices, 0, sizeof (self->cached_mslices));

    return allocator;
}

static void *
_hev_memory_allocator_alloc (HevMemoryAllocator *allocator, size_t size)
{
    HevMemoryAllocatorSlice *self = (HevMemoryAllocatorSlice *)allocator;
    HevMemorySlice *slice, **owner;
    int index;

    size = ALIGN_UP (size, CACHED_SLICE_ALIGN);
    index = size / CACHED_SLICE_ALIGN;

    switch (index) {
    case 0:
        return NULL;
    case 1 ... MAX_CACHED_SLICE_INDEX:
        owner = &self->cached_mslices[index - 1];
        break;
    default:
#ifdef _DEBUG
        printf ("[%p] default alloc size: %lu\n", self, size);
#endif
        slice = malloc (sizeof (HevMemorySlice) + size);
        if (!slice)
            return NULL;
        slice->index = -1;
        return slice + 1;
    }

    if (*owner) {
        HevMemoryLRUNode *node;

        slice = *owner;
        *owner = slice->next;
        self->cached_count--;

        node = &self->lru_nodes[index - 1];
        _hev_memory_allocator_lru_remove (self, node);
        if (*owner)
            _hev_memory_allocator_lru_insert (self, node);
    } else {
#ifdef _DEBUG
        printf ("[%p] alloc size: %lu\n", self, size);
#endif
        slice = malloc (sizeof (HevMemorySlice) + size);
        if (!slice)
            return NULL;

        slice->index = index - 1;
    }

    return slice + 1;
}

static void *
_hev_memory_allocator_realloc (HevMemoryAllocator *allocator, void *ptr,
                               size_t size)
{
    HevMemorySlice *slice;
    int index;

    if (0 == size) {
        _hev_memory_allocator_free (allocator, ptr);
        return NULL;
    }

    slice = (HevMemorySlice *)ptr - 1;
    size = ALIGN_UP (size, CACHED_SLICE_ALIGN);
    index = size / CACHED_SLICE_ALIGN;

    slice = realloc (slice, sizeof (HevMemorySlice) + size);
    if (!slice)
        return NULL;

    switch (index) {
    case 1 ... MAX_CACHED_SLICE_INDEX:
        slice->index = index - 1;
#ifdef _DEBUG
        printf ("[%p] realloc size: %lu\n", allocator, size);
#endif
        break;
    default:
        slice->index = -1;
#ifdef _DEBUG
        printf ("[%p] default realloc size: %lu\n", allocator, size);
#endif
    }

    return slice + 1;
}

static void
_hev_memory_allocator_free (HevMemoryAllocator *allocator, void *ptr)
{
    HevMemoryAllocatorSlice *self = (HevMemoryAllocatorSlice *)allocator;
    HevMemorySlice *slice = (HevMemorySlice *)ptr - 1;

    if (slice->index < 0) {
        free (slice);
        return;
    }

    if (self->cached_count >= MAX_CACHED_SLICE_COUNT) {
        HevMemoryLRUNode *node = self->lru_tail;
        HevMemorySlice **owner = &self->cached_mslices[node - self->lru_nodes];
        HevMemorySlice *free_slice = *owner;

        *owner = free_slice->next;
        self->cached_count--;

        if (!*owner)
            _hev_memory_allocator_lru_remove (self, node);

        free (free_slice);
    }

    slice->next = self->cached_mslices[slice->index];
    self->cached_mslices[slice->index] = slice;
    self->cached_count++;

    if (!slice->next) {
        HevMemoryLRUNode *node;

        node = &self->lru_nodes[slice->index];
        _hev_memory_allocator_lru_insert (self, node);
    }

#ifdef _DEBUG
    printf ("[%p] cached_count: %u\n", self, self->cached_count);
#endif
}

static void
_hev_memory_allocator_destroy (HevMemoryAllocator *allocator)
{
    HevMemoryAllocatorSlice *self = (HevMemoryAllocatorSlice *)allocator;
    HevMemoryLRUNode *node;

    for (node = self->lru_head; node; node = node->next) {
        HevMemorySlice *iter = self->cached_mslices[node - self->lru_nodes];

        while (iter) {
            HevMemorySlice *next = iter->next;
            free (iter);
            iter = next;
        }
    }
}

static void
_hev_memory_allocator_lru_insert (HevMemoryAllocatorSlice *self,
                                  HevMemoryLRUNode *node)
{
    node->prev = NULL;
    if (self->lru_head) {
        node->next = self->lru_head;
        self->lru_head->prev = node;
    } else {
        node->next = NULL;
    }
    self->lru_head = node;
    if (!self->lru_tail)
        self->lru_tail = node;
}

static void
_hev_memory_allocator_lru_remove (HevMemoryAllocatorSlice *self,
                                  HevMemoryLRUNode *node)
{
    if (node->prev)
        node->prev->next = node->next;
    if (node->next)
        node->next->prev = node->prev;
    if (self->lru_head == node)
        self->lru_head = node->next;
    if (self->lru_tail == node)
        self->lru_tail = node->prev;
}