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