detect.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. import methods
  2. import os
  3. # To match other platforms
  4. STACK_SIZE = 8388608
  5. def is_active():
  6. return True
  7. def get_name():
  8. return "Windows"
  9. def can_build():
  10. if os.name == "nt":
  11. # Building natively on Windows
  12. # If VCINSTALLDIR is set in the OS environ, use traditional Godot logic to set up MSVC
  13. if os.getenv("VCINSTALLDIR"): # MSVC, manual setup
  14. return True
  15. # Otherwise, let SCons find MSVC if installed, or else Mingw.
  16. # Since we're just returning True here, if there's no compiler
  17. # installed, we'll get errors when it tries to build with the
  18. # null compiler.
  19. return True
  20. if os.name == "posix":
  21. # Cross-compiling with MinGW-w64 (old MinGW32 is not supported)
  22. mingw32 = "i686-w64-mingw32-"
  23. mingw64 = "x86_64-w64-mingw32-"
  24. if os.getenv("MINGW32_PREFIX"):
  25. mingw32 = os.getenv("MINGW32_PREFIX")
  26. if os.getenv("MINGW64_PREFIX"):
  27. mingw64 = os.getenv("MINGW64_PREFIX")
  28. test = "gcc --version > /dev/null 2>&1"
  29. if os.system(mingw64 + test) == 0 or os.system(mingw32 + test) == 0:
  30. return True
  31. return False
  32. def get_opts():
  33. from SCons.Variables import BoolVariable, EnumVariable
  34. mingw32 = ""
  35. mingw64 = ""
  36. if os.name == "posix":
  37. mingw32 = "i686-w64-mingw32-"
  38. mingw64 = "x86_64-w64-mingw32-"
  39. if os.getenv("MINGW32_PREFIX"):
  40. mingw32 = os.getenv("MINGW32_PREFIX")
  41. if os.getenv("MINGW64_PREFIX"):
  42. mingw64 = os.getenv("MINGW64_PREFIX")
  43. return [
  44. ("mingw_prefix_32", "MinGW prefix (Win32)", mingw32),
  45. ("mingw_prefix_64", "MinGW prefix (Win64)", mingw64),
  46. # Targeted Windows version: 7 (and later), minimum supported version
  47. # XP support dropped after EOL due to missing API for IPv6 and other issues
  48. # Vista support dropped after EOL due to GH-10243
  49. ("target_win_version", "Targeted Windows version, >= 0x0601 (Windows 7)", "0x0601"),
  50. BoolVariable("debug_symbols", "Add debugging symbols to release/release_debug builds", True),
  51. EnumVariable("windows_subsystem", "Windows subsystem", "gui", ("gui", "console")),
  52. BoolVariable("separate_debug_symbols", "Create a separate file containing debugging symbols", False),
  53. ("msvc_version", "MSVC version to use. Ignored if VCINSTALLDIR is set in shell env.", None),
  54. BoolVariable("use_mingw", "Use the Mingw compiler, even if MSVC is installed.", False),
  55. BoolVariable("use_llvm", "Use the LLVM compiler", False),
  56. BoolVariable("use_static_cpp", "Link MinGW/MSVC C++ runtime libraries statically", True),
  57. BoolVariable("use_asan", "Use address sanitizer (ASAN)", False),
  58. BoolVariable("incremental_link", "Use MSVC incremental linking. May increase or decrease build times.", False),
  59. ]
  60. def get_flags():
  61. return []
  62. def build_res_file(target, source, env):
  63. if env["bits"] == "32":
  64. cmdbase = env["mingw_prefix_32"]
  65. else:
  66. cmdbase = env["mingw_prefix_64"]
  67. cmdbase = cmdbase + "windres --include-dir . "
  68. import subprocess
  69. for x in range(len(source)):
  70. cmd = cmdbase + "-i " + str(source[x]) + " -o " + str(target[x])
  71. try:
  72. out = subprocess.Popen(cmd, shell=True, stderr=subprocess.PIPE).communicate()
  73. if len(out[1]):
  74. return 1
  75. except Exception:
  76. return 1
  77. return 0
  78. def setup_msvc_manual(env):
  79. """Set up env to use MSVC manually, using VCINSTALLDIR"""
  80. if env["bits"] != "default":
  81. print(
  82. """
  83. Bits argument is not supported for MSVC compilation. Architecture depends on the Native/Cross Compile Tools Prompt/Developer Console
  84. (or Visual Studio settings) that is being used to run SCons. As a consequence, bits argument is disabled. Run scons again without bits
  85. argument (example: scons p=windows) and SCons will attempt to detect what MSVC compiler will be executed and inform you.
  86. """
  87. )
  88. raise SCons.Errors.UserError("Bits argument should not be used when using VCINSTALLDIR")
  89. # Force bits arg
  90. # (Actually msys2 mingw can support 64-bit, we could detect that)
  91. env["bits"] = "32"
  92. env["x86_libtheora_opt_vc"] = True
  93. # find compiler manually
  94. compiler_version_str = methods.detect_visual_c_compiler_version(env["ENV"])
  95. print("Found MSVC compiler: " + compiler_version_str)
  96. # If building for 64bit architecture, disable assembly optimisations for 32 bit builds (theora as of writing)... vc compiler for 64bit can not compile _asm
  97. if compiler_version_str == "amd64" or compiler_version_str == "x86_amd64":
  98. env["bits"] = "64"
  99. env["x86_libtheora_opt_vc"] = False
  100. print("Compiled program architecture will be a 64 bit executable (forcing bits=64).")
  101. elif compiler_version_str == "x86" or compiler_version_str == "amd64_x86":
  102. print("Compiled program architecture will be a 32 bit executable. (forcing bits=32).")
  103. else:
  104. print(
  105. "Failed to manually detect MSVC compiler architecture version... Defaulting to 32bit executable settings (forcing bits=32). Compilation attempt will continue, but SCons can not detect for what architecture this build is compiled for. You should check your settings/compilation setup, or avoid setting VCINSTALLDIR."
  106. )
  107. def setup_msvc_auto(env):
  108. """Set up MSVC using SCons's auto-detection logic"""
  109. # If MSVC_VERSION is set by SCons, we know MSVC is installed.
  110. # But we may want a different version or target arch.
  111. # The env may have already been set up with default MSVC tools, so
  112. # reset a few things so we can set it up with the tools we want.
  113. # (Ideally we'd decide on the tool config before configuring any
  114. # environment, and just set the env up once, but this function runs
  115. # on an existing env so this is the simplest way.)
  116. env["MSVC_SETUP_RUN"] = False # Need to set this to re-run the tool
  117. env["MSVS_VERSION"] = None
  118. env["MSVC_VERSION"] = None
  119. env["TARGET_ARCH"] = None
  120. if env["bits"] != "default":
  121. env["TARGET_ARCH"] = {"32": "x86", "64": "x86_64"}[env["bits"]]
  122. if "msvc_version" in env:
  123. env["MSVC_VERSION"] = env["msvc_version"]
  124. env.Tool("msvc")
  125. env.Tool("mssdk") # we want the MS SDK
  126. # Note: actual compiler version can be found in env['MSVC_VERSION'], e.g. "14.1" for VS2015
  127. # Get actual target arch into bits (it may be "default" at this point):
  128. if env["TARGET_ARCH"] in ("amd64", "x86_64"):
  129. env["bits"] = "64"
  130. else:
  131. env["bits"] = "32"
  132. print("Found MSVC version %s, arch %s, bits=%s" % (env["MSVC_VERSION"], env["TARGET_ARCH"], env["bits"]))
  133. if env["TARGET_ARCH"] in ("amd64", "x86_64"):
  134. env["x86_libtheora_opt_vc"] = False
  135. def setup_mingw(env):
  136. """Set up env for use with mingw"""
  137. # Nothing to do here
  138. print("Using MinGW")
  139. def configure_msvc(env, manual_msvc_config):
  140. """Configure env to work with MSVC"""
  141. # Build type
  142. if env["target"] == "release":
  143. if env["optimize"] == "speed": # optimize for speed (default)
  144. env.Append(CCFLAGS=["/O2"])
  145. env.Append(LINKFLAGS=["/OPT:REF"])
  146. elif env["optimize"] == "size": # optimize for size
  147. env.Append(CCFLAGS=["/O1"])
  148. env.Append(LINKFLAGS=["/OPT:REF"])
  149. elif env["target"] == "release_debug":
  150. if env["optimize"] == "speed": # optimize for speed (default)
  151. env.Append(CCFLAGS=["/O2"])
  152. env.Append(LINKFLAGS=["/OPT:REF"])
  153. elif env["optimize"] == "size": # optimize for size
  154. env.Append(CCFLAGS=["/O1"])
  155. env.Append(LINKFLAGS=["/OPT:REF"])
  156. elif env["target"] == "debug":
  157. env.AppendUnique(CCFLAGS=["/Zi", "/FS", "/Od", "/EHsc"])
  158. env.Append(LINKFLAGS=["/DEBUG"])
  159. if env["windows_subsystem"] == "gui":
  160. env.Append(LINKFLAGS=["/SUBSYSTEM:WINDOWS"])
  161. else:
  162. env.Append(LINKFLAGS=["/SUBSYSTEM:CONSOLE"])
  163. env.AppendUnique(CPPDEFINES=["WINDOWS_SUBSYSTEM_CONSOLE"])
  164. env.Append(LINKFLAGS=["/ENTRY:mainCRTStartup"])
  165. if env["debug_symbols"]:
  166. env.AppendUnique(CCFLAGS=["/Zi", "/FS"])
  167. env.AppendUnique(LINKFLAGS=["/DEBUG"])
  168. ## Compile/link flags
  169. if env["use_static_cpp"]:
  170. env.AppendUnique(CCFLAGS=["/MT"])
  171. else:
  172. env.AppendUnique(CCFLAGS=["/MD"])
  173. # MSVC incremental linking is broken and may _increase_ link time (GH-77968).
  174. if not env["incremental_link"]:
  175. env.Append(LINKFLAGS=["/INCREMENTAL:NO"])
  176. env.AppendUnique(CCFLAGS=["/Gd", "/GR", "/nologo"])
  177. env.AppendUnique(CCFLAGS=["/utf-8"]) # Force to use Unicode encoding.
  178. env.AppendUnique(CXXFLAGS=["/TP"]) # assume all sources are C++
  179. # Once it was thought that only debug builds would be too large,
  180. # but this has recently stopped being true. See the mingw function
  181. # for notes on why this shouldn't be enabled for gcc
  182. env.AppendUnique(CCFLAGS=["/bigobj"])
  183. if manual_msvc_config: # should be automatic if SCons found it
  184. if os.getenv("WindowsSdkDir") is not None:
  185. env.Prepend(CPPPATH=[os.getenv("WindowsSdkDir") + "/Include"])
  186. else:
  187. print("Missing environment variable: WindowsSdkDir")
  188. env.AppendUnique(
  189. CPPDEFINES=[
  190. "WINDOWS_ENABLED",
  191. "OPENGL_ENABLED",
  192. "WASAPI_ENABLED",
  193. "WINMIDI_ENABLED",
  194. "TYPED_METHOD_BIND",
  195. "WIN32",
  196. "MSVC",
  197. "WINVER=%s" % env["target_win_version"],
  198. "_WIN32_WINNT=%s" % env["target_win_version"],
  199. ]
  200. )
  201. env.AppendUnique(CPPDEFINES=["NOMINMAX"]) # disable bogus min/max WinDef.h macros
  202. if env["bits"] == "64":
  203. env.AppendUnique(CPPDEFINES=["_WIN64"])
  204. ## Libs
  205. LIBS = [
  206. "winmm",
  207. "opengl32",
  208. "dsound",
  209. "kernel32",
  210. "ole32",
  211. "oleaut32",
  212. "sapi",
  213. "user32",
  214. "gdi32",
  215. "IPHLPAPI",
  216. "Shlwapi",
  217. "wsock32",
  218. "Ws2_32",
  219. "shell32",
  220. "advapi32",
  221. "dinput8",
  222. "dxguid",
  223. "imm32",
  224. "bcrypt",
  225. "Avrt",
  226. "dwmapi",
  227. ]
  228. env.Append(LINKFLAGS=[p + env["LIBSUFFIX"] for p in LIBS])
  229. if manual_msvc_config:
  230. if os.getenv("WindowsSdkDir") is not None:
  231. env.Append(LIBPATH=[os.getenv("WindowsSdkDir") + "/Lib"])
  232. else:
  233. print("Missing environment variable: WindowsSdkDir")
  234. ## LTO
  235. if env["lto"] == "auto": # No LTO by default for MSVC, doesn't help.
  236. env["lto"] = "none"
  237. if env["lto"] != "none":
  238. if env["lto"] == "thin":
  239. print("ThinLTO is only compatible with LLVM, use `use_llvm=yes` or `lto=full`.")
  240. sys.exit(255)
  241. env.AppendUnique(CCFLAGS=["/GL"])
  242. env.AppendUnique(ARFLAGS=["/LTCG"])
  243. if env["progress"]:
  244. env.AppendUnique(LINKFLAGS=["/LTCG:STATUS"])
  245. else:
  246. env.AppendUnique(LINKFLAGS=["/LTCG"])
  247. if manual_msvc_config:
  248. env.Prepend(CPPPATH=[p for p in os.getenv("INCLUDE").split(";")])
  249. env.Append(LIBPATH=[p for p in os.getenv("LIB").split(";")])
  250. # Sanitizers
  251. if env["use_asan"]:
  252. env.extra_suffix += ".s"
  253. env.Append(LINKFLAGS=["/INFERASANLIBS"])
  254. env.Append(CCFLAGS=["/fsanitize=address"])
  255. # Incremental linking fix
  256. env["BUILDERS"]["ProgramOriginal"] = env["BUILDERS"]["Program"]
  257. env["BUILDERS"]["Program"] = methods.precious_program
  258. env.AppendUnique(LINKFLAGS=["/STACK:" + str(STACK_SIZE)])
  259. def configure_mingw(env):
  260. # Workaround for MinGW. See:
  261. # http://www.scons.org/wiki/LongCmdLinesOnWin32
  262. env.use_windows_spawn_fix()
  263. ## Build type
  264. if env["target"] == "release":
  265. env.Append(CCFLAGS=["-msse2"])
  266. if env["optimize"] == "speed": # optimize for speed (default)
  267. if env["bits"] == "64":
  268. env.Append(CCFLAGS=["-O3"])
  269. else:
  270. env.Append(CCFLAGS=["-O2"])
  271. else: # optimize for size
  272. env.Prepend(CCFLAGS=["-Os"])
  273. if env["debug_symbols"]:
  274. env.Prepend(CCFLAGS=["-g2"])
  275. elif env["target"] == "release_debug":
  276. env.Append(CCFLAGS=["-O2"])
  277. if env["debug_symbols"]:
  278. env.Prepend(CCFLAGS=["-g2"])
  279. if env["optimize"] == "speed": # optimize for speed (default)
  280. env.Append(CCFLAGS=["-O2"])
  281. else: # optimize for size
  282. env.Prepend(CCFLAGS=["-Os"])
  283. elif env["target"] == "debug":
  284. env.Append(CCFLAGS=["-g3"])
  285. # Allow big objects. It's supposed not to have drawbacks but seems to break
  286. # GCC LTO, so enabling for debug builds only (which are not built with LTO
  287. # and are the only ones with too big objects).
  288. env.Append(CCFLAGS=["-Wa,-mbig-obj"])
  289. if env["windows_subsystem"] == "gui":
  290. env.Append(LINKFLAGS=["-Wl,--subsystem,windows"])
  291. else:
  292. env.Append(LINKFLAGS=["-Wl,--subsystem,console"])
  293. env.AppendUnique(CPPDEFINES=["WINDOWS_SUBSYSTEM_CONSOLE"])
  294. ## Compiler configuration
  295. if os.name == "nt":
  296. # Force splitting libmodules.a in multiple chunks to work around
  297. # issues reaching the linker command line size limit, which also
  298. # seem to induce huge slowdown for 'ar' (GH-30892).
  299. env["split_libmodules"] = True
  300. else:
  301. env["PROGSUFFIX"] = env["PROGSUFFIX"] + ".exe" # for linux cross-compilation
  302. if env["bits"] == "default":
  303. if os.name == "nt":
  304. env["bits"] = "64" if "PROGRAMFILES(X86)" in os.environ else "32"
  305. else: # default to 64-bit on Linux
  306. env["bits"] = "64"
  307. mingw_prefix = ""
  308. if env["bits"] == "32":
  309. if env["use_static_cpp"]:
  310. env.Append(LINKFLAGS=["-static"])
  311. env.Append(LINKFLAGS=["-static-libgcc"])
  312. env.Append(LINKFLAGS=["-static-libstdc++"])
  313. mingw_prefix = env["mingw_prefix_32"]
  314. else:
  315. if env["use_static_cpp"]:
  316. env.Append(LINKFLAGS=["-static"])
  317. mingw_prefix = env["mingw_prefix_64"]
  318. if env["use_llvm"]:
  319. env["CC"] = mingw_prefix + "clang"
  320. env["CXX"] = mingw_prefix + "clang++"
  321. env["AS"] = mingw_prefix + "as"
  322. env["AR"] = mingw_prefix + "ar"
  323. env["RANLIB"] = mingw_prefix + "ranlib"
  324. else:
  325. env["CC"] = mingw_prefix + "gcc"
  326. env["CXX"] = mingw_prefix + "g++"
  327. env["AS"] = mingw_prefix + "as"
  328. env["AR"] = mingw_prefix + "gcc-ar"
  329. env["RANLIB"] = mingw_prefix + "gcc-ranlib"
  330. env["x86_libtheora_opt_gcc"] = True
  331. ## LTO
  332. if env["lto"] == "auto": # Full LTO for production with MinGW.
  333. env["lto"] = "full"
  334. if env["lto"] != "none":
  335. if env["lto"] == "thin":
  336. if not env["use_llvm"]:
  337. print("ThinLTO is only compatible with LLVM, use `use_llvm=yes` or `lto=full`.")
  338. sys.exit(255)
  339. env.Append(CCFLAGS=["-flto=thin"])
  340. env.Append(LINKFLAGS=["-flto=thin"])
  341. elif not env["use_llvm"] and env.GetOption("num_jobs") > 1:
  342. env.Append(CCFLAGS=["-flto"])
  343. env.Append(LINKFLAGS=["-flto=" + str(env.GetOption("num_jobs"))])
  344. else:
  345. env.Append(CCFLAGS=["-flto"])
  346. env.Append(LINKFLAGS=["-flto"])
  347. env.Append(LINKFLAGS=["-Wl,--stack," + str(STACK_SIZE)])
  348. ## Compile flags
  349. env.Append(CCFLAGS=["-mwindows"])
  350. env.Append(LINKFLAGS=["-Wl,--nxcompat"]) # DEP protection. Not enabling ASLR for now, Mono crashes.
  351. env.Append(CPPDEFINES=["WINDOWS_ENABLED", "OPENGL_ENABLED", "WASAPI_ENABLED", "WINMIDI_ENABLED"])
  352. env.Append(CPPDEFINES=[("WINVER", env["target_win_version"]), ("_WIN32_WINNT", env["target_win_version"])])
  353. env.Append(
  354. LIBS=[
  355. "mingw32",
  356. "opengl32",
  357. "dsound",
  358. "ole32",
  359. "d3d9",
  360. "winmm",
  361. "gdi32",
  362. "iphlpapi",
  363. "shlwapi",
  364. "wsock32",
  365. "ws2_32",
  366. "kernel32",
  367. "oleaut32",
  368. "sapi",
  369. "dinput8",
  370. "dxguid",
  371. "ksuser",
  372. "imm32",
  373. "bcrypt",
  374. "avrt",
  375. "uuid",
  376. "dwmapi",
  377. ]
  378. )
  379. env.Append(CPPDEFINES=["MINGW_ENABLED", ("MINGW_HAS_SECURE_API", 1)])
  380. # resrc
  381. env.Append(BUILDERS={"RES": env.Builder(action=build_res_file, suffix=".o", src_suffix=".rc")})
  382. def configure(env):
  383. # At this point the env has been set up with basic tools/compilers.
  384. env.Prepend(CPPPATH=["#platform/windows"])
  385. print("Configuring for Windows: target=%s, bits=%s" % (env["target"], env["bits"]))
  386. if os.name == "nt":
  387. env["ENV"] = os.environ # this makes build less repeatable, but simplifies some things
  388. env["ENV"]["TMP"] = os.environ["TMP"]
  389. # First figure out which compiler, version, and target arch we're using
  390. if os.getenv("VCINSTALLDIR") and not env["use_mingw"]:
  391. # Manual setup of MSVC
  392. setup_msvc_manual(env)
  393. env.msvc = True
  394. manual_msvc_config = True
  395. elif env.get("MSVC_VERSION", "") and not env["use_mingw"]:
  396. setup_msvc_auto(env)
  397. env.msvc = True
  398. manual_msvc_config = False
  399. else:
  400. setup_mingw(env)
  401. env.msvc = False
  402. # Now set compiler/linker flags
  403. if env.msvc:
  404. configure_msvc(env, manual_msvc_config)
  405. else: # MinGW
  406. configure_mingw(env)