os/ossrv/glib/tests/slice-concurrent.c
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/glib/tests/slice-concurrent.c	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,131 @@
     1.4 +/* test for gslice cross thread allocation/free
     1.5 + * Copyright (C) 2006 Stefan Westerfeld
     1.6 + * Copyright (C) 2007 Tim Janik
     1.7 + * Portions copyright (c) 2009 Nokia Corporation.  All rights reserved.
     1.8 + * This library is free software; you can redistribute it and/or
     1.9 + * modify it under the terms of the GNU Lesser General Public
    1.10 + * License as published by the Free Software Foundation; either
    1.11 + * version 2 of the License, or (at your option) any later version.
    1.12 + *
    1.13 + * This library is distributed in the hope that it will be useful,
    1.14 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
    1.15 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
    1.16 + * Lesser General Public License for more details.
    1.17 + *
    1.18 + * You should have received a copy of the GNU Lesser General Public
    1.19 + * License along with this library; if not, write to the
    1.20 + * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
    1.21 + * Boston, MA 02111-1307, USA.
    1.22 + */
    1.23 +#include <glib.h>
    1.24 +#include <stdlib.h>
    1.25 +#include <unistd.h>
    1.26 +#ifdef __SYMBIAN32__
    1.27 +#include <glib_global.h>
    1.28 +#include "mrt2_glib2_test.h"
    1.29 +#endif /*__SYMBIAN32__*/
    1.30 +#define N_THREADS	2
    1.31 +#define N_ALLOCS	500
    1.32 +#define MAX_BLOCK_SIZE  64
    1.33 +
    1.34 +
    1.35 +struct ThreadData
    1.36 +{
    1.37 +  int	   thread_id;
    1.38 +  GThread* gthread;
    1.39 +
    1.40 +  GMutex*  to_free_mutex;
    1.41 +  void*    to_free [N_THREADS * N_ALLOCS];
    1.42 +  int      bytes_to_free [N_THREADS * N_ALLOCS];
    1.43 +  int      n_to_free;
    1.44 +  int      n_freed;
    1.45 +} tdata[N_THREADS];
    1.46 +
    1.47 +void*
    1.48 +thread_func (void *arg)
    1.49 +{
    1.50 +  struct ThreadData *td = arg;
    1.51 +  int i, bytes, f, t;
    1.52 +  char *mem;
    1.53 +  // g_print ("Thread %d starting\n", td->thread_id);
    1.54 +  for (i = 0; i < N_ALLOCS; i++)
    1.55 +    {
    1.56 +      if (rand() % (N_ALLOCS / 20) == 0)
    1.57 +	g_print ("%c", 'a' - 1 + td->thread_id);
    1.58 +
    1.59 +      /* allocate block of random size and randomly fill */
    1.60 +        bytes = rand() % MAX_BLOCK_SIZE + 1;
    1.61 +/*      char **/mem = g_slice_alloc (bytes);
    1.62 +//      int f;
    1.63 +      for (f = 0; f < bytes; f++)
    1.64 +	mem[f] = rand();
    1.65 +
    1.66 +      /* associate block with random thread */
    1.67 +      /*int*/ t = rand() % N_THREADS;
    1.68 +      g_mutex_lock (tdata[t].to_free_mutex);
    1.69 +      tdata[t].to_free[tdata[t].n_to_free] = mem;
    1.70 +      tdata[t].bytes_to_free[tdata[t].n_to_free] = bytes;
    1.71 +      tdata[t].n_to_free++;
    1.72 +      g_mutex_unlock (tdata[t].to_free_mutex);
    1.73 +
    1.74 +      /* shuffle thread execution order every once in a while */
    1.75 +      if (rand() % 97 == 0)
    1.76 +        {
    1.77 +          if (rand() % 2)
    1.78 +            g_thread_yield();   /* concurrent shuffling for single core */
    1.79 +          else
    1.80 +            g_usleep (1000);    /* concurrent shuffling for multi core */
    1.81 +        }
    1.82 +
    1.83 +      /* free a block associated with this thread */
    1.84 +      g_mutex_lock (td->to_free_mutex);
    1.85 +      if (td->n_to_free > 0)
    1.86 +	{
    1.87 +	  td->n_to_free--;
    1.88 +	  g_slice_free1 (td->bytes_to_free[td->n_to_free], td->to_free[td->n_to_free]);
    1.89 +	  td->n_freed++;
    1.90 +	}
    1.91 +      g_mutex_unlock (td->to_free_mutex);
    1.92 +    }
    1.93 +
    1.94 +  return NULL;
    1.95 +}
    1.96 +
    1.97 +int
    1.98 +main()
    1.99 +{
   1.100 +  int t;
   1.101 +#ifdef __SYMBIAN32__
   1.102 +  g_log_set_handler (NULL,  G_LOG_FLAG_FATAL| G_LOG_FLAG_RECURSION | G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_WARNING | G_LOG_LEVEL_MESSAGE | G_LOG_LEVEL_INFO | G_LOG_LEVEL_DEBUG, &mrtLogHandler, NULL);
   1.103 +  g_set_print_handler(mrtPrintHandler);
   1.104 +  #endif /*__SYMBIAN32__*/
   1.105 +  g_thread_init (NULL);
   1.106 +
   1.107 +  for (t = 0; t < N_THREADS; t++)
   1.108 +    {
   1.109 +      tdata[t].thread_id = t + 1;
   1.110 +      tdata[t].n_to_free = 0;
   1.111 +      tdata[t].n_freed = 0;
   1.112 +      tdata[t].to_free_mutex = g_mutex_new();
   1.113 +    }
   1.114 +  g_print ("Starting %d threads for concurrent GSlice usage...\n", N_THREADS);
   1.115 +  for (t = 0; t < N_THREADS; t++)
   1.116 +    {
   1.117 +      tdata[t].gthread   = g_thread_create (thread_func, &tdata[t], TRUE, NULL);
   1.118 +      g_assert (tdata[t].gthread != NULL);
   1.119 +    }
   1.120 +  for (t = 0; t < N_THREADS; t++)
   1.121 +    {
   1.122 +      g_thread_join (tdata[t].gthread);
   1.123 +    }
   1.124 +  g_print ("\n");
   1.125 +  for (t = 0; t < N_THREADS; t++)
   1.126 +    {
   1.127 +      g_print ("Thread %d: %d blocks freed, %d blocks not freed\n",
   1.128 +		    tdata[t].thread_id, tdata[t].n_freed, tdata[t].n_to_free);
   1.129 +    }
   1.130 +#if __SYMBIAN32__
   1.131 +  testResultXml("slice-concurrent");
   1.132 +  #endif /* EMULATOR */	
   1.133 +  return 0;
   1.134 +}