detect.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  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. EnumVariable('debug_symbols', 'Add debugging symbols to release builds', 'yes', ('yes', 'no', 'full')),
  51. BoolVariable('separate_debug_symbols', 'Create a separate file containing debugging symbols', False),
  52. ('msvc_version', 'MSVC version to use. Ignored if VCINSTALLDIR is set in shell env.', None),
  53. BoolVariable('use_mingw', 'Use the Mingw compiler, even if MSVC is installed. Only used on Windows.', False),
  54. BoolVariable('use_llvm', 'Use the LLVM compiler', False),
  55. BoolVariable('use_thinlto', 'Use ThinLTO', False),
  56. ]
  57. def get_flags():
  58. return [
  59. ]
  60. def build_res_file(target, source, env):
  61. if (env["bits"] == "32"):
  62. cmdbase = env['mingw_prefix_32']
  63. else:
  64. cmdbase = env['mingw_prefix_64']
  65. cmdbase = cmdbase + 'windres --include-dir . '
  66. import subprocess
  67. for x in range(len(source)):
  68. cmd = cmdbase + '-i ' + str(source[x]) + ' -o ' + str(target[x])
  69. try:
  70. out = subprocess.Popen(cmd, shell=True, stderr=subprocess.PIPE).communicate()
  71. if len(out[1]):
  72. return 1
  73. except:
  74. return 1
  75. return 0
  76. def setup_msvc_manual(env):
  77. """Set up env to use MSVC manually, using VCINSTALLDIR"""
  78. if (env["bits"] != "default"):
  79. print("""
  80. Bits argument is not supported for MSVC compilation. Architecture depends on the Native/Cross Compile Tools Prompt/Developer Console
  81. (or Visual Studio settings) that is being used to run SCons. As a consequence, bits argument is disabled. Run scons again without bits
  82. argument (example: scons p=windows) and SCons will attempt to detect what MSVC compiler will be executed and inform you.
  83. """)
  84. raise SCons.Errors.UserError("Bits argument should not be used when using VCINSTALLDIR")
  85. # Force bits arg
  86. # (Actually msys2 mingw can support 64-bit, we could detect that)
  87. env["bits"] = "32"
  88. env["x86_libtheora_opt_vc"] = True
  89. # find compiler manually
  90. compiler_version_str = methods.detect_visual_c_compiler_version(env['ENV'])
  91. print("Found MSVC compiler: " + compiler_version_str)
  92. # If building for 64bit architecture, disable assembly optimisations for 32 bit builds (theora as of writing)... vc compiler for 64bit can not compile _asm
  93. if(compiler_version_str == "amd64" or compiler_version_str == "x86_amd64"):
  94. env["bits"] = "64"
  95. env["x86_libtheora_opt_vc"] = False
  96. print("Compiled program architecture will be a 64 bit executable (forcing bits=64).")
  97. elif (compiler_version_str == "x86" or compiler_version_str == "amd64_x86"):
  98. print("Compiled program architecture will be a 32 bit executable. (forcing bits=32).")
  99. else:
  100. print("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.")
  101. def setup_msvc_auto(env):
  102. """Set up MSVC using SCons's auto-detection logic"""
  103. # If MSVC_VERSION is set by SCons, we know MSVC is installed.
  104. # But we may want a different version or target arch.
  105. # The env may have already been set up with default MSVC tools, so
  106. # reset a few things so we can set it up with the tools we want.
  107. # (Ideally we'd decide on the tool config before configuring any
  108. # environment, and just set the env up once, but this function runs
  109. # on an existing env so this is the simplest way.)
  110. env['MSVC_SETUP_RUN'] = False # Need to set this to re-run the tool
  111. env['MSVS_VERSION'] = None
  112. env['MSVC_VERSION'] = None
  113. env['TARGET_ARCH'] = None
  114. if env['bits'] != 'default':
  115. env['TARGET_ARCH'] = {'32': 'x86', '64': 'x86_64'}[env['bits']]
  116. if env.has_key('msvc_version'):
  117. env['MSVC_VERSION'] = env['msvc_version']
  118. env.Tool('msvc')
  119. env.Tool('mssdk') # we want the MS SDK
  120. # Note: actual compiler version can be found in env['MSVC_VERSION'], e.g. "14.1" for VS2015
  121. # Get actual target arch into bits (it may be "default" at this point):
  122. if env['TARGET_ARCH'] in ('amd64', 'x86_64'):
  123. env['bits'] = '64'
  124. else:
  125. env['bits'] = '32'
  126. print("Found MSVC version %s, arch %s, bits=%s" % (env['MSVC_VERSION'], env['TARGET_ARCH'], env['bits']))
  127. if env['TARGET_ARCH'] in ('amd64', 'x86_64'):
  128. env["x86_libtheora_opt_vc"] = False
  129. def setup_mingw(env):
  130. """Set up env for use with mingw"""
  131. # Nothing to do here
  132. print("Using MinGW")
  133. pass
  134. def configure_msvc(env, manual_msvc_config):
  135. """Configure env to work with MSVC"""
  136. # Build type
  137. if (env["target"] == "release"):
  138. if (env["optimize"] == "speed"): #optimize for speed (default)
  139. env.Append(CCFLAGS=['/O2'])
  140. else: # optimize for size
  141. env.Append(CCFLAGS=['/O1'])
  142. env.Append(LINKFLAGS=['/SUBSYSTEM:WINDOWS'])
  143. env.Append(LINKFLAGS=['/ENTRY:mainCRTStartup'])
  144. env.Append(LINKFLAGS=['/OPT:REF'])
  145. elif (env["target"] == "release_debug"):
  146. if (env["optimize"] == "speed"): #optimize for speed (default)
  147. env.Append(CCFLAGS=['/O2'])
  148. else: # optimize for size
  149. env.Append(CCFLAGS=['/O1'])
  150. env.AppendUnique(CPPDEFINES = ['DEBUG_ENABLED'])
  151. env.Append(LINKFLAGS=['/SUBSYSTEM:CONSOLE'])
  152. env.Append(LINKFLAGS=['/OPT:REF'])
  153. elif (env["target"] == "debug"):
  154. env.AppendUnique(CCFLAGS=['/Z7', '/Od', '/EHsc'])
  155. env.AppendUnique(CPPDEFINES = ['DEBUG_ENABLED', 'DEBUG_MEMORY_ENABLED',
  156. 'D3D_DEBUG_INFO'])
  157. env.Append(LINKFLAGS=['/SUBSYSTEM:CONSOLE'])
  158. env.Append(LINKFLAGS=['/DEBUG'])
  159. if (env["debug_symbols"] == "full" or env["debug_symbols"] == "yes"):
  160. env.AppendUnique(CCFLAGS=['/Z7'])
  161. env.AppendUnique(LINKFLAGS=['/DEBUG'])
  162. ## Compile/link flags
  163. env.AppendUnique(CCFLAGS=['/MT', '/Gd', '/GR', '/nologo'])
  164. if int(env['MSVC_VERSION'].split('.')[0]) >= 14: #vs2015 and later
  165. env.AppendUnique(CCFLAGS=['/utf-8'])
  166. env.AppendUnique(CXXFLAGS=['/TP']) # assume all sources are C++
  167. if manual_msvc_config: # should be automatic if SCons found it
  168. if os.getenv("WindowsSdkDir") is not None:
  169. env.Prepend(CPPPATH=[os.getenv("WindowsSdkDir") + "/Include"])
  170. else:
  171. print("Missing environment variable: WindowsSdkDir")
  172. env.AppendUnique(CPPDEFINES = ['WINDOWS_ENABLED', 'OPENGL_ENABLED',
  173. 'WASAPI_ENABLED', 'WINMIDI_ENABLED',
  174. 'TYPED_METHOD_BIND',
  175. 'WIN32', 'MSVC',
  176. 'WINVER=%s' % env["target_win_version"],
  177. '_WIN32_WINNT=%s' % env["target_win_version"]])
  178. env.AppendUnique(CPPDEFINES=['NOMINMAX']) # disable bogus min/max WinDef.h macros
  179. if env["bits"] == "64":
  180. env.AppendUnique(CPPDEFINES=['_WIN64'])
  181. ## Libs
  182. LIBS = ['winmm', 'opengl32', 'dsound', 'kernel32', 'ole32', 'oleaut32',
  183. 'user32', 'gdi32', 'IPHLPAPI', 'Shlwapi', 'wsock32', 'Ws2_32',
  184. 'shell32', 'advapi32', 'dinput8', 'dxguid', 'imm32', 'bcrypt','Avrt',
  185. 'dwmapi']
  186. env.Append(LINKFLAGS=[p + env["LIBSUFFIX"] for p in LIBS])
  187. if manual_msvc_config:
  188. if os.getenv("WindowsSdkDir") is not None:
  189. env.Append(LIBPATH=[os.getenv("WindowsSdkDir") + "/Lib"])
  190. else:
  191. print("Missing environment variable: WindowsSdkDir")
  192. ## LTO
  193. if (env["use_lto"]):
  194. env.AppendUnique(CCFLAGS=['/GL'])
  195. env.AppendUnique(ARFLAGS=['/LTCG'])
  196. if env["progress"]:
  197. env.AppendUnique(LINKFLAGS=['/LTCG:STATUS'])
  198. else:
  199. env.AppendUnique(LINKFLAGS=['/LTCG'])
  200. if manual_msvc_config:
  201. env.Prepend(CPPPATH=[p for p in os.getenv("INCLUDE").split(";")])
  202. env.Append(LIBPATH=[p for p in os.getenv("LIB").split(";")])
  203. # Incremental linking fix
  204. env['BUILDERS']['ProgramOriginal'] = env['BUILDERS']['Program']
  205. env['BUILDERS']['Program'] = methods.precious_program
  206. env.AppendUnique(LINKFLAGS=['/STACK:' + str(STACK_SIZE)])
  207. def configure_mingw(env):
  208. # Workaround for MinGW. See:
  209. # http://www.scons.org/wiki/LongCmdLinesOnWin32
  210. env.use_windows_spawn_fix()
  211. ## Build type
  212. if (env["target"] == "release"):
  213. env.Append(CCFLAGS=['-msse2'])
  214. if (env["optimize"] == "speed"): #optimize for speed (default)
  215. if (env["bits"] == "64"):
  216. env.Append(CCFLAGS=['-O3'])
  217. else:
  218. env.Append(CCFLAGS=['-O2'])
  219. else: #optimize for size
  220. env.Prepend(CCFLAGS=['-Os'])
  221. env.Append(LINKFLAGS=['-Wl,--subsystem,windows'])
  222. if (env["debug_symbols"] == "yes"):
  223. env.Prepend(CCFLAGS=['-g1'])
  224. if (env["debug_symbols"] == "full"):
  225. env.Prepend(CCFLAGS=['-g2'])
  226. elif (env["target"] == "release_debug"):
  227. env.Append(CCFLAGS=['-O2'])
  228. env.Append(CPPDEFINES=['DEBUG_ENABLED'])
  229. if (env["debug_symbols"] == "yes"):
  230. env.Prepend(CCFLAGS=['-g1'])
  231. if (env["debug_symbols"] == "full"):
  232. env.Prepend(CCFLAGS=['-g2'])
  233. if (env["optimize"] == "speed"): #optimize for speed (default)
  234. env.Append(CCFLAGS=['-O2'])
  235. else: #optimize for size
  236. env.Prepend(CCFLAGS=['-Os'])
  237. elif (env["target"] == "debug"):
  238. env.Append(CCFLAGS=['-g3'])
  239. env.Append(CPPDEFINES=['DEBUG_ENABLED', 'DEBUG_MEMORY_ENABLED'])
  240. ## Compiler configuration
  241. if (os.name == "nt"):
  242. # Force splitting libmodules.a in multiple chunks to work around
  243. # issues reaching the linker command line size limit, which also
  244. # seem to induce huge slowdown for 'ar' (GH-30892).
  245. env['split_libmodules'] = True
  246. else:
  247. env["PROGSUFFIX"] = env["PROGSUFFIX"] + ".exe" # for linux cross-compilation
  248. if (env["bits"] == "default"):
  249. if (os.name == "nt"):
  250. env["bits"] = "64" if "PROGRAMFILES(X86)" in os.environ else "32"
  251. else: # default to 64-bit on Linux
  252. env["bits"] = "64"
  253. mingw_prefix = ""
  254. if (env["bits"] == "32"):
  255. env.Append(LINKFLAGS=['-static'])
  256. env.Append(LINKFLAGS=['-static-libgcc'])
  257. env.Append(LINKFLAGS=['-static-libstdc++'])
  258. mingw_prefix = env["mingw_prefix_32"]
  259. else:
  260. env.Append(LINKFLAGS=['-static'])
  261. mingw_prefix = env["mingw_prefix_64"]
  262. if env['use_llvm']:
  263. env["CC"] = mingw_prefix + "clang"
  264. env['AS'] = mingw_prefix + "as"
  265. env["CXX"] = mingw_prefix + "clang++"
  266. env['AR'] = mingw_prefix + "ar"
  267. env['RANLIB'] = mingw_prefix + "ranlib"
  268. env["LINK"] = mingw_prefix + "clang++"
  269. else:
  270. env["CC"] = mingw_prefix + "gcc"
  271. env['AS'] = mingw_prefix + "as"
  272. env['CXX'] = mingw_prefix + "g++"
  273. env['AR'] = mingw_prefix + "gcc-ar"
  274. env['RANLIB'] = mingw_prefix + "gcc-ranlib"
  275. env['LINK'] = mingw_prefix + "g++"
  276. env["x86_libtheora_opt_gcc"] = True
  277. if env['use_lto']:
  278. if not env['use_llvm'] and env.GetOption("num_jobs") > 1:
  279. env.Append(CCFLAGS=['-flto'])
  280. env.Append(LINKFLAGS=['-flto=' + str(env.GetOption("num_jobs"))])
  281. else:
  282. if env['use_thinlto']:
  283. env.Append(CCFLAGS=['-flto=thin'])
  284. env.Append(LINKFLAGS=['-flto=thin'])
  285. else:
  286. env.Append(CCFLAGS=['-flto'])
  287. env.Append(LINKFLAGS=['-flto'])
  288. env.Append(LINKFLAGS=['-Wl,--stack,' + str(STACK_SIZE)])
  289. ## Compile flags
  290. env.Append(CCFLAGS=['-mwindows'])
  291. env.Append(CPPDEFINES=['WINDOWS_ENABLED', 'OPENGL_ENABLED', 'WASAPI_ENABLED', 'WINMIDI_ENABLED'])
  292. env.Append(CPPDEFINES=[('WINVER', env['target_win_version']), ('_WIN32_WINNT', env['target_win_version'])])
  293. env.Append(LIBS=['mingw32', 'opengl32', 'dsound', 'ole32', 'd3d9', 'winmm', 'gdi32', 'iphlpapi', 'shlwapi', 'wsock32', 'ws2_32', 'kernel32', 'oleaut32', 'dinput8', 'dxguid', 'ksuser', 'imm32', 'bcrypt', 'avrt', 'uuid', 'dwmapi'])
  294. env.Append(CPPDEFINES=['MINGW_ENABLED', ('MINGW_HAS_SECURE_API', 1)])
  295. # resrc
  296. env.Append(BUILDERS={'RES': env.Builder(action=build_res_file, suffix='.o', src_suffix='.rc')})
  297. def configure(env):
  298. # At this point the env has been set up with basic tools/compilers.
  299. env.Prepend(CPPPATH=['#platform/windows'])
  300. print("Configuring for Windows: target=%s, bits=%s" % (env['target'], env['bits']))
  301. if (os.name == "nt"):
  302. env['ENV'] = os.environ # this makes build less repeatable, but simplifies some things
  303. env['ENV']['TMP'] = os.environ['TMP']
  304. # First figure out which compiler, version, and target arch we're using
  305. if os.getenv("VCINSTALLDIR") and not env["use_mingw"]:
  306. # Manual setup of MSVC
  307. setup_msvc_manual(env)
  308. env.msvc = True
  309. manual_msvc_config = True
  310. elif env.get('MSVC_VERSION', '') and not env["use_mingw"]:
  311. setup_msvc_auto(env)
  312. env.msvc = True
  313. manual_msvc_config = False
  314. else:
  315. setup_mingw(env)
  316. env.msvc = False
  317. # Now set compiler/linker flags
  318. if env.msvc:
  319. configure_msvc(env, manual_msvc_config)
  320. else: # MinGW
  321. configure_mingw(env)