run-qemu.sh 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. check_dependencies()
  2. {
  3. # Check if qemu is installed
  4. if [ -z "$(which qemu-system-x86_64)" ]; then
  5. echo "Please install qemu first!"
  6. exit 1
  7. fi
  8. if [ -z "$(which ${QEMU})" ]; then
  9. if [ "$ARCH" == "loongarch64" ]; then
  10. echo -e "\nPlease install qemu-system-loongarch64 first!"
  11. echo -e "\nYou can install it by running: (if you are using ubuntu)"
  12. echo -e " ${ROOT_PATH}/tools/qemu/build-qemu-la64-for-ubuntu.sh"
  13. echo -e ""
  14. exit 1
  15. fi
  16. fi
  17. # Check if brctl is installed
  18. if [ -z "$(which brctl)" ]; then
  19. echo "Please install bridge-utils first!"
  20. exit 1
  21. fi
  22. # Check if dnsmasq is installed
  23. if [ -z "$(which dnsmasq)" ]; then
  24. echo "Please install dnsmasq first!"
  25. exit 1
  26. fi
  27. # Check if iptable is installed
  28. if [ -z "$(which iptables)" ]; then
  29. echo "Please install iptables first!"
  30. exit 1
  31. fi
  32. }
  33. # 进行启动前检查
  34. flag_can_run=1
  35. ARGS=`getopt -o p -l bios:,display: -- "$@"`
  36. eval set -- "${ARGS}"
  37. echo "$@"
  38. allflags=
  39. # allflags=$(qemu-system-x86_64 -cpu help | awk '/flags/ {y=1; getline}; y {print}' | tr ' ' '\n' | grep -Ev "^$" | sed -r 's|^|+|' | tr '\n' ',' | sed -r "s|,$||")
  40. # 设置ARCH环境变量,如果没有设置,就默认为x86_64
  41. export ARCH=${ARCH:=x86_64}
  42. echo "ARCH=${ARCH}"
  43. #ARCH="i386"
  44. # 请根据自己的需要,在-d 后方加入所需的 trace 事件
  45. # 标准的trace events
  46. qemu_trace_std=cpu_reset,guest_errors,trace:virtio*,trace:e1000e_rx*,trace:e1000e_tx*,trace:e1000e_irq*
  47. # 调试usb的trace
  48. qemu_trace_usb=trace:usb_xhci_reset,trace:usb_xhci_run,trace:usb_xhci_stop,trace:usb_xhci_irq_msi,trace:usb_xhci_irq_msix,trace:usb_xhci_port_reset,trace:msix_write_config,trace:usb_xhci_irq_msix,trace:usb_xhci_irq_msix_use,trace:usb_xhci_irq_msix_unuse,trace:usb_xhci_irq_msi,trace:usb_xhci_*
  49. # 根据架构设置qemu的加速方式
  50. if [ ${ARCH} == "i386" ] || [ ${ARCH} == "x86_64" ]; then
  51. qemu_accel="kvm"
  52. if [ $(uname) == Darwin ]; then
  53. qemu_accel=hvf
  54. else
  55. # 判断系统kvm模块是否加载
  56. if [ ! -e /dev/kvm ]; then
  57. # kvm模块未加载,使用tcg加速
  58. qemu_accel="tcg"
  59. fi
  60. fi
  61. fi
  62. # uboot版本
  63. UBOOT_VERSION="v2023.10"
  64. RISCV64_UBOOT_PATH="arch/riscv64/u-boot-${UBOOT_VERSION}-riscv64"
  65. DISK_NAME="disk-image-${ARCH}.img"
  66. EXT4_DISK_NAME="ext4.img"
  67. FAT_DISK_NAME="fat.img"
  68. QEMU=$(which qemu-system-${ARCH})
  69. QEMU_DISK_IMAGE="../bin/${DISK_NAME}"
  70. QEMU_EXT4_DISK_IMAGE="../bin/${EXT4_DISK_NAME}"
  71. QEMU_FAT_DISK_IMAGE="../bin/${FAT_DISK_NAME}"
  72. QEMU_MEMORY="512M"
  73. QEMU_MEMORY_BACKEND="dragonos-qemu-shm.ram"
  74. QEMU_MEMORY_BACKEND_PATH_PREFIX="/dev/shm"
  75. QEMU_SHM_OBJECT="-object memory-backend-file,size=${QEMU_MEMORY},id=${QEMU_MEMORY_BACKEND},mem-path=${QEMU_MEMORY_BACKEND_PATH_PREFIX}/${QEMU_MEMORY_BACKEND},share=on "
  76. QEMU_SMP="2,cores=2,threads=1,sockets=1"
  77. QEMU_MONITOR="-monitor stdio"
  78. QEMU_TRACE="${qemu_trace_std}"
  79. QEMU_CPU_FEATURES=""
  80. QEMU_RTC_CLOCK=""
  81. QEMU_SERIAL_LOG_FILE="../serial_opt.txt"
  82. QEMU_SERIAL="-serial file:${QEMU_SERIAL_LOG_FILE}"
  83. QEMU_DRIVE="id=disk,file=${QEMU_DISK_IMAGE},if=none"
  84. QEMU_ACCELARATE=""
  85. QEMU_ARGUMENT=" -no-reboot "
  86. QEMU_DEVICES=""
  87. if [ -f "${QEMU_EXT4_DISK_IMAGE}" ]; then
  88. QEMU_DRIVE+=" -drive id=ext4disk,file=${QEMU_EXT4_DISK_IMAGE},if=none,format=raw"
  89. fi
  90. if [ -f "${QEMU_FAT_DISK_IMAGE}" ]; then
  91. QEMU_DRIVE+=" -drive id=fatdisk,file=${QEMU_FAT_DISK_IMAGE},if=none,format=raw"
  92. fi
  93. check_dependencies
  94. # 设置无图形界面模式
  95. QEMU_NOGRAPHIC=false
  96. KERNEL_CMDLINE=" "
  97. BIOS_TYPE=""
  98. #这个变量为true则使用virtio磁盘
  99. VIRTIO_BLK_DEVICE=true
  100. # 如果qemu_accel不为空
  101. if [ -n "${qemu_accel}" ]; then
  102. QEMU_ACCELARATE=" -machine accel=${qemu_accel} "
  103. if [ "${qemu_accel}" == "kvm" ]; then
  104. QEMU_ACCELARATE+=" -enable-kvm "
  105. fi
  106. fi
  107. if [ ${ARCH} == "i386" ] || [ ${ARCH} == "x86_64" ]; then
  108. QEMU_MACHINE=" -machine q35,memory-backend=${QEMU_MEMORY_BACKEND} "
  109. QEMU_CPU_FEATURES+="-cpu IvyBridge,apic,x2apic,+fpu,check,+vmx,${allflags}"
  110. QEMU_RTC_CLOCK+=" -rtc clock=host,base=localtime"
  111. if [ ${VIRTIO_BLK_DEVICE} == false ]; then
  112. QEMU_DEVICES_DISK+="-device ahci,id=ahci -device ide-hd,drive=disk,bus=ahci.0 "
  113. else
  114. QEMU_DEVICES_DISK="-device virtio-blk-pci,drive=disk -device pci-bridge,chassis_nr=1,id=pci.1 -device pcie-root-port "
  115. fi
  116. if [ -f "${QEMU_EXT4_DISK_IMAGE}" ]; then
  117. QEMU_DEVICES_DISK+=" -device virtio-blk-pci,drive=ext4disk"
  118. fi
  119. if [ -f "${QEMU_FAT_DISK_IMAGE}" ]; then
  120. QEMU_DEVICES_DISK+=" -device virtio-blk-pci,drive=fatdisk"
  121. fi
  122. elif [ ${ARCH} == "riscv64" ]; then
  123. QEMU_MACHINE=" -machine virt,memory-backend=${QEMU_MEMORY_BACKEND} -cpu sifive-u54 "
  124. QEMU_DEVICES_DISK="-device virtio-blk-device,drive=disk "
  125. elif [ ${ARCH} == "loongarch64" ]; then
  126. QEMU_MACHINE=" -machine virt"
  127. QEMU_DEVICES_DISK="-device virtio-blk-pci,drive=disk -device pci-bridge,chassis_nr=1,id=pci.1 -device pcie-root-port "
  128. else
  129. echo "Unsupported architecture: ${ARCH}"
  130. exit 1
  131. fi
  132. if [ ${ARCH} == "riscv64" ]; then
  133. # 如果是riscv64架构,就不需要图形界面
  134. QEMU_NOGRAPHIC=true
  135. fi
  136. while true;do
  137. case "$1" in
  138. --bios)
  139. case "$2" in
  140. uefi) #uefi启动新增ovmf.fd固件
  141. BIOS_TYPE=uefi
  142. ;;
  143. legacy)
  144. BIOS_TYPE=legacy
  145. ;;
  146. esac;shift 2;;
  147. --display)
  148. case "$2" in
  149. vnc)
  150. QEMU_ARGUMENT+=" -display vnc=:00 "
  151. ;;
  152. window)
  153. ;;
  154. nographic)
  155. QEMU_NOGRAPHIC=true
  156. ;;
  157. esac;shift 2;;
  158. *) break
  159. esac
  160. done
  161. setup_kernel_init_program() {
  162. if [ ${ARCH} == "x86_64" ]; then
  163. KERNEL_CMDLINE+=" init=/bin/busybox init "
  164. # KERNEL_CMDLINE+=" init=/bin/dragonreach "
  165. elif [ ${ARCH} == "riscv64" ]; then
  166. KERNEL_CMDLINE+=" init=/bin/riscv_rust_init "
  167. fi
  168. }
  169. # 设置内核init程序
  170. setup_kernel_init_program
  171. if [ ${QEMU_NOGRAPHIC} == true ]; then
  172. QEMU_SERIAL=" -serial chardev:mux -monitor chardev:mux -chardev stdio,id=mux,mux=on,signal=off,logfile=${QEMU_SERIAL_LOG_FILE} "
  173. # 添加 virtio console 设备
  174. if [ ${ARCH} == "x86_64" ]; then
  175. QEMU_DEVICES+=" -device virtio-serial -device virtconsole,chardev=mux "
  176. elif [ ${ARCH} == "loongarch64" ]; then
  177. QEMU_DEVICES+=" -device virtio-serial -device virtconsole,chardev=mux "
  178. elif [ ${ARCH} == "riscv64" ]; then
  179. QEMU_DEVICES+=" -device virtio-serial-device -device virtconsole,chardev=mux "
  180. fi
  181. KERNEL_CMDLINE=" console=/dev/hvc0 ${KERNEL_CMDLINE}"
  182. QEMU_MONITOR=""
  183. QEMU_ARGUMENT+=" --nographic "
  184. KERNEL_CMDLINE=$(echo "${KERNEL_CMDLINE}" | sed 's/^[ \t]*//;s/[ \t]*$//')
  185. if [ ${ARCH} == "x86_64" ]; then
  186. QEMU_ARGUMENT+=" -kernel ../bin/kernel/kernel.elf -append \"${KERNEL_CMDLINE}\" "
  187. elif [ ${ARCH} == "loongarch64" ]; then
  188. QEMU_ARGUMENT+=" -kernel ../bin/kernel/kernel.elf -append \"${KERNEL_CMDLINE}\" "
  189. elif [ ${ARCH} == "riscv64" ]; then
  190. QEMU_ARGUMENT+=" -append \"${KERNEL_CMDLINE}\" "
  191. fi
  192. fi
  193. # ps: 下面这条使用tap的方式,无法dhcp获取到ip,暂时不知道为什么
  194. # QEMU_DEVICES="-device ahci,id=ahci -device ide-hd,drive=disk,bus=ahci.0 -net nic,netdev=nic0 -netdev tap,id=nic0,model=virtio-net-pci,script=qemu/ifup-nat,downscript=qemu/ifdown-nat -usb -device qemu-xhci,id=xhci,p2=8,p3=4 "
  195. QEMU_DEVICES+="${QEMU_DEVICES_DISK} "
  196. QEMU_DEVICES+=" -netdev user,id=hostnet0,hostfwd=tcp::12580-:12580 -device virtio-net-pci,vectors=5,netdev=hostnet0,id=net0 -usb -device qemu-xhci,id=xhci,p2=8,p3=4 "
  197. # E1000E
  198. # QEMU_DEVICES="-device ahci,id=ahci -device ide-hd,drive=disk,bus=ahci.0 -netdev user,id=hostnet0,hostfwd=tcp::12580-:12580 -net nic,model=e1000e,netdev=hostnet0,id=net0 -netdev user,id=hostnet1,hostfwd=tcp::12581-:12581 -device virtio-net-pci,vectors=5,netdev=hostnet1,id=net1 -usb -device qemu-xhci,id=xhci,p2=8,p3=4 "
  199. QEMU_ARGUMENT+="-d ${QEMU_DISK_IMAGE} -m ${QEMU_MEMORY} -smp ${QEMU_SMP} -boot order=d ${QEMU_MONITOR} -d ${qemu_trace_std} "
  200. QEMU_ARGUMENT+="-s ${QEMU_MACHINE} ${QEMU_CPU_FEATURES} ${QEMU_RTC_CLOCK} ${QEMU_SERIAL} -drive ${QEMU_DRIVE} ${QEMU_DEVICES} "
  201. QEMU_ARGUMENT+=" ${QEMU_SHM_OBJECT} "
  202. QEMU_ARGUMENT+=" ${QEMU_ACCELARATE} "
  203. QEMU_ARGUMENT+=" -D ../qemu.log "
  204. # 安装riscv64的uboot
  205. install_riscv_uboot()
  206. {
  207. if [ ! -d ${RISCV64_UBOOT_PATH} ]; then
  208. echo "正在下载u-boot..."
  209. uboot_tar_name="u-boot-${UBOOT_VERSION}-riscv64.tar.xz"
  210. uboot_parent_path=$(dirname ${RISCV64_UBOOT_PATH}) || (echo "获取riscv u-boot 版本 ${UBOOT_VERSION} 的父目录失败" && exit 1)
  211. if [ ! -f ${uboot_tar_name} ]; then
  212. wget https://mirrors.dragonos.org.cn/pub/third_party/u-boot/${uboot_tar_name} || (echo "下载riscv u-boot 版本 ${UBOOT_VERSION} 失败" && exit 1)
  213. fi
  214. echo "下载完成"
  215. echo "正在解压u-boot到 '$uboot_parent_path'..."
  216. mkdir -p $uboot_parent_path
  217. tar xvf u-boot-${UBOOT_VERSION}-riscv64.tar.xz -C ${uboot_parent_path} || (echo "解压riscv u-boot 版本 ${UBOOT_VERSION} 失败" && exit 1)
  218. echo "解压完成"
  219. rm -rf u-boot-${UBOOT_VERSION}-riscv64.tar.xz
  220. fi
  221. echo "riscv u-boot 版本 ${UBOOT_VERSION} 已经安装"
  222. }
  223. if [ $flag_can_run -eq 1 ]; then
  224. # 删除共享内存
  225. sudo rm -rf ${QEMU_MEMORY_BACKEND_PATH_PREFIX}/${QEMU_MEMORY_BACKEND}
  226. if [ ${BIOS_TYPE} == uefi ] ;then
  227. if [ ${ARCH} == x86_64 ] ;then
  228. sh -c "sudo ${QEMU} -bios arch/x86_64/efi/OVMF-pure-efi.fd ${QEMU_ARGUMENT}"
  229. elif [ ${ARCH} == i386 ] ;then
  230. sh -c "sudo ${QEMU} -bios arch/i386/efi/OVMF-pure-efi.fd ${QEMU_ARGUMENT}"
  231. elif [ ${ARCH} == riscv64 ] ;then
  232. install_riscv_uboot
  233. sh -c "sudo ${QEMU} -kernel ${RISCV64_UBOOT_PATH}/u-boot.bin ${QEMU_ARGUMENT}"
  234. else
  235. echo "不支持的架构: ${ARCH}"
  236. fi
  237. else
  238. # 如果是i386架构或者x86_64架构,就直接启动
  239. if [ ${ARCH} == x86_64 ] || [ ${ARCH} == i386 ] ;then
  240. sh -c "sudo ${QEMU} ${QEMU_ARGUMENT}"
  241. elif [ ${ARCH} == riscv64 ] ;then
  242. # 如果是riscv64架构,就与efi启动一样
  243. install_riscv_uboot
  244. sh -c "sudo ${QEMU} -kernel ${RISCV64_UBOOT_PATH}/u-boot.bin ${QEMU_ARGUMENT}"
  245. elif [ ${ARCH} == loongarch64 ] ;then
  246. sh -c "sudo ${QEMU} ${QEMU_ARGUMENT}"
  247. else
  248. echo "不支持的架构: ${ARCH}"
  249. fi
  250. fi
  251. # 删除共享内存
  252. sudo rm -rf ${QEMU_MEMORY_BACKEND_PATH_PREFIX}/${QEMU_MEMORY_BACKEND}
  253. else
  254. echo "不满足运行条件"
  255. fi