os/ossrv/glib/tests/memchunks.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* GLIB - Library of useful routines for C programming
sl@0
     2
 * Copyright (C) 1995-1997  Peter Mattis, Spencer Kimball and Josh MacDonald
sl@0
     3
 * Portions copyright (c) 2009 Nokia Corporation.  All rights reserved.
sl@0
     4
 * This library is free software; you can redistribute it and/or
sl@0
     5
 * modify it under the terms of the GNU Lesser General Public
sl@0
     6
 * License as published by the Free Software Foundation; either
sl@0
     7
 * version 2 of the License, or (at your option) any later version.
sl@0
     8
 *
sl@0
     9
 * This library is distributed in the hope that it will be useful,
sl@0
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
sl@0
    12
 * Lesser General Public License for more details.
sl@0
    13
 *
sl@0
    14
 * You should have received a copy of the GNU Lesser General Public
sl@0
    15
 * License along with this library; if not, write to the
sl@0
    16
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
sl@0
    17
 * Boston, MA 02111-1307, USA.
sl@0
    18
 */
sl@0
    19
sl@0
    20
/*
sl@0
    21
 * Modified by the GLib Team and others 1997-2000.  See the AUTHORS
sl@0
    22
 * file for a list of people on the GLib Team.  See the ChangeLog
sl@0
    23
 * files for a list of changes.  These files are distributed with
sl@0
    24
 * GLib at ftp://ftp.gtk.org/pub/gtk/. 
sl@0
    25
 */
sl@0
    26
sl@0
    27
/* 
sl@0
    28
 * MT safe
sl@0
    29
 */
sl@0
    30
sl@0
    31
#include "config.h"
sl@0
    32
sl@0
    33
#include <stdlib.h>
sl@0
    34
#include <string.h>
sl@0
    35
#include <signal.h>
sl@0
    36
sl@0
    37
#include "glib.h"
sl@0
    38
sl@0
    39
#ifdef __SYMBIAN32__
sl@0
    40
#include <glib_global.h>
sl@0
    41
#endif//__SYMBIAN32__
sl@0
    42
sl@0
    43
/* notes on macros:
sl@0
    44
 * if ENABLE_GC_FRIENDLY is defined, freed memory should be 0-wiped.
sl@0
    45
 */
sl@0
    46
sl@0
    47
#define MEM_PROFILE_TABLE_SIZE 4096
sl@0
    48
sl@0
    49
#define MEM_AREA_SIZE 4L
sl@0
    50
sl@0
    51
static guint mem_chunk_recursion = 0;
sl@0
    52
#  define MEM_CHUNK_ROUTINE_COUNT()	(mem_chunk_recursion)
sl@0
    53
#  define ENTER_MEM_CHUNK_ROUTINE()	(mem_chunk_recursion = MEM_CHUNK_ROUTINE_COUNT () + 1)
sl@0
    54
#  define LEAVE_MEM_CHUNK_ROUTINE()	(mem_chunk_recursion = MEM_CHUNK_ROUTINE_COUNT () - 1)
sl@0
    55
sl@0
    56
/* --- old memchunk prototypes --- */
sl@0
    57
void            old_mem_chunks_init     (void);
sl@0
    58
GMemChunk*      old_mem_chunk_new       (const gchar  *name,
sl@0
    59
                                         gint          atom_size,
sl@0
    60
                                         gulong        area_size,
sl@0
    61
                                         gint          type);
sl@0
    62
void            old_mem_chunk_destroy   (GMemChunk *mem_chunk);
sl@0
    63
gpointer        old_mem_chunk_alloc     (GMemChunk *mem_chunk);
sl@0
    64
gpointer        old_mem_chunk_alloc0    (GMemChunk *mem_chunk);
sl@0
    65
void            old_mem_chunk_free      (GMemChunk *mem_chunk,
sl@0
    66
                                         gpointer   mem);
sl@0
    67
void            old_mem_chunk_clean     (GMemChunk *mem_chunk);
sl@0
    68
void            old_mem_chunk_reset     (GMemChunk *mem_chunk);
sl@0
    69
void            old_mem_chunk_print     (GMemChunk *mem_chunk);
sl@0
    70
