create-test-plan.py 39 KB

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