graphicscomposition/openwfcompositionengine/common/src/owfpool.c
author jakl.martin@cell-telecom.com
Mon, 06 Dec 2010 18:07:30 +0100
branchNewGraphicsArchitecture
changeset 218 99b3451c560e
parent 0 5d03bc08d59c
child 152 9f1c3fea0f87
permissions -rw-r--r--
Fix for Bug 3890

/* Copyright (c) 2009 The Khronos Group Inc.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and/or associated documentation files (the
 * "Materials"), to deal in the Materials without restriction, including
 * without limitation the rights to use, copy, modify, merge, publish,
 * distribute, sublicense, and/or sell copies of the Materials, and to
 * permit persons to whom the Materials are furnished to do so, subject to
 * the following conditions:
 *
 * The above copyright notice and this permission notice shall be included
 * in all copies or substantial portions of the Materials.
 *
 * THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
 * MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
 */

 /*
 * owfpool.c
 */


#ifdef __cplusplus
extern "C"
{
#endif


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

#include "owfdebug.h"
#include "owftypes.h"
#include "owfpool.h"
#include "owfmemory.h"




OWF_API_CALL OWF_POOL*
OWF_Pool_Create(size_t objectSize, size_t objectCount)
{
    OWF_POOL*               pool;
    char*                   chunk;
    OWFuint32*               entries;

    /* we be needing some memories, aye */
    pool    = (OWF_POOL*)xalloc(sizeof(OWF_POOL), 1);
    entries = (OWFuint32*)xalloc(sizeof(OWFuint32), objectCount);
    chunk   = (char*)xalloc((objectSize + sizeof(OWFuint32*)), objectCount);

    if (pool && entries && chunk)
    {
        size_t              i;

        /* setup pool */
        pool->entries   = entries;
        pool->chunk     = chunk;
        pool->capacity  =
        pool->free      = objectCount;
        pool->firstFree = 0;
        pool->entrySize = objectSize;

        /* initially all objects are marked as free */
        for (i = 0; i < objectCount-1; i++) {
            entries[i] = i+1;
        }
        /* terminate chain */
        entries[objectCount-1] = EOC;
    }
    else
    {
        /* failed miserably. bail out. */
        xfree(pool);
        pool = NULL;
        xfree(entries);
        xfree(chunk);
    }

    return pool;
}

OWF_API_CALL void*
OWF_Pool_GetObject(OWF_POOL* pool)
{
    void*                   object = NULL;

    if (pool)
    {
        /* we have objects left in the pool? */
        if (pool->firstFree != EOC)
        {
            OWFuint32*       temp;
            OWFuint32        chunk;

            chunk = (OWFuint32) pool->chunk;

            /* calculate offset inside the chunk */
            temp = (OWFuint32*) (chunk + (pool->entrySize +
                    sizeof(OWFuint32*)) * pool->firstFree);
            /* remember way back to home */
            temp[0] = (OWFuint32) pool;
            object = (void*) &temp[1];
            pool->firstFree = pool->entries[pool->firstFree];
            --pool->free;

            OWF_ASSERT((OWFuint32) object > (OWFuint32) pool->chunk);
            OWF_ASSERT((int)object != 0x5eaf00d);
        }
    }
    return object;
}

OWF_API_CALL void
OWF_Pool_PutObject(void* object)
{
    OWFuint32* temp = (OWFuint32*) object;

    if (temp && temp[-1])
    {
        OWF_POOL*           pool;
        OWFuint32            addr;
        OWFuint32            chunk;
        OWFuint32            entrySize;

        addr    = (OWFuint32) object - 4;
        pool    = (OWF_POOL*) temp[-1];
        chunk   = (OWFuint32) pool->chunk;
        entrySize = pool->entrySize + sizeof(OWFuint32*);

        if (addr >= chunk &&
            addr < (chunk + pool->capacity * entrySize))
        {
            OWFuint32 index = (addr - chunk) / entrySize;

            pool->entries[index] = pool->firstFree;
            pool->firstFree = index;
            memset(object, 0x0, pool->entrySize);
            ++pool->free;
        }
        else
        {
            /* invalid object */
            OWF_ASSERT(0);
        }
    }
}

OWF_API_CALL void
OWF_Pool_Destroy(OWF_POOL* pool)
{
    if (pool)
    {
        xfree(pool->entries);
        xfree(pool->chunk);
        memset(pool, 0, sizeof(OWF_POOL));
        xfree(pool);
    }
}


#ifdef __cplusplus
}
#endif