void            old_mem_chunk_info      (void);
sl@0
    71
sl@0
    72
sl@0
    73
/* --- MemChunks --- */
sl@0
    74
#ifndef G_ALLOC_AND_FREE
sl@0
    75
typedef struct _GAllocator GAllocator;
sl@0
    76
typedef struct _GMemChunk  GMemChunk;
sl@0
    77
#define G_ALLOC_ONLY	  1
sl@0
    78
#define G_ALLOC_AND_FREE  2
sl@0
    79
#endif
sl@0
    80
sl@0
    81
typedef struct _GFreeAtom      GFreeAtom;
sl@0
    82
typedef struct _GMemArea       GMemArea;
sl@0
    83
sl@0
    84
struct _GFreeAtom
sl@0
    85
{
sl@0
    86
  GFreeAtom *next;
sl@0
    87
};
sl@0
    88
sl@0
    89
struct _GMemArea
sl@0
    90
{
sl@0
    91
  GMemArea *next;            /* the next mem area */
sl@0
    92
  GMemArea *prev;            /* the previous mem area */
sl@0
    93
  gulong index;              /* the current index into the "mem" array */
sl@0
    94
  gulong free;               /* the number of free bytes in this mem area */
sl@0
    95
  gulong allocated;          /* the number of atoms allocated from this area */
sl@0
    96
  gulong mark;               /* is this mem area marked for deletion */
sl@0
    97
  gchar mem[MEM_AREA_SIZE];  /* the mem array from which atoms get allocated
sl@0
    98
			      * the actual size of this array is determined by
sl@0
    99
			      *  the mem chunk "area_size". ANSI says that it
sl@0
   100
			      *  must be declared to be the maximum size it
sl@0
   101
			      *  can possibly be (even though the actual size
sl@0
   102
			      *  may be less).
sl@0
   103
			      */
sl@0
   104
};
sl@0
   105
sl@0
   106
struct _GMemChunk
sl@0
   107
{
sl@0
   108
  const gchar *name;         /* name of this MemChunk...used for debugging output */
sl@0
   109
  gint type;                 /* the type of MemChunk: ALLOC_ONLY or ALLOC_AND_FREE */
sl@0
   110
  gint num_mem_areas;        /* the number of memory areas */
sl@0
   111
  gint num_marked_areas;     /* the number of areas marked for deletion */
sl@0
   112
  guint atom_size;           /* the size of an atom */
sl@0
   113
  gulong area_size;          /* the size of a memory area */
sl@0
   114
  GMemArea *mem_area;        /* the current memory area */
sl@0
   115
  GMemArea *mem_areas;       /* a list of all the mem areas owned by this chunk */
sl@0
   116
  GMemArea *free_mem_area;   /* the free area...which is about to be destroyed */
sl@0
   117
  GFreeAtom *free_atoms;     /* the free atoms list */
sl@0
   118
  GTree *mem_tree;           /* tree of mem areas sorted by memory address */
sl@0
   119
  GMemChunk *next;           /* pointer to the next chunk */
sl@0
   120
  GMemChunk *prev;           /* pointer to the previous chunk */
sl@0
   121
};
sl@0
   122
sl@0
   123
sl@0
   124
static gulong old_mem_chunk_compute_size (gulong    size,
sl@0
   125
                                          gulong    min_size) G_GNUC_CONST;
sl@0
   126
static gint   old_mem_chunk_area_compare (GMemArea *a,
sl@0
   127
                                          GMemArea *b);
sl@0
   128
static gint   old_mem_chunk_area_search  (GMemArea *a,
sl@0
   129
                                          gchar    *addr);
sl@0
   130
sl@0
   131
/* here we can't use StaticMutexes, as they depend upon a working
sl@0
   132
 * g_malloc, the same holds true for StaticPrivate
sl@0
   133
 */
sl@0
   134
static GMutex        *mem_chunks_lock = NULL;
sl@0
   135
static GMemChunk     *mem_chunks = NULL;
sl@0
   136
sl@0
   137
void
sl@0
   138
old_mem_chunks_init (void)
sl@0
   139
{
sl@0
   140
  mem_chunks_lock = g_mutex_new ();
sl@0
   141
}
sl@0
   142
