create-test-plan.py 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865
  1. #!/usr/bin/env python
  2. import argparse
  3. import dataclasses
  4. import fnmatch
  5. from enum import Enum
  6. import json
  7. import logging
  8. import os
  9. import re
  10. from typing import Optional
  11. logger = logging.getLogger(__name__)
  12. class AppleArch(Enum):
  13. Aarch64 = "aarch64"
  14. X86_64 = "x86_64"
  15. class MsvcArch(Enum):
  16. X86 = "x86"
  17. X64 = "x64"
  18. Arm32 = "arm"
  19. Arm64 = "arm64"
  20. class JobOs(Enum):
  21. WindowsLatest = "windows-latest"
  22. UbuntuLatest = "ubuntu-latest"
  23. MacosLatest = "macos-latest"
  24. Ubuntu20_04 = "ubuntu-20.04"
  25. Ubuntu22_04 = "ubuntu-22.04"
  26. Ubuntu24_04 = "ubuntu-24.04"
  27. Ubuntu24_04_arm = "ubuntu-24.04-arm"
  28. Macos13 = "macos-13"
  29. class SdlPlatform(Enum):
  30. Android = "android"
  31. Emscripten = "emscripten"
  32. Haiku = "haiku"
  33. LoongArch64 = "loongarch64"
  34. Msys2 = "msys2"
  35. Linux = "linux"
  36. MacOS = "macos"
  37. Ios = "ios"
  38. Tvos = "tvos"
  39. Msvc = "msvc"
  40. N3ds = "n3ds"
  41. PowerPC = "powerpc"
  42. PowerPC64 = "powerpc64"
  43. Ps2 = "ps2"
  44. Psp = "psp"
  45. Vita = "vita"
  46. Riscos = "riscos"
  47. FreeBSD = "freebsd"
  48. NetBSD = "netbsd"
  49. class Msys2Platform(Enum):
  50. Mingw32 = "mingw32"
  51. Mingw64 = "mingw64"
  52. Clang32 = "clang32"
  53. Clang64 = "clang64"
  54. Ucrt64 = "ucrt64"
  55. class IntelCompiler(Enum):
  56. Icc = "icc"
  57. Icx = "icx"
  58. class VitaGLES(Enum):
  59. Pib = "pib"
  60. Pvr = "pvr"
  61. @dataclasses.dataclass(slots=True)
  62. class JobSpec:
  63. name: str
  64. os: JobOs
  65. platform: SdlPlatform
  66. artifact: Optional[str]
  67. container: Optional[str] = None
  68. no_cmake: bool = False
  69. xcode: bool = False
  70. android_mk: bool = False
  71. android_gradle: bool = False
  72. lean: bool = False
  73. android_arch: Optional[str] = None
  74. android_abi: Optional[str] = None
  75. android_platform: Optional[int] = None
  76. msys2_platform: Optional[Msys2Platform] = None
  77. intel: Optional[IntelCompiler] = None
  78. apple_framework: Optional[bool] = None
  79. apple_archs: Optional[set[AppleArch]] = None
  80. msvc_project: Optional[str] = None
  81. msvc_arch: Optional[MsvcArch] = None
  82. clang_cl: bool = False
  83. gdk: bool = False
  84. vita_gles: Optional[VitaGLES] = None
  85. JOB_SPECS = {
  86. "msys2-mingw32": JobSpec(name="Windows (msys2, mingw32)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msys2, artifact="SDL-mingw32", msys2_platform=Msys2Platform.Mingw32, ),
  87. "msys2-mingw64": JobSpec(name="Windows (msys2, mingw64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msys2, artifact="SDL-mingw64", msys2_platform=Msys2Platform.Mingw64, ),
  88. "msys2-clang32": JobSpec(name="Windows (msys2, clang32)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msys2, artifact="SDL-mingw32-clang", msys2_platform=Msys2Platform.Clang32, ),
  89. "msys2-clang64": JobSpec(name="Windows (msys2, clang64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msys2, artifact="SDL-mingw64-clang", msys2_platform=Msys2Platform.Clang64, ),
  90. "msys2-ucrt64": JobSpec(name="Windows (msys2, ucrt64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msys2, artifact="SDL-mingw64-ucrt", msys2_platform=Msys2Platform.Ucrt64, ),
  91. "msvc-x64": JobSpec(name="Windows (MSVC, x64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-VC-x64", msvc_arch=MsvcArch.X64, msvc_project="VisualC/SDL.sln", ),
  92. "msvc-x86": JobSpec(name="Windows (MSVC, x86)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-VC-x86", msvc_arch=MsvcArch.X86, msvc_project="VisualC/SDL.sln", ),
  93. "msvc-clang-x64": JobSpec(name="Windows (MSVC, clang-cl x64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-clang-cl-x64", msvc_arch=MsvcArch.X64, clang_cl=True, ),
  94. "msvc-clang-x86": JobSpec(name="Windows (MSVC, clang-cl x86)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-clang-cl-x86", msvc_arch=MsvcArch.X86, clang_cl=True, ),
  95. "msvc-arm32": JobSpec(name="Windows (MSVC, ARM)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-VC-arm32", msvc_arch=MsvcArch.Arm32, ),
  96. "msvc-arm64": JobSpec(name="Windows (MSVC, ARM64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-VC-arm64", msvc_arch=MsvcArch.Arm64, ),
  97. "msvc-gdk-x64": JobSpec(name="GDK (MSVC, x64)", os=JobOs.WindowsLatest, platform=SdlPlatform.Msvc, artifact="SDL-VC-GDK", msvc_arch=MsvcArch.X64, msvc_project="VisualC-GDK/SDL.sln", gdk=True, no_cmake=True, ),
  98. "ubuntu-20.04": JobSpec(name="Ubuntu 20.04", os=JobOs.Ubuntu20_04, platform=SdlPlatform.Linux, artifact="SDL-ubuntu20.04", ),
  99. "ubuntu-22.04": JobSpec(name="Ubuntu 22.04", os=JobOs.Ubuntu22_04, platform=SdlPlatform.Linux, artifact="SDL-ubuntu22.04", ),
  100. "ubuntu-24.04-arm64": JobSpec(name="Ubuntu 24.04 (ARM64)", os=JobOs.Ubuntu24_04_arm, platform=SdlPlatform.Linux, artifact="SDL-ubuntu24.04-arm64", ),
  101. "steamrt-sniper": JobSpec(name="Steam Linux Runtime (Sniper)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Linux, artifact="SDL-slrsniper", container="registry.gitlab.steamos.cloud/steamrt/sniper/sdk:beta", ),
  102. "ubuntu-intel-icx": JobSpec(name="Ubuntu 20.04 (Intel oneAPI)", os=JobOs.Ubuntu20_04, platform=SdlPlatform.Linux, artifact="SDL-ubuntu20.04-oneapi", intel=IntelCompiler.Icx, ),
  103. "ubuntu-intel-icc": JobSpec(name="Ubuntu 20.04 (Intel Compiler)", os=JobOs.Ubuntu20_04, platform=SdlPlatform.Linux, artifact="SDL-ubuntu20.04-icc", intel=IntelCompiler.Icc, ),
  104. "macos-framework-x64": JobSpec(name="MacOS (Framework) (x64)", os=JobOs.Macos13, platform=SdlPlatform.MacOS, artifact="SDL-macos-framework", apple_framework=True, apple_archs={AppleArch.Aarch64, AppleArch.X86_64, }, xcode=True, ),
  105. "macos-framework-arm64": JobSpec(name="MacOS (Framework) (arm64)", os=JobOs.MacosLatest, platform=SdlPlatform.MacOS, artifact=None, apple_framework=True, apple_archs={AppleArch.Aarch64, AppleArch.X86_64, }, ),
  106. "macos-gnu-arm64": JobSpec(name="MacOS (GNU prefix)", os=JobOs.MacosLatest, platform=SdlPlatform.MacOS, artifact="SDL-macos-arm64-gnu", apple_framework=False, apple_archs={AppleArch.Aarch64, }, ),
  107. "ios": JobSpec(name="iOS (CMake & xcode)", os=JobOs.MacosLatest, platform=SdlPlatform.Ios, artifact="SDL-ios-arm64", xcode=True, ),
  108. "tvos": JobSpec(name="tvOS (CMake & xcode)", os=JobOs.MacosLatest, platform=SdlPlatform.Tvos, artifact="SDL-tvos-arm64", xcode=True, ),
  109. "android-cmake": JobSpec(name="Android (CMake)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Android, artifact="SDL-android-arm64", android_abi="arm64-v8a", android_arch="aarch64", android_platform=23, ),
  110. "android-cmake-lean": JobSpec(name="Android (CMake, lean)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Android, artifact="SDL-lean-android-arm64", android_abi="arm64-v8a", android_arch="aarch64", android_platform=23, lean=True, ),
  111. "android-mk": JobSpec(name="Android (Android.mk)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Android, artifact=None, no_cmake=True, android_mk=True, ),
  112. "android-gradle": JobSpec(name="Android (Gradle)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Android, artifact=None, no_cmake=True, android_gradle=True, ),
  113. "emscripten": JobSpec(name="Emscripten", os=JobOs.UbuntuLatest, platform=SdlPlatform.Emscripten, artifact="SDL-emscripten", ),
  114. "haiku": JobSpec(name="Haiku", os=JobOs.UbuntuLatest, platform=SdlPlatform.Haiku, artifact="SDL-haiku-x64", container="ghcr.io/haiku/cross-compiler:x86_64-r1beta5", ),
  115. "loongarch64": JobSpec(name="LoongArch64", os=JobOs.UbuntuLatest, platform=SdlPlatform.LoongArch64, artifact="SDL-loongarch64", ),
  116. "n3ds": JobSpec(name="Nintendo 3DS", os=JobOs.UbuntuLatest, platform=SdlPlatform.N3ds, artifact="SDL-n3ds", container="devkitpro/devkitarm:latest", ),
  117. "ppc": JobSpec(name="PowerPC", os=JobOs.UbuntuLatest, platform=SdlPlatform.PowerPC, artifact="SDL-ppc", container="dockcross/linux-ppc:latest", ),
  118. "ppc64": JobSpec(name="PowerPC64", os=JobOs.UbuntuLatest, platform=SdlPlatform.PowerPC64, artifact="SDL-ppc64le", container="dockcross/linux-ppc64le:latest", ),
  119. "ps2": JobSpec(name="Sony PlayStation 2", os=JobOs.UbuntuLatest, platform=SdlPlatform.Ps2, artifact="SDL-ps2", container="ps2dev/ps2dev:latest", ),
  120. "psp": JobSpec(name="Sony PlayStation Portable", os=JobOs.UbuntuLatest, platform=SdlPlatform.Psp, artifact="SDL-psp", container="pspdev/pspdev:latest", ),
  121. "vita-pib": JobSpec(name="Sony PlayStation Vita (GLES w/ pib)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Vita, artifact="SDL-vita-pib", container="vitasdk/vitasdk:latest", vita_gles=VitaGLES.Pib, ),
  122. "vita-pvr": JobSpec(name="Sony PlayStation Vita (GLES w/ PVR_PSP2)", os=JobOs.UbuntuLatest, platform=SdlPlatform.Vita, artifact="SDL-vita-pvr", container="vitasdk/vitasdk:latest", vita_gles=VitaGLES.Pvr, ),
  123. "riscos": JobSpec(name="RISC OS", os=JobOs.UbuntuLatest, platform=SdlPlatform.Riscos, artifact="SDL-riscos", container="riscosdotinfo/riscos-gccsdk-4.7:latest", ),
  124. "netbsd": JobSpec(name="NetBSD", os=JobOs.UbuntuLatest, platform=SdlPlatform.NetBSD, artifact="SDL-netbsd-x64", ),
  125. "freebsd": JobSpec(name="FreeBSD", os=JobOs.UbuntuLatest, platform=SdlPlatform.FreeBSD, artifact="SDL-freebsd-x64", ),
  126. }
  127. class StaticLibType(Enum):
  128. MSVC = "SDL3-static.lib"
  129. A = "libSDL3.a"
  130. class SharedLibType(Enum):
  131. WIN32 = "SDL3.dll"
  132. SO_0 = "libSDL3.so.0"
  133. SO = "libSDL3.so"
  134. DYLIB = "libSDL3.0.dylib"
  135. FRAMEWORK = "SDL3.framework/Versions/A/SDL3"
  136. @dataclasses.dataclass(slots=True)
  137. class JobDetails:
  138. name: str
  139. key: str
  140. os: str
  141. platform: str
  142. artifact: str
  143. no_cmake: bool
  144. ccache: bool = False
  145. build_tests: bool = True
  146. container: str = ""
  147. cmake_build_type: str = "RelWithDebInfo"
  148. shell: str = "sh"
  149. sudo: str = "sudo"
  150. cmake_config_emulator: str = ""
  151. apk_packages: list[str] = dataclasses.field(default_factory=list)
  152. apt_packages: list[str] = dataclasses.field(default_factory=list)
  153. brew_packages: list[str] = dataclasses.field(default_factory=list)
  154. cmake_toolchain_file: str = ""
  155. cmake_arguments: list[str] = dataclasses.field(default_factory=list)
  156. cmake_build_arguments: list[str] = dataclasses.field(default_factory=list)
  157. clang_tidy: bool = True
  158. cppflags: list[str] = dataclasses.field(default_factory=list)
  159. cc: str = ""
  160. cxx: str = ""
  161. cflags: list[str] = dataclasses.field(default_factory=list)
  162. cxxflags: list[str] = dataclasses.field(default_factory=list)
  163. ldflags: list[str] = dataclasses.field(default_factory=list)
  164. pollute_directories: list[str] = dataclasses.field(default_factory=list)
  165. use_cmake: bool = True
  166. shared: bool = True
  167. static: bool = True
  168. shared_lib: Optional[SharedLibType] = None
  169. static_lib: Optional[StaticLibType] = None
  170. run_tests: bool = True
  171. test_pkg_config: bool = True
  172. cc_from_cmake: bool = False
  173. source_cmd: str = ""
  174. pretest_cmd: str = ""
  175. java: bool = False
  176. android_apks: list[str] = dataclasses.field(default_factory=list)
  177. android_ndk: bool = False
  178. android_mk: bool = False
  179. android_gradle: bool = False
  180. minidump: bool = False
  181. intel: bool = False
  182. msys2_msystem: str = ""
  183. msys2_env: str = ""
  184. msys2_no_perl: bool = False
  185. werror: bool = True
  186. msvc_vcvars_arch: str = ""
  187. msvc_vcvars_sdk: str = ""
  188. msvc_project: str = ""
  189. msvc_project_flags: list[str] = dataclasses.field(default_factory=list)
  190. setup_ninja: bool = False
  191. setup_libusb_arch: str = ""
  192. xcode_sdk: str = ""
  193. cpactions: bool = False
  194. setup_gdk_folder: str = ""
  195. cpactions_os: str = ""
  196. cpactions_version: str = ""
  197. cpactions_arch: str = ""
  198. cpactions_setup_cmd: str = ""
  199. cpactions_install_cmd: str = ""
  200. setup_vita_gles_type: str = ""
  201. check_sources: bool = False
  202. setup_python: bool = False
  203. pypi_packages: list[str] = dataclasses.field(default_factory=list)
  204. def to_workflow(self, enable_artifacts: bool) -> dict[str, str|bool]:
  205. data = {
  206. "name": self.name,
  207. "key": self.key,
  208. "os": self.os,
  209. "ccache": self.ccache,
  210. "container": self.container if self.container else "",
  211. "platform": self.platform,
  212. "artifact": self.artifact,
  213. "enable-artifacts": enable_artifacts,
  214. "shell": self.shell,
  215. "msys2-msystem": self.msys2_msystem,
  216. "msys2-env": self.msys2_env,
  217. "msys2-no-perl": self.msys2_no_perl,
  218. "android-ndk": self.android_ndk,
  219. "java": self.java,
  220. "intel": self.intel,
  221. "apk-packages": my_shlex_join(self.apk_packages),
  222. "apt-packages": my_shlex_join(self.apt_packages),
  223. "test-pkg-config": self.test_pkg_config,
  224. "brew-packages": my_shlex_join(self.brew_packages),
  225. "pollute-directories": my_shlex_join(self.pollute_directories),
  226. "no-cmake": self.no_cmake,
  227. "build-tests": self.build_tests,
  228. "source-cmd": self.source_cmd,
  229. "pretest-cmd": self.pretest_cmd,
  230. "cmake-config-emulator": self.cmake_config_emulator,
  231. "cc": self.cc,
  232. "cxx": self.cxx,
  233. "cflags": my_shlex_join(self.cppflags + self.cflags),
  234. "cxxflags": my_shlex_join(self.cppflags + self.cxxflags),
  235. "ldflags": my_shlex_join(self.ldflags),
  236. "cmake-toolchain-file": self.cmake_toolchain_file,
  237. "clang-tidy": self.clang_tidy,
  238. "cmake-arguments": my_shlex_join(self.cmake_arguments),
  239. "cmake-build-arguments": my_shlex_join(self.cmake_build_arguments),
  240. "shared": self.shared,
  241. "static": self.static,
  242. "shared-lib": self.shared_lib.value if self.shared_lib else None,
  243. "static-lib": self.static_lib.value if self.static_lib else None,
  244. "cmake-build-type": self.cmake_build_type,
  245. "run-tests": self.run_tests,
  246. "android-apks": my_shlex_join(self.android_apks),
  247. "android-gradle": self.android_gradle,
  248. "android-mk": self.android_mk,
  249. "werror": self.werror,
  250. "sudo": self.sudo,
  251. "msvc-vcvars-arch": self.msvc_vcvars_arch,
  252. "msvc-vcvars-sdk": self.msvc_vcvars_sdk,
  253. "msvc-project": self.msvc_project,
  254. "msvc-project-flags": my_shlex_join(self.msvc_project_flags),
  255. "setup-ninja": self.setup_ninja,
  256. "setup-libusb-arch": self.setup_libusb_arch,
  257. "cc-from-cmake": self.cc_from_cmake,
  258. "xcode-sdk": self.xcode_sdk,
  259. "cpactions": self.cpactions,
  260. "cpactions-os": self.cpactions_os,
  261. "cpactions-version": self.cpactions_version,
  262. "cpactions-arch": self.cpactions_arch,
  263. "cpactions-setup-cmd": self.cpactions_setup_cmd,
  264. "cpactions-install-cmd": self.cpactions_install_cmd,
  265. "setup-vita-gles-type": self.setup_vita_gles_type,
  266. "setup-gdk-folder": self.setup_gdk_folder,
  267. "check-sources": self.check_sources,
  268. "setup-python": self.setup_python,
  269. "pypi-packages": my_shlex_join(self.pypi_packages),
  270. }
  271. return {k: v for k, v in data.items() if v != ""}
  272. def my_shlex_join(s):
  273. def escape(s):
  274. if s[:1] == "'" and s[-1:] == "'":
  275. return s
  276. if set(s).intersection(set("; \t")):
  277. return f'"{s}"'
  278. return s
  279. return " ".join(escape(s))
  280. def spec_to_job(spec: JobSpec, key: str, trackmem_symbol_names: bool) -> JobDetails:
  281. job = JobDetails(
  282. name=spec.name,
  283. key=key,
  284. os=spec.os.value,
  285. artifact=spec.artifact or "",
  286. container=spec.container or "",
  287. platform=spec.platform.value,
  288. sudo="sudo",
  289. no_cmake=spec.no_cmake,
  290. )
  291. if job.os.startswith("ubuntu"):
  292. job.apt_packages.extend([
  293. "ninja-build",
  294. "pkg-config",
  295. ])
  296. pretest_cmd = []
  297. if trackmem_symbol_names:
  298. pretest_cmd.append("export SDL_TRACKMEM_SYMBOL_NAMES=1")
  299. else:
  300. pretest_cmd.append("export SDL_TRACKMEM_SYMBOL_NAMES=0")
  301. win32 = spec.platform in (SdlPlatform.Msys2, SdlPlatform.Msvc)
  302. fpic = None
  303. build_parallel = True
  304. if spec.lean:
  305. job.cppflags.append("-DSDL_LEAN_AND_MEAN=1")
  306. if win32:
  307. job.cmake_arguments.append("-DSDLTEST_PROCDUMP=ON")
  308. job.minidump = True
  309. if spec.intel is not None:
  310. match spec.intel:
  311. case IntelCompiler.Icx:
  312. job.cc = "icx"
  313. job.cxx = "icpx"
  314. case IntelCompiler.Icc:
  315. job.cc = "icc"
  316. job.cxx = "icpc"
  317. job.cppflags.append("-diag-disable=10441")
  318. job.clang_tidy = False
  319. case _:
  320. raise ValueError(f"Invalid intel={spec.intel}")
  321. job.source_cmd = f"source /opt/intel/oneapi/setvars.sh;"
  322. job.intel = True
  323. job.shell = "bash"
  324. job.cmake_arguments.extend((
  325. f"-DCMAKE_C_COMPILER={job.cc}",
  326. f"-DCMAKE_CXX_COMPILER={job.cxx}",
  327. "-DCMAKE_SYSTEM_NAME=Linux",
  328. ))
  329. match spec.platform:
  330. case SdlPlatform.Msvc:
  331. job.setup_ninja = not spec.gdk
  332. job.clang_tidy = False # complains about \threadsafety: "unknown command tag name [clang-diagnostic-documentation-unknown-command]"
  333. job.msvc_project = spec.msvc_project if spec.msvc_project else ""
  334. job.msvc_project_flags.append("-p:TreatWarningsAsError=true")
  335. job.test_pkg_config = False
  336. job.shared_lib = SharedLibType.WIN32
  337. job.static_lib = StaticLibType.MSVC
  338. job.cmake_arguments.extend((
  339. "-DCMAKE_MSVC_DEBUG_INFORMATION_FORMAT=ProgramDatabase",
  340. "-DCMAKE_EXE_LINKER_FLAGS=-DEBUG",
  341. "-DCMAKE_SHARED_LINKER_FLAGS=-DEBUG",
  342. ))
  343. job.cmake_arguments.append("'-DCMAKE_MSVC_RUNTIME_LIBRARY=MultiThreaded$<$<CONFIG:Debug>:Debug>'")
  344. if spec.clang_cl:
  345. job.cmake_arguments.extend((
  346. "-DCMAKE_C_COMPILER=clang-cl",
  347. "-DCMAKE_CXX_COMPILER=clang-cl",
  348. ))
  349. match spec.msvc_arch:
  350. case MsvcArch.X86:
  351. job.cflags.append("/clang:-m32")
  352. job.ldflags.append("/MACHINE:X86")
  353. case MsvcArch.X64:
  354. job.cflags.append("/clang:-m64")
  355. job.ldflags.append("/MACHINE:X64")
  356. case _:
  357. raise ValueError(f"Unsupported clang-cl architecture (arch={spec.msvc_arch})")
  358. if spec.msvc_project:
  359. match spec.msvc_arch:
  360. case MsvcArch.X86:
  361. msvc_platform = "Win32"
  362. case MsvcArch.X64:
  363. msvc_platform = "x64"
  364. case _:
  365. raise ValueError(f"Unsupported vcxproj architecture (arch={spec.msvc_arch})")
  366. if spec.gdk:
  367. msvc_platform = f"Gaming.Desktop.{msvc_platform}"
  368. job.msvc_project_flags.append(f"-p:Platform={msvc_platform}")
  369. match spec.msvc_arch:
  370. case MsvcArch.X86:
  371. job.msvc_vcvars_arch = "x64_x86"
  372. case MsvcArch.X64:
  373. job.msvc_vcvars_arch = "x64"
  374. case MsvcArch.Arm32:
  375. job.msvc_vcvars_arch = "x64_arm"
  376. job.msvc_vcvars_sdk = "10.0.22621.0" # 10.0.26100.0 dropped ARM32 um and ucrt libraries
  377. job.run_tests = False
  378. case MsvcArch.Arm64:
  379. job.msvc_vcvars_arch = "x64_arm64"
  380. job.run_tests = False
  381. if spec.gdk:
  382. job.setup_gdk_folder = "VisualC-GDK"
  383. else:
  384. match spec.msvc_arch:
  385. case MsvcArch.X86:
  386. job.setup_libusb_arch = "x86"
  387. case MsvcArch.X64:
  388. job.setup_libusb_arch = "x64"
  389. case SdlPlatform.Linux:
  390. if spec.name.startswith("Ubuntu"):
  391. assert spec.os.value.startswith("ubuntu-")
  392. job.apt_packages.extend((
  393. "ccache",
  394. "gnome-desktop-testing",
  395. "libasound2-dev",
  396. "libpulse-dev",
  397. "libaudio-dev",
  398. "libjack-dev",
  399. "libsndio-dev",
  400. "libusb-1.0-0-dev",
  401. "libx11-dev",
  402. "libxext-dev",
  403. "libxrandr-dev",
  404. "libxcursor-dev",
  405. "libxfixes-dev",
  406. "libxi-dev",
  407. "libxss-dev",
  408. "libwayland-dev",
  409. "libxkbcommon-dev",
  410. "libdrm-dev",
  411. "libgbm-dev",
  412. "libgl1-mesa-dev",
  413. "libgles2-mesa-dev",
  414. "libegl1-mesa-dev",
  415. "libdbus-1-dev",
  416. "libibus-1.0-dev",
  417. "libudev-dev",
  418. "fcitx-libs-dev",
  419. ))
  420. match = re.match(r"ubuntu-(?P<year>[0-9]+)\.(?P<month>[0-9]+).*", spec.os.value)
  421. ubuntu_year, ubuntu_month = [int(match["year"]), int(match["month"])]
  422. if ubuntu_year >= 22:
  423. job.apt_packages.extend(("libpipewire-0.3-dev", "libdecor-0-dev"))
  424. job.apt_packages.extend((
  425. "libunwind-dev", # For SDL_test memory tracking
  426. ))
  427. job.ccache = True
  428. if trackmem_symbol_names:
  429. # older libunwind is slow
  430. job.cmake_arguments.append("-DSDLTEST_TIMEOUT_MULTIPLIER=2")
  431. job.shared_lib = SharedLibType.SO_0
  432. job.static_lib = StaticLibType.A
  433. fpic = True
  434. case SdlPlatform.Ios | SdlPlatform.Tvos:
  435. job.brew_packages.extend([
  436. "ccache",
  437. "ninja",
  438. ])
  439. job.ccache = True
  440. job.clang_tidy = False
  441. job.run_tests = False
  442. job.test_pkg_config = False
  443. job.shared_lib = SharedLibType.DYLIB
  444. job.static_lib = StaticLibType.A
  445. match spec.platform:
  446. case SdlPlatform.Ios:
  447. if spec.xcode:
  448. job.xcode_sdk = 'iphoneos'
  449. job.cmake_arguments.extend([
  450. "-DCMAKE_SYSTEM_NAME=iOS",
  451. "-DCMAKE_OSX_ARCHITECTURES=\"arm64\"",
  452. "-DCMAKE_OSX_DEPLOYMENT_TARGET=11.0",
  453. ])
  454. case SdlPlatform.Tvos:
  455. if spec.xcode:
  456. job.xcode_sdk = 'appletvos'
  457. job.cmake_arguments.extend([
  458. "-DCMAKE_SYSTEM_NAME=tvOS",
  459. "-DCMAKE_OSX_ARCHITECTURES=\"arm64\"",
  460. "-DCMAKE_OSX_DEPLOYMENT_TARGET=11.0",
  461. ])
  462. case SdlPlatform.MacOS:
  463. if spec.apple_framework:
  464. job.static = False
  465. job.clang_tidy = False
  466. job.test_pkg_config = False
  467. job.cmake_arguments.extend((
  468. "'-DCMAKE_OSX_ARCHITECTURES=x86_64;arm64'",
  469. "-DCMAKE_OSX_DEPLOYMENT_TARGET=10.13",
  470. "-DSDL_FRAMEWORK=ON",
  471. ))
  472. job.shared_lib = SharedLibType.FRAMEWORK
  473. else:
  474. job.clang_tidy = True
  475. job.cmake_arguments.extend((
  476. "-DCMAKE_OSX_ARCHITECTURES=arm64",
  477. "-DCMAKE_OSX_DEPLOYMENT_TARGET=10.13",
  478. "-DCLANG_TIDY_BINARY=$(brew --prefix llvm)/bin/clang-tidy",
  479. ))
  480. job.shared_lib = SharedLibType.DYLIB
  481. job.static_lib = StaticLibType.A
  482. job.ccache = True
  483. job.apt_packages = []
  484. job.brew_packages.extend((
  485. "ccache",
  486. "ninja",
  487. ))
  488. if job.clang_tidy:
  489. job.brew_packages.append("llvm")
  490. if spec.xcode:
  491. job.xcode_sdk = "macosx"
  492. case SdlPlatform.Android:
  493. job.android_gradle = spec.android_gradle
  494. job.android_mk = spec.android_mk
  495. job.apt_packages.append("ccache")
  496. job.run_tests = False
  497. job.shared_lib = SharedLibType.SO
  498. job.static_lib = StaticLibType.A
  499. if spec.android_mk or not spec.no_cmake:
  500. job.android_ndk = True
  501. if spec.android_gradle or not spec.no_cmake:
  502. job.java = True
  503. if spec.android_mk or spec.android_gradle:
  504. job.apt_packages = []
  505. if not spec.no_cmake:
  506. job.ccache = True
  507. job.cmake_arguments.extend((
  508. f"-DANDROID_PLATFORM={spec.android_platform}",
  509. f"-DANDROID_ABI={spec.android_abi}",
  510. ))
  511. job.cmake_toolchain_file = "${ANDROID_NDK_HOME}/build/cmake/android.toolchain.cmake"
  512. job.cc = f"${{ANDROID_NDK_HOME}}/toolchains/llvm/prebuilt/linux-x86_64/bin/clang --target={spec.android_arch}-none-linux-androideabi{spec.android_platform}"
  513. job.android_apks = [
  514. "testaudiorecording-apk",
  515. "testautomation-apk",
  516. "testcontroller-apk",
  517. "testmultiaudio-apk",
  518. "testsprite-apk",
  519. ]
  520. case SdlPlatform.Emscripten:
  521. job.clang_tidy = False # clang-tidy does not understand -gsource-map
  522. job.shared = False
  523. job.ccache = True
  524. job.apt_packages.append("ccache")
  525. job.cmake_config_emulator = "emcmake"
  526. job.cmake_build_type = "Debug"
  527. job.test_pkg_config = False
  528. job.cmake_arguments.extend((
  529. "-DSDLTEST_BROWSER=chrome",
  530. "-DSDLTEST_TIMEOUT_MULTIPLIER=4",
  531. "-DSDLTEST_CHROME_BINARY=${CHROME_BINARY}",
  532. ))
  533. job.cflags.extend((
  534. "-gsource-map",
  535. "-ffile-prefix-map=${PWD}=/SDL",
  536. ))
  537. job.ldflags.extend((
  538. "--source-map-base", "/",
  539. ))
  540. pretest_cmd.extend((
  541. "# Start local HTTP server",
  542. "cmake --build build --target serve-sdl-tests --verbose &",
  543. "chrome --version",
  544. "chromedriver --version",
  545. ))
  546. job.static_lib = StaticLibType.A
  547. job.setup_python = True
  548. job.pypi_packages.append("selenium")
  549. case SdlPlatform.Ps2:
  550. job.ccache = False # actions/ccache does not work in psp container (incompatible tar of busybox)
  551. build_parallel = False
  552. job.shared = False
  553. job.sudo = ""
  554. job.apt_packages = []
  555. job.apk_packages = ["ccache", "cmake", "gmp", "mpc1", "mpfr4", "ninja", "pkgconf", "git", ]
  556. job.cmake_toolchain_file = "${PS2DEV}/ps2sdk/ps2dev.cmake"
  557. job.clang_tidy = False
  558. job.run_tests = False
  559. job.shared = False
  560. job.cc = "mips64r5900el-ps2-elf-gcc"
  561. job.ldflags = ["-L${PS2DEV}/ps2sdk/ee/lib", "-L${PS2DEV}/gsKit/lib", "-L${PS2DEV}/ps2sdk/ports/lib", ]
  562. job.static_lib = StaticLibType.A
  563. case SdlPlatform.Psp:
  564. job.ccache = False # actions/ccache does not work in psp container (incompatible tar of busybox)
  565. build_parallel = False
  566. job.sudo = ""
  567. job.apt_packages = []
  568. job.apk_packages = ["ccache", "cmake", "gmp", "mpc1", "mpfr4", "ninja", "pkgconf", ]
  569. job.cmake_toolchain_file = "${PSPDEV}/psp/share/pspdev.cmake"
  570. job.clang_tidy = False
  571. job.run_tests = False
  572. job.shared = False
  573. job.cc = "psp-gcc"
  574. job.ldflags = ["-L${PSPDEV}/lib", "-L${PSPDEV}/psp/lib", "-L${PSPDEV}/psp/sdk/lib", ]
  575. job.pollute_directories = ["${PSPDEV}/include", "${PSPDEV}/psp/include", "${PSPDEV}/psp/sdk/include", ]
  576. job.static_lib = StaticLibType.A
  577. case SdlPlatform.Vita:
  578. job.ccache = True
  579. job.sudo = ""
  580. job.apt_packages = []
  581. job.apk_packages = ["ccache", "cmake", "ninja", "pkgconf", "bash", "tar"]
  582. job.cmake_toolchain_file = "${VITASDK}/share/vita.toolchain.cmake"
  583. assert spec.vita_gles is not None
  584. job.setup_vita_gles_type = {
  585. VitaGLES.Pib: "pib",
  586. VitaGLES.Pvr: "pvr",
  587. }[spec.vita_gles]
  588. job.cmake_arguments.extend((
  589. f"-DVIDEO_VITA_PIB={ 'true' if spec.vita_gles == VitaGLES.Pib else 'false' }",
  590. f"-DVIDEO_VITA_PVR={ 'true' if spec.vita_gles == VitaGLES.Pvr else 'false' }",
  591. "-DSDL_ARMNEON=ON",
  592. "-DSDL_ARMSIMD=ON",
  593. ))
  594. # Fix vita.toolchain.cmake (https://github.com/vitasdk/vita-toolchain/pull/253)
  595. job.source_cmd = r"""sed -i -E "s#set\\( PKG_CONFIG_EXECUTABLE \"\\$\\{VITASDK}/bin/arm-vita-eabi-pkg-config\" \\)#set\\( PKG_CONFIG_EXECUTABLE \"${VITASDK}/bin/arm-vita-eabi-pkg-config\" CACHE PATH \"Path of pkg-config executable\" \\)#" ${VITASDK}/share/vita.toolchain.cmake"""
  596. job.clang_tidy = False
  597. job.run_tests = False
  598. job.shared = False
  599. job.cc = "arm-vita-eabi-gcc"
  600. job.static_lib = StaticLibType.A
  601. case SdlPlatform.Haiku:
  602. job.ccache = True
  603. fpic = False
  604. job.run_tests = False
  605. job.apt_packages.append("ccache")
  606. job.cc = "x86_64-unknown-haiku-gcc"
  607. job.cxx = "x86_64-unknown-haiku-g++"
  608. job.sudo = ""
  609. job.cmake_arguments.extend((
  610. f"-DCMAKE_C_COMPILER={job.cc}",
  611. f"-DCMAKE_CXX_COMPILER={job.cxx}",
  612. "-DSDL_UNIX_CONSOLE_BUILD=ON",
  613. ))
  614. job.shared_lib = SharedLibType.SO_0
  615. job.static_lib = StaticLibType.A
  616. case SdlPlatform.PowerPC64 | SdlPlatform.PowerPC:
  617. job.ccache = True
  618. # FIXME: Enable SDL_WERROR
  619. job.werror = False
  620. job.clang_tidy = False
  621. job.run_tests = False
  622. job.sudo = ""
  623. job.apt_packages = ["ccache"]
  624. job.shared_lib = SharedLibType.SO_0
  625. job.static_lib = StaticLibType.A
  626. job.cmake_arguments.extend((
  627. "-DSDL_UNIX_CONSOLE_BUILD=ON",
  628. ))
  629. case SdlPlatform.LoongArch64:
  630. job.ccache = True
  631. fpic = True
  632. job.run_tests = False
  633. job.apt_packages.append("ccache")
  634. job.cc = "${LOONGARCH64_CC}"
  635. job.cxx = "${LOONGARCH64_CXX}"
  636. job.cmake_arguments.extend((
  637. f"-DCMAKE_C_COMPILER={job.cc}",
  638. f"-DCMAKE_CXX_COMPILER={job.cxx}",
  639. "-DSDL_UNIX_CONSOLE_BUILD=ON",
  640. "-DCMAKE_SYSTEM_NAME=Linux",
  641. ))
  642. job.shared_lib = SharedLibType.SO_0
  643. job.static_lib = StaticLibType.A
  644. case SdlPlatform.N3ds:
  645. job.ccache = True
  646. job.shared = False
  647. job.apt_packages = ["ccache", "ninja-build", "binutils"]
  648. job.clang_tidy = False
  649. job.run_tests = False
  650. job.cc_from_cmake = True
  651. job.cmake_toolchain_file = "${DEVKITPRO}/cmake/3DS.cmake"
  652. job.static_lib = StaticLibType.A
  653. case SdlPlatform.Msys2:
  654. job.ccache = True
  655. job.shell = "msys2 {0}"
  656. assert spec.msys2_platform
  657. job.msys2_msystem = spec.msys2_platform.value
  658. job.msys2_env = {
  659. "mingw32": "mingw-w64-i686",
  660. "mingw64": "mingw-w64-x86_64",
  661. "clang32": "mingw-w64-clang-i686",
  662. "clang64": "mingw-w64-clang-x86_64",
  663. "ucrt64": "mingw-w64-ucrt-x86_64",
  664. }[spec.msys2_platform.value]
  665. job.msys2_no_perl = spec.msys2_platform in (Msys2Platform.Mingw32, Msys2Platform.Clang32)
  666. job.shared_lib = SharedLibType.WIN32
  667. job.static_lib = StaticLibType.A
  668. case SdlPlatform.Riscos:
  669. job.ccache = False # FIXME: enable when container gets upgrade
  670. # FIXME: Enable SDL_WERROR
  671. job.werror = False
  672. job.apt_packages = ["ccache", "cmake", "ninja-build"]
  673. job.test_pkg_config = False
  674. job.shared = False
  675. job.run_tests = False
  676. job.sudo = ""
  677. job.cmake_arguments.extend((
  678. "-DRISCOS:BOOL=ON",
  679. "-DCMAKE_DISABLE_PRECOMPILE_HEADERS:BOOL=ON",
  680. "-DSDL_GCC_ATOMICS:BOOL=OFF",
  681. ))
  682. job.cmake_toolchain_file = "/home/riscos/env/toolchain-riscos.cmake"
  683. job.static_lib = StaticLibType.A
  684. case SdlPlatform.FreeBSD | SdlPlatform.NetBSD:
  685. job.cpactions = True
  686. job.no_cmake = True
  687. job.run_tests = False
  688. job.apt_packages = []
  689. job.shared_lib = SharedLibType.SO_0
  690. job.static_lib = StaticLibType.A
  691. match spec.platform:
  692. case SdlPlatform.FreeBSD:
  693. job.cpactions_os = "freebsd"
  694. job.cpactions_version = "14.2"
  695. job.cpactions_arch = "x86-64"
  696. job.cpactions_setup_cmd = "sudo pkg update"
  697. job.cpactions_install_cmd = "sudo pkg install -y cmake ninja pkgconf libXcursor libXext libXinerama libXi libXfixes libXrandr libXScrnSaver libXxf86vm wayland wayland-protocols libxkbcommon mesa-libs libglvnd evdev-proto libinotify alsa-lib jackit pipewire pulseaudio sndio dbus zh-fcitx ibus libudev-devd"
  698. job.cmake_arguments.extend((
  699. "-DSDL_CHECK_REQUIRED_INCLUDES=/usr/local/include",
  700. "-DSDL_CHECK_REQUIRED_LINK_OPTIONS=-L/usr/local/lib",
  701. ))
  702. case SdlPlatform.NetBSD:
  703. job.cpactions_os = "netbsd"
  704. job.cpactions_version = "10.1"
  705. job.cpactions_arch = "x86-64"
  706. job.cpactions_setup_cmd = "export PATH=\"/usr/pkg/sbin:/usr/pkg/bin:/sbin:$PATH\"; export PKG_CONFIG_PATH=\"/usr/pkg/lib/pkgconfig\";export PKG_PATH=\"https://cdn.netBSD.org/pub/pkgsrc/packages/NetBSD/$(uname -p)/$(uname -r|cut -f \"1 2\" -d.)/All/\";echo \"PKG_PATH=$PKG_PATH\";echo \"uname -a -> \"$(uname -a)\"\";sudo -E sysctl -w security.pax.aslr.enabled=0;sudo -E sysctl -w security.pax.aslr.global=0;sudo -E pkgin clean;sudo -E pkgin update"
  707. job.cpactions_install_cmd = "sudo -E pkgin -y install cmake dbus pkgconf ninja-build pulseaudio libxkbcommon wayland wayland-protocols libinotify libusb1"
  708. case _:
  709. raise ValueError(f"Unsupported platform={spec.platform}")
  710. if "ubuntu" in spec.name.lower():
  711. job.check_sources = True
  712. job.setup_python = True
  713. if job.ccache:
  714. job.cmake_arguments.extend((
  715. "-DCMAKE_C_COMPILER_LAUNCHER=ccache",
  716. "-DCMAKE_CXX_COMPILER_LAUNCHER=ccache",
  717. ))
  718. if not build_parallel:
  719. job.cmake_build_arguments.append("-j1")
  720. if job.cflags:
  721. job.cmake_arguments.append(f"-DCMAKE_C_FLAGS=\"{my_shlex_join(job.cflags)}\"")
  722. if job.cxxflags:
  723. job.cmake_arguments.append(f"-DCMAKE_CXX_FLAGS=\"{my_shlex_join(job.cxxflags)}\"")
  724. if job.ldflags:
  725. job.cmake_arguments.append(f"-DCMAKE_SHARED_LINKER_FLAGS=\"{my_shlex_join(job.ldflags)}\"")
  726. job.cmake_arguments.append(f"-DCMAKE_EXE_LINKER_FLAGS=\"{my_shlex_join(job.ldflags)}\"")
  727. job.pretest_cmd = "\n".join(pretest_cmd)
  728. def tf(b):
  729. return "ON" if b else "OFF"
  730. if fpic is not None:
  731. job.cmake_arguments.append(f"-DCMAKE_POSITION_INDEPENDENT_CODE={tf(fpic)}")
  732. if job.no_cmake:
  733. job.cmake_arguments = []
  734. return job
  735. def spec_to_platform(spec: JobSpec, key: str, enable_artifacts: bool, trackmem_symbol_names: bool) -> dict[str, str|bool]:
  736. logger.info("spec=%r", spec)
  737. job = spec_to_job(spec, key=key, trackmem_symbol_names=trackmem_symbol_names)
  738. logger.info("job=%r", job)
  739. platform = job.to_workflow(enable_artifacts=enable_artifacts)
  740. logger.info("platform=%r", platform)
  741. return platform
  742. def main():
  743. parser = argparse.ArgumentParser(allow_abbrev=False)
  744. parser.add_argument("--github-variable-prefix", default="platforms")
  745. parser.add_argument("--github-ci", action="store_true")
  746. parser.add_argument("--verbose", action="store_true")
  747. parser.add_argument("--commit-message-file")
  748. parser.add_argument("--no-artifact", dest="enable_artifacts", action="store_false")
  749. parser.add_argument("--trackmem-symbol-names", dest="trackmem_symbol_names", action="store_true")
  750. args = parser.parse_args()
  751. logging.basicConfig(level=logging.INFO if args.verbose else logging.WARNING)
  752. remaining_keys = set(JOB_SPECS.keys())
  753. all_level_keys = (
  754. # Level 1
  755. (
  756. "haiku",
  757. ),
  758. )
  759. filters = []
  760. if args.commit_message_file:
  761. with open(args.commit_message_file, "r") as f:
  762. commit_message = f.read()
  763. for m in re.finditer(r"\[sdl-ci-filter (.*)]", commit_message, flags=re.M):
  764. filters.append(m.group(1).strip(" \t\n\r\t'\""))
  765. if re.search(r"\[sdl-ci-artifacts?]", commit_message, flags=re.M):
  766. args.enable_artifacts = True
  767. if re.search(r"\[sdl-ci-(full-)?trackmem(-symbol-names)?]", commit_message, flags=re.M):
  768. args.trackmem_symbol_names = True
  769. if not filters:
  770. filters.append("*")
  771. logger.info("filters: %r", filters)
  772. all_level_platforms = {}
  773. all_platforms = {key: spec_to_platform(spec, key=key, enable_artifacts=args.enable_artifacts, trackmem_symbol_names=args.trackmem_symbol_names) for key, spec in JOB_SPECS.items()}
  774. for level_i, level_keys in enumerate(all_level_keys, 1):
  775. level_key = f"level{level_i}"
  776. logger.info("Level %d: keys=%r", level_i, level_keys)
  777. assert all(k in remaining_keys for k in level_keys)
  778. level_platforms = tuple(all_platforms[key] for key in level_keys)
  779. remaining_keys.difference_update(level_keys)
  780. all_level_platforms[level_key] = level_platforms
  781. logger.info("=" * 80)
  782. logger.info("Keys before filter: %r", remaining_keys)
  783. filtered_remaining_keys = set()
  784. for filter in filters:
  785. filtered_remaining_keys.update(fnmatch.filter(remaining_keys, filter))
  786. logger.info("Keys after filter: %r", filtered_remaining_keys)
  787. remaining_keys = filtered_remaining_keys
  788. logger.info("Remaining: %r", remaining_keys)
  789. all_level_platforms["others"] = tuple(all_platforms[key] for key in remaining_keys)
  790. if args.github_ci:
  791. for level, platforms in all_level_platforms.items():
  792. platforms_json = json.dumps(platforms)
  793. txt = f"{args.github_variable_prefix}-{level}={platforms_json}"
  794. logger.info("%s", txt)
  795. if "GITHUB_OUTPUT" in os.environ:
  796. with open(os.environ["GITHUB_OUTPUT"], "a") as f:
  797. f.write(txt)
  798. f.write("\n")
  799. else:
  800. logger.warning("GITHUB_OUTPUT not defined")
  801. return 0
  802. if __name__ == "__main__":
  803. raise SystemExit(main())