• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* GLib testing framework examples and tests
2  * Copyright (C) 2008 Imendio AB
3  * Authors: Tim Janik
4  *
5  * This work is provided "as is"; redistribution and modification
6  * in whole or in part, in any medium, physical or electronic is
7  * permitted without restriction.
8  *
9  * This work is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12  *
13  * In no event shall the authors or contributors be liable for any
14  * direct, indirect, incidental, special, exemplary, or consequential
15  * damages (including, but not limited to, procurement of substitute
16  * goods or services; loss of use, data, or profits; or business
17  * interruption) however caused and on any theory of liability, whether
18  * in contract, strict liability, or tort (including negligence or
19  * otherwise) arising in any way out of the use of this software, even
20  * if advised of the possibility of such damage.
21  */
22 #include <glib.h>
23 #include <glib-object.h>
24 
25 /* This test tests the macros for defining dynamic types.
26  */
27 
28 static GMutex sync_mutex;
29 static gboolean loaded = FALSE;
30 
31 /* MODULE */
32 typedef struct _TestModule      TestModule;
33 typedef struct _TestModuleClass TestModuleClass;
34 
35 #define TEST_TYPE_MODULE              (test_module_get_type ())
36 #define TEST_MODULE(module)           (G_TYPE_CHECK_INSTANCE_CAST ((module), TEST_TYPE_MODULE, TestModule))
37 #define TEST_MODULE_CLASS(class)      (G_TYPE_CHECK_CLASS_CAST ((class), TEST_TYPE_MODULE, TestModuleClass))
38 #define TEST_IS_MODULE(module)        (G_TYPE_CHECK_INSTANCE_TYPE ((module), TEST_TYPE_MODULE))
39 #define TEST_IS_MODULE_CLASS(class)   (G_TYPE_CHECK_CLASS_TYPE ((class), TEST_TYPE_MODULE))
40 #define TEST_MODULE_GET_CLASS(module) (G_TYPE_INSTANCE_GET_CLASS ((module), TEST_TYPE_MODULE, TestModuleClass))
41 typedef void (*TestModuleRegisterFunc) (GTypeModule *module);
42 
43 struct _TestModule
44 {
45   GTypeModule parent_instance;
46 
47   TestModuleRegisterFunc register_func;
48 };
49 
50 struct _TestModuleClass
51 {
52   GTypeModuleClass parent_class;
53 };
54 
55 static GType test_module_get_type (void);
56 
57 static gboolean
test_module_load(GTypeModule * module)58 test_module_load (GTypeModule *module)
59 {
60   TestModule *test_module = TEST_MODULE (module);
61 
62   test_module->register_func (module);
63 
64   return TRUE;
65 }
66 
67 static void
test_module_unload(GTypeModule * module)68 test_module_unload (GTypeModule *module)
69 {
70 }
71 
72 static void
test_module_class_init(TestModuleClass * class)73 test_module_class_init (TestModuleClass *class)
74 {
75   GTypeModuleClass *module_class = G_TYPE_MODULE_CLASS (class);
76 
77   module_class->load = test_module_load;
78   module_class->unload = test_module_unload;
79 }
80 
test_module_get_type(void)81 static GType test_module_get_type (void)
82 {
83   static GType object_type = 0;
84 
85   if (!object_type) {
86     static const GTypeInfo object_info =
87       {
88 	sizeof (TestModuleClass),
89 	(GBaseInitFunc) NULL,
90 	(GBaseFinalizeFunc) NULL,
91 	(GClassInitFunc) test_module_class_init,
92 	(GClassFinalizeFunc) NULL,
93 	NULL,
94 	sizeof (TestModule),
95 	0,
96         (GInstanceInitFunc)NULL,
97         NULL,
98       };
99     object_type = g_type_register_static (G_TYPE_TYPE_MODULE, "TestModule", &object_info, 0);
100   }
101   return object_type;
102 }
103 
104 
105 static GTypeModule *
test_module_new(TestModuleRegisterFunc register_func)106 test_module_new (TestModuleRegisterFunc register_func)
107 {
108   TestModule *test_module = g_object_new (TEST_TYPE_MODULE, NULL);
109   GTypeModule *module = G_TYPE_MODULE (test_module);
110 
111   test_module->register_func = register_func;
112 
113   /* Register the types initially */
114   g_type_module_use (module);
115   g_type_module_unuse (module);
116 
117   return G_TYPE_MODULE (module);
118 }
119 
120 
121 
122 #define DYNAMIC_OBJECT_TYPE (dynamic_object_get_type ())
123 
124 typedef GObject DynamicObject;
125 typedef struct _DynamicObjectClass DynamicObjectClass;
126 
127 struct _DynamicObjectClass
128 {
129   GObjectClass parent_class;
130   guint val;
131 };
132 
133 static GType dynamic_object_get_type (void);
G_DEFINE_DYNAMIC_TYPE(DynamicObject,dynamic_object,G_TYPE_OBJECT)134 G_DEFINE_DYNAMIC_TYPE(DynamicObject, dynamic_object, G_TYPE_OBJECT)
135 
136 static void
137 dynamic_object_class_init (DynamicObjectClass *class)
138 {
139   class->val = 42;
140   g_assert (loaded == FALSE);
141   loaded = TRUE;
142 }
143 
144 static void
dynamic_object_class_finalize(DynamicObjectClass * class)145 dynamic_object_class_finalize (DynamicObjectClass *class)
146 {
147   g_assert (loaded == TRUE);
148   loaded = FALSE;
149 }
150 
151 static void
dynamic_object_init(DynamicObject * dynamic_object)152 dynamic_object_init (DynamicObject *dynamic_object)
153 {
154 }
155 
156 
157 static void
module_register(GTypeModule * module)158 module_register (GTypeModule *module)
159 {
160   dynamic_object_register_type (module);
161 }
162 
163 #define N_THREADS 100
164 #define N_REFS 10000
165 
166 static gpointer
ref_unref_thread(gpointer data)167 ref_unref_thread (gpointer data)
168 {
169   gint i;
170   /* first, synchronize with other threads,
171    */
172   if (g_test_verbose())
173     g_printerr ("WAITING!\n");
174   g_mutex_lock (&sync_mutex);
175   g_mutex_unlock (&sync_mutex);
176   if (g_test_verbose ())
177     g_printerr ("STARTING\n");
178 
179   /* ref/unref the klass 10000000 times */
180   for (i = N_REFS; i; i--) {
181     if (g_test_verbose ())
182       if (i % 10)
183 	g_printerr ("%d\n", i);
184     g_type_class_unref (g_type_class_ref ((GType) data));
185   }
186 
187   if (g_test_verbose())
188     g_printerr ("DONE !\n");
189 
190   return NULL;
191 }
192 
193 static void
test_multithreaded_dynamic_type_init(void)194 test_multithreaded_dynamic_type_init (void)
195 {
196   GTypeModule *module;
197   DynamicObjectClass *class;
198   /* Create N_THREADS threads that are going to just ref/unref a class */
199   GThread *threads[N_THREADS];
200   guint i;
201 
202   module = test_module_new (module_register);
203   g_assert (module != NULL);
204 
205   /* Not loaded until we call ref for the first time */
206   class = g_type_class_peek (DYNAMIC_OBJECT_TYPE);
207   g_assert (class == NULL);
208   g_assert (!loaded);
209 
210   /* pause newly created threads */
211   g_mutex_lock (&sync_mutex);
212 
213   /* create threads */
214   for (i = 0; i < N_THREADS; i++) {
215     threads[i] = g_thread_new ("test", ref_unref_thread, (gpointer) DYNAMIC_OBJECT_TYPE);
216   }
217 
218   /* execute threads */
219   g_mutex_unlock (&sync_mutex);
220 
221   for (i = 0; i < N_THREADS; i++) {
222     g_thread_join (threads[i]);
223   }
224 }
225 
226 enum
227 {
228   PROP_0,
229   PROP_FOO
230 };
231 
232 typedef struct _DynObj DynObj;
233 typedef struct _DynObjClass DynObjClass;
234 typedef struct _DynIfaceInterface DynIfaceInterface;
235 
236 struct _DynObj
237 {
238   GObject obj;
239 
240   gint foo;
241 };
242 
243 struct _DynObjClass
244 {
245   GObjectClass class;
246 };
247 
248 struct _DynIfaceInterface
249 {
250   GTypeInterface iface;
251 };
252 
253 static void dyn_obj_iface_init (DynIfaceInterface *iface);
254 
255 static GType dyn_iface_get_type (void);
256 G_DEFINE_INTERFACE (DynIface, dyn_iface, G_TYPE_OBJECT)
257 
258 static GType dyn_obj_get_type (void);
259 G_DEFINE_DYNAMIC_TYPE_EXTENDED(DynObj, dyn_obj, G_TYPE_OBJECT, 0,
260                       G_IMPLEMENT_INTERFACE_DYNAMIC(dyn_iface_get_type (), dyn_obj_iface_init))
261 
262 
263 static void
dyn_iface_default_init(DynIfaceInterface * iface)264 dyn_iface_default_init (DynIfaceInterface *iface)
265 {
266   g_object_interface_install_property (iface,
267     g_param_spec_int ("foo", NULL, NULL, 0, 100, 0, G_PARAM_READWRITE));
268 }
269 
270 static void
dyn_obj_iface_init(DynIfaceInterface * iface)271 dyn_obj_iface_init (DynIfaceInterface *iface)
272 {
273 }
274 
275 static void
dyn_obj_init(DynObj * obj)276 dyn_obj_init (DynObj *obj)
277 {
278   obj->foo = 0;
279 }
280 
281 static void
set_prop(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)282 set_prop (GObject      *object,
283           guint         prop_id,
284           const GValue *value,
285           GParamSpec   *pspec)
286 {
287   DynObj *obj = (DynObj *)object;
288 
289   switch (prop_id)
290     {
291     case PROP_FOO:
292       obj->foo = g_value_get_int (value);
293       break;
294     default:
295       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
296       break;
297     }
298 }
299 
300 static void
get_prop(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)301 get_prop (GObject    *object,
302           guint       prop_id,
303           GValue     *value,
304           GParamSpec *pspec)
305 {
306   DynObj *obj = (DynObj *)object;
307 
308   switch (prop_id)
309     {
310     case PROP_FOO:
311       g_value_set_int (value, obj->foo);
312       break;
313     default:
314       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
315       break;
316     }
317 }
318 
319 static void
dyn_obj_class_init(DynObjClass * class)320 dyn_obj_class_init (DynObjClass *class)
321 {
322   GObjectClass *object_class = G_OBJECT_CLASS (class);
323 
324   object_class->set_property = set_prop;
325   object_class->get_property = get_prop;
326 
327   g_object_class_override_property (object_class, PROP_FOO, "foo");
328 }
329 
330 static void
dyn_obj_class_finalize(DynObjClass * class)331 dyn_obj_class_finalize (DynObjClass *class)
332 {
333 }
334 
335 static void
mod_register(GTypeModule * module)336 mod_register (GTypeModule *module)
337 {
338   dyn_obj_register_type (module);
339 }
340 
341 static void
test_dynamic_interface_properties(void)342 test_dynamic_interface_properties (void)
343 {
344   GTypeModule *module;
345   DynObj *obj;
346   gint val;
347 
348   module = test_module_new (mod_register);
349   g_assert (module != NULL);
350 
351   obj = g_object_new (dyn_obj_get_type (), "foo", 1, NULL);
352   g_object_get (obj, "foo", &val, NULL);
353   g_assert_cmpint (val, ==, 1);
354 
355   g_object_unref (obj);
356 }
357 
358 int
main(int argc,char * argv[])359 main (int   argc,
360       char *argv[])
361 {
362   g_test_init (&argc, &argv, NULL);
363 
364   g_test_add_func ("/GObject/threaded-dynamic-ref-unref-init", test_multithreaded_dynamic_type_init);
365   g_test_add_func ("/GObject/dynamic-interface-properties", test_dynamic_interface_properties);
366 
367   return g_test_run();
368 }
369