sl@0
   143
GMemChunk*
sl@0
   144
old_mem_chunk_new (const gchar  *name,
sl@0
   145
                   gint          atom_size,
sl@0
   146
                   gulong        area_size,
sl@0
   147
                   gint          type)
sl@0
   148
{
sl@0
   149
  GMemChunk *mem_chunk;
sl@0
   150
  gulong rarea_size;
sl@0
   151
  
sl@0
   152
  g_return_val_if_fail (atom_size > 0, NULL);
sl@0
   153
  g_return_val_if_fail (area_size >= atom_size, NULL);
sl@0
   154
  
sl@0
   155
  ENTER_MEM_CHUNK_ROUTINE ();
sl@0
   156
  
sl@0
   157
  area_size = (area_size + atom_size - 1) / atom_size;
sl@0
   158
  area_size *= atom_size;
sl@0
   159
  
sl@0
   160
  mem_chunk = g_new (GMemChunk, 1);
sl@0
   161
  mem_chunk->name = name;
sl@0
   162
  mem_chunk->type = type;
sl@0
   163
  mem_chunk->num_mem_areas = 0;
sl@0
   164
  mem_chunk->num_marked_areas = 0;
sl@0
   165
  mem_chunk->mem_area = NULL;
sl@0
   166
  mem_chunk->free_mem_area = NULL;
sl@0
   167
  mem_chunk->free_atoms = NULL;
sl@0
   168
  mem_chunk->mem_tree = NULL;
sl@0
   169
  mem_chunk->mem_areas = NULL;
sl@0
   170
  mem_chunk->atom_size = atom_size;
sl@0
   171
  
sl@0
   172
  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   173
    mem_chunk->mem_tree = g_tree_new ((GCompareFunc) old_mem_chunk_area_compare);
sl@0
   174
  
sl@0
   175
  if (mem_chunk->atom_size % G_MEM_ALIGN)
sl@0
   176
    mem_chunk->atom_size += G_MEM_ALIGN - (mem_chunk->atom_size % G_MEM_ALIGN);
sl@0
   177
  
sl@0
   178
  rarea_size = area_size + sizeof (GMemArea) - MEM_AREA_SIZE;
sl@0
   179
  rarea_size = old_mem_chunk_compute_size (rarea_size, atom_size + sizeof (GMemArea) - MEM_AREA_SIZE);
sl@0
   180
  mem_chunk->area_size = rarea_size - (sizeof (GMemArea) - MEM_AREA_SIZE);
sl@0
   181
  
sl@0
   182
  g_mutex_lock (mem_chunks_lock);
sl@0
   183
  mem_chunk->next = mem_chunks;
sl@0
   184
  mem_chunk->prev = NULL;
sl@0
   185
  if (mem_chunks)
sl@0
   186
    mem_chunks->prev = mem_chunk;
sl@0
   187
  mem_chunks = mem_chunk;
sl@0
   188
  g_mutex_unlock (mem_chunks_lock);
sl@0
   189
  
sl@0
   190
  LEAVE_MEM_CHUNK_ROUTINE ();
sl@0
   191
  
sl@0
   192
  return mem_chunk;
sl@0
   193
}
sl@0
   194
sl@0
   195
void
sl@0
   196
