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