| project('glib', 'c', |
| version : '2.79.3', |
| # NOTE: See the policy in docs/meson-version.md before changing the Meson dependency |
| meson_version : '>= 1.2.0', |
| default_options : [ |
| 'buildtype=debugoptimized', |
| 'warning_level=3', |
| 'c_std=gnu99' |
| ] |
| ) |
| |
| fs = import('fs') |
| |
| cc = meson.get_compiler('c') |
| c_standards = {} |
| |
| foreach std : ['90', '99', '11', '17'] |
| arg = (cc.get_id() == 'msvc' ? '/std:' : '-std=') + 'c' + std |
| if cc.has_argument(arg) |
| c_standards += { std: arg } |
| endif |
| endforeach |
| |
| have_cxx = add_languages('cpp', native: false, required: get_option('oss_fuzz').enabled()) |
| if have_cxx |
| 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 |
| |
| cc_can_run = meson.can_run_host_binaries() |
| |
| if cc.get_argument_syntax() == 'msvc' |
| # Ignore several spurious warnings for things glib does very commonly |
| # (also for clang-cl) |
| add_project_arguments('/FIglib/msvc_recommended_pragmas.h', language : 'c') |
| endif |
| |
| if cc.get_id() == 'msvc' |
| # Set the input and exec encoding to utf-8, like is the default with GCC |
| add_project_arguments(cc.get_supported_arguments(['/utf-8']), language: 'c') |
| # Disable SAFESEH with MSVC for plugins and libs that use external deps that |
| # are built with MinGW |
| noseh_link_args = ['/SAFESEH:NO'] |
| else |
| noseh_link_args = [] |
| # -mms-bitfields vs -fnative-struct ? |
| endif |
| |
| host_system = host_machine.system() |
| |
| if host_system == 'darwin' |
| ios_test_code = '''#include <TargetConditionals.h> |
| #if ! TARGET_OS_IPHONE |
| #error "Not iOS/tvOS/watchOS/iPhoneSimulator" |
| #endif''' |
| if cc.compiles(ios_test_code, name : 'building for iOS') |
| host_system = 'ios' |
| endif |
| endif |
| |
| linux_libc = '' |
| if host_system == 'linux' |
| musl_test_code = '''#include <stdlib.h> |
| #if defined __GLIBC__ || defined __UCLIBC__ |
| #error "Not in glibc or uclibc" |
| #endif''' |
| if cc.compiles(musl_test_code, name : 'building for musl libc') |
| linux_libc = 'musl' |
| endif |
| endif |
| |
| glib_version = meson.project_version() |
| glib_api_version = '2.0' |
| version_arr = glib_version.split('.') |
| major_version = version_arr[0].to_int() |
| minor_version = version_arr[1].to_int() |
| micro_version = version_arr[2].to_int() |
| |
| interface_age = minor_version.is_odd() ? 0 : micro_version |
| binary_age = 100 * minor_version + micro_version |
| |
| soversion = 0 |
| # Maintain compatibility with previous libtool versioning |
| # current = minor * 100 + micro |
| current = binary_age - interface_age |
| library_version = '@0@.@1@.@2@'.format(soversion, current, interface_age) |
| darwin_versions = [current + 1, '@0@.@1@'.format(current + 1, interface_age)] |
| |
| configinc = include_directories('.') |
| glibinc = include_directories('glib') |
| gobjectinc = include_directories('gobject') |
| gmoduleinc = include_directories('gmodule') |
| gioinc = include_directories('gio') |
| girepoinc = include_directories('girepository') |
| |
| glib_prefix = get_option('prefix') |
| glib_bindir = join_paths(glib_prefix, get_option('bindir')) |
| glib_libdir = join_paths(glib_prefix, get_option('libdir')) |
| glib_libexecdir = join_paths(glib_prefix, get_option('libexecdir')) |
| glib_datadir = join_paths(glib_prefix, get_option('datadir')) |
| glib_pkgdatadir = join_paths(glib_datadir, 'glib-2.0') |
| glib_includedir = join_paths(glib_prefix, get_option('includedir'), 'glib-2.0') |
| if get_option('gio_module_dir') != '' |
| glib_giomodulesdir = join_paths(glib_prefix, get_option('gio_module_dir')) |
| else |
| glib_giomodulesdir = join_paths(glib_libdir, 'gio', 'modules') |
| endif |
| |
| if get_option('multiarch') |
| # For multiarch/multilib distributions, install each architecture's |
| # build of executables used in packaging triggers (like gio-querymodules) |
| # to an architecture-dependent location, with a compatibility symlink |
| # in the PATH. |
| multiarch_bindir = get_option('libdir') / 'glib-2.0' |
| multiarch_libexecdir = multiarch_bindir |
| pkgconfig_multiarch_bindir = '${libdir}/glib-2.0' |
| else |
| # For single-architecture distributions, just install them into the PATH |
| # as was traditionally done. |
| multiarch_bindir = get_option('bindir') |
| multiarch_libexecdir = get_option('libexecdir') |
| pkgconfig_multiarch_bindir = '${bindir}' |
| endif |
| |
| glib_pkgconfigreldir = join_paths(glib_libdir, 'pkgconfig') |
| |
| if get_option('charsetalias_dir') != '' |
| glib_charsetaliasdir = join_paths(glib_prefix, get_option('charsetalias_dir')) |
| else |
| glib_charsetaliasdir = glib_libdir |
| endif |
| |
| glib_localstatedir = glib_prefix / get_option('localstatedir') |
| |
| if get_option('runtime_dir') != '' |
| glib_runstatedir = glib_prefix / get_option('runtime_dir') |
| else |
| # While we’d normally prefix directories like this with, for example, |
| # glib_localstatedir, `/run` is a bit different in that it’s for runtime state |
| # rather than data files, so it’s typically functionally useless to use a |
| # prefixed version. No other processes will be using it. So we default to the |
| # unprefixed system `/run` directory. |
| glib_runstatedir = '/run' |
| endif |
| |
| # When building glib and gobject-introspection with subprojects, gobject-introspection |
| # requires to know the path of the sources and the build directory for the subproject. |
| # We provide it here with a variable. |
| glib_source_dir = meson.current_source_dir() |
| glib_build_dir = meson.current_build_dir() |
| |
| installed_tests_metadir = join_paths(glib_datadir, 'installed-tests', meson.project_name()) |
| installed_tests_execdir = join_paths(glib_libexecdir, 'installed-tests', meson.project_name()) |
| installed_tests_enabled = get_option('installed_tests') |
| installed_tests_template = files('tests/template.test.in') |
| installed_tests_template_tap = files('tests/template-tap.test.in') |
| |
| # Don’t build the tests unless we can run them (either natively, in an exe wrapper, or by installing them for later use) |
| build_tests = get_option('tests') and (meson.can_run_host_binaries() or installed_tests_enabled) |
| |
| common_test_env = [ |
| 'G_DEBUG=gc-friendly', |
| 'G_ENABLE_DIAGNOSTIC=1', |
| 'MALLOC_CHECK_=2', |
| ] |
| |
| if get_option('werror') |
| common_test_env += 'LINT_WARNINGS_ARE_ERRORS=1' |
| endif |
| |
| # Note: this may cause the tests output not to be printed when running in |
| # verbose mode, see https://github.com/mesonbuild/meson/issues/11185 |
| # Can be changed it to 'exitcode' if required during development. |
| test_protocol = 'tap' |
| test_timeout = 30 |
| test_timeout_slow = 90 |
| |
| add_test_setup('default', |
| is_default: true, |
| exclude_suites: ['flaky', 'failing'], |
| env: common_test_env, |
| timeout_multiplier: 2, |
| ) |
| |
| add_test_setup('unstable_tests', |
| env: common_test_env, |
| timeout_multiplier: 2, |
| # Empty test setup, used for having different results set for flaky tests |
| # Sadly we can't use (https://github.com/mesonbuild/meson/issues/10934): |
| #suites: ['flaky', 'unstable'] |
| ) |
| |
| add_test_setup('thorough', |
| exclude_suites: ['flaky', 'failing', 'performance'], |
| env: common_test_env, |
| timeout_multiplier: 20, |
| exe_wrapper: [find_program('./.gitlab-ci/thorough-test-wrapper.sh', required: true)], |
| ) |
| |
| # Allow the tests to be easily run under valgrind using --setup=valgrind |
| valgrind = find_program('valgrind', required: false) |
| valgrind_suppression_file = files('tools' / 'glib.supp')[0] |
| valgrind_suppression_file_install_subdir = 'glib-2.0' / 'valgrind' |
| |
| if valgrind.found() |
| add_test_setup('valgrind', |
| exclude_suites: [ 'no-valgrind', 'flaky' ], |
| exe_wrapper: [ |
| valgrind, |
| '--tool=memcheck', |
| '--error-exitcode=1', |
| '--track-origins=yes', |
| '--leak-check=full', |
| '--leak-resolution=high', |
| '--num-callers=50', |
| '--show-leak-kinds=definite,possible', |
| '--show-error-list=yes', |
| '--suppressions=@0@'.format(meson.project_source_root() / |
| '@0@'.format(valgrind_suppression_file)), |
| ], |
| env: common_test_env, |
| timeout_multiplier: 20, |
| ) |
| endif |
| |
| add_project_arguments('-D_GNU_SOURCE', language: 'c') |
| |
| if host_system == 'qnx' |
| add_project_arguments('-D_QNX_SOURCE', language: 'c') |
| endif |
| |
| if host_system == 'windows' |
| add_project_arguments(['-DUNICODE', '-D_UNICODE'], language: 'c') |
| endif |
| |
| # Disable strict aliasing; |
| # see https://bugzilla.gnome.org/show_bug.cgi?id=791622 |
| if cc.has_argument('-fno-strict-aliasing') |
| add_project_arguments('-fno-strict-aliasing', language: 'c') |
| endif |
| |
| # dummy/empty dependency() object to declare fallbacks and simpler dependencies |
| not_found = dependency('', required: false) |
| |
| ######################## |
| # Configuration begins # |
| ######################## |
| glib_conf = configuration_data() |
| glibconfig_conf = configuration_data() |
| |
| # accumulated list of defines as we check for them, so we can easily |
| # use them later in test programs (autoconf does this automatically) |
| glib_conf_prefix = '' |
| |
| glib_conf.set('GLIB_MAJOR_VERSION', major_version) |
| glib_conf.set('GLIB_MINOR_VERSION', minor_version) |
| glib_conf.set('GLIB_MICRO_VERSION', micro_version) |
| glib_conf.set('GLIB_INTERFACE_AGE', interface_age) |
| glib_conf.set('GLIB_BINARY_AGE', binary_age) |
| glib_conf.set_quoted('GETTEXT_PACKAGE', 'glib20') |
| glib_conf.set_quoted('PACKAGE_BUGREPORT', 'https://gitlab.gnome.org/GNOME/glib/issues/new') |
| glib_conf.set_quoted('PACKAGE_NAME', 'glib') |
| glib_conf.set_quoted('PACKAGE_STRING', 'glib @0@'.format(meson.project_version())) |
| glib_conf.set_quoted('PACKAGE_TARNAME', 'glib') |
| glib_conf.set_quoted('PACKAGE_URL', '') |
| glib_conf.set_quoted('PACKAGE_VERSION', meson.project_version()) |
| glib_conf.set('ENABLE_NLS', 1) |
| |
| # used by the .rc.in files |
| glibconfig_conf.set('LT_CURRENT_MINUS_AGE', soversion) |
| |
| glib_conf.set('_GNU_SOURCE', 1) |
| |
| if host_system in ['windows', 'darwin'] |
| # Poll doesn't work on devices on Windows, and macOS's poll() implementation is known to be broken |
| glib_conf.set('BROKEN_POLL', true) |
| endif |
| |
| if host_system == 'windows' and cc.get_id() != 'msvc' and cc.get_id() != 'clang-cl' |
| # FIXME: Ideally we shouldn't depend on this on Windows and should use |
| # 64 bit capable Windows API that also works with MSVC. |
| # The autotools build did set this for mingw and while meson sets it |
| # for gcc/clang by default, it doesn't do so on Windows. |
| glib_conf.set('_FILE_OFFSET_BITS', 64) |
| endif |
| |
| glib_build_shared = false |
| glib_build_static = false |
| if get_option('default_library') == 'both' |
| glib_build_static = true |
| glib_build_shared = true |
| elif get_option('default_library') == 'static' |
| glib_build_static = true |
| elif get_option('default_library') == 'shared' |
| glib_build_shared = true |
| endif |
| |
| glib_build_both = glib_build_static and glib_build_shared |
| glib_build_static_only = glib_build_static and not glib_build_shared |
| glib_build_shared_only = glib_build_shared and not glib_build_static |
| |
| if glib_build_shared and glib_build_static and ( |
| host_system == 'windows' or host_system == 'cygwin') |
| error('On Windows default_library must be "shared" or "static" but not "both"') |
| endif |
| |
| if glib_build_static_only |
| glibconfig_conf.set('GLIB_STATIC_COMPILATION', '1') |
| glibconfig_conf.set('GOBJECT_STATIC_COMPILATION', '1') |
| glibconfig_conf.set('GIO_STATIC_COMPILATION', '1') |
| glibconfig_conf.set('GMODULE_STATIC_COMPILATION', '1') |
| glibconfig_conf.set('GI_STATIC_COMPILATION', '1') |
| glibconfig_conf.set('G_INTL_STATIC_COMPILATION', '1') |
| glibconfig_conf.set('FFI_STATIC_BUILD', '1') |
| endif |
| |
| # Cygwin glib port maintainers made it clear |
| # (via the patches they apply) that they want no |
| # part of glib W32 code, therefore we do not define |
| # G_PLATFORM_WIN32 for host_system == 'cygwin'. |
| # This makes G_PLATFORM_WIN32 a synonym for |
| # G_OS_WIN32. |
| if host_system == 'windows' |
| glib_os = '''#define G_OS_WIN32 |
| #define G_PLATFORM_WIN32''' |
| elif host_system == 'cygwin' |
| glib_os = '''#define G_OS_UNIX |
| #define G_WITH_CYGWIN''' |
| else |
| glib_os = '#define G_OS_UNIX' |
| endif |
| glibconfig_conf.set('glib_os', glib_os) |
| |
| # We need to know the CRT being used to determine what .lib files we need on |
| # Visual Studio for dependencies that don't normally come with pkg-config files |
| vs_crt = 'release' |
| vs_crt_opt = get_option('b_vscrt') |
| if vs_crt_opt in ['mdd', 'mtd'] |
| vs_crt = 'debug' |
| elif vs_crt_opt == 'from_buildtype' |
| if get_option('buildtype') == 'debug' |
| vs_crt = 'debug' |
| endif |
| endif |
| |
| # Use debug/optimization flags to determine whether to enable debug or disable |
| # cast checks. We have a non-production (debug) build if debug is true and if |
| # optimization is 0 or g; otherwise, we have a production build. |
| glib_debug_cflags = [] |
| glib_debug = get_option('glib_debug') |
| if (glib_debug.enabled() or ( |
| glib_debug.auto() and get_option('debug') and get_option('optimization') in [ '0', 'g' ])) |
| glib_debug_cflags += ['-DG_ENABLE_DEBUG'] |
| message('Enabling various debug infrastructure') |
| else |
| glib_debug_cflags += ['-DG_DISABLE_CAST_CHECKS'] |
| message('Disabling cast checks') |
| endif |
| |
| if not get_option('glib_assert') |
| glib_debug_cflags += ['-DG_DISABLE_ASSERT'] |
| message('Disabling GLib asserts') |
| endif |
| |
| if not get_option('glib_checks') |
| glib_debug_cflags += ['-DG_DISABLE_CHECKS'] |
| message('Disabling GLib checks') |
| endif |
| |
| add_project_arguments(glib_debug_cflags, language: 'c') |
| |
| # check for header files |
| |
| headers = [ |
| 'alloca.h', |
| 'afunix.h', |
| 'crt_externs.h', |
| 'dirent.h', # MSC does not come with this by default |
| 'float.h', |
| 'fstab.h', |
| 'grp.h', |
| 'inttypes.h', |
| 'libproc.h', |
| 'limits.h', |
| 'locale.h', |
| 'mach/mach_time.h', |
| 'memory.h', |
| 'mntent.h', |
| 'poll.h', |
| 'pwd.h', |
| 'sched.h', |
| 'spawn.h', |
| 'stdatomic.h', |
| 'stdint.h', |
| 'stdlib.h', |
| 'string.h', |
| 'strings.h', |
| 'sys/auxv.h', |
| 'sys/event.h', |
| 'sys/filio.h', |
| 'sys/inotify.h', |
| 'sys/mkdev.h', |
| 'sys/mntctl.h', |
| 'sys/mnttab.h', |
| 'sys/mount.h', |
| 'sys/param.h', |
| 'sys/prctl.h', |
| 'sys/resource.h', |
| 'sys/select.h', |
| 'sys/statfs.h', |
| 'sys/stat.h', |
| 'sys/statvfs.h', |
| 'sys/sysctl.h', |
| 'sys/time.h', # MSC does not come with this by default |
| 'sys/times.h', |
| 'sys/types.h', |
| 'sys/uio.h', |
| 'sys/vfs.h', |
| 'sys/vfstab.h', |
| 'sys/vmount.h', |
| 'sys/wait.h', |
| 'syslog.h', |
| 'termios.h', |
| 'unistd.h', |
| 'values.h', |
| 'wchar.h', |
| 'xlocale.h', |
| ] |
| |
| foreach h : headers |
| if cc.has_header(h) |
| define = 'HAVE_' + h.underscorify().to_upper() |
| glib_conf.set(define, 1) |
| glib_conf_prefix = glib_conf_prefix + '#define @0@ 1\n'.format(define) |
| endif |
| endforeach |
| |
| # FreeBSD includes a malloc.h which always throws compilation error, so we have |
| # to use check_header() rather than has_header(). |
| if cc.check_header('malloc.h') |
| glib_conf.set('HAVE_MALLOC_H', 1) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_MALLOC_H 1\n' |
| endif |
| |
| if cc.check_header('linux/netlink.h') |
| glib_conf.set('HAVE_NETLINK', 1) |
| endif |
| |
| # Is statx() supported? Android systems don’t reliably support it as of August 2020. |
| statx_code = ''' |
| #ifndef _GNU_SOURCE |
| #define _GNU_SOURCE |
| #endif |
| #include <sys/stat.h> |
| #include <fcntl.h> |
| int main (void) |
| { |
| struct statx stat_buf; |
| return statx (AT_FDCWD, "/", AT_SYMLINK_NOFOLLOW, STATX_BASIC_STATS | STATX_BTIME, &stat_buf); |
| } |
| ''' |
| if host_system != 'android' and cc.compiles(statx_code, name : 'statx() test') |
| glib_conf.set('HAVE_STATX', 1) |
| endif |
| |
| if glib_conf.has('HAVE_LOCALE_H') |
| if cc.has_header_symbol('locale.h', 'LC_MESSAGES') |
| glib_conf.set('HAVE_LC_MESSAGES', 1) |
| endif |
| endif |
| |
| struct_stat_blkprefix = ''' |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #ifdef HAVE_UNISTD_H |
| #include <unistd.h> |
| #endif |
| #ifdef HAVE_SYS_STATFS_H |
| #include <sys/statfs.h> |
| #endif |
| #ifdef HAVE_SYS_PARAM_H |
| #include <sys/param.h> |
| #endif |
| #ifdef HAVE_SYS_MOUNT_H |
| #include <sys/mount.h> |
| #endif |
| ''' |
| |
| struct_members = [ |
| [ 'stat', 'st_mtimensec' ], |
| [ 'stat', 'st_mtim.tv_nsec' ], |
| [ 'stat', 'st_atimensec' ], |
| [ 'stat', 'st_atim.tv_nsec' ], |
| [ 'stat', 'st_ctimensec' ], |
| [ 'stat', 'st_ctim.tv_nsec' ], |
| [ 'stat', 'st_birthtime' ], |
| [ 'stat', 'st_birthtimensec' ], |
| [ 'stat', 'st_birthtim' ], |
| [ 'stat', 'st_birthtim.tv_nsec' ], |
| [ 'stat', 'st_blksize', struct_stat_blkprefix ], |
| [ 'stat', 'st_blocks', struct_stat_blkprefix ], |
| [ 'statfs', 'f_fstypename', struct_stat_blkprefix ], |
| [ 'statfs', 'f_bavail', struct_stat_blkprefix ], |
| [ 'dirent', 'd_type', '''#include <sys/types.h> |
| #include <dirent.h>''' ], |
| [ 'statvfs', 'f_basetype', '#include <sys/statvfs.h>' ], |
| [ 'statvfs', 'f_fstypename', '#include <sys/statvfs.h>' ], |
| [ 'statvfs', 'f_type', '#include <sys/statvfs.h>' ], |
| [ 'tm', 'tm_gmtoff', '#include <time.h>' ], |
| [ 'tm', '__tm_gmtoff', '#include <time.h>' ], |
| ] |
| |
| foreach m : struct_members |
| header_check_prefix = glib_conf_prefix |
| if m.length() == 3 |
| header_check_prefix = header_check_prefix + m[2] |
| else |
| header_check_prefix = header_check_prefix + '#include <sys/stat.h>' |
| endif |
| # Reimplement cc.has_member() to workaround compiler warning |
| # FIXME: https://github.com/mesonbuild/meson/pull/12818 |
| code = header_check_prefix + ''' |
| void bar(void) { |
| struct ''' + m[0] + ''' foo; |
| (void) ( foo.''' + m[1] + ''' ); |
| (void) foo; |
| } |
| ''' |
| if cc.compiles(code, name : 'type "struct ' + m[0] + '" has member "' + m[1] + '"') |
| define = 'HAVE_STRUCT_@0@_@1@'.format(m[0].to_upper(), m[1].underscorify().to_upper()) |
| glib_conf.set(define, 1) |
| glib_conf_prefix = glib_conf_prefix + '#define @0@ 1\n'.format(define) |
| else |
| endif |
| endforeach |
| |
| # Compiler flags |
| if cc.get_id() == 'gcc' or cc.get_id() == 'clang' |
| warning_common_args = [ |
| '-Wduplicated-branches', |
| '-Wimplicit-fallthrough', |
| '-Wmisleading-indentation', |
| '-Wmissing-field-initializers', |
| '-Wnonnull', |
| '-Wunused', |
| # Due to maintained deprecated code, we do not want to see unused parameters |
| '-Wno-unused-parameter', |
| # Due to pervasive use of things like GPOINTER_TO_UINT(), we do not support |
| # building with -Wbad-function-cast. |
| '-Wno-cast-function-type', |
| # Due to function casts through (void*) we cannot support -Wpedantic: |
| # ./docs/toolchain-requirements.md#Function_pointer_conversions. |
| '-Wno-pedantic', |
| # A zero-length format string shouldn't be considered an issue. |
| '-Wno-format-zero-length', |
| # We explicitly require variadic macros |
| '-Wno-variadic-macros', |
| '-Werror=format=2', |
| '-Werror=init-self', |
| '-Werror=missing-include-dirs', |
| '-Werror=pointer-arith', |
| '-Werror=unused-result', |
| ] |
| |
| warning_c_args = warning_common_args + [ |
| '-Wstrict-prototypes', |
| # Due to pervasive use of things like GPOINTER_TO_UINT(), we do not support |
| # building with -Wbad-function-cast. |
| '-Wno-bad-function-cast', |
| '-Werror=implicit-function-declaration', |
| '-Werror=missing-prototypes', |
| '-Werror=pointer-sign', |
| '-Wno-string-plus-int', |
| ] |
| warning_cxx_args = warning_common_args |
| warning_objc_args = warning_c_args |
| warning_c_link_args = [ |
| '-Wl,-z,nodelete', |
| ] |
| if get_option('bsymbolic_functions') |
| warning_c_link_args += ['-Wl,-Bsymbolic-functions'] |
| endif |
| elif cc.get_id() == 'msvc' |
| warning_c_args = [ |
| # If a warning is completely useless and spammy, use '/wdXXXX' to suppress it |
| # If a warning is harmless but hard to fix, use '/woXXXX' so it's shown once |
| # NOTE: Only add warnings here if you are sure they're spurious |
| '/wo4057', # 'operator': 'identifier1' differs in indirection to slightly different base types from 'identifier2' |
| '/wd4068', # unknown pragma |
| '/wo4090', # 'operation': different 'modifier' qualifiers |
| '/wd4100', # 'identifier': unreferenced formal parameter |
| '/wd4116', # unnamed type definition in parentheses |
| '/wo4125', # decimal digit terminates octal escape sequence |
| '/wd4127', # conditional expression is constant |
| '/wd4146', # unary minus operator applied to unsigned type, result still unsigned |
| '/wd4152', # nonstandard extension, function/data pointer conversion in expression |
| '/wd4201', # nonstandard extension used: nameless struct/union |
| '/wd4232', # nonstandard extension used: 'identifier': address of dllimport 'dllimport' is not static, identity not guaranteed |
| '/wo4245', # 'conversion_type': conversion from 'type1' to 'type2', signed/unsigned mismatch |
| '/wo4267', # 'variable': conversion from 'size_t' to 'type', possible loss of data |
| '/wd4334', # 'shift_operator': result of 32-bit shift implicitly converted to 64 bits (was 64-bit shift intended?) |
| '/wo4389', # 'operator': signed/unsigned mismatch |
| '/wo4702', # unreachable code |
| '/wd4706', # assignment within conditional expression |
| ] |
| |
| warning_cxx_args = [] |
| warning_objc_args = [] |
| warning_c_link_args = [] |
| else |
| warning_c_args = [] |
| warning_cxx_args = [] |
| warning_objc_args = [] |
| warning_c_link_args = [] |
| endif |
| |
| add_project_arguments(cc.get_supported_arguments(warning_c_args), language: 'c') |
| if have_cxx |
| add_project_arguments(cxx.get_supported_arguments(warning_cxx_args), language: 'cpp') |
| endif |
| |
| # FIXME: We cannot build some of the GResource tests with -z nodelete, which |
| # means we cannot use that flag in add_project_link_arguments(), and must add |
| # it to the relevant targets manually. We do the same with -Bsymbolic-functions |
| # because that is what the autotools build did. |
| # See https://github.com/mesonbuild/meson/pull/3520 for a way to eventually |
| # improve this. |
| glib_link_flags = cc.get_supported_link_arguments(warning_c_link_args) |
| |
| # Windows SDK requirements and checks |
| if host_system == 'windows' |
| # Check whether we're building for UWP apps |
| code = ''' |
| #include <windows.h> |
| #if !(WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_APP) && !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)) |
| #error "Not building for UWP" |
| #endif''' |
| if cc.compiles(code, name : 'building for UWP') |
| glib_conf.set('G_WINAPI_ONLY_APP', true) |
| # We require Windows 10+ on WinRT |
| glib_conf.set('_WIN32_WINNT', '0x0A00') |
| uwp_gio_deps = [cc.find_library('shcore'), |
| cc.find_library('runtimeobject')] |
| else |
| # We require Windows 7+ on Win32 |
| glib_conf.set('_WIN32_WINNT', '0x0601') |
| uwp_gio_deps = [] |
| endif |
| endif |
| |
| functions = [ |
| 'accept4', |
| 'close_range', |
| 'copy_file_range', |
| 'endmntent', |
| 'endservent', |
| 'epoll_create', |
| 'fallocate', |
| 'fchmod', |
| 'fchown', |
| 'fdwalk', |
| 'free_aligned_sized', |
| 'free_sized', |
| 'fsync', |
| 'ftruncate64', |
| 'getauxval', |
| 'getc_unlocked', |
| 'getfsstat', |
| 'getgrgid_r', |
| 'getmntent_r', |
| 'getpwuid_r', |
| 'getresuid', |
| 'getvfsstat', |
| 'gmtime_r', |
| 'hasmntopt', |
| 'inotify_init1', |
| 'issetugid', |
| 'kevent', |
| 'kqueue', |
| 'lchmod', |
| 'lchown', |
| 'link', |
| 'localtime_r', |
| 'lstat', |
| 'mbrtowc', |
| 'memalign', |
| 'mmap', |
| 'newlocale', |
| 'pipe2', |
| 'poll', |
| 'prlimit', |
| 'readlink', |
| 'recvmmsg', |
| 'sendmmsg', |
| 'setenv', |
| 'setmntent', |
| 'strerror_r', |
| 'strnlen', |
| 'strsignal', |
| 'strtod_l', |
| 'strtoll_l', |
| 'strtoull_l', |
| 'symlink', |
| 'timegm', |
| 'unsetenv', |
| 'uselocale', |
| 'utimes', |
| 'utimensat', |
| 'valloc', |
| 'vasprintf', |
| 'vsnprintf', |
| 'wcrtomb', |
| 'wcslen', |
| 'wcsnlen', |
| 'sysctlbyname', |
| ] |
| |
| # _NSGetEnviron is available on iOS too, but its usage gets apps rejected from |
| # the app store since it's considered 'private API' |
| if host_system == 'darwin' |
| functions += ['_NSGetEnviron'] |
| endif |
| |
| if glib_conf.has('HAVE_SYS_STATVFS_H') |
| functions += ['statvfs'] |
| else |
| have_func_statvfs = false |
| endif |
| if glib_conf.has('HAVE_SYS_STATFS_H') or glib_conf.has('HAVE_SYS_MOUNT_H') |
| functions += ['statfs'] |
| else |
| have_func_statfs = false |
| endif |
| if glib_conf.has('HAVE_SYS_PRCTL_H') |
| functions += ['prctl'] |
| else |
| have_func_prctl = false |
| endif |
| |
| if host_system == 'windows' |
| iphlpapi_dep = cc.find_library('iphlpapi') |
| iphlpapi_funcs = ['if_nametoindex', 'if_indextoname'] |
| foreach ifunc : iphlpapi_funcs |
| iphl_prefix = '''#define _WIN32_WINNT @0@ |
| #include <winsock2.h> |
| #include <iphlpapi.h>'''.format(glib_conf.get('_WIN32_WINNT')) |
| if cc.has_function(ifunc, |
| prefix : iphl_prefix, |
| dependencies : iphlpapi_dep) |
| idefine = 'HAVE_' + ifunc.underscorify().to_upper() |
| glib_conf.set(idefine, 1) |
| glib_conf_prefix = glib_conf_prefix + '#define @0@ 1\n'.format(idefine) |
| set_variable('have_func_' + ifunc, true) |
| else |
| set_variable('have_func_' + ifunc, false) |
| endif |
| endforeach |
| else |
| functions += ['if_indextoname', 'if_nametoindex'] |
| endif |
| |
| # AIX splice is something else |
| if host_system != 'aix' |
| functions += ['splice'] |
| endif |
| |
| foreach f : functions |
| if cc.has_function(f) |
| define = 'HAVE_' + f.underscorify().to_upper() |
| glib_conf.set(define, 1) |
| glib_conf_prefix = glib_conf_prefix + '#define @0@ 1\n'.format(define) |
| set_variable('have_func_' + f, true) |
| else |
| set_variable('have_func_' + f, false) |
| endif |
| endforeach |
| |
| # Export the information about free_sized() so we can correctly define a macro |
| # wrapper around g_free()/g_free_sized() depending on whether it’s available |
| glibconfig_conf.set('G_HAVE_FREE_SIZED', have_func_free_sized) |
| |
| # Check that stpcpy() is usable; must use header. |
| # See: |
| # https://github.com/mesonbuild/meson/issues/5628. |
| if cc.has_function('stpcpy', prefix : '#include <string.h>') |
| glib_conf.set('HAVE_STPCPY', 1) |
| endif |
| |
| if cc.has_function('memalign', prefix: '#include <stdlib.h>\n#include <malloc.h>') |
| glib_conf.set('HAVE_MEMALIGN', 1) |
| endif |
| |
| # For example on Openbsd, getservbyname_r() has a different signature. |
| # https://man.openbsd.org/getservbyname.3 |
| if cc.compiles('''#include <netdb.h> |
| int main (int argc, char ** argv) { |
| int (*fcn)(const char *, |
| const char *, |
| struct servent *, |
| char *, |
| size_t, |
| struct servent **) = getservbyname_r; |
| (void) fcn; |
| return 0; |
| }''', |
| name : 'getservbyname_r()', |
| args: '-Werror=incompatible-pointer-types') |
| glib_conf.set('HAVE_GETSERVBYNAME_R', 1) |
| endif |
| |
| if cc.has_function('_aligned_malloc', prefix: '#include <malloc.h>') |
| glib_conf.set('HAVE__ALIGNED_MALLOC', 1) |
| endif |
| |
| if host_system != 'windows' and cc.has_function('aligned_alloc', prefix: '#include <stdlib.h>') |
| glib_conf.set('HAVE_ALIGNED_ALLOC', 1) |
| endif |
| |
| if host_system != 'windows' and cc.has_function('posix_memalign', prefix: '#include <stdlib.h>') |
| glib_conf.set('HAVE_POSIX_MEMALIGN', 1) |
| endif |
| |
| # Check that posix_spawn() is usable; must use header |
| if cc.has_function('posix_spawn', prefix : '#include <spawn.h>') |
| glib_conf.set('HAVE_POSIX_SPAWN', 1) |
| endif |
| |
| # Check whether strerror_r returns char * |
| if have_func_strerror_r |
| if cc.compiles('''#define _GNU_SOURCE |
| #include <string.h> |
| int func (void) { |
| char error_string[256]; |
| char *ptr = strerror_r (-2, error_string, 256); |
| char c = *strerror_r (-2, error_string, 256); |
| return c != 0 && ptr != (void*) 0L; |
| } |
| ''', |
| name : 'strerror_r() returns char *') |
| glib_conf.set('STRERROR_R_CHAR_P', 1, |
| description: 'Defined if strerror_r returns char *') |
| endif |
| endif |
| |
| # Special-case these functions that have alternative names on Windows/MSVC |
| if cc.has_function('snprintf') or cc.has_header_symbol('stdio.h', 'snprintf') |
| glib_conf.set('HAVE_SNPRINTF', 1) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_SNPRINTF 1\n' |
| elif cc.has_function('_snprintf') or cc.has_header_symbol('stdio.h', '_snprintf') |
| hack_define = '1\n#define snprintf _snprintf' |
| glib_conf.set('HAVE_SNPRINTF', hack_define) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_SNPRINTF ' + hack_define |
| endif |
| |
| if cc.has_function('strcasecmp', prefix: '#include <strings.h>') |
| glib_conf.set('HAVE_STRCASECMP', 1) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_STRCASECMP 1\n' |
| elif cc.has_function('_stricmp') |
| hack_define = '1\n#define strcasecmp _stricmp' |
| glib_conf.set('HAVE_STRCASECMP', hack_define) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_STRCASECMP ' + hack_define |
| endif |
| |
| if cc.has_function('strncasecmp', prefix: '#include <strings.h>') |
| glib_conf.set('HAVE_STRNCASECMP', 1) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_STRNCASECMP 1\n' |
| elif cc.has_function('_strnicmp') |
| hack_define = '1\n#define strncasecmp _strnicmp' |
| glib_conf.set('HAVE_STRNCASECMP', hack_define) |
| glib_conf_prefix = glib_conf_prefix + '#define HAVE_STRNCASECMP ' + hack_define |
| endif |
| |
| if cc.has_header_symbol('sys/sysmacros.h', 'major') |
| glib_conf.set('MAJOR_IN_SYSMACROS', 1) |
| elif cc.has_header_symbol('sys/mkdev.h', 'major') |
| glib_conf.set('MAJOR_IN_MKDEV', 1) |
| elif cc.has_header_symbol('sys/types.h', 'major') |
| glib_conf.set('MAJOR_IN_TYPES', 1) |
| endif |
| |
| if cc.has_header_symbol('dlfcn.h', 'RTLD_LAZY') |
| glib_conf.set('HAVE_RTLD_LAZY', 1) |
| endif |
| |
| if cc.has_header_symbol('dlfcn.h', 'RTLD_NOW') |
| glib_conf.set('HAVE_RTLD_NOW', 1) |
| endif |
| |
| if cc.has_header_symbol('dlfcn.h', 'RTLD_GLOBAL') |
| glib_conf.set('HAVE_RTLD_GLOBAL', 1) |
| endif |
| |
| have_rtld_next = false |
| if cc.has_header_symbol('dlfcn.h', 'RTLD_NEXT', args: '-D_GNU_SOURCE') |
| have_rtld_next = true |
| glib_conf.set('HAVE_RTLD_NEXT', 1) |
| endif |
| |
| if cc.has_type('loff_t', prefix: '#include <sys/types.h>') |
| glib_conf.set('HAVE_LOFF_T', 1) |
| endif |
| |
| # Check whether to use statfs or statvfs |
| # Some systems have both statfs and statvfs, pick the most "native" for these |
| if have_func_statfs and have_func_statvfs |
| # on solaris and irix, statfs doesn't even have the f_bavail field |
| if not glib_conf.has('HAVE_STRUCT_STATFS_F_BAVAIL') |
| have_func_statfs = false |
| else |
| # at least on linux, statfs is the actual syscall |
| have_func_statvfs = false |
| endif |
| endif |
| if have_func_statfs |
| glib_conf.set('USE_STATFS', 1) |
| stat_func_to_use = 'statfs' |
| elif have_func_statvfs |
| glib_conf.set('USE_STATVFS', 1) |
| stat_func_to_use = 'statvfs' |
| else |
| stat_func_to_use = 'neither' |
| endif |
| message('Checking whether to use statfs or statvfs .. ' + stat_func_to_use) |
| |
| if host_system == 'linux' |
| if cc.has_function('mkostemp', |
| prefix: '''#define _GNU_SOURCE |
| #include <stdlib.h>''') |
| glib_conf.set('HAVE_MKOSTEMP', 1) |
| endif |
| endif |
| |
| osx_ldflags = [] |
| glib_have_os_x_9_or_later = false |
| glib_have_carbon = false |
| glib_have_cocoa = false |
| if host_system == 'darwin' |
| add_languages('objc', native: false, required: true) |
| objcc = meson.get_compiler('objc') |
| |
| add_project_arguments(objcc.get_supported_arguments(warning_objc_args), language: 'objc') |
| |
| # Mac OS X Carbon support |
| glib_have_carbon = objcc.compiles('''#include <Carbon/Carbon.h> |
| #include <CoreServices/CoreServices.h>''', |
| name : 'Mac OS X Carbon support') |
| |
| if glib_have_carbon |
| glib_conf.set('HAVE_CARBON', true) |
| glib_have_os_x_9_or_later = objcc.compiles('''#include <AvailabilityMacros.h> |
| #if MAC_OS_X_VERSION_MIN_REQUIRED < 1090 |
| #error Compiling for minimum OS X version before 10.9 |
| #endif''', |
| name : 'OS X 9 or later') |
| endif |
| |
| # Mac OS X Cocoa support |
| glib_have_cocoa = objcc.compiles('''#include <Cocoa/Cocoa.h> |
| #ifdef GNUSTEP_BASE_VERSION |
| #error "Detected GNUstep, not Cocoa" |
| #endif''', |
| name : 'Mac OS X Cocoa support') |
| |
| if glib_have_cocoa |
| glib_conf.set('HAVE_COCOA', true) |
| osx_ldflags += ['-Wl,-framework,Foundation', '-Wl,-framework,AppKit'] |
| endif |
| endif |
| |
| if host_system == 'qnx' |
| glib_conf.set('HAVE_QNX', 1) |
| endif |
| |
| # Check for futex(2) |
| if cc.compiles('''#include <linux/futex.h> |
| #include <sys/syscall.h> |
| #include <unistd.h> |
| int main (int argc, char ** argv) { |
| syscall (__NR_futex, NULL, FUTEX_WAKE, FUTEX_WAIT); |
| return 0; |
| }''', name : 'futex(2) system call') |
| glib_conf.set('HAVE_FUTEX', 1) |
| endif |
| if cc.compiles('''#include <linux/futex.h> |
| #include <sys/syscall.h> |
| #include <unistd.h> |
| int main (int argc, char ** argv) { |
| syscall (__NR_futex_time64, NULL, FUTEX_WAKE, FUTEX_WAIT); |
| return 0; |
| }''', name : 'futex(2) system call') |
| glib_conf.set('HAVE_FUTEX_TIME64', 1) |
| endif |
| |
| # Check for eventfd(2) |
| if cc.links('''#include <sys/eventfd.h> |
| #include <unistd.h> |
| int main (int argc, char ** argv) { |
| eventfd (0, EFD_CLOEXEC); |
| return 0; |
| }''', name : 'eventfd(2) system call') |
| glib_conf.set('HAVE_EVENTFD', 1) |
| endif |
| |
| # Check for pidfd_open(2) |
| if cc.links('''#include <sys/syscall.h> |
| #include <sys/wait.h> |
| #include <linux/wait.h> |
| #include <unistd.h> |
| int main (int argc, char ** argv) { |
| siginfo_t child_info = { 0, }; |
| syscall (SYS_pidfd_open, 0, 0); |
| waitid (P_PIDFD, 0, &child_info, WEXITED | WNOHANG); |
| return 0; |
| }''', name : 'pidfd_open(2) system call') |
| glib_conf.set('HAVE_PIDFD', 1) |
| endif |
| |
| # Check for __uint128_t (gcc) by checking for 128-bit division |
| uint128_t_src = '''int main() { |
| static __uint128_t v1 = 100; |
| static __uint128_t v2 = 10; |
| static __uint128_t u; |
| u = v1 / v2; |
| (void) u; |
| }''' |
| if cc.compiles(uint128_t_src, name : '__uint128_t available') |
| glib_conf.set('HAVE_UINT128_T', 1) |
| endif |
| |
| clock_gettime_test_code = ''' |
| #include <time.h> |
| struct timespec t; |
| int main (int argc, char ** argv) { |
| return clock_gettime(CLOCK_REALTIME, &t); |
| }''' |
| librt = [] |
| if cc.links(clock_gettime_test_code, name : 'clock_gettime') |
| glib_conf.set('HAVE_CLOCK_GETTIME', 1) |
| elif cc.links(clock_gettime_test_code, args : '-lrt', name : 'clock_gettime in librt') |
| glib_conf.set('HAVE_CLOCK_GETTIME', 1) |
| librt = cc.find_library('rt') |
| endif |
| |
| dlopen_dlsym_test_code = ''' |
| #include <dlfcn.h> |
| #include <stdio.h> |
| int r; |
| int glib_underscore_test (void) { return 42; } |
| int main (int argc, char ** argv) { |
| void *f1 = (void*)0, *f2 = (void*)0, *handle; |
| handle = dlopen ((void*)0, 0); |
| if (handle) { |
| f1 = dlsym (handle, "glib_underscore_test"); |
| f2 = dlsym (handle, "_glib_underscore_test"); |
| } |
| r = (!f2 || f1) ? puts ("1") : puts ("0"); |
| return r > 0 ? 0 : r; |
| }''' |
| libdl_dep = [] |
| if cc.links(dlopen_dlsym_test_code, name : 'dlopen() and dlsym() in system libraries') |
| have_dlopen_dlsym = true |
| elif cc.links(dlopen_dlsym_test_code, args : '-ldl', name : 'dlopen() and dlsym() in libdl') |
| have_dlopen_dlsym = true |
| libdl_dep = cc.find_library('dl') |
| else |
| have_dlopen_dlsym = false |
| endif |
| |
| # if statfs() takes 2 arguments (Posix) or 4 (Solaris) |
| if have_func_statfs |
| if cc.compiles(glib_conf_prefix + ''' |
| #include <unistd.h> |
| #ifdef HAVE_SYS_PARAM_H |
| #include <sys/param.h> |
| #endif |
| #ifdef HAVE_SYS_VFS_H |
| #include <sys/vfs.h> |
| #endif |
| #ifdef HAVE_SYS_MOUNT_H |
| #include <sys/mount.h> |
| #endif |
| #ifdef HAVE_SYS_STATFS_H |
| #include <sys/statfs.h> |
| #endif |
| void some_func (void) { |
| struct statfs st; |
| statfs("/", &st); |
| }''', name : 'number of arguments to statfs() (n=2)') |
| glib_conf.set('STATFS_ARGS', 2) |
| elif cc.compiles(glib_conf_prefix + ''' |
| #include <unistd.h> |
| #ifdef HAVE_SYS_PARAM_H |
| #include <sys/param.h> |
| #endif |
| #ifdef HAVE_SYS_VFS_H |
| #include <sys/vfs.h> |
| #endif |
| #ifdef HAVE_SYS_MOUNT_H |
| #include <sys/mount.h> |
| #endif |
| #ifdef HAVE_SYS_STATFS_H |
| #include <sys/statfs.h> |
| #endif |
| void some_func (void) { |
| struct statfs st; |
| statfs("/", &st, sizeof (st), 0); |
| }''', name : 'number of arguments to statfs() (n=4)') |
| glib_conf.set('STATFS_ARGS', 4) |
| else |
| error('Unable to determine number of arguments to statfs()') |
| endif |
| endif |
| |
| # open takes O_DIRECTORY as an option |
| #AC_MSG_CHECKING([]) |
| if cc.compiles('''#include <fcntl.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| void some_func (void) { |
| open(".", O_DIRECTORY, 0); |
| }''', name : 'open() option O_DIRECTORY') |
| glib_conf.set('HAVE_OPEN_O_DIRECTORY', 1) |
| endif |
| |
| # fcntl takes F_FULLFSYNC as an option |
| # See https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/fsync.2.html |
| if cc.compiles('''#include <fcntl.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| void some_func (void) { |
| fcntl(0, F_FULLFSYNC, 0); |
| }''', name : 'fcntl() option F_FULLFSYNC') |
| glib_conf.set('HAVE_FCNTL_F_FULLFSYNC', 1) |
| endif |
| |
| # Check whether there is a vsnprintf() function with C99 semantics installed. |
| # (similar tests to AC_FUNC_VSNPRINTF_C99) |
| # Check whether there is a snprintf() function with C99 semantics installed. |
| # (similar tests to AC_FUNC_SNPRINTF_C99) |
| # Check whether there is a printf() function with Unix98 semantics installed. |
| # (similar tests to AC_FUNC_PRINTF_UNIX98) |
| have_good_vsnprintf = false |
| have_good_snprintf = false |
| have_good_printf = false |
| |
| if host_system == 'windows' and (cc.get_id() == 'msvc' or cc.get_id() == 'clang-cl') |
| # Unfortunately the Visual Studio 2015+ implementations of C99-style |
| # snprintf and vsnprintf don't seem to be quite good enough. |
| # (Sorry, I don't know exactly what is the problem, |
| # but it is related to floating point formatting and decimal point vs. comma.) |
| # The simple tests in AC_FUNC_VSNPRINTF_C99 and AC_FUNC_SNPRINTF_C99 aren't |
| # rigorous enough to notice, though. |
| glib_conf.set('HAVE_C99_SNPRINTF', false) |
| glib_conf.set('HAVE_C99_VSNPRINTF', false) |
| glib_conf.set('HAVE_UNIX98_PRINTF', false) |
| elif not cc_can_run and host_system in ['ios', 'darwin'] |
| # All these are true when compiling natively on macOS, so we should use good |
| # defaults when building for iOS and tvOS. |
| glib_conf.set('HAVE_C99_SNPRINTF', true) |
| glib_conf.set('HAVE_C99_VSNPRINTF', true) |
| glib_conf.set('HAVE_UNIX98_PRINTF', true) |
| have_good_vsnprintf = true |
| have_good_snprintf = true |
| have_good_printf = true |
| else |
| vsnprintf_c99_test_code = ''' |
| #include <stdio.h> |
| #include <stdarg.h> |
| #include <stdlib.h> |
| |
| int |
| doit(char * s, ...) |
| { |
| char buffer[32]; |
| va_list args; |
| int r; |
| |
| va_start(args, s); |
| r = vsnprintf(buffer, 5, s, args); |
| va_end(args); |
| |
| if (r != 7) |
| exit(1); |
| |
| /* AIX 5.1 and Solaris seems to have a half-baked vsnprintf() |
| implementation. The above will return 7 but if you replace |
| the size of the buffer with 0, it borks! */ |
| va_start(args, s); |
| r = vsnprintf(buffer, 0, s, args); |
| va_end(args); |
| |
| if (r != 7) |
| exit(1); |
| |
| exit(0); |
| } |
| |
| int |
| main(void) |
| { |
| doit("1234567"); |
| exit(1); |
| }''' |
| |
| if cc_can_run |
| rres = cc.run(vsnprintf_c99_test_code, name : 'C99 vsnprintf') |
| if rres.compiled() and rres.returncode() == 0 |
| glib_conf.set('HAVE_C99_VSNPRINTF', 1) |
| have_good_vsnprintf = true |
| endif |
| else |
| have_good_vsnprintf = meson.get_external_property('have_c99_vsnprintf', false) |
| glib_conf.set('HAVE_C99_VSNPRINTF', have_good_vsnprintf) |
| endif |
| |
| snprintf_c99_test_code = ''' |
| #include <stdio.h> |
| #include <stdarg.h> |
| #include <stdlib.h> |
| |
| int |
| doit() |
| { |
| char buffer[32]; |
| int r; |
| |
| r = snprintf(buffer, 5, "1234567"); |
| |
| if (r != 7) |
| exit(1); |
| |
| r = snprintf(buffer, 0, "1234567"); |
| |
| if (r != 7) |
| exit(1); |
| |
| r = snprintf(NULL, 0, "1234567"); |
| |
| if (r != 7) |
| exit(1); |
| |
| exit(0); |
| } |
| |
| int |
| main(void) |
| { |
| doit(); |
| exit(1); |
| }''' |
| |
| if cc_can_run |
| rres = cc.run(snprintf_c99_test_code, name : 'C99 snprintf') |
| if rres.compiled() and rres.returncode() == 0 |
| glib_conf.set('HAVE_C99_SNPRINTF', 1) |
| have_good_snprintf = true |
| endif |
| else |
| have_good_snprintf = meson.get_external_property('have_c99_snprintf', false) |
| glib_conf.set('HAVE_C99_SNPRINTF', have_good_snprintf) |
| endif |
| |
| printf_unix98_test_code = ''' |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| int |
| main (void) |
| { |
| char buffer[128]; |
| |
| sprintf (buffer, "%2\$d %3\$d %1\$d", 1, 2, 3); |
| if (strcmp ("2 3 1", buffer) == 0) |
| exit (0); |
| exit (1); |
| }''' |
| |
| if cc_can_run |
| rres = cc.run(printf_unix98_test_code, name : 'Unix98 printf positional parameters') |
| if rres.compiled() and rres.returncode() == 0 |
| glib_conf.set('HAVE_UNIX98_PRINTF', 1) |
| have_good_printf = true |
| endif |
| else |
| have_good_printf = meson.get_external_property('have_unix98_printf', false) |
| glib_conf.set('HAVE_UNIX98_PRINTF', have_good_printf) |
| endif |
| endif |
| |
| if host_system == 'windows' |
| glib_conf.set_quoted('EXEEXT', '.exe') |
| else |
| glib_conf.set('EXEEXT', '') |
| endif |
| |
| # Our printf is 'good' only if vsnpintf()/snprintf()/printf() supports C99 well enough |
| use_system_printf = have_good_vsnprintf and have_good_snprintf and have_good_printf |
| glib_conf.set('USE_SYSTEM_PRINTF', use_system_printf) |
| glibconfig_conf.set('GLIB_USING_SYSTEM_PRINTF', use_system_printf) |
| |
| if not use_system_printf |
| # gnulib has vasprintf so override the previous check |
| glib_conf.set('HAVE_VASPRINTF', 1) |
| endif |
| |
| # Check for nl_langinfo and CODESET |
| if cc.links('''#include <langinfo.h> |
| int main (int argc, char ** argv) { |
| char *codeset = nl_langinfo (CODESET); |
| (void) codeset; |
| return 0; |
| }''', name : 'nl_langinfo and CODESET') |
| glib_conf.set('HAVE_LANGINFO_CODESET', 1) |
| glib_conf.set('HAVE_CODESET', 1) |
| endif |
| |
| # Check for nl_langinfo and LC_TIME parts that are needed in gdatetime.c |
| have_langinfo_time = false |
| if cc.links('''#include <langinfo.h> |
| int main (int argc, char ** argv) { |
| char *str; |
| str = nl_langinfo (PM_STR); |
| str = nl_langinfo (D_T_FMT); |
| str = nl_langinfo (D_FMT); |
| str = nl_langinfo (T_FMT); |
| str = nl_langinfo (T_FMT_AMPM); |
| str = nl_langinfo (MON_1); |
| str = nl_langinfo (ABMON_12); |
| str = nl_langinfo (DAY_1); |
| str = nl_langinfo (ABDAY_7); |
| (void) str; |
| return 0; |
| }''', name : 'nl_langinfo (PM_STR)') |
| have_langinfo_time = true |
| glib_conf.set('HAVE_LANGINFO_TIME', 1) |
| endif |
| |
| # Linux glibc supports ERA, but FreeBSD and macOS don’t |
| if cc.links('''#include <langinfo.h> |
| int main (int argc, char **argv) { |
| char *str; |
| str = nl_langinfo (ERA); |
| str = nl_langinfo (ERA_D_T_FMT); |
| str = nl_langinfo (ERA_D_FMT); |
| str = nl_langinfo (ERA_T_FMT); |
| str = nl_langinfo (_NL_TIME_ERA_NUM_ENTRIES); |
| (void) str; |
| return 0; |
| }''', name : 'nl_langinfo (ERA)') |
| glib_conf.set('HAVE_LANGINFO_ERA', 1) |
| if not have_langinfo_time |
| error('nl_langinfo(ERA) is supported but more basic nl_langinfo() functionality like PM_STR is not') |
| endif |
| endif |
| |
| if cc.links('''#include <langinfo.h> |
| int main (int argc, char ** argv) { |
| char *str; |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT0_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT1_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT2_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT3_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT4_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT5_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT6_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT7_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT8_MB); |
| str = nl_langinfo (_NL_CTYPE_OUTDIGIT9_MB); |
| (void) str; |
| return 0; |
| }''', name : 'nl_langinfo (_NL_CTYPE_OUTDIGITn_MB)') |
| glib_conf.set('HAVE_LANGINFO_OUTDIGIT', 1) |
| if not have_langinfo_time |
| error('nl_langinfo(_NL_CTYPE_OUTDIGITn_MB) is supported but more basic nl_langinfo() functionality like PM_STR is not') |
| endif |
| endif |
| |
| # Check for nl_langinfo and alternative month names |
| if cc.links('''#ifndef _GNU_SOURCE |
| # define _GNU_SOURCE |
| #endif |
| #include <langinfo.h> |
| int main (int argc, char ** argv) { |
| char *str; |
| str = nl_langinfo (ALTMON_1); |
| str = nl_langinfo (ALTMON_2); |
| str = nl_langinfo (ALTMON_3); |
| str = nl_langinfo (ALTMON_4); |
| str = nl_langinfo (ALTMON_5); |
| str = nl_langinfo (ALTMON_6); |
| str = nl_langinfo (ALTMON_7); |
| str = nl_langinfo (ALTMON_8); |
| str = nl_langinfo (ALTMON_9); |
| str = nl_langinfo (ALTMON_10); |
| str = nl_langinfo (ALTMON_11); |
| str = nl_langinfo (ALTMON_12); |
| (void) str; |
| return 0; |
| }''', name : 'nl_langinfo (ALTMON_n)') |
| glib_conf.set('HAVE_LANGINFO_ALTMON', 1) |
| if not have_langinfo_time |
| error('nl_langinfo(ALTMON_n) is supported but more basic nl_langinfo() functionality like PM_STR is not') |
| endif |
| endif |
| |
| # Check for nl_langinfo and abbreviated alternative month names |
| if cc.links('''#ifndef _GNU_SOURCE |
| # define _GNU_SOURCE |
| #endif |
| #include <langinfo.h> |
| int main (int argc, char ** argv) { |
| char *str; |
| str = nl_langinfo (_NL_ABALTMON_1); |
| str = nl_langinfo (_NL_ABALTMON_2); |
| str = nl_langinfo (_NL_ABALTMON_3); |
| str = nl_langinfo (_NL_ABALTMON_4); |
| str = nl_langinfo (_NL_ABALTMON_5); |
| str = nl_langinfo (_NL_ABALTMON_6); |
| str = nl_langinfo (_NL_ABALTMON_7); |
| str = nl_langinfo (_NL_ABALTMON_8); |
| str = nl_langinfo (_NL_ABALTMON_9); |
| str = nl_langinfo (_NL_ABALTMON_10); |
| str = nl_langinfo (_NL_ABALTMON_11); |
| str = nl_langinfo (_NL_ABALTMON_12); |
| (void) str; |
| return 0; |
| }''', name : 'nl_langinfo (_NL_ABALTMON_n)') |
| glib_conf.set('HAVE_LANGINFO_ABALTMON', 1) |
| if not have_langinfo_time |
| error('nl_langinfo(_NL_ABALTMON_n) is supported but more basic nl_langinfo() functionality like PM_STR is not') |
| endif |
| endif |
| |
| # Check for nl_langinfo and _NL_TIME_CODESET |
| if cc.links('''#include <langinfo.h> |
| int main (int argc, char ** argv) { |
| char *codeset = nl_langinfo (_NL_TIME_CODESET); |
| (void) codeset; |
| return 0; |
| }''', name : 'nl_langinfo and _NL_TIME_CODESET') |
| glib_conf.set('HAVE_LANGINFO_TIME_CODESET', 1) |
| if not have_langinfo_time |
| error('nl_langinfo(_NL_TIME_CODESET) is supported but more basic nl_langinfo() functionality like PM_STR is not') |
| endif |
| endif |
| |
| # Check if C compiler supports the 'signed' keyword |
| if not cc.compiles('''signed char x;''', name : 'signed') |
| glib_conf.set('signed', '/* NOOP */') |
| endif |
| |
| # Check if the ptrdiff_t type exists |
| if cc.has_header_symbol('stddef.h', 'ptrdiff_t') |
| glib_conf.set('HAVE_PTRDIFF_T', 1) |
| endif |
| |
| # Check for sig_atomic_t type |
| if cc.links('''#include <signal.h> |
| #include <sys/types.h> |
| sig_atomic_t val = 42; |
| int main (int argc, char ** argv) { |
| return val == 42 ? 0 : 1; |
| }''', name : 'sig_atomic_t') |
| glib_conf.set('HAVE_SIG_ATOMIC_T', 1) |
| endif |
| |
| # Check if 'long long' works |
| # jm_AC_TYPE_LONG_LONG |
| if cc.compiles('''long long ll = 1LL; |
| int i = 63; |
| int some_func (void) { |
| long long llmax = (long long) -1; |
| return ll << i | ll >> i | llmax / ll | llmax % ll; |
| }''', name : 'long long') |
| glib_conf.set('HAVE_LONG_LONG', 1) |
| have_long_long = true |
| else |
| have_long_long = false |
| endif |
| |
| # Test whether the compiler supports the 'long double' type. |
| if cc.compiles('''/* The Stardent Vistra knows sizeof(long double), but does not support it. */ |
| long double foo = 0.0; |
| /* On Ultrix 4.3 cc, long double is 4 and double is 8. */ |
| int array [2*(sizeof(long double) >= sizeof(double)) - 1];''', |
| name : 'long double') |
| glib_conf.set('HAVE_LONG_DOUBLE', 1) |
| endif |
| |
| # Test whether <stddef.h> has the 'wchar_t' type. |
| if cc.has_header_symbol('stddef.h', 'wchar_t') |
| glib_conf.set('HAVE_WCHAR_T', 1) |
| endif |
| |
| # Test whether <wchar.h> has the 'wint_t' type. |
| if cc.has_header_symbol('wchar.h', 'wint_t') |
| glib_conf.set('HAVE_WINT_T', 1) |
| endif |
| |
| found_uintmax_t = false |
| |
| # Define HAVE_INTTYPES_H_WITH_UINTMAX if <inttypes.h> exists, |
| # doesn't clash with <sys/types.h>, and declares uintmax_t. |
| # jm_AC_HEADER_INTTYPES_H |
| if cc.compiles('''#include <sys/types.h> |
| #include <inttypes.h> |
| void some_func (void) { |
| uintmax_t i = (uintmax_t) -1; |
| (void) i; |
| }''', name : 'uintmax_t in inttypes.h') |
| glib_conf.set('HAVE_INTTYPES_H_WITH_UINTMAX', 1) |
| found_uintmax_t = true |
| endif |
| |
| # Define HAVE_STDINT_H_WITH_UINTMAX if <stdint.h> exists, |
| # doesn't clash with <sys/types.h>, and declares uintmax_t. |
| # jm_AC_HEADER_STDINT_H |
| if cc.compiles('''#include <sys/types.h> |
| #include <stdint.h> |
| void some_func (void) { |
| uintmax_t i = (uintmax_t) -1; |
| (void) i; |
| }''', name : 'uintmax_t in stdint.h') |
| glib_conf.set('HAVE_STDINT_H_WITH_UINTMAX', 1) |
| found_uintmax_t = true |
| endif |
| |
| # Define intmax_t to 'long' or 'long long' |
| # if it is not already defined in <stdint.h> or <inttypes.h>. |
| # For simplicity, we assume that a header file defines 'intmax_t' if and |
| # only if it defines 'uintmax_t'. |
| if found_uintmax_t |
| glib_conf.set('HAVE_INTMAX_T', 1) |
| elif have_long_long |
| glib_conf.set('intmax_t', 'long long') |
| else |
| glib_conf.set('intmax_t', 'long') |
| endif |
| |
| char_size = cc.sizeof('char') |
| short_size = cc.sizeof('short') |
| int_size = cc.sizeof('int') |
| voidp_size = cc.sizeof('void*') |
| long_size = cc.sizeof('long') |
| if have_long_long |
| long_long_size = cc.sizeof('long long') |
| else |
| long_long_size = 0 |
| endif |
| sizet_size = cc.sizeof('size_t') |
| if cc.get_id() == 'msvc' or cc.get_id() == 'clang-cl' |
| ssizet_size = cc.sizeof('SSIZE_T', prefix : '#include <BaseTsd.h>') |
| else |
| ssizet_size = cc.sizeof('ssize_t', prefix : '#include <unistd.h>') |
| endif |
| |
| # Some platforms (Apple) hard-code int64_t to long long instead of |
| # using long on 64-bit architectures. This can cause type mismatch |
| # warnings when trying to interface with code using the standard |
| # library type. Test for the warnings and set gint64 to whichever |
| # works. |
| if long_long_size == long_size |
| if cc.compiles('''#if defined(_AIX) && !defined(__GNUC__) |
| #pragma options langlvl=stdc99 |
| #endif |
| #pragma GCC diagnostic error "-Wincompatible-pointer-types" |
| #include <stdint.h> |
| #include <stdio.h> |
| int main () { |
| int64_t i1 = 1; |
| long *i2 = &i1; |
| (void) i2; |
| return 1; |
| }''', name : 'int64_t is long') |
| int64_t_typedef = 'long' |
| elif cc.compiles('''#if defined(_AIX) && !defined(__GNUC__) |
| #pragma options langlvl=stdc99 |
| #endif |
| #pragma GCC diagnostic error "-Wincompatible-pointer-types" |
| #include <stdint.h> |
| #include <stdio.h> |
| int main () { |
| int64_t i1 = 1; |
| long long *i2 = &i1; |
| (void) i2; |
| return 1; |
| }''', name : 'int64_t is long long') |
| int64_t_typedef = 'long long' |
| else |
| error('Cannot detect int64_t typedef') |
| endif |
| endif |
| |
| int64_m = 'll' |
| char_align = cc.alignment('char') |
| short_align = cc.alignment('short') |
| int_align = cc.alignment('int') |
| voidp_align = cc.alignment('void*') |
| long_align = cc.alignment('long') |
| long_long_align = cc.alignment('long long') |
| # NOTE: We don't check for size of __int64 because long long is guaranteed to |
| # be 64-bit in C99, and it is available on all supported compilers |
| sizet_align = cc.alignment('size_t') |
| |
| glib_conf.set('SIZEOF_CHAR', char_size) |
| glib_conf.set('SIZEOF_INT', int_size) |
| glib_conf.set('SIZEOF_SHORT', short_size) |
| glib_conf.set('SIZEOF_LONG', long_size) |
| glib_conf.set('SIZEOF_LONG_LONG', long_long_size) |
| glib_conf.set('SIZEOF_SIZE_T', sizet_size) |
| glib_conf.set('SIZEOF_SSIZE_T', ssizet_size) |
| glib_conf.set('SIZEOF_VOID_P', voidp_size) |
| glib_conf.set('SIZEOF_WCHAR_T', cc.sizeof('wchar_t', prefix: '#include <stddef.h>')) |
| |
| if short_size == 2 |
| gint16 = 'short' |
| gint16_modifier='h' |
| gint16_format='hi' |
| guint16_format='hu' |
| elif int_size == 2 |
| gint16 = 'int' |
| gint16_modifier='' |
| gint16_format='i' |
| guint16_format='u' |
| else |
| error('Compiler provides no native 16-bit integer type') |
| endif |
| glibconfig_conf.set('gint16', gint16) |
| glibconfig_conf.set_quoted('gint16_modifier', gint16_modifier) |
| glibconfig_conf.set_quoted('gint16_format', gint16_format) |
| glibconfig_conf.set_quoted('guint16_format', guint16_format) |
| |
| if short_size == 4 |
| gint32 = 'short' |
| gint32_modifier='h' |
| gint32_format='hi' |
| guint32_format='hu' |
| guint32_align = short_align |
| elif int_size == 4 |
| gint32 = 'int' |
| gint32_modifier='' |
| gint32_format='i' |
| guint32_format='u' |
| guint32_align = int_align |
| elif long_size == 4 |
| gint32 = 'long' |
| gint32_modifier='l' |
| gint32_format='li' |
| guint32_format='lu' |
| guint32_align = long_align |
| else |
| error('Compiler provides no native 32-bit integer type') |
| endif |
| glibconfig_conf.set('gint32', gint32) |
| glibconfig_conf.set_quoted('gint32_modifier', gint32_modifier) |
| glibconfig_conf.set_quoted('gint32_format', gint32_format) |
| glibconfig_conf.set_quoted('guint32_format', guint32_format) |
| |
| if int_size == 8 |
| gint64 = 'int' |
| gint64_modifier='' |
| gint64_format='i' |
| guint64_format='u' |
| glib_extension='' |
| gint64_constant='(val)' |
| guint64_constant='(val)' |
| guint64_align = int_align |
| elif long_size == 8 and (long_long_size != long_size or int64_t_typedef == 'long') |
| gint64 = 'long' |
| glib_extension='' |
| gint64_modifier='l' |
| gint64_format='li' |
| guint64_format='lu' |
| gint64_constant='(val##L)' |
| guint64_constant='(val##UL)' |
| guint64_align = long_align |
| elif long_long_size == 8 and (long_long_size != long_size or int64_t_typedef == 'long long') |
| gint64 = 'long long' |
| glib_extension='G_GNUC_EXTENSION ' |
| gint64_modifier=int64_m |
| gint64_format=int64_m + 'i' |
| guint64_format=int64_m + 'u' |
| gint64_constant='(G_GNUC_EXTENSION (val##LL))' |
| guint64_constant='(G_GNUC_EXTENSION (val##ULL))' |
| guint64_align = long_long_align |
| else |
| error('Compiler provides no native 64-bit integer type') |
| endif |
| glibconfig_conf.set('glib_extension', glib_extension) |
| glibconfig_conf.set('gint64', gint64) |
| glibconfig_conf.set_quoted('gint64_modifier', gint64_modifier) |
| glibconfig_conf.set_quoted('gint64_format', gint64_format) |
| glibconfig_conf.set_quoted('guint64_format', guint64_format) |
| glibconfig_conf.set('gint64_constant', gint64_constant) |
| glibconfig_conf.set('guint64_constant', guint64_constant) |
| |
| if host_system == 'windows' |
| glibconfig_conf.set('g_pid_type', 'void*') |
| glibconfig_conf.set_quoted('g_pid_format', 'p') |
| if host_machine.cpu_family() == 'x86_64' |
| glibconfig_conf.set_quoted('g_pollfd_format', '%#' + int64_m + 'x') |
| else |
| glibconfig_conf.set_quoted('g_pollfd_format', '%#x') |
| endif |
| glibconfig_conf.set('g_dir_separator', '\\\\') |
| glibconfig_conf.set('g_searchpath_separator', ';') |
| else |
| glibconfig_conf.set('g_pid_type', 'int') |
| glibconfig_conf.set_quoted('g_pid_format', 'i') |
| glibconfig_conf.set_quoted('g_pollfd_format', '%d') |
| glibconfig_conf.set('g_dir_separator', '/') |
| glibconfig_conf.set('g_searchpath_separator', ':') |
| endif |
| |
| g_sizet_compatibility = { |
| 'short': sizet_size == short_size, |
| 'int': sizet_size == int_size, |
| 'long': sizet_size == long_size, |
| 'long long': sizet_size == long_long_size, |
| } |
| |
| # Do separate checks for gcc/clang (and ignore other compilers for now), since |
| # we need to explicitly pass -Werror to the compilers. |
| # FIXME: https://github.com/mesonbuild/meson/issues/5399 |
| if cc.get_id() == 'gcc' or cc.get_id() == 'clang' |
| foreach type_name, size_compatibility : g_sizet_compatibility |
| g_sizet_compatibility += { type_name: size_compatibility and |
| cc.compiles( |
| '''#include <stddef.h> |
| static size_t f (size_t *i) { return *i + 1; } |
| int main (void) { |
| unsigned ''' + type_name + ''' i = 0; |
| f (&i); |
| return 0; |
| }''', |
| args: ['-Werror'], |
| name : 'GCC size_t typedef is ' + type_name), } |
| endforeach |
| endif |
| |
| if g_sizet_compatibility['short'] |
| glibconfig_conf.set('glib_size_type_define', 'short') |
| glibconfig_conf.set_quoted('gsize_modifier', 'h') |
| glibconfig_conf.set_quoted('gssize_modifier', 'h') |
| glibconfig_conf.set_quoted('gsize_format', 'hu') |
| glibconfig_conf.set_quoted('gssize_format', 'hi') |
| glibconfig_conf.set('glib_msize_type', 'SHRT') |
| elif g_sizet_compatibility['int'] |
| glibconfig_conf.set('glib_size_type_define', 'int') |
| glibconfig_conf.set_quoted('gsize_modifier', '') |
| glibconfig_conf.set_quoted('gssize_modifier', '') |
| glibconfig_conf.set_quoted('gsize_format', 'u') |
| glibconfig_conf.set_quoted('gssize_format', 'i') |
| glibconfig_conf.set('glib_msize_type', 'INT') |
| elif g_sizet_compatibility['long'] |
| glibconfig_conf.set('glib_size_type_define', 'long') |
| glibconfig_conf.set_quoted('gsize_modifier', 'l') |
| glibconfig_conf.set_quoted('gssize_modifier', 'l') |
| glibconfig_conf.set_quoted('gsize_format', 'lu') |
| glibconfig_conf.set_quoted('gssize_format', 'li') |
| glibconfig_conf.set('glib_msize_type', 'LONG') |
| elif g_sizet_compatibility['long long'] |
| glibconfig_conf.set('glib_size_type_define', 'long long') |
| glibconfig_conf.set_quoted('gsize_modifier', int64_m) |
| glibconfig_conf.set_quoted('gssize_modifier', int64_m) |
| glibconfig_conf.set_quoted('gsize_format', int64_m + 'u') |
| glibconfig_conf.set_quoted('gssize_format', int64_m + 'i') |
| glibconfig_conf.set('glib_msize_type', 'INT64') |
| else |
| error('Could not determine size of size_t.') |
| endif |
| |
| if voidp_size == int_size |
| glibconfig_conf.set('glib_intptr_type_define', 'int') |
| glibconfig_conf.set_quoted('gintptr_modifier', '') |
| glibconfig_conf.set_quoted('gintptr_format', 'i') |
| glibconfig_conf.set_quoted('guintptr_format', 'u') |
| glibconfig_conf.set('glib_gpi_cast', '(gint)') |
| glibconfig_conf.set('glib_gpui_cast', '(guint)') |
| elif voidp_size == long_size |
| glibconfig_conf.set('glib_intptr_type_define', 'long') |
| glibconfig_conf.set_quoted('gintptr_modifier', 'l') |
| glibconfig_conf.set_quoted('gintptr_format', 'li') |
| glibconfig_conf.set_quoted('guintptr_format', 'lu') |
| glibconfig_conf.set('glib_gpi_cast', '(glong)') |
| glibconfig_conf.set('glib_gpui_cast', '(gulong)') |
| elif voidp_size == long_long_size |
| glibconfig_conf.set('glib_intptr_type_define', 'long long') |
| glibconfig_conf.set_quoted('gintptr_modifier', int64_m) |
| glibconfig_conf.set_quoted('gintptr_format', int64_m + 'i') |
| glibconfig_conf.set_quoted('guintptr_format', int64_m + 'u') |
| glibconfig_conf.set('glib_gpi_cast', '(gint64)') |
| glibconfig_conf.set('glib_gpui_cast', '(guint64)') |
| else |
| error('Could not determine size of void *') |
| endif |
| |
| if long_size != 8 and long_long_size != 8 and int_size != 8 |
| error('GLib requires a 64-bit type. You might want to consider using the GNU C compiler.') |
| endif |
| |
| glibconfig_conf.set('gintbits', int_size * 8) |
| glibconfig_conf.set('glongbits', long_size * 8) |
| glibconfig_conf.set('gsizebits', sizet_size * 8) |
| glibconfig_conf.set('gssizebits', ssizet_size * 8) |
| |
| # XXX: https://gitlab.gnome.org/GNOME/glib/issues/1413 |
| if host_system == 'windows' |
| g_module_suffix = 'dll' |
| else |
| g_module_suffix = 'so' |
| endif |
| glibconfig_conf.set('g_module_suffix', g_module_suffix) |
| |
| glibconfig_conf.set('GLIB_MAJOR_VERSION', major_version) |
| glibconfig_conf.set('GLIB_MINOR_VERSION', minor_version) |
| glibconfig_conf.set('GLIB_MICRO_VERSION', micro_version) |
| glibconfig_conf.set('GLIB_VERSION', glib_version) |
| |
| glibconfig_conf.set('glib_void_p', voidp_size) |
| glibconfig_conf.set('glib_long', long_size) |
| glibconfig_conf.set('glib_size_t', sizet_size) |
| glibconfig_conf.set('glib_ssize_t', ssizet_size) |
| if host_machine.endian() == 'big' |
| glibconfig_conf.set('g_byte_order', 'G_BIG_ENDIAN') |
| glibconfig_conf.set('g_bs_native', 'BE') |
| glibconfig_conf.set('g_bs_alien', 'LE') |
| else |
| glibconfig_conf.set('g_byte_order', 'G_LITTLE_ENDIAN') |
| glibconfig_conf.set('g_bs_native', 'LE') |
| glibconfig_conf.set('g_bs_alien', 'BE') |
| endif |
| |
| # === va_copy checks === |
| |
| glib_vacopy = '' |
| |
| # We check for G_VA_COPY_AS_ARRAY for historical reasons, but we no longer |
| # use it: use Standard C va_copy() instead. |
| |
| va_list_val_copy_prog = ''' |
| #include <stdarg.h> |
| #include <stdlib.h> |
| void f (int i, ...) { |
| va_list args1, args2; |
| va_start (args1, i); |
| args2 = args1; |
| if (va_arg (args2, int) != 42 || va_arg (args1, int) != 42) |
| exit (1); |
| va_end (args1); va_end (args2); |
| } |
| int main() { |
| f (0, 42); |
| return 0; |
| }''' |
| |
| if cc_can_run |
| rres = cc.run(va_list_val_copy_prog, name : 'va_lists can be copied as values') |
| glib_va_val_copy = rres.compiled() and rres.returncode() == 0 |
| else |
| glib_va_val_copy = meson.get_external_property('va_val_copy', true) |
| endif |
| if not glib_va_val_copy |
| glib_vacopy = glib_vacopy + '\n#define G_VA_COPY_AS_ARRAY 1' |
| glib_conf.set('G_VA_COPY_AS_ARRAY', 1) |
| endif |
| glibconfig_conf.set('glib_vacopy', glib_vacopy) |
| |
| # check for flavours of varargs macros |
| g_have_iso_c_varargs = cc.compiles(''' |
| void some_func (void) { |
| int a(int p1, int p2, int p3); |
| #define call_a(...) a(1,__VA_ARGS__) |
| call_a(2,3); |
| }''', name : 'ISO C99 varargs macros in C') |
| |
| if not g_have_iso_c_varargs |
| error('GLib requires a C compiler with support for C99 __VA_ARG__ in macros.') |
| endif |
| |
| if have_cxx |
| g_have_iso_cxx_varargs = cxx.compiles(''' |
| void some_func (void) { |
| int a(int p1, int p2, int p3); |
| #define call_a(...) a(1,__VA_ARGS__) |
| call_a(2,3); |
| }''', name : 'ISO C99 varargs macros in C++') |
| |
| if not g_have_iso_cxx_varargs |
| error('GLib requires a C++ compiler with support for C99 __VA_ARG__ in macros.') |
| endif |
| endif |
| |
| g_have_gnuc_varargs = cc.compiles(''' |
| void some_func (void) { |
| int a(int p1, int p2, int p3); |
| #define call_a(params...) a(1,params) |
| call_a(2,3); |
| }''', name : 'GNUC varargs macros') |
| |
| if cc.has_header('alloca.h') |
| glibconfig_conf.set('GLIB_HAVE_ALLOCA_H', true) |
| endif |
| has_syspoll = cc.has_header('sys/poll.h') |
| has_systypes = cc.has_header('sys/types.h') |
| if has_syspoll |
| glibconfig_conf.set('GLIB_HAVE_SYS_POLL_H', true) |
| endif |
| has_winsock2 = cc.has_header('winsock2.h') |
| |
| if has_syspoll and has_systypes |
| poll_includes = ''' |
| #include<sys/poll.h> |
| #include<sys/types.h>''' |
| elif has_winsock2 |
| poll_includes = ''' |
| #define _WIN32_WINNT @0@ |
| #include <winsock2.h>'''.format(glib_conf.get('_WIN32_WINNT')) |
| else |
| # FIXME? |
| error('FIX POLL* defines') |
| endif |
| |
| poll_defines = [ |
| [ 'POLLIN', 'g_pollin', 1 ], |
| [ 'POLLOUT', 'g_pollout', 4 ], |
| [ 'POLLPRI', 'g_pollpri', 2 ], |
| [ 'POLLERR', 'g_pollerr', 8 ], |
| [ 'POLLHUP', 'g_pollhup', 16 ], |
| [ 'POLLNVAL', 'g_pollnval', 32 ], |
| ] |
| |
| if has_syspoll and has_systypes |
| foreach d : poll_defines |
| val = cc.compute_int(d[0], prefix: poll_includes) |
| glibconfig_conf.set(d[1], val) |
| endforeach |
| elif has_winsock2 |
| # Due to a missed bug in configure.ac the poll test |
| # never succeeded on Windows and used some pre-defined |
| # values as a fallback. Keep using them to maintain |
| # ABI compatibility with autotools builds of glibs |
| # and with *any* glib-using code compiled against them, |
| # since these values end up in a public header glibconfig.h. |
| foreach d : poll_defines |
| glibconfig_conf.set(d[1], d[2]) |
| endforeach |
| endif |
| |
| # Internet address families |
| # FIXME: what about Cygwin (G_WITH_CYGWIN) |
| if host_system == 'windows' |
| inet_includes = ''' |
| #include <winsock2.h>''' |
| else |
| inet_includes = ''' |
| #include <sys/types.h> |
| #include <sys/socket.h>''' |
| endif |
| |
| inet_defines = [ |
| [ 'AF_UNIX', 'g_af_unix' ], |
| [ 'AF_INET', 'g_af_inet' ], |
| [ 'AF_INET6', 'g_af_inet6' ], |
| [ 'MSG_OOB', 'g_msg_oob' ], |
| [ 'MSG_PEEK', 'g_msg_peek' ], |
| [ 'MSG_DONTROUTE', 'g_msg_dontroute' ], |
| ] |
| foreach d : inet_defines |
| val = cc.compute_int(d[0], prefix: inet_includes) |
| glibconfig_conf.set(d[1], val) |
| endforeach |
| |
| if host_system == 'windows' |
| have_ipv6 = true |
| else |
| have_ipv6 = cc.has_type('struct in6_addr', prefix: '#include <netinet/in.h>') |
| endif |
| glib_conf.set('HAVE_IPV6', have_ipv6) |
| |
| # We need to decide at configure time if GLib will use real atomic |
| # operations ("lock free") or emulated ones with a mutex. This is |
| # because we must put this information in glibconfig.h so we know if |
| # it is safe or not to inline using compiler intrinsics directly from |
| # the header. |
| # |
| # We also publish the information via G_ATOMIC_LOCK_FREE in case the |
| # user is interested in knowing if they can use the atomic ops across |
| # processes. |
| # |
| # We can currently support the atomic ops natively when building GLib |
| # with recent versions of GCC or MSVC. |
| # |
| # Note that the atomic ops are only available with GCC on x86 when |
| # using -march=i486 or higher. If we detect that the atomic ops are |
| # not available but would be available given the right flags, we want |
| # to abort and advise the user to fix their CFLAGS. It's better to do |
| # that then to silently fall back on emulated atomic ops just because |
| # the user had the wrong build environment. |
| atomictest = '''int main() { |
| int atomic = 2; |
| __sync_bool_compare_and_swap (&atomic, 2, 3); |
| return 0; |
| } |
| ''' |
| |
| atomicdefine = ''' |
| #ifndef __GCC_HAVE_SYNC_COMPARE_AND_SWAP_4 |
| #error "compiler has atomic ops, but doesn't define __GCC_HAVE_SYNC_COMPARE_AND_SWAP_4" |
| #endif |
| ''' |
| |
| # We know that we can always use real ("lock free") atomic operations with MSVC |
| if cc.get_id() == 'msvc' or cc.get_id() == 'clang-cl' or cc.links(atomictest, name : 'atomic ops') |
| have_atomic_lock_free = true |
| if cc.get_id() == 'gcc' and not cc.compiles(atomicdefine, name : 'atomic ops define') |
| # Old gcc release may provide |
| # __sync_bool_compare_and_swap but doesn't define |
| # __GCC_HAVE_SYNC_COMPARE_AND_SWAP_4 |
| glib_conf.set('__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4', true) |
| endif |
| |
| if cc.get_id() == 'gcc' or cc.get_id() == 'clang' |
| sync_swap_test = ''' |
| int main() { |
| int atomic = 2; |
| __sync_swap (&atomic, 2); |
| return 0; |
| } |
| ''' |
| |
| glib_conf.set('_GLIB_GCC_HAVE_SYNC_SWAP', cc.links(sync_swap_test, name : 'sync swap')) |
| endif |
| else |
| have_atomic_lock_free = false |
| if host_machine.cpu_family() == 'x86' and cc.links(atomictest, args : '-march=i486') |
| error('GLib must be built with -march=i486 or later.') |
| endif |
| endif |
| glibconfig_conf.set('G_ATOMIC_LOCK_FREE', have_atomic_lock_free) |
| |
| # === Threads === |
| |
| if get_option('force_posix_threads') |
| warning('DEPRECATION: Option \'force_posix_threads\' is deprecated and will be removed after GLib 2.72; please file an issue with your use case if you still require it') |
| endif |
| |
| # Determination of thread implementation |
| if host_system == 'windows' and not get_option('force_posix_threads') |
| thread_dep = [] |
| threads_implementation = 'win32' |
| glibconfig_conf.set('g_threads_impl_def', 'WIN32') |
| glib_conf.set('THREADS_WIN32', 1) |
| else |
| thread_dep = dependency('threads') |
| threads_implementation = 'posix' |
| pthread_prefix = ''' |
| #ifndef _GNU_SOURCE |
| # define _GNU_SOURCE |
| #endif |
| #include <pthread.h>''' |
| glibconfig_conf.set('g_threads_impl_def', 'POSIX') |
| glib_conf.set('THREADS_POSIX', 1) |
| if cc.has_header_symbol('pthread.h', 'pthread_attr_setstacksize') |
| glib_conf.set('HAVE_PTHREAD_ATTR_SETSTACKSIZE', 1) |
| endif |
| if cc.has_header_symbol('pthread.h', 'pthread_attr_setinheritsched') |
| glib_conf.set('HAVE_PTHREAD_ATTR_SETINHERITSCHED', 1) |
| endif |
| if cc.has_header_symbol('pthread.h', 'pthread_condattr_setclock') |
| glib_conf.set('HAVE_PTHREAD_CONDATTR_SETCLOCK', 1) |
| endif |
| if cc.has_header_symbol('pthread.h', 'pthread_cond_timedwait_relative_np') |
| glib_conf.set('HAVE_PTHREAD_COND_TIMEDWAIT_RELATIVE_NP', 1) |
| endif |
| if cc.has_header_symbol('pthread.h', 'pthread_getname_np', prefix : pthread_prefix) |
| glib_conf.set('HAVE_PTHREAD_GETNAME_NP', 1) |
| endif |
| if cc.has_header_symbol('pthread.h', 'pthread_getaffinity_np', prefix : pthread_prefix) |
| glib_conf.set('HAVE_PTHREAD_GETAFFINITY_NP', 1) |
| endif |
| |
| # Assume that pthread_setname_np is available in some form; same as configure |
| if cc.links(pthread_prefix + ''' |
| int main() { |
| pthread_setname_np("example"); |
| return 0; |
| }''', |
| name : 'pthread_setname_np(const char*)', |
| dependencies : thread_dep) |
| # macOS and iOS |
| glib_conf.set('HAVE_PTHREAD_SETNAME_NP_WITHOUT_TID', 1) |
| elif cc.links(pthread_prefix + ''' |
| int main() { |
| pthread_setname_np(pthread_self(), "example"); |
| return 0; |
| }''', |
| name : 'pthread_setname_np(pthread_t, const char*)', |
| dependencies : thread_dep) |
| # Linux, Solaris, etc. |
| glib_conf.set('HAVE_PTHREAD_SETNAME_NP_WITH_TID', 1) |
| elif cc.links(pthread_prefix + ''' |
| int main() { |
| pthread_setname_np(pthread_self(), "%s", "example"); |
| return 0; |
| }''', |
| name : 'pthread_setname_np(pthread_t, const char*, void*)', |
| dependencies : thread_dep) |
| # NetBSD |
| glib_conf.set('HAVE_PTHREAD_SETNAME_NP_WITH_TID_AND_ARG', 1) |
| elif cc.links(pthread_prefix + ''' |
| #include <pthread_np.h> |
| int main() { |
| pthread_set_name_np(pthread_self(), "example"); |
| return 0; |
| }''', |
| name : 'pthread_set_name_np(pthread_t, const char*)', |
| dependencies : thread_dep) |
| # FreeBSD, DragonFlyBSD, OpenBSD, etc. |
| glib_conf.set('HAVE_PTHREAD_SET_NAME_NP', 1) |
| endif |
| endif |
| |
| # FIXME: we should make it print the result and always return 0, so that |
| # the output in meson shows up as green |
| # volatile is needed here to avoid optimisations in the test |
| stack_grows_check_prog = ''' |
| volatile int *a = 0, *b = 0; |
| void f (int i) { |
| volatile int x = 5; |
| if (i == 0) |
| b = &x; |
| else |
| f (i - 1); |
| } |
| int main () { |
| volatile int y = 7; |
| a = &y; |
| f (100); |
| return b > a ? 0 : 1; |
| }''' |
| |
| if cc_can_run |
| rres = cc.run(stack_grows_check_prog, name : 'stack grows check') |
| growing_stack = rres.compiled() and rres.returncode() == 0 |
| else |
| growing_stack = meson.get_external_property('growing_stack', false) |
| endif |
| |
| glibconfig_conf.set10('G_HAVE_GROWING_STACK', growing_stack) |
| |
| # Tests for iconv |
| # |
| # We should never use the MinGW C library's iconv because it may not be |
| # available in the actual runtime environment. On Windows, we always use |
| # the built-in implementation |
| if host_system == 'windows' |
| # We have a #include "win_iconv.c" in gconvert.c on Windows, so we don't need |
| # any external library for it |
| libiconv = [] |
| else |
| libiconv = dependency('iconv') |
| endif |
| |
| pcre2_req = '>=10.32' |
| |
| # Pick up pcre from the system, or if "--force-fallback-for libpcre2-8" was specified |
| pcre2 = dependency('libpcre2-8', version: pcre2_req, required: false, default_options: ['default_library=static']) |
| if not pcre2.found() |
| if cc.get_id() == 'msvc' or cc.get_id() == 'clang-cl' |
| # MSVC: Search for the PCRE2 library by the configuration, which corresponds |
| # to the output of CMake builds of PCRE2. Note that debugoptimized |
| # is really a Release build with .PDB files. |
| if vs_crt == 'debug' |
| pcre2 = cc.find_library('pcre2d-8', required : false) |
| else |
| pcre2 = cc.find_library('pcre2-8', required : false) |
| endif |
| endif |
| endif |
| |
| # Try again with the fallback |
| if not pcre2.found() |
| pcre2 = dependency('libpcre2-8', version: pcre2_req, allow_fallback: true, default_options: ['default_library=static']) |
| assert(pcre2.type_name() == 'internal') |
| # static flags are automatically enabled by the subproject if it's built |
| # with default_library=static |
| use_pcre2_static_flag = false |
| elif host_system == 'windows' and pcre2.type_name() != 'internal' |
| pcre2_static = cc.links('''#define PCRE2_STATIC |
| #define PCRE2_CODE_UNIT_WIDTH 8 |
| #include <pcre2.h> |
| int main() { |
| void *p = NULL; |
| pcre2_code_free(p); |
| return 0; |
| }''', |
| dependencies: pcre2, |
| name : 'Windows system PCRE2 is a static build') |
| use_pcre2_static_flag = pcre2_static |
| else |
| use_pcre2_static_flag = false |
| endif |
| |
| # Import the gvdb sources as a subproject to avoid having the copylib in-tree |
| subproject('gvdb') |
| gvdb_dep = dependency('gvdb') |
| |
| libm = cc.find_library('m', required : false) |
| libffi_dep = dependency('libffi', version : '>= 3.0.0') |
| |
| libz_dep = dependency('zlib') |
| |
| # First check in libc, fallback to libintl, and as last chance build |
| # proxy-libintl subproject. |
| # FIXME: glib-gettext.m4 has much more checks to detect broken/uncompatible |
| # implementations. This could be extended if issues are found in some platforms. |
| libintl_deps = [] |
| libintl_prefix = '#include <libintl.h>' |
| libintl = dependency('intl', required: false) |
| if libintl.found() and libintl.type_name() != 'internal' |
| # libintl supports different threading APIs, which may not |
| # require additional flags, but it defaults to using pthreads if |
| # found. Meson's "threads" dependency does not allow you to |
| # prefer pthreads. We may not be using pthreads for glib itself |
| # either so just link the library to satisfy libintl rather than |
| # also defining the macros with the -pthread flag. |
| # |
| # Meson's builtin dependency lookup as of 0.60.0 doesn't check for |
| # pthread, so we do this manually here. |
| if cc.has_function('ngettext', dependencies : libintl, prefix: libintl_prefix) |
| libintl_deps += [libintl] |
| else |
| libintl_iconv = cc.find_library('iconv', required : false) |
| if libintl_iconv.found() and cc.has_function('ngettext', args : osx_ldflags, dependencies : [libintl, libintl_iconv]) |
| libintl_deps += [libintl, libintl_iconv] |
| else |
| libintl_pthread = cc.find_library('pthread', required : false) |
| if libintl_pthread.found() and cc.has_function('ngettext', dependencies : [libintl, libintl_pthread], prefix: libintl_prefix) |
| libintl_deps += [libintl, libintl_pthread] |
| else |
| libintl = disabler() |
| endif |
| endif |
| endif |
| endif |
| |
| if libintl.found() and libintl.type_name() != 'internal' |
| have_bind_textdomain_codeset = cc.has_function('bind_textdomain_codeset', dependencies: libintl_deps, prefix: libintl_prefix) |
| else |
| # using proxy-libintl fallback |
| libintl = dependency('intl', allow_fallback: true) |
| assert(libintl.type_name() == 'internal') |
| libintl_deps = [libintl] |
| have_bind_textdomain_codeset = true # proxy-libintl supports it |
| endif |
| |
| glib_conf.set('HAVE_BIND_TEXTDOMAIN_CODESET', have_bind_textdomain_codeset) |
| |
| # We require gettext to always be present |
| glib_conf.set('HAVE_DCGETTEXT', 1) |
| glib_conf.set('HAVE_GETTEXT', 1) |
| |
| glib_conf.set_quoted('GLIB_LOCALE_DIR', join_paths(glib_datadir, 'locale')) |
| |
| glib_conf.set_quoted('GLIB_LOCALSTATEDIR', glib_localstatedir) |
| glib_conf.set_quoted('GLIB_RUNSTATEDIR', glib_runstatedir) |
| |
| # libmount is only used by gio, but we need to fetch the libs to generate the |
| # pkg-config file below |
| libmount_dep = [] |
| if host_system == 'linux' |
| libmount_dep = dependency('mount', version : '>=2.23', required : get_option('libmount')) |
| glib_conf.set('HAVE_LIBMOUNT', libmount_dep.found()) |
| |
| if libmount_dep.found() and cc.has_function('mnt_monitor_veil_kernel', dependencies: libmount_dep) |
| glib_conf.set('HAVE_MNT_MONITOR_VEIL_KERNEL', 1) |
| endif |
| endif |
| |
| if host_system == 'windows' |
| winsock2 = cc.find_library('ws2_32') |
| else |
| winsock2 = not_found |
| endif |
| |
| selinux_dep = [] |
| if host_system == 'linux' |
| selinux_dep = dependency('libselinux', version: '>=2.2', required: get_option('selinux')) |
| |
| glib_conf.set('HAVE_SELINUX', selinux_dep.found()) |
| endif |
| |
| xattr_dep = [] |
| if host_system != 'windows' and get_option('xattr') |
| # either glibc or libattr can provide xattr support |
| # for both of them, we check for getxattr being in |
| # the library and a valid xattr header. |
| |
| # try glibc |
| if cc.has_function('getxattr') and cc.has_header('sys/xattr.h') |
| glib_conf.set('HAVE_SYS_XATTR_H', 1) |
| glib_conf_prefix = glib_conf_prefix + '#define @0@ 1\n'.format('HAVE_SYS_XATTR_H') |
| #failure. try libattr |
| elif cc.has_header_symbol('attr/xattr.h', 'getxattr') |
| glib_conf.set('HAVE_ATTR_XATTR_H', 1) |
| glib_conf_prefix = glib_conf_prefix + '#define @0@ 1\n'.format('HAVE_ATTR_XATTR_H') |
| xattr_dep = [cc.find_library('xattr')] |
| else |
| error('No getxattr implementation found in C library or libxattr') |
| endif |
| |
| glib_conf.set('HAVE_XATTR', 1) |
| if cc.compiles(glib_conf_prefix + ''' |
| #include <stdio.h> |
| #ifdef HAVE_SYS_TYPES_H |
| #include <sys/types.h> |
| #endif |
| #ifdef HAVE_SYS_XATTR_H |
| #include <sys/xattr.h> |
| #elif HAVE_ATTR_XATTR_H |
| #include <attr/xattr.h> |
| #endif |
| |
| int main (void) { |
| ssize_t len = getxattr("", "", NULL, 0, 0, XATTR_NOFOLLOW); |
| return len; |
| }''', |
| name : 'XATTR_NOFOLLOW') |
| glib_conf.set('HAVE_XATTR_NOFOLLOW', 1) |
| endif |
| endif |
| |
| # If strlcpy is present (BSD and similar), check that it conforms to the BSD |
| # specification. Specifically Solaris 8's strlcpy() does not, see |
| # https://bugzilla.gnome.org/show_bug.cgi?id=53933 for further context. |
| if cc.has_function('strlcpy') |
| if cc_can_run |
| rres = cc.run('''#include <stdlib.h> |
| #include <string.h> |
| int main() { |
| char p[10]; |
| (void) strlcpy (p, "hi", 10); |
| if (strlcat (p, "bye", 0) != 3) |
| return 1; |
| return 0; |
| }''', |
| name : 'OpenBSD strlcpy/strlcat') |
| if rres.compiled() and rres.returncode() == 0 |
| glib_conf.set('HAVE_STRLCPY', 1) |
| endif |
| elif meson.get_external_property('have_strlcpy', false) |
| glib_conf.set('HAVE_STRLCPY', 1) |
| endif |
| endif |
| |
| cmdline_test_code = ''' |
| #include <fcntl.h> |
| #include <sys/stat.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <unistd.h> |
| |
| static int |
| __getcmdline (void) |
| { |
| /* This code is a dumbed-down version of g_file_get_contents() */ |
| #ifndef O_BINARY |
| #define O_BINARY 0 |
| #endif |
| #define BUFSIZE 1024 |
| char result[BUFSIZE]; |
| struct stat stat_buf; |
| |
| int fd = open ("/proc/self/cmdline", O_RDONLY|O_BINARY); |
| if (fd < 0) |
| exit (1); |
| if (fstat (fd, &stat_buf)) |
| exit (1); |
| |
| if (stat_buf.st_size > 0 && S_ISREG (stat_buf.st_mode)) |
| { |
| if (read (fd, result, BUFSIZE) <= 0) |
| exit (1); |
| } |
| else |
| { |
| FILE *f = fdopen (fd, "r"); |
| if (f == NULL) |
| exit (1); |
| |
| if (fread (result, 1, BUFSIZE, f) <= 0) |
| exit (1); |
| } |
| |
| return 0; |
| } |
| |
| int |
| main (void) |
| { |
| exit (__getcmdline ()); |
| }''' |
| |
| if cc_can_run |
| rres = cc.run(cmdline_test_code, name : '/proc/self/cmdline') |
| have_proc_self_cmdline = rres.compiled() and rres.returncode() == 0 |
| else |
| have_proc_self_cmdline = meson.get_external_property('have_proc_self_cmdline', false) |
| endif |
| |
| glib_conf.set('HAVE_PROC_SELF_CMDLINE', have_proc_self_cmdline) |
| |
| python = import('python').find_installation(modules: ['packaging']) |
| # used for '#!/usr/bin/env <name>' |
| python_name = 'python3' |
| |
| python_version = python.language_version() |
| python_version_req = '>=3.7' |
| if not python_version.version_compare(python_version_req) |
| error('Requires Python @0@, @1@ found.'.format(python_version_req, python_version)) |
| endif |
| |
| # Determine which user environment-dependent files that we want to install |
| bash = find_program('bash', required : false) |
| have_bash = bash.found() # For completion scripts |
| bash_comp_dep = dependency('bash-completion', version: '>=2.0', required: false) |
| have_sh = find_program('sh', required : false).found() # For glib-gettextize |
| have_pkg_config = find_program('pkg-config', required: false).found() |
| |
| # Some installed tests require a custom environment |
| env_program = find_program('env', required: installed_tests_enabled) |
| |
| # FIXME: How to detect Solaris? https://github.com/mesonbuild/meson/issues/1578 |
| if host_system == 'sunos' |
| glib_conf.set('_XOPEN_SOURCE_EXTENDED', 1) |
| glib_conf.set('_XOPEN_SOURCE', 2) |
| glib_conf.set('__EXTENSIONS__',1) |
| endif |
| |
| # Sadly Meson does not expose this value: |
| # https://github.com/mesonbuild/meson/pull/3460 |
| if host_system == 'windows' |
| # Autotools explicitly removed --Wl,--export-all-symbols from windows builds, |
| # with no explanation. Do the same here for now but this could be revisited if |
| # if causes issues. |
| export_dynamic_ldflags = [] |
| elif host_system == 'cygwin' |
| export_dynamic_ldflags = ['-Wl,--export-all-symbols'] |
| elif host_system in ['darwin', 'ios'] |
| export_dynamic_ldflags = [] |
| elif host_system == 'sunos' |
| export_dynamic_ldflags = [] |
| else |
| export_dynamic_ldflags = ['-Wl,--export-dynamic'] |
| endif |
| |
| win32_cflags = [] |
| win32_ldflags = [] |
| if host_system == 'windows' and cc.get_id() != 'msvc' and cc.get_id() != 'clang-cl' |
| # Ensure MSVC-compatible struct packing convention is used when |
| # compiling for Win32 with gcc. It is used for the whole project and exposed |
| # in glib-2.0.pc. |
| if not cc.compiles(''' |
| struct _GTestMSBitfields |
| { |
| int a : 1; |
| short b : 1; |
| }; |
| |
| typedef char _StaticCheck[sizeof(struct _GTestMSBitfields) != sizeof(int) ? 1 : -1]; |
| ''') |
| warning(''' |
| Your compiler does not have ms-bitfields packing by default. |
| Please use gcc >= 4.7 or clang >= 12: GLib will drop -mms-bitfields in the future. |
| ''') |
| endif |
| win32_cflags = ['-mms-bitfields'] |
| add_project_arguments(win32_cflags, language : 'c') |
| |
| # Win32 API libs, used only by libglib and exposed in glib-2.0.pc |
| win32_ldflags = ['-lws2_32', '-lole32', '-lwinmm', '-lshlwapi', '-luuid'] |
| elif host_system == 'cygwin' |
| win32_ldflags = ['-luser32', '-lkernel32'] |
| endif |
| |
| # Tracing: dtrace |
| want_dtrace = get_option('dtrace') |
| enable_dtrace = false |
| |
| # Since dtrace support is opt-in we just error out if it was requested but |
| # is not available. We don't bother with autodetection yet. |
| if want_dtrace |
| if glib_have_carbon |
| error('GLib dtrace support not yet compatible with macOS dtrace') |
| endif |
| dtrace = find_program('dtrace', required : true) # error out if not found |
| if not cc.has_header('sys/sdt.h') |
| error('dtrace support needs sys/sdt.h header') |
| endif |
| # FIXME: autotools build also passes -fPIC -DPIC but is it needed in this case? |
| dtrace_obj_gen = generator(dtrace, |
| output : '@BASENAME@.o', |
| arguments : ['-G', '-s', '@INPUT@', '-o', '@OUTPUT@']) |
| dtrace_hdr_gen = generator(python, |
| output : '@BASENAME@.h', |
| arguments : ['-c', ''' |
| import subprocess, sys |
| subprocess.run(sys.argv[1:], check=True) |
| output = sys.argv[6] |
| with open(output) as f: |
| contents = f.read() |
| contents = contents.replace("define STAP_HAS_SEMAPHORES 1", |
| "undef STAP_HAS_SEMAPHORES") |
| contents = contents.replace("define _SDT_HAS_SEMAPHORES 1", |
| "undef _SDT_HAS_SEMAPHORES") |
| with open(output, "w") as f: |
| f.write(contents) |
| ''', dtrace.full_path(), '-h', '-s', '@INPUT@', '-o', '@OUTPUT@']) |
| glib_conf.set('HAVE_DTRACE', 1) |
| enable_dtrace = true |
| endif |
| |
| if cc.has_header_symbol('sys/ptrace.h', 'PTRACE_O_EXITKILL') |
| glib_conf.set('HAVE_PTRACE_O_EXITKILL', 1) |
| endif |
| |
| # systemtap |
| want_systemtap = get_option('systemtap') |
| enable_systemtap = false |
| |
| if want_systemtap and enable_dtrace |
| tapset_install_dir = get_option('tapset_install_dir') |
| if tapset_install_dir == '' |
| tapset_install_dir = join_paths(get_option('datadir'), 'systemtap/tapset', host_machine.cpu_family()) |
| endif |
| stp_cdata = configuration_data() |
| stp_cdata.set('ABS_GLIB_RUNTIME_LIBDIR', glib_libdir) |
| stp_cdata.set('LT_CURRENT', minor_version * 100) |
| stp_cdata.set('LT_REVISION', micro_version) |
| enable_systemtap = true |
| endif |
| |
| # introspection |
| gir_scanner = find_program('g-ir-scanner', required: get_option('introspection')) |
| enable_gir = get_option('introspection').allowed() and gir_scanner.found() and meson.can_run_host_binaries() |
| |
| if get_option('introspection').enabled() and not meson.can_run_host_binaries() |
| error('Running binaries on the build host needs to be supported to build with -Dintrospection=enabled') |
| endif |
| |
| gir_args = [ |
| '--quiet', |
| ] |
| |
| pkg = import('pkgconfig') |
| windows = import('windows') |
| gnome = import('gnome') |
| |
| subdir('tools') |
| subdir('glib') |
| subdir('gobject') |
| subdir('gthread') |
| subdir('gmodule') |
| subdir('gio') |
| subdir('girepository') |
| subdir('fuzzing') |
| subdir('tests') |
| |
| # xgettext is optional (on Windows for instance) |
| if find_program('xgettext', required : get_option('nls')).found() |
| subdir('po') |
| endif |
| |
| # Install m4 macros that other projects use |
| install_data('m4macros/glib-2.0.m4', 'm4macros/glib-gettext.m4', 'm4macros/gsettings.m4', |
| install_dir : get_option('datadir') / 'aclocal', |
| install_tag : 'devel', |
| ) |
| |
| # Check whether we support overriding the invalid parameter handler on Windows for _get_osfhandle(), |
| # g_fsync() (i.e. _commit()), etc |
| if host_system == 'windows' |
| if cc.has_function('_set_thread_local_invalid_parameter_handler', prefix: '#include <stdlib.h>') |
| glib_conf.set('HAVE__SET_THREAD_LOCAL_INVALID_PARAMETER_HANDLER', 1) |
| endif |
| if cc.has_function('_set_invalid_parameter_handler', prefix: '#include <stdlib.h>') |
| glib_conf.set('HAVE__SET_INVALID_PARAMETER_HANDLER', 1) |
| endif |
| if cc.has_header_symbol('crtdbg.h', '_CrtSetReportMode') |
| glib_conf.set('HAVE__CRT_SET_REPORT_MODE', 1) |
| endif |
| endif |
| |
| configure_file(output : 'config.h', configuration : glib_conf) |
| |
| rst2man = find_program('rst2man', 'rst2man.py', required: get_option('man-pages')) |
| if rst2man.found() |
| rst2man_flags = [ |
| '--syntax-highlight=none', |
| ] |
| man1_dir = join_paths(glib_prefix, get_option('mandir'), 'man1') |
| endif |
| |
| gnome = import('gnome') |
| subdir('docs/reference') |
| |
| summary({ |
| 'host cpu' : host_machine.cpu_family(), |
| 'host endian' : host_machine.endian(), |
| 'host system' : host_system, |
| 'C Compiler' : cc.get_id(), |
| 'C++ Compiler' : have_cxx ? cxx.get_id() : 'none', |
| 'shared build' : glib_build_shared, |
| 'static build' : glib_build_static, |
| }, section: 'Build environment') |
| |
| if build_machine.system() != host_system |
| summary({ |
| 'build cpu' : build_machine.cpu_family(), |
| 'build endian' : build_machine.endian(), |
| 'build system' : build_machine.system(), |
| }, section: 'Build environment') |
| endif |
| |
| if linux_libc != '' |
| summary({ |
| 'linux_libc' : linux_libc |
| }, section: 'Build environment') |
| endif |
| |
| summary({ |
| 'prefix' : glib_prefix, |
| 'bindir' : glib_bindir, |
| 'libexecdir' : glib_libexecdir, |
| 'pkgdatadir' : glib_pkgdatadir, |
| 'datadir' : glib_datadir, |
| 'includedir' : glib_includedir, |
| 'giomodulesdir' : glib_giomodulesdir, |
| 'localstatedir' : glib_localstatedir, |
| 'runstatedir' : glib_runstatedir, |
| }, section: 'Directories') |
| |
| if get_option('multiarch') |
| summary({ |
| 'multiarch bindir' : glib_bindir, |
| 'multiarch libexecdir' : glib_libexecdir, |
| }, section: 'Directories') |
| endif |
| |
| if enable_systemtap |
| summary('tapset dir', get_option('tapset_install_dir'), section: 'Directories') |
| endif |
| |
| if host_system == 'linux' |
| summary({ |
| 'selinux' : selinux_dep.found(), |
| 'libmount' : libmount_dep.found(), |
| }, section: 'Options') |
| endif |
| |
| summary({ |
| 'xattr' : xattr_dep.length() > 0, |
| 'man-pages' : get_option('man-pages'), |
| 'dtrace' : get_option('dtrace'), |
| 'systemtap' : enable_systemtap, |
| 'sysprof' : libsysprof_capture_dep.found(), |
| 'documentation' : get_option('documentation'), |
| 'bsymbolic_functions' : get_option('bsymbolic_functions'), |
| 'force_posix_threads' : get_option('force_posix_threads'), |
| 'tests' : get_option('tests'), |
| 'installed_tests' : get_option('installed_tests'), |
| 'nls' : get_option('nls'), |
| 'oss_fuzz' : get_option('oss_fuzz'), |
| 'glib_debug' : get_option('glib_debug'), |
| 'glib_assert' : get_option('glib_assert'), |
| 'glib_checks' : get_option('glib_checks'), |
| 'libelf' : get_option('libelf'), |
| 'multiarch' : get_option('multiarch'), |
| 'introspection' : enable_gir, |
| }, section: 'Options') |