old_mem_chunk_destroy (GMemChunk *mem_chunk)
sl@0
   197
{
sl@0
   198
  GMemArea *mem_areas;
sl@0
   199
  GMemArea *temp_area;
sl@0
   200
  
sl@0
   201
  g_return_if_fail (mem_chunk != NULL);
sl@0
   202
  
sl@0
   203
  ENTER_MEM_CHUNK_ROUTINE ();
sl@0
   204
  
sl@0
   205
  mem_areas = mem_chunk->mem_areas;
sl@0
   206
  while (mem_areas)
sl@0
   207
    {
sl@0
   208
      temp_area = mem_areas;
sl@0
   209
      mem_areas = mem_areas->next;
sl@0
   210
      g_free (temp_area);
sl@0
   211
    }
sl@0
   212
  
sl@0
   213
  g_mutex_lock (mem_chunks_lock);
sl@0
   214
  if (mem_chunk->next)
sl@0
   215
    mem_chunk->next->prev = mem_chunk->prev;
sl@0
   216
  if (mem_chunk->prev)
sl@0
   217
    mem_chunk->prev->next = mem_chunk->next;
sl@0
   218
  
sl@0
   219
  if (mem_chunk == mem_chunks)
sl@0
   220
    mem_chunks = mem_chunks->next;
sl@0
   221
  g_mutex_unlock (mem_chunks_lock);
sl@0
   222
  
sl@0
   223
  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   224
    g_tree_destroy (mem_chunk->mem_tree);  
sl@0
   225
  
sl@0
   226
  g_free (mem_chunk);
sl@0
   227
  
sl@0
   228
  LEAVE_MEM_CHUNK_ROUTINE ();
sl@0
   229
}
sl@0
   230
sl@0
   231
gpointer
sl@0
   232
