Blame glib/tests/gwakeuptest.c

Packit Service d3d246
#include <glib.h>
Packit Service d3d246
#include <glib/gwakeup.h>
Packit Service d3d246
#ifdef G_OS_UNIX
Packit Service d3d246
#include <unistd.h>
Packit Service d3d246
#endif
Packit Service d3d246
Packit Service d3d246
#ifdef _WIN32
Packit Service d3d246
static void alarm (int sec) { }
Packit Service d3d246
#endif
Packit Service d3d246
Packit Service d3d246
static gboolean
Packit Service d3d246
check_signaled (GWakeup *wakeup)
Packit Service d3d246
{
Packit Service d3d246
  GPollFD fd;
Packit Service d3d246
Packit Service d3d246
  g_wakeup_get_pollfd (wakeup, &fd;;
Packit Service d3d246
  return g_poll (&fd, 1, 0);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
wait_for_signaled (GWakeup *wakeup)
Packit Service d3d246
{
Packit Service d3d246
  GPollFD fd;
Packit Service d3d246
Packit Service d3d246
  g_wakeup_get_pollfd (wakeup, &fd;;
Packit Service d3d246
  g_poll (&fd, 1, -1);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
test_semantics (void)
Packit Service d3d246
{
Packit Service d3d246
  GWakeup *wakeup;
Packit Service d3d246
  gint i;
Packit Service d3d246
Packit Service d3d246
  /* prevent the test from deadlocking */
Packit Service d3d246
  alarm (60);
Packit Service d3d246
Packit Service d3d246
  wakeup = g_wakeup_new ();
Packit Service d3d246
  g_assert (!check_signaled (wakeup));
Packit Service d3d246
Packit Service d3d246
  g_wakeup_signal (wakeup);
Packit Service d3d246
  g_assert (check_signaled (wakeup));
Packit Service d3d246
Packit Service d3d246
  g_wakeup_acknowledge (wakeup);
Packit Service d3d246
  g_assert (!check_signaled (wakeup));
Packit Service d3d246
Packit Service d3d246
  g_wakeup_free (wakeup);
Packit Service d3d246
Packit Service d3d246
  /* free unused */
Packit Service d3d246
  wakeup = g_wakeup_new ();
Packit Service d3d246
  g_wakeup_free (wakeup);
Packit Service d3d246
Packit Service d3d246
  /* free while signaled */
Packit Service d3d246
  wakeup = g_wakeup_new ();
Packit Service d3d246
  g_wakeup_signal (wakeup);
Packit Service d3d246
  g_wakeup_free (wakeup);
Packit Service d3d246
Packit Service d3d246
  /* ensure excessive signalling doesn't deadlock */
Packit Service d3d246
  wakeup = g_wakeup_new ();
Packit Service d3d246
  for (i = 0; i < 1000000; i++)
Packit Service d3d246
    g_wakeup_signal (wakeup);
Packit Service d3d246
  g_assert (check_signaled (wakeup));
Packit Service d3d246
Packit Service d3d246
  /* ensure a single acknowledgement is sufficient */
Packit Service d3d246
  g_wakeup_acknowledge (wakeup);
Packit Service d3d246
  g_assert (!check_signaled (wakeup));
Packit Service d3d246
Packit Service d3d246
  g_wakeup_free (wakeup);
Packit Service d3d246
Packit Service d3d246
  /* cancel the alarm */
Packit Service d3d246
  alarm (0);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
struct token
Packit Service d3d246
{
Packit Service d3d246
  gpointer owner;
Packit Service d3d246
  gint ttl;
Packit Service d3d246
};
Packit Service d3d246
Packit Service d3d246
struct context
Packit Service d3d246
{
Packit Service d3d246
  GSList *pending_tokens;
Packit Service d3d246
  GMutex lock;
Packit Service d3d246
  GWakeup *wakeup;
Packit Service d3d246
  gboolean quit;
Packit Service d3d246
};
Packit Service d3d246
Packit Service d3d246
#define NUM_THREADS     50
Packit Service d3d246
#define NUM_TOKENS       5
Packit Service d3d246
#define TOKEN_TTL   100000
Packit Service d3d246
Packit Service d3d246
static struct context contexts[NUM_THREADS];
Packit Service d3d246
static GThread *threads[NUM_THREADS];
Packit Service d3d246
static GWakeup *last_token_wakeup;
Packit Service d3d246
static volatile gint tokens_alive;
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
context_init (struct context *ctx)
Packit Service d3d246
{
Packit Service d3d246
  ctx->pending_tokens = NULL;
Packit Service d3d246
  g_mutex_init (&ctx->lock);
Packit Service d3d246
  ctx->wakeup = g_wakeup_new ();
Packit Service d3d246
  ctx->quit = FALSE;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
context_clear (struct context *ctx)
Packit Service d3d246
{
Packit Service d3d246
  g_assert (ctx->pending_tokens == NULL);
Packit Service d3d246
  g_assert (ctx->quit);
Packit Service d3d246
Packit Service d3d246
  g_mutex_clear (&ctx->lock);
Packit Service d3d246
  g_wakeup_free (ctx->wakeup);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
context_quit (struct context *ctx)
Packit Service d3d246
{
Packit Service d3d246
  ctx->quit = TRUE;
Packit Service d3d246
  g_wakeup_signal (ctx->wakeup);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static struct token *
Packit Service d3d246
context_pop_token (struct context *ctx)
Packit Service d3d246
{
Packit Service d3d246
  struct token *token;
Packit Service d3d246
Packit Service d3d246
  g_mutex_lock (&ctx->lock);
Packit Service d3d246
  token = ctx->pending_tokens->data;
Packit Service d3d246
  ctx->pending_tokens = g_slist_delete_link (ctx->pending_tokens,
Packit Service d3d246
                                             ctx->pending_tokens);
Packit Service d3d246
  g_mutex_unlock (&ctx->lock);
Packit Service d3d246
Packit Service d3d246
  return token;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
context_push_token (struct context *ctx,
Packit Service d3d246
                    struct token   *token)
Packit Service d3d246
{
Packit Service d3d246
  g_assert (token->owner == ctx);
Packit Service d3d246
Packit Service d3d246
  g_mutex_lock (&ctx->lock);
Packit Service d3d246
  ctx->pending_tokens = g_slist_prepend (ctx->pending_tokens, token);
Packit Service d3d246
  g_mutex_unlock (&ctx->lock);
Packit Service d3d246
Packit Service d3d246
  g_wakeup_signal (ctx->wakeup);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
dispatch_token (struct token *token)
Packit Service d3d246
{
Packit Service d3d246
  if (token->ttl > 0)
Packit Service d3d246
    {
Packit Service d3d246
      struct context *ctx;
Packit Service d3d246
      gint next_ctx;
Packit Service d3d246
Packit Service d3d246
      next_ctx = g_test_rand_int_range (0, NUM_THREADS);
Packit Service d3d246
      ctx = &contexts[next_ctx];
Packit Service d3d246
      token->owner = ctx;
Packit Service d3d246
      token->ttl--;
Packit Service d3d246
Packit Service d3d246
      context_push_token (ctx, token);
Packit Service d3d246
    }
Packit Service d3d246
  else
Packit Service d3d246
    {
Packit Service d3d246
      g_slice_free (struct token, token);
Packit Service d3d246
Packit Service d3d246
      if (g_atomic_int_dec_and_test (&tokens_alive))
Packit Service d3d246
        g_wakeup_signal (last_token_wakeup);
Packit Service d3d246
    }
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static struct token *
Packit Service d3d246
token_new (int ttl)
Packit Service d3d246
{
Packit Service d3d246
  struct token *token;
Packit Service d3d246
Packit Service d3d246
  token = g_slice_new (struct token);
Packit Service d3d246
  token->ttl = ttl;
Packit Service d3d246
Packit Service d3d246
  g_atomic_int_inc (&tokens_alive);
Packit Service d3d246
Packit Service d3d246
  return token;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static gpointer
Packit Service d3d246
thread_func (gpointer data)
Packit Service d3d246
{
Packit Service d3d246
  struct context *ctx = data;
Packit Service d3d246
Packit Service d3d246
  while (!ctx->quit)
Packit Service d3d246
    {
Packit Service d3d246
      wait_for_signaled (ctx->wakeup);
Packit Service d3d246
      g_wakeup_acknowledge (ctx->wakeup);
Packit Service d3d246
Packit Service d3d246
      while (ctx->pending_tokens)
Packit Service d3d246
        {
Packit Service d3d246
          struct token *token;
Packit Service d3d246
Packit Service d3d246
          token = context_pop_token (ctx);
Packit Service d3d246
          g_assert (token->owner == ctx);
Packit Service d3d246
          dispatch_token (token);
Packit Service d3d246
        }
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  return NULL;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
test_threaded (void)
Packit Service d3d246
{
Packit Service d3d246
  gint i;
Packit Service d3d246
Packit Service d3d246
  /* make sure we don't block forever */
Packit Service d3d246
  alarm (60);
Packit Service d3d246
Packit Service d3d246
  /* simple mainloop test based on GWakeup.
Packit Service d3d246
   *
Packit Service d3d246
   * create a bunch of contexts and a thread to 'run' each one.  create
Packit Service d3d246
   * some tokens and randomly pass them between the threads, until the
Packit Service d3d246
   * TTL on each token is zero.
Packit Service d3d246
   *
Packit Service d3d246
   * when no tokens are left, signal that we are done.  the mainthread
Packit Service d3d246
   * will then signal each worker thread to exit and join them to make
Packit Service d3d246
   * sure that works.
Packit Service d3d246
   */
Packit Service d3d246
Packit Service d3d246
  last_token_wakeup = g_wakeup_new ();
Packit Service d3d246
Packit Service d3d246
  /* create contexts, assign to threads */
Packit Service d3d246
  for (i = 0; i < NUM_THREADS; i++)
Packit Service d3d246
    {
Packit Service d3d246
      context_init (&contexts[i]);
Packit Service d3d246
      threads[i] = g_thread_new ("test", thread_func, &contexts[i]);
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  /* dispatch tokens */
Packit Service d3d246
  for (i = 0; i < NUM_TOKENS; i++)
Packit Service d3d246
    dispatch_token (token_new (TOKEN_TTL));
Packit Service d3d246
Packit Service d3d246
  /* wait until all tokens are gone */
Packit Service d3d246
  wait_for_signaled (last_token_wakeup);
Packit Service d3d246
Packit Service d3d246
  /* ask threads to quit, join them, cleanup */
Packit Service d3d246
  for (i = 0; i < NUM_THREADS; i++)
Packit Service d3d246
    {
Packit Service d3d246
      context_quit (&contexts[i]);
Packit Service d3d246
      g_thread_join (threads[i]);
Packit Service d3d246
      context_clear (&contexts[i]);
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  g_wakeup_free (last_token_wakeup);
Packit Service d3d246
Packit Service d3d246
  /* cancel alarm */
Packit Service d3d246
  alarm (0);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
int
Packit Service d3d246
main (int argc, char **argv)
Packit Service d3d246
{
Packit Service d3d246
  g_test_init (&argc, &argv, NULL);
Packit Service d3d246
Packit Service d3d246
#ifdef TEST_EVENTFD_FALLBACK
Packit Service d3d246
#define TESTNAME_SUFFIX "-fallback"
Packit Service d3d246
#else
Packit Service d3d246
#define TESTNAME_SUFFIX
Packit Service d3d246
#endif
Packit Service d3d246
Packit Service d3d246
Packit Service d3d246
  g_test_add_func ("/gwakeup/semantics" TESTNAME_SUFFIX, test_semantics);
Packit Service d3d246
  g_test_add_func ("/gwakeup/threaded" TESTNAME_SUFFIX, test_threaded);
Packit Service d3d246
Packit Service d3d246
  return g_test_run ();
Packit Service d3d246
}