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