old_mem_chunk_alloc (GMemChunk *mem_chunk)
sl@0
   233
{
sl@0
   234
  GMemArea *temp_area;
sl@0
   235
  gpointer mem;
sl@0
   236
  
sl@0
   237
  ENTER_MEM_CHUNK_ROUTINE ();
sl@0
   238
  
sl@0
   239
  g_return_val_if_fail (mem_chunk != NULL, NULL);
sl@0
   240
  
sl@0
   241
  while (mem_chunk->free_atoms)
sl@0
   242
    {
sl@0
   243
      /* Get the first piece of memory on the "free_atoms" list.
sl@0
   244
       * We can go ahead and destroy the list node we used to keep
sl@0
   245
       *  track of it with and to update the "free_atoms" list to
sl@0
   246
       *  point to its next element.
sl@0
   247
       */
sl@0
   248
      mem = mem_chunk->free_atoms;
sl@0
   249
      mem_chunk->free_atoms = mem_chunk->free_atoms->next;
sl@0
   250
      
sl@0
   251
      /* Determine which area this piece of memory is allocated from */
sl@0
   252
      temp_area = g_tree_search (mem_chunk->mem_tree,
sl@0
   253
				 (GCompareFunc) old_mem_chunk_area_search,
sl@0
   254
				 mem);
sl@0
   255
      
sl@0
   256
      /* If the area has been marked, then it is being destroyed.
sl@0
   257
       *  (ie marked to be destroyed).
sl@0
   258
       * We check to see if all of the segments on the free list that
sl@0
   259
       *  reference this area have been removed. This occurs when
sl@0
   260
       *  the ammount of free memory is less than the allocatable size.
sl@0
   261
       * If the chunk should be freed, then we place it in the "free_mem_area".
sl@0
   262
       * This is so we make sure not to free the mem area here and then
sl@0
   263
       *  allocate it again a few lines down.
sl@0
   264
       * If we don't allocate a chunk a few lines down then the "free_mem_area"
sl@0
   265
       *  will be freed.
sl@0
   266
       * If there is already a "free_mem_area" then we'll just free this mem area.
sl@0
   267
       */
sl@0
   268
      if (temp_area->mark)
sl@0
   269
        {
sl@0
   270
          /* Update the "free" memory available in that area */
sl@0
   271
          temp_area->free += mem_chunk->atom_size;
sl@0
   272
	  
sl@0
   273
          if (temp_area->free == mem_chunk->area_size)
sl@0
   274
            {
sl@0
   275
              if (temp_area == mem_chunk->mem_area)
sl@0
   276
                mem_chunk->mem_area = NULL;
sl@0
   277
	      
sl@0
   278
              if (mem_chunk->free_mem_area)
sl@0
   279
                {
sl@0
   280
                  mem_chunk->num_mem_areas -= 1;
sl@0
   281
		  
sl@0
   282
                  if (temp_area->next)
sl@0
   283
                    temp_area->next->prev = temp_area->prev;
sl@0
   284
                  if (temp_area->prev)
sl@0
   285
                    temp_area->prev->next = temp_area->next;
sl@0
   286
                  if (temp_area == mem_chunk->mem_areas)
sl@0
   287
                    mem_chunk->mem_areas = mem_chunk->mem_areas->next;
sl@0
   288
		  
sl@0
   289
		  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   290
		    g_tree_remove (mem_chunk->mem_tree, temp_area);
sl@0
   291
                  g_free (temp_area);
sl@0
   292
                }
sl@0
   293
              else
sl@0
   294
                mem_chunk->free_mem_area = temp_area;
sl@0
   295
	      
sl@0
   296
	      mem_chunk->num_marked_areas -= 1;
sl@0
   297
	    }
sl@0
   298
	}
sl@0
   299
      else
sl@0
   300
        {
sl@0
   301
          /* Update the number of allocated atoms count.
sl@0
   302
	   */
sl@0
   303
          temp_area->allocated += 1;
sl@0
   304
	  
sl@0
   305
          /* The area wasn't marked...return the memory
sl@0
   306
	   */
sl@0
   307
	  goto outa_here;
sl@0
   308
        }
sl@0
   309
    }
sl@0
   310
  
sl@0
   311
  /* If there isn't a current mem area or the current mem area is out of space
sl@0
   312
   *  then allocate a new mem area. We'll first check and see if we can use
sl@0
   313
   *  the "free_mem_area". Otherwise we'll just malloc the mem area.
sl@0
   314
   */
sl@0
   315
  if ((!mem_chunk->mem_area) ||
sl@0
   316
      ((mem_chunk->mem_area->index + mem_chunk->atom_size) > mem_chunk->area_size))
sl@0
   317
    {
sl@0
   318
      if (mem_chunk->free_mem_area)
sl@0
   319
        {
sl@0
   320
          mem_chunk->mem_area = mem_chunk->free_mem_area;
sl@0
   321
	  mem_chunk->free_mem_area = NULL;
sl@0
   322
        }
sl@0
   323
      else
sl@0
   324
        {
sl@0
   325
#ifdef ENABLE_GC_FRIENDLY
sl@0
   326
	  mem_chunk->mem_area = (GMemArea*) g_malloc0 (sizeof (GMemArea) -
sl@0
   327
						       MEM_AREA_SIZE +
sl@0
   328
						       mem_chunk->area_size); 
sl@0
   329
#else /* !ENABLE_GC_FRIENDLY */
sl@0
   330
	  mem_chunk->mem_area = (GMemArea*) g_malloc (sizeof (GMemArea) -
sl@0
   331
						      MEM_AREA_SIZE +
sl@0
   332
						      mem_chunk->area_size);
sl@0
   333
#endif /* ENABLE_GC_FRIENDLY */
sl@0
   334
	  
sl@0
   335
	  mem_chunk->num_mem_areas += 1;
sl@0
   336
	  mem_chunk->mem_area->next = mem_chunk->mem_areas;
sl@0
   337
	  mem_chunk->mem_area->prev = NULL;
sl@0
   338
	  
sl@0
   339
	  if (mem_chunk->mem_areas)
sl@0
   340
	    mem_chunk->mem_areas->prev = mem_chunk->mem_area;
sl@0
   341
	  mem_chunk->mem_areas = mem_chunk->mem_area;
sl@0
   342
	  
sl@0
   343
	  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   344
	    g_tree_insert (mem_chunk->mem_tree, mem_chunk->mem_area, mem_chunk->mem_area);
sl@0
   345
        }
sl@0
   346
      
sl@0
   347
      mem_chunk->mem_area->index = 0;
sl@0
   348
      mem_chunk->mem_area->free = mem_chunk->area_size;
sl@0
   349
      mem_chunk->mem_area->allocated = 0;
sl@0
   350
      mem_chunk->mem_area->mark = 0;
sl@0
   351
    }
sl@0
   352
  
sl@0
   353
  /* Get the memory and modify the state variables appropriately.
sl@0
   354
   */
sl@0
   355
  mem = (gpointer) &mem_chunk->mem_area->mem[mem_chunk->mem_area->index];
sl@0
   356
  mem_chunk->mem_area->index += mem_chunk->atom_size;
sl@0
   357
  mem_chunk->mem_area->free -= mem_chunk->atom_size;
sl@0
   358
  mem_chunk->mem_area->allocated += 1;
sl@0
   359
  
sl@0
   360
 outa_here:
sl@0
   361
  
sl@0
   362
  LEAVE_MEM_CHUNK_ROUTINE ();
sl@0
   363
  
sl@0
   364
  return mem;
sl@0
   365
}
sl@0
   366
