detect.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. #
  2. # tested on | Windows native | Linux cross-compilation
  3. # ------------------------+-------------------+---------------------------
  4. # MSVS C++ 2010 Express | WORKS | n/a
  5. # Mingw-w64 | WORKS | WORKS
  6. # Mingw-w32 | WORKS | WORKS
  7. # MinGW | WORKS | untested
  8. #
  9. #####
  10. # Notes about MSVS C++ :
  11. #
  12. # - MSVC2010-Express compiles to 32bits only.
  13. #
  14. #####
  15. # Notes about Mingw-w64 and Mingw-w32 under Windows :
  16. #
  17. # - both can be installed using the official installer :
  18. # http://mingw-w64.sourceforge.net/download.php#mingw-builds
  19. #
  20. # - if you want to compile both 32bits and 64bits, don't forget to
  21. # run the installer twice to install them both.
  22. #
  23. # - install them into a path that does not contain spaces
  24. # ( example : "C:/Mingw-w32", "C:/Mingw-w64" )
  25. #
  26. # - if you want to compile faster using the "-j" option, don't forget
  27. # to install the appropriate version of the Pywin32 python extension
  28. # available from : http://sourceforge.net/projects/pywin32/files/
  29. #
  30. # - before running scons, you must add into the environment path
  31. # the path to the "/bin" directory of the Mingw version you want
  32. # to use :
  33. #
  34. # set PATH=C:/Mingw-w32/bin;%PATH%
  35. #
  36. # - then, scons should be able to detect gcc.
  37. # - Mingw-w32 only compiles 32bits.
  38. # - Mingw-w64 only compiles 64bits.
  39. #
  40. # - it is possible to add them both at the same time into the PATH env,
  41. # if you also define the MINGW32_PREFIX and MINGW64_PREFIX environment
  42. # variables.
  43. # For instance, you could store that set of commands into a .bat script
  44. # that you would run just before scons :
  45. #
  46. # set PATH=C:\mingw-w32\bin;%PATH%
  47. # set PATH=C:\mingw-w64\bin;%PATH%
  48. # set MINGW32_PREFIX=C:\mingw-w32\bin\
  49. # set MINGW64_PREFIX=C:\mingw-w64\bin\
  50. #
  51. #####
  52. # Notes about Mingw, Mingw-w64 and Mingw-w32 under Linux :
  53. #
  54. # - default toolchain prefixes are :
  55. # "i586-mingw32msvc-" for MinGW
  56. # "i686-w64-mingw32-" for Mingw-w32
  57. # "x86_64-w64-mingw32-" for Mingw-w64
  58. #
  59. # - if both MinGW and Mingw-w32 are installed on your system
  60. # Mingw-w32 should take the priority over MinGW.
  61. #
  62. # - it is possible to manually override prefixes by defining
  63. # the MINGW32_PREFIX and MINGW64_PREFIX environment variables.
  64. #
  65. #####
  66. # Notes about Mingw under Windows :
  67. #
  68. # - this is the MinGW version from http://mingw.org/
  69. # - install it into a path that does not contain spaces
  70. # ( example : "C:/MinGW" )
  71. # - several DirectX headers might be missing. You can copy them into
  72. # the C:/MinGW/include" directory from this page :
  73. # https://code.google.com/p/mingw-lib/source/browse/trunk/working/avcodec_to_widget_5/directx_include/
  74. # - before running scons, add the path to the "/bin" directory :
  75. # set PATH=C:/MinGW/bin;%PATH%
  76. # - scons should be able to detect gcc.
  77. #
  78. #####
  79. # TODO :
  80. #
  81. # - finish to cleanup this script to remove all the remains of previous hacks and workarounds
  82. # - make it work with the Windows7 SDK that is supposed to enable 64bits compilation for MSVC2010-Express
  83. # - confirm it works well with other Visual Studio versions.
  84. # - update the wiki about the pywin32 extension required for the "-j" option under Windows.
  85. # - update the wiki to document MINGW32_PREFIX and MINGW64_PREFIX
  86. #
  87. import os
  88. import sys
  89. def is_active():
  90. return True
  91. def get_name():
  92. return "Windows"
  93. def can_build():
  94. if (os.name=="nt"):
  95. #building natively on windows!
  96. if (os.getenv("VSINSTALLDIR")):
  97. return True
  98. else:
  99. print("\nMSVC not detected, attempting Mingw.")
  100. mingw32 = ""
  101. mingw64 = ""
  102. if ( os.getenv("MINGW32_PREFIX") ) :
  103. mingw32 = os.getenv("MINGW32_PREFIX")
  104. if ( os.getenv("MINGW64_PREFIX") ) :
  105. mingw64 = os.getenv("MINGW64_PREFIX")
  106. test = "gcc --version > NUL 2>&1"
  107. if os.system(test)!= 0 and os.system(mingw32+test)!=0 and os.system(mingw64+test)!=0 :
  108. print("- could not detect gcc.")
  109. print("Please, make sure a path to a Mingw /bin directory is accessible into the environment PATH.\n")
  110. return False
  111. else:
  112. print("- gcc detected.")
  113. return True
  114. if (os.name=="posix"):
  115. mingw = "i586-mingw32msvc-"
  116. mingw64 = "x86_64-w64-mingw32-"
  117. mingw32 = "i686-w64-mingw32-"
  118. if (os.getenv("MINGW32_PREFIX")):
  119. mingw32=os.getenv("MINGW32_PREFIX")
  120. mingw = mingw32
  121. if (os.getenv("MINGW64_PREFIX")):
  122. mingw64=os.getenv("MINGW64_PREFIX")
  123. test = "gcc --version &>/dev/null"
  124. if (os.system(mingw+test) == 0 or os.system(mingw64+test) == 0 or os.system(mingw32+test) == 0):
  125. return True
  126. return False
  127. def get_opts():
  128. mingw=""
  129. mingw32=""
  130. mingw64=""
  131. if ( os.name == "posix" ):
  132. mingw = "i586-mingw32msvc-"
  133. mingw32 = "i686-w64-mingw32-"
  134. mingw64 = "x86_64-w64-mingw32-"
  135. if os.system(mingw32+"gcc --version &>/dev/null") != 0 :
  136. mingw32 = mingw
  137. if (os.getenv("MINGW32_PREFIX")):
  138. mingw32=os.getenv("MINGW32_PREFIX")
  139. mingw = mingw32
  140. if (os.getenv("MINGW64_PREFIX")):
  141. mingw64=os.getenv("MINGW64_PREFIX")
  142. return [
  143. ('mingw_prefix','Mingw Prefix',mingw32),
  144. ('mingw_prefix_64','Mingw Prefix 64 bits',mingw64),
  145. ]
  146. def get_flags():
  147. return [
  148. ('freetype','builtin'), #use builtin freetype
  149. ('openssl','builtin'), #use builtin openssl
  150. ]
  151. def build_res_file( target, source, env ):
  152. cmdbase = ""
  153. if (env["bits"] == "32"):
  154. cmdbase = env['mingw_prefix']
  155. else:
  156. cmdbase = env['mingw_prefix_64']
  157. CPPPATH = env['CPPPATH']
  158. cmdbase = cmdbase + 'windres --include-dir . '
  159. import subprocess
  160. for x in range(len(source)):
  161. cmd = cmdbase + '-i ' + str(source[x]) + ' -o ' + str(target[x])
  162. try:
  163. out = subprocess.Popen(cmd,shell = True,stderr = subprocess.PIPE).communicate()
  164. if len(out[1]):
  165. return 1
  166. except:
  167. return 1
  168. return 0
  169. def configure(env):
  170. env.Append(CPPPATH=['#platform/windows'])
  171. env['is_mingw']=False
  172. if (os.name=="nt" and os.getenv("VSINSTALLDIR")!=None):
  173. #build using visual studio
  174. env['ENV']['TMP'] = os.environ['TMP']
  175. env.Append(CPPPATH=['#platform/windows/include'])
  176. env.Append(LIBPATH=['#platform/windows/lib'])
  177. if (env["freetype"]!="no"):
  178. env.Append(CCFLAGS=['/DFREETYPE_ENABLED'])
  179. env.Append(CPPPATH=['#tools/freetype'])
  180. env.Append(CPPPATH=['#tools/freetype/freetype/include'])
  181. if (env["target"]=="release"):
  182. env.Append(CCFLAGS=['/O2'])
  183. env.Append(LINKFLAGS=['/SUBSYSTEM:WINDOWS'])
  184. env.Append(LINKFLAGS=['/ENTRY:mainCRTStartup'])
  185. elif (env["target"]=="release_debug"):
  186. env.Append(CCFLAGS=['/O2','/DDEBUG_ENABLED'])
  187. env.Append(LINKFLAGS=['/SUBSYSTEM:CONSOLE'])
  188. elif (env["target"]=="debug_release"):
  189. env.Append(CCFLAGS=['/Z7','/Od'])
  190. env.Append(LINKFLAGS=['/DEBUG'])
  191. env.Append(LINKFLAGS=['/SUBSYSTEM:WINDOWS'])
  192. env.Append(LINKFLAGS=['/ENTRY:mainCRTStartup'])
  193. elif (env["target"]=="debug"):
  194. env.Append(CCFLAGS=['/Z7','/DDEBUG_ENABLED','/DDEBUG_MEMORY_ENABLED','/DD3D_DEBUG_INFO','/Od'])
  195. env.Append(LINKFLAGS=['/SUBSYSTEM:CONSOLE'])
  196. env.Append(LINKFLAGS=['/DEBUG'])
  197. env.Append(CCFLAGS=['/MT','/Gd','/GR','/nologo'])
  198. env.Append(CXXFLAGS=['/TP'])
  199. env.Append(CPPFLAGS=['/DMSVC', '/GR', ])
  200. env.Append(CCFLAGS=['/I'+os.getenv("WindowsSdkDir")+"/Include"])
  201. env.Append(CCFLAGS=['/DWINDOWS_ENABLED'])
  202. env.Append(CCFLAGS=['/DRTAUDIO_ENABLED'])
  203. env.Append(CCFLAGS=['/DWIN32'])
  204. env.Append(CCFLAGS=['/DTYPED_METHOD_BIND'])
  205. env.Append(CCFLAGS=['/DGLES2_ENABLED'])
  206. env.Append(CCFLAGS=['/DGLEW_ENABLED'])
  207. LIBS=['winmm','opengl32','dsound','kernel32','ole32','oleaut32','user32','gdi32', 'IPHLPAPI','Shlwapi', 'wsock32', 'shell32','advapi32','dinput8','dxguid']
  208. env.Append(LINKFLAGS=[p+env["LIBSUFFIX"] for p in LIBS])
  209. env.Append(LIBPATH=[os.getenv("WindowsSdkDir")+"/Lib"])
  210. if (os.getenv("DXSDK_DIR")):
  211. DIRECTX_PATH=os.getenv("DXSDK_DIR")
  212. else:
  213. DIRECTX_PATH="C:/Program Files/Microsoft DirectX SDK (March 2009)"
  214. if (os.getenv("VCINSTALLDIR")):
  215. VC_PATH=os.getenv("VCINSTALLDIR")
  216. else:
  217. VC_PATH=""
  218. env.Append(CCFLAGS=["/I" + p for p in os.getenv("INCLUDE").split(";")])
  219. env.Append(LIBPATH=[p for p in os.getenv("LIB").split(";")])
  220. env.Append(CCFLAGS=["/I"+DIRECTX_PATH+"/Include"])
  221. env.Append(LIBPATH=[DIRECTX_PATH+"/Lib/x86"])
  222. env['ENV'] = os.environ;
  223. env["x86_opt_vc"]=True
  224. else:
  225. # Workaround for MinGW. See:
  226. # http://www.scons.org/wiki/LongCmdLinesOnWin32
  227. if (os.name=="nt"):
  228. import subprocess
  229. def mySubProcess(cmdline,env):
  230. #print "SPAWNED : " + cmdline
  231. startupinfo = subprocess.STARTUPINFO()
  232. startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
  233. proc = subprocess.Popen(cmdline, stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  234. stderr=subprocess.PIPE, startupinfo=startupinfo, shell = False, env = env)
  235. data, err = proc.communicate()
  236. rv = proc.wait()
  237. if rv:
  238. print "====="
  239. print err
  240. print "====="
  241. return rv
  242. def mySpawn(sh, escape, cmd, args, env):
  243. newargs = ' '.join(args[1:])
  244. cmdline = cmd + " " + newargs
  245. rv=0
  246. if len(cmdline) > 32000 and cmd.endswith("ar") :
  247. cmdline = cmd + " " + args[1] + " " + args[2] + " "
  248. for i in range(3,len(args)) :
  249. rv = mySubProcess( cmdline + args[i], env )
  250. if rv :
  251. break
  252. else:
  253. rv = mySubProcess( cmdline, env )
  254. return rv
  255. env['SPAWN'] = mySpawn
  256. #build using mingw
  257. if (os.name=="nt"):
  258. env['ENV']['TMP'] = os.environ['TMP'] #way to go scons, you can be so stupid sometimes
  259. else:
  260. env["PROGSUFFIX"]=env["PROGSUFFIX"]+".exe" # for linux cross-compilation
  261. mingw_prefix=""
  262. if (env["bits"]=="default"):
  263. env["bits"]="32"
  264. if (env["bits"]=="32"):
  265. env.Append(LINKFLAGS=['-static'])
  266. env.Append(LINKFLAGS=['-static-libgcc'])
  267. env.Append(LINKFLAGS=['-static-libstdc++'])
  268. mingw_prefix=env["mingw_prefix"];
  269. else:
  270. env.Append(LINKFLAGS=['-static'])
  271. mingw_prefix=env["mingw_prefix_64"];
  272. nulstr=""
  273. if (os.name=="posix"):
  274. nulstr=">/dev/null"
  275. else:
  276. nulstr=">nul"
  277. # if os.system(mingw_prefix+"gcc --version"+nulstr)!=0:
  278. # #not really super consistent but..
  279. # print("Can't find Windows compiler: "+mingw_prefix)
  280. # sys.exit(255)
  281. if (env["target"]=="release"):
  282. env.Append(CCFLAGS=['-O3','-ffast-math','-fomit-frame-pointer','-msse2'])
  283. env.Append(LINKFLAGS=['-Wl,--subsystem,windows'])
  284. elif (env["target"]=="release_debug"):
  285. env.Append(CCFLAGS=['-O2','-DDEBUG_ENABLED'])
  286. elif (env["target"]=="debug"):
  287. env.Append(CCFLAGS=['-g', '-Wall','-DDEBUG_ENABLED','-DDEBUG_MEMORY_ENABLED'])
  288. if (env["freetype"]!="no"):
  289. env.Append(CCFLAGS=['-DFREETYPE_ENABLED'])
  290. env.Append(CPPPATH=['#tools/freetype'])
  291. env.Append(CPPPATH=['#tools/freetype/freetype/include'])
  292. env["CC"]=mingw_prefix+"gcc"
  293. env['AS']=mingw_prefix+"as"
  294. env['CXX'] = mingw_prefix+"g++"
  295. env['AR'] = mingw_prefix+"ar"
  296. env['RANLIB'] = mingw_prefix+"ranlib"
  297. env['LD'] = mingw_prefix+"g++"
  298. env["x86_opt_gcc"]=True
  299. #env['CC'] = "winegcc"
  300. #env['CXX'] = "wineg++"
  301. env.Append(CCFLAGS=['-DWINDOWS_ENABLED','-mwindows'])
  302. env.Append(CPPFLAGS=['-DRTAUDIO_ENABLED'])
  303. env.Append(CCFLAGS=['-DGLES2_ENABLED','-DGLEW_ENABLED'])
  304. env.Append(LIBS=['mingw32','opengl32', 'dsound', 'ole32', 'd3d9','winmm','gdi32','iphlpapi','shlwapi','wsock32','kernel32', 'oleaut32', 'dinput8', 'dxguid'])
  305. # if (env["bits"]=="32"):
  306. # env.Append(LIBS=['gcc_s'])
  307. # #--with-arch=i686
  308. # env.Append(CPPFLAGS=['-march=i686'])
  309. # env.Append(LINKFLAGS=['-march=i686'])
  310. #'d3dx9d'
  311. env.Append(CPPFLAGS=['-DMINGW_ENABLED'])
  312. env.Append(LINKFLAGS=['-g'])
  313. # resrc
  314. env['is_mingw']=True
  315. env.Append( BUILDERS = { 'RES' : env.Builder(action = build_res_file, suffix = '.o',src_suffix = '.rc') } )
  316. import methods
  317. env.Append( BUILDERS = { 'GLSL120' : env.Builder(action = methods.build_legacygl_headers, suffix = 'glsl.h',src_suffix = '.glsl') } )
  318. env.Append( BUILDERS = { 'GLSL' : env.Builder(action = methods.build_glsl_headers, suffix = 'glsl.h',src_suffix = '.glsl') } )
  319. env.Append( BUILDERS = { 'HLSL9' : env.Builder(action = methods.build_hlsl_dx9_headers, suffix = 'hlsl.h',src_suffix = '.hlsl') } )
  320. env.Append( BUILDERS = { 'GLSL120GLES' : env.Builder(action = methods.build_gles2_headers, suffix = 'glsl.h',src_suffix = '.glsl') } )