1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
|
/* Unit tests for GOnce and friends
* Copyright (C) 2011 Red Hat, Inc
* Author: Matthias Clasen
*
* This work is provided "as is"; redistribution and modification
* in whole or in part, in any medium, physical or electronic is
* permitted without restriction.
*
* This work 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.
*
* In no event shall the authors or contributors be liable for any
* direct, indirect, incidental, special, exemplary, or consequential
* damages (including, but not limited to, procurement of substitute
* goods or services; loss of use, data, or profits; or business
* interruption) however caused and on any theory of liability, whether
* in contract, strict liability, or tort (including negligence or
* otherwise) arising in any way out of the use of this software, even
* if advised of the possibility of such damage.
*/
#include <glib.h>
static gpointer
do_once (gpointer data)
{
static gint i = 0;
i++;
return GINT_TO_POINTER (i);
}
static void
test_once1 (void)
{
GOnce once = G_ONCE_INIT;
gpointer res;
g_assert (once.status == G_ONCE_STATUS_NOTCALLED);
res = g_once (&once, do_once, NULL);
g_assert_cmpint (GPOINTER_TO_INT (res), ==, 1);
g_assert (once.status == G_ONCE_STATUS_READY);
res = g_once (&once, do_once, NULL);
g_assert_cmpint (GPOINTER_TO_INT (res), ==, 1);
}
static void
test_once2 (void)
{
static gsize init = 0;
if (g_once_init_enter (&init))
{
g_assert (TRUE);
g_once_init_leave (&init, 1);
}
g_assert_cmpint (init, ==, 1);
if (g_once_init_enter (&init))
{
g_assert_not_reached ();
g_once_init_leave (&init, 2);
}
g_assert_cmpint (init, ==, 1);
}
#define THREADS 100
static gint64 shared;
static void
init_shared (void)
{
static gsize init = 0;
if (g_once_init_enter (&init))
{
shared += 42;
g_once_init_leave (&init, 1);
}
}
static gpointer
thread_func (gpointer data)
{
init_shared ();
return NULL;
}
static void
test_once3 (void)
{
gint i;
GThread *threads[THREADS];
shared = 0;
for (i = 0; i < THREADS; i++)
threads[i] = g_thread_new ("once3", thread_func, NULL);
for (i = 0; i < THREADS; i++)
g_thread_join (threads[i]);
g_assert_cmpint (shared, ==, 42);
}
static void
test_once4 (void)
{
static const gchar *val;
if (g_once_init_enter (&val))
g_once_init_leave (&val, "foo");
g_assert_cmpstr (val, ==, "foo");
}
int
main (int argc, char *argv[])
{
g_test_init (&argc, &argv, NULL);
g_test_add_func ("/thread/once1", test_once1);
g_test_add_func ("/thread/once2", test_once2);
g_test_add_func ("/thread/once3", test_once3);
g_test_add_func ("/thread/once4", test_once4);
return g_test_run ();
}
|