sl@0
   367
gpointer
sl@0
   368
old_mem_chunk_alloc0 (GMemChunk *mem_chunk)
sl@0
   369
{
sl@0
   370
  gpointer mem;
sl@0
   371
  
sl@0
   372
  mem = old_mem_chunk_alloc (mem_chunk);
sl@0
   373
  if (mem)
sl@0
   374
    {
sl@0
   375
      memset (mem, 0, mem_chunk->atom_size);
sl@0
   376
    }
sl@0
   377
  
sl@0
   378
  return mem;
sl@0
   379
}
sl@0
   380
sl@0
   381
void
sl@0
   382
old_mem_chunk_free (GMemChunk *mem_chunk,
sl@0
   383
                    gpointer   mem)
sl@0
   384
{
sl@0
   385
  GMemArea *temp_area;
sl@0
   386
  GFreeAtom *free_atom;
sl@0
   387
  
sl@0
   388
  g_return_if_fail (mem_chunk != NULL);
sl@0
   389
  g_return_if_fail (mem != NULL);
sl@0
   390
  
sl@0
   391
  ENTER_MEM_CHUNK_ROUTINE ();
sl@0
   392
  
sl@0
   393
#ifdef ENABLE_GC_FRIENDLY
sl@0
   394
  memset (mem, 0, mem_chunk->atom_size);
sl@0
   395
#endif /* ENABLE_GC_FRIENDLY */
sl@0
   396
  
sl@0
   397
  /* Don't do anything if this is an ALLOC_ONLY chunk
sl@0
   398
   */
sl@0
   399
  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   400
    {
sl@0
   401
      /* Place the memory on the "free_atoms" list
sl@0
   402
       */
sl@0
   403
      free_atom = (GFreeAtom*) mem;
sl@0
   404
      free_atom->next = mem_chunk->free_atoms;
sl@0
   405
      mem_chunk->free_atoms = free_atom;
sl@0
   406
      
sl@0
   407
      temp_area = g_tree_search (mem_chunk->mem_tree,
sl@0
   408
				 (GCompareFunc) old_mem_chunk_area_search,
sl@0
   409
				 mem);
sl@0
   410
      
sl@0
   411
      temp_area->allocated -= 1;
sl@0
   412
      
sl@0
   413
      if (temp_area->allocated == 0)
sl@0
   414
	{
sl@0
   415
	  temp_area->mark = 1;
sl@0
   416
	  mem_chunk->num_marked_areas += 1;
sl@0
   417
	}
sl@0
   418
    }
sl@0
   419
  
sl@0
   420
  LEAVE_MEM_CHUNK_ROUTINE ();
sl@0
   421
}
sl@0
   422
sl@0
   423
/* This doesn't free the free_area if there is one */
sl@0
   424
void
sl@0
   425
