123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 |
- # SPDX-License-Identifier: Apache-2.0
- if("${ARCH}" STREQUAL "x86")
- set_ifndef(QEMU_binary_suffix i386)
- elseif(DEFINED QEMU_ARCH)
- set_ifndef(QEMU_binary_suffix ${QEMU_ARCH})
- else()
- set_ifndef(QEMU_binary_suffix ${ARCH})
- endif()
- set(qemu_alternate_path $ENV{QEMU_BIN_PATH})
- if(qemu_alternate_path)
- find_program(
- QEMU
- PATHS ${qemu_alternate_path}
- NO_DEFAULT_PATH
- NAMES qemu-system-${QEMU_binary_suffix}
- )
- else()
- find_program(
- QEMU
- qemu-system-${QEMU_binary_suffix}
- )
- endif()
- # We need to set up uefi-run and OVMF environment
- # for testing UEFI method on qemu platforms
- if(CONFIG_QEMU_UEFI_BOOT)
- find_program(UEFI NAMES uefi-run REQUIRED)
- if(DEFINED ENV{OVMF_FD_PATH})
- set(OVMF_FD_PATH $ENV{OVMF_FD_PATH})
- else()
- message(FATAL_ERROR "Couldn't find an valid OVMF_FD_PATH.")
- endif()
- list(APPEND UEFI -b ${OVMF_FD_PATH} -q ${QEMU})
- set(QEMU ${UEFI})
- endif()
- set(qemu_targets
- run
- debugserver
- )
- set(QEMU_FLAGS -pidfile)
- if(${CMAKE_GENERATOR} STREQUAL "Unix Makefiles")
- list(APPEND QEMU_FLAGS qemu\${QEMU_INSTANCE}.pid)
- else()
- list(APPEND QEMU_FLAGS qemu${QEMU_INSTANCE}.pid)
- endif()
- # Set up chardev for console.
- if(QEMU_PTY)
- # Redirect console to a pseudo-tty, used for running automated tests.
- list(APPEND QEMU_FLAGS -chardev pty,id=con,mux=on)
- elseif(QEMU_PIPE)
- # Redirect console to a pipe, used for running automated tests.
- list(APPEND QEMU_FLAGS -chardev pipe,id=con,mux=on,path=${QEMU_PIPE})
- else()
- # Redirect console to stdio, used for manual debugging.
- list(APPEND QEMU_FLAGS -chardev stdio,id=con,mux=on)
- endif()
- # Connect main serial port to the console chardev.
- list(APPEND QEMU_FLAGS -serial chardev:con)
- # Connect semihosting console to the console chardev if configured.
- if(CONFIG_SEMIHOST_CONSOLE)
- list(APPEND QEMU_FLAGS
- -semihosting-config enable=on,target=auto,chardev=con
- )
- endif()
- # Connect monitor to the console chardev.
- list(APPEND QEMU_FLAGS -mon chardev=con,mode=readline)
- if(CONFIG_QEMU_ICOUNT)
- list(APPEND QEMU_FLAGS
- -icount shift=${CONFIG_QEMU_ICOUNT_SHIFT},align=off,sleep=off
- -rtc clock=vm)
- endif()
- # Add a BT serial device when building for bluetooth, unless the
- # application explicitly opts out with NO_QEMU_SERIAL_BT_SERVER.
- if(CONFIG_BT)
- if(CONFIG_BT_NO_DRIVER)
- set(NO_QEMU_SERIAL_BT_SERVER 1)
- endif()
- if(NOT NO_QEMU_SERIAL_BT_SERVER)
- list(APPEND QEMU_FLAGS -serial unix:/tmp/bt-server-bredr)
- endif()
- endif()
- # If we are running a networking application in QEMU, then set proper
- # QEMU variables. This also allows two QEMUs to be hooked together and
- # pass data between them. The QEMU flags are not set for standalone
- # tests defined by CONFIG_NET_TEST. For PPP, the serial port file is
- # not available if we run unit tests which define CONFIG_NET_TEST.
- if(CONFIG_NETWORKING)
- if(CONFIG_NET_QEMU_SLIP)
- if((CONFIG_NET_SLIP_TAP) OR (CONFIG_IEEE802154_UPIPE))
- set(QEMU_NET_STACK 1)
- endif()
- elseif((CONFIG_NET_QEMU_PPP) AND NOT (CONFIG_NET_TEST))
- set(QEMU_NET_STACK 1)
- endif()
- endif()
- # TO create independent pipes for each QEMU application set QEMU_PIPE_STACK
- if(QEMU_PIPE_STACK)
- list(APPEND qemu_targets
- node
- )
- if(NOT QEMU_PIPE_ID)
- set(QEMU_PIPE_ID 1)
- endif()
- list(APPEND QEMU_FLAGS
- -serial none
- )
- list(APPEND MORE_FLAGS_FOR_node
- -serial pipe:/tmp/hub/ip-stack-node${QEMU_PIPE_ID}
- -pidfile qemu-node${QEMU_PIPE_ID}.pid
- )
- set(PIPE_NODE_IN /tmp/hub/ip-stack-node${QEMU_PIPE_ID}.in)
- set(PIPE_NODE_OUT /tmp/hub/ip-stack-node${QEMU_PIPE_ID}.out)
- set(pipes
- ${PIPE_NODE_IN}
- ${PIPE_NODE_OUT}
- )
- set(destroy_pipe_commands
- COMMAND ${CMAKE_COMMAND} -E remove -f ${pipes}
- )
- set(create_pipe_commands
- COMMAND ${CMAKE_COMMAND} -E make_directory /tmp/hub
- COMMAND mkfifo ${PIPE_NODE_IN}
- COMMAND mkfifo ${PIPE_NODE_OUT}
- )
- set(PRE_QEMU_COMMANDS_FOR_node
- ${destroy_pipe_commands}
- ${create_pipe_commands}
- )
- elseif(QEMU_NET_STACK)
- list(APPEND qemu_targets
- client
- server
- )
- foreach(target ${qemu_targets})
- if((${target} STREQUAL client) OR (${target} STREQUAL server))
- list(APPEND MORE_FLAGS_FOR_${target}
- -serial pipe:/tmp/ip-stack-${target}
- -pidfile qemu-${target}.pid
- )
- else()
- # QEMU_INSTANCE is a command line argument to *make* (not cmake). By
- # appending the instance name to the pid file we can easily run more
- # instances of the same sample.
- if(CONFIG_NET_QEMU_PPP)
- if(${CMAKE_GENERATOR} STREQUAL "Unix Makefiles")
- set(ppp_path unix:/tmp/ppp\${QEMU_INSTANCE})
- else()
- set(ppp_path unix:/tmp/ppp${QEMU_INSTANCE})
- endif()
- list(APPEND MORE_FLAGS_FOR_${target}
- -serial ${ppp_path}
- )
- else()
- if(${CMAKE_GENERATOR} STREQUAL "Unix Makefiles")
- set(tmp_file unix:/tmp/slip.sock\${QEMU_INSTANCE})
- else()
- set(tmp_file unix:/tmp/slip.sock${QEMU_INSTANCE})
- endif()
- list(APPEND MORE_FLAGS_FOR_${target}
- -serial ${tmp_file}
- )
- endif()
- endif()
- endforeach()
- set(PIPE_SERVER_IN /tmp/ip-stack-server.in)
- set(PIPE_SERVER_OUT /tmp/ip-stack-server.out)
- set(PIPE_CLIENT_IN /tmp/ip-stack-client.in)
- set(PIPE_CLIENT_OUT /tmp/ip-stack-client.out)
- set(pipes
- ${PIPE_SERVER_IN}
- ${PIPE_SERVER_OUT}
- ${PIPE_CLIENT_IN}
- ${PIPE_CLIENT_OUT}
- )
- set(destroy_pipe_commands
- COMMAND ${CMAKE_COMMAND} -E remove -f ${pipes}
- )
- # TODO: Port to Windows. Perhaps using python? Or removing the
- # need for mkfifo and create_symlink somehow.
- set(create_pipe_commands
- COMMAND mkfifo ${PIPE_SERVER_IN}
- COMMAND mkfifo ${PIPE_SERVER_OUT}
- )
- if(PCAP)
- list(APPEND create_pipe_commands
- COMMAND mkfifo ${PIPE_CLIENT_IN}
- COMMAND mkfifo ${PIPE_CLIENT_OUT}
- )
- else()
- list(APPEND create_pipe_commands
- COMMAND ${CMAKE_COMMAND} -E create_symlink ${PIPE_SERVER_IN} ${PIPE_CLIENT_OUT}
- COMMAND ${CMAKE_COMMAND} -E create_symlink ${PIPE_SERVER_OUT} ${PIPE_CLIENT_IN}
- )
- endif()
- set(PRE_QEMU_COMMANDS_FOR_server
- ${destroy_pipe_commands}
- ${create_pipe_commands}
- )
- if(PCAP)
- # Start a monitor application to capture traffic
- #
- # Assumes;
- # PCAP has been set to the file where traffic should be captured
- # NET_TOOLS has been set to the net-tools repo path
- # net-tools/monitor_15_4 has been built beforehand
- set_ifndef(NET_TOOLS ${ZEPHYR_BASE}/../net-tools) # Default if not set
- list(APPEND PRE_QEMU_COMMANDS_FOR_server
- COMMAND
- #This command is run in the background using '&'. This prevents
- #chaining other commands with '&&'. The command is enclosed in '{}'
- #to fix this.
- {
- ${NET_TOOLS}/monitor_15_4
- ${PCAP}
- /tmp/ip-stack-server
- /tmp/ip-stack-client
- > /dev/null &
- }
- # TODO: Support cleanup of the monitor_15_4 process
- )
- endif()
- endif(QEMU_PIPE_STACK)
- if(CONFIG_X86_64 AND NOT CONFIG_QEMU_UEFI_BOOT)
- # QEMU doesn't like 64-bit ELF files. Since we don't use any >4GB
- # addresses, converting it to 32-bit is safe enough for emulation.
- add_custom_target(qemu_image_target
- COMMAND
- ${CMAKE_OBJCOPY}
- -O elf32-i386
- $<TARGET_FILE:${logical_target_for_zephyr_elf}>
- ${ZEPHYR_BINARY_DIR}/zephyr-qemu.elf
- DEPENDS ${logical_target_for_zephyr_elf}
- )
- # Split the 'locore' and 'main' memory regions into separate executable
- # images and specify the 'locore' as the boot kernel, in order to prevent
- # the QEMU direct multiboot kernel loader from overwriting the BIOS and
- # option ROM areas located in between the two memory regions.
- # (for more details, refer to the issue zephyrproject-rtos/sdk-ng#168)
- add_custom_target(qemu_locore_image_target
- COMMAND
- ${CMAKE_OBJCOPY}
- -j .locore
- ${ZEPHYR_BINARY_DIR}/zephyr-qemu.elf
- ${ZEPHYR_BINARY_DIR}/zephyr-qemu-locore.elf
- 2>&1 | grep -iv \"empty loadable segment detected\" || true
- DEPENDS qemu_image_target
- )
- add_custom_target(qemu_main_image_target
- COMMAND
- ${CMAKE_OBJCOPY}
- -R .locore
- ${ZEPHYR_BINARY_DIR}/zephyr-qemu.elf
- ${ZEPHYR_BINARY_DIR}/zephyr-qemu-main.elf
- 2>&1 | grep -iv \"empty loadable segment detected\" || true
- DEPENDS qemu_image_target
- )
- add_custom_target(
- qemu_kernel_target
- DEPENDS qemu_locore_image_target qemu_main_image_target
- )
- set(QEMU_KERNEL_FILE "${ZEPHYR_BINARY_DIR}/zephyr-qemu-locore.elf")
- list(APPEND QEMU_EXTRA_FLAGS
- "-device;loader,file=${ZEPHYR_BINARY_DIR}/zephyr-qemu-main.elf"
- )
- endif()
- if(CONFIG_IVSHMEM)
- if(CONFIG_IVSHMEM_DOORBELL)
- list(APPEND QEMU_FLAGS
- -device ivshmem-doorbell,vectors=${CONFIG_IVSHMEM_MSI_X_VECTORS},chardev=ivshmem
- -chardev socket,path=/tmp/ivshmem_socket,id=ivshmem
- )
- else()
- list(APPEND QEMU_FLAGS
- -device ivshmem-plain,memdev=hostmem
- -object memory-backend-file,size=${CONFIG_QEMU_IVSHMEM_PLAIN_MEM_SIZE}M,share,mem-path=/dev/shm/ivshmem,id=hostmem
- )
- endif()
- endif()
- if(NOT QEMU_PIPE)
- set(QEMU_PIPE_COMMENT "\nTo exit from QEMU enter: 'CTRL+a, x'\n")
- endif()
- # Don't just test CONFIG_SMP, there is at least one test of the lower
- # level multiprocessor API that wants an auxiliary CPU but doesn't
- # want SMP using it.
- if(NOT CONFIG_MP_NUM_CPUS MATCHES "1")
- list(APPEND QEMU_SMP_FLAGS -smp cpus=${CONFIG_MP_NUM_CPUS})
- endif()
- # Use flags passed in from the environment
- set(env_qemu $ENV{QEMU_EXTRA_FLAGS})
- separate_arguments(env_qemu)
- list(APPEND QEMU_EXTRA_FLAGS ${env_qemu})
- list(APPEND MORE_FLAGS_FOR_debugserver -s -S)
- # Architectures can define QEMU_KERNEL_FILE to use a specific output
- # file to pass to qemu (and a "qemu_kernel_target" target to generate
- # it), or set QEMU_KERNEL_OPTION if they want to replace the "-kernel
- # ..." option entirely.
- if(CONFIG_QEMU_UEFI_BOOT)
- set(QEMU_UEFI_OPTION ${PROJECT_BINARY_DIR}/${CONFIG_KERNEL_BIN_NAME}.efi)
- list(APPEND QEMU_UEFI_OPTION --)
- elseif(DEFINED QEMU_KERNEL_FILE)
- set(QEMU_KERNEL_OPTION "-kernel;${QEMU_KERNEL_FILE}")
- elseif(NOT DEFINED QEMU_KERNEL_OPTION)
- set(QEMU_KERNEL_OPTION "-kernel;$<TARGET_FILE:${logical_target_for_zephyr_elf}>")
- elseif(DEFINED QEMU_KERNEL_OPTION)
- string(CONFIGURE "${QEMU_KERNEL_OPTION}" QEMU_KERNEL_OPTION)
- endif()
- foreach(target ${qemu_targets})
- add_custom_target(${target}
- ${PRE_QEMU_COMMANDS}
- ${PRE_QEMU_COMMANDS_FOR_${target}}
- COMMAND
- ${QEMU}
- ${QEMU_UEFI_OPTION}
- ${QEMU_FLAGS_${ARCH}}
- ${QEMU_FLAGS}
- ${QEMU_EXTRA_FLAGS}
- ${MORE_FLAGS_FOR_${target}}
- ${QEMU_SMP_FLAGS}
- ${QEMU_KERNEL_OPTION}
- DEPENDS ${logical_target_for_zephyr_elf}
- WORKING_DIRECTORY ${APPLICATION_BINARY_DIR}
- COMMENT "${QEMU_PIPE_COMMENT}[QEMU] CPU: ${QEMU_CPU_TYPE_${ARCH}}"
- USES_TERMINAL
- )
- if(DEFINED QEMU_KERNEL_FILE)
- add_dependencies(${target} qemu_kernel_target)
- endif()
- endforeach()
|