gstreamer/test/memchunk/gstmemchunk.c
Wim Taymans 7bb4b2cefe Added a fast non-blocking memchunk implementation, it's about 20x faster than a mutex based implementation.
Original commit message from CVS:
Added a fast non-blocking memchunk implementation, it's about 20x
faster than a mutex based implementation.
2001-09-30 01:33:28 +00:00

96 lines
2.1 KiB
C

#include "gstmemchunk.h"
#ifdef __SMP__
#define CHUNK_LOCK "lock ; "
#else
#define CHUNK_LOCK ""
#endif
static void
setup_area (guint8 *area, gint chunk_size, gulong area_size)
{
gint i;
g_print ("setup area at %p, real area_size %lu, chunk size %d\n", area, area_size, chunk_size);
for (i = 0; i < area_size; i += chunk_size) {
guint8 *next;
next = area + chunk_size;
((GstMemChunkElement *)area)->link = (gpointer)(next);
area = next;
}
}
GstMemChunk *
gst_mem_chunk_new (gchar * name, gint atom_size, gulong area_size, gint type)
{
GstMemChunk *mem_chunk;
gint chunk_size;
g_print ("create: atom size %d, area_size %lu\n", atom_size, area_size);
chunk_size = atom_size + sizeof (gpointer);
area_size = (area_size/atom_size) * chunk_size;
g_print ("chunk size %d, real area_size %lu\n", chunk_size, area_size);
mem_chunk = g_malloc (sizeof (GstMemChunk));
mem_chunk->free = g_malloc0 (area_size);
mem_chunk->cnt = 0;
mem_chunk->atom_size = atom_size;
mem_chunk->chunk_size = chunk_size;
mem_chunk->area_size = area_size;
setup_area ((guint8 *)mem_chunk->free, chunk_size, area_size);
return mem_chunk;
}
gpointer
gst_mem_chunk_alloc (GstMemChunk *mem_chunk)
{
guint8 *chunk = NULL;
g_return_val_if_fail (mem_chunk != NULL, NULL);
__asm__ __volatile__ ("testl %%eax, %%eax \n\t"
"jz 20f \n"
"10:\t"
"movl (%%eax), %%ebx \n\t"
"movl %%edx, %%ecx \n\t"
"incl %%ecx \n\t"
CHUNK_LOCK "cmpxchg8b %1 \n\t"
"jz 20f \n\t"
"testl %%eax, %%eax \n\t"
"jnz 10b \n"
"20:\t":"=a" (chunk):"m" (*mem_chunk),
"a" (mem_chunk->free), "d" (mem_chunk->cnt):"ecx",
"ebx");
if (chunk)
chunk += sizeof (gpointer);
else {
g_print ("empty\n");
exit (-1);
}
return (gpointer) chunk;
}
void
gst_mem_chunk_free (GstMemChunk *mem_chunk, gpointer mem)
{
guint8 *chunk = ((guint8 *)mem) - sizeof (gpointer);
g_return_if_fail (mem_chunk != NULL);
g_return_if_fail (mem != NULL);
__asm__ __volatile__ ( "1:\t"
"movl %2, (%1) \n"
CHUNK_LOCK "cmpxchg %1, %0 \n\t"
"jnz 1b \n\t"::"m" (*mem_chunk), "r" (chunk), "a" (mem_chunk->free));
}