old_mem_chunk_clean (GMemChunk *mem_chunk)
sl@0
   426
{
sl@0
   427
  GMemArea *mem_area;
sl@0
   428
  GFreeAtom *prev_free_atom;
sl@0
   429
  GFreeAtom *temp_free_atom;
sl@0
   430
  gpointer mem;
sl@0
   431
  
sl@0
   432
  g_return_if_fail (mem_chunk != NULL);
sl@0
   433
  
sl@0
   434
  ENTER_MEM_CHUNK_ROUTINE ();
sl@0
   435
  
sl@0
   436
  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   437
    {
sl@0
   438
      prev_free_atom = NULL;
sl@0
   439
      temp_free_atom = mem_chunk->free_atoms;
sl@0
   440
      
sl@0
   441
      while (temp_free_atom)
sl@0
   442
	{
sl@0
   443
	  mem = (gpointer) temp_free_atom;
sl@0
   444
	  
sl@0
   445
	  mem_area = g_tree_search (mem_chunk->mem_tree,
sl@0
   446
				    (GCompareFunc) old_mem_chunk_area_search,
sl@0
   447
				    mem);
sl@0
   448
	  
sl@0
   449
          /* If this mem area is marked for destruction then delete the
sl@0
   450
	   *  area and list node and decrement the free mem.
sl@0
   451
           */
sl@0
   452
	  if (mem_area->mark)
sl@0
   453
	    {
sl@0
   454
	      if (prev_free_atom)
sl@0
   455
		prev_free_atom->next = temp_free_atom->next;
sl@0
   456
	      else
sl@0
   457
		mem_chunk->free_atoms = temp_free_atom->next;
sl@0
   458
	      temp_free_atom = temp_free_atom->next;
sl@0
   459
	      
sl@0
   460
	      mem_area->free += mem_chunk->atom_size;
sl@0
   461
	      if (mem_area->free == mem_chunk->area_size)
sl@0
   462
		{
sl@0
   463
		  mem_chunk->num_mem_areas -= 1;
sl@0
   464
		  mem_chunk->num_marked_areas -= 1;
sl@0
   465
		  
sl@0
   466
		  if (mem_area->next)
sl@0
   467
		    mem_area->next->prev = mem_area->prev;
sl@0
   468
		  if (mem_area->prev)
sl@0
   469
		    mem_area->prev->next = mem_area->next;
sl@0
   470
		  if (mem_area == mem_chunk->mem_areas)
sl@0
   471
		    mem_chunk->mem_areas = mem_chunk->mem_areas->next;
sl@0
   472
		  if (mem_area == mem_chunk->mem_area)
sl@0
   473
		    mem_chunk->mem_area = NULL;
sl@0
   474
		  
sl@0
   475
		  if (mem_chunk->type == G_ALLOC_AND_FREE)
sl@0
   476
		    g_tree_remove (mem_chunk->mem_tree, mem_area);
sl@0
   477
		  g_free (mem_area);
sl@0
   478
		}
sl@0
   479
	    }
sl@0
   480
	  else
sl@0
   481
	    {
sl@0
   482
	      prev_free_atom = temp_free_atom;
sl@0
   483
	      temp_free_atom = temp_free_atom->next;
sl@0
   484
	    }
sl@0
   485
	}
sl@0
   486
    }
sl@0
   487
  LEAVE_MEM_CHUNK_ROUTINE ();
sl@0
   488
}
sl@0
   489
sl@0
   490
void
sl@0
   491
old_mem_chunk_reset (GMemChunk *mem_chunk)
sl@0
   492
{
sl@0
   493
  GMemArea *mem_areas;
sl@0
   494
  GMemArea *temp_area;
sl@0
   495
  
sl@0
   496
  g_return_if_fail (mem_chunk != NULL);
sl@0
   497
  
sl@0
   498
  ENTER_MEM_CHUNK_ROUTINE ();
sl@0
   499
  
sl@0
   500
  mem_areas = mem_chunk->mem_areas;
sl@0
   501
  mem_chunk->num_mem_areas = 0;
sl@0
   502
  mem_chunk->mem_areas = NULL;
sl@0
   503
  mem_chunk->mem_area = NULL;
sl@0
   504
  
sl@0
   505
  while (mem_areas)
sl@0
   506
    {
sl@0
   507
      temp_area = mem_areas;
sl@0
   508
      mem_areas = mem_areas->next;
sl@0
   509
      g_free (temp_area);
sl@0
   510
    }
sl@0
   511
  
sl@0
   512
  mem_chunk->free_atoms = NULL;
sl@0
   513
  
sl@0
   514
  if (mem_chunk->mem_tree)
sl@0
   515
    {
sl@0
   516
      g_tree_destroy (mem_chunk->mem_tree);
sl@0
   517
      mem_chunk->mem_tree = g_tree_new ((GCompareFunc) old_mem_chunk_area_compare);
sl@0
   518
    }
sl@0
   519
  
sl@0
   520
  LEAVE_MEM_CHUNK_ROUTINE ();
sl@0
   521
}
sl@0
   522
sl@0
   523
void
sl@0
   524
