mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-22 23:28:16 +00:00
74b6b732f1
Original commit message from CVS: - Added a bunch of fast atomic operations on x86 with C fallbacks - Added the code for refcounting in GstData - Added a gstmemchunk allocator using atomic operations
159 lines
3.9 KiB
C
159 lines
3.9 KiB
C
/* GStreamer
|
|
* Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <string.h> /* memset */
|
|
|
|
#include "gstlog.h"
|
|
#include "gstmemchunk.h"
|
|
#include "gstutils.h"
|
|
|
|
#define GST_MEM_CHUNK_AREA(chunk) (((GstMemChunkElement*)(chunk))->area)
|
|
#define GST_MEM_CHUNK_DATA(chunk) ((gpointer)(((GstMemChunkElement*)(chunk)) + 1))
|
|
#define GST_MEM_CHUNK_LINK(mem) ((GstMemChunkElement*)((guint8*)(mem) - sizeof (GstMemChunkElement)))
|
|
|
|
/*******************************************************
|
|
* area size
|
|
* +-----------------------------------------+
|
|
* chunk size
|
|
* +------------+
|
|
*
|
|
* !next!data... !next!data.... !next!data...
|
|
* ! ^ ! ^ !
|
|
* +-------------+ +------------+ +---> NULL
|
|
*
|
|
*/
|
|
static gboolean
|
|
populate (GstMemChunk *mem_chunk)
|
|
{
|
|
guint8 *area;
|
|
gint i;
|
|
|
|
if (mem_chunk->cleanup)
|
|
return FALSE;
|
|
|
|
area = (guint8 *) g_malloc0 (mem_chunk->area_size);
|
|
|
|
for (i=0; i < mem_chunk->area_size; i += mem_chunk->chunk_size) {
|
|
GST_MEM_CHUNK_AREA (area + i) = (GstMemChunkElement *)area;
|
|
gst_mem_chunk_free (mem_chunk, GST_MEM_CHUNK_DATA (area + i));
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
GstMemChunk*
|
|
gst_mem_chunk_new (gchar* name, gint atom_size, gulong area_size, gint type)
|
|
{
|
|
GstMemChunk *mem_chunk;
|
|
|
|
g_return_val_if_fail (atom_size > 0, NULL);
|
|
g_return_val_if_fail (area_size >= atom_size, NULL);
|
|
|
|
mem_chunk = g_malloc (sizeof (GstMemChunk));
|
|
|
|
mem_chunk->chunk_size = atom_size + sizeof (GstMemChunkElement);
|
|
area_size = (area_size/atom_size) * mem_chunk->chunk_size;
|
|
|
|
mem_chunk->name = g_strdup (name);
|
|
mem_chunk->atom_size = atom_size;
|
|
mem_chunk->area_size = area_size;
|
|
mem_chunk->cleanup = FALSE;
|
|
GST_ATOMIC_SWAP_INIT (&mem_chunk->swap, NULL);
|
|
|
|
populate (mem_chunk);
|
|
|
|
return mem_chunk;
|
|
}
|
|
|
|
static gboolean
|
|
free_area (gpointer key, gpointer value, gpointer user_data)
|
|
{
|
|
g_free (key);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gst_mem_chunk_destroy (GstMemChunk *mem_chunk)
|
|
{
|
|
GHashTable *elements = g_hash_table_new (NULL, NULL);
|
|
gpointer data;
|
|
|
|
mem_chunk->cleanup = TRUE;
|
|
|
|
data = gst_mem_chunk_alloc (mem_chunk);
|
|
while (data) {
|
|
GstMemChunkElement *elem = GST_MEM_CHUNK_LINK (data);
|
|
|
|
g_hash_table_insert (elements, GST_MEM_CHUNK_AREA (elem), NULL);
|
|
|
|
data = gst_mem_chunk_alloc (mem_chunk);
|
|
}
|
|
g_hash_table_foreach_remove (elements, free_area, NULL);
|
|
|
|
g_hash_table_destroy (elements);
|
|
g_free (mem_chunk->name);
|
|
g_free (mem_chunk);
|
|
}
|
|
|
|
gpointer
|
|
gst_mem_chunk_alloc (GstMemChunk *mem_chunk)
|
|
{
|
|
GstMemChunkElement *chunk = NULL;
|
|
|
|
g_return_val_if_fail (mem_chunk != NULL, NULL);
|
|
|
|
again:
|
|
GST_ATOMIC_SWAP_GET (&mem_chunk->swap,
|
|
GST_ATOMIC_SWAP_VALUE (&mem_chunk->swap),
|
|
&chunk);
|
|
|
|
if (!chunk) {
|
|
if (populate (mem_chunk))
|
|
goto again;
|
|
else
|
|
return NULL;
|
|
}
|
|
return GST_MEM_CHUNK_DATA (chunk);
|
|
}
|
|
|
|
gpointer
|
|
gst_mem_chunk_alloc0 (GstMemChunk *mem_chunk)
|
|
{
|
|
gpointer mem = gst_mem_chunk_alloc (mem_chunk);
|
|
|
|
if (mem)
|
|
memset (mem, 0, mem_chunk->atom_size);
|
|
|
|
return mem;
|
|
}
|
|
|
|
void
|
|
gst_mem_chunk_free (GstMemChunk *mem_chunk, gpointer mem)
|
|
{
|
|
GstMemChunkElement *chunk;
|
|
|
|
g_return_if_fail (mem_chunk != NULL);
|
|
g_return_if_fail (mem != NULL);
|
|
|
|
chunk = GST_MEM_CHUNK_LINK (mem);
|
|
|
|
GST_ATOMIC_SWAP (&mem_chunk->swap, &chunk->link);
|
|
}
|