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