roms 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459
  1. #!/usr/bin/env sh
  2. # SPDX-License-Identifier: GPL-3.0-or-later
  3. # SPDX-FileCopyrightText: 2014-2016,2020,2021,2023,2024 Leah Rowe <leah@libreboot.org>
  4. # SPDX-FileCopyrightText: 2021,2022 Ferass El Hafidi <vitali64pmemail@protonmail.com>
  5. # SPDX-FileCopyrightText: 2022 Caleb La Grange <thonkpeasant@protonmail.com>
  6. # SPDX-FileCopyrightText: 2022-2023 Alper Nebi Yasak <alpernebiyasak@gmail.com>
  7. # SPDX-FileCopyrightText: 2023 Riku Viitanen <riku.viitanen@protonmail.com>
  8. set -u -e
  9. . "include/option.sh"
  10. seavgabiosrom="elf/seabios/default/libgfxinit/vgabios.bin"
  11. grub_background="background1280x800.png"
  12. grubelf="elf/grub/grub.elf"
  13. cfgsdir="config/coreboot"
  14. pico_src_dir="src/pico-serprog"
  15. pico_sdk_dir="src/pico-sdk"
  16. stm32_src_dir="src/stm32-vserprog"
  17. # Disable all payloads by default.
  18. # target.cfg files have to specifically enable [a] payload(s)
  19. pv="payload_grub payload_grub_withseabios payload_seabios payload_memtest t"
  20. pv="${pv} payload_seabios_withgrub payload_seabios_grubonly payload_uboot memtest_bin"
  21. v="romdir cbrom initmode displaymode cbcfg targetdir tree keymaps release"
  22. v="${v} grub_timeout ubdir board grub_scan_disk uboot_config"
  23. eval "$(setvars "n" ${pv} serprog)"
  24. eval "$(setvars "" ${v} boards _displaymode _payload _keyboard all targets \
  25. serprog_boards_dir)"
  26. main()
  27. {
  28. while [ $# -gt 0 ]; do
  29. case ${1} in
  30. list)
  31. x_ items config/coreboot
  32. return 0 ;;
  33. serprog)
  34. serprog="y"
  35. shift 1; break ;;
  36. -d) _displaymode="${2}" ;;
  37. -p) _payload="${2}" ;;
  38. -k) _keyboard="${2}" ;;
  39. *)
  40. [ "${1}" = "all" ] && all="y"
  41. boards="${1} ${boards}"
  42. shift && continue ;;
  43. esac
  44. shift 2
  45. done
  46. if [ "$serprog" = "y" ]; then
  47. handle_serprog $@
  48. return 0
  49. else
  50. [ "${all}" != "y" ] || boards=$(items config/coreboot) || \
  51. $err "Cannot generate list of boards for building"
  52. for x in ${boards}; do
  53. [ -d "config/coreboot/$x/config" ] && \
  54. handle_coreboot_target "$x" && continue
  55. printf "No configs for %s; skipping\n" "$x"
  56. done
  57. fi
  58. bstr="directories"
  59. [ "$xbmk_release" = "y" ] && bstr="tarballs"
  60. [ -z "${targets}" ] && $err "No ROM images were compiled"
  61. printf "\nROM images available in these %s:\n" "$bstr"
  62. eval "printf \"${targets}\""
  63. printf "^^ ROM images available in these %s.\n\n" "$bstr"
  64. [ "$xbmk_release" = "y" ] && \
  65. printf "Always run the inject command on release images!\n"
  66. printf "DO NOT flash images from elf/ - please use bin/ instead.\n"
  67. }
  68. handle_serprog()
  69. {
  70. [ -z "${1+x}" ] && $err "bad command. Check $projectname docs."
  71. [ "$1" != "rp2040" ] && [ "$1" != "stm32" ] && $err "bad command"
  72. if [ "${1}" = "rp2040" ]; then
  73. serprog_boards_dir=${pico_sdk_dir}/src/boards/include/boards
  74. [ -d "$pico_src_dir" ] || x_ ./update trees -f "pico-serprog"
  75. elif [ "${1}" = "stm32" ]; then
  76. serprog_boards_dir=${stm32_src_dir}/boards
  77. [ -d "$stm32_src_dir" ] || x_ ./update trees -f "stm32-vserprog"
  78. fi
  79. x_ mkdir -p "bin/serprog_${1}"
  80. if [ $# -gt 1 ] && [ "${2}" = "list" ]; then
  81. list_serprog_boards ${serprog_boards_dir}
  82. return 0
  83. elif [ $# -gt 1 ]; then
  84. build_${1}_rom "${2}"
  85. else
  86. printf "Building all serprog targets\n"
  87. list_serprog_boards "${serprog_boards_dir}" | \
  88. while read -r board; do
  89. build_${1}_rom "${board}"
  90. done
  91. fi
  92. [ "$xbmk_release" = "y" ] && mkrom_tarball "bin/serprog_$1"; return 0
  93. }
  94. build_rp2040_rom()
  95. {
  96. board=${1}
  97. printf "Building pico-serprog for %s\n" "${board}"
  98. x_ cmake -DPICO_BOARD="$board" -DPICO_SDK_PATH="$pico_sdk_dir" \
  99. -B "${pico_src_dir}/build" "${pico_src_dir}"
  100. x_ cmake --build "${pico_src_dir}/build"
  101. x_ mv ${pico_src_dir}/build/pico_serprog.uf2 \
  102. bin/serprog_rp2040/serprog_${board}.uf2
  103. printf "output to bin/serprog_rp2040/serprog_%s.uf2\n" "$board"
  104. }
  105. build_stm32_rom()
  106. {
  107. board=${1}
  108. printf "Building stm32-vserprog for %s\n" "${board}"
  109. x_ make -C $stm32_src_dir libopencm3-just-make BOARD=$board
  110. x_ make -C ${stm32_src_dir} BOARD=${board}
  111. x_ mv ${stm32_src_dir}/stm32-vserprog.hex \
  112. bin/serprog_stm32/serprog_${board}.hex
  113. printf "output to bin/serprog_stm32/serprog_%s.hex\n" "$board"
  114. }
  115. list_serprog_boards()
  116. {
  117. basename -a -s .h "${1}/"*.h || $err "list_boards $1: can't list boards"
  118. }
  119. handle_coreboot_target()
  120. {
  121. eval "$(setvars "n" ${pv}) $(setvars "" ${v})"
  122. grub_background="background1280x800.png"
  123. board="$1"
  124. configure_target
  125. [ "$board" = "$tree" ] && return 0
  126. if [ "$xbmk_release" = "y" ] && [ "$release" = "n" ]; then
  127. printf "Target '%s' disabled for release.\n" "$board"
  128. return 0
  129. fi
  130. build_payloads
  131. build_target_mainboard
  132. [ -d "bin/${board}" ] || return 0
  133. [ "$xbmk_release" = "y" ] || targets="* bin/${board}\n${targets}"
  134. [ "$xbmk_release" = "y" ] && mkrom_tarball "bin/$board" && \
  135. targets="* bin/${relname}_$board.tar.xz\n$targets"; return 0
  136. }
  137. configure_target()
  138. {
  139. targetdir="${cfgsdir}/${board}"
  140. [ -f "${targetdir}/target.cfg" ] || \
  141. $err "Missing target.cfg for target: ${board}"
  142. # Override the above defaults using target.cfg
  143. . "${targetdir}/target.cfg"
  144. [ -z "${grub_scan_disk}" ] && grub_scan_disk="both"
  145. [ "$grub_scan_disk" != "both" ] && [ "$grub_scan_disk" != "ata" ] \
  146. && [ "${grub_scan_disk}" != "ahci" ] && \
  147. grub_scan_disk="both"
  148. [ -z "$tree" ] && $err "$board: tree not defined"
  149. [ "${payload_memtest}" != "y" ] && payload_memtest="n"
  150. [ "$(uname -m)" = "x86_64" ] || payload_memtest="n"
  151. [ "${payload_grub_withseabios}" = "y" ] && payload_grub="y"
  152. [ "${payload_grub_withseabios}" = "y" ] && \
  153. eval "$(setvars "y" payload_seabios payload_seabios_withgrub)"
  154. [ "$payload_seabios_withgrub" = "y" ] && payload_seabios="y"
  155. [ "$payload_seabios_grubonly" = "y" ] && payload_seabios="y"
  156. [ "$payload_seabios_grubonly" = "y" ] && payload_seabios_withgrub="y"
  157. # The reverse logic must not be applied. If SeaBIOS-with-GRUB works,
  158. # that doesn't mean GRUB-with-SeaBIOS will, e.g. VGA ROM execution
  159. [ "$payload_grub" != "y" ] && [ "$payload_seabios" != "y" ] && \
  160. [ "${payload_uboot}" != "y" ] && \
  161. $err "target '$board' defines no payload"
  162. [ "$payload_uboot" != "n" ] && [ "$payload_uboot" != "y" ] && \
  163. payload_uboot="n"
  164. [ "$payload_uboot" = "y" ] && [ -z "$uboot_config" ] && \
  165. uboot_config="default"
  166. # Override all payload directives with cmdline args
  167. [ -z "${_payload}" ] && return 0
  168. eval "$(setvars "n" payload_grub payload_memtest payload_seabios \
  169. payload_seabios_withgrub payload_uboot payload_grub_withseabios \
  170. payload_seabios_grubonly)"
  171. eval "payload_${_payload}=y"
  172. }
  173. build_payloads()
  174. {
  175. romdir="bin/${board}"
  176. cbdir="src/coreboot/${board}"
  177. [ "${board}" = "${tree}" ] || cbdir="src/coreboot/${tree}"
  178. cbfstool="cbutils/${tree}/cbfstool"
  179. cbrom="${cbdir}/build/coreboot.rom"
  180. [ -f "$cbfstool" ] || x_ ./update trees -b coreboot utils $tree
  181. memtest_bin="memtest86plus/build64/memtest.bin"
  182. [ "${payload_memtest}" != "y" ] || [ -f "src/${memtest_bin}" ] || \
  183. x_ ./update trees -b memtest86plus
  184. [ "$payload_seabios" = "y" ] && x_ ./update trees -b seabios
  185. if [ "$payload_grub" = "y" ] || [ "$payload_seabios_withgrub" = "y" ] \
  186. || [ "$payload_seabios_grubonly" = "y" ]; then build_grub_payload
  187. fi
  188. [ "${payload_uboot}" = "y" ] && build_uboot_payload; return 0
  189. }
  190. build_grub_payload()
  191. {
  192. x_ mkdir -p elf/grub
  193. for keymapfile in config/grub/keymap/*.gkb; do
  194. [ -f "${keymapfile}" ] || continue
  195. keymaps="${keymaps} ${keymapfile}"
  196. done
  197. [ -z "$_keyboard" ] || [ -f "$grubcfgsdir/keymap/$_keyboard.gkb" ] || \
  198. $err "build_grub_payload: $_keyboard layout not defined"
  199. [ -n "$_keyboard" ] && keymaps="${grubcfgsdir}/keymap/${_keyboard}.gkb"
  200. [ -f "$grubelf" ] && return 0
  201. [ -f "src/grub/grub-mkstandalone" ] || x_ ./update trees -b grub
  202. ./src/grub/grub-mkstandalone --grub-mkimage="src/grub/grub-mkimage" \
  203. -O i386-coreboot -o "elf/grub/grub.elf" -d "src/grub/grub-core/" \
  204. --fonts= --themes= --locales= --modules="${grub_modules}" \
  205. --install-modules="${grub_install_modules}" \
  206. "/boot/grub/grub.cfg=${grubcfgsdir}/config/grub_memdisk.cfg" \
  207. "/boot/grub/grub_default.cfg=${grubcfgsdir}/config/grub.cfg" || \
  208. $err "could not generate grub.elf"
  209. }
  210. build_uboot_payload()
  211. {
  212. x_ ./update trees -b u-boot ${board}
  213. ubdir="elf/u-boot/${board}/${uboot_config}"
  214. ubootelf="${ubdir}/u-boot.elf"
  215. [ ! -f "${ubootelf}" ] && [ -f "${ubdir}/u-boot" ] && \
  216. ubootelf="${ubdir}/u-boot"
  217. [ -f "${ubootelf}" ] || $err "$board: Can't find u-boot"; return 0
  218. }
  219. build_target_mainboard()
  220. {
  221. x_ rm -Rf "${romdir}"
  222. for x in "normal" "vgarom" "libgfxinit"; do
  223. initmode="${x}"
  224. hmode="vesafb"
  225. [ "${initmode}" = "vgarom" ] || hmode="corebootfb"
  226. modes="${hmode} txtmode"
  227. [ -z "${_displaymode}" ] || modes="${_displaymode}"
  228. for y in ${modes}; do
  229. displaymode="${y}"
  230. [ "${initmode}" = "normal" ] && \
  231. [ "$displaymode" != "txtmode" ] && continue
  232. cbcfg="${targetdir}/config/${initmode}_${displaymode}"
  233. [ "${initmode}" = "normal" ] && cbcfg="${cbcfg%_*}"
  234. build_roms "${cbcfg}"
  235. x_ rm -f "$cbrom"
  236. done
  237. done
  238. }
  239. build_roms()
  240. {
  241. cbcfg="${1}"
  242. if [ ! -f "${cbcfg}" ]; then
  243. printf "'%s' does not exist. Skipping build for %s %s %s\n" \
  244. "$cbcfg" "$board" "$displaymode" "$initmode" 1>&2
  245. return 0
  246. fi
  247. x_ ./update trees -b coreboot ${board}
  248. _cbrom="elf/coreboot_nopayload_DO_NOT_FLASH"
  249. _cbrom="${_cbrom}/${board}/${initmode}_${displaymode}"
  250. [ "${initmode}" = "normal" ] && \
  251. _cbrom="${_cbrom%"_${displaymode}"}"
  252. _cbrom="${_cbrom}/coreboot.rom"
  253. cbrom="$(mktemp -t coreboot_rom.XXXXXXXXXX)"
  254. x_ cp "${_cbrom}" "${cbrom}"
  255. [ "${payload_memtest}" != "y" ] || \
  256. x_ "${cbfstool}" "${cbrom}" add-payload \
  257. -f "src/${memtest_bin}" -n img/memtest -c lzma
  258. [ "${payload_seabios}" = "y" ] && build_seabios_roms
  259. [ "$payload_grub" != "y" ] || x_ build_grub_roms "$cbrom" "grub"
  260. [ "${payload_uboot}" = "y" ] || return 0
  261. x_ cp "${_cbrom}" "${cbrom}"
  262. build_uboot_roms
  263. }
  264. build_seabios_roms()
  265. {
  266. if [ "${payload_seabios_withgrub}" = "y" ]; then
  267. t=$(mktemp -t coreboot_rom.XXXXXXXXXX)
  268. x_ cp "${cbrom}" "${t}"
  269. x_ build_grub_roms "${t}" "seabios_withgrub"
  270. else
  271. t=$(mkSeabiosRom "${cbrom}" "fallback/payload") || \
  272. $err "build_seabios_roms: cannot build tmprom"
  273. newrom="${romdir}/seabios_${board}_${initmode}_${displaymode}"
  274. [ "${initmode}" = "normal" ] && newrom="${romdir}/seabios" \
  275. && newrom="${newrom}_${board}_${initmode}"
  276. x_ copyrom "${t}" "${newrom}.rom"
  277. fi
  278. x_ rm -f "${t}"
  279. }
  280. # Make separate ROM images with GRUB payload, for each supported keymap
  281. build_grub_roms()
  282. {
  283. tmprom="${1}"
  284. payload1="${2}" # allow values: grub, seabios, seabios_withgrub
  285. grub_cbfs="fallback/payload"
  286. if [ "$payload1" = "grub" ] && [ "$payload_grub_withseabios" = "y" ]
  287. then
  288. _tmpmvrom=$(mkSeabiosRom "$tmprom" "seabios.elf") || \
  289. $err "build_grub_roms 1 $board: can't build tmprom"
  290. x_ mv "$_tmpmvrom" "$tmprom"
  291. elif [ "$payload1" != "grub" ] && [ "$payload_seabios_withgrub" = "y" ]
  292. then
  293. grub_cbfs="img/grub2"
  294. _tmpmvrom=$(mkSeabiosRom "$tmprom" fallback/payload) || \
  295. $err "build_grub_roms 2 $board: can't build tmprom"
  296. x_ mv "$_tmpmvrom" "$tmprom"
  297. fi
  298. # we only need insert grub.elf once, for each coreboot config:
  299. x_ "${cbfstool}" "${tmprom}" add-payload -f "${grubelf}" \
  300. -n ${grub_cbfs} -c lzma
  301. # we only need insert background.png once, for each coreboot config:
  302. if [ "${displaymode}" = "vesafb" ] || \
  303. [ "${displaymode}" = "corebootfb" ]; then
  304. backgroundfile="config/grub/background/${grub_background}"
  305. "${cbfstool}" "${tmprom}" add -f ${backgroundfile} \
  306. -n background.png -t raw || \
  307. $err "insert background, ${backgroundfile}"
  308. fi
  309. tmpcfg=$(mktemp -t coreboot_rom.XXXXXXXXXX)
  310. printf "set grub_scan_disk=\"%s\"\n" "$grub_scan_disk" >"$tmpcfg" \
  311. || $err "set grub_scandisk, $grub_scan_disk, $tmpcfg"
  312. [ "${grub_scan_disk}" = "both" ] || \
  313. x_ "$cbfstool" "$tmprom" add -f "$tmpcfg" -n scan.cfg -t raw
  314. printf "set timeout=%s\n" "${grub_timeout}" > "${tmpcfg}" || \
  315. $err "set timeout, ${grub_timeout}, ${tmpcfg}"
  316. [ -z "${grub_timeout}" ] || x_ "${cbfstool}" "${tmprom}" add \
  317. -f "${tmpcfg}" -n timeout.cfg -t raw
  318. x_ rm -f "${tmpcfg}"
  319. for keymapfile in ${keymaps}; do
  320. [ -f "${keymapfile}" ] || continue
  321. keymap="${keymapfile##*/}"
  322. keymap="${keymap%.gkb}"
  323. tmpgrubrom="$(mktemp -t coreboot_rom.XXXXXXXXXX)"
  324. x_ cp "${tmprom}" "${tmpgrubrom}"
  325. x_ "$cbfstool" "$tmpgrubrom" add -f "$keymapfile" \
  326. -n keymap.gkb -t raw
  327. newrom="${romdir}/${payload1}_${board}_${initmode}_"
  328. newrom="${newrom}${displaymode}_${keymap}.rom"
  329. [ "${initmode}" = "normal" ] && \
  330. newrom="${romdir}/${payload1}_${board}_" && \
  331. newrom="${newrom}${initmode}_${keymap}.rom"
  332. x_ copyrom "${tmpgrubrom}" "${newrom}"
  333. if [ "${payload_seabios_grubonly}" = "y" ]; then
  334. x_ "$cbfstool" "$tmpgrubrom" add \
  335. -f "config/grub/bootorder" -n bootorder -t raw
  336. x_ copyrom "$tmpgrubrom" "${newrom%.rom}_grubfirst.rom"
  337. x_ "$cbfstool" "$tmpgrubrom" add-int -i 0 \
  338. -n etc/show-boot-menu
  339. x_ copyrom "$tmpgrubrom" "${newrom%.rom}_grubonly.rom"
  340. fi
  341. x_ rm -f "${tmpgrubrom}"
  342. done
  343. }
  344. # make a rom in /tmp/ and then print the path of that ROM
  345. mkSeabiosRom() {
  346. _cbrom="${1}" # rom to insert seabios in. will not be touched
  347. # (a tmpfile will be made instead)
  348. _seabios_cbfs_path="${2}" # e.g. fallback/payload
  349. _seabioself="elf/seabios/default/${initmode}/bios.bin.elf"
  350. tmprom=$(mktemp -t coreboot_rom.XXXXXXXXXX)
  351. x_ cp "${_cbrom}" "${tmprom}"
  352. x_ "$cbfstool" "$tmprom" add-payload -f "$_seabioself" \
  353. -n "${_seabios_cbfs_path}" -c lzma
  354. x_ "$cbfstool" "$tmprom" add-int -i 3000 -n etc/ps2-keyboard-spinup
  355. z="2"; [ "$initmode" = "vgarom" ] && z="0"
  356. x_ "$cbfstool" "$tmprom" add-int -i $z -n etc/pci-optionrom-exec
  357. x_ "$cbfstool" "$tmprom" add-int -i 0 -n etc/optionroms-checksum
  358. [ "$initmode" != "libgfxinit" ] || \
  359. x_ "$cbfstool" "$tmprom" add -f "$seavgabiosrom" \
  360. -n vgaroms/seavgabios.bin -t raw
  361. printf "%s\n" "${tmprom}"
  362. }
  363. build_uboot_roms()
  364. {
  365. tmprom=$(mkUbootRom "${cbrom}" "fallback/payload") || \
  366. $err "build_uboot_roms $board: could not create tmprom"
  367. newrom="${romdir}/uboot_payload_${board}_${initmode}_${displaymode}.rom"
  368. x_ copyrom "${tmprom}" "${newrom}"
  369. x_ rm -f "${tmprom}"
  370. }
  371. # make a rom in /tmp/ and then print the path of that ROM
  372. mkUbootRom() {
  373. _cbrom="${1}"
  374. _uboot_cbfs_path="${2}"
  375. _ubdir="elf/u-boot/${board}/${uboot_config}"
  376. _ubootelf="${_ubdir}/u-boot.elf"
  377. [ -f "${_ubootelf}" ] || _ubootelf="${_ubdir}/u-boot"
  378. [ -f "$_ubootelf" ] || $err "mkUbootRom: $board: cant find u-boot"
  379. tmprom=$(mktemp -t coreboot_rom.XXXXXXXXXX)
  380. x_ cp "${_cbrom}" "${tmprom}"
  381. x_ "$cbfstool" "$tmprom" add-payload -f "$_ubootelf" \
  382. -n "${_uboot_cbfs_path}" -c lzma
  383. printf "%s\n" "${tmprom}"
  384. }
  385. copyrom()
  386. {
  387. printf "Creating target image: %s\n" "$2"
  388. x_ mkdir -p "${2%/*}"
  389. x_ cp "$1" "$2"
  390. [ "$xbmk_release" = "y" ] && mksha512sum "${2}" "vendorhashes" && \
  391. x_ ./vendor inject -r "${2}" -b "$board" -n nuke; return 0
  392. }
  393. main $@