qemu.cmake 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. # SPDX-License-Identifier: Apache-2.0
  2. if("${ARCH}" STREQUAL "x86")
  3. set_ifndef(QEMU_binary_suffix i386)
  4. elseif(DEFINED QEMU_ARCH)
  5. set_ifndef(QEMU_binary_suffix ${QEMU_ARCH})
  6. else()
  7. set_ifndef(QEMU_binary_suffix ${ARCH})
  8. endif()
  9. set(qemu_alternate_path $ENV{QEMU_BIN_PATH})
  10. if(qemu_alternate_path)
  11. find_program(
  12. QEMU
  13. PATHS ${qemu_alternate_path}
  14. NO_DEFAULT_PATH
  15. NAMES qemu-system-${QEMU_binary_suffix}
  16. )
  17. else()
  18. find_program(
  19. QEMU
  20. qemu-system-${QEMU_binary_suffix}
  21. )
  22. endif()
  23. # We need to set up uefi-run and OVMF environment
  24. # for testing UEFI method on qemu platforms
  25. if(CONFIG_QEMU_UEFI_BOOT)
  26. find_program(UEFI NAMES uefi-run REQUIRED)
  27. if(DEFINED ENV{OVMF_FD_PATH})
  28. set(OVMF_FD_PATH $ENV{OVMF_FD_PATH})
  29. else()
  30. message(FATAL_ERROR "Couldn't find an valid OVMF_FD_PATH.")
  31. endif()
  32. list(APPEND UEFI -b ${OVMF_FD_PATH} -q ${QEMU})
  33. set(QEMU ${UEFI})
  34. endif()
  35. set(qemu_targets
  36. run
  37. debugserver
  38. )
  39. set(QEMU_FLAGS -pidfile)
  40. if(${CMAKE_GENERATOR} STREQUAL "Unix Makefiles")
  41. list(APPEND QEMU_FLAGS qemu\${QEMU_INSTANCE}.pid)
  42. else()
  43. list(APPEND QEMU_FLAGS qemu${QEMU_INSTANCE}.pid)
  44. endif()
  45. # Set up chardev for console.
  46. if(QEMU_PTY)
  47. # Redirect console to a pseudo-tty, used for running automated tests.
  48. list(APPEND QEMU_FLAGS -chardev pty,id=con,mux=on)
  49. elseif(QEMU_PIPE)
  50. # Redirect console to a pipe, used for running automated tests.
  51. list(APPEND QEMU_FLAGS -chardev pipe,id=con,mux=on,path=${QEMU_PIPE})
  52. else()
  53. # Redirect console to stdio, used for manual debugging.
  54. list(APPEND QEMU_FLAGS -chardev stdio,id=con,mux=on)
  55. endif()
  56. # Connect main serial port to the console chardev.
  57. list(APPEND QEMU_FLAGS -serial chardev:con)
  58. # Connect semihosting console to the console chardev if configured.
  59. if(CONFIG_SEMIHOST_CONSOLE)
  60. list(APPEND QEMU_FLAGS
  61. -semihosting-config enable=on,target=auto,chardev=con
  62. )
  63. endif()
  64. # Connect monitor to the console chardev.
  65. list(APPEND QEMU_FLAGS -mon chardev=con,mode=readline)
  66. if(CONFIG_QEMU_ICOUNT)
  67. list(APPEND QEMU_FLAGS
  68. -icount shift=${CONFIG_QEMU_ICOUNT_SHIFT},align=off,sleep=off
  69. -rtc clock=vm)
  70. endif()
  71. # Add a BT serial device when building for bluetooth, unless the
  72. # application explicitly opts out with NO_QEMU_SERIAL_BT_SERVER.
  73. if(CONFIG_BT)
  74. if(CONFIG_BT_NO_DRIVER)
  75. set(NO_QEMU_SERIAL_BT_SERVER 1)
  76. endif()
  77. if(NOT NO_QEMU_SERIAL_BT_SERVER)
  78. list(APPEND QEMU_FLAGS -serial unix:/tmp/bt-server-bredr)
  79. endif()
  80. endif()
  81. # If we are running a networking application in QEMU, then set proper
  82. # QEMU variables. This also allows two QEMUs to be hooked together and
  83. # pass data between them. The QEMU flags are not set for standalone
  84. # tests defined by CONFIG_NET_TEST. For PPP, the serial port file is
  85. # not available if we run unit tests which define CONFIG_NET_TEST.
  86. if(CONFIG_NETWORKING)
  87. if(CONFIG_NET_QEMU_SLIP)
  88. if((CONFIG_NET_SLIP_TAP) OR (CONFIG_IEEE802154_UPIPE))
  89. set(QEMU_NET_STACK 1)
  90. endif()
  91. elseif((CONFIG_NET_QEMU_PPP) AND NOT (CONFIG_NET_TEST))
  92. set(QEMU_NET_STACK 1)
  93. endif()
  94. endif()
  95. # TO create independent pipes for each QEMU application set QEMU_PIPE_STACK
  96. if(QEMU_PIPE_STACK)
  97. list(APPEND qemu_targets
  98. node
  99. )
  100. if(NOT QEMU_PIPE_ID)
  101. set(QEMU_PIPE_ID 1)
  102. endif()
  103. list(APPEND QEMU_FLAGS
  104. -serial none
  105. )
  106. list(APPEND MORE_FLAGS_FOR_node
  107. -serial pipe:/tmp/hub/ip-stack-node${QEMU_PIPE_ID}
  108. -pidfile qemu-node${QEMU_PIPE_ID}.pid
  109. )
  110. set(PIPE_NODE_IN /tmp/hub/ip-stack-node${QEMU_PIPE_ID}.in)
  111. set(PIPE_NODE_OUT /tmp/hub/ip-stack-node${QEMU_PIPE_ID}.out)
  112. set(pipes
  113. ${PIPE_NODE_IN}
  114. ${PIPE_NODE_OUT}
  115. )
  116. set(destroy_pipe_commands
  117. COMMAND ${CMAKE_COMMAND} -E remove -f ${pipes}
  118. )
  119. set(create_pipe_commands
  120. COMMAND ${CMAKE_COMMAND} -E make_directory /tmp/hub
  121. COMMAND mkfifo ${PIPE_NODE_IN}
  122. COMMAND mkfifo ${PIPE_NODE_OUT}
  123. )
  124. set(PRE_QEMU_COMMANDS_FOR_node
  125. ${destroy_pipe_commands}
  126. ${create_pipe_commands}
  127. )
  128. elseif(QEMU_NET_STACK)
  129. list(APPEND qemu_targets
  130. client
  131. server
  132. )
  133. foreach(target ${qemu_targets})
  134. if((${target} STREQUAL client) OR (${target} STREQUAL server))
  135. list(APPEND MORE_FLAGS_FOR_${target}
  136. -serial pipe:/tmp/ip-stack-${target}
  137. -pidfile qemu-${target}.pid
  138. )
  139. else()
  140. # QEMU_INSTANCE is a command line argument to *make* (not cmake). By
  141. # appending the instance name to the pid file we can easily run more
  142. # instances of the same sample.
  143. if(CONFIG_NET_QEMU_PPP)
  144. if(${CMAKE_GENERATOR} STREQUAL "Unix Makefiles")
  145. set(ppp_path unix:/tmp/ppp\${QEMU_INSTANCE})
  146. else()
  147. set(ppp_path unix:/tmp/ppp${QEMU_INSTANCE})
  148. endif()
  149. list(APPEND MORE_FLAGS_FOR_${target}
  150. -serial ${ppp_path}
  151. )
  152. else()
  153. if(${CMAKE_GENERATOR} STREQUAL "Unix Makefiles")
  154. set(tmp_file unix:/tmp/slip.sock\${QEMU_INSTANCE})
  155. else()
  156. set(tmp_file unix:/tmp/slip.sock${QEMU_INSTANCE})
  157. endif()
  158. list(APPEND MORE_FLAGS_FOR_${target}
  159. -serial ${tmp_file}
  160. )
  161. endif()
  162. endif()
  163. endforeach()
  164. set(PIPE_SERVER_IN /tmp/ip-stack-server.in)
  165. set(PIPE_SERVER_OUT /tmp/ip-stack-server.out)
  166. set(PIPE_CLIENT_IN /tmp/ip-stack-client.in)
  167. set(PIPE_CLIENT_OUT /tmp/ip-stack-client.out)
  168. set(pipes
  169. ${PIPE_SERVER_IN}
  170. ${PIPE_SERVER_OUT}
  171. ${PIPE_CLIENT_IN}
  172. ${PIPE_CLIENT_OUT}
  173. )
  174. set(destroy_pipe_commands
  175. COMMAND ${CMAKE_COMMAND} -E remove -f ${pipes}
  176. )
  177. # TODO: Port to Windows. Perhaps using python? Or removing the
  178. # need for mkfifo and create_symlink somehow.
  179. set(create_pipe_commands
  180. COMMAND mkfifo ${PIPE_SERVER_IN}
  181. COMMAND mkfifo ${PIPE_SERVER_OUT}
  182. )
  183. if(PCAP)
  184. list(APPEND create_pipe_commands
  185. COMMAND mkfifo ${PIPE_CLIENT_IN}
  186. COMMAND mkfifo ${PIPE_CLIENT_OUT}
  187. )
  188. else()
  189. list(APPEND create_pipe_commands
  190. COMMAND ${CMAKE_COMMAND} -E create_symlink ${PIPE_SERVER_IN} ${PIPE_CLIENT_OUT}
  191. COMMAND ${CMAKE_COMMAND} -E create_symlink ${PIPE_SERVER_OUT} ${PIPE_CLIENT_IN}
  192. )
  193. endif()
  194. set(PRE_QEMU_COMMANDS_FOR_server
  195. ${destroy_pipe_commands}
  196. ${create_pipe_commands}
  197. )
  198. if(PCAP)
  199. # Start a monitor application to capture traffic
  200. #
  201. # Assumes;
  202. # PCAP has been set to the file where traffic should be captured
  203. # NET_TOOLS has been set to the net-tools repo path
  204. # net-tools/monitor_15_4 has been built beforehand
  205. set_ifndef(NET_TOOLS ${ZEPHYR_BASE}/../net-tools) # Default if not set
  206. list(APPEND PRE_QEMU_COMMANDS_FOR_server
  207. COMMAND
  208. #This command is run in the background using '&'. This prevents
  209. #chaining other commands with '&&'. The command is enclosed in '{}'
  210. #to fix this.
  211. {
  212. ${NET_TOOLS}/monitor_15_4
  213. ${PCAP}
  214. /tmp/ip-stack-server
  215. /tmp/ip-stack-client
  216. > /dev/null &
  217. }
  218. # TODO: Support cleanup of the monitor_15_4 process
  219. )
  220. endif()
  221. endif(QEMU_PIPE_STACK)
  222. if(CONFIG_X86_64 AND NOT CONFIG_QEMU_UEFI_BOOT)
  223. # QEMU doesn't like 64-bit ELF files. Since we don't use any >4GB
  224. # addresses, converting it to 32-bit is safe enough for emulation.
  225. add_custom_target(qemu_image_target
  226. COMMAND
  227. ${CMAKE_OBJCOPY}
  228. -O elf32-i386
  229. $<TARGET_FILE:${logical_target_for_zephyr_elf}>
  230. ${ZEPHYR_BINARY_DIR}/zephyr-qemu.elf
  231. DEPENDS ${logical_target_for_zephyr_elf}
  232. )
  233. # Split the 'locore' and 'main' memory regions into separate executable
  234. # images and specify the 'locore' as the boot kernel, in order to prevent
  235. # the QEMU direct multiboot kernel loader from overwriting the BIOS and
  236. # option ROM areas located in between the two memory regions.
  237. # (for more details, refer to the issue zephyrproject-rtos/sdk-ng#168)
  238. add_custom_target(qemu_locore_image_target
  239. COMMAND
  240. ${CMAKE_OBJCOPY}
  241. -j .locore
  242. ${ZEPHYR_BINARY_DIR}/zephyr-qemu.elf
  243. ${ZEPHYR_BINARY_DIR}/zephyr-qemu-locore.elf
  244. 2>&1 | grep -iv \"empty loadable segment detected\" || true
  245. DEPENDS qemu_image_target
  246. )
  247. add_custom_target(qemu_main_image_target
  248. COMMAND
  249. ${CMAKE_OBJCOPY}
  250. -R .locore
  251. ${ZEPHYR_BINARY_DIR}/zephyr-qemu.elf
  252. ${ZEPHYR_BINARY_DIR}/zephyr-qemu-main.elf
  253. 2>&1 | grep -iv \"empty loadable segment detected\" || true
  254. DEPENDS qemu_image_target
  255. )
  256. add_custom_target(
  257. qemu_kernel_target
  258. DEPENDS qemu_locore_image_target qemu_main_image_target
  259. )
  260. set(QEMU_KERNEL_FILE "${ZEPHYR_BINARY_DIR}/zephyr-qemu-locore.elf")
  261. list(APPEND QEMU_EXTRA_FLAGS
  262. "-device;loader,file=${ZEPHYR_BINARY_DIR}/zephyr-qemu-main.elf"
  263. )
  264. endif()
  265. if(CONFIG_IVSHMEM)
  266. if(CONFIG_IVSHMEM_DOORBELL)
  267. list(APPEND QEMU_FLAGS
  268. -device ivshmem-doorbell,vectors=${CONFIG_IVSHMEM_MSI_X_VECTORS},chardev=ivshmem
  269. -chardev socket,path=/tmp/ivshmem_socket,id=ivshmem
  270. )
  271. else()
  272. list(APPEND QEMU_FLAGS
  273. -device ivshmem-plain,memdev=hostmem
  274. -object memory-backend-file,size=${CONFIG_QEMU_IVSHMEM_PLAIN_MEM_SIZE}M,share,mem-path=/dev/shm/ivshmem,id=hostmem
  275. )
  276. endif()
  277. endif()
  278. if(NOT QEMU_PIPE)
  279. set(QEMU_PIPE_COMMENT "\nTo exit from QEMU enter: 'CTRL+a, x'\n")
  280. endif()
  281. # Don't just test CONFIG_SMP, there is at least one test of the lower
  282. # level multiprocessor API that wants an auxiliary CPU but doesn't
  283. # want SMP using it.
  284. if(NOT CONFIG_MP_NUM_CPUS MATCHES "1")
  285. list(APPEND QEMU_SMP_FLAGS -smp cpus=${CONFIG_MP_NUM_CPUS})
  286. endif()
  287. # Use flags passed in from the environment
  288. set(env_qemu $ENV{QEMU_EXTRA_FLAGS})
  289. separate_arguments(env_qemu)
  290. list(APPEND QEMU_EXTRA_FLAGS ${env_qemu})
  291. list(APPEND MORE_FLAGS_FOR_debugserver -s -S)
  292. # Architectures can define QEMU_KERNEL_FILE to use a specific output
  293. # file to pass to qemu (and a "qemu_kernel_target" target to generate
  294. # it), or set QEMU_KERNEL_OPTION if they want to replace the "-kernel
  295. # ..." option entirely.
  296. if(CONFIG_QEMU_UEFI_BOOT)
  297. set(QEMU_UEFI_OPTION ${PROJECT_BINARY_DIR}/${CONFIG_KERNEL_BIN_NAME}.efi)
  298. list(APPEND QEMU_UEFI_OPTION --)
  299. elseif(DEFINED QEMU_KERNEL_FILE)
  300. set(QEMU_KERNEL_OPTION "-kernel;${QEMU_KERNEL_FILE}")
  301. elseif(NOT DEFINED QEMU_KERNEL_OPTION)
  302. set(QEMU_KERNEL_OPTION "-kernel;$<TARGET_FILE:${logical_target_for_zephyr_elf}>")
  303. elseif(DEFINED QEMU_KERNEL_OPTION)
  304. string(CONFIGURE "${QEMU_KERNEL_OPTION}" QEMU_KERNEL_OPTION)
  305. endif()
  306. foreach(target ${qemu_targets})
  307. add_custom_target(${target}
  308. ${PRE_QEMU_COMMANDS}
  309. ${PRE_QEMU_COMMANDS_FOR_${target}}
  310. COMMAND
  311. ${QEMU}
  312. ${QEMU_UEFI_OPTION}
  313. ${QEMU_FLAGS_${ARCH}}
  314. ${QEMU_FLAGS}
  315. ${QEMU_EXTRA_FLAGS}
  316. ${MORE_FLAGS_FOR_${target}}
  317. ${QEMU_SMP_FLAGS}
  318. ${QEMU_KERNEL_OPTION}
  319. DEPENDS ${logical_target_for_zephyr_elf}
  320. WORKING_DIRECTORY ${APPLICATION_BINARY_DIR}
  321. COMMENT "${QEMU_PIPE_COMMENT}[QEMU] CPU: ${QEMU_CPU_TYPE_${ARCH}}"
  322. USES_TERMINAL
  323. )
  324. if(DEFINED QEMU_KERNEL_FILE)
  325. add_dependencies(${target} qemu_kernel_target)
  326. endif()
  327. endforeach()