old_mem_chunk_print (GMemChunk *mem_chunk)
sl@0
   525
{
sl@0
   526
  GMemArea *mem_areas;
sl@0
   527
  gulong mem;
sl@0
   528
  
sl@0
   529
  g_return_if_fail (mem_chunk != NULL);
sl@0
   530
  
sl@0
   531
  mem_areas = mem_chunk->mem_areas;
sl@0
   532
  mem = 0;
sl@0
   533
  
sl@0
   534
  while (mem_areas)
sl@0
   535
    {
sl@0
   536
      mem += mem_chunk->area_size - mem_areas->free;
sl@0
   537
      mem_areas = mem_areas->next;
sl@0
   538
    }
sl@0
   539
  
sl@0
   540
  g_log (G_LOG_DOMAIN, G_LOG_LEVEL_INFO,
sl@0
   541
	 "%s: %ld bytes using %d mem areas",
sl@0
   542
	 mem_chunk->name, mem, mem_chunk->num_mem_areas);
sl@0
   543
}
sl@0
   544
sl@0
   545
void
sl@0
   546
old_mem_chunk_info (void)
sl@0
   547
{
sl@0
   548
  GMemChunk *mem_chunk;
sl@0
   549
  gint count;
sl@0
   550
  
sl@0
   551
  count = 0;
sl@0
   552
  g_mutex_lock (mem_chunks_lock);
sl@0
   553
  mem_chunk = mem_chunks;
sl@0
   554
  while (mem_chunk)
sl@0
   555
    {
sl@0
   556
      count += 1;
sl@0
   557
      mem_chunk = mem_chunk->next;
sl@0
   558
    }
sl@0
   559
  g_mutex_unlock (mem_chunks_lock);
sl@0
   560
  
sl@0
   561
  g_log (G_LOG_DOMAIN, G_LOG_LEVEL_INFO, "%d mem chunks", count);
sl@0
   562
  
sl@0
   563
  g_mutex_lock (mem_chunks_lock);
sl@0
   564
  mem_chunk = mem_chunks;
sl@0
   565
  g_mutex_unlock (mem_chunks_lock);
sl@0
   566
  
sl@0
   567
  while (mem_chunk)
sl@0
   568
    {
sl@0
   569
      old_mem_chunk_print ((GMemChunk*) mem_chunk);
sl@0
   570
      mem_chunk = mem_chunk->next;
sl@0
   571
    }  
sl@0
   572
}
sl@0
   573
sl@0
   574
static gulong
sl@0
   575
old_mem_chunk_compute_size (gulong size,
sl@0
   576
                            gulong min_size)
sl@0
   577
{
sl@0
   578
  gulong power_of_2;
sl@0
   579
  gulong lower, upper;
sl@0
   580
  
sl@0
   581
  power_of_2 = 16;
sl@0
   582
  while (power_of_2 < size)
sl@0
   583
    power_of_2 <<= 1;
sl@0
   584
  
sl@0
   585
  lower = power_of_2 >> 1;
sl@0
   586
  upper = power_of_2;
sl@0
   587
  
sl@0
   588
  if (size - lower < upper - size && lower >= min_size)
sl@0
   589
    return lower;
sl@0
   590
  else
sl@0
   591
    return upper;
sl@0
   592
}
sl@0
   593
sl@0
   594
static gint
sl@0
   595
old_mem_chunk_area_compare (GMemArea *a,
sl@0
   596
                            GMemArea *b)
sl@0
   597
{
sl@0
   598
  if (a->mem > b->mem)
sl@0
   599
    return 1;
sl@0
   600
  else if (a->mem < b->mem)
sl@0
   601
    return -1;
sl@0
   602
  return 0;
sl@0
   603
}
sl@0
   604
sl@0
   605
static gint
sl@0
   606
old_mem_chunk_area_search (GMemArea *a,
sl@0
   607
                           gchar    *addr)
sl@0
   608
{
sl@0
   609
  if (a->mem <= addr)
sl@0
   610
    {
sl@0
   611
      if (addr < &a->mem[a->index])
sl@0
   612
	return 0;
sl@0
   613
      return 1;
sl@0
   614
    }
sl@0
   615
  return -1;
sl@0
   616
}