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