Merge branch 'g-clear-pointer-no-side-effects' into 'master'
[glib.git] / glib / tests / utils.c
blob79bc08184997111712f269f0cedd89a6c6dc8190
1 /* Unit tests for utilities
2 * Copyright (C) 2010 Red Hat, Inc.
4 * This work is provided "as is"; redistribution and modification
5 * in whole or in part, in any medium, physical or electronic is
6 * permitted without restriction.
8 * This work is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12 * In no event shall the authors or contributors be liable for any
13 * direct, indirect, incidental, special, exemplary, or consequential
14 * damages (including, but not limited to, procurement of substitute
15 * goods or services; loss of use, data, or profits; or business
16 * interruption) however caused and on any theory of liability, whether
17 * in contract, strict liability, or tort (including negligence or
18 * otherwise) arising in any way out of the use of this software, even
19 * if advised of the possibility of such damage.
21 * Author: Matthias Clasen
24 #define GLIB_DISABLE_DEPRECATION_WARNINGS
26 #include "glib.h"
27 #include "glib-private.h"
29 #include <stdlib.h>
30 #include <string.h>
31 #include <stdarg.h>
33 static gboolean
34 strv_check (const gchar * const *strv, ...)
36 va_list args;
37 gchar *s;
38 gint i;
40 va_start (args, strv);
41 for (i = 0; strv[i]; i++)
43 s = va_arg (args, gchar*);
44 if (g_strcmp0 (strv[i], s) != 0)
46 va_end (args);
47 return FALSE;
51 va_end (args);
53 return TRUE;
56 static void
57 test_language_names (void)
59 const gchar * const *names;
61 g_setenv ("LANGUAGE", "de:en_US", TRUE);
62 names = g_get_language_names ();
63 g_assert (strv_check (names, "de", "en_US", "en", "C", NULL));
65 g_setenv ("LANGUAGE", "tt_RU.UTF-8@iqtelif", TRUE);
66 names = g_get_language_names ();
67 g_assert (strv_check (names,
68 "tt_RU.UTF-8@iqtelif",
69 "tt_RU@iqtelif",
70 "tt.UTF-8@iqtelif",
71 "tt@iqtelif",
72 "tt_RU.UTF-8",
73 "tt_RU",
74 "tt.UTF-8",
75 "tt",
76 "C",
77 NULL));
80 static void
81 test_locale_variants (void)
83 char **v;
85 v = g_get_locale_variants ("fr_BE");
86 g_assert (strv_check ((const gchar * const *) v, "fr_BE", "fr", NULL));
87 g_strfreev (v);
89 v = g_get_locale_variants ("sr_SR@latin");
90 g_assert (strv_check ((const gchar * const *) v, "sr_SR@latin", "sr@latin", "sr_SR", "sr", NULL));
91 g_strfreev (v);
94 static void
95 test_version (void)
97 if (g_test_verbose ())
98 g_printerr ("(header %d.%d.%d library %d.%d.%d) ",
99 GLIB_MAJOR_VERSION, GLIB_MINOR_VERSION, GLIB_MICRO_VERSION,
100 glib_major_version, glib_minor_version, glib_micro_version);
102 g_assert (glib_check_version (GLIB_MAJOR_VERSION,
103 GLIB_MINOR_VERSION,
104 GLIB_MICRO_VERSION) == NULL);
105 g_assert (glib_check_version (GLIB_MAJOR_VERSION,
106 GLIB_MINOR_VERSION,
107 0) == NULL);
108 g_assert (glib_check_version (GLIB_MAJOR_VERSION - 1,
110 0) != NULL);
111 g_assert (glib_check_version (GLIB_MAJOR_VERSION + 1,
113 0) != NULL);
114 g_assert (glib_check_version (GLIB_MAJOR_VERSION,
115 GLIB_MINOR_VERSION + 1,
116 0) != NULL);
117 /* don't use + 1 here, since a +/-1 difference can
118 * happen due to post-release version bumps in git
120 g_assert (glib_check_version (GLIB_MAJOR_VERSION,
121 GLIB_MINOR_VERSION,
122 GLIB_MICRO_VERSION + 3) != NULL);
125 static const gchar *argv0;
127 static void
128 test_appname (void)
130 const gchar *prgname;
131 const gchar *appname;
133 prgname = g_get_prgname ();
134 appname = g_get_application_name ();
135 g_assert_cmpstr (prgname, ==, argv0);
136 g_assert_cmpstr (appname, ==, prgname);
138 g_set_prgname ("prgname");
140 prgname = g_get_prgname ();
141 appname = g_get_application_name ();
142 g_assert_cmpstr (prgname, ==, "prgname");
143 g_assert_cmpstr (appname, ==, "prgname");
145 g_set_application_name ("appname");
147 prgname = g_get_prgname ();
148 appname = g_get_application_name ();
149 g_assert_cmpstr (prgname, ==, "prgname");
150 g_assert_cmpstr (appname, ==, "appname");
153 static void
154 test_tmpdir (void)
156 g_test_bug ("627969");
157 g_assert_cmpstr (g_get_tmp_dir (), !=, "");
160 static void
161 test_bits (void)
163 gulong mask;
164 gint max_bit;
165 gint i, pos;
167 pos = g_bit_nth_lsf (0, -1);
168 g_assert_cmpint (pos, ==, -1);
170 max_bit = sizeof (gulong) * 8;
171 for (i = 0; i < max_bit; i++)
173 mask = 1UL << i;
175 pos = g_bit_nth_lsf (mask, -1);
176 g_assert_cmpint (pos, ==, i);
178 pos = g_bit_nth_lsf (mask, i - 3);
179 g_assert_cmpint (pos , ==, i);
181 pos = g_bit_nth_lsf (mask, i);
182 g_assert_cmpint (pos , ==, -1);
184 pos = g_bit_nth_lsf (mask, i + 1);
185 g_assert_cmpint (pos , ==, -1);
188 pos = g_bit_nth_msf (0, -1);
189 g_assert_cmpint (pos, ==, -1);
191 for (i = 0; i < max_bit; i++)
193 mask = 1UL << i;
195 pos = g_bit_nth_msf (mask, -1);
196 g_assert_cmpint (pos, ==, i);
198 pos = g_bit_nth_msf (mask, i + 3);
199 g_assert_cmpint (pos , ==, i);
201 pos = g_bit_nth_msf (mask, i);
202 g_assert_cmpint (pos , ==, -1);
204 if (i > 0)
206 pos = g_bit_nth_msf (mask, i - 1);
207 g_assert_cmpint (pos , ==, -1);
212 static void
213 test_swap (void)
215 guint16 a16, b16;
216 guint32 a32, b32;
217 guint64 a64, b64;
219 a16 = 0xaabb;
220 b16 = 0xbbaa;
222 g_assert_cmpint (GUINT16_SWAP_LE_BE (a16), ==, b16);
224 a32 = 0xaaaabbbb;
225 b32 = 0xbbbbaaaa;
227 g_assert_cmpint (GUINT32_SWAP_LE_BE (a32), ==, b32);
229 a64 = G_GUINT64_CONSTANT(0xaaaaaaaabbbbbbbb);
230 b64 = G_GUINT64_CONSTANT(0xbbbbbbbbaaaaaaaa);
232 g_assert_cmpint (GUINT64_SWAP_LE_BE (a64), ==, b64);
235 static void
236 test_find_program (void)
238 gchar *res;
240 #ifdef G_OS_UNIX
241 res = g_find_program_in_path ("sh");
242 g_assert (res != NULL);
243 g_free (res);
245 res = g_find_program_in_path ("/bin/sh");
246 g_assert (res != NULL);
247 g_free (res);
248 #else
249 /* There's not a lot we can search for that would reliably work both
250 * on real Windows and mingw.
252 #endif
254 res = g_find_program_in_path ("this_program_does_not_exit");
255 g_assert (res == NULL);
257 res = g_find_program_in_path ("/bin");
258 g_assert (res == NULL);
260 res = g_find_program_in_path ("/etc/passwd");
261 g_assert (res == NULL);
264 static void
265 test_debug (void)
267 GDebugKey keys[] = {
268 { "key1", 1 },
269 { "key2", 2 },
270 { "key3", 4 },
272 guint res;
274 res = g_parse_debug_string (NULL, keys, G_N_ELEMENTS (keys));
275 g_assert_cmpint (res, ==, 0);
277 res = g_parse_debug_string ("foobabla;#!%!$%112 223", keys, G_N_ELEMENTS (keys));
278 g_assert_cmpint (res, ==, 0);
280 res = g_parse_debug_string ("key1:key2", keys, G_N_ELEMENTS (keys));
281 g_assert_cmpint (res, ==, 3);
283 res = g_parse_debug_string ("key1;key2", keys, G_N_ELEMENTS (keys));
284 g_assert_cmpint (res, ==, 3);
286 res = g_parse_debug_string ("key1,key2", keys, G_N_ELEMENTS (keys));
287 g_assert_cmpint (res, ==, 3);
289 res = g_parse_debug_string ("key1 key2", keys, G_N_ELEMENTS (keys));
290 g_assert_cmpint (res, ==, 3);
292 res = g_parse_debug_string ("key1\tkey2", keys, G_N_ELEMENTS (keys));
293 g_assert_cmpint (res, ==, 3);
295 res = g_parse_debug_string ("all", keys, G_N_ELEMENTS (keys));
296 g_assert_cmpint (res, ==, 7);
298 if (g_test_subprocess ())
300 res = g_parse_debug_string ("help", keys, G_N_ELEMENTS (keys));
301 g_assert_cmpint (res, ==, 0);
302 return;
304 g_test_trap_subprocess (NULL, 0, 0);
305 g_test_trap_assert_passed ();
306 g_test_trap_assert_stderr ("*Supported debug values: key1 key2 key3 all help*");
309 static void
310 test_codeset (void)
312 gchar *c;
313 const gchar *c2;
315 c = g_get_codeset ();
316 g_get_charset (&c2);
318 g_assert_cmpstr (c, ==, c2);
320 g_free (c);
323 static void
324 test_codeset2 (void)
326 if (g_test_subprocess ())
328 const gchar *c;
329 g_setenv ("CHARSET", "UTF-8", TRUE);
330 g_get_charset (&c);
331 g_assert_cmpstr (c, ==, "UTF-8");
332 return;
334 g_test_trap_subprocess (NULL, 0, 0);
335 g_test_trap_assert_passed ();
338 static void
339 test_basename (void)
341 const gchar *path = "/path/to/a/file/deep/down.sh";
342 const gchar *b;
344 b = g_basename (path);
346 g_assert_cmpstr (b, ==, "down.sh");
349 extern const gchar *glib_pgettext (const gchar *msgidctxt, gsize msgidoffset);
351 static void
352 test_gettext (void)
354 const gchar *am0, *am1, *am2, *am3;
356 am0 = glib_pgettext ("GDateTime\004AM", strlen ("GDateTime") + 1);
357 am1 = g_dpgettext ("glib20", "GDateTime\004AM", strlen ("GDateTime") + 1);
358 am2 = g_dpgettext ("glib20", "GDateTime|AM", 0);
359 am3 = g_dpgettext2 ("glib20", "GDateTime", "AM");
361 g_assert_cmpstr (am0, ==, am1);
362 g_assert_cmpstr (am1, ==, am2);
363 g_assert_cmpstr (am2, ==, am3);
366 static void
367 test_username (void)
369 const gchar *name;
371 name = g_get_user_name ();
373 g_assert (name != NULL);
376 static void
377 test_realname (void)
379 const gchar *name;
381 name = g_get_real_name ();
383 g_assert (name != NULL);
386 static void
387 test_hostname (void)
389 const gchar *name;
391 name = g_get_host_name ();
393 g_assert (name != NULL);
394 g_assert_true (g_utf8_validate (name, -1, NULL));
397 #ifdef G_OS_UNIX
398 static void
399 test_xdg_dirs (void)
401 gchar *xdg;
402 const gchar *dir;
403 const gchar * const *dirs;
404 gchar *s;
406 xdg = g_strdup (g_getenv ("XDG_CONFIG_HOME"));
407 if (!xdg)
408 xdg = g_build_filename (g_get_home_dir (), ".config", NULL);
410 dir = g_get_user_config_dir ();
412 g_assert_cmpstr (dir, ==, xdg);
413 g_free (xdg);
415 xdg = g_strdup (g_getenv ("XDG_DATA_HOME"));
416 if (!xdg)
417 xdg = g_build_filename (g_get_home_dir (), ".local", "share", NULL);
419 dir = g_get_user_data_dir ();
421 g_assert_cmpstr (dir, ==, xdg);
422 g_free (xdg);
424 xdg = g_strdup (g_getenv ("XDG_CACHE_HOME"));
425 if (!xdg)
426 xdg = g_build_filename (g_get_home_dir (), ".cache", NULL);
428 dir = g_get_user_cache_dir ();
430 g_assert_cmpstr (dir, ==, xdg);
431 g_free (xdg);
433 xdg = g_strdup (g_getenv ("XDG_RUNTIME_DIR"));
434 if (!xdg)
435 xdg = g_strdup (g_get_user_cache_dir ());
437 dir = g_get_user_runtime_dir ();
439 g_assert_cmpstr (dir, ==, xdg);
440 g_free (xdg);
442 xdg = (gchar *)g_getenv ("XDG_CONFIG_DIRS");
443 if (!xdg)
444 xdg = "/etc/xdg";
446 dirs = g_get_system_config_dirs ();
448 s = g_strjoinv (":", (gchar **)dirs);
450 g_assert_cmpstr (s, ==, xdg);
452 g_free (s);
454 #endif
456 static void
457 test_special_dir (void)
459 const gchar *dir, *dir2;
461 dir = g_get_user_special_dir (G_USER_DIRECTORY_DESKTOP);
462 g_reload_user_special_dirs_cache ();
463 dir2 = g_get_user_special_dir (G_USER_DIRECTORY_DESKTOP);
465 g_assert_cmpstr (dir, ==, dir2);
468 static void
469 test_desktop_special_dir (void)
471 const gchar *dir, *dir2;
473 dir = g_get_user_special_dir (G_USER_DIRECTORY_DESKTOP);
474 g_assert (dir != NULL);
476 g_reload_user_special_dirs_cache ();
477 dir2 = g_get_user_special_dir (G_USER_DIRECTORY_DESKTOP);
478 g_assert (dir2 != NULL);
481 static gboolean
482 source_test (gpointer data)
484 g_assert_not_reached ();
485 return G_SOURCE_REMOVE;
488 static void
489 test_clear_source (void)
491 guint id;
493 id = g_idle_add (source_test, NULL);
494 g_assert_cmpuint (id, >, 0);
496 g_clear_handle_id (&id, g_source_remove);
497 g_assert_cmpuint (id, ==, 0);
499 id = g_timeout_add (100, source_test, NULL);
500 g_assert_cmpuint (id, >, 0);
502 g_clear_handle_id (&id, g_source_remove);
503 g_assert_cmpuint (id, ==, 0);
506 static void
507 test_clear_pointer (void)
509 gpointer a;
511 a = g_malloc (5);
512 g_clear_pointer (&a, g_free);
513 g_assert (a == NULL);
515 a = g_malloc (5);
516 (g_clear_pointer) (&a, g_free);
517 g_assert (a == NULL);
520 /* Test that g_clear_pointer() works with a GDestroyNotify which contains a cast.
521 * See https://gitlab.gnome.org/GNOME/glib/issues/1425 */
522 static void
523 test_clear_pointer_cast (void)
525 GHashTable *hash_table = NULL;
527 hash_table = g_hash_table_new (g_str_hash, g_str_equal);
529 g_assert_nonnull (hash_table);
531 g_clear_pointer (&hash_table, (void (*) (GHashTable *)) g_hash_table_destroy);
533 g_assert_null (hash_table);
536 /* Test that the macro version of g_clear_pointer() only evaluates its argument
537 * once, just like the function version would. */
538 static void
539 test_clear_pointer_side_effects (void)
541 gchar **my_string_array, **i;
543 my_string_array = g_new0 (gchar*, 3);
544 my_string_array[0] = g_strdup ("hello");
545 my_string_array[1] = g_strdup ("there");
546 my_string_array[2] = NULL;
548 i = my_string_array;
550 g_clear_pointer (i++, g_free);
552 g_assert_true (i == &my_string_array[1]);
553 g_assert_null (my_string_array[0]);
554 g_assert_nonnull (my_string_array[1]);
555 g_assert_null (my_string_array[2]);
557 g_free (my_string_array[1]);
558 g_free (my_string_array[2]);
559 g_free (my_string_array);
562 static int obj_count;
564 static void
565 get_obj (gpointer *obj_out)
567 gpointer obj = g_malloc (5);
568 obj_count++;
570 if (obj_out)
571 *obj_out = g_steal_pointer (&obj);
573 if (obj)
575 g_free (obj);
576 obj_count--;
580 static void
581 test_take_pointer (void)
583 gpointer a;
584 gpointer b;
586 get_obj (NULL);
588 get_obj (&a);
589 g_assert (a);
591 /* ensure that it works to skip the macro */
592 b = (g_steal_pointer) (&a);
593 g_assert (!a);
594 obj_count--;
595 g_free (b);
597 g_assert (!obj_count);
600 static void
601 test_misc_mem (void)
603 gpointer a;
605 a = g_try_malloc (0);
606 g_assert (a == NULL);
608 a = g_try_malloc0 (0);
609 g_assert (a == NULL);
611 a = g_malloc (16);
612 a = g_try_realloc (a, 20);
613 a = g_try_realloc (a, 0);
615 g_assert (a == NULL);
618 static void
619 test_nullify (void)
621 gpointer p = &test_nullify;
623 g_assert (p != NULL);
625 g_nullify_pointer (&p);
627 g_assert (p == NULL);
630 static void
631 atexit_func (void)
633 g_print ("atexit called");
636 static void
637 test_atexit (void)
639 if (g_test_subprocess ())
641 g_atexit (atexit_func);
642 return;
644 g_test_trap_subprocess (NULL, 0, 0);
645 g_test_trap_assert_passed ();
646 g_test_trap_assert_stdout ("*atexit called*");
649 static void
650 test_check_setuid (void)
652 gboolean res;
654 res = GLIB_PRIVATE_CALL(g_check_setuid) ();
655 g_assert (!res);
659 main (int argc,
660 char *argv[])
662 argv0 = argv[0];
664 /* for tmpdir test, need to do this early before g_get_any_init */
665 g_setenv ("TMPDIR", "", TRUE);
666 g_unsetenv ("TMP");
667 g_unsetenv ("TEMP");
669 /* g_test_init() only calls g_set_prgname() if g_get_prgname()
670 * returns %NULL, but g_get_prgname() on Windows never returns NULL.
671 * So we need to do this by hand to make test_appname() work on
672 * Windows.
674 g_set_prgname (argv[0]);
676 g_test_init (&argc, &argv, NULL);
677 g_test_bug_base ("http://bugzilla.gnome.org/");
679 g_test_add_func ("/utils/language-names", test_language_names);
680 g_test_add_func ("/utils/locale-variants", test_locale_variants);
681 g_test_add_func ("/utils/version", test_version);
682 g_test_add_func ("/utils/appname", test_appname);
683 g_test_add_func ("/utils/tmpdir", test_tmpdir);
684 g_test_add_func ("/utils/bits", test_bits);
685 g_test_add_func ("/utils/swap", test_swap);
686 g_test_add_func ("/utils/find-program", test_find_program);
687 g_test_add_func ("/utils/debug", test_debug);
688 g_test_add_func ("/utils/codeset", test_codeset);
689 g_test_add_func ("/utils/codeset2", test_codeset2);
690 g_test_add_func ("/utils/basename", test_basename);
691 g_test_add_func ("/utils/gettext", test_gettext);
692 g_test_add_func ("/utils/username", test_username);
693 g_test_add_func ("/utils/realname", test_realname);
694 g_test_add_func ("/utils/hostname", test_hostname);
695 #ifdef G_OS_UNIX
696 g_test_add_func ("/utils/xdgdirs", test_xdg_dirs);
697 #endif
698 g_test_add_func ("/utils/specialdir", test_special_dir);
699 g_test_add_func ("/utils/specialdir/desktop", test_desktop_special_dir);
700 g_test_add_func ("/utils/clear-pointer", test_clear_pointer);
701 g_test_add_func ("/utils/clear-pointer-cast", test_clear_pointer_cast);
702 g_test_add_func ("/utils/clear-pointer/side-effects", test_clear_pointer_side_effects);
703 g_test_add_func ("/utils/take-pointer", test_take_pointer);
704 g_test_add_func ("/utils/clear-source", test_clear_source);
705 g_test_add_func ("/utils/misc-mem", test_misc_mem);
706 g_test_add_func ("/utils/nullify", test_nullify);
707 g_test_add_func ("/utils/atexit", test_atexit);
708 g_test_add_func ("/utils/check-setuid", test_check_setuid);
710 return g_test_run ();