xref: /qemu/meson.build (revision 73b49878)
1project('qemu', ['c'], meson_version: '>=0.63.0',
2        default_options: ['warning_level=1', 'c_std=gnu11', 'cpp_std=gnu++11', 'b_colorout=auto',
3                          'b_staticpic=false', 'stdsplit=false', 'optimization=2', 'b_pie=true'],
4        version: files('VERSION'))
5
6add_test_setup('quick', exclude_suites: ['slow', 'thorough'], is_default: true)
7add_test_setup('slow', exclude_suites: ['thorough'], env: ['G_TEST_SLOW=1', 'SPEED=slow'])
8add_test_setup('thorough', env: ['G_TEST_SLOW=1', 'SPEED=thorough'])
9
10meson.add_postconf_script(find_program('scripts/symlink-install-tree.py'))
11
12####################
13# Global variables #
14####################
15
16not_found = dependency('', required: false)
17keyval = import('keyval')
18ss = import('sourceset')
19fs = import('fs')
20
21host_os = host_machine.system()
22config_host = keyval.load(meson.current_build_dir() / 'config-host.mak')
23
24# Temporary directory used for files created while
25# configure runs. Since it is in the build directory
26# we can safely blow away any previous version of it
27# (and we need not jump through hoops to try to delete
28# it when configure exits.)
29tmpdir = meson.current_build_dir() / 'meson-private/temp'
30
31if get_option('qemu_suffix').startswith('/')
32  error('qemu_suffix cannot start with a /')
33endif
34
35qemu_confdir = get_option('sysconfdir') / get_option('qemu_suffix')
36qemu_datadir = get_option('datadir') / get_option('qemu_suffix')
37qemu_docdir = get_option('docdir') / get_option('qemu_suffix')
38qemu_moddir = get_option('libdir') / get_option('qemu_suffix')
39
40qemu_desktopdir = get_option('datadir') / 'applications'
41qemu_icondir = get_option('datadir') / 'icons'
42
43genh = []
44qapi_trace_events = []
45
46bsd_oses = ['gnu/kfreebsd', 'freebsd', 'netbsd', 'openbsd', 'dragonfly', 'darwin']
47supported_oses = ['windows', 'freebsd', 'netbsd', 'openbsd', 'darwin', 'sunos', 'linux']
48supported_cpus = ['ppc', 'ppc64', 's390x', 'riscv32', 'riscv64', 'x86', 'x86_64',
49  'arm', 'aarch64', 'loongarch64', 'mips', 'mips64', 'sparc64']
50
51cpu = host_machine.cpu_family()
52
53target_dirs = config_host['TARGET_DIRS'].split()
54
55############
56# Programs #
57############
58
59sh = find_program('sh')
60python = import('python').find_installation()
61
62cc = meson.get_compiler('c')
63all_languages = ['c']
64if host_os == 'windows' and add_languages('cpp', required: false, native: false)
65  all_languages += ['cpp']
66  cxx = meson.get_compiler('cpp')
67endif
68if host_os == 'darwin' and \
69   add_languages('objc', required: get_option('cocoa'), native: false)
70  all_languages += ['objc']
71  objc = meson.get_compiler('objc')
72endif
73
74dtrace = not_found
75stap = not_found
76if 'dtrace' in get_option('trace_backends')
77  dtrace = find_program('dtrace', required: true)
78  stap = find_program('stap', required: false)
79  if stap.found()
80    # Workaround to avoid dtrace(1) producing a file with 'hidden' symbol
81    # visibility. Define STAP_SDT_V2 to produce 'default' symbol visibility
82    # instead. QEMU --enable-modules depends on this because the SystemTap
83    # semaphores are linked into the main binary and not the module's shared
84    # object.
85    add_global_arguments('-DSTAP_SDT_V2',
86                         native: false, language: all_languages)
87  endif
88endif
89
90if get_option('iasl') == ''
91  iasl = find_program('iasl', required: false)
92else
93  iasl = find_program(get_option('iasl'), required: true)
94endif
95
96edk2_targets = [ 'arm-softmmu', 'aarch64-softmmu', 'i386-softmmu', 'x86_64-softmmu' ]
97unpack_edk2_blobs = false
98foreach target : edk2_targets
99  if target in target_dirs
100    bzip2 = find_program('bzip2', required: get_option('install_blobs'))
101    unpack_edk2_blobs = bzip2.found()
102    break
103  endif
104endforeach
105
106#####################
107# Option validation #
108#####################
109
110# Fuzzing
111if get_option('fuzzing') and get_option('fuzzing_engine') == '' and \
112    not cc.links('''
113          #include <stdint.h>
114          #include <sys/types.h>
115          int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size);
116          int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { return 0; }
117        ''',
118        args: ['-Werror', '-fsanitize=fuzzer'])
119  error('Your compiler does not support -fsanitize=fuzzer')
120endif
121
122# Tracing backends
123if 'ftrace' in get_option('trace_backends') and host_os != 'linux'
124  error('ftrace is supported only on Linux')
125endif
126if 'syslog' in get_option('trace_backends') and not cc.compiles('''
127    #include <syslog.h>
128    int main(void) {
129        openlog("qemu", LOG_PID, LOG_DAEMON);
130        syslog(LOG_INFO, "configure");
131        return 0;
132    }''')
133  error('syslog is not supported on this system')
134endif
135
136# Miscellaneous Linux-only features
137get_option('mpath') \
138  .require(host_os == 'linux', error_message: 'Multipath is supported only on Linux')
139
140multiprocess_allowed = get_option('multiprocess') \
141  .require(host_os == 'linux', error_message: 'Multiprocess QEMU is supported only on Linux') \
142  .allowed()
143
144vfio_user_server_allowed = get_option('vfio_user_server') \
145  .require(host_os == 'linux', error_message: 'vfio-user server is supported only on Linux') \
146  .allowed()
147
148have_tpm = get_option('tpm') \
149  .require(host_os != 'windows', error_message: 'TPM emulation only available on POSIX systems') \
150  .allowed()
151
152# vhost
153have_vhost_user = get_option('vhost_user') \
154  .disable_auto_if(host_os != 'linux') \
155  .require(host_os != 'windows',
156           error_message: 'vhost-user is not available on Windows').allowed()
157have_vhost_vdpa = get_option('vhost_vdpa') \
158  .require(host_os == 'linux',
159           error_message: 'vhost-vdpa is only available on Linux').allowed()
160have_vhost_kernel = get_option('vhost_kernel') \
161  .require(host_os == 'linux',
162           error_message: 'vhost-kernel is only available on Linux').allowed()
163have_vhost_user_crypto = get_option('vhost_crypto') \
164  .require(have_vhost_user,
165           error_message: 'vhost-crypto requires vhost-user to be enabled').allowed()
166
167have_vhost = have_vhost_user or have_vhost_vdpa or have_vhost_kernel
168
169have_vhost_net_user = have_vhost_user and get_option('vhost_net').allowed()
170have_vhost_net_vdpa = have_vhost_vdpa and get_option('vhost_net').allowed()
171have_vhost_net_kernel = have_vhost_kernel and get_option('vhost_net').allowed()
172have_vhost_net = have_vhost_net_kernel or have_vhost_net_user or have_vhost_net_vdpa
173
174# type of binaries to build
175have_linux_user = false
176have_bsd_user = false
177have_system = false
178foreach target : target_dirs
179  have_linux_user = have_linux_user or target.endswith('linux-user')
180  have_bsd_user = have_bsd_user or target.endswith('bsd-user')
181  have_system = have_system or target.endswith('-softmmu')
182endforeach
183have_user = have_linux_user or have_bsd_user
184
185have_tools = get_option('tools') \
186  .disable_auto_if(not have_system) \
187  .allowed()
188have_ga = get_option('guest_agent') \
189  .disable_auto_if(not have_system and not have_tools) \
190  .require(host_os in ['sunos', 'linux', 'windows', 'freebsd', 'netbsd', 'openbsd'],
191           error_message: 'unsupported OS for QEMU guest agent') \
192  .allowed()
193have_block = have_system or have_tools
194
195enable_modules = get_option('modules') \
196  .require(host_os != 'windows',
197           error_message: 'Modules are not available for Windows') \
198  .require(not get_option('prefer_static'),
199           error_message: 'Modules are incompatible with static linking') \
200  .allowed()
201
202#######################################
203# Variables for host and accelerators #
204#######################################
205
206if cpu not in supported_cpus
207  host_arch = 'unknown'
208elif cpu == 'x86'
209  host_arch = 'i386'
210elif cpu == 'mips64'
211  host_arch = 'mips'
212elif cpu in ['riscv32', 'riscv64']
213  host_arch = 'riscv'
214else
215  host_arch = cpu
216endif
217
218if cpu in ['x86', 'x86_64']
219  kvm_targets = ['i386-softmmu', 'x86_64-softmmu']
220elif cpu == 'aarch64'
221  kvm_targets = ['aarch64-softmmu']
222elif cpu == 's390x'
223  kvm_targets = ['s390x-softmmu']
224elif cpu in ['ppc', 'ppc64']
225  kvm_targets = ['ppc-softmmu', 'ppc64-softmmu']
226elif cpu in ['mips', 'mips64']
227  kvm_targets = ['mips-softmmu', 'mipsel-softmmu', 'mips64-softmmu', 'mips64el-softmmu']
228elif cpu in ['riscv32']
229  kvm_targets = ['riscv32-softmmu']
230elif cpu in ['riscv64']
231  kvm_targets = ['riscv64-softmmu']
232elif cpu in ['loongarch64']
233  kvm_targets = ['loongarch64-softmmu']
234else
235  kvm_targets = []
236endif
237accelerator_targets = { 'CONFIG_KVM': kvm_targets }
238
239if cpu in ['x86', 'x86_64']
240  xen_targets = ['i386-softmmu', 'x86_64-softmmu']
241elif cpu in ['arm', 'aarch64']
242  # i386 emulator provides xenpv machine type for multiple architectures
243  xen_targets = ['i386-softmmu', 'x86_64-softmmu', 'aarch64-softmmu']
244else
245  xen_targets = []
246endif
247accelerator_targets += { 'CONFIG_XEN': xen_targets }
248
249if cpu in ['aarch64']
250  accelerator_targets += {
251    'CONFIG_HVF': ['aarch64-softmmu']
252  }
253endif
254
255if cpu in ['x86', 'x86_64']
256  accelerator_targets += {
257    'CONFIG_HVF': ['x86_64-softmmu'],
258    'CONFIG_NVMM': ['i386-softmmu', 'x86_64-softmmu'],
259    'CONFIG_WHPX': ['i386-softmmu', 'x86_64-softmmu'],
260  }
261endif
262
263modular_tcg = []
264# Darwin does not support references to thread-local variables in modules
265if host_os != 'darwin'
266  modular_tcg = ['i386-softmmu', 'x86_64-softmmu']
267endif
268
269##################
270# Compiler flags #
271##################
272
273foreach lang : all_languages
274  compiler = meson.get_compiler(lang)
275  if compiler.get_id() == 'gcc' and compiler.version().version_compare('>=7.4')
276    # ok
277  elif compiler.get_id() == 'clang' and compiler.compiles('''
278      #ifdef __apple_build_version__
279      # if __clang_major__ < 12 || (__clang_major__ == 12 && __clang_minor__ < 0)
280      #  error You need at least XCode Clang v12.0 to compile QEMU
281      # endif
282      #else
283      # if __clang_major__ < 10 || (__clang_major__ == 10 && __clang_minor__ < 0)
284      #  error You need at least Clang v10.0 to compile QEMU
285      # endif
286      #endif''')
287    # ok
288  else
289    error('You either need GCC v7.4 or Clang v10.0 (or XCode Clang v12.0) to compile QEMU')
290  endif
291endforeach
292
293# default flags for all hosts
294# We use -fwrapv to tell the compiler that we require a C dialect where
295# left shift of signed integers is well defined and has the expected
296# 2s-complement style results. (Both clang and gcc agree that it
297# provides these semantics.)
298
299qemu_common_flags = [
300  '-D_GNU_SOURCE', '-D_FILE_OFFSET_BITS=64', '-D_LARGEFILE_SOURCE',
301  '-fno-strict-aliasing', '-fno-common', '-fwrapv' ]
302qemu_cflags = []
303qemu_ldflags = []
304
305if host_os == 'darwin'
306  # Disable attempts to use ObjectiveC features in os/object.h since they
307  # won't work when we're compiling with gcc as a C compiler.
308  if compiler.get_id() == 'gcc'
309    qemu_common_flags += '-DOS_OBJECT_USE_OBJC=0'
310  endif
311elif host_os == 'sunos'
312  # needed for CMSG_ macros in sys/socket.h
313  qemu_common_flags += '-D_XOPEN_SOURCE=600'
314  # needed for TIOCWIN* defines in termios.h
315  qemu_common_flags += '-D__EXTENSIONS__'
316elif host_os == 'haiku'
317  qemu_common_flags += ['-DB_USE_POSITIVE_POSIX_ERRORS', '-D_BSD_SOURCE', '-fPIC']
318endif
319
320# __sync_fetch_and_and requires at least -march=i486. Many toolchains
321# use i686 as default anyway, but for those that don't, an explicit
322# specification is necessary
323if host_arch == 'i386' and not cc.links('''
324  static int sfaa(int *ptr)
325  {
326    return __sync_fetch_and_and(ptr, 0);
327  }
328
329  int main(void)
330  {
331    int val = 42;
332    val = __sync_val_compare_and_swap(&val, 0, 1);
333    sfaa(&val);
334    return val;
335  }''')
336  qemu_common_flags = ['-march=i486'] + qemu_common_flags
337endif
338
339if get_option('prefer_static')
340  qemu_ldflags += get_option('b_pie') ? '-static-pie' : '-static'
341endif
342
343# Meson currently only handles pie as a boolean for now, so if the user
344# has explicitly disabled PIE we need to extend our cflags.
345#
346# -no-pie is supposedly a linker flag that has no effect on the compiler
347# command line, but some distros, that didn't quite know what they were
348# doing, made local changes to gcc's specs file that turned it into
349# a compiler command-line flag.
350#
351# What about linker flags?  For a static build, no PIE is implied by -static
352# which we added above (and if it's not because of the same specs patching,
353# there's nothing we can do: compilation will fail, report a bug to your
354# distro and do not use --disable-pie in the meanwhile).  For dynamic linking,
355# instead, we can't add -no-pie because it overrides -shared: the linker then
356# tries to build an executable instead of a shared library and fails.  So
357# don't add -no-pie anywhere and cross fingers. :(
358if not get_option('b_pie')
359  qemu_common_flags += cc.get_supported_arguments('-fno-pie', '-no-pie')
360endif
361
362if not get_option('stack_protector').disabled()
363  stack_protector_probe = '''
364    int main(int argc, char *argv[])
365    {
366      char arr[64], *p = arr, *c = argv[argc - 1];
367      while (*c) {
368          *p++ = *c++;
369      }
370      return 0;
371    }'''
372  have_stack_protector = false
373  foreach arg : ['-fstack-protector-strong', '-fstack-protector-all']
374    # We need to check both a compile and a link, since some compiler
375    # setups fail only on a .c->.o compile and some only at link time
376    if cc.compiles(stack_protector_probe, args: ['-Werror', arg]) and \
377       cc.links(stack_protector_probe, args: ['-Werror', arg])
378      have_stack_protector = true
379      qemu_cflags += arg
380      qemu_ldflags += arg
381      break
382    endif
383  endforeach
384  get_option('stack_protector') \
385    .require(have_stack_protector, error_message: 'Stack protector not supported')
386endif
387
388coroutine_backend = get_option('coroutine_backend')
389ucontext_probe = '''
390  #include <ucontext.h>
391  #ifdef __stub_makecontext
392  #error Ignoring glibc stub makecontext which will always fail
393  #endif
394  int main(void) { makecontext(0, 0, 0); return 0; }'''
395
396# On Windows the only valid backend is the Windows specific one.
397# For POSIX prefer ucontext, but it's not always possible. The fallback
398# is sigcontext.
399supported_backends = []
400if host_os == 'windows'
401  supported_backends += ['windows']
402else
403  if host_os != 'darwin' and cc.links(ucontext_probe)
404    supported_backends += ['ucontext']
405  endif
406  supported_backends += ['sigaltstack']
407endif
408
409if coroutine_backend == 'auto'
410  coroutine_backend = supported_backends[0]
411elif coroutine_backend not in supported_backends
412  error('"@0@" backend requested but not available.  Available backends: @1@' \
413        .format(coroutine_backend, ', '.join(supported_backends)))
414endif
415
416# Compiles if SafeStack *not* enabled
417safe_stack_probe = '''
418  int main(void)
419  {
420  #if defined(__has_feature)
421  #if __has_feature(safe_stack)
422  #error SafeStack Enabled
423  #endif
424  #endif
425      return 0;
426  }'''
427if get_option('safe_stack') != not cc.compiles(safe_stack_probe)
428  safe_stack_arg = get_option('safe_stack') ? '-fsanitize=safe-stack' : '-fno-sanitize=safe-stack'
429  if get_option('safe_stack') != not cc.compiles(safe_stack_probe, args: safe_stack_arg)
430    error(get_option('safe_stack') \
431          ? 'SafeStack not supported by your compiler' \
432          : 'Cannot disable SafeStack')
433  endif
434  qemu_cflags += safe_stack_arg
435  qemu_ldflags += safe_stack_arg
436endif
437if get_option('safe_stack') and coroutine_backend != 'ucontext'
438  error('SafeStack is only supported with the ucontext coroutine backend')
439endif
440
441if get_option('sanitizers')
442  if cc.has_argument('-fsanitize=address')
443    qemu_cflags = ['-fsanitize=address'] + qemu_cflags
444    qemu_ldflags = ['-fsanitize=address'] + qemu_ldflags
445  endif
446
447  # Detect static linking issue with ubsan - https://gcc.gnu.org/bugzilla/show_bug.cgi?id=84285
448  if cc.links('int main(int argc, char **argv) { return argc + 1; }',
449              args: [qemu_ldflags, '-fsanitize=undefined'])
450    qemu_cflags = ['-fsanitize=undefined'] + qemu_cflags
451    qemu_ldflags = ['-fsanitize=undefined'] + qemu_ldflags
452  endif
453endif
454
455# Thread sanitizer is, for now, much noisier than the other sanitizers;
456# keep it separate until that is not the case.
457if get_option('tsan')
458  if get_option('sanitizers')
459    error('TSAN is not supported with other sanitizers')
460  endif
461  if not cc.has_function('__tsan_create_fiber',
462                         args: '-fsanitize=thread',
463                         prefix: '#include <sanitizer/tsan_interface.h>')
464    error('Cannot enable TSAN due to missing fiber annotation interface')
465  endif
466  qemu_cflags = ['-fsanitize=thread'] + qemu_cflags
467  qemu_ldflags = ['-fsanitize=thread'] + qemu_ldflags
468endif
469
470# Detect support for PT_GNU_RELRO + DT_BIND_NOW.
471# The combination is known as "full relro", because .got.plt is read-only too.
472qemu_ldflags += cc.get_supported_link_arguments('-Wl,-z,relro', '-Wl,-z,now')
473
474if host_os == 'windows'
475  qemu_ldflags += cc.get_supported_link_arguments('-Wl,--no-seh', '-Wl,--nxcompat')
476  qemu_ldflags += cc.get_supported_link_arguments('-Wl,--dynamicbase', '-Wl,--high-entropy-va')
477endif
478
479# Exclude --warn-common with TSan to suppress warnings from the TSan libraries.
480if host_os != 'sunos' and not get_option('tsan')
481  qemu_ldflags += cc.get_supported_link_arguments('-Wl,--warn-common')
482endif
483
484if get_option('fuzzing')
485  # Specify a filter to only instrument code that is directly related to
486  # virtual-devices.
487  configure_file(output: 'instrumentation-filter',
488                 input: 'scripts/oss-fuzz/instrumentation-filter-template',
489                 copy: true)
490
491  if cc.compiles('int main () { return 0; }',
492                  name: '-fsanitize-coverage-allowlist=/dev/null',
493                 args: ['-fsanitize-coverage-allowlist=/dev/null',
494                        '-fsanitize-coverage=trace-pc'] )
495    qemu_common_flags += ['-fsanitize-coverage-allowlist=instrumentation-filter']
496  endif
497
498  if get_option('fuzzing_engine') == ''
499    # Add CFLAGS to tell clang to add fuzzer-related instrumentation to all the
500    # compiled code.  To build non-fuzzer binaries with --enable-fuzzing, link
501    # everything with fsanitize=fuzzer-no-link. Otherwise, the linker will be
502    # unable to bind the fuzzer-related callbacks added by instrumentation.
503    qemu_common_flags += ['-fsanitize=fuzzer-no-link']
504    qemu_ldflags += ['-fsanitize=fuzzer-no-link']
505    # For the actual fuzzer binaries, we need to link against the libfuzzer
506    # library. They need to be configurable, to support OSS-Fuzz
507    fuzz_exe_ldflags = ['-fsanitize=fuzzer']
508  else
509    # LIB_FUZZING_ENGINE was set; assume we are running on OSS-Fuzz, and
510    # the needed CFLAGS have already been provided
511    fuzz_exe_ldflags = get_option('fuzzing_engine').split()
512  endif
513endif
514
515if get_option('cfi')
516  cfi_flags=[]
517  # Check for dependency on LTO
518  if not get_option('b_lto')
519    error('Selected Control-Flow Integrity but LTO is disabled')
520  endif
521  if enable_modules
522    error('Selected Control-Flow Integrity is not compatible with modules')
523  endif
524  # Check for cfi flags. CFI requires LTO so we can't use
525  # get_supported_arguments, but need a more complex "compiles" which allows
526  # custom arguments
527  if cc.compiles('int main () { return 0; }', name: '-fsanitize=cfi-icall',
528                 args: ['-flto', '-fsanitize=cfi-icall'] )
529    cfi_flags += '-fsanitize=cfi-icall'
530  else
531    error('-fsanitize=cfi-icall is not supported by the compiler')
532  endif
533  if cc.compiles('int main () { return 0; }',
534                 name: '-fsanitize-cfi-icall-generalize-pointers',
535                 args: ['-flto', '-fsanitize=cfi-icall',
536                        '-fsanitize-cfi-icall-generalize-pointers'] )
537    cfi_flags += '-fsanitize-cfi-icall-generalize-pointers'
538  else
539    error('-fsanitize-cfi-icall-generalize-pointers is not supported by the compiler')
540  endif
541  if get_option('cfi_debug')
542    if cc.compiles('int main () { return 0; }',
543                   name: '-fno-sanitize-trap=cfi-icall',
544                   args: ['-flto', '-fsanitize=cfi-icall',
545                          '-fno-sanitize-trap=cfi-icall'] )
546      cfi_flags += '-fno-sanitize-trap=cfi-icall'
547    else
548      error('-fno-sanitize-trap=cfi-icall is not supported by the compiler')
549    endif
550  endif
551  add_global_arguments(cfi_flags, native: false, language: all_languages)
552  add_global_link_arguments(cfi_flags, native: false, language: all_languages)
553endif
554
555# Check further flags that make QEMU more robust against malicious parties
556
557hardening_flags = [
558    # Zero out registers used during a function call
559    # upon its return. This makes it harder to assemble
560    # ROP gadgets into something usable
561    '-fzero-call-used-regs=used-gpr',
562
563    # Initialize all stack variables to zero. This makes
564    # it harder to take advantage of uninitialized stack
565    # data to drive exploits
566    '-ftrivial-auto-var-init=zero',
567]
568
569qemu_common_flags += cc.get_supported_arguments(hardening_flags)
570
571add_global_arguments(qemu_common_flags, native: false, language: all_languages)
572add_global_link_arguments(qemu_ldflags, native: false, language: all_languages)
573
574# Collect warnings that we want to enable
575
576warn_flags = [
577  '-Wundef',
578  '-Wwrite-strings',
579  '-Wmissing-prototypes',
580  '-Wstrict-prototypes',
581  '-Wredundant-decls',
582  '-Wold-style-declaration',
583  '-Wold-style-definition',
584  '-Wtype-limits',
585  '-Wformat-security',
586  '-Wformat-y2k',
587  '-Winit-self',
588  '-Wignored-qualifiers',
589  '-Wempty-body',
590  '-Wnested-externs',
591  '-Wendif-labels',
592  '-Wexpansion-to-defined',
593  '-Wimplicit-fallthrough=2',
594  '-Wmissing-format-attribute',
595  '-Wno-initializer-overrides',
596  '-Wno-missing-include-dirs',
597  '-Wno-shift-negative-value',
598  '-Wno-string-plus-int',
599  '-Wno-typedef-redefinition',
600  '-Wno-tautological-type-limit-compare',
601  '-Wno-psabi',
602  '-Wno-gnu-variable-sized-type-not-at-end',
603  '-Wshadow=local',
604]
605
606if host_os != 'darwin'
607  warn_flags += ['-Wthread-safety']
608endif
609
610# Set up C++ compiler flags
611qemu_cxxflags = []
612if 'cpp' in all_languages
613  qemu_cxxflags = ['-D__STDC_LIMIT_MACROS', '-D__STDC_CONSTANT_MACROS', '-D__STDC_FORMAT_MACROS'] + qemu_cflags
614endif
615
616add_project_arguments(qemu_cflags, native: false, language: 'c')
617add_project_arguments(cc.get_supported_arguments(warn_flags), native: false, language: 'c')
618if 'cpp' in all_languages
619  add_project_arguments(qemu_cxxflags, native: false, language: 'cpp')
620  add_project_arguments(cxx.get_supported_arguments(warn_flags), native: false, language: 'cpp')
621endif
622if 'objc' in all_languages
623  # Note sanitizer flags are not applied to Objective-C sources!
624  add_project_arguments(objc.get_supported_arguments(warn_flags), native: false, language: 'objc')
625endif
626if host_os == 'linux'
627  add_project_arguments('-isystem', meson.current_source_dir() / 'linux-headers',
628                        '-isystem', 'linux-headers',
629                        language: all_languages)
630endif
631
632add_project_arguments('-iquote', '.',
633                      '-iquote', meson.current_source_dir(),
634                      '-iquote', meson.current_source_dir() / 'include',
635                      language: all_languages)
636
637# If a host-specific include directory exists, list that first...
638host_include = meson.current_source_dir() / 'host/include/'
639if fs.is_dir(host_include / host_arch)
640  add_project_arguments('-iquote', host_include / host_arch,
641                        language: all_languages)
642endif
643# ... followed by the generic fallback.
644add_project_arguments('-iquote', host_include / 'generic',
645                      language: all_languages)
646
647sparse = find_program('cgcc', required: get_option('sparse'))
648if sparse.found()
649  run_target('sparse',
650             command: [find_program('scripts/check_sparse.py'),
651                       'compile_commands.json', sparse.full_path(), '-Wbitwise',
652                       '-Wno-transparent-union', '-Wno-old-initializer',
653                       '-Wno-non-pointer-null'])
654endif
655
656#####################################
657# Host-specific libraries and flags #
658#####################################
659
660libm = cc.find_library('m', required: false)
661threads = dependency('threads')
662util = cc.find_library('util', required: false)
663winmm = []
664socket = []
665version_res = []
666coref = []
667iokit = []
668emulator_link_args = []
669midl = not_found
670widl = not_found
671pathcch = not_found
672host_dsosuf = '.so'
673if host_os == 'windows'
674  midl = find_program('midl', required: false)
675  widl = find_program('widl', required: false)
676  pathcch = cc.find_library('pathcch')
677  socket = cc.find_library('ws2_32')
678  winmm = cc.find_library('winmm')
679
680  win = import('windows')
681  version_res = win.compile_resources('version.rc',
682                                      depend_files: files('pc-bios/qemu-nsis.ico'),
683                                      include_directories: include_directories('.'))
684  host_dsosuf = '.dll'
685elif host_os == 'darwin'
686  coref = dependency('appleframeworks', modules: 'CoreFoundation')
687  iokit = dependency('appleframeworks', modules: 'IOKit', required: false)
688  host_dsosuf = '.dylib'
689elif host_os == 'sunos'
690  socket = [cc.find_library('socket'),
691            cc.find_library('nsl'),
692            cc.find_library('resolv')]
693elif host_os == 'haiku'
694  socket = [cc.find_library('posix_error_mapper'),
695            cc.find_library('network'),
696            cc.find_library('bsd')]
697elif host_os == 'openbsd'
698  if get_option('tcg').allowed() and target_dirs.length() > 0
699    # Disable OpenBSD W^X if available
700    emulator_link_args = cc.get_supported_link_arguments('-Wl,-z,wxneeded')
701  endif
702endif
703
704###############################################
705# Host-specific configuration of accelerators #
706###############################################
707
708accelerators = []
709if get_option('kvm').allowed() and host_os == 'linux'
710  accelerators += 'CONFIG_KVM'
711endif
712if get_option('whpx').allowed() and host_os == 'windows'
713  if get_option('whpx').enabled() and host_machine.cpu() != 'x86_64'
714    error('WHPX requires 64-bit host')
715  elif cc.has_header('winhvplatform.h', required: get_option('whpx')) and \
716       cc.has_header('winhvemulation.h', required: get_option('whpx'))
717    accelerators += 'CONFIG_WHPX'
718  endif
719endif
720
721hvf = not_found
722if get_option('hvf').allowed()
723  hvf = dependency('appleframeworks', modules: 'Hypervisor',
724                   required: get_option('hvf'))
725  if hvf.found()
726    accelerators += 'CONFIG_HVF'
727  endif
728endif
729
730nvmm = not_found
731if host_os == 'netbsd'
732  nvmm = cc.find_library('nvmm', required: get_option('nvmm'))
733  if nvmm.found()
734    accelerators += 'CONFIG_NVMM'
735  endif
736endif
737
738tcg_arch = host_arch
739if get_option('tcg').allowed()
740  if host_arch == 'unknown'
741    if not get_option('tcg_interpreter')
742      error('Unsupported CPU @0@, try --enable-tcg-interpreter'.format(cpu))
743    endif
744  elif get_option('tcg_interpreter')
745    warning('Use of the TCG interpreter is not recommended on this host')
746    warning('architecture. There is a native TCG execution backend available')
747    warning('which provides substantially better performance and reliability.')
748    warning('It is strongly recommended to remove the --enable-tcg-interpreter')
749    warning('configuration option on this architecture to use the native')
750    warning('backend.')
751  endif
752  if get_option('tcg_interpreter')
753    tcg_arch = 'tci'
754  elif host_arch == 'x86_64'
755    tcg_arch = 'i386'
756  elif host_arch == 'ppc64'
757    tcg_arch = 'ppc'
758  endif
759  add_project_arguments('-iquote', meson.current_source_dir() / 'tcg' / tcg_arch,
760                        language: all_languages)
761
762  accelerators += 'CONFIG_TCG'
763endif
764
765if 'CONFIG_KVM' not in accelerators and get_option('kvm').enabled()
766  error('KVM not available on this platform')
767endif
768if 'CONFIG_HVF' not in accelerators and get_option('hvf').enabled()
769  error('HVF not available on this platform')
770endif
771if 'CONFIG_NVMM' not in accelerators and get_option('nvmm').enabled()
772  error('NVMM not available on this platform')
773endif
774if 'CONFIG_WHPX' not in accelerators and get_option('whpx').enabled()
775  error('WHPX not available on this platform')
776endif
777
778xen = not_found
779if get_option('xen').enabled() or (get_option('xen').auto() and have_system)
780  xencontrol = dependency('xencontrol', required: false,
781                          method: 'pkg-config')
782  if xencontrol.found()
783    xen_pc = declare_dependency(version: xencontrol.version(),
784      dependencies: [
785        xencontrol,
786        # disabler: true makes xen_pc.found() return false if any is not found
787        dependency('xenstore', required: false,
788                   method: 'pkg-config',
789                   disabler: true),
790        dependency('xenforeignmemory', required: false,
791                   method: 'pkg-config',
792                   disabler: true),
793        dependency('xengnttab', required: false,
794                   method: 'pkg-config',
795                   disabler: true),
796        dependency('xenevtchn', required: false,
797                   method: 'pkg-config',
798                   disabler: true),
799        dependency('xendevicemodel', required: false,
800                   method: 'pkg-config',
801                   disabler: true),
802        # optional, no "disabler: true"
803        dependency('xentoolcore', required: false,
804                   method: 'pkg-config')])
805    if xen_pc.found()
806      xen = xen_pc
807    endif
808  endif
809  if not xen.found()
810    xen_tests = [ '4.11.0', '4.10.0', '4.9.0', '4.8.0', '4.7.1' ]
811    xen_libs = {
812      '4.11.0': [ 'xenstore', 'xenctrl', 'xendevicemodel', 'xenforeignmemory', 'xengnttab', 'xenevtchn', 'xentoolcore' ],
813      '4.10.0': [ 'xenstore', 'xenctrl', 'xendevicemodel', 'xenforeignmemory', 'xengnttab', 'xenevtchn', 'xentoolcore' ],
814      '4.9.0': [ 'xenstore', 'xenctrl', 'xendevicemodel', 'xenforeignmemory', 'xengnttab', 'xenevtchn' ],
815      '4.8.0': [ 'xenstore', 'xenctrl', 'xenforeignmemory', 'xengnttab', 'xenevtchn' ],
816      '4.7.1': [ 'xenstore', 'xenctrl', 'xenforeignmemory', 'xengnttab', 'xenevtchn' ],
817    }
818    xen_deps = {}
819    foreach ver: xen_tests
820      # cache the various library tests to avoid polluting the logs
821      xen_test_deps = []
822      foreach l: xen_libs[ver]
823        if l not in xen_deps
824          xen_deps += { l: cc.find_library(l, required: false) }
825        endif
826        xen_test_deps += xen_deps[l]
827      endforeach
828
829      # Use -D to pick just one of the test programs in scripts/xen-detect.c
830      xen_version = ver.split('.')
831      xen_ctrl_version = xen_version[0] + \
832        ('0' + xen_version[1]).substring(-2) + \
833        ('0' + xen_version[2]).substring(-2)
834      if cc.links(files('scripts/xen-detect.c'),
835                  args: '-DCONFIG_XEN_CTRL_INTERFACE_VERSION=' + xen_ctrl_version,
836                  dependencies: xen_test_deps)
837        xen = declare_dependency(version: ver, dependencies: xen_test_deps)
838        break
839      endif
840    endforeach
841  endif
842  if xen.found()
843    accelerators += 'CONFIG_XEN'
844  elif get_option('xen').enabled()
845    error('could not compile and link Xen test program')
846  endif
847endif
848have_xen_pci_passthrough = get_option('xen_pci_passthrough') \
849  .require(xen.found(),
850           error_message: 'Xen PCI passthrough requested but Xen not enabled') \
851  .require(host_os == 'linux',
852           error_message: 'Xen PCI passthrough not available on this platform') \
853  .require(cpu == 'x86'  or cpu == 'x86_64',
854           error_message: 'Xen PCI passthrough not available on this platform') \
855  .allowed()
856
857################
858# Dependencies #
859################
860
861# When bumping glib minimum version, please check also whether to increase
862# the _WIN32_WINNT setting in osdep.h according to the value from glib
863glib_req_ver = '>=2.56.0'
864glib_pc = dependency('glib-2.0', version: glib_req_ver, required: true,
865                    method: 'pkg-config')
866glib_cflags = []
867if enable_modules
868  gmodule = dependency('gmodule-export-2.0', version: glib_req_ver, required: true,
869                       method: 'pkg-config')
870elif get_option('plugins')
871  gmodule = dependency('gmodule-no-export-2.0', version: glib_req_ver, required: true,
872                       method: 'pkg-config')
873else
874  gmodule = not_found
875endif
876
877# This workaround is required due to a bug in pkg-config file for glib as it
878# doesn't define GLIB_STATIC_COMPILATION for pkg-config --static
879if host_os == 'windows' and get_option('prefer_static')
880  glib_cflags += ['-DGLIB_STATIC_COMPILATION']
881endif
882
883# Sanity check that the current size_t matches the
884# size that glib thinks it should be. This catches
885# problems on multi-arch where people try to build
886# 32-bit QEMU while pointing at 64-bit glib headers
887
888if not cc.compiles('''
889  #include <glib.h>
890  #include <unistd.h>
891
892  #define QEMU_BUILD_BUG_ON(x) \
893  typedef char qemu_build_bug_on[(x)?-1:1] __attribute__((unused));
894
895  int main(void) {
896     QEMU_BUILD_BUG_ON(sizeof(size_t) != GLIB_SIZEOF_SIZE_T);
897     return 0;
898  }''', dependencies: glib_pc, args: glib_cflags)
899  error('''sizeof(size_t) doesn't match GLIB_SIZEOF_SIZE_T.
900        You probably need to set PKG_CONFIG_LIBDIR" to point
901        to the right pkg-config files for your build target.''')
902endif
903
904# Silence clang warnings triggered by glib < 2.57.2
905if not cc.compiles('''
906  #include <glib.h>
907  typedef struct Foo {
908    int i;
909  } Foo;
910  static void foo_free(Foo *f)
911  {
912    g_free(f);
913  }
914  G_DEFINE_AUTOPTR_CLEANUP_FUNC(Foo, foo_free)
915  int main(void) { return 0; }''', dependencies: glib_pc, args: ['-Wunused-function', '-Werror'])
916  glib_cflags += cc.get_supported_arguments('-Wno-unused-function')
917endif
918glib = declare_dependency(dependencies: [glib_pc, gmodule],
919                          compile_args: glib_cflags,
920                          version: glib_pc.version())
921
922# Check whether glib has gslice, which we have to avoid for correctness.
923# TODO: remove this check and the corresponding workaround (qtree) when
924# the minimum supported glib is >= 2.75.3
925glib_has_gslice = glib.version().version_compare('<2.75.3')
926
927# override glib dep to include the above refinements
928meson.override_dependency('glib-2.0', glib)
929
930# The path to glib.h is added to all compilation commands.
931add_project_dependencies(glib.partial_dependency(compile_args: true, includes: true),
932                         native: false, language: all_languages)
933
934gio = not_found
935gdbus_codegen = not_found
936gdbus_codegen_error = '@0@ requires gdbus-codegen, please install libgio'
937if not get_option('gio').auto() or have_system
938  gio = dependency('gio-2.0', required: get_option('gio'),
939                   method: 'pkg-config')
940  if gio.found() and not cc.links('''
941    #include <gio/gio.h>
942    int main(void)
943    {
944      g_dbus_proxy_new_sync(0, 0, 0, 0, 0, 0, 0, 0);
945      return 0;
946    }''', dependencies: [glib, gio])
947    if get_option('gio').enabled()
948      error('The installed libgio is broken for static linking')
949    endif
950    gio = not_found
951  endif
952  if gio.found()
953    gdbus_codegen = find_program(gio.get_variable('gdbus_codegen'),
954                                 required: get_option('gio'))
955    gio_unix = dependency('gio-unix-2.0', required: get_option('gio'),
956                          method: 'pkg-config')
957    gio = declare_dependency(dependencies: [gio, gio_unix],
958                             version: gio.version())
959  endif
960endif
961if gdbus_codegen.found() and get_option('cfi')
962  gdbus_codegen = not_found
963  gdbus_codegen_error = '@0@ uses gdbus-codegen, which does not support control flow integrity'
964endif
965
966xml_pp = find_program('scripts/xml-preprocess.py')
967
968lttng = not_found
969if 'ust' in get_option('trace_backends')
970  lttng = dependency('lttng-ust', required: true, version: '>= 2.1',
971                     method: 'pkg-config')
972endif
973pixman = not_found
974if not get_option('pixman').auto() or have_system or have_tools
975  pixman = dependency('pixman-1', required: get_option('pixman'), version:'>=0.21.8',
976                      method: 'pkg-config')
977endif
978
979zlib = dependency('zlib', required: true)
980
981libaio = not_found
982if not get_option('linux_aio').auto() or have_block
983  libaio = cc.find_library('aio', has_headers: ['libaio.h'],
984                           required: get_option('linux_aio'))
985endif
986
987linux_io_uring_test = '''
988  #include <liburing.h>
989  #include <linux/errqueue.h>
990
991  int main(void) { return 0; }'''
992
993linux_io_uring = not_found
994if not get_option('linux_io_uring').auto() or have_block
995  linux_io_uring = dependency('liburing', version: '>=0.3',
996                              required: get_option('linux_io_uring'),
997                              method: 'pkg-config')
998  if not cc.links(linux_io_uring_test)
999    linux_io_uring = not_found
1000  endif
1001endif
1002
1003libnfs = not_found
1004if not get_option('libnfs').auto() or have_block
1005  libnfs = dependency('libnfs', version: '>=1.9.3',
1006                      required: get_option('libnfs'),
1007                      method: 'pkg-config')
1008endif
1009
1010libattr_test = '''
1011  #include <stddef.h>
1012  #include <sys/types.h>
1013  #ifdef CONFIG_LIBATTR
1014  #include <attr/xattr.h>
1015  #else
1016  #include <sys/xattr.h>
1017  #endif
1018  int main(void) { getxattr(NULL, NULL, NULL, 0); setxattr(NULL, NULL, NULL, 0, 0); return 0; }'''
1019
1020libattr = not_found
1021have_old_libattr = false
1022if get_option('attr').allowed()
1023  if cc.links(libattr_test)
1024    libattr = declare_dependency()
1025  else
1026    libattr = cc.find_library('attr', has_headers: ['attr/xattr.h'],
1027                              required: get_option('attr'))
1028    if libattr.found() and not \
1029      cc.links(libattr_test, dependencies: libattr, args: '-DCONFIG_LIBATTR')
1030      libattr = not_found
1031      if get_option('attr').enabled()
1032        error('could not link libattr')
1033      else
1034        warning('could not link libattr, disabling')
1035      endif
1036    else
1037      have_old_libattr = libattr.found()
1038    endif
1039  endif
1040endif
1041
1042cocoa = dependency('appleframeworks', modules: ['Cocoa', 'CoreVideo'],
1043                   required: get_option('cocoa'))
1044
1045vmnet = dependency('appleframeworks', modules: 'vmnet', required: get_option('vmnet'))
1046if vmnet.found() and not cc.has_header_symbol('vmnet/vmnet.h',
1047                                              'VMNET_BRIDGED_MODE',
1048                                              dependencies: vmnet)
1049  vmnet = not_found
1050  if get_option('vmnet').enabled()
1051    error('vmnet.framework API is outdated')
1052  else
1053    warning('vmnet.framework API is outdated, disabling')
1054  endif
1055endif
1056
1057seccomp = not_found
1058seccomp_has_sysrawrc = false
1059if not get_option('seccomp').auto() or have_system or have_tools
1060  seccomp = dependency('libseccomp', version: '>=2.3.0',
1061                       required: get_option('seccomp'),
1062                       method: 'pkg-config')
1063  if seccomp.found()
1064    seccomp_has_sysrawrc = cc.has_header_symbol('seccomp.h',
1065                                                'SCMP_FLTATR_API_SYSRAWRC',
1066                                                dependencies: seccomp)
1067  endif
1068endif
1069
1070libcap_ng = not_found
1071if not get_option('cap_ng').auto() or have_system or have_tools
1072  libcap_ng = cc.find_library('cap-ng', has_headers: ['cap-ng.h'],
1073                              required: get_option('cap_ng'))
1074endif
1075if libcap_ng.found() and not cc.links('''
1076   #include <cap-ng.h>
1077   int main(void)
1078   {
1079     capng_capability_to_name(CAPNG_EFFECTIVE);
1080     return 0;
1081   }''', dependencies: libcap_ng)
1082  libcap_ng = not_found
1083  if get_option('cap_ng').enabled()
1084    error('could not link libcap-ng')
1085  else
1086    warning('could not link libcap-ng, disabling')
1087  endif
1088endif
1089
1090if get_option('xkbcommon').auto() and not have_system and not have_tools
1091  xkbcommon = not_found
1092else
1093  xkbcommon = dependency('xkbcommon', required: get_option('xkbcommon'),
1094                         method: 'pkg-config')
1095endif
1096
1097slirp = not_found
1098if not get_option('slirp').auto() or have_system
1099  slirp = dependency('slirp', required: get_option('slirp'),
1100                     method: 'pkg-config')
1101  # slirp < 4.7 is incompatible with CFI support in QEMU.  This is because
1102  # it passes function pointers within libslirp as callbacks for timers.
1103  # When using a system-wide shared libslirp, the type information for the
1104  # callback is missing and the timer call produces a false positive with CFI.
1105  # Do not use the "version" keyword argument to produce a better error.
1106  # with control-flow integrity.
1107  if get_option('cfi') and slirp.found() and slirp.version().version_compare('<4.7')
1108    if get_option('slirp').enabled()
1109      error('Control-Flow Integrity requires libslirp 4.7.')
1110    else
1111      warning('Cannot use libslirp since Control-Flow Integrity requires libslirp >= 4.7.')
1112      slirp = not_found
1113    endif
1114  endif
1115endif
1116
1117vde = not_found
1118if not get_option('vde').auto() or have_system or have_tools
1119  vde = cc.find_library('vdeplug', has_headers: ['libvdeplug.h'],
1120                           required: get_option('vde'))
1121endif
1122if vde.found() and not cc.links('''
1123   #include <libvdeplug.h>
1124   int main(void)
1125   {
1126     struct vde_open_args a = {0, 0, 0};
1127     char s[] = "";
1128     vde_open(s, s, &a);
1129     return 0;
1130   }''', dependencies: vde)
1131  vde = not_found
1132  if get_option('cap_ng').enabled()
1133    error('could not link libvdeplug')
1134  else
1135    warning('could not link libvdeplug, disabling')
1136  endif
1137endif
1138
1139pulse = not_found
1140if not get_option('pa').auto() or (host_os == 'linux' and have_system)
1141  pulse = dependency('libpulse', required: get_option('pa'),
1142                     method: 'pkg-config')
1143endif
1144alsa = not_found
1145if not get_option('alsa').auto() or (host_os == 'linux' and have_system)
1146  alsa = dependency('alsa', required: get_option('alsa'),
1147                    method: 'pkg-config')
1148endif
1149jack = not_found
1150if not get_option('jack').auto() or have_system
1151  jack = dependency('jack', required: get_option('jack'),
1152                    method: 'pkg-config')
1153endif
1154pipewire = not_found
1155if not get_option('pipewire').auto() or (host_os == 'linux' and have_system)
1156  pipewire = dependency('libpipewire-0.3', version: '>=0.3.60',
1157                    required: get_option('pipewire'),
1158                    method: 'pkg-config')
1159endif
1160sndio = not_found
1161if not get_option('sndio').auto() or have_system
1162  sndio = dependency('sndio', required: get_option('sndio'),
1163                    method: 'pkg-config')
1164endif
1165
1166spice_protocol = not_found
1167if not get_option('spice_protocol').auto() or have_system
1168  spice_protocol = dependency('spice-protocol', version: '>=0.14.0',
1169                              required: get_option('spice_protocol'),
1170                              method: 'pkg-config')
1171endif
1172spice = not_found
1173if get_option('spice') \
1174             .disable_auto_if(not have_system) \
1175             .require(pixman.found(),
1176                      error_message: 'cannot enable SPICE if pixman is not available') \
1177             .allowed()
1178  spice = dependency('spice-server', version: '>=0.14.0',
1179                     required: get_option('spice'),
1180                     method: 'pkg-config')
1181endif
1182spice_headers = spice.partial_dependency(compile_args: true, includes: true)
1183
1184rt = cc.find_library('rt', required: false)
1185
1186libiscsi = not_found
1187if not get_option('libiscsi').auto() or have_block
1188  libiscsi = dependency('libiscsi', version: '>=1.9.0',
1189                         required: get_option('libiscsi'),
1190                         method: 'pkg-config')
1191endif
1192zstd = not_found
1193if not get_option('zstd').auto() or have_block
1194  zstd = dependency('libzstd', version: '>=1.4.0',
1195                    required: get_option('zstd'),
1196                    method: 'pkg-config')
1197endif
1198virgl = not_found
1199
1200have_vhost_user_gpu = have_tools and host_os == 'linux' and pixman.found()
1201if not get_option('virglrenderer').auto() or have_system or have_vhost_user_gpu
1202  virgl = dependency('virglrenderer',
1203                     method: 'pkg-config',
1204                     required: get_option('virglrenderer'))
1205endif
1206rutabaga = not_found
1207if not get_option('rutabaga_gfx').auto() or have_system or have_vhost_user_gpu
1208  rutabaga = dependency('rutabaga_gfx_ffi',
1209                         method: 'pkg-config',
1210                         required: get_option('rutabaga_gfx'))
1211endif
1212blkio = not_found
1213if not get_option('blkio').auto() or have_block
1214  blkio = dependency('blkio',
1215                     method: 'pkg-config',
1216                     required: get_option('blkio'))
1217endif
1218curl = not_found
1219if not get_option('curl').auto() or have_block
1220  curl = dependency('libcurl', version: '>=7.29.0',
1221                    method: 'pkg-config',
1222                    required: get_option('curl'))
1223endif
1224libudev = not_found
1225if host_os == 'linux' and (have_system or have_tools)
1226  libudev = dependency('libudev',
1227                       method: 'pkg-config',
1228                       required: get_option('libudev'))
1229endif
1230
1231mpathlibs = [libudev]
1232mpathpersist = not_found
1233if host_os == 'linux' and have_tools and get_option('mpath').allowed()
1234  mpath_test_source = '''
1235    #include <libudev.h>
1236    #include <mpath_persist.h>
1237    unsigned mpath_mx_alloc_len = 1024;
1238    int logsink;
1239    static struct config *multipath_conf;
1240    extern struct udev *udev;
1241    extern struct config *get_multipath_config(void);
1242    extern void put_multipath_config(struct config *conf);
1243    struct udev *udev;
1244    struct config *get_multipath_config(void) { return multipath_conf; }
1245    void put_multipath_config(struct config *conf) { }
1246    int main(void) {
1247        udev = udev_new();
1248        multipath_conf = mpath_lib_init();
1249        return 0;
1250    }'''
1251  libmpathpersist = cc.find_library('mpathpersist',
1252                                    required: get_option('mpath'))
1253  if libmpathpersist.found()
1254    mpathlibs += libmpathpersist
1255    if get_option('prefer_static')
1256      mpathlibs += cc.find_library('devmapper',
1257                                     required: get_option('mpath'))
1258    endif
1259    mpathlibs += cc.find_library('multipath',
1260                                 required: get_option('mpath'))
1261    foreach lib: mpathlibs
1262      if not lib.found()
1263        mpathlibs = []
1264        break
1265      endif
1266    endforeach
1267    if mpathlibs.length() == 0
1268      msg = 'Dependencies missing for libmpathpersist'
1269    elif cc.links(mpath_test_source, dependencies: mpathlibs)
1270      mpathpersist = declare_dependency(dependencies: mpathlibs)
1271    else
1272      msg = 'Cannot detect libmpathpersist API'
1273    endif
1274    if not mpathpersist.found()
1275      if get_option('mpath').enabled()
1276        error(msg)
1277      else
1278        warning(msg + ', disabling')
1279      endif
1280    endif
1281  endif
1282endif
1283
1284iconv = not_found
1285curses = not_found
1286if have_system and get_option('curses').allowed()
1287  curses_test = '''
1288    #if defined(__APPLE__) || defined(__OpenBSD__)
1289    #define _XOPEN_SOURCE_EXTENDED 1
1290    #endif
1291    #include <locale.h>
1292    #include <curses.h>
1293    #include <wchar.h>
1294    int main(void) {
1295      wchar_t wch = L'w';
1296      setlocale(LC_ALL, "");
1297      resize_term(0, 0);
1298      addwstr(L"wide chars\n");
1299      addnwstr(&wch, 1);
1300      add_wch(WACS_DEGREE);
1301      return 0;
1302    }'''
1303
1304  curses_dep_list = host_os == 'windows' ? ['ncurses', 'ncursesw'] : ['ncursesw']
1305  curses = dependency(curses_dep_list,
1306                      required: false,
1307                      method: 'pkg-config')
1308  msg = get_option('curses').enabled() ? 'curses library not found' : ''
1309  curses_compile_args = ['-DNCURSES_WIDECHAR=1']
1310  if curses.found()
1311    if cc.links(curses_test, args: curses_compile_args, dependencies: [curses])
1312      curses = declare_dependency(compile_args: curses_compile_args, dependencies: [curses],
1313                                  version: curses.version())
1314    else
1315      msg = 'curses package not usable'
1316      curses = not_found
1317    endif
1318  endif
1319  if not curses.found()
1320    has_curses_h = cc.has_header('curses.h', args: curses_compile_args)
1321    if host_os != 'windows' and not has_curses_h
1322      message('Trying with /usr/include/ncursesw')
1323      curses_compile_args += ['-I/usr/include/ncursesw']
1324      has_curses_h = cc.has_header('curses.h', args: curses_compile_args)
1325    endif
1326    if has_curses_h
1327      curses_libname_list = (host_os == 'windows' ? ['pdcurses'] : ['ncursesw', 'cursesw'])
1328      foreach curses_libname : curses_libname_list
1329        libcurses = cc.find_library(curses_libname,
1330                                    required: false)
1331        if libcurses.found()
1332          if cc.links(curses_test, args: curses_compile_args, dependencies: libcurses)
1333            curses = declare_dependency(compile_args: curses_compile_args,
1334                                        dependencies: [libcurses])
1335            break
1336          else
1337            msg = 'curses library not usable'
1338          endif
1339        endif
1340      endforeach
1341    endif
1342  endif
1343  if get_option('iconv').allowed()
1344    foreach link_args : [ ['-liconv'], [] ]
1345      # Programs will be linked with glib and this will bring in libiconv on FreeBSD.
1346      # We need to use libiconv if available because mixing libiconv's headers with
1347      # the system libc does not work.
1348      # However, without adding glib to the dependencies -L/usr/local/lib will not be
1349      # included in the command line and libiconv will not be found.
1350      if cc.links('''
1351        #include <iconv.h>
1352        int main(void) {
1353          iconv_t conv = iconv_open("WCHAR_T", "UCS-2");
1354          return conv != (iconv_t) -1;
1355        }''', args: link_args, dependencies: glib)
1356        iconv = declare_dependency(link_args: link_args, dependencies: glib)
1357        break
1358      endif
1359    endforeach
1360  endif
1361  if curses.found() and not iconv.found()
1362    if get_option('iconv').enabled()
1363      error('iconv not available')
1364    endif
1365    msg = 'iconv required for curses UI but not available'
1366    curses = not_found
1367  endif
1368  if not curses.found() and msg != ''
1369    if get_option('curses').enabled()
1370      error(msg)
1371    else
1372      warning(msg + ', disabling')
1373    endif
1374  endif
1375endif
1376
1377brlapi = not_found
1378if not get_option('brlapi').auto() or have_system
1379  brlapi = cc.find_library('brlapi', has_headers: ['brlapi.h'],
1380                         required: get_option('brlapi'))
1381  if brlapi.found() and not cc.links('''
1382     #include <brlapi.h>
1383     #include <stddef.h>
1384     int main(void) { return brlapi__openConnection (NULL, NULL, NULL); }''', dependencies: brlapi)
1385    brlapi = not_found
1386    if get_option('brlapi').enabled()
1387      error('could not link brlapi')
1388    else
1389      warning('could not link brlapi, disabling')
1390    endif
1391  endif
1392endif
1393
1394sdl = not_found
1395if not get_option('sdl').auto() or have_system
1396  sdl = dependency('sdl2', required: get_option('sdl'))
1397  sdl_image = not_found
1398endif
1399if sdl.found()
1400  # Some versions of SDL have problems with -Wundef
1401  if not cc.compiles('''
1402                     #include <SDL.h>
1403                     #include <SDL_syswm.h>
1404                     int main(int argc, char *argv[]) { return 0; }
1405                     ''', dependencies: sdl, args: '-Werror=undef')
1406    sdl = declare_dependency(compile_args: '-Wno-undef',
1407                             dependencies: sdl,
1408                             version: sdl.version())
1409  endif
1410  sdl_image = dependency('SDL2_image', required: get_option('sdl_image'),
1411                         method: 'pkg-config')
1412else
1413  if get_option('sdl_image').enabled()
1414    error('sdl-image required, but SDL was @0@'.format(
1415          get_option('sdl').disabled() ? 'disabled' : 'not found'))
1416  endif
1417  sdl_image = not_found
1418endif
1419
1420rbd = not_found
1421if not get_option('rbd').auto() or have_block
1422  librados = cc.find_library('rados', required: get_option('rbd'))
1423  librbd = cc.find_library('rbd', has_headers: ['rbd/librbd.h'],
1424                           required: get_option('rbd'))
1425  if librados.found() and librbd.found()
1426    if cc.links('''
1427      #include <stdio.h>
1428      #include <rbd/librbd.h>
1429      int main(void) {
1430        rados_t cluster;
1431        rados_create(&cluster, NULL);
1432        #if LIBRBD_VERSION_CODE < LIBRBD_VERSION(1, 12, 0)
1433        #error
1434        #endif
1435        return 0;
1436      }''', dependencies: [librbd, librados])
1437      rbd = declare_dependency(dependencies: [librbd, librados])
1438    elif get_option('rbd').enabled()
1439      error('librbd >= 1.12.0 required')
1440    else
1441      warning('librbd >= 1.12.0 not found, disabling')
1442    endif
1443  endif
1444endif
1445
1446glusterfs = not_found
1447glusterfs_ftruncate_has_stat = false
1448glusterfs_iocb_has_stat = false
1449if not get_option('glusterfs').auto() or have_block
1450  glusterfs = dependency('glusterfs-api', version: '>=3',
1451                         required: get_option('glusterfs'),
1452                         method: 'pkg-config')
1453  if glusterfs.found()
1454    glusterfs_ftruncate_has_stat = cc.links('''
1455      #include <glusterfs/api/glfs.h>
1456
1457      int
1458      main(void)
1459      {
1460          /* new glfs_ftruncate() passes two additional args */
1461          return glfs_ftruncate(NULL, 0, NULL, NULL);
1462      }
1463    ''', dependencies: glusterfs)
1464    glusterfs_iocb_has_stat = cc.links('''
1465      #include <glusterfs/api/glfs.h>
1466
1467      /* new glfs_io_cbk() passes two additional glfs_stat structs */
1468      static void
1469      glusterfs_iocb(glfs_fd_t *fd, ssize_t ret, struct glfs_stat *prestat, struct glfs_stat *poststat, void *data)
1470      {}
1471
1472      int
1473      main(void)
1474      {
1475          glfs_io_cbk iocb = &glusterfs_iocb;
1476          iocb(NULL, 0 , NULL, NULL, NULL);
1477          return 0;
1478      }
1479    ''', dependencies: glusterfs)
1480  endif
1481endif
1482
1483hv_balloon = false
1484if get_option('hv_balloon').allowed() and have_system
1485  if cc.links('''
1486    #include <string.h>
1487    #include <gmodule.h>
1488    int main(void) {
1489        GTree *tree;
1490
1491        tree = g_tree_new((GCompareFunc)strcmp);
1492        (void)g_tree_node_first(tree);
1493        g_tree_destroy(tree);
1494        return 0;
1495    }
1496  ''', dependencies: glib)
1497    hv_balloon = true
1498  else
1499    if get_option('hv_balloon').enabled()
1500      error('could not enable hv-balloon, update your glib')
1501    else
1502      warning('could not find glib support for hv-balloon, disabling')
1503    endif
1504  endif
1505endif
1506
1507libssh = not_found
1508if not get_option('libssh').auto() or have_block
1509  libssh = dependency('libssh', version: '>=0.8.7',
1510                    method: 'pkg-config',
1511                    required: get_option('libssh'))
1512endif
1513
1514libbzip2 = not_found
1515if not get_option('bzip2').auto() or have_block
1516  libbzip2 = cc.find_library('bz2', has_headers: ['bzlib.h'],
1517                             required: get_option('bzip2'))
1518  if libbzip2.found() and not cc.links('''
1519     #include <bzlib.h>
1520     int main(void) { BZ2_bzlibVersion(); return 0; }''', dependencies: libbzip2)
1521    libbzip2 = not_found
1522    if get_option('bzip2').enabled()
1523      error('could not link libbzip2')
1524    else
1525      warning('could not link libbzip2, disabling')
1526    endif
1527  endif
1528endif
1529
1530liblzfse = not_found
1531if not get_option('lzfse').auto() or have_block
1532  liblzfse = cc.find_library('lzfse', has_headers: ['lzfse.h'],
1533                             required: get_option('lzfse'))
1534endif
1535if liblzfse.found() and not cc.links('''
1536   #include <lzfse.h>
1537   int main(void) { lzfse_decode_scratch_size(); return 0; }''', dependencies: liblzfse)
1538  liblzfse = not_found
1539  if get_option('lzfse').enabled()
1540    error('could not link liblzfse')
1541  else
1542    warning('could not link liblzfse, disabling')
1543  endif
1544endif
1545
1546oss = not_found
1547if get_option('oss').allowed() and have_system
1548  if not cc.has_header('sys/soundcard.h')
1549    # not found
1550  elif host_os == 'netbsd'
1551    oss = cc.find_library('ossaudio', required: get_option('oss'))
1552  else
1553    oss = declare_dependency()
1554  endif
1555
1556  if not oss.found()
1557    if get_option('oss').enabled()
1558      error('OSS not found')
1559    endif
1560  endif
1561endif
1562dsound = not_found
1563if not get_option('dsound').auto() or (host_os == 'windows' and have_system)
1564  if cc.has_header('dsound.h')
1565    dsound = declare_dependency(link_args: ['-lole32', '-ldxguid'])
1566  endif
1567
1568  if not dsound.found()
1569    if get_option('dsound').enabled()
1570      error('DirectSound not found')
1571    endif
1572  endif
1573endif
1574
1575coreaudio = not_found
1576if not get_option('coreaudio').auto() or (host_os == 'darwin' and have_system)
1577  coreaudio = dependency('appleframeworks', modules: 'CoreAudio',
1578                         required: get_option('coreaudio'))
1579endif
1580
1581opengl = not_found
1582if not get_option('opengl').auto() or have_system or have_vhost_user_gpu
1583  epoxy = dependency('epoxy', method: 'pkg-config',
1584                      required: get_option('opengl'))
1585  if cc.has_header('epoxy/egl.h', dependencies: epoxy)
1586    opengl = epoxy
1587  elif get_option('opengl').enabled()
1588    error('epoxy/egl.h not found')
1589  endif
1590endif
1591gbm = not_found
1592if (have_system or have_tools) and (virgl.found() or opengl.found())
1593  gbm = dependency('gbm', method: 'pkg-config', required: false)
1594endif
1595have_vhost_user_gpu = have_vhost_user_gpu and virgl.found() and opengl.found() and gbm.found()
1596
1597gnutls = not_found
1598gnutls_crypto = not_found
1599if get_option('gnutls').enabled() or (get_option('gnutls').auto() and have_system)
1600  # For general TLS support our min gnutls matches
1601  # that implied by our platform support matrix
1602  #
1603  # For the crypto backends, we look for a newer
1604  # gnutls:
1605  #
1606  #   Version 3.6.8  is needed to get XTS
1607  #   Version 3.6.13 is needed to get PBKDF
1608  #   Version 3.6.14 is needed to get HW accelerated XTS
1609  #
1610  # If newer enough gnutls isn't available, we can
1611  # still use a different crypto backend to satisfy
1612  # the platform support requirements
1613  gnutls_crypto = dependency('gnutls', version: '>=3.6.14',
1614                             method: 'pkg-config',
1615                             required: false)
1616  if gnutls_crypto.found()
1617    gnutls = gnutls_crypto
1618  else
1619    # Our min version if all we need is TLS
1620    gnutls = dependency('gnutls', version: '>=3.5.18',
1621                        method: 'pkg-config',
1622                        required: get_option('gnutls'))
1623  endif
1624endif
1625
1626# We prefer use of gnutls for crypto, unless the options
1627# explicitly asked for nettle or gcrypt.
1628#
1629# If gnutls isn't available for crypto, then we'll prefer
1630# gcrypt over nettle for performance reasons.
1631gcrypt = not_found
1632nettle = not_found
1633hogweed = not_found
1634xts = 'none'
1635
1636if get_option('nettle').enabled() and get_option('gcrypt').enabled()
1637  error('Only one of gcrypt & nettle can be enabled')
1638endif
1639
1640# Explicit nettle/gcrypt request, so ignore gnutls for crypto
1641if get_option('nettle').enabled() or get_option('gcrypt').enabled()
1642  gnutls_crypto = not_found
1643endif
1644
1645if not gnutls_crypto.found()
1646  if (not get_option('gcrypt').auto() or have_system) and not get_option('nettle').enabled()
1647    gcrypt = dependency('libgcrypt', version: '>=1.8',
1648                        method: 'config-tool',
1649                        required: get_option('gcrypt'))
1650    # Debian has removed -lgpg-error from libgcrypt-config
1651    # as it "spreads unnecessary dependencies" which in
1652    # turn breaks static builds...
1653    if gcrypt.found() and get_option('prefer_static')
1654      gcrypt = declare_dependency(dependencies:
1655        [gcrypt,
1656         cc.find_library('gpg-error', required: true)],
1657        version: gcrypt.version())
1658    endif
1659  endif
1660  if (not get_option('nettle').auto() or have_system) and not gcrypt.found()
1661    nettle = dependency('nettle', version: '>=3.4',
1662                        method: 'pkg-config',
1663                        required: get_option('nettle'))
1664    if nettle.found() and not cc.has_header('nettle/xts.h', dependencies: nettle)
1665      xts = 'private'
1666    endif
1667  endif
1668endif
1669
1670capstone = not_found
1671if not get_option('capstone').auto() or have_system or have_user
1672  capstone = dependency('capstone', version: '>=3.0.5',
1673                        method: 'pkg-config',
1674                        required: get_option('capstone'))
1675
1676  # Some versions of capstone have broken pkg-config file
1677  # that reports a wrong -I path, causing the #include to
1678  # fail later. If the system has such a broken version
1679  # do not use it.
1680  if capstone.found() and not cc.compiles('#include <capstone.h>',
1681                                          dependencies: [capstone])
1682    capstone = not_found
1683    if get_option('capstone').enabled()
1684      error('capstone requested, but it does not appear to work')
1685    endif
1686  endif
1687endif
1688
1689gmp = dependency('gmp', required: false, method: 'pkg-config')
1690if nettle.found() and gmp.found()
1691  hogweed = dependency('hogweed', version: '>=3.4',
1692                       method: 'pkg-config',
1693                       required: get_option('nettle'))
1694endif
1695
1696
1697gtk = not_found
1698gtkx11 = not_found
1699vte = not_found
1700have_gtk_clipboard = get_option('gtk_clipboard').enabled()
1701
1702if get_option('gtk') \
1703             .disable_auto_if(not have_system) \
1704             .require(pixman.found(),
1705                      error_message: 'cannot enable GTK if pixman is not available') \
1706             .allowed()
1707  gtk = dependency('gtk+-3.0', version: '>=3.22.0',
1708                   method: 'pkg-config',
1709                   required: get_option('gtk'))
1710  if gtk.found()
1711    gtkx11 = dependency('gtk+-x11-3.0', version: '>=3.22.0',
1712                        method: 'pkg-config',
1713                        required: false)
1714    gtk = declare_dependency(dependencies: [gtk, gtkx11],
1715                             version: gtk.version())
1716
1717    if not get_option('vte').auto() or have_system
1718      vte = dependency('vte-2.91',
1719                       method: 'pkg-config',
1720                       required: get_option('vte'))
1721    endif
1722  elif have_gtk_clipboard
1723    error('GTK clipboard requested, but GTK not found')
1724  endif
1725endif
1726
1727x11 = not_found
1728if gtkx11.found()
1729  x11 = dependency('x11', method: 'pkg-config', required: gtkx11.found())
1730endif
1731png = not_found
1732if get_option('png').allowed() and have_system
1733   png = dependency('libpng', version: '>=1.6.34', required: get_option('png'),
1734                    method: 'pkg-config')
1735endif
1736vnc = not_found
1737jpeg = not_found
1738sasl = not_found
1739if get_option('vnc') \
1740             .disable_auto_if(not have_system) \
1741             .require(pixman.found(),
1742                      error_message: 'cannot enable VNC if pixman is not available') \
1743             .allowed()
1744  vnc = declare_dependency() # dummy dependency
1745  jpeg = dependency('libjpeg', required: get_option('vnc_jpeg'),
1746                    method: 'pkg-config')
1747  sasl = cc.find_library('sasl2', has_headers: ['sasl/sasl.h'],
1748                         required: get_option('vnc_sasl'))
1749  if sasl.found()
1750    sasl = declare_dependency(dependencies: sasl,
1751                              compile_args: '-DSTRUCT_IOVEC_DEFINED')
1752  endif
1753endif
1754
1755pam = not_found
1756if not get_option('auth_pam').auto() or have_system
1757  pam = cc.find_library('pam', has_headers: ['security/pam_appl.h'],
1758                        required: get_option('auth_pam'))
1759endif
1760if pam.found() and not cc.links('''
1761   #include <stddef.h>
1762   #include <security/pam_appl.h>
1763   int main(void) {
1764     const char *service_name = "qemu";
1765     const char *user = "frank";
1766     const struct pam_conv pam_conv = { 0 };
1767     pam_handle_t *pamh = NULL;
1768     pam_start(service_name, user, &pam_conv, &pamh);
1769     return 0;
1770   }''', dependencies: pam)
1771  pam = not_found
1772  if get_option('auth_pam').enabled()
1773    error('could not link libpam')
1774  else
1775    warning('could not link libpam, disabling')
1776  endif
1777endif
1778
1779snappy = not_found
1780if not get_option('snappy').auto() or have_system
1781  snappy = cc.find_library('snappy', has_headers: ['snappy-c.h'],
1782                           required: get_option('snappy'))
1783endif
1784if snappy.found() and not cc.links('''
1785   #include <snappy-c.h>
1786   int main(void) { snappy_max_compressed_length(4096); return 0; }''', dependencies: snappy)
1787  snappy = not_found
1788  if get_option('snappy').enabled()
1789    error('could not link libsnappy')
1790  else
1791    warning('could not link libsnappy, disabling')
1792  endif
1793endif
1794
1795lzo = not_found
1796if not get_option('lzo').auto() or have_system
1797  lzo = cc.find_library('lzo2', has_headers: ['lzo/lzo1x.h'],
1798                        required: get_option('lzo'))
1799endif
1800if lzo.found() and not cc.links('''
1801   #include <lzo/lzo1x.h>
1802   int main(void) { lzo_version(); return 0; }''', dependencies: lzo)
1803  lzo = not_found
1804  if get_option('lzo').enabled()
1805    error('could not link liblzo2')
1806  else
1807    warning('could not link liblzo2, disabling')
1808  endif
1809endif
1810
1811numa = not_found
1812if not get_option('numa').auto() or have_system or have_tools
1813  numa = cc.find_library('numa', has_headers: ['numa.h'],
1814                              required: get_option('numa'))
1815endif
1816if numa.found() and not cc.links('''
1817   #include <numa.h>
1818   int main(void) { return numa_available(); }
1819   ''', dependencies: numa)
1820  numa = not_found
1821  if get_option('numa').enabled()
1822    error('could not link numa')
1823  else
1824    warning('could not link numa, disabling')
1825  endif
1826endif
1827
1828rdma = not_found
1829if not get_option('rdma').auto() or have_system
1830  libumad = cc.find_library('ibumad', required: get_option('rdma'))
1831  rdma_libs = [cc.find_library('rdmacm', has_headers: ['rdma/rdma_cma.h'],
1832                               required: get_option('rdma')),
1833               cc.find_library('ibverbs', required: get_option('rdma')),
1834               libumad]
1835  rdma = declare_dependency(dependencies: rdma_libs)
1836  foreach lib: rdma_libs
1837    if not lib.found()
1838      rdma = not_found
1839    endif
1840  endforeach
1841endif
1842
1843cacard = not_found
1844if not get_option('smartcard').auto() or have_system
1845  cacard = dependency('libcacard', required: get_option('smartcard'),
1846                      version: '>=2.5.1', method: 'pkg-config')
1847endif
1848u2f = not_found
1849if not get_option('u2f').auto() or have_system
1850  u2f = dependency('u2f-emu', required: get_option('u2f'),
1851                   method: 'pkg-config')
1852endif
1853canokey = not_found
1854if not get_option('canokey').auto() or have_system
1855  canokey = dependency('canokey-qemu', required: get_option('canokey'),
1856                   method: 'pkg-config')
1857endif
1858usbredir = not_found
1859if not get_option('usb_redir').auto() or have_system
1860  usbredir = dependency('libusbredirparser-0.5', required: get_option('usb_redir'),
1861                        version: '>=0.6', method: 'pkg-config')
1862endif
1863libusb = not_found
1864if not get_option('libusb').auto() or have_system
1865  libusb = dependency('libusb-1.0', required: get_option('libusb'),
1866                      version: '>=1.0.13', method: 'pkg-config')
1867endif
1868
1869libpmem = not_found
1870if not get_option('libpmem').auto() or have_system
1871  libpmem = dependency('libpmem', required: get_option('libpmem'),
1872                       method: 'pkg-config')
1873endif
1874libdaxctl = not_found
1875if not get_option('libdaxctl').auto() or have_system
1876  libdaxctl = dependency('libdaxctl', required: get_option('libdaxctl'),
1877                         version: '>=57', method: 'pkg-config')
1878endif
1879tasn1 = not_found
1880if gnutls.found()
1881  tasn1 = dependency('libtasn1',
1882                     method: 'pkg-config')
1883endif
1884keyutils = not_found
1885if not get_option('libkeyutils').auto() or have_block
1886  keyutils = dependency('libkeyutils', required: get_option('libkeyutils'),
1887                        method: 'pkg-config')
1888endif
1889
1890has_gettid = cc.has_function('gettid')
1891
1892# libselinux
1893selinux = dependency('libselinux',
1894                     required: get_option('selinux'),
1895                     method: 'pkg-config')
1896
1897# Malloc tests
1898
1899malloc = []
1900if get_option('malloc') == 'system'
1901  has_malloc_trim = \
1902    get_option('malloc_trim').allowed() and \
1903    cc.has_function('malloc_trim', prefix: '#include <malloc.h>')
1904else
1905  has_malloc_trim = false
1906  malloc = cc.find_library(get_option('malloc'), required: true)
1907endif
1908if not has_malloc_trim and get_option('malloc_trim').enabled()
1909  if get_option('malloc') == 'system'
1910    error('malloc_trim not available on this platform.')
1911  else
1912    error('malloc_trim not available with non-libc memory allocator')
1913  endif
1914endif
1915
1916gnu_source_prefix = '''
1917  #ifndef _GNU_SOURCE
1918  #define _GNU_SOURCE
1919  #endif
1920'''
1921
1922# Check whether the glibc provides STATX_BASIC_STATS
1923
1924has_statx = cc.has_header_symbol('sys/stat.h', 'STATX_BASIC_STATS', prefix: gnu_source_prefix)
1925
1926# Check whether statx() provides mount ID information
1927
1928has_statx_mnt_id = cc.has_header_symbol('sys/stat.h', 'STATX_MNT_ID', prefix: gnu_source_prefix)
1929
1930have_vhost_user_blk_server = get_option('vhost_user_blk_server') \
1931  .require(host_os == 'linux',
1932           error_message: 'vhost_user_blk_server requires linux') \
1933  .require(have_vhost_user,
1934           error_message: 'vhost_user_blk_server requires vhost-user support') \
1935  .disable_auto_if(not have_tools and not have_system) \
1936  .allowed()
1937
1938if get_option('fuse').disabled() and get_option('fuse_lseek').enabled()
1939  error('Cannot enable fuse-lseek while fuse is disabled')
1940endif
1941
1942fuse = dependency('fuse3', required: get_option('fuse'),
1943                  version: '>=3.1', method: 'pkg-config')
1944
1945fuse_lseek = not_found
1946if get_option('fuse_lseek').allowed()
1947  if fuse.version().version_compare('>=3.8')
1948    # Dummy dependency
1949    fuse_lseek = declare_dependency()
1950  elif get_option('fuse_lseek').enabled()
1951    if fuse.found()
1952      error('fuse-lseek requires libfuse >=3.8, found ' + fuse.version())
1953    else
1954      error('fuse-lseek requires libfuse, which was not found')
1955    endif
1956  endif
1957endif
1958
1959have_libvduse = (host_os == 'linux')
1960if get_option('libvduse').enabled()
1961    if host_os != 'linux'
1962        error('libvduse requires linux')
1963    endif
1964elif get_option('libvduse').disabled()
1965    have_libvduse = false
1966endif
1967
1968have_vduse_blk_export = (have_libvduse and host_os == 'linux')
1969if get_option('vduse_blk_export').enabled()
1970    if host_os != 'linux'
1971        error('vduse_blk_export requires linux')
1972    elif not have_libvduse
1973        error('vduse_blk_export requires libvduse support')
1974    endif
1975elif get_option('vduse_blk_export').disabled()
1976    have_vduse_blk_export = false
1977endif
1978
1979# libbpf
1980libbpf = dependency('libbpf', required: get_option('bpf'), method: 'pkg-config')
1981if libbpf.found() and not cc.links('''
1982   #include <bpf/libbpf.h>
1983   int main(void)
1984   {
1985     bpf_object__destroy_skeleton(NULL);
1986     return 0;
1987   }''', dependencies: libbpf)
1988  libbpf = not_found
1989  if get_option('bpf').enabled()
1990    error('libbpf skeleton test failed')
1991  else
1992    warning('libbpf skeleton test failed, disabling')
1993  endif
1994endif
1995
1996# libxdp
1997libxdp = not_found
1998if not get_option('af_xdp').auto() or have_system
1999    libxdp = dependency('libxdp', required: get_option('af_xdp'),
2000                        version: '>=1.4.0', method: 'pkg-config')
2001endif
2002
2003# libdw
2004libdw = not_found
2005if not get_option('libdw').auto() or \
2006        (not get_option('prefer_static') and (have_system or have_user))
2007    libdw = dependency('libdw',
2008                       method: 'pkg-config',
2009                       required: get_option('libdw'))
2010endif
2011
2012#################
2013# config-host.h #
2014#################
2015
2016config_host_data = configuration_data()
2017
2018audio_drivers_selected = []
2019if have_system
2020  audio_drivers_available = {
2021    'alsa': alsa.found(),
2022    'coreaudio': coreaudio.found(),
2023    'dsound': dsound.found(),
2024    'jack': jack.found(),
2025    'oss': oss.found(),
2026    'pa': pulse.found(),
2027    'pipewire': pipewire.found(),
2028    'sdl': sdl.found(),
2029    'sndio': sndio.found(),
2030  }
2031  foreach k, v: audio_drivers_available
2032    config_host_data.set('CONFIG_AUDIO_' + k.to_upper(), v)
2033  endforeach
2034
2035  # Default to native drivers first, OSS second, SDL third
2036  audio_drivers_priority = \
2037    [ 'pa', 'coreaudio', 'dsound', 'sndio', 'oss' ] + \
2038    (host_os == 'linux' ? [] : [ 'sdl' ])
2039  audio_drivers_default = []
2040  foreach k: audio_drivers_priority
2041    if audio_drivers_available[k]
2042      audio_drivers_default += k
2043    endif
2044  endforeach
2045
2046  foreach k: get_option('audio_drv_list')
2047    if k == 'default'
2048      audio_drivers_selected += audio_drivers_default
2049    elif not audio_drivers_available[k]
2050      error('Audio driver "@0@" not available.'.format(k))
2051    else
2052      audio_drivers_selected += k
2053    endif
2054  endforeach
2055endif
2056config_host_data.set('CONFIG_AUDIO_DRIVERS',
2057                     '"' + '", "'.join(audio_drivers_selected) + '", ')
2058
2059have_host_block_device = (host_os != 'darwin' or
2060    cc.has_header('IOKit/storage/IOMedia.h'))
2061
2062dbus_display = get_option('dbus_display') \
2063  .require(gio.version().version_compare('>=2.64'),
2064           error_message: '-display dbus requires glib>=2.64') \
2065  .require(gdbus_codegen.found(),
2066           error_message: gdbus_codegen_error.format('-display dbus')) \
2067  .allowed()
2068
2069have_virtfs = get_option('virtfs') \
2070    .require(host_os == 'linux' or host_os == 'darwin',
2071             error_message: 'virtio-9p (virtfs) requires Linux or macOS') \
2072    .require(host_os == 'linux' or cc.has_function('pthread_fchdir_np'),
2073             error_message: 'virtio-9p (virtfs) on macOS requires the presence of pthread_fchdir_np') \
2074    .require(host_os == 'darwin' or libattr.found(),
2075             error_message: 'virtio-9p (virtfs) on Linux requires libattr-devel') \
2076    .disable_auto_if(not have_tools and not have_system) \
2077    .allowed()
2078
2079have_virtfs_proxy_helper = get_option('virtfs_proxy_helper') \
2080    .require(host_os != 'darwin', error_message: 'the virtfs proxy helper is incompatible with macOS') \
2081    .require(have_virtfs, error_message: 'the virtfs proxy helper requires that virtfs is enabled') \
2082    .disable_auto_if(not have_tools) \
2083    .require(libcap_ng.found(), error_message: 'the virtfs proxy helper requires libcap-ng') \
2084    .allowed()
2085
2086if get_option('block_drv_ro_whitelist') == ''
2087  config_host_data.set('CONFIG_BDRV_RO_WHITELIST', '')
2088else
2089  config_host_data.set('CONFIG_BDRV_RO_WHITELIST',
2090        '"' + get_option('block_drv_ro_whitelist').replace(',', '", "') + '", ')
2091endif
2092if get_option('block_drv_rw_whitelist') == ''
2093  config_host_data.set('CONFIG_BDRV_RW_WHITELIST', '')
2094else
2095  config_host_data.set('CONFIG_BDRV_RW_WHITELIST',
2096        '"' + get_option('block_drv_rw_whitelist').replace(',', '", "') + '", ')
2097endif
2098
2099foreach k : get_option('trace_backends')
2100  config_host_data.set('CONFIG_TRACE_' + k.to_upper(), true)
2101endforeach
2102config_host_data.set_quoted('CONFIG_TRACE_FILE', get_option('trace_file'))
2103config_host_data.set_quoted('CONFIG_TLS_PRIORITY', get_option('tls_priority'))
2104if iasl.found()
2105  config_host_data.set_quoted('CONFIG_IASL', iasl.full_path())
2106endif
2107config_host_data.set_quoted('CONFIG_BINDIR', get_option('prefix') / get_option('bindir'))
2108config_host_data.set_quoted('CONFIG_PREFIX', get_option('prefix'))
2109config_host_data.set_quoted('CONFIG_QEMU_CONFDIR', get_option('prefix') / qemu_confdir)
2110config_host_data.set_quoted('CONFIG_QEMU_DATADIR', get_option('prefix') / qemu_datadir)
2111config_host_data.set_quoted('CONFIG_QEMU_DESKTOPDIR', get_option('prefix') / qemu_desktopdir)
2112
2113qemu_firmwarepath = ''
2114foreach k : get_option('qemu_firmwarepath')
2115  qemu_firmwarepath += '"' + get_option('prefix') / k + '", '
2116endforeach
2117config_host_data.set('CONFIG_QEMU_FIRMWAREPATH', qemu_firmwarepath)
2118
2119config_host_data.set_quoted('CONFIG_QEMU_HELPERDIR', get_option('prefix') / get_option('libexecdir'))
2120config_host_data.set_quoted('CONFIG_QEMU_ICONDIR', get_option('prefix') / qemu_icondir)
2121config_host_data.set_quoted('CONFIG_QEMU_LOCALEDIR', get_option('prefix') / get_option('localedir'))
2122config_host_data.set_quoted('CONFIG_QEMU_LOCALSTATEDIR', get_option('prefix') / get_option('localstatedir'))
2123config_host_data.set_quoted('CONFIG_QEMU_MODDIR', get_option('prefix') / qemu_moddir)
2124config_host_data.set_quoted('CONFIG_SYSCONFDIR', get_option('prefix') / get_option('sysconfdir'))
2125
2126if enable_modules
2127  config_host_data.set('CONFIG_STAMP', run_command(
2128      meson.current_source_dir() / 'scripts/qemu-stamp.py',
2129      meson.project_version(), get_option('pkgversion'), '--',
2130      meson.current_source_dir() / 'configure',
2131      capture: true, check: true).stdout().strip())
2132endif
2133
2134have_slirp_smbd = get_option('slirp_smbd') \
2135  .require(host_os != 'windows', error_message: 'Host smbd not supported on this platform.') \
2136  .allowed()
2137if have_slirp_smbd
2138  smbd_path = get_option('smbd')
2139  if smbd_path == ''
2140    smbd_path = (host_os == 'sunos' ? '/usr/sfw/sbin/smbd' : '/usr/sbin/smbd')
2141  endif
2142  config_host_data.set_quoted('CONFIG_SMBD_COMMAND', smbd_path)
2143endif
2144
2145config_host_data.set('HOST_' + host_arch.to_upper(), 1)
2146
2147kvm_targets_c = '""'
2148if get_option('kvm').allowed() and host_os == 'linux'
2149  kvm_targets_c = '"' + '" ,"'.join(kvm_targets) + '"'
2150endif
2151config_host_data.set('CONFIG_KVM_TARGETS', kvm_targets_c)
2152
2153if get_option('module_upgrades') and not enable_modules
2154  error('Cannot enable module-upgrades as modules are not enabled')
2155endif
2156config_host_data.set('CONFIG_MODULE_UPGRADES', get_option('module_upgrades'))
2157
2158config_host_data.set('CONFIG_ATTR', libattr.found())
2159config_host_data.set('CONFIG_BDRV_WHITELIST_TOOLS', get_option('block_drv_whitelist_in_tools'))
2160config_host_data.set('CONFIG_BRLAPI', brlapi.found())
2161config_host_data.set('CONFIG_BSD', host_os in bsd_oses)
2162config_host_data.set('CONFIG_CAPSTONE', capstone.found())
2163config_host_data.set('CONFIG_COCOA', cocoa.found())
2164config_host_data.set('CONFIG_DARWIN', host_os == 'darwin')
2165config_host_data.set('CONFIG_FUZZ', get_option('fuzzing'))
2166config_host_data.set('CONFIG_GCOV', get_option('b_coverage'))
2167config_host_data.set('CONFIG_LIBUDEV', libudev.found())
2168config_host_data.set('CONFIG_LINUX', host_os == 'linux')
2169config_host_data.set('CONFIG_POSIX', host_os != 'windows')
2170config_host_data.set('CONFIG_WIN32', host_os == 'windows')
2171config_host_data.set('CONFIG_LZO', lzo.found())
2172config_host_data.set('CONFIG_MPATH', mpathpersist.found())
2173config_host_data.set('CONFIG_BLKIO', blkio.found())
2174if blkio.found()
2175  config_host_data.set('CONFIG_BLKIO_VHOST_VDPA_FD',
2176                       blkio.version().version_compare('>=1.3.0'))
2177endif
2178config_host_data.set('CONFIG_CURL', curl.found())
2179config_host_data.set('CONFIG_CURSES', curses.found())
2180config_host_data.set('CONFIG_GBM', gbm.found())
2181config_host_data.set('CONFIG_GIO', gio.found())
2182config_host_data.set('CONFIG_GLUSTERFS', glusterfs.found())
2183if glusterfs.found()
2184  config_host_data.set('CONFIG_GLUSTERFS_XLATOR_OPT', glusterfs.version().version_compare('>=4'))
2185  config_host_data.set('CONFIG_GLUSTERFS_DISCARD', glusterfs.version().version_compare('>=5'))
2186  config_host_data.set('CONFIG_GLUSTERFS_FALLOCATE', glusterfs.version().version_compare('>=6'))
2187  config_host_data.set('CONFIG_GLUSTERFS_ZEROFILL', glusterfs.version().version_compare('>=6'))
2188  config_host_data.set('CONFIG_GLUSTERFS_FTRUNCATE_HAS_STAT', glusterfs_ftruncate_has_stat)
2189  config_host_data.set('CONFIG_GLUSTERFS_IOCB_HAS_STAT', glusterfs_iocb_has_stat)
2190endif
2191config_host_data.set('CONFIG_GTK', gtk.found())
2192config_host_data.set('CONFIG_VTE', vte.found())
2193config_host_data.set('CONFIG_GTK_CLIPBOARD', have_gtk_clipboard)
2194config_host_data.set('CONFIG_HEXAGON_IDEF_PARSER', get_option('hexagon_idef_parser'))
2195config_host_data.set('CONFIG_LIBATTR', have_old_libattr)
2196config_host_data.set('CONFIG_LIBCAP_NG', libcap_ng.found())
2197config_host_data.set('CONFIG_EBPF', libbpf.found())
2198config_host_data.set('CONFIG_AF_XDP', libxdp.found())
2199config_host_data.set('CONFIG_LIBDAXCTL', libdaxctl.found())
2200config_host_data.set('CONFIG_LIBISCSI', libiscsi.found())
2201config_host_data.set('CONFIG_LIBNFS', libnfs.found())
2202config_host_data.set('CONFIG_LIBSSH', libssh.found())
2203config_host_data.set('CONFIG_LINUX_AIO', libaio.found())
2204config_host_data.set('CONFIG_LINUX_IO_URING', linux_io_uring.found())
2205config_host_data.set('CONFIG_LIBPMEM', libpmem.found())
2206config_host_data.set('CONFIG_MODULES', enable_modules)
2207config_host_data.set('CONFIG_NUMA', numa.found())
2208if numa.found()
2209  config_host_data.set('HAVE_NUMA_HAS_PREFERRED_MANY',
2210                       cc.has_function('numa_has_preferred_many',
2211                                       dependencies: numa))
2212endif
2213config_host_data.set('CONFIG_OPENGL', opengl.found())
2214config_host_data.set('CONFIG_PLUGIN', get_option('plugins'))
2215config_host_data.set('CONFIG_RBD', rbd.found())
2216config_host_data.set('CONFIG_RDMA', rdma.found())
2217config_host_data.set('CONFIG_RELOCATABLE', get_option('relocatable'))
2218config_host_data.set('CONFIG_SAFESTACK', get_option('safe_stack'))
2219config_host_data.set('CONFIG_SDL', sdl.found())
2220config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found())
2221config_host_data.set('CONFIG_SECCOMP', seccomp.found())
2222if seccomp.found()
2223  config_host_data.set('CONFIG_SECCOMP_SYSRAWRC', seccomp_has_sysrawrc)
2224endif
2225config_host_data.set('CONFIG_PIXMAN', pixman.found())
2226config_host_data.set('CONFIG_SLIRP', slirp.found())
2227config_host_data.set('CONFIG_SNAPPY', snappy.found())
2228config_host_data.set('CONFIG_SOLARIS', host_os == 'sunos')
2229if get_option('tcg').allowed()
2230  config_host_data.set('CONFIG_TCG', 1)
2231  config_host_data.set('CONFIG_TCG_INTERPRETER', tcg_arch == 'tci')
2232endif
2233config_host_data.set('CONFIG_TPM', have_tpm)
2234config_host_data.set('CONFIG_TSAN', get_option('tsan'))
2235config_host_data.set('CONFIG_USB_LIBUSB', libusb.found())
2236config_host_data.set('CONFIG_VDE', vde.found())
2237config_host_data.set('CONFIG_VHOST', have_vhost)
2238config_host_data.set('CONFIG_VHOST_NET', have_vhost_net)
2239config_host_data.set('CONFIG_VHOST_NET_USER', have_vhost_net_user)
2240config_host_data.set('CONFIG_VHOST_NET_VDPA', have_vhost_net_vdpa)
2241config_host_data.set('CONFIG_VHOST_KERNEL', have_vhost_kernel)
2242config_host_data.set('CONFIG_VHOST_USER', have_vhost_user)
2243config_host_data.set('CONFIG_VHOST_CRYPTO', have_vhost_user_crypto)
2244config_host_data.set('CONFIG_VHOST_VDPA', have_vhost_vdpa)
2245config_host_data.set('CONFIG_VMNET', vmnet.found())
2246config_host_data.set('CONFIG_VHOST_USER_BLK_SERVER', have_vhost_user_blk_server)
2247config_host_data.set('CONFIG_VDUSE_BLK_EXPORT', have_vduse_blk_export)
2248config_host_data.set('CONFIG_PNG', png.found())
2249config_host_data.set('CONFIG_VNC', vnc.found())
2250config_host_data.set('CONFIG_VNC_JPEG', jpeg.found())
2251config_host_data.set('CONFIG_VNC_SASL', sasl.found())
2252if virgl.found()
2253  config_host_data.set('HAVE_VIRGL_D3D_INFO_EXT',
2254                       cc.has_member('struct virgl_renderer_resource_info_ext', 'd3d_tex2d',
2255                                     prefix: '#include <virglrenderer.h>',
2256                                     dependencies: virgl))
2257endif
2258config_host_data.set('CONFIG_VIRTFS', have_virtfs)
2259config_host_data.set('CONFIG_VTE', vte.found())
2260config_host_data.set('CONFIG_XKBCOMMON', xkbcommon.found())
2261config_host_data.set('CONFIG_KEYUTILS', keyutils.found())
2262config_host_data.set('CONFIG_GETTID', has_gettid)
2263config_host_data.set('CONFIG_GNUTLS', gnutls.found())
2264config_host_data.set('CONFIG_GNUTLS_CRYPTO', gnutls_crypto.found())
2265config_host_data.set('CONFIG_TASN1', tasn1.found())
2266config_host_data.set('CONFIG_GCRYPT', gcrypt.found())
2267config_host_data.set('CONFIG_NETTLE', nettle.found())
2268config_host_data.set('CONFIG_HOGWEED', hogweed.found())
2269config_host_data.set('CONFIG_QEMU_PRIVATE_XTS', xts == 'private')
2270config_host_data.set('CONFIG_MALLOC_TRIM', has_malloc_trim)
2271config_host_data.set('CONFIG_STATX', has_statx)
2272config_host_data.set('CONFIG_STATX_MNT_ID', has_statx_mnt_id)
2273config_host_data.set('CONFIG_ZSTD', zstd.found())
2274config_host_data.set('CONFIG_FUSE', fuse.found())
2275config_host_data.set('CONFIG_FUSE_LSEEK', fuse_lseek.found())
2276config_host_data.set('CONFIG_SPICE_PROTOCOL', spice_protocol.found())
2277if spice_protocol.found()
2278config_host_data.set('CONFIG_SPICE_PROTOCOL_MAJOR', spice_protocol.version().split('.')[0])
2279config_host_data.set('CONFIG_SPICE_PROTOCOL_MINOR', spice_protocol.version().split('.')[1])
2280config_host_data.set('CONFIG_SPICE_PROTOCOL_MICRO', spice_protocol.version().split('.')[2])
2281endif
2282config_host_data.set('CONFIG_SPICE', spice.found())
2283config_host_data.set('CONFIG_X11', x11.found())
2284config_host_data.set('CONFIG_DBUS_DISPLAY', dbus_display)
2285config_host_data.set('CONFIG_CFI', get_option('cfi'))
2286config_host_data.set('CONFIG_SELINUX', selinux.found())
2287config_host_data.set('CONFIG_XEN_BACKEND', xen.found())
2288config_host_data.set('CONFIG_LIBDW', libdw.found())
2289if xen.found()
2290  # protect from xen.version() having less than three components
2291  xen_version = xen.version().split('.') + ['0', '0']
2292  xen_ctrl_version = xen_version[0] + \
2293    ('0' + xen_version[1]).substring(-2) + \
2294    ('0' + xen_version[2]).substring(-2)
2295  config_host_data.set('CONFIG_XEN_CTRL_INTERFACE_VERSION', xen_ctrl_version)
2296endif
2297config_host_data.set('QEMU_VERSION', '"@0@"'.format(meson.project_version()))
2298config_host_data.set('QEMU_VERSION_MAJOR', meson.project_version().split('.')[0])
2299config_host_data.set('QEMU_VERSION_MINOR', meson.project_version().split('.')[1])
2300config_host_data.set('QEMU_VERSION_MICRO', meson.project_version().split('.')[2])
2301
2302config_host_data.set_quoted('CONFIG_HOST_DSOSUF', host_dsosuf)
2303config_host_data.set('HAVE_HOST_BLOCK_DEVICE', have_host_block_device)
2304
2305have_coroutine_pool = get_option('coroutine_pool')
2306if get_option('debug_stack_usage') and have_coroutine_pool
2307  message('Disabling coroutine pool to measure stack usage')
2308  have_coroutine_pool = false
2309endif
2310config_host_data.set('CONFIG_COROUTINE_POOL', have_coroutine_pool)
2311config_host_data.set('CONFIG_DEBUG_GRAPH_LOCK', get_option('debug_graph_lock'))
2312config_host_data.set('CONFIG_DEBUG_MUTEX', get_option('debug_mutex'))
2313config_host_data.set('CONFIG_DEBUG_STACK_USAGE', get_option('debug_stack_usage'))
2314config_host_data.set('CONFIG_DEBUG_TCG', get_option('debug_tcg'))
2315config_host_data.set('CONFIG_LIVE_BLOCK_MIGRATION', get_option('live_block_migration').allowed())
2316config_host_data.set('CONFIG_QOM_CAST_DEBUG', get_option('qom_cast_debug'))
2317config_host_data.set('CONFIG_REPLICATION', get_option('replication').allowed())
2318
2319# has_header
2320config_host_data.set('CONFIG_EPOLL', cc.has_header('sys/epoll.h'))
2321config_host_data.set('CONFIG_LINUX_MAGIC_H', cc.has_header('linux/magic.h'))
2322config_host_data.set('CONFIG_VALGRIND_H', cc.has_header('valgrind/valgrind.h'))
2323config_host_data.set('HAVE_BTRFS_H', cc.has_header('linux/btrfs.h'))
2324config_host_data.set('HAVE_DRM_H', cc.has_header('libdrm/drm.h'))
2325config_host_data.set('HAVE_PTY_H', cc.has_header('pty.h'))
2326config_host_data.set('HAVE_SYS_DISK_H', cc.has_header('sys/disk.h'))
2327config_host_data.set('HAVE_SYS_IOCCOM_H', cc.has_header('sys/ioccom.h'))
2328config_host_data.set('HAVE_SYS_KCOV_H', cc.has_header('sys/kcov.h'))
2329if host_os == 'windows'
2330  config_host_data.set('HAVE_AFUNIX_H', cc.has_header('afunix.h'))
2331endif
2332
2333# has_function
2334config_host_data.set('CONFIG_CLOSE_RANGE', cc.has_function('close_range'))
2335config_host_data.set('CONFIG_ACCEPT4', cc.has_function('accept4'))
2336config_host_data.set('CONFIG_CLOCK_ADJTIME', cc.has_function('clock_adjtime'))
2337config_host_data.set('CONFIG_DUP3', cc.has_function('dup3'))
2338config_host_data.set('CONFIG_FALLOCATE', cc.has_function('fallocate'))
2339config_host_data.set('CONFIG_POSIX_FALLOCATE', cc.has_function('posix_fallocate'))
2340config_host_data.set('CONFIG_GETCPU', cc.has_function('getcpu', prefix: gnu_source_prefix))
2341config_host_data.set('CONFIG_SCHED_GETCPU', cc.has_function('sched_getcpu', prefix: '#include <sched.h>'))
2342# Note that we need to specify prefix: here to avoid incorrectly
2343# thinking that Windows has posix_memalign()
2344config_host_data.set('CONFIG_POSIX_MEMALIGN', cc.has_function('posix_memalign', prefix: '#include <stdlib.h>'))
2345config_host_data.set('CONFIG_ALIGNED_MALLOC', cc.has_function('_aligned_malloc'))
2346config_host_data.set('CONFIG_VALLOC', cc.has_function('valloc'))
2347config_host_data.set('CONFIG_MEMALIGN', cc.has_function('memalign'))
2348config_host_data.set('CONFIG_PPOLL', cc.has_function('ppoll'))
2349config_host_data.set('CONFIG_PREADV', cc.has_function('preadv', prefix: '#include <sys/uio.h>'))
2350config_host_data.set('CONFIG_PTHREAD_FCHDIR_NP', cc.has_function('pthread_fchdir_np'))
2351config_host_data.set('CONFIG_SENDFILE', cc.has_function('sendfile'))
2352config_host_data.set('CONFIG_SETNS', cc.has_function('setns') and cc.has_function('unshare'))
2353config_host_data.set('CONFIG_SYNCFS', cc.has_function('syncfs'))
2354config_host_data.set('CONFIG_SYNC_FILE_RANGE', cc.has_function('sync_file_range'))
2355config_host_data.set('CONFIG_TIMERFD', cc.has_function('timerfd_create'))
2356config_host_data.set('HAVE_COPY_FILE_RANGE', cc.has_function('copy_file_range'))
2357config_host_data.set('HAVE_GETIFADDRS', cc.has_function('getifaddrs'))
2358config_host_data.set('HAVE_GLIB_WITH_SLICE_ALLOCATOR', glib_has_gslice)
2359config_host_data.set('HAVE_OPENPTY', cc.has_function('openpty', dependencies: util))
2360config_host_data.set('HAVE_STRCHRNUL', cc.has_function('strchrnul'))
2361config_host_data.set('HAVE_SYSTEM_FUNCTION', cc.has_function('system', prefix: '#include <stdlib.h>'))
2362if rbd.found()
2363  config_host_data.set('HAVE_RBD_NAMESPACE_EXISTS',
2364                       cc.has_function('rbd_namespace_exists',
2365                                       dependencies: rbd,
2366                                       prefix: '#include <rbd/librbd.h>'))
2367endif
2368if rdma.found()
2369  config_host_data.set('HAVE_IBV_ADVISE_MR',
2370                       cc.has_function('ibv_advise_mr',
2371                                       dependencies: rdma,
2372                                       prefix: '#include <infiniband/verbs.h>'))
2373endif
2374
2375have_asan_fiber = false
2376if get_option('sanitizers') and \
2377   not cc.has_function('__sanitizer_start_switch_fiber',
2378                         args: '-fsanitize=address',
2379                         prefix: '#include <sanitizer/asan_interface.h>')
2380  warning('Missing ASAN due to missing fiber annotation interface')
2381  warning('Without code annotation, the report may be inferior.')
2382else
2383  have_asan_fiber = true
2384endif
2385config_host_data.set('CONFIG_ASAN_IFACE_FIBER', have_asan_fiber)
2386
2387# has_header_symbol
2388config_host_data.set('CONFIG_BLKZONED',
2389                     cc.has_header_symbol('linux/blkzoned.h', 'BLKOPENZONE'))
2390config_host_data.set('CONFIG_EPOLL_CREATE1',
2391                     cc.has_header_symbol('sys/epoll.h', 'epoll_create1'))
2392config_host_data.set('CONFIG_FALLOCATE_PUNCH_HOLE',
2393                     cc.has_header_symbol('linux/falloc.h', 'FALLOC_FL_PUNCH_HOLE') and
2394                     cc.has_header_symbol('linux/falloc.h', 'FALLOC_FL_KEEP_SIZE'))
2395config_host_data.set('CONFIG_FALLOCATE_ZERO_RANGE',
2396                     cc.has_header_symbol('linux/falloc.h', 'FALLOC_FL_ZERO_RANGE'))
2397config_host_data.set('CONFIG_FIEMAP',
2398                     cc.has_header('linux/fiemap.h') and
2399                     cc.has_header_symbol('linux/fs.h', 'FS_IOC_FIEMAP'))
2400config_host_data.set('CONFIG_GETRANDOM',
2401                     cc.has_function('getrandom') and
2402                     cc.has_header_symbol('sys/random.h', 'GRND_NONBLOCK'))
2403config_host_data.set('CONFIG_INOTIFY',
2404                     cc.has_header_symbol('sys/inotify.h', 'inotify_init'))
2405config_host_data.set('CONFIG_INOTIFY1',
2406                     cc.has_header_symbol('sys/inotify.h', 'inotify_init1'))
2407config_host_data.set('CONFIG_PRCTL_PR_SET_TIMERSLACK',
2408                     cc.has_header_symbol('sys/prctl.h', 'PR_SET_TIMERSLACK'))
2409config_host_data.set('CONFIG_RTNETLINK',
2410                     cc.has_header_symbol('linux/rtnetlink.h', 'IFLA_PROTO_DOWN'))
2411config_host_data.set('CONFIG_SYSMACROS',
2412                     cc.has_header_symbol('sys/sysmacros.h', 'makedev'))
2413config_host_data.set('HAVE_OPTRESET',
2414                     cc.has_header_symbol('getopt.h', 'optreset'))
2415config_host_data.set('HAVE_IPPROTO_MPTCP',
2416                     cc.has_header_symbol('netinet/in.h', 'IPPROTO_MPTCP'))
2417
2418# has_member
2419config_host_data.set('HAVE_SIGEV_NOTIFY_THREAD_ID',
2420                     cc.has_member('struct sigevent', 'sigev_notify_thread_id',
2421                                   prefix: '#include <signal.h>'))
2422config_host_data.set('HAVE_STRUCT_STAT_ST_ATIM',
2423                     cc.has_member('struct stat', 'st_atim',
2424                                   prefix: '#include <sys/stat.h>'))
2425config_host_data.set('HAVE_BLK_ZONE_REP_CAPACITY',
2426                     cc.has_member('struct blk_zone', 'capacity',
2427                                   prefix: '#include <linux/blkzoned.h>'))
2428
2429# has_type
2430config_host_data.set('CONFIG_IOVEC',
2431                     cc.has_type('struct iovec',
2432                                 prefix: '#include <sys/uio.h>'))
2433config_host_data.set('HAVE_UTMPX',
2434                     cc.has_type('struct utmpx',
2435                                 prefix: '#include <utmpx.h>'))
2436
2437config_host_data.set('CONFIG_EVENTFD', cc.links('''
2438  #include <sys/eventfd.h>
2439  int main(void) { return eventfd(0, EFD_NONBLOCK | EFD_CLOEXEC); }'''))
2440config_host_data.set('CONFIG_FDATASYNC', cc.links(gnu_source_prefix + '''
2441  #include <unistd.h>
2442  int main(void) {
2443  #if defined(_POSIX_SYNCHRONIZED_IO) && _POSIX_SYNCHRONIZED_IO > 0
2444  return fdatasync(0);
2445  #else
2446  #error Not supported
2447  #endif
2448  }'''))
2449
2450has_madvise = cc.links(gnu_source_prefix + '''
2451  #include <sys/types.h>
2452  #include <sys/mman.h>
2453  #include <stddef.h>
2454  int main(void) { return madvise(NULL, 0, MADV_DONTNEED); }''')
2455missing_madvise_proto = false
2456if has_madvise
2457  # Some platforms (illumos and Solaris before Solaris 11) provide madvise()
2458  # but forget to prototype it. In this case, has_madvise will be true (the
2459  # test program links despite a compile warning). To detect the
2460  # missing-prototype case, we try again with a definitely-bogus prototype.
2461  # This will only compile if the system headers don't provide the prototype;
2462  # otherwise the conflicting prototypes will cause a compiler error.
2463  missing_madvise_proto = cc.links(gnu_source_prefix + '''
2464    #include <sys/types.h>
2465    #include <sys/mman.h>
2466    #include <stddef.h>
2467    extern int madvise(int);
2468    int main(void) { return madvise(0); }''')
2469endif
2470config_host_data.set('CONFIG_MADVISE', has_madvise)
2471config_host_data.set('HAVE_MADVISE_WITHOUT_PROTOTYPE', missing_madvise_proto)
2472
2473config_host_data.set('CONFIG_MEMFD', cc.links(gnu_source_prefix + '''
2474  #include <sys/mman.h>
2475  int main(void) { return memfd_create("foo", MFD_ALLOW_SEALING); }'''))
2476config_host_data.set('CONFIG_OPEN_BY_HANDLE', cc.links(gnu_source_prefix + '''
2477  #include <fcntl.h>
2478  #if !defined(AT_EMPTY_PATH)
2479  # error missing definition
2480  #else
2481  int main(void) { struct file_handle fh; return open_by_handle_at(0, &fh, 0); }
2482  #endif'''))
2483config_host_data.set('CONFIG_POSIX_MADVISE', cc.links(gnu_source_prefix + '''
2484  #include <sys/mman.h>
2485  #include <stddef.h>
2486  int main(void) { return posix_madvise(NULL, 0, POSIX_MADV_DONTNEED); }'''))
2487
2488config_host_data.set('CONFIG_PTHREAD_SETNAME_NP_W_TID', cc.links(gnu_source_prefix + '''
2489  #include <pthread.h>
2490
2491  static void *f(void *p) { return NULL; }
2492  int main(void)
2493  {
2494    pthread_t thread;
2495    pthread_create(&thread, 0, f, 0);
2496    pthread_setname_np(thread, "QEMU");
2497    return 0;
2498  }''', dependencies: threads))
2499config_host_data.set('CONFIG_PTHREAD_SETNAME_NP_WO_TID', cc.links(gnu_source_prefix + '''
2500  #include <pthread.h>
2501
2502  static void *f(void *p) { pthread_setname_np("QEMU"); return NULL; }
2503  int main(void)
2504  {
2505    pthread_t thread;
2506    pthread_create(&thread, 0, f, 0);
2507    return 0;
2508  }''', dependencies: threads))
2509config_host_data.set('CONFIG_PTHREAD_SET_NAME_NP', cc.links(gnu_source_prefix + '''
2510  #include <pthread.h>
2511  #include <pthread_np.h>
2512
2513  static void *f(void *p) { return NULL; }
2514  int main(void)
2515  {
2516    pthread_t thread;
2517    pthread_create(&thread, 0, f, 0);
2518    pthread_set_name_np(thread, "QEMU");
2519    return 0;
2520  }''', dependencies: threads))
2521config_host_data.set('CONFIG_PTHREAD_CONDATTR_SETCLOCK', cc.links(gnu_source_prefix + '''
2522  #include <pthread.h>
2523  #include <time.h>
2524
2525  int main(void)
2526  {
2527    pthread_condattr_t attr
2528    pthread_condattr_init(&attr);
2529    pthread_condattr_setclock(&attr, CLOCK_MONOTONIC);
2530    return 0;
2531  }''', dependencies: threads))
2532config_host_data.set('CONFIG_PTHREAD_AFFINITY_NP', cc.links(gnu_source_prefix + '''
2533  #include <pthread.h>
2534
2535  static void *f(void *p) { return NULL; }
2536  int main(void)
2537  {
2538    int setsize = CPU_ALLOC_SIZE(64);
2539    pthread_t thread;
2540    cpu_set_t *cpuset;
2541    pthread_create(&thread, 0, f, 0);
2542    cpuset = CPU_ALLOC(64);
2543    CPU_ZERO_S(setsize, cpuset);
2544    pthread_setaffinity_np(thread, setsize, cpuset);
2545    pthread_getaffinity_np(thread, setsize, cpuset);
2546    CPU_FREE(cpuset);
2547    return 0;
2548  }''', dependencies: threads))
2549config_host_data.set('CONFIG_SIGNALFD', cc.links(gnu_source_prefix + '''
2550  #include <sys/signalfd.h>
2551  #include <stddef.h>
2552  int main(void) { return signalfd(-1, NULL, SFD_CLOEXEC); }'''))
2553config_host_data.set('CONFIG_SPLICE', cc.links(gnu_source_prefix + '''
2554  #include <unistd.h>
2555  #include <fcntl.h>
2556  #include <limits.h>
2557
2558  int main(void)
2559  {
2560    int len, fd = 0;
2561    len = tee(STDIN_FILENO, STDOUT_FILENO, INT_MAX, SPLICE_F_NONBLOCK);
2562    splice(STDIN_FILENO, NULL, fd, NULL, len, SPLICE_F_MOVE);
2563    return 0;
2564  }'''))
2565
2566config_host_data.set('HAVE_MLOCKALL', cc.links(gnu_source_prefix + '''
2567  #include <sys/mman.h>
2568  int main(void) {
2569    return mlockall(MCL_FUTURE);
2570  }'''))
2571
2572have_l2tpv3 = false
2573if get_option('l2tpv3').allowed() and have_system
2574  have_l2tpv3 = cc.has_type('struct mmsghdr',
2575    prefix: gnu_source_prefix + '''
2576      #include <sys/socket.h>
2577      #include <linux/ip.h>''')
2578endif
2579config_host_data.set('CONFIG_L2TPV3', have_l2tpv3)
2580
2581have_netmap = false
2582if get_option('netmap').allowed() and have_system
2583  have_netmap = cc.compiles('''
2584    #include <inttypes.h>
2585    #include <net/if.h>
2586    #include <net/netmap.h>
2587    #include <net/netmap_user.h>
2588    #if (NETMAP_API < 11) || (NETMAP_API > 15)
2589    #error
2590    #endif
2591    int main(void) { return 0; }''')
2592  if not have_netmap and get_option('netmap').enabled()
2593    error('Netmap headers not available')
2594  endif
2595endif
2596config_host_data.set('CONFIG_NETMAP', have_netmap)
2597
2598# Work around a system header bug with some kernel/XFS header
2599# versions where they both try to define 'struct fsxattr':
2600# xfs headers will not try to redefine structs from linux headers
2601# if this macro is set.
2602config_host_data.set('HAVE_FSXATTR', cc.links('''
2603  #include <linux/fs.h>
2604  struct fsxattr foo;
2605  int main(void) {
2606    return 0;
2607  }'''))
2608
2609# Some versions of Mac OS X incorrectly define SIZE_MAX
2610config_host_data.set('HAVE_BROKEN_SIZE_MAX', not cc.compiles('''
2611    #include <stdint.h>
2612    #include <stdio.h>
2613    int main(void) {
2614        return printf("%zu", SIZE_MAX);
2615    }''', args: ['-Werror']))
2616
2617# See if 64-bit atomic operations are supported.
2618# Note that without __atomic builtins, we can only
2619# assume atomic loads/stores max at pointer size.
2620config_host_data.set('CONFIG_ATOMIC64', cc.links('''
2621  #include <stdint.h>
2622  int main(void)
2623  {
2624    uint64_t x = 0, y = 0;
2625    y = __atomic_load_n(&x, __ATOMIC_RELAXED);
2626    __atomic_store_n(&x, y, __ATOMIC_RELAXED);
2627    __atomic_compare_exchange_n(&x, &y, x, 0, __ATOMIC_RELAXED, __ATOMIC_RELAXED);
2628    __atomic_exchange_n(&x, y, __ATOMIC_RELAXED);
2629    __atomic_fetch_add(&x, y, __ATOMIC_RELAXED);
2630    return 0;
2631  }'''))
2632
2633has_int128_type = cc.compiles('''
2634  __int128_t a;
2635  __uint128_t b;
2636  int main(void) { b = a; }''')
2637config_host_data.set('CONFIG_INT128_TYPE', has_int128_type)
2638
2639has_int128 = has_int128_type and cc.links('''
2640  __int128_t a;
2641  __uint128_t b;
2642  int main (void) {
2643    a = a + b;
2644    b = a * b;
2645    a = a * a;
2646    return 0;
2647  }''')
2648config_host_data.set('CONFIG_INT128', has_int128)
2649
2650if has_int128_type
2651  # "do we have 128-bit atomics which are handled inline and specifically not
2652  # via libatomic". The reason we can't use libatomic is documented in the
2653  # comment starting "GCC is a house divided" in include/qemu/atomic128.h.
2654  # We only care about these operations on 16-byte aligned pointers, so
2655  # force 16-byte alignment of the pointer, which may be greater than
2656  # __alignof(unsigned __int128) for the host.
2657  atomic_test_128 = '''
2658    int main(int ac, char **av) {
2659      __uint128_t *p = __builtin_assume_aligned(av[ac - 1], 16);
2660      p[1] = __atomic_load_n(&p[0], __ATOMIC_RELAXED);
2661      __atomic_store_n(&p[2], p[3], __ATOMIC_RELAXED);
2662      __atomic_compare_exchange_n(&p[4], &p[5], p[6], 0, __ATOMIC_RELAXED, __ATOMIC_RELAXED);
2663      return 0;
2664    }'''
2665  has_atomic128 = cc.links(atomic_test_128)
2666
2667  config_host_data.set('CONFIG_ATOMIC128', has_atomic128)
2668
2669  if not has_atomic128
2670    # Even with __builtin_assume_aligned, the above test may have failed
2671    # without optimization enabled.  Try again with optimizations locally
2672    # enabled for the function.  See
2673    #   https://gcc.gnu.org/bugzilla/show_bug.cgi?id=107389
2674    has_atomic128_opt = cc.links('__attribute__((optimize("O1")))' + atomic_test_128)
2675    config_host_data.set('CONFIG_ATOMIC128_OPT', has_atomic128_opt)
2676
2677    if not has_atomic128_opt
2678      config_host_data.set('CONFIG_CMPXCHG128', cc.links('''
2679        int main(void)
2680        {
2681          __uint128_t x = 0, y = 0;
2682          __sync_val_compare_and_swap_16(&x, y, x);
2683          return 0;
2684        }
2685      '''))
2686    endif
2687  endif
2688endif
2689
2690config_host_data.set('CONFIG_GETAUXVAL', cc.links(gnu_source_prefix + '''
2691  #include <sys/auxv.h>
2692  int main(void) {
2693    return getauxval(AT_HWCAP) == 0;
2694  }'''))
2695
2696config_host_data.set('CONFIG_USBFS', have_linux_user and cc.compiles('''
2697  #include <linux/usbdevice_fs.h>
2698
2699  #ifndef USBDEVFS_GET_CAPABILITIES
2700  #error "USBDEVFS_GET_CAPABILITIES undefined"
2701  #endif
2702
2703  #ifndef USBDEVFS_DISCONNECT_CLAIM
2704  #error "USBDEVFS_DISCONNECT_CLAIM undefined"
2705  #endif
2706
2707  int main(void) { return 0; }'''))
2708
2709have_keyring = get_option('keyring') \
2710  .require(host_os == 'linux', error_message: 'keyring is only available on Linux') \
2711  .require(cc.compiles('''
2712    #include <errno.h>
2713    #include <asm/unistd.h>
2714    #include <linux/keyctl.h>
2715    #include <sys/syscall.h>
2716    #include <unistd.h>
2717    int main(void) {
2718        return syscall(__NR_keyctl, KEYCTL_READ, 0, NULL, NULL, 0);
2719    }'''), error_message: 'keyctl syscall not available on this system').allowed()
2720config_host_data.set('CONFIG_SECRET_KEYRING', have_keyring)
2721
2722have_cpuid_h = cc.links('''
2723  #include <cpuid.h>
2724  int main(void) {
2725    unsigned a, b, c, d;
2726    unsigned max = __get_cpuid_max(0, 0);
2727
2728    if (max >= 1) {
2729        __cpuid(1, a, b, c, d);
2730    }
2731
2732    if (max >= 7) {
2733        __cpuid_count(7, 0, a, b, c, d);
2734    }
2735
2736    return 0;
2737  }''')
2738config_host_data.set('CONFIG_CPUID_H', have_cpuid_h)
2739
2740config_host_data.set('CONFIG_AVX2_OPT', get_option('avx2') \
2741  .require(have_cpuid_h, error_message: 'cpuid.h not available, cannot enable AVX2') \
2742  .require(cc.links('''
2743    #include <cpuid.h>
2744    #include <immintrin.h>
2745    static int __attribute__((target("avx2"))) bar(void *a) {
2746      __m256i x = *(__m256i *)a;
2747      return _mm256_testz_si256(x, x);
2748    }
2749    int main(int argc, char *argv[]) { return bar(argv[argc - 1]); }
2750  '''), error_message: 'AVX2 not available').allowed())
2751
2752config_host_data.set('CONFIG_AVX512F_OPT', get_option('avx512f') \
2753  .require(have_cpuid_h, error_message: 'cpuid.h not available, cannot enable AVX512F') \
2754  .require(cc.links('''
2755    #include <cpuid.h>
2756    #include <immintrin.h>
2757    static int __attribute__((target("avx512f"))) bar(void *a) {
2758      __m512i x = *(__m512i *)a;
2759      return _mm512_test_epi64_mask(x, x);
2760    }
2761    int main(int argc, char *argv[]) { return bar(argv[argc - 1]); }
2762  '''), error_message: 'AVX512F not available').allowed())
2763
2764config_host_data.set('CONFIG_AVX512BW_OPT', get_option('avx512bw') \
2765  .require(have_cpuid_h, error_message: 'cpuid.h not available, cannot enable AVX512BW') \
2766  .require(cc.links('''
2767    #include <cpuid.h>
2768    #include <immintrin.h>
2769    static int __attribute__((target("avx512bw"))) bar(void *a) {
2770      __m512i *x = a;
2771      __m512i res= _mm512_abs_epi8(*x);
2772      return res[1];
2773    }
2774    int main(int argc, char *argv[]) { return bar(argv[0]); }
2775  '''), error_message: 'AVX512BW not available').allowed())
2776
2777# For both AArch64 and AArch32, detect if builtins are available.
2778config_host_data.set('CONFIG_ARM_AES_BUILTIN', cc.compiles('''
2779    #include <arm_neon.h>
2780    #ifndef __ARM_FEATURE_AES
2781    __attribute__((target("+crypto")))
2782    #endif
2783    void foo(uint8x16_t *p) { *p = vaesmcq_u8(*p); }
2784  '''))
2785
2786have_pvrdma = get_option('pvrdma') \
2787  .require(rdma.found(), error_message: 'PVRDMA requires OpenFabrics libraries') \
2788  .require(cc.compiles(gnu_source_prefix + '''
2789    #include <sys/mman.h>
2790    int main(void)
2791    {
2792      char buf = 0;
2793      void *addr = &buf;
2794      addr = mremap(addr, 0, 1, MREMAP_MAYMOVE | MREMAP_FIXED);
2795
2796      return 0;
2797    }'''), error_message: 'PVRDMA requires mremap').allowed()
2798
2799if have_pvrdma
2800  config_host_data.set('LEGACY_RDMA_REG_MR', not cc.links('''
2801    #include <infiniband/verbs.h>
2802    int main(void)
2803    {
2804      struct ibv_mr *mr;
2805      struct ibv_pd *pd = NULL;
2806      size_t length = 10;
2807      uint64_t iova = 0;
2808      int access = 0;
2809      void *addr = NULL;
2810
2811      mr = ibv_reg_mr_iova(pd, addr, length, iova, access);
2812      ibv_dereg_mr(mr);
2813      return 0;
2814    }'''))
2815endif
2816
2817if get_option('membarrier').disabled()
2818  have_membarrier = false
2819elif host_os == 'windows'
2820  have_membarrier = true
2821elif host_os == 'linux'
2822  have_membarrier = cc.compiles('''
2823    #include <linux/membarrier.h>
2824    #include <sys/syscall.h>
2825    #include <unistd.h>
2826    #include <stdlib.h>
2827    int main(void) {
2828        syscall(__NR_membarrier, MEMBARRIER_CMD_QUERY, 0);
2829        syscall(__NR_membarrier, MEMBARRIER_CMD_SHARED, 0);
2830        exit(0);
2831    }''')
2832endif
2833config_host_data.set('CONFIG_MEMBARRIER', get_option('membarrier') \
2834  .require(have_membarrier, error_message: 'membarrier system call not available') \
2835  .allowed())
2836
2837have_afalg = get_option('crypto_afalg') \
2838  .require(cc.compiles(gnu_source_prefix + '''
2839    #include <errno.h>
2840    #include <sys/types.h>
2841    #include <sys/socket.h>
2842    #include <linux/if_alg.h>
2843    int main(void) {
2844      int sock;
2845      sock = socket(AF_ALG, SOCK_SEQPACKET, 0);
2846      return sock;
2847    }
2848  '''), error_message: 'AF_ALG requested but could not be detected').allowed()
2849config_host_data.set('CONFIG_AF_ALG', have_afalg)
2850
2851config_host_data.set('CONFIG_AF_VSOCK', cc.has_header_symbol(
2852  'linux/vm_sockets.h', 'AF_VSOCK',
2853  prefix: '#include <sys/socket.h>',
2854))
2855
2856have_vss = false
2857have_vss_sdk = false # old xp/2003 SDK
2858if host_os == 'windows' and 'cpp' in all_languages
2859  have_vss = cxx.compiles('''
2860    #define __MIDL_user_allocate_free_DEFINED__
2861    #include <vss.h>
2862    int main(void) { return VSS_CTX_BACKUP; }''')
2863  have_vss_sdk = cxx.has_header('vscoordint.h')
2864endif
2865config_host_data.set('HAVE_VSS_SDK', have_vss_sdk)
2866
2867# Older versions of MinGW do not import _lock_file and _unlock_file properly.
2868# This was fixed for v6.0.0 with commit b48e3ac8969d.
2869if host_os == 'windows'
2870  config_host_data.set('HAVE__LOCK_FILE', cc.links('''
2871    #include <stdio.h>
2872    int main(void) {
2873      _lock_file(NULL);
2874      _unlock_file(NULL);
2875      return 0;
2876    }''', name: '_lock_file and _unlock_file'))
2877endif
2878
2879if host_os == 'windows'
2880  mingw_has_setjmp_longjmp = cc.links('''
2881    #include <setjmp.h>
2882    int main(void) {
2883      /*
2884       * These functions are not available in setjmp header, but may be
2885       * available at link time, from libmingwex.a.
2886       */
2887      extern int __mingw_setjmp(jmp_buf);
2888      extern void __attribute__((noreturn)) __mingw_longjmp(jmp_buf, int);
2889      jmp_buf env;
2890      __mingw_setjmp(env);
2891      __mingw_longjmp(env, 0);
2892    }
2893  ''', name: 'mingw setjmp and longjmp')
2894
2895  if cpu == 'aarch64' and not mingw_has_setjmp_longjmp
2896    error('mingw must provide setjmp/longjmp for windows-arm64')
2897  endif
2898endif
2899
2900########################
2901# Target configuration #
2902########################
2903
2904minikconf = find_program('scripts/minikconf.py')
2905
2906config_all_accel = {}
2907config_all_devices = {}
2908config_devices_mak_list = []
2909config_devices_h = {}
2910config_target_h = {}
2911config_target_mak = {}
2912
2913disassemblers = {
2914  'alpha' : ['CONFIG_ALPHA_DIS'],
2915  'avr' : ['CONFIG_AVR_DIS'],
2916  'cris' : ['CONFIG_CRIS_DIS'],
2917  'hexagon' : ['CONFIG_HEXAGON_DIS'],
2918  'hppa' : ['CONFIG_HPPA_DIS'],
2919  'i386' : ['CONFIG_I386_DIS'],
2920  'x86_64' : ['CONFIG_I386_DIS'],
2921  'm68k' : ['CONFIG_M68K_DIS'],
2922  'microblaze' : ['CONFIG_MICROBLAZE_DIS'],
2923  'mips' : ['CONFIG_MIPS_DIS'],
2924  'nios2' : ['CONFIG_NIOS2_DIS'],
2925  'or1k' : ['CONFIG_OPENRISC_DIS'],
2926  'ppc' : ['CONFIG_PPC_DIS'],
2927  'riscv' : ['CONFIG_RISCV_DIS'],
2928  'rx' : ['CONFIG_RX_DIS'],
2929  's390' : ['CONFIG_S390_DIS'],
2930  'sh4' : ['CONFIG_SH4_DIS'],
2931  'sparc' : ['CONFIG_SPARC_DIS'],
2932  'xtensa' : ['CONFIG_XTENSA_DIS'],
2933  'loongarch' : ['CONFIG_LOONGARCH_DIS'],
2934}
2935
2936have_ivshmem = config_host_data.get('CONFIG_EVENTFD')
2937host_kconfig = \
2938  (get_option('fuzzing') ? ['CONFIG_FUZZ=y'] : []) + \
2939  (have_tpm ? ['CONFIG_TPM=y'] : []) + \
2940  (pixman.found() ? ['CONFIG_PIXMAN=y'] : []) + \
2941  (spice.found() ? ['CONFIG_SPICE=y'] : []) + \
2942  (have_ivshmem ? ['CONFIG_IVSHMEM=y'] : []) + \
2943  (opengl.found() ? ['CONFIG_OPENGL=y'] : []) + \
2944  (x11.found() ? ['CONFIG_X11=y'] : []) + \
2945  (have_vhost_user ? ['CONFIG_VHOST_USER=y'] : []) + \
2946  (have_vhost_vdpa ? ['CONFIG_VHOST_VDPA=y'] : []) + \
2947  (have_vhost_kernel ? ['CONFIG_VHOST_KERNEL=y'] : []) + \
2948  (have_virtfs ? ['CONFIG_VIRTFS=y'] : []) + \
2949  (host_os == 'linux' ? ['CONFIG_LINUX=y'] : []) + \
2950  (have_pvrdma ? ['CONFIG_PVRDMA=y'] : []) + \
2951  (multiprocess_allowed ? ['CONFIG_MULTIPROCESS_ALLOWED=y'] : []) + \
2952  (vfio_user_server_allowed ? ['CONFIG_VFIO_USER_SERVER_ALLOWED=y'] : []) + \
2953  (hv_balloon ? ['CONFIG_HV_BALLOON_POSSIBLE=y'] : [])
2954
2955ignored = [ 'TARGET_XML_FILES', 'TARGET_ABI_DIR', 'TARGET_ARCH' ]
2956
2957default_targets = 'CONFIG_DEFAULT_TARGETS' in config_host
2958actual_target_dirs = []
2959fdt_required = []
2960foreach target : target_dirs
2961  config_target = { 'TARGET_NAME': target.split('-')[0] }
2962  if target.endswith('linux-user')
2963    if host_os != 'linux'
2964      if default_targets
2965        continue
2966      endif
2967      error('Target @0@ is only available on a Linux host'.format(target))
2968    endif
2969    config_target += { 'CONFIG_LINUX_USER': 'y' }
2970  elif target.endswith('bsd-user')
2971    if host_os not in bsd_oses
2972      if default_targets
2973        continue
2974      endif
2975      error('Target @0@ is only available on a BSD host'.format(target))
2976    endif
2977    config_target += { 'CONFIG_BSD_USER': 'y' }
2978  elif target.endswith('softmmu')
2979    config_target += { 'CONFIG_SYSTEM_ONLY': 'y' }
2980    config_target += { 'CONFIG_SOFTMMU': 'y' }
2981  endif
2982  if target.endswith('-user')
2983    config_target += {
2984      'CONFIG_USER_ONLY': 'y',
2985      'CONFIG_QEMU_INTERP_PREFIX':
2986        get_option('interp_prefix').replace('%M', config_target['TARGET_NAME'])
2987    }
2988  endif
2989
2990  accel_kconfig = []
2991  foreach sym: accelerators
2992    if sym == 'CONFIG_TCG' or target in accelerator_targets.get(sym, [])
2993      config_target += { sym: 'y' }
2994      config_all_accel += { sym: 'y' }
2995      if target in modular_tcg
2996        config_target += { 'CONFIG_TCG_MODULAR': 'y' }
2997      else
2998        config_target += { 'CONFIG_TCG_BUILTIN': 'y' }
2999      endif
3000      accel_kconfig += [ sym + '=y' ]
3001    endif
3002  endforeach
3003  if accel_kconfig.length() == 0
3004    if default_targets
3005      continue
3006    endif
3007    error('No accelerator available for target @0@'.format(target))
3008  endif
3009
3010  actual_target_dirs += target
3011  config_target += keyval.load('configs/targets' / target + '.mak')
3012  config_target += { 'TARGET_' + config_target['TARGET_ARCH'].to_upper(): 'y' }
3013
3014  if 'TARGET_NEED_FDT' in config_target
3015    fdt_required += target
3016  endif
3017
3018  # Add default keys
3019  if 'TARGET_BASE_ARCH' not in config_target
3020    config_target += {'TARGET_BASE_ARCH': config_target['TARGET_ARCH']}
3021  endif
3022  if 'TARGET_ABI_DIR' not in config_target
3023    config_target += {'TARGET_ABI_DIR': config_target['TARGET_ARCH']}
3024  endif
3025  if 'TARGET_BIG_ENDIAN' not in config_target
3026    config_target += {'TARGET_BIG_ENDIAN': 'n'}
3027  endif
3028
3029  foreach k, v: disassemblers
3030    if host_arch.startswith(k) or config_target['TARGET_BASE_ARCH'].startswith(k)
3031      foreach sym: v
3032        config_target += { sym: 'y' }
3033      endforeach
3034    endif
3035  endforeach
3036
3037  config_target_data = configuration_data()
3038  foreach k, v: config_target
3039    if not k.startswith('TARGET_') and not k.startswith('CONFIG_')
3040      # do nothing
3041    elif ignored.contains(k)
3042      # do nothing
3043    elif k == 'TARGET_BASE_ARCH'
3044      # Note that TARGET_BASE_ARCH ends up in config-target.h but it is
3045      # not used to select files from sourcesets.
3046      config_target_data.set('TARGET_' + v.to_upper(), 1)
3047    elif k == 'TARGET_NAME' or k == 'CONFIG_QEMU_INTERP_PREFIX'
3048      config_target_data.set_quoted(k, v)
3049    elif v == 'y'
3050      config_target_data.set(k, 1)
3051    elif v == 'n'
3052      config_target_data.set(k, 0)
3053    else
3054      config_target_data.set(k, v)
3055    endif
3056  endforeach
3057  config_target_data.set('QEMU_ARCH',
3058                         'QEMU_ARCH_' + config_target['TARGET_BASE_ARCH'].to_upper())
3059  config_target_h += {target: configure_file(output: target + '-config-target.h',
3060                                               configuration: config_target_data)}
3061
3062  if target.endswith('-softmmu')
3063    config_input = meson.get_external_property(target, 'default')
3064    config_devices_mak = target + '-config-devices.mak'
3065    config_devices_mak = configure_file(
3066      input: ['configs/devices' / target / config_input + '.mak', 'Kconfig'],
3067      output: config_devices_mak,
3068      depfile: config_devices_mak + '.d',
3069      capture: true,
3070      command: [minikconf,
3071                get_option('default_devices') ? '--defconfig' : '--allnoconfig',
3072                config_devices_mak, '@DEPFILE@', '@INPUT@',
3073                host_kconfig, accel_kconfig,
3074                'CONFIG_' + config_target['TARGET_ARCH'].to_upper() + '=y'])
3075
3076    config_devices_data = configuration_data()
3077    config_devices = keyval.load(config_devices_mak)
3078    foreach k, v: config_devices
3079      config_devices_data.set(k, 1)
3080    endforeach
3081    config_devices_mak_list += config_devices_mak
3082    config_devices_h += {target: configure_file(output: target + '-config-devices.h',
3083                                                configuration: config_devices_data)}
3084    config_target += config_devices
3085    config_all_devices += config_devices
3086  endif
3087  config_target_mak += {target: config_target}
3088endforeach
3089target_dirs = actual_target_dirs
3090
3091target_configs_h = []
3092foreach target: target_dirs
3093  target_configs_h += config_target_h[target]
3094  target_configs_h += config_devices_h.get(target, [])
3095endforeach
3096genh += custom_target('config-poison.h',
3097                      input: [target_configs_h],
3098                      output: 'config-poison.h',
3099                      capture: true,
3100                      command: [find_program('scripts/make-config-poison.sh'),
3101                                target_configs_h])
3102
3103###############
3104# Subprojects #
3105###############
3106
3107libvfio_user_dep = not_found
3108if have_system and vfio_user_server_allowed
3109  libvfio_user_proj = subproject('libvfio-user', required: true)
3110  libvfio_user_dep = libvfio_user_proj.get_variable('libvfio_user_dep')
3111endif
3112
3113fdt = not_found
3114fdt_opt = get_option('fdt')
3115if fdt_required.length() > 0 or fdt_opt == 'enabled'
3116  if fdt_opt == 'disabled'
3117    error('fdt disabled but required by targets ' + ', '.join(fdt_required))
3118  endif
3119
3120  if fdt_opt in ['enabled', 'auto', 'system']
3121    if get_option('wrap_mode') == 'nodownload'
3122      fdt_opt = 'system'
3123    endif
3124    fdt = cc.find_library('fdt', required: fdt_opt == 'system')
3125    if fdt.found() and cc.links('''
3126       #include <libfdt.h>
3127       #include <libfdt_env.h>
3128       int main(void) { fdt_find_max_phandle(NULL, NULL); return 0; }''',
3129         dependencies: fdt)
3130      fdt_opt = 'system'
3131    elif fdt_opt == 'system'
3132       error('system libfdt requested, but it is too old (1.5.1 or newer required)')
3133    else
3134      fdt_opt = 'internal'
3135      fdt = not_found
3136    endif
3137  endif
3138  if not fdt.found()
3139    assert(fdt_opt == 'internal')
3140    libfdt_proj = subproject('dtc', required: true,
3141                             default_options: ['tools=false',  'yaml=disabled',
3142                                               'python=disabled', 'default_library=static'])
3143    fdt = libfdt_proj.get_variable('libfdt_dep')
3144  endif
3145else
3146  fdt_opt = 'disabled'
3147endif
3148
3149config_host_data.set('CONFIG_FDT', fdt.found())
3150
3151vhost_user = not_found
3152if host_os == 'linux' and have_vhost_user
3153  libvhost_user = subproject('libvhost-user')
3154  vhost_user = libvhost_user.get_variable('vhost_user_dep')
3155endif
3156
3157libvduse = not_found
3158if have_libvduse
3159  libvduse_proj = subproject('libvduse')
3160  libvduse = libvduse_proj.get_variable('libvduse_dep')
3161endif
3162
3163#####################
3164# Generated sources #
3165#####################
3166
3167genh += configure_file(output: 'config-host.h', configuration: config_host_data)
3168
3169hxtool = find_program('scripts/hxtool')
3170shaderinclude = find_program('scripts/shaderinclude.py')
3171qapi_gen = find_program('scripts/qapi-gen.py')
3172qapi_gen_depends = [ meson.current_source_dir() / 'scripts/qapi/__init__.py',
3173                     meson.current_source_dir() / 'scripts/qapi/commands.py',
3174                     meson.current_source_dir() / 'scripts/qapi/common.py',
3175                     meson.current_source_dir() / 'scripts/qapi/error.py',
3176                     meson.current_source_dir() / 'scripts/qapi/events.py',
3177                     meson.current_source_dir() / 'scripts/qapi/expr.py',
3178                     meson.current_source_dir() / 'scripts/qapi/gen.py',
3179                     meson.current_source_dir() / 'scripts/qapi/introspect.py',
3180                     meson.current_source_dir() / 'scripts/qapi/main.py',
3181                     meson.current_source_dir() / 'scripts/qapi/parser.py',
3182                     meson.current_source_dir() / 'scripts/qapi/schema.py',
3183                     meson.current_source_dir() / 'scripts/qapi/source.py',
3184                     meson.current_source_dir() / 'scripts/qapi/types.py',
3185                     meson.current_source_dir() / 'scripts/qapi/visit.py',
3186                     meson.current_source_dir() / 'scripts/qapi-gen.py'
3187]
3188
3189tracetool = [
3190  python, files('scripts/tracetool.py'),
3191   '--backend=' + ','.join(get_option('trace_backends'))
3192]
3193tracetool_depends = files(
3194  'scripts/tracetool/backend/log.py',
3195  'scripts/tracetool/backend/__init__.py',
3196  'scripts/tracetool/backend/dtrace.py',
3197  'scripts/tracetool/backend/ftrace.py',
3198  'scripts/tracetool/backend/simple.py',
3199  'scripts/tracetool/backend/syslog.py',
3200  'scripts/tracetool/backend/ust.py',
3201  'scripts/tracetool/format/ust_events_c.py',
3202  'scripts/tracetool/format/ust_events_h.py',
3203  'scripts/tracetool/format/__init__.py',
3204  'scripts/tracetool/format/d.py',
3205  'scripts/tracetool/format/simpletrace_stap.py',
3206  'scripts/tracetool/format/c.py',
3207  'scripts/tracetool/format/h.py',
3208  'scripts/tracetool/format/log_stap.py',
3209  'scripts/tracetool/format/stap.py',
3210  'scripts/tracetool/__init__.py',
3211  'scripts/tracetool/vcpu.py'
3212)
3213
3214qemu_version_cmd = [find_program('scripts/qemu-version.sh'),
3215                    meson.current_source_dir(),
3216                    get_option('pkgversion'), meson.project_version()]
3217qemu_version = custom_target('qemu-version.h',
3218                             output: 'qemu-version.h',
3219                             command: qemu_version_cmd,
3220                             capture: true,
3221                             build_by_default: true,
3222                             build_always_stale: true)
3223genh += qemu_version
3224
3225hxdep = []
3226hx_headers = [
3227  ['qemu-options.hx', 'qemu-options.def'],
3228  ['qemu-img-cmds.hx', 'qemu-img-cmds.h'],
3229]
3230if have_system
3231  hx_headers += [
3232    ['hmp-commands.hx', 'hmp-commands.h'],
3233    ['hmp-commands-info.hx', 'hmp-commands-info.h'],
3234  ]
3235endif
3236foreach d : hx_headers
3237  hxdep += custom_target(d[1],
3238                input: files(d[0]),
3239                output: d[1],
3240                capture: true,
3241                command: [hxtool, '-h', '@INPUT0@'])
3242endforeach
3243genh += hxdep
3244
3245###############
3246# Trace files #
3247###############
3248
3249# TODO: add each directory to the subdirs from its own meson.build, once
3250# we have those
3251trace_events_subdirs = [
3252  'crypto',
3253  'qapi',
3254  'qom',
3255  'monitor',
3256  'util',
3257  'gdbstub',
3258]
3259if have_linux_user
3260  trace_events_subdirs += [ 'linux-user' ]
3261endif
3262if have_bsd_user
3263  trace_events_subdirs += [ 'bsd-user' ]
3264endif
3265if have_block
3266  trace_events_subdirs += [
3267    'authz',
3268    'block',
3269    'io',
3270    'nbd',
3271    'scsi',
3272  ]
3273endif
3274if have_system
3275  trace_events_subdirs += [
3276    'accel/kvm',
3277    'audio',
3278    'backends',
3279    'backends/tpm',
3280    'chardev',
3281    'ebpf',
3282    'hw/9pfs',
3283    'hw/acpi',
3284    'hw/adc',
3285    'hw/alpha',
3286    'hw/arm',
3287    'hw/audio',
3288    'hw/block',
3289    'hw/char',
3290    'hw/display',
3291    'hw/dma',
3292    'hw/hyperv',
3293    'hw/i2c',
3294    'hw/i386',
3295    'hw/i386/xen',
3296    'hw/i386/kvm',
3297    'hw/ide',
3298    'hw/input',
3299    'hw/intc',
3300    'hw/isa',
3301    'hw/mem',
3302    'hw/mips',
3303    'hw/misc',
3304    'hw/misc/macio',
3305    'hw/net',
3306    'hw/net/can',
3307    'hw/nubus',
3308    'hw/nvme',
3309    'hw/nvram',
3310    'hw/pci',
3311    'hw/pci-host',
3312    'hw/ppc',
3313    'hw/rdma',
3314    'hw/rdma/vmw',
3315    'hw/rtc',
3316    'hw/s390x',
3317    'hw/scsi',
3318    'hw/sd',
3319    'hw/sh4',
3320    'hw/sparc',
3321    'hw/sparc64',
3322    'hw/ssi',
3323    'hw/timer',
3324    'hw/tpm',
3325    'hw/ufs',
3326    'hw/usb',
3327    'hw/vfio',
3328    'hw/virtio',
3329    'hw/watchdog',
3330    'hw/xen',
3331    'hw/gpio',
3332    'migration',
3333    'net',
3334    'system',
3335    'ui',
3336    'hw/remote',
3337  ]
3338endif
3339if have_system or have_user
3340  trace_events_subdirs += [
3341    'accel/tcg',
3342    'hw/core',
3343    'target/arm',
3344    'target/arm/hvf',
3345    'target/hppa',
3346    'target/i386',
3347    'target/i386/kvm',
3348    'target/loongarch',
3349    'target/mips/tcg',
3350    'target/nios2',
3351    'target/ppc',
3352    'target/riscv',
3353    'target/s390x',
3354    'target/s390x/kvm',
3355    'target/sparc',
3356  ]
3357endif
3358
3359###################
3360# Collect sources #
3361###################
3362
3363authz_ss = ss.source_set()
3364blockdev_ss = ss.source_set()
3365block_ss = ss.source_set()
3366chardev_ss = ss.source_set()
3367common_ss = ss.source_set()
3368crypto_ss = ss.source_set()
3369hwcore_ss = ss.source_set()
3370io_ss = ss.source_set()
3371qmp_ss = ss.source_set()
3372qom_ss = ss.source_set()
3373system_ss = ss.source_set()
3374specific_fuzz_ss = ss.source_set()
3375specific_ss = ss.source_set()
3376stub_ss = ss.source_set()
3377trace_ss = ss.source_set()
3378user_ss = ss.source_set()
3379util_ss = ss.source_set()
3380
3381# accel modules
3382qtest_module_ss = ss.source_set()
3383tcg_module_ss = ss.source_set()
3384
3385modules = {}
3386target_modules = {}
3387hw_arch = {}
3388target_arch = {}
3389target_system_arch = {}
3390target_user_arch = {}
3391
3392# NOTE: the trace/ subdirectory needs the qapi_trace_events variable
3393# that is filled in by qapi/.
3394subdir('qapi')
3395subdir('qobject')
3396subdir('stubs')
3397subdir('trace')
3398subdir('util')
3399subdir('qom')
3400subdir('authz')
3401subdir('crypto')
3402subdir('ui')
3403subdir('hw')
3404subdir('gdbstub')
3405
3406if enable_modules
3407  libmodulecommon = static_library('module-common', files('module-common.c') + genh, pic: true, c_args: '-DBUILD_DSO')
3408  modulecommon = declare_dependency(link_whole: libmodulecommon, compile_args: '-DBUILD_DSO')
3409endif
3410
3411qom_ss = qom_ss.apply({})
3412libqom = static_library('qom', qom_ss.sources() + genh,
3413                        dependencies: [qom_ss.dependencies()],
3414                        name_suffix: 'fa',
3415                        build_by_default: false)
3416qom = declare_dependency(link_whole: libqom)
3417
3418event_loop_base = files('event-loop-base.c')
3419event_loop_base = static_library('event-loop-base',
3420                                 sources: event_loop_base + genh,
3421                                 name_suffix: 'fa',
3422                                 build_by_default: false)
3423event_loop_base = declare_dependency(link_whole: event_loop_base,
3424                                     dependencies: [qom])
3425
3426stub_ss = stub_ss.apply({})
3427
3428util_ss.add_all(trace_ss)
3429util_ss = util_ss.apply({})
3430libqemuutil = static_library('qemuutil',
3431                             build_by_default: false,
3432                             sources: util_ss.sources() + stub_ss.sources() + genh,
3433                             dependencies: [util_ss.dependencies(), libm, threads, glib, socket, malloc, pixman])
3434qemuutil = declare_dependency(link_with: libqemuutil,
3435                              sources: genh + version_res,
3436                              dependencies: [event_loop_base])
3437
3438if have_system or have_user
3439  decodetree = generator(find_program('scripts/decodetree.py'),
3440                         output: 'decode-@BASENAME@.c.inc',
3441                         arguments: ['@INPUT@', '@EXTRA_ARGS@', '-o', '@OUTPUT@'])
3442  subdir('libdecnumber')
3443  subdir('target')
3444endif
3445
3446subdir('audio')
3447subdir('io')
3448subdir('chardev')
3449subdir('fsdev')
3450subdir('dump')
3451
3452if have_block
3453  block_ss.add(files(
3454    'block.c',
3455    'blockjob.c',
3456    'job.c',
3457    'qemu-io-cmds.c',
3458  ))
3459  if config_host_data.get('CONFIG_REPLICATION')
3460    block_ss.add(files('replication.c'))
3461  endif
3462
3463  subdir('nbd')
3464  subdir('scsi')
3465  subdir('block')
3466
3467  blockdev_ss.add(files(
3468    'blockdev.c',
3469    'blockdev-nbd.c',
3470    'iothread.c',
3471    'job-qmp.c',
3472  ), gnutls)
3473
3474  # os-posix.c contains POSIX-specific functions used by qemu-storage-daemon,
3475  # os-win32.c does not
3476  if host_os == 'windows'
3477    system_ss.add(files('os-win32.c'))
3478  else
3479    blockdev_ss.add(files('os-posix.c'))
3480  endif
3481endif
3482
3483common_ss.add(files('cpu-common.c'))
3484specific_ss.add(files('cpu-target.c'))
3485
3486subdir('system')
3487
3488# Work around a gcc bug/misfeature wherein constant propagation looks
3489# through an alias:
3490#   https://gcc.gnu.org/bugzilla/show_bug.cgi?id=99696
3491# to guess that a const variable is always zero.  Without lto, this is
3492# impossible, as the alias is restricted to page-vary-common.c.  Indeed,
3493# without lto, not even the alias is required -- we simply use different
3494# declarations in different compilation units.
3495pagevary = files('page-vary-common.c')
3496if get_option('b_lto')
3497  pagevary_flags = ['-fno-lto']
3498  if get_option('cfi')
3499    pagevary_flags += '-fno-sanitize=cfi-icall'
3500  endif
3501  pagevary = static_library('page-vary-common', sources: pagevary + genh,
3502                            c_args: pagevary_flags)
3503  pagevary = declare_dependency(link_with: pagevary)
3504endif
3505common_ss.add(pagevary)
3506specific_ss.add(files('page-vary-target.c'))
3507
3508subdir('backends')
3509subdir('disas')
3510subdir('migration')
3511subdir('monitor')
3512subdir('net')
3513subdir('replay')
3514subdir('semihosting')
3515subdir('stats')
3516subdir('tcg')
3517subdir('fpu')
3518subdir('accel')
3519subdir('plugins')
3520subdir('ebpf')
3521
3522common_user_inc = []
3523
3524subdir('common-user')
3525subdir('bsd-user')
3526subdir('linux-user')
3527
3528# needed for fuzzing binaries
3529subdir('tests/qtest/libqos')
3530subdir('tests/qtest/fuzz')
3531
3532# accel modules
3533tcg_real_module_ss = ss.source_set()
3534tcg_real_module_ss.add_all(when: 'CONFIG_TCG_MODULAR', if_true: tcg_module_ss)
3535specific_ss.add_all(when: 'CONFIG_TCG_BUILTIN', if_true: tcg_module_ss)
3536target_modules += { 'accel' : { 'qtest': qtest_module_ss,
3537                                'tcg': tcg_real_module_ss }}
3538
3539##############################################
3540# Internal static_libraries and dependencies #
3541##############################################
3542
3543modinfo_collect = find_program('scripts/modinfo-collect.py')
3544modinfo_generate = find_program('scripts/modinfo-generate.py')
3545modinfo_files = []
3546
3547block_mods = []
3548system_mods = []
3549foreach d, list : modules
3550  if not (d == 'block' ? have_block : have_system)
3551    continue
3552  endif
3553
3554  foreach m, module_ss : list
3555    if enable_modules
3556      module_ss = module_ss.apply(config_all_devices, strict: false)
3557      sl = static_library(d + '-' + m, [genh, module_ss.sources()],
3558                          dependencies: [modulecommon, module_ss.dependencies()], pic: true)
3559      if d == 'block'
3560        block_mods += sl
3561      else
3562        system_mods += sl
3563      endif
3564      if module_ss.sources() != []
3565        # FIXME: Should use sl.extract_all_objects(recursive: true) as
3566        # input. Sources can be used multiple times but objects are
3567        # unique when it comes to lookup in compile_commands.json.
3568        # Depnds on a mesion version with
3569        # https://github.com/mesonbuild/meson/pull/8900
3570        modinfo_files += custom_target(d + '-' + m + '.modinfo',
3571                                       output: d + '-' + m + '.modinfo',
3572                                       input: module_ss.sources() + genh,
3573                                       capture: true,
3574                                       command: [modinfo_collect, module_ss.sources()])
3575      endif
3576    else
3577      if d == 'block'
3578        block_ss.add_all(module_ss)
3579      else
3580        system_ss.add_all(module_ss)
3581      endif
3582    endif
3583  endforeach
3584endforeach
3585
3586foreach d, list : target_modules
3587  foreach m, module_ss : list
3588    if enable_modules
3589      foreach target : target_dirs
3590        if target.endswith('-softmmu')
3591          config_target = config_target_mak[target]
3592          target_inc = [include_directories('target' / config_target['TARGET_BASE_ARCH'])]
3593          c_args = ['-DNEED_CPU_H',
3594                    '-DCONFIG_TARGET="@0@-config-target.h"'.format(target),
3595                    '-DCONFIG_DEVICES="@0@-config-devices.h"'.format(target)]
3596          target_module_ss = module_ss.apply(config_target, strict: false)
3597          if target_module_ss.sources() != []
3598            module_name = d + '-' + m + '-' + config_target['TARGET_NAME']
3599            sl = static_library(module_name,
3600                                [genh, target_module_ss.sources()],
3601                                dependencies: [modulecommon, target_module_ss.dependencies()],
3602                                include_directories: target_inc,
3603                                c_args: c_args,
3604                                pic: true)
3605            system_mods += sl
3606            # FIXME: Should use sl.extract_all_objects(recursive: true) too.
3607            modinfo_files += custom_target(module_name + '.modinfo',
3608                                           output: module_name + '.modinfo',
3609                                           input: target_module_ss.sources() + genh,
3610                                           capture: true,
3611                                           command: [modinfo_collect, '--target', target, target_module_ss.sources()])
3612          endif
3613        endif
3614      endforeach
3615    else
3616      specific_ss.add_all(module_ss)
3617    endif
3618  endforeach
3619endforeach
3620
3621if enable_modules
3622  foreach target : target_dirs
3623    if target.endswith('-softmmu')
3624      config_target = config_target_mak[target]
3625      config_devices_mak = target + '-config-devices.mak'
3626      modinfo_src = custom_target('modinfo-' + target + '.c',
3627                                  output: 'modinfo-' + target + '.c',
3628                                  input: modinfo_files,
3629                                  command: [modinfo_generate, '--devices', config_devices_mak, '@INPUT@'],
3630                                  capture: true)
3631
3632      modinfo_lib = static_library('modinfo-' + target + '.c', modinfo_src)
3633      modinfo_dep = declare_dependency(link_with: modinfo_lib)
3634
3635      arch = config_target['TARGET_NAME'] == 'sparc64' ? 'sparc64' : config_target['TARGET_BASE_ARCH']
3636      hw_arch[arch].add(modinfo_dep)
3637    endif
3638  endforeach
3639endif
3640
3641nm = find_program('nm')
3642undefsym = find_program('scripts/undefsym.py')
3643block_syms = custom_target('block.syms', output: 'block.syms',
3644                             input: [libqemuutil, block_mods],
3645                             capture: true,
3646                             command: [undefsym, nm, '@INPUT@'])
3647qemu_syms = custom_target('qemu.syms', output: 'qemu.syms',
3648                             input: [libqemuutil, system_mods],
3649                             capture: true,
3650                             command: [undefsym, nm, '@INPUT@'])
3651
3652authz_ss = authz_ss.apply({})
3653libauthz = static_library('authz', authz_ss.sources() + genh,
3654                          dependencies: [authz_ss.dependencies()],
3655                          name_suffix: 'fa',
3656                          build_by_default: false)
3657
3658authz = declare_dependency(link_whole: libauthz,
3659                           dependencies: qom)
3660
3661crypto_ss = crypto_ss.apply({})
3662libcrypto = static_library('crypto', crypto_ss.sources() + genh,
3663                           dependencies: [crypto_ss.dependencies()],
3664                           name_suffix: 'fa',
3665                           build_by_default: false)
3666
3667crypto = declare_dependency(link_whole: libcrypto,
3668                            dependencies: [authz, qom])
3669
3670io_ss = io_ss.apply({})
3671libio = static_library('io', io_ss.sources() + genh,
3672                       dependencies: [io_ss.dependencies()],
3673                       link_with: libqemuutil,
3674                       name_suffix: 'fa',
3675                       build_by_default: false)
3676
3677io = declare_dependency(link_whole: libio, dependencies: [crypto, qom])
3678
3679libmigration = static_library('migration', sources: migration_files + genh,
3680                              name_suffix: 'fa',
3681                              build_by_default: false)
3682migration = declare_dependency(link_with: libmigration,
3683                               dependencies: [zlib, qom, io])
3684system_ss.add(migration)
3685
3686block_ss = block_ss.apply({})
3687libblock = static_library('block', block_ss.sources() + genh,
3688                          dependencies: block_ss.dependencies(),
3689                          link_depends: block_syms,
3690                          name_suffix: 'fa',
3691                          build_by_default: false)
3692
3693block = declare_dependency(link_whole: [libblock],
3694                           link_args: '@block.syms',
3695                           dependencies: [crypto, io])
3696
3697blockdev_ss = blockdev_ss.apply({})
3698libblockdev = static_library('blockdev', blockdev_ss.sources() + genh,
3699                             dependencies: blockdev_ss.dependencies(),
3700                             name_suffix: 'fa',
3701                             build_by_default: false)
3702
3703blockdev = declare_dependency(link_whole: [libblockdev],
3704                              dependencies: [block, event_loop_base])
3705
3706qmp_ss = qmp_ss.apply({})
3707libqmp = static_library('qmp', qmp_ss.sources() + genh,
3708                        dependencies: qmp_ss.dependencies(),
3709                        name_suffix: 'fa',
3710                        build_by_default: false)
3711
3712qmp = declare_dependency(link_whole: [libqmp])
3713
3714libchardev = static_library('chardev', chardev_ss.sources() + genh,
3715                            name_suffix: 'fa',
3716                            dependencies: chardev_ss.dependencies(),
3717                            build_by_default: false)
3718
3719chardev = declare_dependency(link_whole: libchardev)
3720
3721hwcore_ss = hwcore_ss.apply({})
3722libhwcore = static_library('hwcore', sources: hwcore_ss.sources() + genh,
3723                           name_suffix: 'fa',
3724                           build_by_default: false)
3725hwcore = declare_dependency(link_whole: libhwcore)
3726common_ss.add(hwcore)
3727
3728###########
3729# Targets #
3730###########
3731
3732emulator_modules = []
3733foreach m : block_mods + system_mods
3734  emulator_modules += shared_module(m.name(),
3735                build_by_default: true,
3736                name_prefix: '',
3737                link_whole: m,
3738                install: true,
3739                install_dir: qemu_moddir)
3740endforeach
3741if emulator_modules.length() > 0
3742  alias_target('modules', emulator_modules)
3743endif
3744
3745system_ss.add(authz, blockdev, chardev, crypto, io, qmp)
3746common_ss.add(qom, qemuutil)
3747
3748common_ss.add_all(when: 'CONFIG_SYSTEM_ONLY', if_true: [system_ss])
3749common_ss.add_all(when: 'CONFIG_USER_ONLY', if_true: user_ss)
3750
3751# Note that this library is never used directly (only through extract_objects)
3752# and is not built by default; therefore, source files not used by the build
3753# configuration will be in build.ninja, but are never built by default.
3754common_all = static_library('common',
3755                            build_by_default: false,
3756                            sources: common_ss.all_sources() + genh,
3757                            include_directories: common_user_inc,
3758                            implicit_include_directories: false,
3759                            dependencies: common_ss.all_dependencies(),
3760                            name_suffix: 'fa')
3761
3762feature_to_c = find_program('scripts/feature_to_c.py')
3763
3764if host_os == 'darwin'
3765  entitlement = find_program('scripts/entitlement.sh')
3766endif
3767
3768emulators = {}
3769foreach target : target_dirs
3770  config_target = config_target_mak[target]
3771  target_name = config_target['TARGET_NAME']
3772  target_base_arch = config_target['TARGET_BASE_ARCH']
3773  arch_srcs = [config_target_h[target]]
3774  arch_deps = []
3775  c_args = ['-DNEED_CPU_H',
3776            '-DCONFIG_TARGET="@0@-config-target.h"'.format(target),
3777            '-DCONFIG_DEVICES="@0@-config-devices.h"'.format(target)]
3778  link_args = emulator_link_args
3779
3780  target_inc = [include_directories('target' / config_target['TARGET_BASE_ARCH'])]
3781  if host_os == 'linux'
3782    target_inc += include_directories('linux-headers', is_system: true)
3783  endif
3784  if target.endswith('-softmmu')
3785    target_type='system'
3786    t = target_system_arch[target_base_arch].apply(config_target, strict: false)
3787    arch_srcs += t.sources()
3788    arch_deps += t.dependencies()
3789
3790    hw_dir = target_name == 'sparc64' ? 'sparc64' : target_base_arch
3791    if hw_arch.has_key(hw_dir)
3792      hw = hw_arch[hw_dir].apply(config_target, strict: false)
3793      arch_srcs += hw.sources()
3794      arch_deps += hw.dependencies()
3795    endif
3796
3797    arch_srcs += config_devices_h[target]
3798    link_args += ['@block.syms', '@qemu.syms']
3799  else
3800    abi = config_target['TARGET_ABI_DIR']
3801    target_type='user'
3802    target_inc += common_user_inc
3803    if target_base_arch in target_user_arch
3804      t = target_user_arch[target_base_arch].apply(config_target, strict: false)
3805      arch_srcs += t.sources()
3806      arch_deps += t.dependencies()
3807    endif
3808    if 'CONFIG_LINUX_USER' in config_target
3809      base_dir = 'linux-user'
3810    endif
3811    if 'CONFIG_BSD_USER' in config_target
3812      base_dir = 'bsd-user'
3813      target_inc += include_directories('bsd-user/' / host_os)
3814      target_inc += include_directories('bsd-user/host/' / host_arch)
3815      dir = base_dir / abi
3816      arch_srcs += files(dir / 'signal.c', dir / 'target_arch_cpu.c')
3817    endif
3818    target_inc += include_directories(
3819      base_dir,
3820      base_dir / abi,
3821    )
3822    if 'CONFIG_LINUX_USER' in config_target
3823      dir = base_dir / abi
3824      arch_srcs += files(dir / 'signal.c', dir / 'cpu_loop.c')
3825      if config_target.has_key('TARGET_SYSTBL_ABI')
3826        arch_srcs += \
3827          syscall_nr_generators[abi].process(base_dir / abi / config_target['TARGET_SYSTBL'],
3828                                             extra_args : config_target['TARGET_SYSTBL_ABI'])
3829      endif
3830    endif
3831  endif
3832
3833  if 'TARGET_XML_FILES' in config_target
3834    gdbstub_xml = custom_target(target + '-gdbstub-xml.c',
3835                                output: target + '-gdbstub-xml.c',
3836                                input: files(config_target['TARGET_XML_FILES'].split()),
3837                                command: [feature_to_c, '@INPUT@'],
3838                                capture: true)
3839    arch_srcs += gdbstub_xml
3840  endif
3841
3842  t = target_arch[target_base_arch].apply(config_target, strict: false)
3843  arch_srcs += t.sources()
3844  arch_deps += t.dependencies()
3845
3846  target_common = common_ss.apply(config_target, strict: false)
3847  objects = common_all.extract_objects(target_common.sources())
3848  deps = target_common.dependencies()
3849
3850  target_specific = specific_ss.apply(config_target, strict: false)
3851  arch_srcs += target_specific.sources()
3852  arch_deps += target_specific.dependencies()
3853
3854  lib = static_library('qemu-' + target,
3855                 sources: arch_srcs + genh,
3856                 dependencies: arch_deps,
3857                 objects: objects,
3858                 include_directories: target_inc,
3859                 c_args: c_args,
3860                 build_by_default: false,
3861                 name_suffix: 'fa')
3862
3863  if target.endswith('-softmmu')
3864    execs = [{
3865      'name': 'qemu-system-' + target_name,
3866      'win_subsystem': 'console',
3867      'sources': files('system/main.c'),
3868      'dependencies': []
3869    }]
3870    if host_os == 'windows' and (sdl.found() or gtk.found())
3871      execs += [{
3872        'name': 'qemu-system-' + target_name + 'w',
3873        'win_subsystem': 'windows',
3874        'sources': files('system/main.c'),
3875        'dependencies': []
3876      }]
3877    endif
3878    if get_option('fuzzing')
3879      specific_fuzz = specific_fuzz_ss.apply(config_target, strict: false)
3880      execs += [{
3881        'name': 'qemu-fuzz-' + target_name,
3882        'win_subsystem': 'console',
3883        'sources': specific_fuzz.sources(),
3884        'dependencies': specific_fuzz.dependencies(),
3885      }]
3886    endif
3887  else
3888    execs = [{
3889      'name': 'qemu-' + target_name,
3890      'win_subsystem': 'console',
3891      'sources': [],
3892      'dependencies': []
3893    }]
3894  endif
3895  foreach exe: execs
3896    exe_name = exe['name']
3897    if host_os == 'darwin'
3898      exe_name += '-unsigned'
3899    endif
3900
3901    emulator = executable(exe_name, exe['sources'],
3902               install: true,
3903               c_args: c_args,
3904               dependencies: arch_deps + deps + exe['dependencies'],
3905               objects: lib.extract_all_objects(recursive: true),
3906               link_depends: [block_syms, qemu_syms] + exe.get('link_depends', []),
3907               link_args: link_args,
3908               win_subsystem: exe['win_subsystem'])
3909
3910    if host_os == 'darwin'
3911      icon = 'pc-bios/qemu.rsrc'
3912      build_input = [emulator, files(icon)]
3913      install_input = [
3914        get_option('bindir') / exe_name,
3915        meson.current_source_dir() / icon
3916      ]
3917      if 'CONFIG_HVF' in config_target
3918        entitlements = 'accel/hvf/entitlements.plist'
3919        build_input += files(entitlements)
3920        install_input += meson.current_source_dir() / entitlements
3921      endif
3922
3923      emulators += {exe['name'] : custom_target(exe['name'],
3924                   input: build_input,
3925                   output: exe['name'],
3926                   command: [entitlement, '@OUTPUT@', '@INPUT@'])
3927      }
3928
3929      meson.add_install_script(entitlement, '--install',
3930                               get_option('bindir') / exe['name'],
3931                               install_input)
3932    else
3933      emulators += {exe['name']: emulator}
3934    endif
3935
3936    if stap.found()
3937      foreach stp: [
3938        {'ext': '.stp-build', 'fmt': 'stap', 'bin': meson.current_build_dir() / exe['name'], 'install': false},
3939        {'ext': '.stp', 'fmt': 'stap', 'bin': get_option('prefix') / get_option('bindir') / exe['name'], 'install': true},
3940        {'ext': '-simpletrace.stp', 'fmt': 'simpletrace-stap', 'bin': '', 'install': true},
3941        {'ext': '-log.stp', 'fmt': 'log-stap', 'bin': '', 'install': true},
3942      ]
3943        custom_target(exe['name'] + stp['ext'],
3944                      input: trace_events_all,
3945                      output: exe['name'] + stp['ext'],
3946                      install: stp['install'],
3947                      install_dir: get_option('datadir') / 'systemtap/tapset',
3948                      command: [
3949                        tracetool, '--group=all', '--format=' + stp['fmt'],
3950                        '--binary=' + stp['bin'],
3951                        '--target-name=' + target_name,
3952                        '--target-type=' + target_type,
3953                        '--probe-prefix=qemu.' + target_type + '.' + target_name,
3954                        '@INPUT@', '@OUTPUT@'
3955                      ],
3956                      depend_files: tracetool_depends)
3957      endforeach
3958    endif
3959  endforeach
3960endforeach
3961
3962# Other build targets
3963
3964if get_option('plugins')
3965  install_headers('include/qemu/qemu-plugin.h')
3966  if host_os == 'windows'
3967    # On windows, we want to deliver the qemu_plugin_api.lib file in the qemu installer,
3968    # so that plugin authors can compile against it.
3969    install_data(win32_qemu_plugin_api_lib, install_dir: 'lib')
3970  endif
3971endif
3972
3973subdir('qga')
3974
3975# Don't build qemu-keymap if xkbcommon is not explicitly enabled
3976# when we don't build tools or system
3977if xkbcommon.found()
3978  # used for the update-keymaps target, so include rules even if !have_tools
3979  qemu_keymap = executable('qemu-keymap', files('qemu-keymap.c', 'ui/input-keymap.c') + genh,
3980                           dependencies: [qemuutil, xkbcommon], install: have_tools)
3981endif
3982
3983if have_tools
3984  qemu_img = executable('qemu-img', [files('qemu-img.c'), hxdep],
3985             dependencies: [authz, block, crypto, io, qom, qemuutil], install: true)
3986  qemu_io = executable('qemu-io', files('qemu-io.c'),
3987             dependencies: [block, qemuutil], install: true)
3988  qemu_nbd = executable('qemu-nbd', files('qemu-nbd.c'),
3989               dependencies: [blockdev, qemuutil, gnutls, selinux],
3990               install: true)
3991
3992  subdir('storage-daemon')
3993  subdir('contrib/rdmacm-mux')
3994  subdir('contrib/elf2dmp')
3995
3996  executable('qemu-edid', files('qemu-edid.c', 'hw/display/edid-generate.c'),
3997             dependencies: qemuutil,
3998             install: true)
3999
4000  if have_vhost_user
4001    subdir('contrib/vhost-user-blk')
4002    subdir('contrib/vhost-user-gpu')
4003    subdir('contrib/vhost-user-input')
4004    subdir('contrib/vhost-user-scsi')
4005  endif
4006
4007  if host_os == 'linux'
4008    executable('qemu-bridge-helper', files('qemu-bridge-helper.c'),
4009               dependencies: [qemuutil, libcap_ng],
4010               install: true,
4011               install_dir: get_option('libexecdir'))
4012
4013    executable('qemu-pr-helper', files('scsi/qemu-pr-helper.c', 'scsi/utils.c'),
4014               dependencies: [authz, crypto, io, qom, qemuutil,
4015                              libcap_ng, mpathpersist],
4016               install: true)
4017  endif
4018
4019  if have_ivshmem
4020    subdir('contrib/ivshmem-client')
4021    subdir('contrib/ivshmem-server')
4022  endif
4023endif
4024
4025subdir('scripts')
4026subdir('tools')
4027subdir('pc-bios')
4028subdir('docs')
4029subdir('tests')
4030if gtk.found()
4031  subdir('po')
4032endif
4033
4034if host_machine.system() == 'windows'
4035  nsis_cmd = [
4036    find_program('scripts/nsis.py'),
4037    '@OUTPUT@',
4038    get_option('prefix'),
4039    meson.current_source_dir(),
4040    glib_pc.get_variable('bindir'),
4041    host_machine.cpu(),
4042    '--',
4043    '-DDISPLAYVERSION=' + meson.project_version(),
4044  ]
4045  if build_docs
4046    nsis_cmd += '-DCONFIG_DOCUMENTATION=y'
4047  endif
4048  if gtk.found()
4049    nsis_cmd += '-DCONFIG_GTK=y'
4050  endif
4051
4052  nsis = custom_target('nsis',
4053                       output: 'qemu-setup-' + meson.project_version() + '.exe',
4054                       input: files('qemu.nsi'),
4055                       build_always_stale: true,
4056                       command: nsis_cmd + ['@INPUT@'])
4057  alias_target('installer', nsis)
4058endif
4059
4060#########################
4061# Configuration summary #
4062#########################
4063
4064# Build environment
4065summary_info = {}
4066summary_info += {'Build directory':   meson.current_build_dir()}
4067summary_info += {'Source path':       meson.current_source_dir()}
4068summary_info += {'Download dependencies': get_option('wrap_mode') != 'nodownload'}
4069summary(summary_info, bool_yn: true, section: 'Build environment')
4070
4071# Directories
4072summary_info += {'Install prefix':    get_option('prefix')}
4073summary_info += {'BIOS directory':    qemu_datadir}
4074pathsep = host_os == 'windows' ? ';' : ':'
4075summary_info += {'firmware path':     pathsep.join(get_option('qemu_firmwarepath'))}
4076summary_info += {'binary directory':  get_option('prefix') / get_option('bindir')}
4077summary_info += {'library directory': get_option('prefix') / get_option('libdir')}
4078summary_info += {'module directory':  qemu_moddir}
4079summary_info += {'libexec directory': get_option('prefix') / get_option('libexecdir')}
4080summary_info += {'include directory': get_option('prefix') / get_option('includedir')}
4081summary_info += {'config directory':  get_option('prefix') / get_option('sysconfdir')}
4082if host_os != 'windows'
4083  summary_info += {'local state directory': get_option('prefix') / get_option('localstatedir')}
4084  summary_info += {'Manual directory':      get_option('prefix') / get_option('mandir')}
4085else
4086  summary_info += {'local state directory': 'queried at runtime'}
4087endif
4088summary_info += {'Doc directory':     get_option('prefix') / get_option('docdir')}
4089summary(summary_info, bool_yn: true, section: 'Directories')
4090
4091# Host binaries
4092summary_info = {}
4093summary_info += {'python':            '@0@ (version: @1@)'.format(python.full_path(), python.language_version())}
4094summary_info += {'sphinx-build':      sphinx_build}
4095
4096# FIXME: the [binaries] section of machine files, which can be probed
4097# with find_program(), would be great for passing gdb and genisoimage
4098# paths from configure to Meson.  However, there seems to be no way to
4099# hide a program (for example if gdb is too old).
4100if config_host.has_key('GDB')
4101  summary_info += {'gdb':             config_host['GDB']}
4102endif
4103summary_info += {'iasl':              iasl}
4104summary_info += {'genisoimage':       config_host['GENISOIMAGE']}
4105if host_os == 'windows' and have_ga
4106  summary_info += {'wixl':            wixl}
4107endif
4108if slirp.found() and have_system
4109  summary_info += {'smbd':            have_slirp_smbd ? smbd_path : false}
4110endif
4111summary(summary_info, bool_yn: true, section: 'Host binaries')
4112
4113# Configurable features
4114summary_info = {}
4115summary_info += {'Documentation':     build_docs}
4116summary_info += {'system-mode emulation': have_system}
4117summary_info += {'user-mode emulation': have_user}
4118summary_info += {'block layer':       have_block}
4119summary_info += {'Install blobs':     get_option('install_blobs')}
4120summary_info += {'module support':    enable_modules}
4121if enable_modules
4122  summary_info += {'alternative module path': get_option('module_upgrades')}
4123endif
4124summary_info += {'fuzzing support':   get_option('fuzzing')}
4125if have_system
4126  summary_info += {'Audio drivers':     ' '.join(audio_drivers_selected)}
4127endif
4128summary_info += {'Trace backends':    ','.join(get_option('trace_backends'))}
4129if 'simple' in get_option('trace_backends')
4130  summary_info += {'Trace output file': get_option('trace_file') + '-<pid>'}
4131endif
4132summary_info += {'D-Bus display':     dbus_display}
4133summary_info += {'QOM debugging':     get_option('qom_cast_debug')}
4134summary_info += {'Relocatable install': get_option('relocatable')}
4135summary_info += {'vhost-kernel support': have_vhost_kernel}
4136summary_info += {'vhost-net support': have_vhost_net}
4137summary_info += {'vhost-user support': have_vhost_user}
4138summary_info += {'vhost-user-crypto support': have_vhost_user_crypto}
4139summary_info += {'vhost-user-blk server support': have_vhost_user_blk_server}
4140summary_info += {'vhost-vdpa support': have_vhost_vdpa}
4141summary_info += {'build guest agent': have_ga}
4142summary(summary_info, bool_yn: true, section: 'Configurable features')
4143
4144# Compilation information
4145summary_info = {}
4146summary_info += {'host CPU':          cpu}
4147summary_info += {'host endianness':   build_machine.endian()}
4148summary_info += {'C compiler':        ' '.join(meson.get_compiler('c').cmd_array())}
4149summary_info += {'Host C compiler':   ' '.join(meson.get_compiler('c', native: true).cmd_array())}
4150if 'cpp' in all_languages
4151  summary_info += {'C++ compiler':    ' '.join(meson.get_compiler('cpp').cmd_array())}
4152else
4153  summary_info += {'C++ compiler':      false}
4154endif
4155if 'objc' in all_languages
4156  summary_info += {'Objective-C compiler': ' '.join(meson.get_compiler('objc').cmd_array())}
4157else
4158  summary_info += {'Objective-C compiler': false}
4159endif
4160option_cflags = (get_option('debug') ? ['-g'] : [])
4161if get_option('optimization') != 'plain'
4162  option_cflags += ['-O' + get_option('optimization')]
4163endif
4164summary_info += {'CFLAGS':            ' '.join(get_option('c_args') + option_cflags)}
4165if 'cpp' in all_languages
4166  summary_info += {'CXXFLAGS':        ' '.join(get_option('cpp_args') + option_cflags)}
4167endif
4168if 'objc' in all_languages
4169  summary_info += {'OBJCFLAGS':       ' '.join(get_option('objc_args') + option_cflags)}
4170endif
4171link_args = get_option('c_link_args')
4172if link_args.length() > 0
4173  summary_info += {'LDFLAGS':         ' '.join(link_args)}
4174endif
4175summary_info += {'QEMU_CFLAGS':       ' '.join(qemu_common_flags + qemu_cflags)}
4176if 'cpp' in all_languages
4177  summary_info += {'QEMU_CXXFLAGS':     ' '.join(qemu_common_flags + qemu_cxxflags)}
4178endif
4179if 'objc' in all_languages
4180  summary_info += {'QEMU_OBJCFLAGS':    ' '.join(qemu_common_flags)}
4181endif
4182summary_info += {'QEMU_LDFLAGS':      ' '.join(qemu_ldflags)}
4183summary_info += {'link-time optimization (LTO)': get_option('b_lto')}
4184summary_info += {'PIE':               get_option('b_pie')}
4185summary_info += {'static build':      get_option('prefer_static')}
4186summary_info += {'malloc trim support': has_malloc_trim}
4187summary_info += {'membarrier':        have_membarrier}
4188summary_info += {'debug graph lock':  get_option('debug_graph_lock')}
4189summary_info += {'debug stack usage': get_option('debug_stack_usage')}
4190summary_info += {'mutex debugging':   get_option('debug_mutex')}
4191summary_info += {'memory allocator':  get_option('malloc')}
4192summary_info += {'avx2 optimization': config_host_data.get('CONFIG_AVX2_OPT')}
4193summary_info += {'avx512bw optimization': config_host_data.get('CONFIG_AVX512BW_OPT')}
4194summary_info += {'avx512f optimization': config_host_data.get('CONFIG_AVX512F_OPT')}
4195summary_info += {'gcov':              get_option('b_coverage')}
4196summary_info += {'thread sanitizer':  get_option('tsan')}
4197summary_info += {'CFI support':       get_option('cfi')}
4198if get_option('cfi')
4199  summary_info += {'CFI debug support': get_option('cfi_debug')}
4200endif
4201summary_info += {'strip binaries':    get_option('strip')}
4202summary_info += {'sparse':            sparse}
4203summary_info += {'mingw32 support':   host_os == 'windows'}
4204summary(summary_info, bool_yn: true, section: 'Compilation')
4205
4206# snarf the cross-compilation information for tests
4207summary_info = {}
4208have_cross = false
4209foreach target: target_dirs
4210  tcg_mak = meson.current_build_dir() / 'tests/tcg' / target / 'config-target.mak'
4211  if fs.exists(tcg_mak)
4212    config_cross_tcg = keyval.load(tcg_mak)
4213    if 'CC' in config_cross_tcg
4214      summary_info += {config_cross_tcg['TARGET_NAME']: config_cross_tcg['CC']}
4215      have_cross = true
4216    endif
4217  endif
4218endforeach
4219if have_cross
4220  summary(summary_info, bool_yn: true, section: 'Cross compilers')
4221endif
4222
4223# Targets and accelerators
4224summary_info = {}
4225if have_system
4226  summary_info += {'KVM support':       config_all_accel.has_key('CONFIG_KVM')}
4227  summary_info += {'HVF support':       config_all_accel.has_key('CONFIG_HVF')}
4228  summary_info += {'WHPX support':      config_all_accel.has_key('CONFIG_WHPX')}
4229  summary_info += {'NVMM support':      config_all_accel.has_key('CONFIG_NVMM')}
4230  summary_info += {'Xen support':       xen.found()}
4231  if xen.found()
4232    summary_info += {'xen ctrl version':  xen.version()}
4233  endif
4234  summary_info += {'Xen emulation':     config_all_devices.has_key('CONFIG_XEN_EMU')}
4235endif
4236summary_info += {'TCG support':       config_all_accel.has_key('CONFIG_TCG')}
4237if config_all_accel.has_key('CONFIG_TCG')
4238  if get_option('tcg_interpreter')
4239    summary_info += {'TCG backend':   'TCI (TCG with bytecode interpreter, slow)'}
4240  else
4241    summary_info += {'TCG backend':   'native (@0@)'.format(cpu)}
4242  endif
4243  summary_info += {'TCG plugins':       get_option('plugins')}
4244  summary_info += {'TCG debug enabled': get_option('debug_tcg')}
4245endif
4246summary_info += {'target list':       ' '.join(target_dirs)}
4247if have_system
4248  summary_info += {'default devices':   get_option('default_devices')}
4249  summary_info += {'out of process emulation': multiprocess_allowed}
4250  summary_info += {'vfio-user server': vfio_user_server_allowed}
4251endif
4252summary(summary_info, bool_yn: true, section: 'Targets and accelerators')
4253
4254# Block layer
4255summary_info = {}
4256summary_info += {'coroutine backend': coroutine_backend}
4257summary_info += {'coroutine pool':    have_coroutine_pool}
4258if have_block
4259  summary_info += {'Block whitelist (rw)': get_option('block_drv_rw_whitelist')}
4260  summary_info += {'Block whitelist (ro)': get_option('block_drv_ro_whitelist')}
4261  summary_info += {'Use block whitelist in tools': get_option('block_drv_whitelist_in_tools')}
4262  summary_info += {'VirtFS (9P) support':    have_virtfs}
4263  summary_info += {'VirtFS (9P) Proxy Helper support (deprecated)': have_virtfs_proxy_helper}
4264  summary_info += {'Live block migration': config_host_data.get('CONFIG_LIVE_BLOCK_MIGRATION')}
4265  summary_info += {'replication support': config_host_data.get('CONFIG_REPLICATION')}
4266  summary_info += {'bochs support':     get_option('bochs').allowed()}
4267  summary_info += {'cloop support':     get_option('cloop').allowed()}
4268  summary_info += {'dmg support':       get_option('dmg').allowed()}
4269  summary_info += {'qcow v1 support':   get_option('qcow1').allowed()}
4270  summary_info += {'vdi support':       get_option('vdi').allowed()}
4271  summary_info += {'vhdx support':      get_option('vhdx').allowed()}
4272  summary_info += {'vmdk support':      get_option('vmdk').allowed()}
4273  summary_info += {'vpc support':       get_option('vpc').allowed()}
4274  summary_info += {'vvfat support':     get_option('vvfat').allowed()}
4275  summary_info += {'qed support':       get_option('qed').allowed()}
4276  summary_info += {'parallels support': get_option('parallels').allowed()}
4277  summary_info += {'FUSE exports':      fuse}
4278  summary_info += {'VDUSE block exports': have_vduse_blk_export}
4279endif
4280summary(summary_info, bool_yn: true, section: 'Block layer support')
4281
4282# Crypto
4283summary_info = {}
4284summary_info += {'TLS priority':      get_option('tls_priority')}
4285summary_info += {'GNUTLS support':    gnutls}
4286if gnutls.found()
4287  summary_info += {'  GNUTLS crypto':   gnutls_crypto.found()}
4288endif
4289summary_info += {'libgcrypt':         gcrypt}
4290summary_info += {'nettle':            nettle}
4291if nettle.found()
4292   summary_info += {'  XTS':             xts != 'private'}
4293endif
4294summary_info += {'AF_ALG support':    have_afalg}
4295summary_info += {'rng-none':          get_option('rng_none')}
4296summary_info += {'Linux keyring':     have_keyring}
4297summary_info += {'Linux keyutils':    keyutils}
4298summary(summary_info, bool_yn: true, section: 'Crypto')
4299
4300# UI
4301summary_info = {}
4302if host_os == 'darwin'
4303  summary_info += {'Cocoa support':           cocoa}
4304endif
4305summary_info += {'SDL support':       sdl}
4306summary_info += {'SDL image support': sdl_image}
4307summary_info += {'GTK support':       gtk}
4308summary_info += {'pixman':            pixman}
4309summary_info += {'VTE support':       vte}
4310summary_info += {'PNG support':       png}
4311summary_info += {'VNC support':       vnc}
4312if vnc.found()
4313  summary_info += {'VNC SASL support':  sasl}
4314  summary_info += {'VNC JPEG support':  jpeg}
4315endif
4316summary_info += {'spice protocol support': spice_protocol}
4317if spice_protocol.found()
4318  summary_info += {'  spice server support': spice}
4319endif
4320summary_info += {'curses support':    curses}
4321summary_info += {'brlapi support':    brlapi}
4322summary(summary_info, bool_yn: true, section: 'User interface')
4323
4324# Graphics backends
4325summary_info = {}
4326summary_info += {'VirGL support':     virgl}
4327summary_info += {'Rutabaga support':  rutabaga}
4328summary(summary_info, bool_yn: true, section: 'Graphics backends')
4329
4330# Audio backends
4331summary_info = {}
4332if host_os not in ['darwin', 'haiku', 'windows']
4333  summary_info += {'OSS support':     oss}
4334  summary_info += {'sndio support':   sndio}
4335elif host_os == 'darwin'
4336  summary_info += {'CoreAudio support': coreaudio}
4337elif host_os == 'windows'
4338  summary_info += {'DirectSound support': dsound}
4339endif
4340if host_os == 'linux'
4341  summary_info += {'ALSA support':    alsa}
4342  summary_info += {'PulseAudio support': pulse}
4343endif
4344summary_info += {'PipeWire support':  pipewire}
4345summary_info += {'JACK support':      jack}
4346summary(summary_info, bool_yn: true, section: 'Audio backends')
4347
4348# Network backends
4349summary_info = {}
4350if host_os == 'darwin'
4351  summary_info += {'vmnet.framework support': vmnet}
4352endif
4353summary_info += {'AF_XDP support':    libxdp}
4354summary_info += {'slirp support':     slirp}
4355summary_info += {'vde support':       vde}
4356summary_info += {'netmap support':    have_netmap}
4357summary_info += {'l2tpv3 support':    have_l2tpv3}
4358summary(summary_info, bool_yn: true, section: 'Network backends')
4359
4360# Libraries
4361summary_info = {}
4362summary_info += {'libtasn1':          tasn1}
4363summary_info += {'PAM':               pam}
4364summary_info += {'iconv support':     iconv}
4365summary_info += {'blkio support':     blkio}
4366summary_info += {'curl support':      curl}
4367summary_info += {'Multipath support': mpathpersist}
4368summary_info += {'Linux AIO support': libaio}
4369summary_info += {'Linux io_uring support': linux_io_uring}
4370summary_info += {'ATTR/XATTR support': libattr}
4371summary_info += {'RDMA support':      rdma}
4372summary_info += {'PVRDMA support':    have_pvrdma}
4373summary_info += {'fdt support':       fdt_opt == 'disabled' ? false : fdt_opt}
4374summary_info += {'libcap-ng support': libcap_ng}
4375summary_info += {'bpf support':       libbpf}
4376summary_info += {'rbd support':       rbd}
4377summary_info += {'smartcard support': cacard}
4378summary_info += {'U2F support':       u2f}
4379summary_info += {'libusb':            libusb}
4380summary_info += {'usb net redir':     usbredir}
4381summary_info += {'OpenGL support (epoxy)': opengl}
4382summary_info += {'GBM':               gbm}
4383summary_info += {'libiscsi support':  libiscsi}
4384summary_info += {'libnfs support':    libnfs}
4385if host_os == 'windows'
4386  if have_ga
4387    summary_info += {'QGA VSS support':   have_qga_vss}
4388  endif
4389endif
4390summary_info += {'seccomp support':   seccomp}
4391summary_info += {'GlusterFS support': glusterfs}
4392summary_info += {'hv-balloon support': hv_balloon}
4393summary_info += {'TPM support':       have_tpm}
4394summary_info += {'libssh support':    libssh}
4395summary_info += {'lzo support':       lzo}
4396summary_info += {'snappy support':    snappy}
4397summary_info += {'bzip2 support':     libbzip2}
4398summary_info += {'lzfse support':     liblzfse}
4399summary_info += {'zstd support':      zstd}
4400summary_info += {'NUMA host support': numa}
4401summary_info += {'capstone':          capstone}
4402summary_info += {'libpmem support':   libpmem}
4403summary_info += {'libdaxctl support': libdaxctl}
4404summary_info += {'libudev':           libudev}
4405# Dummy dependency, keep .found()
4406summary_info += {'FUSE lseek':        fuse_lseek.found()}
4407summary_info += {'selinux':           selinux}
4408summary_info += {'libdw':             libdw}
4409summary(summary_info, bool_yn: true, section: 'Dependencies')
4410
4411if host_arch == 'unknown'
4412  message()
4413  warning('UNSUPPORTED HOST CPU')
4414  message()
4415  message('Support for CPU host architecture ' + cpu + ' is not currently')
4416  message('maintained. The QEMU project does not guarantee that QEMU will')
4417  message('compile or work on this host CPU. You can help by volunteering')
4418  message('to maintain it and providing a build host for our continuous')
4419  message('integration setup.')
4420  if get_option('tcg').allowed() and target_dirs.length() > 0
4421    message()
4422    message('configure has succeeded and you can continue to build, but')
4423    message('QEMU will use a slow interpreter to emulate the target CPU.')
4424  endif
4425endif
4426
4427if not supported_oses.contains(host_os)
4428  message()
4429  warning('UNSUPPORTED HOST OS')
4430  message()
4431  message('Support for host OS ' + host_os + 'is not currently maintained.')
4432  message('configure has succeeded and you can continue to build, but')
4433  message('the QEMU project does not guarantee that QEMU will compile or')
4434  message('work on this operating system. You can help by volunteering')
4435  message('to maintain it and providing a build host for our continuous')
4436  message('integration setup. This will ensure that future versions of QEMU')
4437  message('will keep working on ' + host_os + '.')
4438endif
4439
4440if host_arch == 'unknown' or not supported_oses.contains(host_os)
4441  message()
4442  message('If you want to help supporting QEMU on this platform, please')
4443  message('contact the developers at qemu-devel@nongnu.org.')
4444endif
4445
4446actually_reloc = get_option('relocatable')
4447# check if get_relocated_path() is actually able to relocate paths
4448if get_option('relocatable') and \
4449  not (get_option('prefix') / get_option('bindir')).startswith(get_option('prefix') / '')
4450  message()
4451  warning('bindir not included within prefix, the installation will not be relocatable.')
4452  actually_reloc = false
4453endif
4454if not actually_reloc and (host_os == 'windows' or get_option('relocatable'))
4455  if host_os == 'windows'
4456    message()
4457    warning('Windows installs should usually be relocatable.')
4458  endif
4459  message()
4460  message('QEMU will have to be installed under ' + get_option('prefix') + '.')
4461  message('Use --disable-relocatable to remove this warning.')
4462endif
4463