create-test-plan.py 41 KB

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