]> git.proxmox.com Git - mirror_qemu.git/blob - meson.build
linux-aio: add `dev_max_batch` parameter to laio_co_submit()
[mirror_qemu.git] / meson.build
1 project('qemu', ['c'], meson_version: '>=0.58.2',
2 default_options: ['warning_level=1', 'c_std=gnu11', 'cpp_std=gnu++11', 'b_colorout=auto',
3 'b_staticpic=false'],
4 version: files('VERSION'))
5
6 not_found = dependency('', required: false)
7 keyval = import('keyval')
8 ss = import('sourceset')
9 fs = import('fs')
10
11 sh = find_program('sh')
12 cc = meson.get_compiler('c')
13 config_host = keyval.load(meson.current_build_dir() / 'config-host.mak')
14 enable_modules = 'CONFIG_MODULES' in config_host
15 enable_static = 'CONFIG_STATIC' in config_host
16
17 # Allow both shared and static libraries unless --enable-static
18 static_kwargs = enable_static ? {'static': true} : {}
19
20 # Temporary directory used for files created while
21 # configure runs. Since it is in the build directory
22 # we can safely blow away any previous version of it
23 # (and we need not jump through hoops to try to delete
24 # it when configure exits.)
25 tmpdir = meson.current_build_dir() / 'meson-private/temp'
26
27 if get_option('qemu_suffix').startswith('/')
28 error('qemu_suffix cannot start with a /')
29 endif
30
31 qemu_confdir = get_option('sysconfdir') / get_option('qemu_suffix')
32 qemu_datadir = get_option('datadir') / get_option('qemu_suffix')
33 qemu_docdir = get_option('docdir') / get_option('qemu_suffix')
34 qemu_moddir = get_option('libdir') / get_option('qemu_suffix')
35
36 qemu_desktopdir = get_option('datadir') / 'applications'
37 qemu_icondir = get_option('datadir') / 'icons'
38
39 config_host_data = configuration_data()
40 genh = []
41
42 target_dirs = config_host['TARGET_DIRS'].split()
43 have_linux_user = false
44 have_bsd_user = false
45 have_system = false
46 foreach target : target_dirs
47 have_linux_user = have_linux_user or target.endswith('linux-user')
48 have_bsd_user = have_bsd_user or target.endswith('bsd-user')
49 have_system = have_system or target.endswith('-softmmu')
50 endforeach
51 have_user = have_linux_user or have_bsd_user
52 have_tools = 'CONFIG_TOOLS' in config_host
53 have_block = have_system or have_tools
54
55 python = import('python').find_installation()
56
57 supported_oses = ['windows', 'freebsd', 'netbsd', 'openbsd', 'darwin', 'sunos', 'linux']
58 supported_cpus = ['ppc', 'ppc64', 's390x', 'riscv32', 'riscv64', 'x86', 'x86_64',
59 'arm', 'aarch64', 'mips', 'mips64', 'sparc', 'sparc64']
60
61 cpu = host_machine.cpu_family()
62 targetos = host_machine.system()
63
64 if cpu in ['x86', 'x86_64']
65 kvm_targets = ['i386-softmmu', 'x86_64-softmmu']
66 elif cpu == 'aarch64'
67 kvm_targets = ['aarch64-softmmu']
68 elif cpu == 's390x'
69 kvm_targets = ['s390x-softmmu']
70 elif cpu in ['ppc', 'ppc64']
71 kvm_targets = ['ppc-softmmu', 'ppc64-softmmu']
72 elif cpu in ['mips', 'mips64']
73 kvm_targets = ['mips-softmmu', 'mipsel-softmmu', 'mips64-softmmu', 'mips64el-softmmu']
74 else
75 kvm_targets = []
76 endif
77
78 kvm_targets_c = ''
79 if not get_option('kvm').disabled() and targetos == 'linux'
80 kvm_targets_c = '"' + '" ,"'.join(kvm_targets) + '"'
81 endif
82 config_host_data.set('CONFIG_KVM_TARGETS', kvm_targets_c)
83
84 accelerator_targets = { 'CONFIG_KVM': kvm_targets }
85
86 if cpu in ['aarch64']
87 accelerator_targets += {
88 'CONFIG_HVF': ['aarch64-softmmu']
89 }
90 endif
91
92 if cpu in ['x86', 'x86_64', 'arm', 'aarch64']
93 # i386 emulator provides xenpv machine type for multiple architectures
94 accelerator_targets += {
95 'CONFIG_XEN': ['i386-softmmu', 'x86_64-softmmu'],
96 }
97 endif
98 if cpu in ['x86', 'x86_64']
99 accelerator_targets += {
100 'CONFIG_HAX': ['i386-softmmu', 'x86_64-softmmu'],
101 'CONFIG_HVF': ['x86_64-softmmu'],
102 'CONFIG_NVMM': ['i386-softmmu', 'x86_64-softmmu'],
103 'CONFIG_WHPX': ['i386-softmmu', 'x86_64-softmmu'],
104 }
105 endif
106
107 modular_tcg = []
108 # Darwin does not support references to thread-local variables in modules
109 if targetos != 'darwin'
110 modular_tcg = ['i386-softmmu', 'x86_64-softmmu']
111 endif
112
113 edk2_targets = [ 'arm-softmmu', 'aarch64-softmmu', 'i386-softmmu', 'x86_64-softmmu' ]
114 unpack_edk2_blobs = false
115 foreach target : edk2_targets
116 if target in target_dirs
117 bzip2 = find_program('bzip2', required: get_option('install_blobs'))
118 unpack_edk2_blobs = bzip2.found()
119 break
120 endif
121 endforeach
122
123 dtrace = not_found
124 stap = not_found
125 if 'dtrace' in get_option('trace_backends')
126 dtrace = find_program('dtrace', required: true)
127 stap = find_program('stap', required: false)
128 if stap.found()
129 # Workaround to avoid dtrace(1) producing a file with 'hidden' symbol
130 # visibility. Define STAP_SDT_V2 to produce 'default' symbol visibility
131 # instead. QEMU --enable-modules depends on this because the SystemTap
132 # semaphores are linked into the main binary and not the module's shared
133 # object.
134 add_global_arguments('-DSTAP_SDT_V2',
135 native: false, language: ['c', 'cpp', 'objc'])
136 endif
137 endif
138
139 ##################
140 # Compiler flags #
141 ##################
142
143 # Specify linker-script with add_project_link_arguments so that it is not placed
144 # within a linker --start-group/--end-group pair
145 if get_option('fuzzing')
146 add_project_link_arguments(['-Wl,-T,',
147 (meson.current_source_dir() / 'tests/qtest/fuzz/fork_fuzz.ld')],
148 native: false, language: ['c', 'cpp', 'objc'])
149
150 # Specify a filter to only instrument code that is directly related to
151 # virtual-devices.
152 configure_file(output: 'instrumentation-filter',
153 input: 'scripts/oss-fuzz/instrumentation-filter-template',
154 copy: true)
155 add_global_arguments(
156 cc.get_supported_arguments('-fsanitize-coverage-allowlist=instrumentation-filter'),
157 native: false, language: ['c', 'cpp', 'objc'])
158
159 if get_option('fuzzing_engine') == ''
160 # Add CFLAGS to tell clang to add fuzzer-related instrumentation to all the
161 # compiled code. To build non-fuzzer binaries with --enable-fuzzing, link
162 # everything with fsanitize=fuzzer-no-link. Otherwise, the linker will be
163 # unable to bind the fuzzer-related callbacks added by instrumentation.
164 add_global_arguments('-fsanitize=fuzzer-no-link',
165 native: false, language: ['c', 'cpp', 'objc'])
166 add_global_link_arguments('-fsanitize=fuzzer-no-link',
167 native: false, language: ['c', 'cpp', 'objc'])
168 # For the actual fuzzer binaries, we need to link against the libfuzzer
169 # library. They need to be configurable, to support OSS-Fuzz
170 fuzz_exe_ldflags = ['-fsanitize=fuzzer']
171 else
172 # LIB_FUZZING_ENGINE was set; assume we are running on OSS-Fuzz, and
173 # the needed CFLAGS have already been provided
174 fuzz_exe_ldflags = get_option('fuzzing_engine').split()
175 endif
176 endif
177
178 add_global_arguments(config_host['QEMU_CFLAGS'].split(),
179 native: false, language: ['c', 'objc'])
180 add_global_arguments(config_host['QEMU_CXXFLAGS'].split(),
181 native: false, language: 'cpp')
182 add_global_link_arguments(config_host['QEMU_LDFLAGS'].split(),
183 native: false, language: ['c', 'cpp', 'objc'])
184
185 if targetos == 'linux'
186 add_project_arguments('-isystem', meson.current_source_dir() / 'linux-headers',
187 '-isystem', 'linux-headers',
188 language: ['c', 'cpp'])
189 endif
190
191 add_project_arguments('-iquote', '.',
192 '-iquote', meson.current_source_dir(),
193 '-iquote', meson.current_source_dir() / 'include',
194 '-iquote', meson.current_source_dir() / 'disas/libvixl',
195 language: ['c', 'cpp', 'objc'])
196
197 link_language = meson.get_external_property('link_language', 'cpp')
198 if link_language == 'cpp'
199 add_languages('cpp', required: true, native: false)
200 endif
201 if host_machine.system() == 'darwin'
202 add_languages('objc', required: false, native: false)
203 endif
204
205 sparse = find_program('cgcc', required: get_option('sparse'))
206 if sparse.found()
207 run_target('sparse',
208 command: [find_program('scripts/check_sparse.py'),
209 'compile_commands.json', sparse.full_path(), '-Wbitwise',
210 '-Wno-transparent-union', '-Wno-old-initializer',
211 '-Wno-non-pointer-null'])
212 endif
213
214 ###########################################
215 # Target-specific checks and dependencies #
216 ###########################################
217
218 if get_option('fuzzing') and get_option('fuzzing_engine') == '' and \
219 not cc.links('''
220 #include <stdint.h>
221 #include <sys/types.h>
222 int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size);
223 int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { return 0; }
224 ''',
225 args: ['-Werror', '-fsanitize=fuzzer'])
226 error('Your compiler does not support -fsanitize=fuzzer')
227 endif
228
229 if 'ftrace' in get_option('trace_backends') and targetos != 'linux'
230 error('ftrace is supported only on Linux')
231 endif
232 if 'syslog' in get_option('trace_backends') and not cc.compiles('''
233 #include <syslog.h>
234 int main(void) {
235 openlog("qemu", LOG_PID, LOG_DAEMON);
236 syslog(LOG_INFO, "configure");
237 return 0;
238 }''')
239 error('syslog is not supported on this system')
240 endif
241
242 if targetos != 'linux' and get_option('mpath').enabled()
243 error('Multipath is supported only on Linux')
244 endif
245
246 if targetos != 'linux' and get_option('multiprocess').enabled()
247 error('Multiprocess QEMU is supported only on Linux')
248 endif
249 multiprocess_allowed = targetos == 'linux' and not get_option('multiprocess').disabled()
250
251 libm = cc.find_library('m', required: false)
252 threads = dependency('threads')
253 util = cc.find_library('util', required: false)
254 winmm = []
255 socket = []
256 version_res = []
257 coref = []
258 iokit = []
259 emulator_link_args = []
260 nvmm =not_found
261 hvf = not_found
262 host_dsosuf = '.so'
263 if targetos == 'windows'
264 socket = cc.find_library('ws2_32')
265 winmm = cc.find_library('winmm')
266
267 win = import('windows')
268 version_res = win.compile_resources('version.rc',
269 depend_files: files('pc-bios/qemu-nsis.ico'),
270 include_directories: include_directories('.'))
271 host_dsosuf = '.dll'
272 elif targetos == 'darwin'
273 coref = dependency('appleframeworks', modules: 'CoreFoundation')
274 iokit = dependency('appleframeworks', modules: 'IOKit', required: false)
275 host_dsosuf = '.dylib'
276 elif targetos == 'sunos'
277 socket = [cc.find_library('socket'),
278 cc.find_library('nsl'),
279 cc.find_library('resolv')]
280 elif targetos == 'haiku'
281 socket = [cc.find_library('posix_error_mapper'),
282 cc.find_library('network'),
283 cc.find_library('bsd')]
284 elif targetos == 'openbsd'
285 if not get_option('tcg').disabled() and target_dirs.length() > 0
286 # Disable OpenBSD W^X if available
287 emulator_link_args = cc.get_supported_link_arguments('-Wl,-z,wxneeded')
288 endif
289 endif
290
291 accelerators = []
292 if not get_option('kvm').disabled() and targetos == 'linux'
293 accelerators += 'CONFIG_KVM'
294 endif
295 if not get_option('xen').disabled() and 'CONFIG_XEN_BACKEND' in config_host
296 accelerators += 'CONFIG_XEN'
297 have_xen_pci_passthrough = not get_option('xen_pci_passthrough').disabled() and targetos == 'linux'
298 else
299 have_xen_pci_passthrough = false
300 endif
301 if not get_option('whpx').disabled() and targetos == 'windows'
302 if get_option('whpx').enabled() and host_machine.cpu() != 'x86_64'
303 error('WHPX requires 64-bit host')
304 elif cc.has_header('WinHvPlatform.h', required: get_option('whpx')) and \
305 cc.has_header('WinHvEmulation.h', required: get_option('whpx'))
306 accelerators += 'CONFIG_WHPX'
307 endif
308 endif
309 if not get_option('hvf').disabled()
310 hvf = dependency('appleframeworks', modules: 'Hypervisor',
311 required: get_option('hvf'))
312 if hvf.found()
313 accelerators += 'CONFIG_HVF'
314 endif
315 endif
316 if not get_option('hax').disabled()
317 if get_option('hax').enabled() or targetos in ['windows', 'darwin', 'netbsd']
318 accelerators += 'CONFIG_HAX'
319 endif
320 endif
321 if targetos == 'netbsd'
322 if cc.has_header_symbol('nvmm.h', 'nvmm_cpu_stop', required: get_option('nvmm'))
323 nvmm = cc.find_library('nvmm', required: get_option('nvmm'))
324 endif
325 if nvmm.found()
326 accelerators += 'CONFIG_NVMM'
327 endif
328 endif
329
330 tcg_arch = config_host['ARCH']
331 if not get_option('tcg').disabled()
332 if cpu not in supported_cpus
333 if get_option('tcg_interpreter')
334 warning('Unsupported CPU @0@, will use TCG with TCI (experimental and slow)'.format(cpu))
335 else
336 error('Unsupported CPU @0@, try --enable-tcg-interpreter'.format(cpu))
337 endif
338 elif get_option('tcg_interpreter')
339 warning('Use of the TCG interpretor is not recommended on this host')
340 warning('architecture. There is a native TCG execution backend available')
341 warning('which provides substantially better performance and reliability.')
342 warning('It is strongly recommended to remove the --enable-tcg-interpreter')
343 warning('configuration option on this architecture to use the native')
344 warning('backend.')
345 endif
346 if get_option('tcg_interpreter')
347 tcg_arch = 'tci'
348 elif config_host['ARCH'] == 'sparc64'
349 tcg_arch = 'sparc'
350 elif config_host['ARCH'] in ['x86_64', 'x32']
351 tcg_arch = 'i386'
352 elif config_host['ARCH'] == 'ppc64'
353 tcg_arch = 'ppc'
354 elif config_host['ARCH'] in ['riscv32', 'riscv64']
355 tcg_arch = 'riscv'
356 endif
357 add_project_arguments('-iquote', meson.current_source_dir() / 'tcg' / tcg_arch,
358 language: ['c', 'cpp', 'objc'])
359
360 accelerators += 'CONFIG_TCG'
361 config_host += { 'CONFIG_TCG': 'y' }
362 endif
363
364 if 'CONFIG_KVM' not in accelerators and get_option('kvm').enabled()
365 error('KVM not available on this platform')
366 endif
367 if 'CONFIG_HVF' not in accelerators and get_option('hvf').enabled()
368 error('HVF not available on this platform')
369 endif
370 if 'CONFIG_NVMM' not in accelerators and get_option('nvmm').enabled()
371 error('NVMM not available on this platform')
372 endif
373 if 'CONFIG_WHPX' not in accelerators and get_option('whpx').enabled()
374 error('WHPX not available on this platform')
375 endif
376 if not have_xen_pci_passthrough and get_option('xen_pci_passthrough').enabled()
377 if 'CONFIG_XEN' in accelerators
378 error('Xen PCI passthrough not available on this platform')
379 else
380 error('Xen PCI passthrough requested but Xen not enabled')
381 endif
382 endif
383
384 ################
385 # Dependencies #
386 ################
387
388 # The path to glib.h is added to all compilation commands. This was
389 # grandfathered in from the QEMU Makefiles.
390 add_project_arguments(config_host['GLIB_CFLAGS'].split(),
391 native: false, language: ['c', 'cpp', 'objc'])
392 glib = declare_dependency(compile_args: config_host['GLIB_CFLAGS'].split(),
393 link_args: config_host['GLIB_LIBS'].split())
394 # override glib dep with the configure results (for subprojects)
395 meson.override_dependency('glib-2.0', glib)
396
397 gio = not_found
398 if 'CONFIG_GIO' in config_host
399 gio = declare_dependency(compile_args: config_host['GIO_CFLAGS'].split(),
400 link_args: config_host['GIO_LIBS'].split())
401 endif
402 lttng = not_found
403 if 'ust' in get_option('trace_backends')
404 lttng = dependency('lttng-ust', required: true, method: 'pkg-config',
405 kwargs: static_kwargs)
406 endif
407 pixman = not_found
408 if have_system or have_tools
409 pixman = dependency('pixman-1', required: have_system, version:'>=0.21.8',
410 method: 'pkg-config', kwargs: static_kwargs)
411 endif
412 zlib = dependency('zlib', required: true, kwargs: static_kwargs)
413
414 libaio = not_found
415 if not get_option('linux_aio').auto() or have_block
416 libaio = cc.find_library('aio', has_headers: ['libaio.h'],
417 required: get_option('linux_aio'),
418 kwargs: static_kwargs)
419 endif
420 linux_io_uring = not_found
421 if not get_option('linux_io_uring').auto() or have_block
422 linux_io_uring = dependency('liburing', required: get_option('linux_io_uring'),
423 method: 'pkg-config', kwargs: static_kwargs)
424 endif
425 libxml2 = not_found
426 if not get_option('libxml2').auto() or have_block
427 libxml2 = dependency('libxml-2.0', required: get_option('libxml2'),
428 method: 'pkg-config', kwargs: static_kwargs)
429 endif
430 libnfs = not_found
431 if not get_option('libnfs').auto() or have_block
432 libnfs = dependency('libnfs', version: '>=1.9.3',
433 required: get_option('libnfs'),
434 method: 'pkg-config', kwargs: static_kwargs)
435 endif
436
437 libattr_test = '''
438 #include <stddef.h>
439 #include <sys/types.h>
440 #ifdef CONFIG_LIBATTR
441 #include <attr/xattr.h>
442 #else
443 #include <sys/xattr.h>
444 #endif
445 int main(void) { getxattr(NULL, NULL, NULL, 0); setxattr(NULL, NULL, NULL, 0, 0); return 0; }'''
446
447 libattr = not_found
448 have_old_libattr = false
449 if not get_option('attr').disabled()
450 if cc.links(libattr_test)
451 libattr = declare_dependency()
452 else
453 libattr = cc.find_library('attr', has_headers: ['attr/xattr.h'],
454 required: get_option('attr'),
455 kwargs: static_kwargs)
456 if libattr.found() and not \
457 cc.links(libattr_test, dependencies: libattr, args: '-DCONFIG_LIBATTR')
458 libattr = not_found
459 if get_option('attr').enabled()
460 error('could not link libattr')
461 else
462 warning('could not link libattr, disabling')
463 endif
464 else
465 have_old_libattr = libattr.found()
466 endif
467 endif
468 endif
469
470 cocoa = dependency('appleframeworks', modules: 'Cocoa', required: get_option('cocoa'))
471 if cocoa.found() and get_option('sdl').enabled()
472 error('Cocoa and SDL cannot be enabled at the same time')
473 endif
474 if cocoa.found() and get_option('gtk').enabled()
475 error('Cocoa and GTK+ cannot be enabled at the same time')
476 endif
477
478 seccomp = not_found
479 if not get_option('seccomp').auto() or have_system or have_tools
480 seccomp = dependency('libseccomp', version: '>=2.3.0',
481 required: get_option('seccomp'),
482 method: 'pkg-config', kwargs: static_kwargs)
483 endif
484
485 libcap_ng = not_found
486 if not get_option('cap_ng').auto() or have_system or have_tools
487 libcap_ng = cc.find_library('cap-ng', has_headers: ['cap-ng.h'],
488 required: get_option('cap_ng'),
489 kwargs: static_kwargs)
490 endif
491 if libcap_ng.found() and not cc.links('''
492 #include <cap-ng.h>
493 int main(void)
494 {
495 capng_capability_to_name(CAPNG_EFFECTIVE);
496 return 0;
497 }''', dependencies: libcap_ng)
498 libcap_ng = not_found
499 if get_option('cap_ng').enabled()
500 error('could not link libcap-ng')
501 else
502 warning('could not link libcap-ng, disabling')
503 endif
504 endif
505
506 if get_option('xkbcommon').auto() and not have_system and not have_tools
507 xkbcommon = not_found
508 else
509 xkbcommon = dependency('xkbcommon', required: get_option('xkbcommon'),
510 method: 'pkg-config', kwargs: static_kwargs)
511 endif
512
513 vde = not_found
514 if not get_option('vde').auto() or have_system or have_tools
515 vde = cc.find_library('vdeplug', has_headers: ['libvdeplug.h'],
516 required: get_option('vde'),
517 kwargs: static_kwargs)
518 endif
519 if vde.found() and not cc.links('''
520 #include <libvdeplug.h>
521 int main(void)
522 {
523 struct vde_open_args a = {0, 0, 0};
524 char s[] = "";
525 vde_open(s, s, &a);
526 return 0;
527 }''', dependencies: vde)
528 vde = not_found
529 if get_option('cap_ng').enabled()
530 error('could not link libvdeplug')
531 else
532 warning('could not link libvdeplug, disabling')
533 endif
534 endif
535
536 pulse = not_found
537 if not get_option('pa').auto() or (targetos == 'linux' and have_system)
538 pulse = dependency('libpulse', required: get_option('pa'),
539 method: 'pkg-config', kwargs: static_kwargs)
540 endif
541 alsa = not_found
542 if not get_option('alsa').auto() or (targetos == 'linux' and have_system)
543 alsa = dependency('alsa', required: get_option('alsa'),
544 method: 'pkg-config', kwargs: static_kwargs)
545 endif
546 jack = not_found
547 if not get_option('jack').auto() or have_system
548 jack = dependency('jack', required: get_option('jack'),
549 method: 'pkg-config', kwargs: static_kwargs)
550 endif
551
552 spice_protocol = not_found
553 if not get_option('spice_protocol').auto() or have_system
554 spice_protocol = dependency('spice-protocol', version: '>=0.12.3',
555 required: get_option('spice_protocol'),
556 method: 'pkg-config', kwargs: static_kwargs)
557 endif
558 spice = not_found
559 if not get_option('spice').auto() or have_system
560 spice = dependency('spice-server', version: '>=0.12.5',
561 required: get_option('spice'),
562 method: 'pkg-config', kwargs: static_kwargs)
563 endif
564 spice_headers = spice.partial_dependency(compile_args: true, includes: true)
565
566 rt = cc.find_library('rt', required: false)
567 libdl = not_found
568 if 'CONFIG_PLUGIN' in config_host
569 libdl = cc.find_library('dl', required: false)
570 if not cc.has_function('dlopen', dependencies: libdl)
571 error('dlopen not found')
572 endif
573 endif
574 libiscsi = not_found
575 if not get_option('libiscsi').auto() or have_block
576 libiscsi = dependency('libiscsi', version: '>=1.9.0',
577 required: get_option('libiscsi'),
578 method: 'pkg-config', kwargs: static_kwargs)
579 endif
580 zstd = not_found
581 if not get_option('zstd').auto() or have_block
582 zstd = dependency('libzstd', version: '>=1.4.0',
583 required: get_option('zstd'),
584 method: 'pkg-config', kwargs: static_kwargs)
585 endif
586 virgl = not_found
587 if not get_option('virglrenderer').auto() or have_system
588 virgl = dependency('virglrenderer',
589 method: 'pkg-config',
590 required: get_option('virglrenderer'),
591 kwargs: static_kwargs)
592 endif
593 curl = not_found
594 if not get_option('curl').auto() or have_block
595 curl = dependency('libcurl', version: '>=7.29.0',
596 method: 'pkg-config',
597 required: get_option('curl'),
598 kwargs: static_kwargs)
599 endif
600 libudev = not_found
601 if targetos == 'linux' and (have_system or have_tools)
602 libudev = dependency('libudev',
603 method: 'pkg-config',
604 required: get_option('libudev'),
605 kwargs: static_kwargs)
606 endif
607
608 mpathlibs = [libudev]
609 mpathpersist = not_found
610 mpathpersist_new_api = false
611 if targetos == 'linux' and have_tools and not get_option('mpath').disabled()
612 mpath_test_source_new = '''
613 #include <libudev.h>
614 #include <mpath_persist.h>
615 unsigned mpath_mx_alloc_len = 1024;
616 int logsink;
617 static struct config *multipath_conf;
618 extern struct udev *udev;
619 extern struct config *get_multipath_config(void);
620 extern void put_multipath_config(struct config *conf);
621 struct udev *udev;
622 struct config *get_multipath_config(void) { return multipath_conf; }
623 void put_multipath_config(struct config *conf) { }
624 int main(void) {
625 udev = udev_new();
626 multipath_conf = mpath_lib_init();
627 return 0;
628 }'''
629 mpath_test_source_old = '''
630 #include <libudev.h>
631 #include <mpath_persist.h>
632 unsigned mpath_mx_alloc_len = 1024;
633 int logsink;
634 int main(void) {
635 struct udev *udev = udev_new();
636 mpath_lib_init(udev);
637 return 0;
638 }'''
639 libmpathpersist = cc.find_library('mpathpersist',
640 required: get_option('mpath'),
641 kwargs: static_kwargs)
642 if libmpathpersist.found()
643 mpathlibs += libmpathpersist
644 if enable_static
645 mpathlibs += cc.find_library('devmapper',
646 required: get_option('mpath'),
647 kwargs: static_kwargs)
648 endif
649 mpathlibs += cc.find_library('multipath',
650 required: get_option('mpath'),
651 kwargs: static_kwargs)
652 foreach lib: mpathlibs
653 if not lib.found()
654 mpathlibs = []
655 break
656 endif
657 endforeach
658 if mpathlibs.length() == 0
659 msg = 'Dependencies missing for libmpathpersist'
660 elif cc.links(mpath_test_source_new, dependencies: mpathlibs)
661 mpathpersist = declare_dependency(dependencies: mpathlibs)
662 mpathpersist_new_api = true
663 elif cc.links(mpath_test_source_old, dependencies: mpathlibs)
664 mpathpersist = declare_dependency(dependencies: mpathlibs)
665 else
666 msg = 'Cannot detect libmpathpersist API'
667 endif
668 if not mpathpersist.found()
669 if get_option('mpath').enabled()
670 error(msg)
671 else
672 warning(msg + ', disabling')
673 endif
674 endif
675 endif
676 endif
677
678 iconv = not_found
679 curses = not_found
680 if have_system and not get_option('curses').disabled()
681 curses_test = '''
682 #include <locale.h>
683 #include <curses.h>
684 #include <wchar.h>
685 int main(void) {
686 wchar_t wch = L'w';
687 setlocale(LC_ALL, "");
688 resize_term(0, 0);
689 addwstr(L"wide chars\n");
690 addnwstr(&wch, 1);
691 add_wch(WACS_DEGREE);
692 return 0;
693 }'''
694
695 curses_dep_list = targetos == 'windows' ? ['ncurses', 'ncursesw'] : ['ncursesw']
696 foreach curses_dep : curses_dep_list
697 if not curses.found()
698 curses = dependency(curses_dep,
699 required: false,
700 method: 'pkg-config',
701 kwargs: static_kwargs)
702 endif
703 endforeach
704 msg = get_option('curses').enabled() ? 'curses library not found' : ''
705 curses_compile_args = ['-DNCURSES_WIDECHAR']
706 if curses.found()
707 if cc.links(curses_test, args: curses_compile_args, dependencies: [curses])
708 curses = declare_dependency(compile_args: curses_compile_args, dependencies: [curses])
709 else
710 msg = 'curses package not usable'
711 curses = not_found
712 endif
713 endif
714 if not curses.found()
715 has_curses_h = cc.has_header('curses.h', args: curses_compile_args)
716 if targetos != 'windows' and not has_curses_h
717 message('Trying with /usr/include/ncursesw')
718 curses_compile_args += ['-I/usr/include/ncursesw']
719 has_curses_h = cc.has_header('curses.h', args: curses_compile_args)
720 endif
721 if has_curses_h
722 curses_libname_list = (targetos == 'windows' ? ['pdcurses'] : ['ncursesw', 'cursesw'])
723 foreach curses_libname : curses_libname_list
724 libcurses = cc.find_library(curses_libname,
725 required: false,
726 kwargs: static_kwargs)
727 if libcurses.found()
728 if cc.links(curses_test, args: curses_compile_args, dependencies: libcurses)
729 curses = declare_dependency(compile_args: curses_compile_args,
730 dependencies: [libcurses])
731 break
732 else
733 msg = 'curses library not usable'
734 endif
735 endif
736 endforeach
737 endif
738 endif
739 if not get_option('iconv').disabled()
740 foreach link_args : [ ['-liconv'], [] ]
741 # Programs will be linked with glib and this will bring in libiconv on FreeBSD.
742 # We need to use libiconv if available because mixing libiconv's headers with
743 # the system libc does not work.
744 # However, without adding glib to the dependencies -L/usr/local/lib will not be
745 # included in the command line and libiconv will not be found.
746 if cc.links('''
747 #include <iconv.h>
748 int main(void) {
749 iconv_t conv = iconv_open("WCHAR_T", "UCS-2");
750 return conv != (iconv_t) -1;
751 }''', args: config_host['GLIB_CFLAGS'].split() + config_host['GLIB_LIBS'].split() + link_args)
752 iconv = declare_dependency(link_args: link_args, dependencies: glib)
753 break
754 endif
755 endforeach
756 endif
757 if curses.found() and not iconv.found()
758 if get_option('iconv').enabled()
759 error('iconv not available')
760 endif
761 msg = 'iconv required for curses UI but not available'
762 curses = not_found
763 endif
764 if not curses.found() and msg != ''
765 if get_option('curses').enabled()
766 error(msg)
767 else
768 warning(msg + ', disabling')
769 endif
770 endif
771 endif
772
773 brlapi = not_found
774 if not get_option('brlapi').auto() or have_system
775 brlapi = cc.find_library('brlapi', has_headers: ['brlapi.h'],
776 required: get_option('brlapi'),
777 kwargs: static_kwargs)
778 if brlapi.found() and not cc.links('''
779 #include <brlapi.h>
780 #include <stddef.h>
781 int main(void) { return brlapi__openConnection (NULL, NULL, NULL); }''', dependencies: brlapi)
782 brlapi = not_found
783 if get_option('brlapi').enabled()
784 error('could not link brlapi')
785 else
786 warning('could not link brlapi, disabling')
787 endif
788 endif
789 endif
790
791 sdl = not_found
792 if not get_option('sdl').auto() or (have_system and not cocoa.found())
793 sdl = dependency('sdl2', required: get_option('sdl'), kwargs: static_kwargs)
794 sdl_image = not_found
795 endif
796 if sdl.found()
797 # work around 2.0.8 bug
798 sdl = declare_dependency(compile_args: '-Wno-undef',
799 dependencies: sdl)
800 sdl_image = dependency('SDL2_image', required: get_option('sdl_image'),
801 method: 'pkg-config', kwargs: static_kwargs)
802 else
803 if get_option('sdl_image').enabled()
804 error('sdl-image required, but SDL was @0@'.format(
805 get_option('sdl').disabled() ? 'disabled' : 'not found'))
806 endif
807 sdl_image = not_found
808 endif
809
810 rbd = not_found
811 if not get_option('rbd').auto() or have_block
812 librados = cc.find_library('rados', required: get_option('rbd'),
813 kwargs: static_kwargs)
814 librbd = cc.find_library('rbd', has_headers: ['rbd/librbd.h'],
815 required: get_option('rbd'),
816 kwargs: static_kwargs)
817 if librados.found() and librbd.found()
818 if cc.links('''
819 #include <stdio.h>
820 #include <rbd/librbd.h>
821 int main(void) {
822 rados_t cluster;
823 rados_create(&cluster, NULL);
824 #if LIBRBD_VERSION_CODE < LIBRBD_VERSION(1, 12, 0)
825 #error
826 #endif
827 return 0;
828 }''', dependencies: [librbd, librados])
829 rbd = declare_dependency(dependencies: [librbd, librados])
830 elif get_option('rbd').enabled()
831 error('librbd >= 1.12.0 required')
832 else
833 warning('librbd >= 1.12.0 not found, disabling')
834 endif
835 endif
836 endif
837
838 glusterfs = not_found
839 glusterfs_ftruncate_has_stat = false
840 glusterfs_iocb_has_stat = false
841 if not get_option('glusterfs').auto() or have_block
842 glusterfs = dependency('glusterfs-api', version: '>=3',
843 required: get_option('glusterfs'),
844 method: 'pkg-config', kwargs: static_kwargs)
845 if glusterfs.found()
846 glusterfs_ftruncate_has_stat = cc.links('''
847 #include <glusterfs/api/glfs.h>
848
849 int
850 main(void)
851 {
852 /* new glfs_ftruncate() passes two additional args */
853 return glfs_ftruncate(NULL, 0, NULL, NULL);
854 }
855 ''', dependencies: glusterfs)
856 glusterfs_iocb_has_stat = cc.links('''
857 #include <glusterfs/api/glfs.h>
858
859 /* new glfs_io_cbk() passes two additional glfs_stat structs */
860 static void
861 glusterfs_iocb(glfs_fd_t *fd, ssize_t ret, struct glfs_stat *prestat, struct glfs_stat *poststat, void *data)
862 {}
863
864 int
865 main(void)
866 {
867 glfs_io_cbk iocb = &glusterfs_iocb;
868 iocb(NULL, 0 , NULL, NULL, NULL);
869 return 0;
870 }
871 ''', dependencies: glusterfs)
872 endif
873 endif
874 libssh = not_found
875 if 'CONFIG_LIBSSH' in config_host
876 libssh = declare_dependency(compile_args: config_host['LIBSSH_CFLAGS'].split(),
877 link_args: config_host['LIBSSH_LIBS'].split())
878 endif
879 libbzip2 = not_found
880 if not get_option('bzip2').auto() or have_block
881 libbzip2 = cc.find_library('bz2', has_headers: ['bzlib.h'],
882 required: get_option('bzip2'),
883 kwargs: static_kwargs)
884 if libbzip2.found() and not cc.links('''
885 #include <bzlib.h>
886 int main(void) { BZ2_bzlibVersion(); return 0; }''', dependencies: libbzip2)
887 libbzip2 = not_found
888 if get_option('bzip2').enabled()
889 error('could not link libbzip2')
890 else
891 warning('could not link libbzip2, disabling')
892 endif
893 endif
894 endif
895
896 liblzfse = not_found
897 if not get_option('lzfse').auto() or have_block
898 liblzfse = cc.find_library('lzfse', has_headers: ['lzfse.h'],
899 required: get_option('lzfse'),
900 kwargs: static_kwargs)
901 endif
902 if liblzfse.found() and not cc.links('''
903 #include <lzfse.h>
904 int main(void) { lzfse_decode_scratch_size(); return 0; }''', dependencies: liblzfse)
905 liblzfse = not_found
906 if get_option('lzfse').enabled()
907 error('could not link liblzfse')
908 else
909 warning('could not link liblzfse, disabling')
910 endif
911 endif
912
913 oss = not_found
914 if not get_option('oss').auto() or have_system
915 if not cc.has_header('sys/soundcard.h')
916 # not found
917 elif targetos == 'netbsd'
918 oss = cc.find_library('ossaudio', required: get_option('oss'),
919 kwargs: static_kwargs)
920 else
921 oss = declare_dependency()
922 endif
923
924 if not oss.found()
925 if get_option('oss').enabled()
926 error('OSS not found')
927 else
928 warning('OSS not found, disabling')
929 endif
930 endif
931 endif
932 dsound = not_found
933 if not get_option('dsound').auto() or (targetos == 'windows' and have_system)
934 if cc.has_header('dsound.h')
935 dsound = declare_dependency(link_args: ['-lole32', '-ldxguid'])
936 endif
937
938 if not dsound.found()
939 if get_option('dsound').enabled()
940 error('DirectSound not found')
941 else
942 warning('DirectSound not found, disabling')
943 endif
944 endif
945 endif
946
947 coreaudio = not_found
948 if not get_option('coreaudio').auto() or (targetos == 'darwin' and have_system)
949 coreaudio = dependency('appleframeworks', modules: 'CoreAudio',
950 required: get_option('coreaudio'))
951 if coreaudio.found() and not cc.links('''
952 #include <CoreAudio/CoreAudio.h>
953 int main(void)
954 {
955 return (int)AudioGetCurrentHostTime();
956 }''')
957 coreaudio = not_found
958 endif
959
960 if not coreaudio.found()
961 if get_option('coreaudio').enabled()
962 error('CoreAudio not found')
963 else
964 warning('CoreAudio not found, disabling')
965 endif
966 endif
967 endif
968
969 opengl = not_found
970 if 'CONFIG_OPENGL' in config_host
971 opengl = declare_dependency(compile_args: config_host['OPENGL_CFLAGS'].split(),
972 link_args: config_host['OPENGL_LIBS'].split())
973 endif
974 gbm = not_found
975 if (have_system or have_tools) and (virgl.found() or opengl.found())
976 gbm = dependency('gbm', method: 'pkg-config', required: false,
977 kwargs: static_kwargs)
978 endif
979
980 gnutls = not_found
981 gnutls_crypto = not_found
982 if get_option('gnutls').enabled() or (get_option('gnutls').auto() and have_system)
983 # For general TLS support our min gnutls matches
984 # that implied by our platform support matrix
985 #
986 # For the crypto backends, we look for a newer
987 # gnutls:
988 #
989 # Version 3.6.8 is needed to get XTS
990 # Version 3.6.13 is needed to get PBKDF
991 # Version 3.6.14 is needed to get HW accelerated XTS
992 #
993 # If newer enough gnutls isn't available, we can
994 # still use a different crypto backend to satisfy
995 # the platform support requirements
996 gnutls_crypto = dependency('gnutls', version: '>=3.6.14',
997 method: 'pkg-config',
998 required: false,
999 kwargs: static_kwargs)
1000 if gnutls_crypto.found()
1001 gnutls = gnutls_crypto
1002 else
1003 # Our min version if all we need is TLS
1004 gnutls = dependency('gnutls', version: '>=3.5.18',
1005 method: 'pkg-config',
1006 required: get_option('gnutls'),
1007 kwargs: static_kwargs)
1008 endif
1009 endif
1010
1011 # We prefer use of gnutls for crypto, unless the options
1012 # explicitly asked for nettle or gcrypt.
1013 #
1014 # If gnutls isn't available for crypto, then we'll prefer
1015 # gcrypt over nettle for performance reasons.
1016 gcrypt = not_found
1017 nettle = not_found
1018 xts = 'none'
1019
1020 if get_option('nettle').enabled() and get_option('gcrypt').enabled()
1021 error('Only one of gcrypt & nettle can be enabled')
1022 endif
1023
1024 # Explicit nettle/gcrypt request, so ignore gnutls for crypto
1025 if get_option('nettle').enabled() or get_option('gcrypt').enabled()
1026 gnutls_crypto = not_found
1027 endif
1028
1029 if not gnutls_crypto.found()
1030 if (not get_option('gcrypt').auto() or have_system) and not get_option('nettle').enabled()
1031 gcrypt = dependency('libgcrypt', version: '>=1.8',
1032 method: 'config-tool',
1033 required: get_option('gcrypt'),
1034 kwargs: static_kwargs)
1035 # Debian has removed -lgpg-error from libgcrypt-config
1036 # as it "spreads unnecessary dependencies" which in
1037 # turn breaks static builds...
1038 if gcrypt.found() and enable_static
1039 gcrypt = declare_dependency(dependencies: [
1040 gcrypt,
1041 cc.find_library('gpg-error', required: true, kwargs: static_kwargs)])
1042 endif
1043 endif
1044 if (not get_option('nettle').auto() or have_system) and not gcrypt.found()
1045 nettle = dependency('nettle', version: '>=3.4',
1046 method: 'pkg-config',
1047 required: get_option('nettle'),
1048 kwargs: static_kwargs)
1049 if nettle.found() and not cc.has_header('nettle/xts.h', dependencies: nettle)
1050 xts = 'private'
1051 endif
1052 endif
1053 endif
1054
1055 gtk = not_found
1056 gtkx11 = not_found
1057 vte = not_found
1058 if not get_option('gtk').auto() or (have_system and not cocoa.found())
1059 gtk = dependency('gtk+-3.0', version: '>=3.22.0',
1060 method: 'pkg-config',
1061 required: get_option('gtk'),
1062 kwargs: static_kwargs)
1063 if gtk.found()
1064 gtkx11 = dependency('gtk+-x11-3.0', version: '>=3.22.0',
1065 method: 'pkg-config',
1066 required: false,
1067 kwargs: static_kwargs)
1068 gtk = declare_dependency(dependencies: [gtk, gtkx11])
1069
1070 if not get_option('vte').auto() or have_system
1071 vte = dependency('vte-2.91',
1072 method: 'pkg-config',
1073 required: get_option('vte'),
1074 kwargs: static_kwargs)
1075 endif
1076 endif
1077 endif
1078
1079 x11 = not_found
1080 if gtkx11.found()
1081 x11 = dependency('x11', method: 'pkg-config', required: gtkx11.found(),
1082 kwargs: static_kwargs)
1083 endif
1084 vnc = not_found
1085 png = not_found
1086 jpeg = not_found
1087 sasl = not_found
1088 if have_system and not get_option('vnc').disabled()
1089 vnc = declare_dependency() # dummy dependency
1090 png = dependency('libpng', required: get_option('vnc_png'),
1091 method: 'pkg-config', kwargs: static_kwargs)
1092 jpeg = dependency('libjpeg', required: get_option('vnc_jpeg'),
1093 method: 'pkg-config', kwargs: static_kwargs)
1094 sasl = cc.find_library('sasl2', has_headers: ['sasl/sasl.h'],
1095 required: get_option('vnc_sasl'),
1096 kwargs: static_kwargs)
1097 if sasl.found()
1098 sasl = declare_dependency(dependencies: sasl,
1099 compile_args: '-DSTRUCT_IOVEC_DEFINED')
1100 endif
1101 endif
1102
1103 pam = not_found
1104 if not get_option('auth_pam').auto() or have_system
1105 pam = cc.find_library('pam', has_headers: ['security/pam_appl.h'],
1106 required: get_option('auth_pam'),
1107 kwargs: static_kwargs)
1108 endif
1109 if pam.found() and not cc.links('''
1110 #include <stddef.h>
1111 #include <security/pam_appl.h>
1112 int main(void) {
1113 const char *service_name = "qemu";
1114 const char *user = "frank";
1115 const struct pam_conv pam_conv = { 0 };
1116 pam_handle_t *pamh = NULL;
1117 pam_start(service_name, user, &pam_conv, &pamh);
1118 return 0;
1119 }''', dependencies: pam)
1120 pam = not_found
1121 if get_option('auth_pam').enabled()
1122 error('could not link libpam')
1123 else
1124 warning('could not link libpam, disabling')
1125 endif
1126 endif
1127
1128 snappy = not_found
1129 if not get_option('snappy').auto() or have_system
1130 snappy = cc.find_library('snappy', has_headers: ['snappy-c.h'],
1131 required: get_option('snappy'),
1132 kwargs: static_kwargs)
1133 endif
1134 if snappy.found() and not cc.links('''
1135 #include <snappy-c.h>
1136 int main(void) { snappy_max_compressed_length(4096); return 0; }''', dependencies: snappy)
1137 snappy = not_found
1138 if get_option('snappy').enabled()
1139 error('could not link libsnappy')
1140 else
1141 warning('could not link libsnappy, disabling')
1142 endif
1143 endif
1144
1145 lzo = not_found
1146 if not get_option('lzo').auto() or have_system
1147 lzo = cc.find_library('lzo2', has_headers: ['lzo/lzo1x.h'],
1148 required: get_option('lzo'),
1149 kwargs: static_kwargs)
1150 endif
1151 if lzo.found() and not cc.links('''
1152 #include <lzo/lzo1x.h>
1153 int main(void) { lzo_version(); return 0; }''', dependencies: lzo)
1154 lzo = not_found
1155 if get_option('lzo').enabled()
1156 error('could not link liblzo2')
1157 else
1158 warning('could not link liblzo2, disabling')
1159 endif
1160 endif
1161
1162 rdma = not_found
1163 if 'CONFIG_RDMA' in config_host
1164 rdma = declare_dependency(link_args: config_host['RDMA_LIBS'].split())
1165 endif
1166 numa = not_found
1167 if 'CONFIG_NUMA' in config_host
1168 numa = declare_dependency(link_args: config_host['NUMA_LIBS'].split())
1169 endif
1170 xen = not_found
1171 if 'CONFIG_XEN_BACKEND' in config_host
1172 xen = declare_dependency(compile_args: config_host['XEN_CFLAGS'].split(),
1173 link_args: config_host['XEN_LIBS'].split())
1174 endif
1175 cacard = not_found
1176 if not get_option('smartcard').auto() or have_system
1177 cacard = dependency('libcacard', required: get_option('smartcard'),
1178 version: '>=2.5.1', method: 'pkg-config',
1179 kwargs: static_kwargs)
1180 endif
1181 u2f = not_found
1182 if have_system
1183 u2f = dependency('u2f-emu', required: get_option('u2f'),
1184 method: 'pkg-config',
1185 kwargs: static_kwargs)
1186 endif
1187 usbredir = not_found
1188 if not get_option('usb_redir').auto() or have_system
1189 usbredir = dependency('libusbredirparser-0.5', required: get_option('usb_redir'),
1190 version: '>=0.6', method: 'pkg-config',
1191 kwargs: static_kwargs)
1192 endif
1193 libusb = not_found
1194 if not get_option('libusb').auto() or have_system
1195 libusb = dependency('libusb-1.0', required: get_option('libusb'),
1196 version: '>=1.0.13', method: 'pkg-config',
1197 kwargs: static_kwargs)
1198 endif
1199
1200 libpmem = not_found
1201 if not get_option('libpmem').auto() or have_system
1202 libpmem = dependency('libpmem', required: get_option('libpmem'),
1203 method: 'pkg-config', kwargs: static_kwargs)
1204 endif
1205 libdaxctl = not_found
1206 if not get_option('libdaxctl').auto() or have_system
1207 libdaxctl = dependency('libdaxctl', required: get_option('libdaxctl'),
1208 version: '>=57', method: 'pkg-config',
1209 kwargs: static_kwargs)
1210 endif
1211 tasn1 = not_found
1212 if gnutls.found()
1213 tasn1 = dependency('libtasn1',
1214 method: 'pkg-config',
1215 kwargs: static_kwargs)
1216 endif
1217 keyutils = dependency('libkeyutils', required: false,
1218 method: 'pkg-config', kwargs: static_kwargs)
1219
1220 has_gettid = cc.has_function('gettid')
1221
1222 # Malloc tests
1223
1224 malloc = []
1225 if get_option('malloc') == 'system'
1226 has_malloc_trim = \
1227 not get_option('malloc_trim').disabled() and \
1228 cc.links('''#include <malloc.h>
1229 int main(void) { malloc_trim(0); return 0; }''')
1230 else
1231 has_malloc_trim = false
1232 malloc = cc.find_library(get_option('malloc'), required: true)
1233 endif
1234 if not has_malloc_trim and get_option('malloc_trim').enabled()
1235 if get_option('malloc') == 'system'
1236 error('malloc_trim not available on this platform.')
1237 else
1238 error('malloc_trim not available with non-libc memory allocator')
1239 endif
1240 endif
1241
1242 # Check whether the glibc provides statx()
1243
1244 gnu_source_prefix = '''
1245 #ifndef _GNU_SOURCE
1246 #define _GNU_SOURCE
1247 #endif
1248 '''
1249 statx_test = gnu_source_prefix + '''
1250 #include <sys/stat.h>
1251 int main(void) {
1252 struct statx statxbuf;
1253 statx(0, "", 0, STATX_BASIC_STATS, &statxbuf);
1254 return 0;
1255 }'''
1256
1257 has_statx = cc.links(statx_test)
1258
1259 have_vhost_user_blk_server = (targetos == 'linux' and
1260 'CONFIG_VHOST_USER' in config_host)
1261
1262 if get_option('vhost_user_blk_server').enabled()
1263 if targetos != 'linux'
1264 error('vhost_user_blk_server requires linux')
1265 elif 'CONFIG_VHOST_USER' not in config_host
1266 error('vhost_user_blk_server requires vhost-user support')
1267 endif
1268 elif get_option('vhost_user_blk_server').disabled() or not have_system
1269 have_vhost_user_blk_server = false
1270 endif
1271
1272
1273 if get_option('fuse').disabled() and get_option('fuse_lseek').enabled()
1274 error('Cannot enable fuse-lseek while fuse is disabled')
1275 endif
1276
1277 fuse = dependency('fuse3', required: get_option('fuse'),
1278 version: '>=3.1', method: 'pkg-config',
1279 kwargs: static_kwargs)
1280
1281 fuse_lseek = not_found
1282 if not get_option('fuse_lseek').disabled()
1283 if fuse.version().version_compare('>=3.8')
1284 # Dummy dependency
1285 fuse_lseek = declare_dependency()
1286 elif get_option('fuse_lseek').enabled()
1287 if fuse.found()
1288 error('fuse-lseek requires libfuse >=3.8, found ' + fuse.version())
1289 else
1290 error('fuse-lseek requires libfuse, which was not found')
1291 endif
1292 endif
1293 endif
1294
1295 # libbpf
1296 libbpf = dependency('libbpf', required: get_option('bpf'), method: 'pkg-config')
1297 if libbpf.found() and not cc.links('''
1298 #include <bpf/libbpf.h>
1299 int main(void)
1300 {
1301 bpf_object__destroy_skeleton(NULL);
1302 return 0;
1303 }''', dependencies: libbpf)
1304 libbpf = not_found
1305 if get_option('bpf').enabled()
1306 error('libbpf skeleton test failed')
1307 else
1308 warning('libbpf skeleton test failed, disabling')
1309 endif
1310 endif
1311
1312 #################
1313 # config-host.h #
1314 #################
1315
1316 audio_drivers_selected = []
1317 if have_system
1318 audio_drivers_available = {
1319 'alsa': alsa.found(),
1320 'coreaudio': coreaudio.found(),
1321 'dsound': dsound.found(),
1322 'jack': jack.found(),
1323 'oss': oss.found(),
1324 'pa': pulse.found(),
1325 'sdl': sdl.found(),
1326 }
1327 foreach k, v: audio_drivers_available
1328 config_host_data.set('CONFIG_AUDIO_' + k.to_upper(), v)
1329 endforeach
1330
1331 # Default to native drivers first, OSS second, SDL third
1332 audio_drivers_priority = \
1333 [ 'pa', 'coreaudio', 'dsound', 'oss' ] + \
1334 (targetos == 'linux' ? [] : [ 'sdl' ])
1335 audio_drivers_default = []
1336 foreach k: audio_drivers_priority
1337 if audio_drivers_available[k]
1338 audio_drivers_default += k
1339 endif
1340 endforeach
1341
1342 foreach k: get_option('audio_drv_list')
1343 if k == 'default'
1344 audio_drivers_selected += audio_drivers_default
1345 elif not audio_drivers_available[k]
1346 error('Audio driver "@0@" not available.'.format(k))
1347 else
1348 audio_drivers_selected += k
1349 endif
1350 endforeach
1351 endif
1352 config_host_data.set('CONFIG_AUDIO_DRIVERS',
1353 '"' + '", "'.join(audio_drivers_selected) + '", ')
1354
1355 if get_option('cfi')
1356 cfi_flags=[]
1357 # Check for dependency on LTO
1358 if not get_option('b_lto')
1359 error('Selected Control-Flow Integrity but LTO is disabled')
1360 endif
1361 if config_host.has_key('CONFIG_MODULES')
1362 error('Selected Control-Flow Integrity is not compatible with modules')
1363 endif
1364 # Check for cfi flags. CFI requires LTO so we can't use
1365 # get_supported_arguments, but need a more complex "compiles" which allows
1366 # custom arguments
1367 if cc.compiles('int main () { return 0; }', name: '-fsanitize=cfi-icall',
1368 args: ['-flto', '-fsanitize=cfi-icall'] )
1369 cfi_flags += '-fsanitize=cfi-icall'
1370 else
1371 error('-fsanitize=cfi-icall is not supported by the compiler')
1372 endif
1373 if cc.compiles('int main () { return 0; }',
1374 name: '-fsanitize-cfi-icall-generalize-pointers',
1375 args: ['-flto', '-fsanitize=cfi-icall',
1376 '-fsanitize-cfi-icall-generalize-pointers'] )
1377 cfi_flags += '-fsanitize-cfi-icall-generalize-pointers'
1378 else
1379 error('-fsanitize-cfi-icall-generalize-pointers is not supported by the compiler')
1380 endif
1381 if get_option('cfi_debug')
1382 if cc.compiles('int main () { return 0; }',
1383 name: '-fno-sanitize-trap=cfi-icall',
1384 args: ['-flto', '-fsanitize=cfi-icall',
1385 '-fno-sanitize-trap=cfi-icall'] )
1386 cfi_flags += '-fno-sanitize-trap=cfi-icall'
1387 else
1388 error('-fno-sanitize-trap=cfi-icall is not supported by the compiler')
1389 endif
1390 endif
1391 add_global_arguments(cfi_flags, native: false, language: ['c', 'cpp', 'objc'])
1392 add_global_link_arguments(cfi_flags, native: false, language: ['c', 'cpp', 'objc'])
1393 endif
1394
1395 have_host_block_device = (targetos != 'darwin' or
1396 cc.has_header('IOKit/storage/IOMedia.h'))
1397
1398 have_virtfs = (targetos == 'linux' and
1399 have_system and
1400 libattr.found() and
1401 libcap_ng.found())
1402
1403 have_virtfs_proxy_helper = have_virtfs and have_tools
1404
1405 if get_option('virtfs').enabled()
1406 if not have_virtfs
1407 if targetos != 'linux'
1408 error('virtio-9p (virtfs) requires Linux')
1409 elif not libcap_ng.found() or not libattr.found()
1410 error('virtio-9p (virtfs) requires libcap-ng-devel and libattr-devel')
1411 elif not have_system
1412 error('virtio-9p (virtfs) needs system emulation support')
1413 endif
1414 endif
1415 elif get_option('virtfs').disabled()
1416 have_virtfs = false
1417 endif
1418
1419 foreach k : get_option('trace_backends')
1420 config_host_data.set('CONFIG_TRACE_' + k.to_upper(), true)
1421 endforeach
1422 config_host_data.set_quoted('CONFIG_TRACE_FILE', get_option('trace_file'))
1423
1424 config_host_data.set_quoted('CONFIG_BINDIR', get_option('prefix') / get_option('bindir'))
1425 config_host_data.set_quoted('CONFIG_PREFIX', get_option('prefix'))
1426 config_host_data.set_quoted('CONFIG_QEMU_CONFDIR', get_option('prefix') / qemu_confdir)
1427 config_host_data.set_quoted('CONFIG_QEMU_DATADIR', get_option('prefix') / qemu_datadir)
1428 config_host_data.set_quoted('CONFIG_QEMU_DESKTOPDIR', get_option('prefix') / qemu_desktopdir)
1429 config_host_data.set_quoted('CONFIG_QEMU_FIRMWAREPATH', get_option('qemu_firmwarepath'))
1430 config_host_data.set_quoted('CONFIG_QEMU_HELPERDIR', get_option('prefix') / get_option('libexecdir'))
1431 config_host_data.set_quoted('CONFIG_QEMU_ICONDIR', get_option('prefix') / qemu_icondir)
1432 config_host_data.set_quoted('CONFIG_QEMU_LOCALEDIR', get_option('prefix') / get_option('localedir'))
1433 config_host_data.set_quoted('CONFIG_QEMU_LOCALSTATEDIR', get_option('prefix') / get_option('localstatedir'))
1434 config_host_data.set_quoted('CONFIG_QEMU_MODDIR', get_option('prefix') / qemu_moddir)
1435 config_host_data.set_quoted('CONFIG_SYSCONFDIR', get_option('prefix') / get_option('sysconfdir'))
1436
1437 config_host_data.set('CONFIG_ATTR', libattr.found())
1438 config_host_data.set('CONFIG_BRLAPI', brlapi.found())
1439 config_host_data.set('CONFIG_COCOA', cocoa.found())
1440 config_host_data.set('CONFIG_FUZZ', get_option('fuzzing'))
1441 config_host_data.set('CONFIG_GCOV', get_option('b_coverage'))
1442 config_host_data.set('CONFIG_LIBUDEV', libudev.found())
1443 config_host_data.set('CONFIG_LZO', lzo.found())
1444 config_host_data.set('CONFIG_MPATH', mpathpersist.found())
1445 config_host_data.set('CONFIG_MPATH_NEW_API', mpathpersist_new_api)
1446 config_host_data.set('CONFIG_CURL', curl.found())
1447 config_host_data.set('CONFIG_CURSES', curses.found())
1448 config_host_data.set('CONFIG_GBM', gbm.found())
1449 config_host_data.set('CONFIG_GLUSTERFS', glusterfs.found())
1450 if glusterfs.found()
1451 config_host_data.set('CONFIG_GLUSTERFS_XLATOR_OPT', glusterfs.version().version_compare('>=4'))
1452 config_host_data.set('CONFIG_GLUSTERFS_DISCARD', glusterfs.version().version_compare('>=5'))
1453 config_host_data.set('CONFIG_GLUSTERFS_FALLOCATE', glusterfs.version().version_compare('>=6'))
1454 config_host_data.set('CONFIG_GLUSTERFS_ZEROFILL', glusterfs.version().version_compare('>=6'))
1455 config_host_data.set('CONFIG_GLUSTERFS_FTRUNCATE_HAS_STAT', glusterfs_ftruncate_has_stat)
1456 config_host_data.set('CONFIG_GLUSTERFS_IOCB_HAS_STAT', glusterfs_iocb_has_stat)
1457 endif
1458 config_host_data.set('CONFIG_GTK', gtk.found())
1459 config_host_data.set('CONFIG_VTE', vte.found())
1460 config_host_data.set('CONFIG_LIBATTR', have_old_libattr)
1461 config_host_data.set('CONFIG_LIBCAP_NG', libcap_ng.found())
1462 config_host_data.set('CONFIG_EBPF', libbpf.found())
1463 config_host_data.set('CONFIG_LIBDAXCTL', libdaxctl.found())
1464 config_host_data.set('CONFIG_LIBISCSI', libiscsi.found())
1465 config_host_data.set('CONFIG_LIBNFS', libnfs.found())
1466 config_host_data.set('CONFIG_LINUX_AIO', libaio.found())
1467 config_host_data.set('CONFIG_LINUX_IO_URING', linux_io_uring.found())
1468 config_host_data.set('CONFIG_LIBPMEM', libpmem.found())
1469 config_host_data.set('CONFIG_RBD', rbd.found())
1470 config_host_data.set('CONFIG_SDL', sdl.found())
1471 config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found())
1472 config_host_data.set('CONFIG_SECCOMP', seccomp.found())
1473 config_host_data.set('CONFIG_SNAPPY', snappy.found())
1474 config_host_data.set('CONFIG_USB_LIBUSB', libusb.found())
1475 config_host_data.set('CONFIG_VDE', vde.found())
1476 config_host_data.set('CONFIG_VHOST_USER_BLK_SERVER', have_vhost_user_blk_server)
1477 config_host_data.set('CONFIG_VNC', vnc.found())
1478 config_host_data.set('CONFIG_VNC_JPEG', jpeg.found())
1479 config_host_data.set('CONFIG_VNC_PNG', png.found())
1480 config_host_data.set('CONFIG_VNC_SASL', sasl.found())
1481 config_host_data.set('CONFIG_VIRTFS', have_virtfs)
1482 config_host_data.set('CONFIG_VTE', vte.found())
1483 config_host_data.set('CONFIG_XKBCOMMON', xkbcommon.found())
1484 config_host_data.set('CONFIG_KEYUTILS', keyutils.found())
1485 config_host_data.set('CONFIG_GETTID', has_gettid)
1486 config_host_data.set('CONFIG_GNUTLS', gnutls.found())
1487 config_host_data.set('CONFIG_GNUTLS_CRYPTO', gnutls_crypto.found())
1488 config_host_data.set('CONFIG_GCRYPT', gcrypt.found())
1489 config_host_data.set('CONFIG_NETTLE', nettle.found())
1490 config_host_data.set('CONFIG_QEMU_PRIVATE_XTS', xts == 'private')
1491 config_host_data.set('CONFIG_MALLOC_TRIM', has_malloc_trim)
1492 config_host_data.set('CONFIG_STATX', has_statx)
1493 config_host_data.set('CONFIG_ZSTD', zstd.found())
1494 config_host_data.set('CONFIG_FUSE', fuse.found())
1495 config_host_data.set('CONFIG_FUSE_LSEEK', fuse_lseek.found())
1496 config_host_data.set('CONFIG_SPICE_PROTOCOL', spice_protocol.found())
1497 config_host_data.set('CONFIG_SPICE', spice.found())
1498 config_host_data.set('CONFIG_X11', x11.found())
1499 config_host_data.set('CONFIG_CFI', get_option('cfi'))
1500 config_host_data.set('QEMU_VERSION', '"@0@"'.format(meson.project_version()))
1501 config_host_data.set('QEMU_VERSION_MAJOR', meson.project_version().split('.')[0])
1502 config_host_data.set('QEMU_VERSION_MINOR', meson.project_version().split('.')[1])
1503 config_host_data.set('QEMU_VERSION_MICRO', meson.project_version().split('.')[2])
1504
1505 config_host_data.set_quoted('CONFIG_HOST_DSOSUF', host_dsosuf)
1506 config_host_data.set('HAVE_HOST_BLOCK_DEVICE', have_host_block_device)
1507 config_host_data.set('HOST_WORDS_BIGENDIAN', host_machine.endian() == 'big')
1508
1509 # has_header
1510 config_host_data.set('CONFIG_EPOLL', cc.has_header('sys/epoll.h'))
1511 config_host_data.set('CONFIG_LINUX_MAGIC_H', cc.has_header('linux/magic.h'))
1512 config_host_data.set('CONFIG_VALGRIND_H', cc.has_header('valgrind/valgrind.h'))
1513 config_host_data.set('HAVE_BTRFS_H', cc.has_header('linux/btrfs.h'))
1514 config_host_data.set('HAVE_DRM_H', cc.has_header('libdrm/drm.h'))
1515 config_host_data.set('HAVE_PTY_H', cc.has_header('pty.h'))
1516 config_host_data.set('HAVE_SYS_DISK_H', cc.has_header('sys/disk.h'))
1517 config_host_data.set('HAVE_SYS_IOCCOM_H', cc.has_header('sys/ioccom.h'))
1518 config_host_data.set('HAVE_SYS_KCOV_H', cc.has_header('sys/kcov.h'))
1519
1520 # has_function
1521 config_host_data.set('CONFIG_ACCEPT4', cc.has_function('accept4'))
1522 config_host_data.set('CONFIG_CLOCK_ADJTIME', cc.has_function('clock_adjtime'))
1523 config_host_data.set('CONFIG_DUP3', cc.has_function('dup3'))
1524 config_host_data.set('CONFIG_FALLOCATE', cc.has_function('fallocate'))
1525 config_host_data.set('CONFIG_POSIX_FALLOCATE', cc.has_function('posix_fallocate'))
1526 config_host_data.set('CONFIG_POSIX_MEMALIGN', cc.has_function('posix_memalign'))
1527 config_host_data.set('CONFIG_PPOLL', cc.has_function('ppoll'))
1528 config_host_data.set('CONFIG_PREADV', cc.has_function('preadv', prefix: '#include <sys/uio.h>'))
1529 config_host_data.set('CONFIG_SEM_TIMEDWAIT', cc.has_function('sem_timedwait', dependencies: threads))
1530 config_host_data.set('CONFIG_SENDFILE', cc.has_function('sendfile'))
1531 config_host_data.set('CONFIG_SETNS', cc.has_function('setns') and cc.has_function('unshare'))
1532 config_host_data.set('CONFIG_SYNCFS', cc.has_function('syncfs'))
1533 config_host_data.set('CONFIG_SYNC_FILE_RANGE', cc.has_function('sync_file_range'))
1534 config_host_data.set('CONFIG_TIMERFD', cc.has_function('timerfd_create'))
1535 config_host_data.set('HAVE_COPY_FILE_RANGE', cc.has_function('copy_file_range'))
1536 config_host_data.set('HAVE_OPENPTY', cc.has_function('openpty', dependencies: util))
1537 config_host_data.set('HAVE_STRCHRNUL', cc.has_function('strchrnul'))
1538 config_host_data.set('HAVE_SYSTEM_FUNCTION', cc.has_function('system', prefix: '#include <stdlib.h>'))
1539 if rdma.found()
1540 config_host_data.set('HAVE_IBV_ADVISE_MR',
1541 cc.has_function('ibv_advise_mr',
1542 args: config_host['RDMA_LIBS'].split(),
1543 prefix: '#include <infiniband/verbs.h>'))
1544 endif
1545
1546 # has_header_symbol
1547 config_host_data.set('CONFIG_BYTESWAP_H',
1548 cc.has_header_symbol('byteswap.h', 'bswap_32'))
1549 config_host_data.set('CONFIG_EPOLL_CREATE1',
1550 cc.has_header_symbol('sys/epoll.h', 'epoll_create1'))
1551 config_host_data.set('CONFIG_HAS_ENVIRON',
1552 cc.has_header_symbol('unistd.h', 'environ', prefix: gnu_source_prefix))
1553 config_host_data.set('CONFIG_FALLOCATE_PUNCH_HOLE',
1554 cc.has_header_symbol('linux/falloc.h', 'FALLOC_FL_PUNCH_HOLE') and
1555 cc.has_header_symbol('linux/falloc.h', 'FALLOC_FL_KEEP_SIZE'))
1556 config_host_data.set('CONFIG_FALLOCATE_ZERO_RANGE',
1557 cc.has_header_symbol('linux/falloc.h', 'FALLOC_FL_ZERO_RANGE'))
1558 config_host_data.set('CONFIG_FIEMAP',
1559 cc.has_header('linux/fiemap.h') and
1560 cc.has_header_symbol('linux/fs.h', 'FS_IOC_FIEMAP'))
1561 config_host_data.set('CONFIG_GETRANDOM',
1562 cc.has_function('getrandom') and
1563 cc.has_header_symbol('sys/random.h', 'GRND_NONBLOCK'))
1564 config_host_data.set('CONFIG_INOTIFY',
1565 cc.has_header_symbol('sys/inotify.h', 'inotify_init'))
1566 config_host_data.set('CONFIG_INOTIFY1',
1567 cc.has_header_symbol('sys/inotify.h', 'inotify_init1'))
1568 config_host_data.set('CONFIG_IOVEC',
1569 cc.has_header_symbol('sys/uio.h', 'struct iovec'))
1570 config_host_data.set('CONFIG_MACHINE_BSWAP_H',
1571 cc.has_header_symbol('machine/bswap.h', 'bswap32',
1572 prefix: '''#include <sys/endian.h>
1573 #include <sys/types.h>'''))
1574 config_host_data.set('CONFIG_PRCTL_PR_SET_TIMERSLACK',
1575 cc.has_header_symbol('sys/prctl.h', 'PR_SET_TIMERSLACK'))
1576 config_host_data.set('CONFIG_RTNETLINK',
1577 cc.has_header_symbol('linux/rtnetlink.h', 'IFLA_PROTO_DOWN'))
1578 config_host_data.set('CONFIG_SYSMACROS',
1579 cc.has_header_symbol('sys/sysmacros.h', 'makedev'))
1580 config_host_data.set('HAVE_OPTRESET',
1581 cc.has_header_symbol('getopt.h', 'optreset'))
1582 config_host_data.set('HAVE_UTMPX',
1583 cc.has_header_symbol('utmpx.h', 'struct utmpx'))
1584 config_host_data.set('HAVE_IPPROTO_MPTCP',
1585 cc.has_header_symbol('netinet/in.h', 'IPPROTO_MPTCP'))
1586
1587 # has_member
1588 config_host_data.set('HAVE_SIGEV_NOTIFY_THREAD_ID',
1589 cc.has_member('struct sigevent', 'sigev_notify_thread_id',
1590 prefix: '#include <signal.h>'))
1591 config_host_data.set('HAVE_STRUCT_STAT_ST_ATIM',
1592 cc.has_member('struct stat', 'st_atim',
1593 prefix: '#include <sys/stat.h>'))
1594
1595 config_host_data.set('CONFIG_EVENTFD', cc.links('''
1596 #include <sys/eventfd.h>
1597 int main(void) { return eventfd(0, EFD_NONBLOCK | EFD_CLOEXEC); }'''))
1598 config_host_data.set('CONFIG_FDATASYNC', cc.links(gnu_source_prefix + '''
1599 #include <unistd.h>
1600 int main(void) {
1601 #if defined(_POSIX_SYNCHRONIZED_IO) && _POSIX_SYNCHRONIZED_IO > 0
1602 return fdatasync(0);
1603 #else
1604 #error Not supported
1605 #endif
1606 }'''))
1607 config_host_data.set('CONFIG_MADVISE', cc.links(gnu_source_prefix + '''
1608 #include <sys/types.h>
1609 #include <sys/mman.h>
1610 #include <stddef.h>
1611 int main(void) { return madvise(NULL, 0, MADV_DONTNEED); }'''))
1612 config_host_data.set('CONFIG_MEMFD', cc.links(gnu_source_prefix + '''
1613 #include <sys/mman.h>
1614 int main(void) { return memfd_create("foo", MFD_ALLOW_SEALING); }'''))
1615 config_host_data.set('CONFIG_OPEN_BY_HANDLE', cc.links(gnu_source_prefix + '''
1616 #include <fcntl.h>
1617 #if !defined(AT_EMPTY_PATH)
1618 # error missing definition
1619 #else
1620 int main(void) { struct file_handle fh; return open_by_handle_at(0, &fh, 0); }
1621 #endif'''))
1622 config_host_data.set('CONFIG_PIPE2', cc.links(gnu_source_prefix + '''
1623 #include <unistd.h>
1624 #include <fcntl.h>
1625
1626 int main(void)
1627 {
1628 int pipefd[2];
1629 return pipe2(pipefd, O_CLOEXEC);
1630 }'''))
1631 config_host_data.set('CONFIG_POSIX_MADVISE', cc.links(gnu_source_prefix + '''
1632 #include <sys/mman.h>
1633 #include <stddef.h>
1634 int main(void) { return posix_madvise(NULL, 0, POSIX_MADV_DONTNEED); }'''))
1635
1636 config_host_data.set('CONFIG_PTHREAD_SETNAME_NP_W_TID', cc.links('''
1637 #include <pthread.h>
1638
1639 static void *f(void *p) { return NULL; }
1640 int main(void)
1641 {
1642 pthread_t thread;
1643 pthread_create(&thread, 0, f, 0);
1644 pthread_setname_np(thread, "QEMU");
1645 return 0;
1646 }''', dependencies: threads))
1647 config_host_data.set('CONFIG_PTHREAD_SETNAME_NP_WO_TID', cc.links('''
1648 #include <pthread.h>
1649
1650 static void *f(void *p) { pthread_setname_np("QEMU"); return NULL; }
1651 int main(void)
1652 {
1653 pthread_t thread;
1654 pthread_create(&thread, 0, f, 0);
1655 return 0;
1656 }''', dependencies: threads))
1657
1658 config_host_data.set('CONFIG_SIGNALFD', cc.links(gnu_source_prefix + '''
1659 #include <sys/signalfd.h>
1660 #include <stddef.h>
1661 int main(void) { return signalfd(-1, NULL, SFD_CLOEXEC); }'''))
1662 config_host_data.set('CONFIG_SPLICE', cc.links(gnu_source_prefix + '''
1663 #include <unistd.h>
1664 #include <fcntl.h>
1665 #include <limits.h>
1666
1667 int main(void)
1668 {
1669 int len, fd = 0;
1670 len = tee(STDIN_FILENO, STDOUT_FILENO, INT_MAX, SPLICE_F_NONBLOCK);
1671 splice(STDIN_FILENO, NULL, fd, NULL, len, SPLICE_F_MOVE);
1672 return 0;
1673 }'''))
1674
1675 config_host_data.set('HAVE_MLOCKALL', cc.links(gnu_source_prefix + '''
1676 #include <sys/mman.h>
1677 int main(int argc, char *argv[]) {
1678 return mlockall(MCL_FUTURE);
1679 }'''))
1680
1681 have_netmap = false
1682 if not get_option('netmap').disabled() and have_system
1683 have_netmap = cc.compiles('''
1684 #include <inttypes.h>
1685 #include <net/if.h>
1686 #include <net/netmap.h>
1687 #include <net/netmap_user.h>
1688 #if (NETMAP_API < 11) || (NETMAP_API > 15)
1689 #error
1690 #endif
1691 int main(void) { return 0; }''')
1692 if not have_netmap and get_option('netmap').enabled()
1693 error('Netmap headers not available')
1694 endif
1695 endif
1696 config_host_data.set('CONFIG_NETMAP', have_netmap)
1697
1698 # Work around a system header bug with some kernel/XFS header
1699 # versions where they both try to define 'struct fsxattr':
1700 # xfs headers will not try to redefine structs from linux headers
1701 # if this macro is set.
1702 config_host_data.set('HAVE_FSXATTR', cc.links('''
1703 #include <linux/fs.h>'
1704 struct fsxattr foo;
1705 int main(void) {
1706 return 0;
1707 }'''))
1708
1709 # Some versions of Mac OS X incorrectly define SIZE_MAX
1710 config_host_data.set('HAVE_BROKEN_SIZE_MAX', not cc.compiles('''
1711 #include <stdint.h>
1712 #include <stdio.h>
1713 int main(int argc, char *argv[]) {
1714 return printf("%zu", SIZE_MAX);
1715 }''', args: ['-Werror']))
1716
1717 # See if 64-bit atomic operations are supported.
1718 # Note that without __atomic builtins, we can only
1719 # assume atomic loads/stores max at pointer size.
1720 config_host_data.set('CONFIG_ATOMIC64', cc.links('''
1721 #include <stdint.h>
1722 int main(void)
1723 {
1724 uint64_t x = 0, y = 0;
1725 y = __atomic_load_n(&x, __ATOMIC_RELAXED);
1726 __atomic_store_n(&x, y, __ATOMIC_RELAXED);
1727 __atomic_compare_exchange_n(&x, &y, x, 0, __ATOMIC_RELAXED, __ATOMIC_RELAXED);
1728 __atomic_exchange_n(&x, y, __ATOMIC_RELAXED);
1729 __atomic_fetch_add(&x, y, __ATOMIC_RELAXED);
1730 return 0;
1731 }'''))
1732
1733 config_host_data.set('CONFIG_GETAUXVAL', cc.links(gnu_source_prefix + '''
1734 #include <sys/auxv.h>
1735 int main(void) {
1736 return getauxval(AT_HWCAP) == 0;
1737 }'''))
1738
1739 config_host_data.set('CONFIG_AF_VSOCK', cc.compiles(gnu_source_prefix + '''
1740 #include <errno.h>
1741 #include <sys/types.h>
1742 #include <sys/socket.h>
1743 #if !defined(AF_VSOCK)
1744 # error missing AF_VSOCK flag
1745 #endif
1746 #include <linux/vm_sockets.h>
1747 int main(void) {
1748 int sock, ret;
1749 struct sockaddr_vm svm;
1750 socklen_t len = sizeof(svm);
1751 sock = socket(AF_VSOCK, SOCK_STREAM, 0);
1752 ret = getpeername(sock, (struct sockaddr *)&svm, &len);
1753 if ((ret == -1) && (errno == ENOTCONN)) {
1754 return 0;
1755 }
1756 return -1;
1757 }'''))
1758
1759 ignored = ['CONFIG_QEMU_INTERP_PREFIX', # actually per-target
1760 'HAVE_GDB_BIN']
1761 arrays = ['CONFIG_BDRV_RW_WHITELIST', 'CONFIG_BDRV_RO_WHITELIST']
1762 strings = ['CONFIG_IASL']
1763 foreach k, v: config_host
1764 if ignored.contains(k)
1765 # do nothing
1766 elif arrays.contains(k)
1767 if v != ''
1768 v = '"' + '", "'.join(v.split()) + '", '
1769 endif
1770 config_host_data.set(k, v)
1771 elif k == 'ARCH'
1772 config_host_data.set('HOST_' + v.to_upper(), 1)
1773 elif strings.contains(k)
1774 config_host_data.set_quoted(k, v)
1775 elif k.startswith('CONFIG_')
1776 config_host_data.set(k, v == 'y' ? 1 : v)
1777 endif
1778 endforeach
1779
1780 ########################
1781 # Target configuration #
1782 ########################
1783
1784 minikconf = find_program('scripts/minikconf.py')
1785 config_all = {}
1786 config_all_devices = {}
1787 config_all_disas = {}
1788 config_devices_mak_list = []
1789 config_devices_h = {}
1790 config_target_h = {}
1791 config_target_mak = {}
1792
1793 disassemblers = {
1794 'alpha' : ['CONFIG_ALPHA_DIS'],
1795 'arm' : ['CONFIG_ARM_DIS'],
1796 'avr' : ['CONFIG_AVR_DIS'],
1797 'cris' : ['CONFIG_CRIS_DIS'],
1798 'hexagon' : ['CONFIG_HEXAGON_DIS'],
1799 'hppa' : ['CONFIG_HPPA_DIS'],
1800 'i386' : ['CONFIG_I386_DIS'],
1801 'x86_64' : ['CONFIG_I386_DIS'],
1802 'x32' : ['CONFIG_I386_DIS'],
1803 'm68k' : ['CONFIG_M68K_DIS'],
1804 'microblaze' : ['CONFIG_MICROBLAZE_DIS'],
1805 'mips' : ['CONFIG_MIPS_DIS'],
1806 'nios2' : ['CONFIG_NIOS2_DIS'],
1807 'or1k' : ['CONFIG_OPENRISC_DIS'],
1808 'ppc' : ['CONFIG_PPC_DIS'],
1809 'riscv' : ['CONFIG_RISCV_DIS'],
1810 'rx' : ['CONFIG_RX_DIS'],
1811 's390' : ['CONFIG_S390_DIS'],
1812 'sh4' : ['CONFIG_SH4_DIS'],
1813 'sparc' : ['CONFIG_SPARC_DIS'],
1814 'xtensa' : ['CONFIG_XTENSA_DIS'],
1815 }
1816 if link_language == 'cpp'
1817 disassemblers += {
1818 'aarch64' : [ 'CONFIG_ARM_A64_DIS'],
1819 'arm' : [ 'CONFIG_ARM_DIS', 'CONFIG_ARM_A64_DIS'],
1820 'mips' : [ 'CONFIG_MIPS_DIS', 'CONFIG_NANOMIPS_DIS'],
1821 }
1822 endif
1823
1824 have_ivshmem = config_host_data.get('CONFIG_EVENTFD')
1825 host_kconfig = \
1826 (get_option('fuzzing') ? ['CONFIG_FUZZ=y'] : []) + \
1827 ('CONFIG_TPM' in config_host ? ['CONFIG_TPM=y'] : []) + \
1828 (spice.found() ? ['CONFIG_SPICE=y'] : []) + \
1829 (have_ivshmem ? ['CONFIG_IVSHMEM=y'] : []) + \
1830 ('CONFIG_OPENGL' in config_host ? ['CONFIG_OPENGL=y'] : []) + \
1831 (x11.found() ? ['CONFIG_X11=y'] : []) + \
1832 ('CONFIG_VHOST_USER' in config_host ? ['CONFIG_VHOST_USER=y'] : []) + \
1833 ('CONFIG_VHOST_VDPA' in config_host ? ['CONFIG_VHOST_VDPA=y'] : []) + \
1834 ('CONFIG_VHOST_KERNEL' in config_host ? ['CONFIG_VHOST_KERNEL=y'] : []) + \
1835 (have_virtfs ? ['CONFIG_VIRTFS=y'] : []) + \
1836 ('CONFIG_LINUX' in config_host ? ['CONFIG_LINUX=y'] : []) + \
1837 ('CONFIG_PVRDMA' in config_host ? ['CONFIG_PVRDMA=y'] : []) + \
1838 (multiprocess_allowed ? ['CONFIG_MULTIPROCESS_ALLOWED=y'] : [])
1839
1840 ignored = [ 'TARGET_XML_FILES', 'TARGET_ABI_DIR', 'TARGET_ARCH' ]
1841
1842 default_targets = 'CONFIG_DEFAULT_TARGETS' in config_host
1843 actual_target_dirs = []
1844 fdt_required = []
1845 foreach target : target_dirs
1846 config_target = { 'TARGET_NAME': target.split('-')[0] }
1847 if target.endswith('linux-user')
1848 if targetos != 'linux'
1849 if default_targets
1850 continue
1851 endif
1852 error('Target @0@ is only available on a Linux host'.format(target))
1853 endif
1854 config_target += { 'CONFIG_LINUX_USER': 'y' }
1855 elif target.endswith('bsd-user')
1856 if 'CONFIG_BSD' not in config_host
1857 if default_targets
1858 continue
1859 endif
1860 error('Target @0@ is only available on a BSD host'.format(target))
1861 endif
1862 config_target += { 'CONFIG_BSD_USER': 'y' }
1863 elif target.endswith('softmmu')
1864 config_target += { 'CONFIG_SOFTMMU': 'y' }
1865 endif
1866 if target.endswith('-user')
1867 config_target += {
1868 'CONFIG_USER_ONLY': 'y',
1869 'CONFIG_QEMU_INTERP_PREFIX':
1870 config_host['CONFIG_QEMU_INTERP_PREFIX'].format(config_target['TARGET_NAME'])
1871 }
1872 endif
1873
1874 accel_kconfig = []
1875 foreach sym: accelerators
1876 if sym == 'CONFIG_TCG' or target in accelerator_targets.get(sym, [])
1877 config_target += { sym: 'y' }
1878 config_all += { sym: 'y' }
1879 if sym == 'CONFIG_TCG' and tcg_arch == 'tci'
1880 config_target += { 'CONFIG_TCG_INTERPRETER': 'y' }
1881 elif sym == 'CONFIG_XEN' and have_xen_pci_passthrough
1882 config_target += { 'CONFIG_XEN_PCI_PASSTHROUGH': 'y' }
1883 endif
1884 if target in modular_tcg
1885 config_target += { 'CONFIG_TCG_MODULAR': 'y' }
1886 else
1887 config_target += { 'CONFIG_TCG_BUILTIN': 'y' }
1888 endif
1889 accel_kconfig += [ sym + '=y' ]
1890 endif
1891 endforeach
1892 if accel_kconfig.length() == 0
1893 if default_targets
1894 continue
1895 endif
1896 error('No accelerator available for target @0@'.format(target))
1897 endif
1898
1899 actual_target_dirs += target
1900 config_target += keyval.load('configs/targets' / target + '.mak')
1901 config_target += { 'TARGET_' + config_target['TARGET_ARCH'].to_upper(): 'y' }
1902
1903 if 'TARGET_NEED_FDT' in config_target
1904 fdt_required += target
1905 endif
1906
1907 # Add default keys
1908 if 'TARGET_BASE_ARCH' not in config_target
1909 config_target += {'TARGET_BASE_ARCH': config_target['TARGET_ARCH']}
1910 endif
1911 if 'TARGET_ABI_DIR' not in config_target
1912 config_target += {'TARGET_ABI_DIR': config_target['TARGET_ARCH']}
1913 endif
1914
1915 foreach k, v: disassemblers
1916 if config_host['ARCH'].startswith(k) or config_target['TARGET_BASE_ARCH'].startswith(k)
1917 foreach sym: v
1918 config_target += { sym: 'y' }
1919 config_all_disas += { sym: 'y' }
1920 endforeach
1921 endif
1922 endforeach
1923
1924 config_target_data = configuration_data()
1925 foreach k, v: config_target
1926 if not k.startswith('TARGET_') and not k.startswith('CONFIG_')
1927 # do nothing
1928 elif ignored.contains(k)
1929 # do nothing
1930 elif k == 'TARGET_BASE_ARCH'
1931 # Note that TARGET_BASE_ARCH ends up in config-target.h but it is
1932 # not used to select files from sourcesets.
1933 config_target_data.set('TARGET_' + v.to_upper(), 1)
1934 elif k == 'TARGET_NAME' or k == 'CONFIG_QEMU_INTERP_PREFIX'
1935 config_target_data.set_quoted(k, v)
1936 elif v == 'y'
1937 config_target_data.set(k, 1)
1938 else
1939 config_target_data.set(k, v)
1940 endif
1941 endforeach
1942 config_target_data.set('QEMU_ARCH',
1943 'QEMU_ARCH_' + config_target['TARGET_BASE_ARCH'].to_upper())
1944 config_target_h += {target: configure_file(output: target + '-config-target.h',
1945 configuration: config_target_data)}
1946
1947 if target.endswith('-softmmu')
1948 config_input = meson.get_external_property(target, 'default')
1949 config_devices_mak = target + '-config-devices.mak'
1950 config_devices_mak = configure_file(
1951 input: ['configs/devices' / target / config_input + '.mak', 'Kconfig'],
1952 output: config_devices_mak,
1953 depfile: config_devices_mak + '.d',
1954 capture: true,
1955 command: [minikconf,
1956 get_option('default_devices') ? '--defconfig' : '--allnoconfig',
1957 config_devices_mak, '@DEPFILE@', '@INPUT@',
1958 host_kconfig, accel_kconfig,
1959 'CONFIG_' + config_target['TARGET_ARCH'].to_upper() + '=y'])
1960
1961 config_devices_data = configuration_data()
1962 config_devices = keyval.load(config_devices_mak)
1963 foreach k, v: config_devices
1964 config_devices_data.set(k, 1)
1965 endforeach
1966 config_devices_mak_list += config_devices_mak
1967 config_devices_h += {target: configure_file(output: target + '-config-devices.h',
1968 configuration: config_devices_data)}
1969 config_target += config_devices
1970 config_all_devices += config_devices
1971 endif
1972 config_target_mak += {target: config_target}
1973 endforeach
1974 target_dirs = actual_target_dirs
1975
1976 # This configuration is used to build files that are shared by
1977 # multiple binaries, and then extracted out of the "common"
1978 # static_library target.
1979 #
1980 # We do not use all_sources()/all_dependencies(), because it would
1981 # build literally all source files, including devices only used by
1982 # targets that are not built for this compilation. The CONFIG_ALL
1983 # pseudo symbol replaces it.
1984
1985 config_all += config_all_devices
1986 config_all += config_host
1987 config_all += config_all_disas
1988 config_all += {
1989 'CONFIG_XEN': config_host.has_key('CONFIG_XEN_BACKEND'),
1990 'CONFIG_SOFTMMU': have_system,
1991 'CONFIG_USER_ONLY': have_user,
1992 'CONFIG_ALL': true,
1993 }
1994
1995 ##############
1996 # Submodules #
1997 ##############
1998
1999 capstone = not_found
2000 capstone_opt = get_option('capstone')
2001 if capstone_opt in ['enabled', 'auto', 'system']
2002 have_internal = fs.exists(meson.current_source_dir() / 'capstone/Makefile')
2003 capstone = dependency('capstone', version: '>=4.0',
2004 kwargs: static_kwargs, method: 'pkg-config',
2005 required: capstone_opt == 'system' or
2006 capstone_opt == 'enabled' and not have_internal)
2007
2008 # Some versions of capstone have broken pkg-config file
2009 # that reports a wrong -I path, causing the #include to
2010 # fail later. If the system has such a broken version
2011 # do not use it.
2012 if capstone.found() and not cc.compiles('#include <capstone.h>',
2013 dependencies: [capstone])
2014 capstone = not_found
2015 if capstone_opt == 'system'
2016 error('system capstone requested, it does not appear to work')
2017 endif
2018 endif
2019
2020 if capstone.found()
2021 capstone_opt = 'system'
2022 elif have_internal
2023 capstone_opt = 'internal'
2024 else
2025 capstone_opt = 'disabled'
2026 endif
2027 endif
2028 if capstone_opt == 'internal'
2029 capstone_data = configuration_data()
2030 capstone_data.set('CAPSTONE_USE_SYS_DYN_MEM', '1')
2031
2032 capstone_files = files(
2033 'capstone/cs.c',
2034 'capstone/MCInst.c',
2035 'capstone/MCInstrDesc.c',
2036 'capstone/MCRegisterInfo.c',
2037 'capstone/SStream.c',
2038 'capstone/utils.c'
2039 )
2040
2041 if 'CONFIG_ARM_DIS' in config_all_disas
2042 capstone_data.set('CAPSTONE_HAS_ARM', '1')
2043 capstone_files += files(
2044 'capstone/arch/ARM/ARMDisassembler.c',
2045 'capstone/arch/ARM/ARMInstPrinter.c',
2046 'capstone/arch/ARM/ARMMapping.c',
2047 'capstone/arch/ARM/ARMModule.c'
2048 )
2049 endif
2050
2051 # FIXME: This config entry currently depends on a c++ compiler.
2052 # Which is needed for building libvixl, but not for capstone.
2053 if 'CONFIG_ARM_A64_DIS' in config_all_disas
2054 capstone_data.set('CAPSTONE_HAS_ARM64', '1')
2055 capstone_files += files(
2056 'capstone/arch/AArch64/AArch64BaseInfo.c',
2057 'capstone/arch/AArch64/AArch64Disassembler.c',
2058 'capstone/arch/AArch64/AArch64InstPrinter.c',
2059 'capstone/arch/AArch64/AArch64Mapping.c',
2060 'capstone/arch/AArch64/AArch64Module.c'
2061 )
2062 endif
2063
2064 if 'CONFIG_PPC_DIS' in config_all_disas
2065 capstone_data.set('CAPSTONE_HAS_POWERPC', '1')
2066 capstone_files += files(
2067 'capstone/arch/PowerPC/PPCDisassembler.c',
2068 'capstone/arch/PowerPC/PPCInstPrinter.c',
2069 'capstone/arch/PowerPC/PPCMapping.c',
2070 'capstone/arch/PowerPC/PPCModule.c'
2071 )
2072 endif
2073
2074 if 'CONFIG_S390_DIS' in config_all_disas
2075 capstone_data.set('CAPSTONE_HAS_SYSZ', '1')
2076 capstone_files += files(
2077 'capstone/arch/SystemZ/SystemZDisassembler.c',
2078 'capstone/arch/SystemZ/SystemZInstPrinter.c',
2079 'capstone/arch/SystemZ/SystemZMapping.c',
2080 'capstone/arch/SystemZ/SystemZModule.c',
2081 'capstone/arch/SystemZ/SystemZMCTargetDesc.c'
2082 )
2083 endif
2084
2085 if 'CONFIG_I386_DIS' in config_all_disas
2086 capstone_data.set('CAPSTONE_HAS_X86', 1)
2087 capstone_files += files(
2088 'capstone/arch/X86/X86Disassembler.c',
2089 'capstone/arch/X86/X86DisassemblerDecoder.c',
2090 'capstone/arch/X86/X86ATTInstPrinter.c',
2091 'capstone/arch/X86/X86IntelInstPrinter.c',
2092 'capstone/arch/X86/X86InstPrinterCommon.c',
2093 'capstone/arch/X86/X86Mapping.c',
2094 'capstone/arch/X86/X86Module.c'
2095 )
2096 endif
2097
2098 configure_file(output: 'capstone-defs.h', configuration: capstone_data)
2099
2100 capstone_cargs = [
2101 # FIXME: There does not seem to be a way to completely replace the c_args
2102 # that come from add_project_arguments() -- we can only add to them.
2103 # So: disable all warnings with a big hammer.
2104 '-Wno-error', '-w',
2105
2106 # Include all configuration defines via a header file, which will wind up
2107 # as a dependency on the object file, and thus changes here will result
2108 # in a rebuild.
2109 '-include', 'capstone-defs.h'
2110 ]
2111
2112 libcapstone = static_library('capstone',
2113 build_by_default: false,
2114 sources: capstone_files,
2115 c_args: capstone_cargs,
2116 include_directories: 'capstone/include')
2117 capstone = declare_dependency(link_with: libcapstone,
2118 include_directories: 'capstone/include/capstone')
2119 endif
2120
2121 slirp = not_found
2122 slirp_opt = 'disabled'
2123 if have_system
2124 slirp_opt = get_option('slirp')
2125 if slirp_opt in ['enabled', 'auto', 'system']
2126 have_internal = fs.exists(meson.current_source_dir() / 'slirp/meson.build')
2127 slirp = dependency('slirp', kwargs: static_kwargs,
2128 method: 'pkg-config',
2129 required: slirp_opt == 'system' or
2130 slirp_opt == 'enabled' and not have_internal)
2131 if slirp.found()
2132 slirp_opt = 'system'
2133 elif have_internal
2134 slirp_opt = 'internal'
2135 else
2136 slirp_opt = 'disabled'
2137 endif
2138 endif
2139 if slirp_opt == 'internal'
2140 slirp_deps = []
2141 if targetos == 'windows'
2142 slirp_deps = cc.find_library('iphlpapi')
2143 elif targetos == 'darwin'
2144 slirp_deps = cc.find_library('resolv')
2145 endif
2146 slirp_conf = configuration_data()
2147 slirp_conf.set('SLIRP_MAJOR_VERSION', meson.project_version().split('.')[0])
2148 slirp_conf.set('SLIRP_MINOR_VERSION', meson.project_version().split('.')[1])
2149 slirp_conf.set('SLIRP_MICRO_VERSION', meson.project_version().split('.')[2])
2150 slirp_conf.set_quoted('SLIRP_VERSION_STRING', meson.project_version())
2151 slirp_cargs = ['-DG_LOG_DOMAIN="Slirp"']
2152 slirp_files = [
2153 'slirp/src/arp_table.c',
2154 'slirp/src/bootp.c',
2155 'slirp/src/cksum.c',
2156 'slirp/src/dhcpv6.c',
2157 'slirp/src/dnssearch.c',
2158 'slirp/src/if.c',
2159 'slirp/src/ip6_icmp.c',
2160 'slirp/src/ip6_input.c',
2161 'slirp/src/ip6_output.c',
2162 'slirp/src/ip_icmp.c',
2163 'slirp/src/ip_input.c',
2164 'slirp/src/ip_output.c',
2165 'slirp/src/mbuf.c',
2166 'slirp/src/misc.c',
2167 'slirp/src/ncsi.c',
2168 'slirp/src/ndp_table.c',
2169 'slirp/src/sbuf.c',
2170 'slirp/src/slirp.c',
2171 'slirp/src/socket.c',
2172 'slirp/src/state.c',
2173 'slirp/src/stream.c',
2174 'slirp/src/tcp_input.c',
2175 'slirp/src/tcp_output.c',
2176 'slirp/src/tcp_subr.c',
2177 'slirp/src/tcp_timer.c',
2178 'slirp/src/tftp.c',
2179 'slirp/src/udp.c',
2180 'slirp/src/udp6.c',
2181 'slirp/src/util.c',
2182 'slirp/src/version.c',
2183 'slirp/src/vmstate.c',
2184 ]
2185
2186 configure_file(
2187 input : 'slirp/src/libslirp-version.h.in',
2188 output : 'libslirp-version.h',
2189 configuration: slirp_conf)
2190
2191 slirp_inc = include_directories('slirp', 'slirp/src')
2192 libslirp = static_library('slirp',
2193 build_by_default: false,
2194 sources: slirp_files,
2195 c_args: slirp_cargs,
2196 include_directories: slirp_inc)
2197 slirp = declare_dependency(link_with: libslirp,
2198 dependencies: slirp_deps,
2199 include_directories: slirp_inc)
2200 endif
2201 endif
2202
2203 # For CFI, we need to compile slirp as a static library together with qemu.
2204 # This is because we register slirp functions as callbacks for QEMU Timers.
2205 # When using a system-wide shared libslirp, the type information for the
2206 # callback is missing and the timer call produces a false positive with CFI.
2207 #
2208 # Now that slirp_opt has been defined, check if the selected slirp is compatible
2209 # with control-flow integrity.
2210 if get_option('cfi') and slirp_opt == 'system'
2211 error('Control-Flow Integrity is not compatible with system-wide slirp.' \
2212 + ' Please configure with --enable-slirp=git')
2213 endif
2214
2215 fdt = not_found
2216 fdt_opt = get_option('fdt')
2217 if have_system
2218 if fdt_opt in ['enabled', 'auto', 'system']
2219 have_internal = fs.exists(meson.current_source_dir() / 'dtc/libfdt/Makefile.libfdt')
2220 fdt = cc.find_library('fdt', kwargs: static_kwargs,
2221 required: fdt_opt == 'system' or
2222 fdt_opt == 'enabled' and not have_internal)
2223 if fdt.found() and cc.links('''
2224 #include <libfdt.h>
2225 #include <libfdt_env.h>
2226 int main(void) { fdt_check_full(NULL, 0); return 0; }''',
2227 dependencies: fdt)
2228 fdt_opt = 'system'
2229 elif fdt_opt == 'system'
2230 error('system libfdt requested, but it is too old (1.5.1 or newer required)')
2231 elif have_internal
2232 fdt_opt = 'internal'
2233 else
2234 fdt_opt = 'disabled'
2235 fdt = not_found
2236 endif
2237 endif
2238 if fdt_opt == 'internal'
2239 fdt_files = files(
2240 'dtc/libfdt/fdt.c',
2241 'dtc/libfdt/fdt_ro.c',
2242 'dtc/libfdt/fdt_wip.c',
2243 'dtc/libfdt/fdt_sw.c',
2244 'dtc/libfdt/fdt_rw.c',
2245 'dtc/libfdt/fdt_strerror.c',
2246 'dtc/libfdt/fdt_empty_tree.c',
2247 'dtc/libfdt/fdt_addresses.c',
2248 'dtc/libfdt/fdt_overlay.c',
2249 'dtc/libfdt/fdt_check.c',
2250 )
2251
2252 fdt_inc = include_directories('dtc/libfdt')
2253 libfdt = static_library('fdt',
2254 build_by_default: false,
2255 sources: fdt_files,
2256 include_directories: fdt_inc)
2257 fdt = declare_dependency(link_with: libfdt,
2258 include_directories: fdt_inc)
2259 endif
2260 endif
2261 if not fdt.found() and fdt_required.length() > 0
2262 error('fdt not available but required by targets ' + ', '.join(fdt_required))
2263 endif
2264
2265 config_host_data.set('CONFIG_CAPSTONE', capstone.found())
2266 config_host_data.set('CONFIG_FDT', fdt.found())
2267 config_host_data.set('CONFIG_SLIRP', slirp.found())
2268
2269 #####################
2270 # Generated sources #
2271 #####################
2272
2273 genh += configure_file(output: 'config-host.h', configuration: config_host_data)
2274
2275 hxtool = find_program('scripts/hxtool')
2276 shaderinclude = find_program('scripts/shaderinclude.pl')
2277 qapi_gen = find_program('scripts/qapi-gen.py')
2278 qapi_gen_depends = [ meson.current_source_dir() / 'scripts/qapi/__init__.py',
2279 meson.current_source_dir() / 'scripts/qapi/commands.py',
2280 meson.current_source_dir() / 'scripts/qapi/common.py',
2281 meson.current_source_dir() / 'scripts/qapi/error.py',
2282 meson.current_source_dir() / 'scripts/qapi/events.py',
2283 meson.current_source_dir() / 'scripts/qapi/expr.py',
2284 meson.current_source_dir() / 'scripts/qapi/gen.py',
2285 meson.current_source_dir() / 'scripts/qapi/introspect.py',
2286 meson.current_source_dir() / 'scripts/qapi/parser.py',
2287 meson.current_source_dir() / 'scripts/qapi/schema.py',
2288 meson.current_source_dir() / 'scripts/qapi/source.py',
2289 meson.current_source_dir() / 'scripts/qapi/types.py',
2290 meson.current_source_dir() / 'scripts/qapi/visit.py',
2291 meson.current_source_dir() / 'scripts/qapi/common.py',
2292 meson.current_source_dir() / 'scripts/qapi-gen.py'
2293 ]
2294
2295 tracetool = [
2296 python, files('scripts/tracetool.py'),
2297 '--backend=' + ','.join(get_option('trace_backends'))
2298 ]
2299 tracetool_depends = files(
2300 'scripts/tracetool/backend/log.py',
2301 'scripts/tracetool/backend/__init__.py',
2302 'scripts/tracetool/backend/dtrace.py',
2303 'scripts/tracetool/backend/ftrace.py',
2304 'scripts/tracetool/backend/simple.py',
2305 'scripts/tracetool/backend/syslog.py',
2306 'scripts/tracetool/backend/ust.py',
2307 'scripts/tracetool/format/tcg_h.py',
2308 'scripts/tracetool/format/ust_events_c.py',
2309 'scripts/tracetool/format/ust_events_h.py',
2310 'scripts/tracetool/format/__init__.py',
2311 'scripts/tracetool/format/d.py',
2312 'scripts/tracetool/format/tcg_helper_c.py',
2313 'scripts/tracetool/format/simpletrace_stap.py',
2314 'scripts/tracetool/format/c.py',
2315 'scripts/tracetool/format/h.py',
2316 'scripts/tracetool/format/tcg_helper_h.py',
2317 'scripts/tracetool/format/log_stap.py',
2318 'scripts/tracetool/format/stap.py',
2319 'scripts/tracetool/format/tcg_helper_wrapper_h.py',
2320 'scripts/tracetool/__init__.py',
2321 'scripts/tracetool/transform.py',
2322 'scripts/tracetool/vcpu.py'
2323 )
2324
2325 qemu_version_cmd = [find_program('scripts/qemu-version.sh'),
2326 meson.current_source_dir(),
2327 config_host['PKGVERSION'], meson.project_version()]
2328 qemu_version = custom_target('qemu-version.h',
2329 output: 'qemu-version.h',
2330 command: qemu_version_cmd,
2331 capture: true,
2332 build_by_default: true,
2333 build_always_stale: true)
2334 genh += qemu_version
2335
2336 hxdep = []
2337 hx_headers = [
2338 ['qemu-options.hx', 'qemu-options.def'],
2339 ['qemu-img-cmds.hx', 'qemu-img-cmds.h'],
2340 ]
2341 if have_system
2342 hx_headers += [
2343 ['hmp-commands.hx', 'hmp-commands.h'],
2344 ['hmp-commands-info.hx', 'hmp-commands-info.h'],
2345 ]
2346 endif
2347 foreach d : hx_headers
2348 hxdep += custom_target(d[1],
2349 input: files(d[0]),
2350 output: d[1],
2351 capture: true,
2352 build_by_default: true, # to be removed when added to a target
2353 command: [hxtool, '-h', '@INPUT0@'])
2354 endforeach
2355 genh += hxdep
2356
2357 ###################
2358 # Collect sources #
2359 ###################
2360
2361 authz_ss = ss.source_set()
2362 blockdev_ss = ss.source_set()
2363 block_ss = ss.source_set()
2364 bsd_user_ss = ss.source_set()
2365 chardev_ss = ss.source_set()
2366 common_ss = ss.source_set()
2367 crypto_ss = ss.source_set()
2368 hwcore_ss = ss.source_set()
2369 io_ss = ss.source_set()
2370 linux_user_ss = ss.source_set()
2371 qmp_ss = ss.source_set()
2372 qom_ss = ss.source_set()
2373 softmmu_ss = ss.source_set()
2374 specific_fuzz_ss = ss.source_set()
2375 specific_ss = ss.source_set()
2376 stub_ss = ss.source_set()
2377 trace_ss = ss.source_set()
2378 user_ss = ss.source_set()
2379 util_ss = ss.source_set()
2380
2381 # accel modules
2382 qtest_module_ss = ss.source_set()
2383 tcg_module_ss = ss.source_set()
2384
2385 modules = {}
2386 target_modules = {}
2387 hw_arch = {}
2388 target_arch = {}
2389 target_softmmu_arch = {}
2390 target_user_arch = {}
2391
2392 ###############
2393 # Trace files #
2394 ###############
2395
2396 # TODO: add each directory to the subdirs from its own meson.build, once
2397 # we have those
2398 trace_events_subdirs = [
2399 'crypto',
2400 'qapi',
2401 'qom',
2402 'monitor',
2403 'util',
2404 ]
2405 if have_user
2406 trace_events_subdirs += [ 'linux-user' ]
2407 endif
2408 if have_block
2409 trace_events_subdirs += [
2410 'authz',
2411 'block',
2412 'io',
2413 'nbd',
2414 'scsi',
2415 ]
2416 endif
2417 if have_system
2418 trace_events_subdirs += [
2419 'accel/kvm',
2420 'audio',
2421 'backends',
2422 'backends/tpm',
2423 'chardev',
2424 'ebpf',
2425 'hw/9pfs',
2426 'hw/acpi',
2427 'hw/adc',
2428 'hw/alpha',
2429 'hw/arm',
2430 'hw/audio',
2431 'hw/block',
2432 'hw/block/dataplane',
2433 'hw/char',
2434 'hw/display',
2435 'hw/dma',
2436 'hw/hppa',
2437 'hw/hyperv',
2438 'hw/i2c',
2439 'hw/i386',
2440 'hw/i386/xen',
2441 'hw/ide',
2442 'hw/input',
2443 'hw/intc',
2444 'hw/isa',
2445 'hw/mem',
2446 'hw/mips',
2447 'hw/misc',
2448 'hw/misc/macio',
2449 'hw/net',
2450 'hw/net/can',
2451 'hw/nubus',
2452 'hw/nvme',
2453 'hw/nvram',
2454 'hw/pci',
2455 'hw/pci-host',
2456 'hw/ppc',
2457 'hw/rdma',
2458 'hw/rdma/vmw',
2459 'hw/rtc',
2460 'hw/s390x',
2461 'hw/scsi',
2462 'hw/sd',
2463 'hw/sh4',
2464 'hw/sparc',
2465 'hw/sparc64',
2466 'hw/ssi',
2467 'hw/timer',
2468 'hw/tpm',
2469 'hw/usb',
2470 'hw/vfio',
2471 'hw/virtio',
2472 'hw/watchdog',
2473 'hw/xen',
2474 'hw/gpio',
2475 'migration',
2476 'net',
2477 'softmmu',
2478 'ui',
2479 'hw/remote',
2480 ]
2481 endif
2482 if have_system or have_user
2483 trace_events_subdirs += [
2484 'accel/tcg',
2485 'hw/core',
2486 'target/arm',
2487 'target/arm/hvf',
2488 'target/hppa',
2489 'target/i386',
2490 'target/i386/kvm',
2491 'target/mips/tcg',
2492 'target/ppc',
2493 'target/riscv',
2494 'target/s390x',
2495 'target/s390x/kvm',
2496 'target/sparc',
2497 ]
2498 endif
2499
2500 vhost_user = not_found
2501 if 'CONFIG_VHOST_USER' in config_host
2502 libvhost_user = subproject('libvhost-user')
2503 vhost_user = libvhost_user.get_variable('vhost_user_dep')
2504 endif
2505
2506 subdir('qapi')
2507 subdir('qobject')
2508 subdir('stubs')
2509 subdir('trace')
2510 subdir('util')
2511 subdir('qom')
2512 subdir('authz')
2513 subdir('crypto')
2514 subdir('ui')
2515
2516
2517 if enable_modules
2518 libmodulecommon = static_library('module-common', files('module-common.c') + genh, pic: true, c_args: '-DBUILD_DSO')
2519 modulecommon = declare_dependency(link_whole: libmodulecommon, compile_args: '-DBUILD_DSO')
2520 endif
2521
2522 stub_ss = stub_ss.apply(config_all, strict: false)
2523
2524 util_ss.add_all(trace_ss)
2525 util_ss = util_ss.apply(config_all, strict: false)
2526 libqemuutil = static_library('qemuutil',
2527 sources: util_ss.sources() + stub_ss.sources() + genh,
2528 dependencies: [util_ss.dependencies(), libm, threads, glib, socket, malloc, pixman])
2529 qemuutil = declare_dependency(link_with: libqemuutil,
2530 sources: genh + version_res)
2531
2532 if have_system or have_user
2533 decodetree = generator(find_program('scripts/decodetree.py'),
2534 output: 'decode-@BASENAME@.c.inc',
2535 arguments: ['@INPUT@', '@EXTRA_ARGS@', '-o', '@OUTPUT@'])
2536 subdir('libdecnumber')
2537 subdir('target')
2538 endif
2539
2540 subdir('audio')
2541 subdir('io')
2542 subdir('chardev')
2543 subdir('fsdev')
2544 subdir('dump')
2545
2546 if have_block
2547 block_ss.add(files(
2548 'block.c',
2549 'blockjob.c',
2550 'job.c',
2551 'qemu-io-cmds.c',
2552 ))
2553 block_ss.add(when: 'CONFIG_REPLICATION', if_true: files('replication.c'))
2554
2555 subdir('nbd')
2556 subdir('scsi')
2557 subdir('block')
2558
2559 blockdev_ss.add(files(
2560 'blockdev.c',
2561 'blockdev-nbd.c',
2562 'iothread.c',
2563 'job-qmp.c',
2564 ), gnutls)
2565
2566 # os-posix.c contains POSIX-specific functions used by qemu-storage-daemon,
2567 # os-win32.c does not
2568 blockdev_ss.add(when: 'CONFIG_POSIX', if_true: files('os-posix.c'))
2569 softmmu_ss.add(when: 'CONFIG_WIN32', if_true: [files('os-win32.c')])
2570 endif
2571
2572 common_ss.add(files('cpus-common.c'))
2573
2574 subdir('softmmu')
2575
2576 common_ss.add(capstone)
2577 specific_ss.add(files('cpu.c', 'disas.c', 'gdbstub.c'), capstone)
2578
2579 # Work around a gcc bug/misfeature wherein constant propagation looks
2580 # through an alias:
2581 # https://gcc.gnu.org/bugzilla/show_bug.cgi?id=99696
2582 # to guess that a const variable is always zero. Without lto, this is
2583 # impossible, as the alias is restricted to page-vary-common.c. Indeed,
2584 # without lto, not even the alias is required -- we simply use different
2585 # declarations in different compilation units.
2586 pagevary = files('page-vary-common.c')
2587 if get_option('b_lto')
2588 pagevary_flags = ['-fno-lto']
2589 if get_option('cfi')
2590 pagevary_flags += '-fno-sanitize=cfi-icall'
2591 endif
2592 pagevary = static_library('page-vary-common', sources: pagevary,
2593 c_args: pagevary_flags)
2594 pagevary = declare_dependency(link_with: pagevary)
2595 endif
2596 common_ss.add(pagevary)
2597 specific_ss.add(files('page-vary.c'))
2598
2599 subdir('backends')
2600 subdir('disas')
2601 subdir('migration')
2602 subdir('monitor')
2603 subdir('net')
2604 subdir('replay')
2605 subdir('semihosting')
2606 subdir('hw')
2607 subdir('tcg')
2608 subdir('fpu')
2609 subdir('accel')
2610 subdir('plugins')
2611 subdir('bsd-user')
2612 subdir('linux-user')
2613 subdir('ebpf')
2614
2615 common_ss.add(libbpf)
2616
2617 specific_ss.add_all(when: 'CONFIG_BSD_USER', if_true: bsd_user_ss)
2618
2619 linux_user_ss.add(files('thunk.c'))
2620 specific_ss.add_all(when: 'CONFIG_LINUX_USER', if_true: linux_user_ss)
2621
2622 # needed for fuzzing binaries
2623 subdir('tests/qtest/libqos')
2624 subdir('tests/qtest/fuzz')
2625
2626 # accel modules
2627 tcg_real_module_ss = ss.source_set()
2628 tcg_real_module_ss.add_all(when: 'CONFIG_TCG_MODULAR', if_true: tcg_module_ss)
2629 specific_ss.add_all(when: 'CONFIG_TCG_BUILTIN', if_true: tcg_module_ss)
2630 target_modules += { 'accel' : { 'qtest': qtest_module_ss,
2631 'tcg': tcg_real_module_ss }}
2632
2633 ########################
2634 # Library dependencies #
2635 ########################
2636
2637 modinfo_collect = find_program('scripts/modinfo-collect.py')
2638 modinfo_generate = find_program('scripts/modinfo-generate.py')
2639 modinfo_files = []
2640
2641 block_mods = []
2642 softmmu_mods = []
2643 foreach d, list : modules
2644 foreach m, module_ss : list
2645 if enable_modules and targetos != 'windows'
2646 module_ss = module_ss.apply(config_all, strict: false)
2647 sl = static_library(d + '-' + m, [genh, module_ss.sources()],
2648 dependencies: [modulecommon, module_ss.dependencies()], pic: true)
2649 if d == 'block'
2650 block_mods += sl
2651 else
2652 softmmu_mods += sl
2653 endif
2654 if module_ss.sources() != []
2655 # FIXME: Should use sl.extract_all_objects(recursive: true) as
2656 # input. Sources can be used multiple times but objects are
2657 # unique when it comes to lookup in compile_commands.json.
2658 # Depnds on a mesion version with
2659 # https://github.com/mesonbuild/meson/pull/8900
2660 modinfo_files += custom_target(d + '-' + m + '.modinfo',
2661 output: d + '-' + m + '.modinfo',
2662 input: module_ss.sources() + genh,
2663 capture: true,
2664 command: [modinfo_collect, module_ss.sources()])
2665 endif
2666 else
2667 if d == 'block'
2668 block_ss.add_all(module_ss)
2669 else
2670 softmmu_ss.add_all(module_ss)
2671 endif
2672 endif
2673 endforeach
2674 endforeach
2675
2676 foreach d, list : target_modules
2677 foreach m, module_ss : list
2678 if enable_modules and targetos != 'windows'
2679 foreach target : target_dirs
2680 if target.endswith('-softmmu')
2681 config_target = config_target_mak[target]
2682 config_target += config_host
2683 target_inc = [include_directories('target' / config_target['TARGET_BASE_ARCH'])]
2684 c_args = ['-DNEED_CPU_H',
2685 '-DCONFIG_TARGET="@0@-config-target.h"'.format(target),
2686 '-DCONFIG_DEVICES="@0@-config-devices.h"'.format(target)]
2687 target_module_ss = module_ss.apply(config_target, strict: false)
2688 if target_module_ss.sources() != []
2689 module_name = d + '-' + m + '-' + config_target['TARGET_NAME']
2690 sl = static_library(module_name,
2691 [genh, target_module_ss.sources()],
2692 dependencies: [modulecommon, target_module_ss.dependencies()],
2693 include_directories: target_inc,
2694 c_args: c_args,
2695 pic: true)
2696 softmmu_mods += sl
2697 # FIXME: Should use sl.extract_all_objects(recursive: true) too.
2698 modinfo_files += custom_target(module_name + '.modinfo',
2699 output: module_name + '.modinfo',
2700 input: target_module_ss.sources() + genh,
2701 capture: true,
2702 command: [modinfo_collect, '--target', target, target_module_ss.sources()])
2703 endif
2704 endif
2705 endforeach
2706 else
2707 specific_ss.add_all(module_ss)
2708 endif
2709 endforeach
2710 endforeach
2711
2712 if enable_modules
2713 modinfo_src = custom_target('modinfo.c',
2714 output: 'modinfo.c',
2715 input: modinfo_files,
2716 command: [modinfo_generate, '@INPUT@'],
2717 capture: true)
2718 modinfo_lib = static_library('modinfo', modinfo_src)
2719 modinfo_dep = declare_dependency(link_whole: modinfo_lib)
2720 softmmu_ss.add(modinfo_dep)
2721 endif
2722
2723 nm = find_program('nm')
2724 undefsym = find_program('scripts/undefsym.py')
2725 block_syms = custom_target('block.syms', output: 'block.syms',
2726 input: [libqemuutil, block_mods],
2727 capture: true,
2728 command: [undefsym, nm, '@INPUT@'])
2729 qemu_syms = custom_target('qemu.syms', output: 'qemu.syms',
2730 input: [libqemuutil, softmmu_mods],
2731 capture: true,
2732 command: [undefsym, nm, '@INPUT@'])
2733
2734 qom_ss = qom_ss.apply(config_host, strict: false)
2735 libqom = static_library('qom', qom_ss.sources() + genh,
2736 dependencies: [qom_ss.dependencies()],
2737 name_suffix: 'fa')
2738
2739 qom = declare_dependency(link_whole: libqom)
2740
2741 authz_ss = authz_ss.apply(config_host, strict: false)
2742 libauthz = static_library('authz', authz_ss.sources() + genh,
2743 dependencies: [authz_ss.dependencies()],
2744 name_suffix: 'fa',
2745 build_by_default: false)
2746
2747 authz = declare_dependency(link_whole: libauthz,
2748 dependencies: qom)
2749
2750 crypto_ss = crypto_ss.apply(config_host, strict: false)
2751 libcrypto = static_library('crypto', crypto_ss.sources() + genh,
2752 dependencies: [crypto_ss.dependencies()],
2753 name_suffix: 'fa',
2754 build_by_default: false)
2755
2756 crypto = declare_dependency(link_whole: libcrypto,
2757 dependencies: [authz, qom])
2758
2759 io_ss = io_ss.apply(config_host, strict: false)
2760 libio = static_library('io', io_ss.sources() + genh,
2761 dependencies: [io_ss.dependencies()],
2762 link_with: libqemuutil,
2763 name_suffix: 'fa',
2764 build_by_default: false)
2765
2766 io = declare_dependency(link_whole: libio, dependencies: [crypto, qom])
2767
2768 libmigration = static_library('migration', sources: migration_files + genh,
2769 name_suffix: 'fa',
2770 build_by_default: false)
2771 migration = declare_dependency(link_with: libmigration,
2772 dependencies: [zlib, qom, io])
2773 softmmu_ss.add(migration)
2774
2775 block_ss = block_ss.apply(config_host, strict: false)
2776 libblock = static_library('block', block_ss.sources() + genh,
2777 dependencies: block_ss.dependencies(),
2778 link_depends: block_syms,
2779 name_suffix: 'fa',
2780 build_by_default: false)
2781
2782 block = declare_dependency(link_whole: [libblock],
2783 link_args: '@block.syms',
2784 dependencies: [crypto, io])
2785
2786 blockdev_ss = blockdev_ss.apply(config_host, strict: false)
2787 libblockdev = static_library('blockdev', blockdev_ss.sources() + genh,
2788 dependencies: blockdev_ss.dependencies(),
2789 name_suffix: 'fa',
2790 build_by_default: false)
2791
2792 blockdev = declare_dependency(link_whole: [libblockdev],
2793 dependencies: [block])
2794
2795 qmp_ss = qmp_ss.apply(config_host, strict: false)
2796 libqmp = static_library('qmp', qmp_ss.sources() + genh,
2797 dependencies: qmp_ss.dependencies(),
2798 name_suffix: 'fa',
2799 build_by_default: false)
2800
2801 qmp = declare_dependency(link_whole: [libqmp])
2802
2803 libchardev = static_library('chardev', chardev_ss.sources() + genh,
2804 name_suffix: 'fa',
2805 dependencies: [gnutls],
2806 build_by_default: false)
2807
2808 chardev = declare_dependency(link_whole: libchardev)
2809
2810 hwcore_ss = hwcore_ss.apply(config_host, strict: false)
2811 libhwcore = static_library('hwcore', sources: hwcore_ss.sources() + genh,
2812 name_suffix: 'fa',
2813 build_by_default: false)
2814 hwcore = declare_dependency(link_whole: libhwcore)
2815 common_ss.add(hwcore)
2816
2817 ###########
2818 # Targets #
2819 ###########
2820
2821 foreach m : block_mods + softmmu_mods
2822 shared_module(m.name(),
2823 name_prefix: '',
2824 link_whole: m,
2825 install: true,
2826 install_dir: qemu_moddir)
2827 endforeach
2828
2829 softmmu_ss.add(authz, blockdev, chardev, crypto, io, qmp)
2830 common_ss.add(qom, qemuutil)
2831
2832 common_ss.add_all(when: 'CONFIG_SOFTMMU', if_true: [softmmu_ss])
2833 common_ss.add_all(when: 'CONFIG_USER_ONLY', if_true: user_ss)
2834
2835 common_all = common_ss.apply(config_all, strict: false)
2836 common_all = static_library('common',
2837 build_by_default: false,
2838 sources: common_all.sources() + genh,
2839 implicit_include_directories: false,
2840 dependencies: common_all.dependencies(),
2841 name_suffix: 'fa')
2842
2843 feature_to_c = find_program('scripts/feature_to_c.sh')
2844
2845 emulators = {}
2846 foreach target : target_dirs
2847 config_target = config_target_mak[target]
2848 target_name = config_target['TARGET_NAME']
2849 arch = config_target['TARGET_BASE_ARCH']
2850 arch_srcs = [config_target_h[target]]
2851 arch_deps = []
2852 c_args = ['-DNEED_CPU_H',
2853 '-DCONFIG_TARGET="@0@-config-target.h"'.format(target),
2854 '-DCONFIG_DEVICES="@0@-config-devices.h"'.format(target)]
2855 link_args = emulator_link_args
2856
2857 config_target += config_host
2858 target_inc = [include_directories('target' / config_target['TARGET_BASE_ARCH'])]
2859 if targetos == 'linux'
2860 target_inc += include_directories('linux-headers', is_system: true)
2861 endif
2862 if target.endswith('-softmmu')
2863 qemu_target_name = 'qemu-system-' + target_name
2864 target_type='system'
2865 t = target_softmmu_arch[arch].apply(config_target, strict: false)
2866 arch_srcs += t.sources()
2867 arch_deps += t.dependencies()
2868
2869 hw_dir = target_name == 'sparc64' ? 'sparc64' : arch
2870 hw = hw_arch[hw_dir].apply(config_target, strict: false)
2871 arch_srcs += hw.sources()
2872 arch_deps += hw.dependencies()
2873
2874 arch_srcs += config_devices_h[target]
2875 link_args += ['@block.syms', '@qemu.syms']
2876 else
2877 abi = config_target['TARGET_ABI_DIR']
2878 target_type='user'
2879 qemu_target_name = 'qemu-' + target_name
2880 if arch in target_user_arch
2881 t = target_user_arch[arch].apply(config_target, strict: false)
2882 arch_srcs += t.sources()
2883 arch_deps += t.dependencies()
2884 endif
2885 if 'CONFIG_LINUX_USER' in config_target
2886 base_dir = 'linux-user'
2887 target_inc += include_directories('linux-user/host/' / config_host['ARCH'])
2888 endif
2889 if 'CONFIG_BSD_USER' in config_target
2890 base_dir = 'bsd-user'
2891 target_inc += include_directories('bsd-user/' / targetos)
2892 dir = base_dir / abi
2893 arch_srcs += files(dir / 'target_arch_cpu.c')
2894 endif
2895 target_inc += include_directories(
2896 base_dir,
2897 base_dir / abi,
2898 )
2899 if 'CONFIG_LINUX_USER' in config_target
2900 dir = base_dir / abi
2901 arch_srcs += files(dir / 'signal.c', dir / 'cpu_loop.c')
2902 if config_target.has_key('TARGET_SYSTBL_ABI')
2903 arch_srcs += \
2904 syscall_nr_generators[abi].process(base_dir / abi / config_target['TARGET_SYSTBL'],
2905 extra_args : config_target['TARGET_SYSTBL_ABI'])
2906 endif
2907 endif
2908 endif
2909
2910 if 'TARGET_XML_FILES' in config_target
2911 gdbstub_xml = custom_target(target + '-gdbstub-xml.c',
2912 output: target + '-gdbstub-xml.c',
2913 input: files(config_target['TARGET_XML_FILES'].split()),
2914 command: [feature_to_c, '@INPUT@'],
2915 capture: true)
2916 arch_srcs += gdbstub_xml
2917 endif
2918
2919 t = target_arch[arch].apply(config_target, strict: false)
2920 arch_srcs += t.sources()
2921 arch_deps += t.dependencies()
2922
2923 target_common = common_ss.apply(config_target, strict: false)
2924 objects = common_all.extract_objects(target_common.sources())
2925 deps = target_common.dependencies()
2926
2927 target_specific = specific_ss.apply(config_target, strict: false)
2928 arch_srcs += target_specific.sources()
2929 arch_deps += target_specific.dependencies()
2930
2931 lib = static_library('qemu-' + target,
2932 sources: arch_srcs + genh,
2933 dependencies: arch_deps,
2934 objects: objects,
2935 include_directories: target_inc,
2936 c_args: c_args,
2937 build_by_default: false,
2938 name_suffix: 'fa')
2939
2940 if target.endswith('-softmmu')
2941 execs = [{
2942 'name': 'qemu-system-' + target_name,
2943 'win_subsystem': 'console',
2944 'sources': files('softmmu/main.c'),
2945 'dependencies': []
2946 }]
2947 if targetos == 'windows' and (sdl.found() or gtk.found())
2948 execs += [{
2949 'name': 'qemu-system-' + target_name + 'w',
2950 'win_subsystem': 'windows',
2951 'sources': files('softmmu/main.c'),
2952 'dependencies': []
2953 }]
2954 endif
2955 if get_option('fuzzing')
2956 specific_fuzz = specific_fuzz_ss.apply(config_target, strict: false)
2957 execs += [{
2958 'name': 'qemu-fuzz-' + target_name,
2959 'win_subsystem': 'console',
2960 'sources': specific_fuzz.sources(),
2961 'dependencies': specific_fuzz.dependencies(),
2962 }]
2963 endif
2964 else
2965 execs = [{
2966 'name': 'qemu-' + target_name,
2967 'win_subsystem': 'console',
2968 'sources': [],
2969 'dependencies': []
2970 }]
2971 endif
2972 foreach exe: execs
2973 exe_name = exe['name']
2974 if targetos == 'darwin'
2975 exe_name += '-unsigned'
2976 endif
2977
2978 emulator = executable(exe_name, exe['sources'],
2979 install: true,
2980 c_args: c_args,
2981 dependencies: arch_deps + deps + exe['dependencies'],
2982 objects: lib.extract_all_objects(recursive: true),
2983 link_language: link_language,
2984 link_depends: [block_syms, qemu_syms] + exe.get('link_depends', []),
2985 link_args: link_args,
2986 win_subsystem: exe['win_subsystem'])
2987
2988 if targetos == 'darwin'
2989 icon = 'pc-bios/qemu.rsrc'
2990 build_input = [emulator, files(icon)]
2991 install_input = [
2992 get_option('bindir') / exe_name,
2993 meson.current_source_dir() / icon
2994 ]
2995 if 'CONFIG_HVF' in config_target
2996 entitlements = 'accel/hvf/entitlements.plist'
2997 build_input += files(entitlements)
2998 install_input += meson.current_source_dir() / entitlements
2999 endif
3000
3001 emulators += {exe['name'] : custom_target(exe['name'],
3002 input: build_input,
3003 output: exe['name'],
3004 command: [
3005 files('scripts/entitlement.sh'),
3006 '@OUTPUT@',
3007 '@INPUT@'
3008 ])
3009 }
3010
3011 meson.add_install_script('scripts/entitlement.sh', '--install',
3012 get_option('bindir') / exe['name'],
3013 install_input)
3014 else
3015 emulators += {exe['name']: emulator}
3016 endif
3017
3018 if stap.found()
3019 foreach stp: [
3020 {'ext': '.stp-build', 'fmt': 'stap', 'bin': meson.current_build_dir() / exe['name'], 'install': false},
3021 {'ext': '.stp', 'fmt': 'stap', 'bin': get_option('prefix') / get_option('bindir') / exe['name'], 'install': true},
3022 {'ext': '-simpletrace.stp', 'fmt': 'simpletrace-stap', 'bin': '', 'install': true},
3023 {'ext': '-log.stp', 'fmt': 'log-stap', 'bin': '', 'install': true},
3024 ]
3025 custom_target(exe['name'] + stp['ext'],
3026 input: trace_events_all,
3027 output: exe['name'] + stp['ext'],
3028 install: stp['install'],
3029 install_dir: get_option('datadir') / 'systemtap/tapset',
3030 command: [
3031 tracetool, '--group=all', '--format=' + stp['fmt'],
3032 '--binary=' + stp['bin'],
3033 '--target-name=' + target_name,
3034 '--target-type=' + target_type,
3035 '--probe-prefix=qemu.' + target_type + '.' + target_name,
3036 '@INPUT@', '@OUTPUT@'
3037 ],
3038 depend_files: tracetool_depends)
3039 endforeach
3040 endif
3041 endforeach
3042 endforeach
3043
3044 # Other build targets
3045
3046 if 'CONFIG_PLUGIN' in config_host
3047 install_headers('include/qemu/qemu-plugin.h')
3048 endif
3049
3050 if 'CONFIG_GUEST_AGENT' in config_host
3051 subdir('qga')
3052 elif get_option('guest_agent_msi').enabled()
3053 error('Guest agent MSI requested, but the guest agent is not being built')
3054 endif
3055
3056 # Don't build qemu-keymap if xkbcommon is not explicitly enabled
3057 # when we don't build tools or system
3058 if xkbcommon.found()
3059 # used for the update-keymaps target, so include rules even if !have_tools
3060 qemu_keymap = executable('qemu-keymap', files('qemu-keymap.c', 'ui/input-keymap.c') + genh,
3061 dependencies: [qemuutil, xkbcommon], install: have_tools)
3062 endif
3063
3064 if have_tools
3065 qemu_img = executable('qemu-img', [files('qemu-img.c'), hxdep],
3066 dependencies: [authz, block, crypto, io, qom, qemuutil], install: true)
3067 qemu_io = executable('qemu-io', files('qemu-io.c'),
3068 dependencies: [block, qemuutil], install: true)
3069 qemu_nbd = executable('qemu-nbd', files('qemu-nbd.c'),
3070 dependencies: [blockdev, qemuutil, gnutls], install: true)
3071
3072 subdir('storage-daemon')
3073 subdir('contrib/rdmacm-mux')
3074 subdir('contrib/elf2dmp')
3075
3076 executable('qemu-edid', files('qemu-edid.c', 'hw/display/edid-generate.c'),
3077 dependencies: qemuutil,
3078 install: true)
3079
3080 if 'CONFIG_VHOST_USER' in config_host
3081 subdir('contrib/vhost-user-blk')
3082 subdir('contrib/vhost-user-gpu')
3083 subdir('contrib/vhost-user-input')
3084 subdir('contrib/vhost-user-scsi')
3085 endif
3086
3087 if targetos == 'linux'
3088 executable('qemu-bridge-helper', files('qemu-bridge-helper.c'),
3089 dependencies: [qemuutil, libcap_ng],
3090 install: true,
3091 install_dir: get_option('libexecdir'))
3092
3093 executable('qemu-pr-helper', files('scsi/qemu-pr-helper.c', 'scsi/utils.c'),
3094 dependencies: [authz, crypto, io, qom, qemuutil,
3095 libcap_ng, mpathpersist],
3096 install: true)
3097 endif
3098
3099 if have_ivshmem
3100 subdir('contrib/ivshmem-client')
3101 subdir('contrib/ivshmem-server')
3102 endif
3103 endif
3104
3105 subdir('scripts')
3106 subdir('tools')
3107 subdir('pc-bios')
3108 subdir('docs')
3109 subdir('tests')
3110 if gtk.found()
3111 subdir('po')
3112 endif
3113
3114 if host_machine.system() == 'windows'
3115 nsis_cmd = [
3116 find_program('scripts/nsis.py'),
3117 '@OUTPUT@',
3118 get_option('prefix'),
3119 meson.current_source_dir(),
3120 host_machine.cpu(),
3121 '--',
3122 '-DDISPLAYVERSION=' + meson.project_version(),
3123 ]
3124 if build_docs
3125 nsis_cmd += '-DCONFIG_DOCUMENTATION=y'
3126 endif
3127 if gtk.found()
3128 nsis_cmd += '-DCONFIG_GTK=y'
3129 endif
3130
3131 nsis = custom_target('nsis',
3132 output: 'qemu-setup-' + meson.project_version() + '.exe',
3133 input: files('qemu.nsi'),
3134 build_always_stale: true,
3135 command: nsis_cmd + ['@INPUT@'])
3136 alias_target('installer', nsis)
3137 endif
3138
3139 #########################
3140 # Configuration summary #
3141 #########################
3142
3143 # Directories
3144 summary_info = {}
3145 summary_info += {'Install prefix': get_option('prefix')}
3146 summary_info += {'BIOS directory': qemu_datadir}
3147 summary_info += {'firmware path': get_option('qemu_firmwarepath')}
3148 summary_info += {'binary directory': get_option('bindir')}
3149 summary_info += {'library directory': get_option('libdir')}
3150 summary_info += {'module directory': qemu_moddir}
3151 summary_info += {'libexec directory': get_option('libexecdir')}
3152 summary_info += {'include directory': get_option('includedir')}
3153 summary_info += {'config directory': get_option('sysconfdir')}
3154 if targetos != 'windows'
3155 summary_info += {'local state directory': get_option('localstatedir')}
3156 summary_info += {'Manual directory': get_option('mandir')}
3157 else
3158 summary_info += {'local state directory': 'queried at runtime'}
3159 endif
3160 summary_info += {'Doc directory': get_option('docdir')}
3161 summary_info += {'Build directory': meson.current_build_dir()}
3162 summary_info += {'Source path': meson.current_source_dir()}
3163 summary_info += {'GIT submodules': config_host['GIT_SUBMODULES']}
3164 summary(summary_info, bool_yn: true, section: 'Directories')
3165
3166 # Host binaries
3167 summary_info = {}
3168 summary_info += {'git': config_host['GIT']}
3169 summary_info += {'make': config_host['MAKE']}
3170 summary_info += {'python': '@0@ (version: @1@)'.format(python.full_path(), python.language_version())}
3171 summary_info += {'sphinx-build': sphinx_build}
3172 if config_host.has_key('HAVE_GDB_BIN')
3173 summary_info += {'gdb': config_host['HAVE_GDB_BIN']}
3174 endif
3175 summary_info += {'genisoimage': config_host['GENISOIMAGE']}
3176 if targetos == 'windows' and config_host.has_key('CONFIG_GUEST_AGENT')
3177 summary_info += {'wixl': wixl}
3178 endif
3179 if slirp_opt != 'disabled' and 'CONFIG_SLIRP_SMBD' in config_host
3180 summary_info += {'smbd': config_host['CONFIG_SMBD_COMMAND']}
3181 endif
3182 summary(summary_info, bool_yn: true, section: 'Host binaries')
3183
3184 # Configurable features
3185 summary_info = {}
3186 summary_info += {'Documentation': build_docs}
3187 summary_info += {'system-mode emulation': have_system}
3188 summary_info += {'user-mode emulation': have_user}
3189 summary_info += {'block layer': have_block}
3190 summary_info += {'Install blobs': get_option('install_blobs')}
3191 summary_info += {'module support': config_host.has_key('CONFIG_MODULES')}
3192 if config_host.has_key('CONFIG_MODULES')
3193 summary_info += {'alternative module path': config_host.has_key('CONFIG_MODULE_UPGRADES')}
3194 endif
3195 summary_info += {'fuzzing support': get_option('fuzzing')}
3196 if have_system
3197 summary_info += {'Audio drivers': ' '.join(audio_drivers_selected)}
3198 endif
3199 summary_info += {'Trace backends': ','.join(get_option('trace_backends'))}
3200 if 'simple' in get_option('trace_backends')
3201 summary_info += {'Trace output file': get_option('trace_file') + '-<pid>'}
3202 endif
3203 summary_info += {'QOM debugging': config_host.has_key('CONFIG_QOM_CAST_DEBUG')}
3204 summary_info += {'vhost-kernel support': config_host.has_key('CONFIG_VHOST_KERNEL')}
3205 summary_info += {'vhost-net support': config_host.has_key('CONFIG_VHOST_NET')}
3206 summary_info += {'vhost-crypto support': config_host.has_key('CONFIG_VHOST_CRYPTO')}
3207 summary_info += {'vhost-scsi support': config_host.has_key('CONFIG_VHOST_SCSI')}
3208 summary_info += {'vhost-vsock support': config_host.has_key('CONFIG_VHOST_VSOCK')}
3209 summary_info += {'vhost-user support': config_host.has_key('CONFIG_VHOST_USER')}
3210 summary_info += {'vhost-user-blk server support': have_vhost_user_blk_server}
3211 summary_info += {'vhost-user-fs support': config_host.has_key('CONFIG_VHOST_USER_FS')}
3212 summary_info += {'vhost-vdpa support': config_host.has_key('CONFIG_VHOST_VDPA')}
3213 summary_info += {'build guest agent': config_host.has_key('CONFIG_GUEST_AGENT')}
3214 summary(summary_info, bool_yn: true, section: 'Configurable features')
3215
3216 # Compilation information
3217 summary_info = {}
3218 summary_info += {'host CPU': cpu}
3219 summary_info += {'host endianness': build_machine.endian()}
3220 summary_info += {'C compiler': ' '.join(meson.get_compiler('c').cmd_array())}
3221 summary_info += {'Host C compiler': ' '.join(meson.get_compiler('c', native: true).cmd_array())}
3222 if link_language == 'cpp'
3223 summary_info += {'C++ compiler': ' '.join(meson.get_compiler('cpp').cmd_array())}
3224 else
3225 summary_info += {'C++ compiler': false}
3226 endif
3227 if targetos == 'darwin'
3228 summary_info += {'Objective-C compiler': ' '.join(meson.get_compiler('objc').cmd_array())}
3229 endif
3230 if targetos == 'windows'
3231 if 'WIN_SDK' in config_host
3232 summary_info += {'Windows SDK': config_host['WIN_SDK']}
3233 endif
3234 endif
3235 summary_info += {'CFLAGS': ' '.join(get_option('c_args')
3236 + ['-O' + get_option('optimization')]
3237 + (get_option('debug') ? ['-g'] : []))}
3238 if link_language == 'cpp'
3239 summary_info += {'CXXFLAGS': ' '.join(get_option('cpp_args')
3240 + ['-O' + get_option('optimization')]
3241 + (get_option('debug') ? ['-g'] : []))}
3242 endif
3243 link_args = get_option(link_language + '_link_args')
3244 if link_args.length() > 0
3245 summary_info += {'LDFLAGS': ' '.join(link_args)}
3246 endif
3247 summary_info += {'QEMU_CFLAGS': config_host['QEMU_CFLAGS']}
3248 summary_info += {'QEMU_LDFLAGS': config_host['QEMU_LDFLAGS']}
3249 summary_info += {'profiler': config_host.has_key('CONFIG_PROFILER')}
3250 summary_info += {'link-time optimization (LTO)': get_option('b_lto')}
3251 summary_info += {'PIE': get_option('b_pie')}
3252 summary_info += {'static build': config_host.has_key('CONFIG_STATIC')}
3253 summary_info += {'malloc trim support': has_malloc_trim}
3254 summary_info += {'membarrier': config_host.has_key('CONFIG_MEMBARRIER')}
3255 summary_info += {'debug stack usage': config_host.has_key('CONFIG_DEBUG_STACK_USAGE')}
3256 summary_info += {'mutex debugging': config_host.has_key('CONFIG_DEBUG_MUTEX')}
3257 summary_info += {'memory allocator': get_option('malloc')}
3258 summary_info += {'avx2 optimization': config_host.has_key('CONFIG_AVX2_OPT')}
3259 summary_info += {'avx512f optimization': config_host.has_key('CONFIG_AVX512F_OPT')}
3260 summary_info += {'gprof enabled': config_host.has_key('CONFIG_GPROF')}
3261 summary_info += {'gcov': get_option('b_coverage')}
3262 summary_info += {'thread sanitizer': config_host.has_key('CONFIG_TSAN')}
3263 summary_info += {'CFI support': get_option('cfi')}
3264 if get_option('cfi')
3265 summary_info += {'CFI debug support': get_option('cfi_debug')}
3266 endif
3267 summary_info += {'strip binaries': get_option('strip')}
3268 summary_info += {'sparse': sparse}
3269 summary_info += {'mingw32 support': targetos == 'windows'}
3270
3271 # snarf the cross-compilation information for tests
3272 foreach target: target_dirs
3273 tcg_mak = meson.current_build_dir() / 'tests/tcg' / 'config-' + target + '.mak'
3274 if fs.exists(tcg_mak)
3275 config_cross_tcg = keyval.load(tcg_mak)
3276 target = config_cross_tcg['TARGET_NAME']
3277 compiler = ''
3278 if 'DOCKER_CROSS_CC_GUEST' in config_cross_tcg
3279 summary_info += {target + ' tests': config_cross_tcg['DOCKER_CROSS_CC_GUEST'] +
3280 ' via ' + config_cross_tcg['DOCKER_IMAGE']}
3281 elif 'CROSS_CC_GUEST' in config_cross_tcg
3282 summary_info += {target + ' tests'
3283 : config_cross_tcg['CROSS_CC_GUEST'] }
3284 endif
3285 endif
3286 endforeach
3287
3288 summary(summary_info, bool_yn: true, section: 'Compilation')
3289
3290 # Targets and accelerators
3291 summary_info = {}
3292 if have_system
3293 summary_info += {'KVM support': config_all.has_key('CONFIG_KVM')}
3294 summary_info += {'HAX support': config_all.has_key('CONFIG_HAX')}
3295 summary_info += {'HVF support': config_all.has_key('CONFIG_HVF')}
3296 summary_info += {'WHPX support': config_all.has_key('CONFIG_WHPX')}
3297 summary_info += {'NVMM support': config_all.has_key('CONFIG_NVMM')}
3298 summary_info += {'Xen support': config_host.has_key('CONFIG_XEN_BACKEND')}
3299 if config_host.has_key('CONFIG_XEN_BACKEND')
3300 summary_info += {'xen ctrl version': config_host['CONFIG_XEN_CTRL_INTERFACE_VERSION']}
3301 endif
3302 endif
3303 summary_info += {'TCG support': config_all.has_key('CONFIG_TCG')}
3304 if config_all.has_key('CONFIG_TCG')
3305 if get_option('tcg_interpreter')
3306 summary_info += {'TCG backend': 'TCI (TCG with bytecode interpreter, experimental and slow)'}
3307 else
3308 summary_info += {'TCG backend': 'native (@0@)'.format(cpu)}
3309 endif
3310 summary_info += {'TCG plugins': config_host.has_key('CONFIG_PLUGIN')}
3311 summary_info += {'TCG debug enabled': config_host.has_key('CONFIG_DEBUG_TCG')}
3312 endif
3313 summary_info += {'target list': ' '.join(target_dirs)}
3314 if have_system
3315 summary_info += {'default devices': get_option('default_devices')}
3316 summary_info += {'out of process emulation': multiprocess_allowed}
3317 endif
3318 summary(summary_info, bool_yn: true, section: 'Targets and accelerators')
3319
3320 # Block layer
3321 summary_info = {}
3322 summary_info += {'coroutine backend': config_host['CONFIG_COROUTINE_BACKEND']}
3323 summary_info += {'coroutine pool': config_host['CONFIG_COROUTINE_POOL'] == '1'}
3324 if have_block
3325 summary_info += {'Block whitelist (rw)': config_host['CONFIG_BDRV_RW_WHITELIST']}
3326 summary_info += {'Block whitelist (ro)': config_host['CONFIG_BDRV_RO_WHITELIST']}
3327 summary_info += {'Use block whitelist in tools': config_host.has_key('CONFIG_BDRV_WHITELIST_TOOLS')}
3328 summary_info += {'VirtFS support': have_virtfs}
3329 summary_info += {'build virtiofs daemon': have_virtiofsd}
3330 summary_info += {'Live block migration': config_host.has_key('CONFIG_LIVE_BLOCK_MIGRATION')}
3331 summary_info += {'replication support': config_host.has_key('CONFIG_REPLICATION')}
3332 summary_info += {'bochs support': config_host.has_key('CONFIG_BOCHS')}
3333 summary_info += {'cloop support': config_host.has_key('CONFIG_CLOOP')}
3334 summary_info += {'dmg support': config_host.has_key('CONFIG_DMG')}
3335 summary_info += {'qcow v1 support': config_host.has_key('CONFIG_QCOW1')}
3336 summary_info += {'vdi support': config_host.has_key('CONFIG_VDI')}
3337 summary_info += {'vvfat support': config_host.has_key('CONFIG_VVFAT')}
3338 summary_info += {'qed support': config_host.has_key('CONFIG_QED')}
3339 summary_info += {'parallels support': config_host.has_key('CONFIG_PARALLELS')}
3340 summary_info += {'FUSE exports': fuse}
3341 endif
3342 summary(summary_info, bool_yn: true, section: 'Block layer support')
3343
3344 # Crypto
3345 summary_info = {}
3346 summary_info += {'TLS priority': config_host['CONFIG_TLS_PRIORITY']}
3347 summary_info += {'GNUTLS support': gnutls}
3348 if gnutls.found()
3349 summary_info += {' GNUTLS crypto': gnutls_crypto.found()}
3350 endif
3351 summary_info += {'libgcrypt': gcrypt}
3352 summary_info += {'nettle': nettle}
3353 if nettle.found()
3354 summary_info += {' XTS': xts != 'private'}
3355 endif
3356 summary_info += {'crypto afalg': config_host.has_key('CONFIG_AF_ALG')}
3357 summary_info += {'rng-none': config_host.has_key('CONFIG_RNG_NONE')}
3358 summary_info += {'Linux keyring': config_host.has_key('CONFIG_SECRET_KEYRING')}
3359 summary(summary_info, bool_yn: true, section: 'Crypto')
3360
3361 # Libraries
3362 summary_info = {}
3363 if targetos == 'darwin'
3364 summary_info += {'Cocoa support': cocoa}
3365 endif
3366 summary_info += {'SDL support': sdl}
3367 summary_info += {'SDL image support': sdl_image}
3368 summary_info += {'GTK support': gtk}
3369 summary_info += {'pixman': pixman}
3370 summary_info += {'VTE support': vte}
3371 summary_info += {'slirp support': slirp_opt == 'internal' ? slirp_opt : slirp}
3372 summary_info += {'libtasn1': tasn1}
3373 summary_info += {'PAM': pam}
3374 summary_info += {'iconv support': iconv}
3375 summary_info += {'curses support': curses}
3376 summary_info += {'virgl support': virgl}
3377 summary_info += {'curl support': curl}
3378 summary_info += {'Multipath support': mpathpersist}
3379 summary_info += {'VNC support': vnc}
3380 if vnc.found()
3381 summary_info += {'VNC SASL support': sasl}
3382 summary_info += {'VNC JPEG support': jpeg}
3383 summary_info += {'VNC PNG support': png}
3384 endif
3385 if targetos not in ['darwin', 'haiku', 'windows']
3386 summary_info += {'OSS support': oss}
3387 elif targetos == 'darwin'
3388 summary_info += {'CoreAudio support': coreaudio}
3389 elif targetos == 'windows'
3390 summary_info += {'DirectSound support': dsound}
3391 endif
3392 if targetos == 'linux'
3393 summary_info += {'ALSA support': alsa}
3394 summary_info += {'PulseAudio support': pulse}
3395 endif
3396 summary_info += {'JACK support': jack}
3397 summary_info += {'brlapi support': brlapi}
3398 summary_info += {'vde support': vde}
3399 summary_info += {'netmap support': have_netmap}
3400 summary_info += {'Linux AIO support': libaio}
3401 summary_info += {'Linux io_uring support': linux_io_uring}
3402 summary_info += {'ATTR/XATTR support': libattr}
3403 summary_info += {'RDMA support': config_host.has_key('CONFIG_RDMA')}
3404 summary_info += {'PVRDMA support': config_host.has_key('CONFIG_PVRDMA')}
3405 summary_info += {'fdt support': fdt_opt == 'disabled' ? false : fdt_opt}
3406 summary_info += {'libcap-ng support': libcap_ng}
3407 summary_info += {'bpf support': libbpf}
3408 summary_info += {'spice protocol support': spice_protocol}
3409 if spice_protocol.found()
3410 summary_info += {' spice server support': spice}
3411 endif
3412 summary_info += {'rbd support': rbd}
3413 summary_info += {'xfsctl support': config_host.has_key('CONFIG_XFS')}
3414 summary_info += {'smartcard support': cacard}
3415 summary_info += {'U2F support': u2f}
3416 summary_info += {'libusb': libusb}
3417 summary_info += {'usb net redir': usbredir}
3418 summary_info += {'OpenGL support': config_host.has_key('CONFIG_OPENGL')}
3419 summary_info += {'GBM': gbm}
3420 summary_info += {'libiscsi support': libiscsi}
3421 summary_info += {'libnfs support': libnfs}
3422 if targetos == 'windows'
3423 if config_host.has_key('CONFIG_GUEST_AGENT')
3424 summary_info += {'QGA VSS support': config_host.has_key('CONFIG_QGA_VSS')}
3425 summary_info += {'QGA w32 disk info': config_host.has_key('CONFIG_QGA_NTDDSCSI')}
3426 endif
3427 endif
3428 summary_info += {'seccomp support': seccomp}
3429 summary_info += {'GlusterFS support': glusterfs}
3430 summary_info += {'TPM support': config_host.has_key('CONFIG_TPM')}
3431 summary_info += {'libssh support': config_host.has_key('CONFIG_LIBSSH')}
3432 summary_info += {'lzo support': lzo}
3433 summary_info += {'snappy support': snappy}
3434 summary_info += {'bzip2 support': libbzip2}
3435 summary_info += {'lzfse support': liblzfse}
3436 summary_info += {'zstd support': zstd}
3437 summary_info += {'NUMA host support': config_host.has_key('CONFIG_NUMA')}
3438 summary_info += {'libxml2': libxml2}
3439 summary_info += {'capstone': capstone_opt == 'internal' ? capstone_opt : capstone}
3440 summary_info += {'libpmem support': libpmem}
3441 summary_info += {'libdaxctl support': libdaxctl}
3442 summary_info += {'libudev': libudev}
3443 # Dummy dependency, keep .found()
3444 summary_info += {'FUSE lseek': fuse_lseek.found()}
3445 summary(summary_info, bool_yn: true, section: 'Dependencies')
3446
3447 if not supported_cpus.contains(cpu)
3448 message()
3449 warning('SUPPORT FOR THIS HOST CPU WILL GO AWAY IN FUTURE RELEASES!')
3450 message()
3451 message('CPU host architecture ' + cpu + ' support is not currently maintained.')
3452 message('The QEMU project intends to remove support for this host CPU in')
3453 message('a future release if nobody volunteers to maintain it and to')
3454 message('provide a build host for our continuous integration setup.')
3455 message('configure has succeeded and you can continue to build, but')
3456 message('if you care about QEMU on this platform you should contact')
3457 message('us upstream at qemu-devel@nongnu.org.')
3458 endif
3459
3460 if not supported_oses.contains(targetos)
3461 message()
3462 warning('WARNING: SUPPORT FOR THIS HOST OS WILL GO AWAY IN FUTURE RELEASES!')
3463 message()
3464 message('Host OS ' + targetos + 'support is not currently maintained.')
3465 message('The QEMU project intends to remove support for this host OS in')
3466 message('a future release if nobody volunteers to maintain it and to')
3467 message('provide a build host for our continuous integration setup.')
3468 message('configure has succeeded and you can continue to build, but')
3469 message('if you care about QEMU on this platform you should contact')
3470 message('us upstream at qemu-devel@nongnu.org.')
3471 endif