Merging r38036 through r38196 from trunk into soc-2011-tomato
[blender.git] / build_files / scons / tools / btools.py
1 import os
2 import os.path
3 import SCons.Options
4
5 import SCons.Variables
6 try:
7     import subprocess
8 except ImportError:
9     pass
10 import string
11 import glob
12 import shutil
13 import sys
14
15 Variables = SCons.Variables
16 BoolVariable = SCons.Variables.BoolVariable
17
18 def get_version():
19     import re
20
21     fname = os.path.join(os.path.dirname(__file__), "..", "..", "..", "source", "blender", "blenkernel", "BKE_blender.h")
22     ver_base = None
23     ver_char = None
24     ver_cycle = None
25
26     re_ver = re.compile("^#\s*define\s+BLENDER_VERSION\s+([0-9]+)")
27     re_ver_char = re.compile("^#\s*define\s+BLENDER_VERSION_CHAR\s*(\S*)") # optional arg
28     re_ver_cycle = re.compile("^#\s*define\s+BLENDER_VERSION_CYCLE\s*(\S*)") # optional arg
29
30     for l in open(fname, "r"):
31         match = re_ver.match(l)
32         if match:
33             ver = int(match.group(1))
34             ver_base = "%d.%d" % (ver / 100, ver % 100)
35
36         match = re_ver_char.match(l)
37         if match:
38             ver_char = match.group(1)
39             if ver_char == "BLENDER_CHAR_VERSION":
40                 ver_char = ""
41
42         match = re_ver_cycle.match(l)
43         if match:
44             ver_cycle = match.group(1)
45             if ver_cycle == "BLENDER_CYCLE_VERSION":
46                 ver_cycle = ""
47
48         if (ver_base is not None) and (ver_char is not None) and (ver_cycle is not None):
49             # eg '2.56a-beta'
50             if ver_cycle:
51                 ver_display = "%s%s-%s" % (ver_base, ver_char, ver_cycle)
52             else:
53                 ver_display = "%s%s" % (ver_base, ver_char)  # assume release
54
55             return ver_base, ver_display
56
57     raise Exception("%s: missing version string" % fname)
58
59 def get_revision():
60     build_rev = os.popen('svnversion').read()[:-1] # remove \n
61     if build_rev == '' or build_rev==None: 
62         build_rev = 'UNKNOWN'
63
64     return 'r' + build_rev
65
66 # This is used in creating the local config directories
67 VERSION, VERSION_DISPLAY = get_version()
68 REVISION = get_revision()
69
70 def print_arguments(args, bc):
71     if len(args):
72         for k,v in args.iteritems():
73             if type(v)==list:
74                 v = ' '.join(v)
75             print '\t'+bc.OKBLUE+k+bc.ENDC+' = '+bc.OKGREEN + v + bc.ENDC
76     else:
77         print '\t'+bc.WARNING+'No  command-line arguments given'+bc.ENDC
78
79 def validate_arguments(args, bc):
80     opts_list = [
81             'WITH_BF_PYTHON', 'WITH_BF_PYTHON_SAFETY', 'BF_PYTHON', 'BF_PYTHON_VERSION', 'BF_PYTHON_INC', 'BF_PYTHON_BINARY', 'BF_PYTHON_LIB', 'BF_PYTHON_LIBPATH', 'WITH_BF_STATICPYTHON', 'BF_PYTHON_LIB_STATIC', 'BF_PYTHON_DLL', 'BF_PYTHON_ABI_FLAGS', 
82             'WITH_BF_OPENAL', 'BF_OPENAL', 'BF_OPENAL_INC', 'BF_OPENAL_LIB', 'BF_OPENAL_LIBPATH', 'WITH_BF_STATICOPENAL', 'BF_OPENAL_LIB_STATIC',
83             'WITH_BF_SDL', 'BF_SDL', 'BF_SDL_INC', 'BF_SDL_LIB', 'BF_SDL_LIBPATH',
84             'BF_LIBSAMPLERATE', 'BF_LIBSAMPLERATE_INC', 'BF_LIBSAMPLERATE_LIB', 'BF_LIBSAMPLERATE_LIBPATH', 'WITH_BF_STATICLIBSAMPLERATE', 'BF_LIBSAMPLERATE_LIB_STATIC',
85             'WITH_BF_JACK', 'BF_JACK', 'BF_JACK_INC', 'BF_JACK_LIB', 'BF_JACK_LIBPATH',
86             'WITH_BF_SNDFILE', 'BF_SNDFILE', 'BF_SNDFILE_INC', 'BF_SNDFILE_LIB', 'BF_SNDFILE_LIBPATH', 'WITH_BF_STATICSNDFILE', 'BF_SNDFILE_LIB_STATIC',
87             'BF_PTHREADS', 'BF_PTHREADS_INC', 'BF_PTHREADS_LIB', 'BF_PTHREADS_LIBPATH',
88             'WITH_BF_OPENEXR', 'BF_OPENEXR', 'BF_OPENEXR_INC', 'BF_OPENEXR_LIB', 'BF_OPENEXR_LIBPATH', 'WITH_BF_STATICOPENEXR', 'BF_OPENEXR_LIB_STATIC',
89             'WITH_BF_DDS', 'WITH_BF_CINEON', 'WITH_BF_HDR',
90             'WITH_BF_FFMPEG', 'BF_FFMPEG_LIB','BF_FFMPEG_EXTRA', 'BF_FFMPEG',  'BF_FFMPEG_INC',
91             'WITH_BF_STATICFFMPEG', 'BF_FFMPEG_LIB_STATIC',
92             'WITH_BF_OGG', 'BF_OGG', 'BF_OGG_LIB',
93             'WITH_BF_JPEG', 'BF_JPEG', 'BF_JPEG_INC', 'BF_JPEG_LIB', 'BF_JPEG_LIBPATH',
94             'WITH_BF_OPENJPEG', 'BF_OPENJPEG', 'BF_OPENJPEG_INC', 'BF_OPENJPEG_LIB', 'BF_OPENJPEG_LIBPATH',
95             'WITH_BF_REDCODE', 'BF_REDCODE', 'BF_REDCODE_INC', 'BF_REDCODE_LIB', 'BF_REDCODE_LIBPATH',
96             'WITH_BF_PNG', 'BF_PNG', 'BF_PNG_INC', 'BF_PNG_LIB', 'BF_PNG_LIBPATH',
97             'WITH_BF_TIFF', 'BF_TIFF', 'BF_TIFF_INC', 'BF_TIFF_LIB', 'BF_TIFF_LIBPATH', 'WITH_BF_STATICTIFF', 'BF_TIFF_LIB_STATIC',
98             'WITH_BF_ZLIB', 'BF_ZLIB', 'BF_ZLIB_INC', 'BF_ZLIB_LIB', 'BF_ZLIB_LIBPATH', 'WITH_BF_STATICZLIB', 'BF_ZLIB_LIB_STATIC',
99             'WITH_BF_INTERNATIONAL',
100             'BF_GETTEXT', 'BF_GETTEXT_INC', 'BF_GETTEXT_LIB', 'WITH_BF_GETTEXT_STATIC', 'BF_GETTEXT_LIB_STATIC', 'BF_GETTEXT_LIBPATH',
101             'WITH_BF_ICONV', 'BF_ICONV', 'BF_ICONV_INC', 'BF_ICONV_LIB', 'BF_ICONV_LIBPATH',
102             'WITH_BF_GAMEENGINE', 'WITH_BF_BULLET', 'WITH_BF_ELTOPO', 'BF_BULLET', 'BF_BULLET_INC', 'BF_BULLET_LIB',
103             'BF_WINTAB', 'BF_WINTAB_INC',
104             'WITH_BF_FREETYPE', 'BF_FREETYPE', 'BF_FREETYPE_INC', 'BF_FREETYPE_LIB', 'BF_FREETYPE_LIBPATH', 'BF_FREETYPE_LIB_STATIC', 'WITH_BF_FREETYPE_STATIC',
105             'WITH_BF_QUICKTIME', 'BF_QUICKTIME', 'BF_QUICKTIME_INC', 'BF_QUICKTIME_LIB', 'BF_QUICKTIME_LIBPATH',
106             'WITH_BF_FFTW3', 'BF_FFTW3', 'BF_FFTW3_INC', 'BF_FFTW3_LIB', 'BF_FFTW3_LIBPATH', 'WITH_BF_STATICFFTW3', 'BF_FFTW3_LIB_STATIC',
107             'WITH_BF_STATICOPENGL', 'BF_OPENGL', 'BF_OPENGL_INC', 'BF_OPENGL_LIB', 'BF_OPENGL_LIBPATH', 'BF_OPENGL_LIB_STATIC',
108             'WITH_BF_COLLADA', 'BF_COLLADA', 'BF_COLLADA_INC', 'BF_COLLADA_LIB', 'BF_OPENCOLLADA', 'BF_OPENCOLLADA_INC', 'BF_OPENCOLLADA_LIB', 'BF_OPENCOLLADA_LIBPATH', 'BF_PCRE', 'BF_PCRE_LIB', 'BF_PCRE_LIBPATH', 'BF_EXPAT', 'BF_EXPAT_LIB', 'BF_EXPAT_LIBPATH',
109             'WITH_BF_PLAYER',
110             'WITH_BF_NOBLENDER',
111             'WITH_BF_BINRELOC',
112             'WITH_BF_LZO', 'WITH_BF_LZMA',
113             'LCGDIR',
114             'BF_CXX', 'WITH_BF_STATICCXX', 'BF_CXX_LIB_STATIC',
115             'BF_TWEAK_MODE', 'BF_SPLIT_SRC',
116             'WITHOUT_BF_INSTALL',
117             'WITHOUT_BF_PYTHON_INSTALL',
118             'WITHOUT_BF_OVERWRITE_INSTALL',
119             'WITH_BF_OPENMP',
120             'BF_OPENMP',
121             'BF_OPENMP_INC',
122             'BF_OPENMP_LIBPATH',
123             'WITH_GHOST_COCOA',
124             'USE_QTKIT',
125             'BF_FANCY', 'BF_QUIET', 'BF_LINE_OVERWRITE',
126             'BF_X264_CONFIG',
127             'BF_XVIDCORE_CONFIG',
128             'WITH_BF_DOCS',
129             'BF_NUMJOBS',
130             'BF_MSVS',
131             'BF_VERSION',
132             'BF_GHOST_DEBUG',
133             'WITH_BF_RAYOPTIMIZATION',
134             'BF_RAYOPTIMIZATION_SSE_FLAGS',
135             'BF_NO_ELBEEM',
136             'WITH_BF_CXX_GUARDEDALLOC',
137             'WITH_BF_JEMALLOC', 'WITH_BF_STATICJEMALLOC', 'BF_JEMALLOC', 'BF_JEMALLOC_INC', 'BF_JEMALLOC_LIBPATH', 'BF_JEMALLOC_LIB', 'BF_JEMALLOC_LIB_STATIC',
138             'BUILDBOT_BRANCH'
139             ]
140     
141     # Have options here that scons expects to be lists
142     opts_list_split = [
143             'BF_PYTHON_LINKFLAGS',
144             'BF_OPENGL_LINKFLAGS',
145             'CFLAGS', 'CCFLAGS', 'CXXFLAGS', 'CPPFLAGS',
146             'REL_CFLAGS', 'REL_CCFLAGS', 'REL_CXXFLAGS',
147             'BGE_CXXFLAGS',
148             'BF_PROFILE_CFLAGS', 'BF_PROFILE_CCFLAGS', 'BF_PROFILE_CXXFLAGS', 'BF_PROFILE_LINKFLAGS',
149             'BF_DEBUG_CFLAGS', 'BF_DEBUG_CCFLAGS', 'BF_DEBUG_CXXFLAGS',
150             'C_WARN', 'CC_WARN', 'CXX_WARN',
151             'LLIBS', 'PLATFORM_LINKFLAGS','MACOSX_ARCHITECTURE',
152     ]
153     
154     
155     arg_list = ['BF_DEBUG', 'BF_QUIET', 'BF_CROSS', 'BF_UPDATE',
156             'BF_INSTALLDIR', 'BF_TOOLSET', 'BF_BINNAME',
157             'BF_BUILDDIR', 'BF_FANCY', 'BF_QUICK', 'BF_PROFILE', 'BF_LINE_OVERWRITE',
158             'BF_BSC', 'BF_CONFIG',
159             'BF_PRIORITYLIST', 'BF_BUILDINFO','CC', 'CXX', 'BF_QUICKDEBUG',
160             'BF_LISTDEBUG', 'LCGDIR', 'BF_X264_CONFIG', 'BF_XVIDCORE_CONFIG',
161             'BF_UNIT_TEST']
162
163     okdict = {}
164
165     for k,v in args.iteritems():
166         if (k in opts_list) or (k in arg_list):
167             okdict[k] = v
168         elif k in opts_list_split:
169             okdict[k] = v.split() # "" have already been stripped
170         else:
171             print '\t'+bc.WARNING+'Invalid argument: '+bc.ENDC+k+'='+v
172
173     return okdict
174
175 def print_targets(targs, bc):
176     if len(targs)>0:
177         for t in targs:
178             print '\t'+bc.OKBLUE+t+bc.ENDC
179     else:
180         print '\t'+bc.WARNING+'No targets given, using '+bc.ENDC+bc.OKGREEN+'default'+bc.ENDC
181
182 def validate_targets(targs, bc):
183     valid_list = ['.', 'blender', 'blenderstatic', 'blenderplayer', 'webplugin',
184             'blendernogame', 'blenderstaticnogame', 'blenderlite', 'release',
185             'everything', 'clean', 'install-bin', 'install', 'nsis','buildslave']
186     oklist = []
187     for t in targs:
188         if t in valid_list:
189             oklist.append(t)
190         else:
191             print '\t'+bc.WARNING+'Invalid target: '+bc.ENDC+t
192     return oklist
193
194 class ourSpawn:
195     def ourspawn(self, sh, escape, cmd, args, env):
196         newargs = string.join(args[1:], ' ')
197         cmdline = cmd + " " + newargs
198         startupinfo = subprocess.STARTUPINFO()
199         startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
200         proc = subprocess.Popen(cmdline, stdin=subprocess.PIPE, stdout=subprocess.PIPE,
201             stderr=subprocess.PIPE, startupinfo=startupinfo, shell = False)
202         data, err = proc.communicate()
203         rv = proc.wait()
204         if rv:
205             print "====="
206             print err
207             print "====="
208         return rv
209
210 def SetupSpawn( env ):
211     buf = ourSpawn()
212     buf.ourenv = env
213     env['SPAWN'] = buf.ourspawn
214
215
216 def read_opts(env, cfg, args):
217     localopts = Variables.Variables(cfg, args)
218     localopts.AddVariables(
219         ('LCGDIR', 'location of cvs lib dir'),
220         (BoolVariable('WITH_BF_PYTHON', 'Compile with python', True)),
221         (BoolVariable('WITH_BF_PYTHON_SAFETY', 'Internal API error checking to track invalid data to prevent crash on access (at the expense of some effeciency)', False)),
222         ('BF_PYTHON', 'base path for python', ''),
223         ('BF_PYTHON_VERSION', 'Python version to use', ''),
224         ('BF_PYTHON_INC', 'include path for Python headers', ''),
225         ('BF_PYTHON_BINARY', 'Path to the Python interpreter', ''),
226         ('BF_PYTHON_LIB', 'Python library', ''),
227         ('BF_PYTHON_DLL', 'Python dll - used on Windows only', ''),
228         ('BF_PYTHON_LIB_STATIC', 'Python static libraries', ''),
229         ('BF_PYTHON_LIBPATH', 'Library path', ''),
230         ('BF_PYTHON_LINKFLAGS', 'Python link flags', ''),
231         (BoolVariable('WITH_BF_STATICPYTHON', 'Staticly link to python', False)),
232         ('BF_PYTHON_ABI_FLAGS', 'Python ABI flags (suffix in library version: m, mu, etc)', ''),
233
234         (BoolVariable('BF_NO_ELBEEM', 'Disable Fluid Sim', False)),
235         ('BF_PROFILE_FLAGS', 'Profiling compiler flags', ''),
236         (BoolVariable('WITH_BF_OPENAL', 'Use OpenAL if true', False)),
237         ('BF_OPENAL', 'base path for OpenAL', ''),
238         ('BF_OPENAL_INC', 'include path for python headers', ''),
239         ('BF_OPENAL_LIB', 'Path to OpenAL library', ''),
240         ('BF_OPENAL_LIB_STATIC', 'Path to OpenAL static library', ''),
241         ('BF_OPENAL_LIBPATH', 'Path to OpenAL library', ''),
242         (BoolVariable('WITH_BF_STATICOPENAL', 'Staticly link to openal', False)),
243
244         (BoolVariable('WITH_BF_SDL', 'Use SDL if true', False)),
245         ('BF_SDL', 'SDL base path', ''),
246         ('BF_SDL_INC', 'SDL include path', ''),
247         ('BF_SDL_LIB', 'SDL library', ''),
248         ('BF_SDL_LIBPATH', 'SDL library path', ''),
249
250         ('BF_LIBSAMPLERATE', 'libsamplerate aka SRC base path', ''),
251         ('BF_LIBSAMPLERATE_INC', 'libsamplerate aka SRC include path', ''),
252         ('BF_LIBSAMPLERATE_LIB', 'libsamplerate aka SRC library', ''),
253         ('BF_LIBSAMPLERATE_LIBPATH', 'libsamplerate aka SRC library path', ''),
254         ('BF_LIBSAMPLERATE_LIB_STATIC', 'Path to libsamplerate static library', ''),
255         (BoolVariable('WITH_BF_STATICLIBSAMPLERATE', 'Staticly link to libsamplerate', False)),
256
257         (BoolVariable('WITH_BF_JACK', 'Enable jack support if true', True)),
258         ('BF_JACK', 'jack base path', ''),
259         ('BF_JACK_INC', 'jack include path', ''),
260         ('BF_JACK_LIB', 'jack library', ''),
261         ('BF_JACK_LIBPATH', 'jack library path', ''),
262
263         (BoolVariable('WITH_BF_SNDFILE', 'Enable sndfile support if true', True)),
264         ('BF_SNDFILE', 'sndfile base path', ''),
265         ('BF_SNDFILE_INC', 'sndfile include path', ''),
266         ('BF_SNDFILE_LIB', 'sndfile library', ''),
267         ('BF_SNDFILE_LIB_STATIC', 'Path to sndfile static library', ''),
268         ('BF_SNDFILE_LIBPATH', 'sndfile library path', ''),
269         (BoolVariable('WITH_BF_STATICSNDFILE', 'Staticly link to sndfile', False)),
270
271         ('BF_PTHREADS', 'Pthreads base path', ''),
272         ('BF_PTHREADS_INC', 'Pthreads include path', ''),
273         ('BF_PTHREADS_LIB', 'Pthreads library', ''),
274         ('BF_PTHREADS_LIBPATH', 'Pthreads library path', ''),
275
276         (BoolVariable('WITH_BF_OPENEXR', 'Use OPENEXR if true', True)),
277         (BoolVariable('WITH_BF_STATICOPENEXR', 'Staticly link to OpenEXR', False)),
278         ('BF_OPENEXR', 'OPENEXR base path', ''),
279         ('BF_OPENEXR_INC', 'OPENEXR include path', ''),
280         ('BF_OPENEXR_LIB', 'OPENEXR library', ''),
281         ('BF_OPENEXR_LIBPATH', 'OPENEXR library path', ''),
282         ('BF_OPENEXR_LIB_STATIC', 'OPENEXR static library', ''),
283
284         (BoolVariable('WITH_BF_DDS', 'Support DDS image format if true', True)),
285
286         (BoolVariable('WITH_BF_CINEON', 'Support CINEON and DPX image formats if true', True)),
287
288         (BoolVariable('WITH_BF_HDR', 'Support HDR image formats if true', True)),
289
290         (BoolVariable('WITH_BF_FFMPEG', 'Use FFMPEG if true', False)),
291         ('BF_FFMPEG', 'FFMPEG base path', ''),
292         ('BF_FFMPEG_LIB', 'FFMPEG library', ''),
293         ('BF_FFMPEG_EXTRA', 'FFMPEG flags that must be preserved', ''),
294
295         ('BF_FFMPEG_INC', 'FFMPEG includes', ''),
296         ('BF_FFMPEG_LIBPATH', 'FFMPEG library path', ''),
297         (BoolVariable('WITH_BF_STATICFFMPEG', 'Use static FFMPEG if true', False)),
298         ('BF_FFMPEG_LIB_STATIC', 'Static FFMPEG libraries', ''),
299         
300         (BoolVariable('WITH_BF_OGG', 'Use OGG, THEORA, VORBIS in FFMPEG if true',
301                     False)),
302         ('BF_OGG', 'OGG base path', ''),
303         ('BF_OGG_LIB', 'OGG library', ''),
304
305         (BoolVariable('WITH_BF_JPEG', 'Use JPEG if true', True)),
306         ('BF_JPEG', 'JPEG base path', ''),
307         ('BF_JPEG_INC', 'JPEG include path', ''),
308         ('BF_JPEG_LIB', 'JPEG library', ''),
309         ('BF_JPEG_LIBPATH', 'JPEG library path', ''),
310
311         (BoolVariable('WITH_BF_OPENJPEG', 'Use OPENJPEG if true', False)),
312         ('BF_OPENJPEG', 'OPENJPEG base path', ''),
313         ('BF_OPENJPEG_INC', 'OPENJPEG include path', ''),
314         ('BF_OPENJPEG_LIB', 'OPENJPEG library', ''),
315         ('BF_OPENJPEG_LIBPATH', 'OPENJPEG library path', ''),
316
317         (BoolVariable('WITH_BF_REDCODE', 'Use REDCODE if true', False)),
318         ('BF_REDCODE', 'REDCODE base path', ''),
319         ('BF_REDCODE_INC', 'REDCODE include path', ''),
320         ('BF_REDCODE_LIB', 'REDCODE library', ''),
321         ('BF_REDCODE_LIBPATH', 'REDCODE library path', ''),
322
323         (BoolVariable('WITH_BF_PNG', 'Use PNG if true', True)),
324         ('BF_PNG', 'PNG base path', ''),
325         ('BF_PNG_INC', 'PNG include path', ''),
326         ('BF_PNG_LIB', 'PNG library', ''),
327         ('BF_PNG_LIBPATH', 'PNG library path', ''),
328
329         (BoolVariable('WITH_BF_TIFF', 'Use TIFF if true', True)),
330         (BoolVariable('WITH_BF_STATICTIFF', 'Staticly link to TIFF', False)),
331         ('BF_TIFF', 'TIFF base path', ''),
332         ('BF_TIFF_INC', 'TIFF include path', ''),
333         ('BF_TIFF_LIB', 'TIFF library', ''),
334         ('BF_TIFF_LIBPATH', 'TIFF library path', ''),
335         ('BF_TIFF_LIB_STATIC', 'TIFF static library', ''),
336
337         (BoolVariable('WITH_BF_ZLIB', 'Use ZLib if true', True)),
338         (BoolVariable('WITH_BF_STATICZLIB', 'Staticly link to ZLib', False)),
339         ('BF_ZLIB', 'ZLib base path', ''),
340         ('BF_ZLIB_INC', 'ZLib include path', ''),
341         ('BF_ZLIB_LIB', 'ZLib library', ''),
342         ('BF_ZLIB_LIBPATH', 'ZLib library path', ''),
343         ('BF_ZLIB_LIB_STATIC', 'ZLib static library', ''),
344
345         (BoolVariable('WITH_BF_INTERNATIONAL', 'Use Gettext if true', True)),
346
347         ('BF_GETTEXT', 'gettext base path', ''),
348         ('BF_GETTEXT_INC', 'gettext include path', ''),
349         ('BF_GETTEXT_LIB', 'gettext library', ''),
350         (BoolVariable('WITH_BF_GETTEXT_STATIC', 'Use static gettext library if true', False)),
351         ('BF_GETTEXT_LIB_STATIC', 'static gettext library', ''),
352         ('BF_GETTEXT_LIBPATH', 'gettext library path', ''),
353         
354         (BoolVariable('WITH_BF_ICONV', 'Use iconv if true', True)),
355         ('BF_ICONV', 'iconv base path', ''),
356         ('BF_ICONV_INC', 'iconv include path', ''),
357         ('BF_ICONV_LIB', 'iconv library', ''),
358         ('BF_ICONV_LIBPATH', 'iconv library path', ''),
359         
360         (BoolVariable('WITH_BF_GAMEENGINE', 'Build with gameengine' , False)),
361
362         (BoolVariable('WITH_BF_BULLET', 'Use Bullet if true', True)),
363         (BoolVariable('WITH_BF_ELTOPO', 'Use Eltopo collision library if true', False)),
364         
365         ('BF_BULLET', 'Bullet base dir', ''),
366         ('BF_BULLET_INC', 'Bullet include path', ''),
367         ('BF_BULLET_LIB', 'Bullet library', ''),
368         
369         ('BF_WINTAB', 'WinTab base dir', ''),
370         ('BF_WINTAB_INC', 'WinTab include dir', ''),
371         ('BF_CXX', 'c++ base path for libstdc++, only used when static linking', ''),
372         (BoolVariable('WITH_BF_STATICCXX', 'static link to stdc++', False)),
373         ('BF_CXX_LIB_STATIC', 'static library path for stdc++', ''),
374
375         (BoolVariable('WITH_BF_FREETYPE', 'Use FreeType2 if true', True)),
376         ('BF_FREETYPE', 'Freetype base path', ''),
377         ('BF_FREETYPE_INC', 'Freetype include path', ''),
378         ('BF_FREETYPE_LIB', 'Freetype library', ''),
379         ('BF_FREETYPE_LIBPATH', 'Freetype library path', ''),
380         (BoolVariable('WITH_BF_FREETYPE_STATIC', 'Use Static Freetype if true', False)),
381         ('BF_FREETYPE_LIB_STATIC', 'Static Freetype library', ''),
382
383         (BoolVariable('WITH_BF_OPENMP', 'Use OpenMP if true', False)),
384         ('BF_OPENMP', 'Base path to OpenMP (used when cross-compiling with older versions of WinGW)', ''),
385         ('BF_OPENMP_INC', 'Path to OpenMP includes (used when cross-compiling with older versions of WinGW)', ''),
386         ('BF_OPENMP_LIBPATH', 'Path to OpenMP libraries (used when cross-compiling with older versions of WinGW)', ''),
387         (BoolVariable('WITH_GHOST_COCOA', 'Use Cocoa-framework if true', False)),
388         (BoolVariable('USE_QTKIT', 'Use QTKIT if true', False)),
389
390         (BoolVariable('WITH_BF_QUICKTIME', 'Use QuickTime if true', False)),
391         ('BF_QUICKTIME', 'QuickTime base path', ''),
392         ('BF_QUICKTIME_INC', 'QuickTime include path', ''),
393         ('BF_QUICKTIME_LIB', 'QuickTime library', ''),
394         ('BF_QUICKTIME_LIBPATH', 'QuickTime library path', ''),
395         
396         (BoolVariable('WITH_BF_FFTW3', 'Use FFTW3 if true', False)),
397         ('BF_FFTW3', 'FFTW3 base path', ''),
398         ('BF_FFTW3_INC', 'FFTW3 include path', ''),
399         ('BF_FFTW3_LIB', 'FFTW3 library', ''),
400         ('BF_FFTW3_LIB_STATIC', 'FFTW3 static libraries', ''),
401         ('BF_FFTW3_LIBPATH', 'FFTW3 library path', ''),
402         (BoolVariable('WITH_BF_STATICFFTW3', 'Staticly link to FFTW3', False)),
403
404         (BoolVariable('WITH_BF_STATICOPENGL', 'Use MESA if true', True)),
405         ('BF_OPENGL', 'OpenGL base path', ''),
406         ('BF_OPENGL_INC', 'OpenGL include path', ''),
407         ('BF_OPENGL_LIB', 'OpenGL libraries', ''),
408         ('BF_OPENGL_LIBPATH', 'OpenGL library path', ''),
409         ('BF_OPENGL_LIB_STATIC', 'OpenGL static libraries', ''),
410         ('BF_OPENGL_LINKFLAGS', 'OpenGL link flags', ''),
411
412         (BoolVariable('WITH_BF_COLLADA', 'Build COLLADA import/export module if true', False)),
413         ('BF_COLLADA', 'COLLADA base path', ''),
414         ('BF_COLLADA_INC', 'COLLADA include path', ''),
415         ('BF_COLLADA_LIB', 'COLLADA library', ''),
416         ('BF_OPENCOLLADA', 'OpenCollada base path', ''),
417         ('BF_OPENCOLLADA_INC', 'OpenCollada base include path', ''),
418         ('BF_OPENCOLLADA_LIB', 'OpenCollada library', ''),
419         ('BF_OPENCOLLADA_LIBPATH', 'OpenCollada library path', ''),
420         ('BF_PCRE', 'PCRE base path', ''),
421         ('BF_PCRE_LIB', 'PCRE library', ''),
422         ('BF_PCRE_LIBPATH', 'PCRE library path', ''),
423         ('BF_EXPAT', 'Expat base path', ''),
424         ('BF_EXPAT_LIB', 'Expat library', ''),
425         ('BF_EXPAT_LIBPATH', 'Expat library path', ''),
426         
427         (BoolVariable('WITH_BF_JEMALLOC', 'Use jemalloc if true', False)),
428         (BoolVariable('WITH_BF_STATICJEMALLOC', 'Staticly link to jemalloc', False)),
429         ('BF_JEMALLOC', 'jemalloc base path', ''),
430         ('BF_JEMALLOC_INC', 'jemalloc include path', ''),
431         ('BF_JEMALLOC_LIB', 'jemalloc library', ''),
432         ('BF_JEMALLOC_LIBPATH', 'jemalloc library path', ''),
433         ('BF_JEMALLOC_LIB_STATIC', 'jemalloc static library', ''),
434
435         (BoolVariable('WITH_BF_PLAYER', 'Build blenderplayer if true', False)),
436         (BoolVariable('WITH_BF_NOBLENDER', 'Do not build blender if true', False)),
437
438         ('CFLAGS', 'C only flags', []),
439         ('CCFLAGS', 'Generic C and C++ flags', []),
440         ('CXXFLAGS', 'C++ only flags', []),
441         ('BGE_CXXFLAGS', 'C++ only flags for BGE', []),
442         ('CPPFLAGS', 'Defines', []),
443         ('REL_CFLAGS', 'C only release flags', []),
444         ('REL_CCFLAGS', 'Generic C and C++ release flags', []),
445         ('REL_CXXFLAGS', 'C++ only release flags', []),
446
447         ('C_WARN', 'C warning flags', []),
448         ('CC_WARN', 'Generic C and C++ warning flags', []),
449         ('CXX_WARN', 'C++ only warning flags', []),
450
451         ('LLIBS', 'Platform libs', []),
452         ('PLATFORM_LINKFLAGS', 'Platform linkflags', []),
453         ('MACOSX_ARCHITECTURE', 'python_arch.zip select', ''),
454
455         (BoolVariable('BF_PROFILE', 'Add profiling information if true', False)),
456         ('BF_PROFILE_CFLAGS', 'C only profiling flags', []),
457         ('BF_PROFILE_CCFLAGS', 'C and C++ profiling flags', []),
458         ('BF_PROFILE_CXXFLAGS', 'C++ only profiling flags', []),
459         ('BF_PROFILE_LINKFLAGS', 'Profile linkflags', []),
460
461         (BoolVariable('BF_DEBUG', 'Add debug flags if true', False)),
462         ('BF_DEBUG_CFLAGS', 'C only debug flags', []),
463         ('BF_DEBUG_CCFLAGS', 'C and C++ debug flags', []),
464         ('BF_DEBUG_CXXFLAGS', 'C++ only debug flags', []),
465
466         (BoolVariable('BF_BSC', 'Create .bsc files (msvc only)', False)),
467
468         ('BF_BUILDDIR', 'Build dir', ''),
469         ('BF_INSTALLDIR', 'Installation dir', ''),
470
471         ('CC', 'C compiler to use', env['CC']),
472         ('CXX', 'C++ compiler to use', env['CXX']),
473
474         (BoolVariable('BF_BUILDINFO', 'Buildtime in splash if true', True)),
475
476         (BoolVariable('BF_TWEAK_MODE', 'Enable tweak mode if true', False)),
477         (BoolVariable('BF_SPLIT_SRC', 'Split src lib into several chunks if true', False)),
478         (BoolVariable('WITHOUT_BF_INSTALL', 'dont install if true', False)),
479         (BoolVariable('WITHOUT_BF_PYTHON_INSTALL', 'dont install Python modules if true', False)),
480         (BoolVariable('WITHOUT_BF_OVERWRITE_INSTALL', 'dont remove existing files before breating the new install directory (set to False when making packages for others)', False)),
481         (BoolVariable('BF_FANCY', 'Enable fancy output if true', True)),
482         (BoolVariable('BF_QUIET', 'Enable silent output if true', True)),
483         (BoolVariable('BF_LINE_OVERWRITE', 'Enable overwriting of compile line in BF_QUIET mode if true', False)),
484         (BoolVariable('WITH_BF_BINRELOC', 'Enable relocatable binary (linux only)', False)),
485         
486         (BoolVariable('WITH_BF_LZO', 'Enable fast LZO pointcache compression', True)),
487         (BoolVariable('WITH_BF_LZMA', 'Enable best LZMA pointcache compression', True)),
488         
489         (BoolVariable('WITH_BF_LIBMV', 'Enable libmv structure from motion library', True)),
490
491         ('BF_X264_CONFIG', 'configuration flags for x264', ''),
492         ('BF_XVIDCORE_CONFIG', 'configuration flags for xvidcore', ''),
493 #        (BoolVariable('WITH_BF_DOCS', 'Generate API documentation', False)),
494         
495         ('BF_CONFIG', 'SCons python config file used to set default options', 'user_config.py'),
496         ('BF_NUMJOBS', 'Number of build processes to spawn', '1'),
497         ('BF_MSVS', 'Generate MSVS project files and solution', False),
498
499         ('BF_VERSION', 'The root path for Unix (non-apple)', '2.5'),
500
501         (BoolVariable('BF_UNIT_TEST', 'Build with unit test support.', False)),
502         
503         (BoolVariable('BF_GHOST_DEBUG', 'Make GHOST print events and info to stdout. (very verbose)', False)),
504         
505         (BoolVariable('WITH_BF_RAYOPTIMIZATION', 'Enable raytracer SSE/SIMD optimization.', False)),
506         ('BF_RAYOPTIMIZATION_SSE_FLAGS', 'SSE flags', ''),
507         (BoolVariable('WITH_BF_CXX_GUARDEDALLOC', 'Enable GuardedAlloc for C++ memory allocation tracking.', False)),
508
509         ('BUILDBOT_BRANCH', 'Buildbot branch name', ''),
510     ) # end of opts.AddOptions()
511
512     return localopts
513
514 def buildbot_zip(src, dest, package_name, extension):
515     import zipfile
516     ln = len(src)+1 # one extra to remove leading os.sep when cleaning root for package_root
517     flist = list()
518
519     # create list of tuples containing file and archive name
520     for root, dirs, files in os.walk(src):
521         package_root = os.path.join(package_name, root[ln:])
522         flist.extend([(os.path.join(root, file), os.path.join(package_root, file)) for file in files])
523
524     if extension == '.zip':
525         package = zipfile.ZipFile(dest, 'w', zipfile.ZIP_DEFLATED)
526         package.comment = package_name + ' is a zip-file containing the Blender software. Visit http://www.blender.org for more information.'
527         for entry in flist:
528             package.write(entry[0], entry[1])
529         package.close()
530     else:
531         import tarfile
532         package = tarfile.open(dest, 'w:bz2')
533         for entry in flist:
534             package.add(entry[0], entry[1], recursive=False)
535         package.close()
536     bb_zip_name = os.path.normpath(src + os.sep + '..' + os.sep + 'buildbot_upload.zip')
537     print("creating %s" % (bb_zip_name))
538     bb_zip = zipfile.ZipFile(bb_zip_name, 'w', zipfile.ZIP_DEFLATED)
539     print("writing %s to %s" % (dest, bb_zip_name))
540     bb_zip.write(dest, os.path.split(dest)[1])
541     bb_zip.close()
542     print("removing unneeded packed file %s (to keep install directory clean)" % (dest))
543     os.remove(dest)
544     print("done.")
545
546 def buildslave_print(target, source, env):
547     return "Running buildslave target"
548
549 def buildslave(target=None, source=None, env=None):
550     """
551     Builder for buildbot integration. Used by buildslaves of http://builder.blender.org only.
552     """
553
554     if env['OURPLATFORM'] in ('win32-vc', 'win64-vc', 'win32-mingw', 'darwin'):
555         extension = '.zip'
556     else:
557         extension = '.tar.bz2'
558
559     platform = env['OURPLATFORM'].split('-')[0]
560     if platform == 'linux2':
561         import platform
562
563         bitness = platform.architecture()[0]
564         if bitness == '64bit':
565             platform = 'linux-glibc27-x86_64'
566         elif bitness == '32bit':
567             platform = 'linux-glibc27-i686'
568     if platform == 'darwin':
569         platform = 'OSX-' + env['MACOSX_ARCHITECTURE']
570
571     branch = env['BUILDBOT_BRANCH']
572
573     outdir = os.path.abspath(env['BF_INSTALLDIR'])
574     package_name = 'blender-' + VERSION+'-'+REVISION + '-' + platform
575     if branch != '':
576         package_name = branch + '-' + package_name
577     package_dir = os.path.normpath(outdir + os.sep + '..' + os.sep + package_name)
578     package_archive = os.path.normpath(outdir + os.sep + '..' + os.sep + package_name + extension)
579
580     try:
581         if os.path.exists(package_archive):
582             os.remove(package_archive)
583         if os.path.exists(package_dir):
584             shutil.rmtree(package_dir)
585     except Exception, ex:
586         sys.stderr.write('Failed to clean up old package files: ' + str(ex) + '\n')
587         return 1
588
589     buildbot_zip(outdir, package_archive, package_name, extension)
590
591     return 0
592
593 def NSIS_print(target, source, env):
594     return "Creating NSIS installer for Blender"
595
596 def NSIS_Installer(target=None, source=None, env=None):
597     print "="*35
598
599     if env['OURPLATFORM'] not in ('win32-vc', 'win32-mingw', 'win64-vc'):
600         print "NSIS installer is only available on Windows."
601         Exit()
602     if env['OURPLATFORM'] == 'win32-vc':
603         bitness = '32'
604     elif env['OURPLATFORM'] == 'win64-vc':
605         bitness = '64'
606     else:
607         bitness = '-mingw'
608
609     start_dir = os.getcwd()
610     rel_dir = os.path.join(start_dir,'release','windows','installer')
611     install_base_dir = start_dir + os.sep
612
613     bf_installdir = os.path.join(os.getcwd(),env['BF_INSTALLDIR'])
614     bf_installdir = os.path.normpath(bf_installdir)
615
616     doneroot = False
617     rootdirconts = []
618     datafiles = ''
619     deldatafiles = ''
620     deldatadirs = ''
621     l = len(bf_installdir)
622     
623     for dp,dn,df in os.walk(bf_installdir):
624         # install
625         if not doneroot:
626             for f in df:
627                 rootdirconts.append(os.path.join(dp,f))
628             doneroot = True
629         else:
630             if len(df)>0:
631                 dp_tmp = dp[l:]
632                 datafiles += "\n" +r'SetOutPath $INSTDIR'+dp[l:]+"\n\n"
633
634                 for f in df:
635                     outfile = os.path.join(dp,f)
636                     datafiles += '  File '+outfile + "\n"
637
638         # uninstall
639         deldir = dp[l+1:]
640
641         if len(deldir)>0:
642             deldatadirs = "RMDir $INSTDIR\\" + deldir + "\n" + deldatadirs
643             deldatadirs = "RMDir /r $INSTDIR\\" + deldir + "\\__pycache__\n" + deldatadirs
644
645             for f in df:
646                 deldatafiles += 'Delete \"$INSTDIR\\' + os.path.join(deldir, f) + "\"\n"
647
648     #### change to suit install dir ####
649     inst_dir = install_base_dir + env['BF_INSTALLDIR']
650     
651     os.chdir(rel_dir)
652
653     ns = open("00.sconsblender.nsi","r")
654
655     ns_cnt = str(ns.read())
656     ns.close()
657
658     # var replacements
659     ns_cnt = string.replace(ns_cnt, "[DISTDIR]", os.path.normpath(inst_dir+os.sep))
660     ns_cnt = string.replace(ns_cnt, "[VERSION]", VERSION_DISPLAY)
661     ns_cnt = string.replace(ns_cnt, "[SHORTVERSION]", VERSION)
662     ns_cnt = string.replace(ns_cnt, "[RELDIR]", os.path.normpath(rel_dir))
663     ns_cnt = string.replace(ns_cnt, "[BITNESS]", bitness)
664
665     # do root
666     rootlist = []
667     for rootitem in rootdirconts:
668         rootlist.append("File \"" + rootitem + "\"")
669     rootstring = string.join(rootlist, "\n  ")
670     rootstring = rootstring
671     rootstring += "\n\n"
672     ns_cnt = string.replace(ns_cnt, "[ROOTDIRCONTS]", rootstring)
673
674
675     # do delete items
676     delrootlist = []
677     for rootitem in rootdirconts:
678         delrootlist.append("Delete $INSTDIR\\" + rootitem[l+1:])
679     delrootstring = string.join(delrootlist, "\n ")
680     delrootstring += "\n"
681     ns_cnt = string.replace(ns_cnt, "[DELROOTDIRCONTS]", delrootstring)
682
683     ns_cnt = string.replace(ns_cnt, "[DODATAFILES]", datafiles)
684     ns_cnt = string.replace(ns_cnt, "[DELDATAFILES]", deldatafiles)
685     ns_cnt = string.replace(ns_cnt, "[DELDATADIRS]", deldatadirs)
686
687     tmpnsi = os.path.normpath(install_base_dir+os.sep+env['BF_BUILDDIR']+os.sep+"00.blender_tmp.nsi")
688     new_nsis = open(tmpnsi, 'w')
689     new_nsis.write(ns_cnt)
690     new_nsis.close()
691     print "NSIS Installer script created"
692
693     os.chdir(start_dir)
694     print "Launching 'makensis'"
695
696     cmdline = "makensis " + "\""+tmpnsi+"\""
697
698     startupinfo = subprocess.STARTUPINFO()
699     startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
700     proc = subprocess.Popen(cmdline, stdin=subprocess.PIPE, stdout=subprocess.PIPE,
701         stderr=subprocess.PIPE, startupinfo=startupinfo, shell = True)
702     data, err = proc.communicate()
703     rv = proc.wait()
704
705     if rv != 0:
706         print
707         print data.strip().split("\n")[-1]
708     return rv
709
710 def check_environ():
711     problematic_envvars = ""
712     for i in os.environ:
713         try:
714             os.environ[i].decode('ascii')
715         except UnicodeDecodeError:
716             problematic_envvars = problematic_envvars + "%s = %s\n" % (i, os.environ[i])
717     if len(problematic_envvars)>0:
718         print("================\n\n")
719         print("@@ ABORTING BUILD @@\n")
720         print("PROBLEM DETECTED WITH ENVIRONMENT")
721         print("---------------------------------\n\n")
722         print("A problem with one or more environment variable was found")
723         print("Their value contain non-ascii characters. Check the below")
724         print("list and override them locally to be ASCII-clean by doing")
725         print("'set VARNAME=cleanvalue' on the command-line prior to")
726         print("starting the build process:\n")
727         print(problematic_envvars)
728         return False
729     else:
730         return True