Merge branch 'atomic-older-cplusplus' into 'main'

gatomic: fix the atomic compare_and_exchange macros on older C++ standard versions

See merge request GNOME/glib!2864
This commit is contained in:
Philip Withnall 2022-10-12 09:56:19 +00:00
commit e90733a457
8 changed files with 110 additions and 36 deletions

View File

@ -27,6 +27,8 @@ if cc.has_header('pty.h')
endif endif
endif endif
test_cpp_args = test_c_args
if host_machine.system() == 'windows' if host_machine.system() == 'windows'
common_gio_tests_deps += [iphlpapi_dep, winsock2, cc.find_library ('secur32')] common_gio_tests_deps += [iphlpapi_dep, winsock2, cc.find_library ('secur32')]
endif endif
@ -130,8 +132,19 @@ if have_cxx
gio_tests += { gio_tests += {
'cxx' : { 'cxx' : {
'source' : ['cxx.cpp'], 'source' : ['cxx.cpp'],
'suite': ['C++'],
}, },
} }
foreach std, arg: cxx_standards
gio_tests += {
'cxx-@0@'.format(std) : {
'source' : ['cxx.cpp'],
'suite' : ['cpp'],
'cpp_args' : [arg],
},
}
endforeach
endif endif
test_extra_programs = { test_extra_programs = {
@ -882,6 +895,7 @@ foreach test_name, extra_args : gio_tests
exe = executable(test_name, [source, extra_sources], exe = executable(test_name, [source, extra_sources],
c_args : test_c_args + extra_args.get('c_args', []), c_args : test_c_args + extra_args.get('c_args', []),
cpp_args : test_cpp_args + extra_args.get('cpp_args', []),
dependencies : common_gio_tests_deps + extra_args.get('dependencies', []), dependencies : common_gio_tests_deps + extra_args.get('dependencies', []),
install_rpath : extra_args.get('install_rpath', ''), install_rpath : extra_args.get('install_rpath', ''),
install_dir: installed_tests_execdir, install_dir: installed_tests_execdir,

View File

@ -170,7 +170,7 @@ G_END_DECLS
(void) (0 ? *(atomic) ^ *(atomic) : 1); \ (void) (0 ? *(atomic) ^ *(atomic) : 1); \
__atomic_fetch_sub ((atomic), 1, __ATOMIC_SEQ_CST) == 1; \ __atomic_fetch_sub ((atomic), 1, __ATOMIC_SEQ_CST) == 1; \
})) }))
#if defined(glib_typeof) && defined(__cplusplus) && __cplusplus >= 201103L #if defined(glib_typeof) && defined(__cplusplus)
/* See comments below about equivalent g_atomic_pointer_compare_and_exchange() /* See comments below about equivalent g_atomic_pointer_compare_and_exchange()
* shenanigans for type-safety when compiling in C++ mode. */ * shenanigans for type-safety when compiling in C++ mode. */
#define g_atomic_int_compare_and_exchange(atomic, oldval, newval) \ #define g_atomic_int_compare_and_exchange(atomic, oldval, newval) \
@ -180,7 +180,7 @@ G_END_DECLS
(void) (0 ? *(atomic) ^ (newval) ^ (oldval) : 1); \ (void) (0 ? *(atomic) ^ (newval) ^ (oldval) : 1); \
__atomic_compare_exchange_n ((atomic), &gaicae_oldval, (newval), FALSE, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST) ? TRUE : FALSE; \ __atomic_compare_exchange_n ((atomic), &gaicae_oldval, (newval), FALSE, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST) ? TRUE : FALSE; \
})) }))
#else /* if !(defined(glib_typeof) && defined(__cplusplus) && __cplusplus >= 201103L) */ #else /* if !(defined(glib_typeof) && defined(__cplusplus)) */
#define g_atomic_int_compare_and_exchange(atomic, oldval, newval) \ #define g_atomic_int_compare_and_exchange(atomic, oldval, newval) \
(G_GNUC_EXTENSION ({ \ (G_GNUC_EXTENSION ({ \
gint gaicae_oldval = (oldval); \ gint gaicae_oldval = (oldval); \
@ -230,7 +230,7 @@ G_END_DECLS
(guint) __atomic_fetch_xor ((atomic), (val), __ATOMIC_SEQ_CST); \ (guint) __atomic_fetch_xor ((atomic), (val), __ATOMIC_SEQ_CST); \
})) }))
#if defined(glib_typeof) && defined(__cplusplus) && __cplusplus >= 201103L #if defined(glib_typeof) && defined(__cplusplus)
/* This is typesafe because we check we can assign oldval to the type of /* This is typesafe because we check we can assign oldval to the type of
* (*atomic). Unfortunately it can only be done in C++ because gcc/clang warn * (*atomic). Unfortunately it can only be done in C++ because gcc/clang warn
* when atomic is volatile and not oldval, or when atomic is gsize* and oldval * when atomic is volatile and not oldval, or when atomic is gsize* and oldval
@ -247,7 +247,7 @@ G_END_DECLS
(void) (0 ? (gpointer) *(atomic) : NULL); \ (void) (0 ? (gpointer) *(atomic) : NULL); \
__atomic_compare_exchange_n ((atomic), &gapcae_oldval, (newval), FALSE, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST) ? TRUE : FALSE; \ __atomic_compare_exchange_n ((atomic), &gapcae_oldval, (newval), FALSE, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST) ? TRUE : FALSE; \
})) }))
#else /* if !(defined(glib_typeof) && defined(__cplusplus) && __cplusplus >= 201103L) */ #else /* if !(defined(glib_typeof) && defined(__cplusplus) */
#define g_atomic_pointer_compare_and_exchange(atomic, oldval, newval) \ #define g_atomic_pointer_compare_and_exchange(atomic, oldval, newval) \
(G_GNUC_EXTENSION ({ \ (G_GNUC_EXTENSION ({ \
G_STATIC_ASSERT (sizeof (oldval) == sizeof (gpointer)); \ G_STATIC_ASSERT (sizeof (oldval) == sizeof (gpointer)); \

View File

@ -34,9 +34,12 @@
* This symbol is private. * This symbol is private.
*/ */
#undef glib_typeof #undef glib_typeof
#if !defined(__cplusplus) && (G_GNUC_CHECK_VERSION(4, 8) || defined(__clang__)) #if (!defined(__cplusplus) || (!defined (_MSVC_LANG) && __cplusplus < 201103L)) && \
(G_GNUC_CHECK_VERSION(4, 8) || defined(__clang__))
#define glib_typeof(t) __typeof__ (t) #define glib_typeof(t) __typeof__ (t)
#elif defined(__cplusplus) && __cplusplus >= 201103L && GLIB_VERSION_MIN_REQUIRED >= GLIB_VERSION_2_68 #elif defined(__cplusplus) && \
(__cplusplus >= 201103L || (defined (_MSVC_LANG) && _MSVC_LANG > 201103L)) && \
GLIB_VERSION_MIN_REQUIRED >= GLIB_VERSION_2_68
/* C++11 decltype() is close enough for our usage */ /* C++11 decltype() is close enough for our usage */
#include <type_traits> #include <type_traits>
#define glib_typeof(t) typename std::remove_reference<decltype (t)>::type #define glib_typeof(t) typename std::remove_reference<decltype (t)>::type

View File

@ -27,9 +27,6 @@ typedef struct
static void static void
test_typeof (void) test_typeof (void)
{ {
#if __cplusplus >= 201103L
// Test that with C++11 we don't get those kind of errors:
// error: invalid conversion from gpointer {aka void*} to MyObject* [-fpermissive]
MyObject *obj = g_rc_box_new0 (MyObject); MyObject *obj = g_rc_box_new0 (MyObject);
MyObject *obj2 = g_rc_box_acquire (obj); MyObject *obj2 = g_rc_box_acquire (obj);
g_assert_true (obj2 == obj); g_assert_true (obj2 == obj);
@ -37,12 +34,21 @@ test_typeof (void)
MyObject *obj3 = g_atomic_pointer_get (&obj2); MyObject *obj3 = g_atomic_pointer_get (&obj2);
g_assert_true (obj3 == obj); g_assert_true (obj3 == obj);
#if __cplusplus >= 201103L
MyObject *obj4 = nullptr; MyObject *obj4 = nullptr;
#else
MyObject *obj4 = NULL;
#endif
g_atomic_pointer_set (&obj4, obj3); g_atomic_pointer_set (&obj4, obj3);
g_assert_true (obj4 == obj); g_assert_true (obj4 == obj);
#if __cplusplus >= 201103L
MyObject *obj5 = nullptr; MyObject *obj5 = nullptr;
g_atomic_pointer_compare_and_exchange (&obj5, nullptr, obj4); g_atomic_pointer_compare_and_exchange (&obj5, nullptr, obj4);
#else
MyObject *obj5 = NULL;
g_atomic_pointer_compare_and_exchange (&obj5, NULL, obj4);
#endif
g_assert_true (obj5 == obj); g_assert_true (obj5 == obj);
MyObject *obj6 = g_steal_pointer (&obj5); MyObject *obj6 = g_steal_pointer (&obj5);
@ -50,15 +56,11 @@ test_typeof (void)
g_clear_pointer (&obj6, g_rc_box_release); g_clear_pointer (&obj6, g_rc_box_release);
g_rc_box_release (obj); g_rc_box_release (obj);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
static void static void
test_atomic_pointer_compare_and_exchange (void) test_atomic_pointer_compare_and_exchange (void)
{ {
#if __cplusplus >= 201103L
const gchar *str1 = "str1"; const gchar *str1 = "str1";
const gchar *str2 = "str2"; const gchar *str2 = "str2";
const gchar *atomic_string = str1; const gchar *atomic_string = str1;
@ -68,15 +70,11 @@ test_atomic_pointer_compare_and_exchange (void)
g_assert_true (g_atomic_pointer_compare_and_exchange (&atomic_string, str1, str2)); g_assert_true (g_atomic_pointer_compare_and_exchange (&atomic_string, str1, str2));
g_assert_true (atomic_string == str2); g_assert_true (atomic_string == str2);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
static void static void
test_atomic_pointer_compare_and_exchange_full (void) test_atomic_pointer_compare_and_exchange_full (void)
{ {
#if __cplusplus >= 201103L
const gchar *str1 = "str1"; const gchar *str1 = "str1";
const gchar *str2 = "str2"; const gchar *str2 = "str2";
const gchar *atomic_string = str1; const gchar *atomic_string = str1;
@ -88,15 +86,11 @@ test_atomic_pointer_compare_and_exchange_full (void)
g_assert_true (g_atomic_pointer_compare_and_exchange_full (&atomic_string, str1, str2, &old)); g_assert_true (g_atomic_pointer_compare_and_exchange_full (&atomic_string, str1, str2, &old));
g_assert_true (atomic_string == str2); g_assert_true (atomic_string == str2);
g_assert_true (old == str1); g_assert_true (old == str1);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
static void static void
test_atomic_int_compare_and_exchange (void) test_atomic_int_compare_and_exchange (void)
{ {
#if __cplusplus >= 201103L
gint atomic_int = 5; gint atomic_int = 5;
g_test_message ("Test that g_atomic_int_compare_and_exchange() doesnt have " g_test_message ("Test that g_atomic_int_compare_and_exchange() doesnt have "
@ -104,15 +98,11 @@ test_atomic_int_compare_and_exchange (void)
g_assert_true (g_atomic_int_compare_and_exchange (&atomic_int, 5, 50)); g_assert_true (g_atomic_int_compare_and_exchange (&atomic_int, 5, 50));
g_assert_cmpint (atomic_int, ==, 50); g_assert_cmpint (atomic_int, ==, 50);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
static void static void
test_atomic_int_compare_and_exchange_full (void) test_atomic_int_compare_and_exchange_full (void)
{ {
#if __cplusplus >= 201103L
gint atomic_int = 5; gint atomic_int = 5;
gint old_value; gint old_value;
@ -122,15 +112,11 @@ test_atomic_int_compare_and_exchange_full (void)
g_assert_true (g_atomic_int_compare_and_exchange_full (&atomic_int, 5, 50, &old_value)); g_assert_true (g_atomic_int_compare_and_exchange_full (&atomic_int, 5, 50, &old_value));
g_assert_cmpint (atomic_int, ==, 50); g_assert_cmpint (atomic_int, ==, 50);
g_assert_cmpint (old_value, ==, 5); g_assert_cmpint (old_value, ==, 5);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
static void static void
test_atomic_pointer_exchange (void) test_atomic_pointer_exchange (void)
{ {
#if __cplusplus >= 201103L
const gchar *str1 = "str1"; const gchar *str1 = "str1";
const gchar *str2 = "str2"; const gchar *str2 = "str2";
const gchar *atomic_string = str1; const gchar *atomic_string = str1;
@ -140,24 +126,17 @@ test_atomic_pointer_exchange (void)
g_assert_true (g_atomic_pointer_exchange (&atomic_string, str2) == str1); g_assert_true (g_atomic_pointer_exchange (&atomic_string, str2) == str1);
g_assert_true (atomic_string == str2); g_assert_true (atomic_string == str2);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
static void static void
test_atomic_int_exchange (void) test_atomic_int_exchange (void)
{ {
#if __cplusplus >= 201103L
gint atomic_int = 5; gint atomic_int = 5;
g_test_message ("Test that g_atomic_int_compare_and_exchange() doesnt have " g_test_message ("Test that g_atomic_int_compare_and_exchange() doesnt have "
"any compiler warnings in C++ mode"); "any compiler warnings in C++ mode");
g_assert_cmpint (g_atomic_int_exchange (&atomic_int, 50), ==, 5); g_assert_cmpint (g_atomic_int_exchange (&atomic_int, 50), ==, 5);
#else
g_test_skip ("This test requires a C++11 compiler");
#endif
} }
G_NO_INLINE G_NO_INLINE
@ -184,6 +163,35 @@ test_inline_no_inline_macros (void)
g_assert_true (do_inline_this ()); g_assert_true (do_inline_this ());
} }
static void
clear_boolean_ptr (gboolean *val)
{
*val = TRUE;
}
static void
test_clear_pointer (void)
{
gboolean value = FALSE;
gboolean *ptr = &value;
g_assert_true (ptr == &value);
g_assert_false (value);
g_clear_pointer (&ptr, clear_boolean_ptr);
g_assert_null (ptr);
g_assert_true (value);
}
static void
test_steal_pointer (void)
{
gpointer ptr = &ptr;
g_assert_true (ptr == &ptr);
g_assert_true (g_steal_pointer (&ptr) == &ptr);
g_assert_null (ptr);
}
int int
main (int argc, char *argv[]) main (int argc, char *argv[])
{ {
@ -201,6 +209,8 @@ main (int argc, char *argv[])
g_test_add_func ("/C++/atomic-pointer-exchange", test_atomic_pointer_exchange); g_test_add_func ("/C++/atomic-pointer-exchange", test_atomic_pointer_exchange);
g_test_add_func ("/C++/atomic-int-exchange", test_atomic_int_exchange); g_test_add_func ("/C++/atomic-int-exchange", test_atomic_int_exchange);
g_test_add_func ("/C++/inlined-not-inlined-functions", test_inline_no_inline_macros); g_test_add_func ("/C++/inlined-not-inlined-functions", test_inline_no_inline_macros);
g_test_add_func ("/C++/clear-pointer", test_clear_pointer);
g_test_add_func ("/C++/steal-pointer", test_steal_pointer);
return g_test_run (); return g_test_run ();
} }

View File

@ -159,8 +159,19 @@ if have_cxx
glib_tests += { glib_tests += {
'cxx' : { 'cxx' : {
'source' : ['cxx.cpp'], 'source' : ['cxx.cpp'],
'suite': ['cpp'],
} }
} }
foreach std, arg: cxx_standards
glib_tests += {
'cxx-@0@'.format(std) : {
'source' : ['cxx.cpp'],
'suite' : ['cpp'],
'cpp_args' : [arg],
},
}
endforeach
endif endif
if cc.get_id() != 'msvc' if cc.get_id() != 'msvc'
@ -255,6 +266,7 @@ test_env.set('MALLOC_CHECK_', '2')
test_deps = [libm, thread_dep, libglib_dep] test_deps = [libm, thread_dep, libglib_dep]
test_cargs = ['-DG_LOG_DOMAIN="GLib"', '-UG_DISABLE_ASSERT'] test_cargs = ['-DG_LOG_DOMAIN="GLib"', '-UG_DISABLE_ASSERT']
test_cpp_args = test_cargs
foreach test_name, extra_args : glib_tests foreach test_name, extra_args : glib_tests
source = extra_args.get('source', test_name + '.c') source = extra_args.get('source', test_name + '.c')
@ -276,6 +288,7 @@ foreach test_name, extra_args : glib_tests
exe = executable(test_name, source, exe = executable(test_name, source,
c_args : test_cargs + extra_args.get('c_args', []), c_args : test_cargs + extra_args.get('c_args', []),
cpp_args : test_cpp_args + extra_args.get('cpp_args', []),
link_args : extra_args.get('link_args', []), link_args : extra_args.get('link_args', []),
dependencies : test_deps + extra_args.get('dependencies', []), dependencies : test_deps + extra_args.get('dependencies', []),
install_dir: installed_tests_execdir, install_dir: installed_tests_execdir,

View File

@ -15,8 +15,19 @@ if have_cxx
gmodule_tests += { gmodule_tests += {
'cxx' : { 'cxx' : {
'source' : ['cxx.cpp'], 'source' : ['cxx.cpp'],
'suite' : ['cpp'],
} }
} }
foreach std, arg: cxx_standards
gmodule_tests += {
'cxx-@0@'.format(std) : {
'source' : ['cxx.cpp'],
'suite' : ['cpp'],
'cpp_args' : [arg],
},
}
endforeach
endif endif
module_suffix = [] module_suffix = []
@ -51,6 +62,7 @@ test_env.set('MALLOC_CHECK_', '2')
test_deps = [libm, thread_dep, libglib_dep, libgmodule_dep] test_deps = [libm, thread_dep, libglib_dep, libgmodule_dep]
test_cargs = ['-DG_LOG_DOMAIN="GModule"', '-UG_DISABLE_ASSERT'] test_cargs = ['-DG_LOG_DOMAIN="GModule"', '-UG_DISABLE_ASSERT']
test_cpp_args = test_cargs
foreach test_name, extra_args : gmodule_tests foreach test_name, extra_args : gmodule_tests
source = extra_args.get('source', test_name + '.c') source = extra_args.get('source', test_name + '.c')
@ -72,6 +84,7 @@ foreach test_name, extra_args : gmodule_tests
exe = executable(test_name, source, exe = executable(test_name, source,
c_args : test_cargs + extra_args.get('c_args', []), c_args : test_cargs + extra_args.get('c_args', []),
cpp_args : test_cpp_args + extra_args.get('cpp_args', []),
link_args : extra_args.get('link_args', []), link_args : extra_args.get('link_args', []),
dependencies : test_deps + extra_args.get('dependencies', []), dependencies : test_deps + extra_args.get('dependencies', []),
export_dynamic : extra_args.get('export_dynamic', false), export_dynamic : extra_args.get('export_dynamic', false),

View File

@ -102,8 +102,19 @@ if have_cxx
gobject_tests += { gobject_tests += {
'cxx' : { 'cxx' : {
'source' : ['cxx.cpp'], 'source' : ['cxx.cpp'],
'suite' : ['cpp'],
}, },
} }
foreach std, arg: cxx_standards
gobject_tests += {
'cxx-@0@'.format(std) : {
'source' : ['cxx.cpp'],
'suite' : ['cpp'],
'cpp_args' : [arg],
},
}
endforeach
endif endif
if cc.get_id() != 'msvc' if cc.get_id() != 'msvc'
@ -125,6 +136,7 @@ test_env.set('MALLOC_CHECK_', '2')
test_deps = [libm, thread_dep, libglib_dep, libgobject_dep] test_deps = [libm, thread_dep, libglib_dep, libgobject_dep]
test_cargs = ['-DG_LOG_DOMAIN="GLib-GObject"', '-UG_DISABLE_ASSERT'] test_cargs = ['-DG_LOG_DOMAIN="GLib-GObject"', '-UG_DISABLE_ASSERT']
test_cpp_args = test_cargs
foreach test_name, extra_args : gobject_tests foreach test_name, extra_args : gobject_tests
source = extra_args.get('source', test_name + '.c') source = extra_args.get('source', test_name + '.c')
@ -146,6 +158,7 @@ foreach test_name, extra_args : gobject_tests
exe = executable(test_name, source, exe = executable(test_name, source,
c_args : test_cargs + extra_args.get('c_args', []), c_args : test_cargs + extra_args.get('c_args', []),
cpp_args : test_cpp_args + extra_args.get('cpp_args', []),
dependencies : test_deps + extra_args.get('dependencies', []), dependencies : test_deps + extra_args.get('dependencies', []),
install_dir: installed_tests_execdir, install_dir: installed_tests_execdir,
install_tag: 'tests', install_tag: 'tests',

View File

@ -13,6 +13,14 @@ cc = meson.get_compiler('c')
have_cxx = add_languages('cpp', native: false, required: get_option('oss_fuzz').enabled()) have_cxx = add_languages('cpp', native: false, required: get_option('oss_fuzz').enabled())
if have_cxx if have_cxx
cxx = meson.get_compiler('cpp') cxx = meson.get_compiler('cpp')
cxx_standards = {}
foreach std : ['98', '03', '11', '14', '17', '20', '2b', 'latest']
arg = (cxx.get_id() == 'msvc' ? '/std:' : '-std=') + 'c++' + std
if cxx.has_argument(arg)
cxx_standards += { std: arg }
endif
endforeach
endif endif
cc_can_run = meson.can_run_host_binaries() cc_can_run = meson.can_run_host_binaries()