108e011621216182d2f6251f7e275f49c4e8d63b
[dcpomatic.git] / wscript
1 #
2 #    Copyright (C) 2012-2019 Carl Hetherington <cth@carlh.net>
3 #
4 #    This file is part of DCP-o-matic.
5 #
6 #    DCP-o-matic is free software; you can redistribute it and/or modify
7 #    it under the terms of the GNU General Public License as published by
8 #    the Free Software Foundation; either version 2 of the License, or
9 #    (at your option) any later version.
10 #
11 #    DCP-o-matic is distributed in the hope that it will be useful,
12 #    but WITHOUT ANY WARRANTY; without even the implied warranty of
13 #    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 #    GNU General Public License for more details.
15 #
16 #    You should have received a copy of the GNU General Public License
17 #    along with DCP-o-matic.  If not, see <http://www.gnu.org/licenses/>.
18 #
19
20 from __future__ import print_function
21
22 import subprocess
23 import os
24 import shlex
25 import sys
26 import glob
27 import distutils
28 import distutils.spawn
29 try:
30     # python 2
31     from urllib import urlencode
32 except ImportError:
33     # python 3
34     from urllib.parse import urlencode
35 from waflib import Logs, Context
36
37 APPNAME = 'dcpomatic'
38
39 this_version = subprocess.Popen(shlex.split('git tag -l --points-at HEAD'), stdout=subprocess.PIPE).communicate()[0]
40 last_version = subprocess.Popen(shlex.split('git describe --tags --abbrev=0'), stdout=subprocess.PIPE).communicate()[0]
41
42 # Python 2/3 compatibility; I don't really understand what's going on here
43 if not isinstance(this_version, str):
44     this_version = this_version.decode('utf-8')
45 if not isinstance(last_version, str):
46     last_version = last_version.decode('utf-8')
47
48 if this_version == '':
49     VERSION = '%sdevel' % last_version[1:].strip()
50 else:
51     VERSION = this_version[1:].strip()
52
53 def options(opt):
54     opt.load('compiler_cxx')
55     opt.load('winres')
56
57     opt.add_option('--enable-debug',      action='store_true', default=False, help='build with debugging information and without optimisation')
58     opt.add_option('--disable-gui',       action='store_true', default=False, help='disable building of GUI tools')
59     opt.add_option('--disable-tests',     action='store_true', default=False, help='disable building of tests')
60     opt.add_option('--install-prefix',                         default=None,  help='prefix of where DCP-o-matic will be installed')
61     opt.add_option('--target-windows',    action='store_true', default=False, help='set up to do a cross-compile to make a Windows package')
62     opt.add_option('--static-dcpomatic',  action='store_true', default=False, help='link to components of DCP-o-matic statically')
63     opt.add_option('--static-boost',      action='store_true', default=False, help='link statically to Boost')
64     opt.add_option('--static-wxwidgets',  action='store_true', default=False, help='link statically to wxWidgets')
65     opt.add_option('--static-ffmpeg',     action='store_true', default=False, help='link statically to FFmpeg')
66     opt.add_option('--static-xmlpp',      action='store_true', default=False, help='link statically to libxml++')
67     opt.add_option('--static-xmlsec',     action='store_true', default=False, help='link statically to xmlsec')
68     opt.add_option('--static-ssh',        action='store_true', default=False, help='link statically to libssh')
69     opt.add_option('--static-cxml',       action='store_true', default=False, help='link statically to libcxml')
70     opt.add_option('--static-dcp',        action='store_true', default=False, help='link statically to libdcp')
71     opt.add_option('--static-sub',        action='store_true', default=False, help='link statically to libsub')
72     opt.add_option('--static-curl',       action='store_true', default=False, help='link statically to libcurl')
73     opt.add_option('--workaround-gssapi', action='store_true', default=False, help='link to gssapi_krb5')
74     opt.add_option('--force-cpp11',       action='store_true', default=False, help='force use of C++11')
75     opt.add_option('--variant',           help='build variant (swaroop-studio, swaroop-theater)', choices=['swaroop-studio', 'swaroop-theater'])
76     opt.add_option('--enable-player-stress-test', action='store_true', default=False, help='build the player with stress testing enabled') 
77     opt.add_option('--use-lld',           action='store_true', default=False, help='use lld linker')
78
79 def configure(conf):
80     conf.load('compiler_cxx')
81     conf.load('clang_compilation_database', tooldir=['waf-tools'])
82     if conf.options.target_windows:
83         conf.load('winres')
84
85     # Save conf.options that we need elsewhere in conf.env
86     conf.env.DISABLE_GUI = conf.options.disable_gui
87     conf.env.DISABLE_TESTS = conf.options.disable_tests
88     conf.env.TARGET_WINDOWS = conf.options.target_windows
89     conf.env.TARGET_OSX = sys.platform == 'darwin'
90     conf.env.TARGET_LINUX = not conf.env.TARGET_WINDOWS and not conf.env.TARGET_OSX
91     conf.env.VERSION = VERSION
92     conf.env.DEBUG = conf.options.enable_debug
93     conf.env.STATIC_DCPOMATIC = conf.options.static_dcpomatic
94     if conf.options.install_prefix is None:
95         conf.env.INSTALL_PREFIX = conf.env.PREFIX
96     else:
97         conf.env.INSTALL_PREFIX = conf.options.install_prefix
98
99     # Common CXXFLAGS
100     conf.env.append_value('CXXFLAGS', ['-D__STDC_CONSTANT_MACROS',
101                                        '-D__STDC_LIMIT_MACROS',
102                                        '-D__STDC_FORMAT_MACROS',
103                                        '-msse',
104                                        '-fno-strict-aliasing',
105                                        '-Wall',
106                                        '-Wextra',
107                                        '-Wwrite-strings',
108                                        # Remove auto_ptr warnings from libxml++-2.6
109                                        '-Wno-deprecated-declarations',
110                                        '-Wno-ignored-qualifiers',
111                                        '-Wno-parentheses',
112                                        '-D_FILE_OFFSET_BITS=64'])
113
114     if conf.options.force_cpp11:
115         conf.env.append_value('CXXFLAGS', ['-std=c++11', '-DBOOST_NO_CXX11_SCOPED_ENUMS'])
116
117     if conf.env['CXX_NAME'] == 'gcc':
118         gcc = conf.env['CC_VERSION']
119         if int(gcc[0]) >= 4 and int(gcc[1]) > 1:
120             conf.env.append_value('CXXFLAGS', ['-Wno-unused-result'])
121         if int(gcc[0]) >= 9:
122             conf.env.append_value('CXXFLAGS', ['-Wno-deprecated-copy'])
123         have_c11 = int(gcc[0]) >= 4 and int(gcc[1]) >= 8 and int(gcc[2]) >= 1
124     else:
125         have_c11 = False
126
127     if conf.options.enable_debug:
128         conf.env.append_value('CXXFLAGS', ['-g', '-DDCPOMATIC_DEBUG', '-fno-omit-frame-pointer'])
129     else:
130         conf.env.append_value('CXXFLAGS', '-O2')
131
132     if conf.options.variant is not None:
133         conf.env.VARIANT = conf.options.variant
134         if conf.options.variant.startswith('swaroop-'):
135             conf.env.append_value('CXXFLAGS', '-DDCPOMATIC_VARIANT_SWAROOP')
136
137     if conf.options.enable_player_stress_test:
138         conf.env.append_value('CXXFLAGS', '-DDCPOMATIC_PLAYER_STRESS_TEST')
139
140     if conf.options.use_lld:
141         try:
142             conf.find_program('ld.lld')
143             conf.env.append_value('LINKFLAGS', '-fuse-ld=lld')
144         except conf.errors.ConfigurationError:
145             pass
146
147     #
148     # Windows/Linux/OS X specific
149     #
150
151     # Windows
152     if conf.env.TARGET_WINDOWS:
153         conf.env.append_value('CXXFLAGS', '-DDCPOMATIC_WINDOWS')
154         conf.env.append_value('CXXFLAGS', '-DWIN32_LEAN_AND_MEAN')
155         conf.env.append_value('CXXFLAGS', '-DBOOST_USE_WINDOWS_H')
156         conf.env.append_value('CXXFLAGS', '-DUNICODE')
157         conf.env.append_value('CXXFLAGS', '-DBOOST_THREAD_PROVIDES_GENERIC_SHARED_MUTEX_ON_WIN')
158         conf.env.append_value('CXXFLAGS', '-mfpmath=sse')
159         conf.env.append_value('CXXFLAGS', '-std=c++11')
160         conf.env.append_value('CXXFLAGS', '-Wcast-align')
161         wxrc = os.popen('wx-config --rescomp').read().split()[1:]
162         conf.env.append_value('WINRCFLAGS', wxrc)
163         if conf.options.enable_debug:
164             conf.env.append_value('CXXFLAGS', ['-mconsole'])
165             conf.env.append_value('LINKFLAGS', ['-mconsole'])
166         conf.check(lib='ws2_32', uselib_store='WINSOCK2', msg="Checking for library winsock2")
167         conf.check(lib='dbghelp', uselib_store='DBGHELP', msg="Checking for library dbghelp")
168         conf.check(lib='shlwapi', uselib_store='SHLWAPI', msg="Checking for library shlwapi")
169         conf.check(lib='mswsock', uselib_store='MSWSOCK', msg="Checking for library mswsock")
170         conf.check(lib='ole32', uselib_store='OLE32', msg="Checking for library ole32")
171         conf.check(lib='dsound', uselib_store='DSOUND', msg="Checking for library dsound")
172         conf.check(lib='winmm', uselib_store='WINMM', msg="Checking for library winmm")
173         conf.check(lib='ksuser', uselib_store='KSUSER', msg="Checking for library ksuser")
174         boost_lib_suffix = '-mt'
175         boost_thread = 'boost_thread-mt'
176         conf.check_cxx(fragment="""
177                                #include <boost/locale.hpp>\n
178                                int main() { std::locale::global (boost::locale::generator().generate ("")); }\n
179                                """,
180                                msg='Checking for boost locale library',
181                                libpath='/usr/local/lib',
182                                lib=['boost_locale%s' % boost_lib_suffix, 'boost_system%s' % boost_lib_suffix],
183                                uselib_store='BOOST_LOCALE')
184         conf.env.append_value('CXXFLAGS', ['-DBOOST_STACKTRACE_LINK', '-DBOOST_STACKTRACE_USE_BACKTRACE'])
185         conf.check(lib='dl', uselib_store='DL', msg="Checking for library dl")
186         conf.check(lib='backtrace', uselib_store='BACKTRACE', msg="Checking for library backtrace")
187
188     # POSIX
189     if conf.env.TARGET_LINUX or conf.env.TARGET_OSX:
190         conf.env.append_value('CXXFLAGS', '-DDCPOMATIC_POSIX')
191         boost_lib_suffix = ''
192         boost_thread = 'boost_thread'
193         conf.env.append_value('LINKFLAGS', ['-pthread'])
194
195     # Linux
196     if conf.env.TARGET_LINUX:
197         conf.env.append_value('CXXFLAGS', '-mfpmath=sse')
198         conf.env.append_value('CXXFLAGS', '-DLINUX_LOCALE_PREFIX="%s/share/locale"' % conf.env['INSTALL_PREFIX'])
199         conf.env.append_value('CXXFLAGS', '-DLINUX_SHARE_PREFIX="%s/share/dcpomatic2"' % conf.env['INSTALL_PREFIX'])
200         conf.env.append_value('CXXFLAGS', '-DDCPOMATIC_LINUX')
201         conf.env.append_value('CXXFLAGS', ['-Wlogical-op', '-Wcast-align'])
202         conf.env.append_value('CXXFLAGS', '-DBOOST_STACKTRACE_USE_BACKTRACE')
203         conf.check(lib='dl', uselib_store='DL', msg="Checking for library dl")
204         conf.check(lib='backtrace', uselib_store='BACKTRACE', msg="Checking for library backtrace")
205         if not conf.env.DISABLE_GUI:
206             conf.check_cfg(package='gtk+-2.0', args='--cflags --libs', uselib_store='GTK', mandatory=True)
207
208     # OSX
209     if conf.env.TARGET_OSX:
210         conf.env.append_value('CXXFLAGS', ['-DDCPOMATIC_OSX', '-Wno-unused-function', '-Wno-unused-parameter', '-Wno-unused-local-typedef', '-Wno-potentially-evaluated-expression'])
211         conf.env.append_value('LINKFLAGS', '-headerpad_max_install_names')
212     else:
213         # Avoid the endless warnings about _t uninitialized in optional<>
214         conf.env.append_value('CXXFLAGS', '-Wno-maybe-uninitialized')
215
216     #
217     # Dependencies.
218     #
219
220     # It should be possible to use check_cfg for both dynamic and static linking, but
221     # e.g. pkg-config --libs --static foo returns some libraries that should be statically
222     # linked and others that should be dynamic.  This doesn't work too well with waf
223     # as it wants them separate.
224
225     # libcurl
226     if conf.options.static_curl:
227         conf.env.STLIB_CURL = ['curl']
228         conf.env.LIB_CURL = ['ssh2', 'idn']
229     else:
230         conf.check_cfg(package='libcurl', args='--cflags --libs', atleast_version='7.19.1', uselib_store='CURL', mandatory=True)
231
232     # libicu
233     if conf.check_cfg(package='icu-i18n', args='--cflags --libs', uselib_store='ICU', mandatory=False) is None:
234         if conf.check_cfg(package='icu', args='--cflags --libs', uselib_store='ICU', mandatory=False) is None:
235             conf.check_cxx(fragment="""
236                             #include <unicode/ucsdet.h>
237                             int main(void) {
238                                 UErrorCode status = U_ZERO_ERROR;
239                                 UCharsetDetector* detector = ucsdet_open (&status);
240                                 return 0; }\n
241                             """,
242                        mandatory=True,
243                        msg='Checking for libicu',
244                        okmsg='yes',
245                        libpath=['/usr/local/lib', '/usr/lib', '/usr/lib/x86_64-linux-gnu'],
246                        lib=['icuio', 'icui18n', 'icudata', 'icuuc'],
247                        uselib_store='ICU')
248
249     # libsamplerate
250     conf.check_cfg(package='samplerate', args='--cflags --libs', uselib_store='SAMPLERATE', mandatory=True)
251
252     # glib
253     conf.check_cfg(package='glib-2.0', args='--cflags --libs', uselib_store='GLIB', mandatory=True)
254
255     # libzip
256     conf.check_cfg(package='libzip', args='--cflags --libs', uselib_store='ZIP', mandatory=True)
257     conf.check_cxx(fragment="""
258                             #include <zip.h>
259                             int main() { zip_source_t* foo; }
260                             """,
261                    mandatory=False,
262                    msg="Checking for zip_source_t",
263                    uselib="ZIP",
264                    define_name='DCPOMATIC_HAVE_ZIP_SOURCE_T'
265                    )
266
267     # fontconfig
268     conf.check_cfg(package='fontconfig', args='--cflags --libs', uselib_store='FONTCONFIG', mandatory=True)
269
270     # pangomm
271     conf.check_cfg(package='pangomm-1.4', args='--cflags --libs', uselib_store='PANGOMM', mandatory=True)
272
273     # cairomm
274     conf.check_cfg(package='cairomm-1.0', args='--cflags --libs', uselib_store='CAIROMM', mandatory=True)
275
276     test_cxxflags = ''
277     if have_c11:
278         test_cxxflags = '-std=c++11'
279
280     # See if we have Cairo::ImageSurface::format_stride_for_width; Centos 5 does not
281     conf.check_cxx(fragment="""
282                             #include <cairomm/cairomm.h>
283                             int main(void) {
284                                 Cairo::ImageSurface::format_stride_for_width (Cairo::FORMAT_ARGB32, 1024);\n
285                                 return 0; }\n
286                             """,
287                        mandatory=False,
288                        cxxflags=test_cxxflags,
289                        msg='Checking for format_stride_for_width',
290                        okmsg='yes',
291                        includes=conf.env['INCLUDES_CAIROMM'],
292                        uselib='CAIROMM',
293                        define_name='DCPOMATIC_HAVE_FORMAT_STRIDE_FOR_WIDTH')
294
295     # See if we have Pango::Layout::show_in_cairo_context; Centos 5 does not
296     conf.check_cxx(fragment="""
297                             #include <pangomm.h>
298                             int main(void) {
299                                 Cairo::RefPtr<Cairo::Context> context;
300                                 Glib::RefPtr<Pango::Layout> layout;
301                                 layout->show_in_cairo_context (context);
302                                 return 0; }\n
303                             """,
304                        mandatory=False,
305                        msg='Checking for show_in_cairo_context',
306                        cxxflags=test_cxxflags,
307                        okmsg='yes',
308                        includes=conf.env['INCLUDES_PANGOMM'],
309                        uselib='PANGOMM',
310                        define_name='DCPOMATIC_HAVE_SHOW_IN_CAIRO_CONTEXT')
311
312
313     # libcxml
314     if conf.options.static_cxml:
315         conf.check_cfg(package='libcxml', atleast_version='0.16.0', args='--cflags', uselib_store='CXML', mandatory=True)
316         conf.env.STLIB_CXML = ['cxml']
317     else:
318         conf.check_cfg(package='libcxml', atleast_version='0.16.0', args='--cflags --libs', uselib_store='CXML', mandatory=True)
319
320     # libssh
321     if conf.options.static_ssh:
322         conf.env.STLIB_SSH = ['ssh']
323         if conf.options.workaround_gssapi:
324             conf.env.LIB_SSH = ['gssapi_krb5']
325     else:
326         conf.check_cc(fragment="""
327                                #include <libssh/libssh.h>\n
328                                int main () {\n
329                                ssh_session s = ssh_new ();\n
330                                return 0;\n
331                                }
332                                """,
333                       msg='Checking for library libssh',
334                       mandatory=True,
335                       lib='ssh',
336                       uselib_store='SSH')
337
338     # libdcp
339     if conf.options.static_dcp:
340         conf.check_cfg(package='libdcp-1.0', atleast_version='1.6.7', args='--cflags', uselib_store='DCP', mandatory=True)
341         conf.env.DEFINES_DCP = [f.replace('\\', '') for f in conf.env.DEFINES_DCP]
342         conf.env.STLIB_DCP = ['dcp-1.0', 'asdcp-carl', 'kumu-carl', 'openjp2']
343         conf.env.LIB_DCP = ['glibmm-2.4', 'ssl', 'crypto', 'bz2', 'xslt', 'xerces-c']
344     else:
345         conf.check_cfg(package='libdcp-1.0', atleast_version='1.6.7', args='--cflags --libs', uselib_store='DCP', mandatory=True)
346         conf.env.DEFINES_DCP = [f.replace('\\', '') for f in conf.env.DEFINES_DCP]
347
348     # libsub
349     if conf.options.static_sub:
350         conf.check_cfg(package='libsub-1.0', atleast_version='1.4.7', args='--cflags', uselib_store='SUB', mandatory=True)
351         conf.env.DEFINES_SUB = [f.replace('\\', '') for f in conf.env.DEFINES_SUB]
352         conf.env.STLIB_SUB = ['sub-1.0']
353     else:
354         conf.check_cfg(package='libsub-1.0', atleast_version='1.4.7', args='--cflags --libs', uselib_store='SUB', mandatory=True)
355         conf.env.DEFINES_SUB = [f.replace('\\', '') for f in conf.env.DEFINES_SUB]
356
357     # libxml++
358     if conf.options.static_xmlpp:
359         conf.env.STLIB_XMLPP = ['xml++-2.6']
360         conf.env.LIB_XMLPP = ['xml2']
361     else:
362         conf.check_cfg(package='libxml++-2.6', args='--cflags --libs', uselib_store='XMLPP', mandatory=True)
363
364     # libxmlsec
365     if conf.options.static_xmlsec:
366         if conf.check_cxx(lib='xmlsec1-openssl', mandatory=False):
367             conf.env.STLIB_XMLSEC = ['xmlsec1-openssl', 'xmlsec1']
368         else:
369             conf.env.STLIB_XMLSEC = ['xmlsec1']
370     else:
371         conf.env.LIB_XMLSEC = ['xmlsec1-openssl', 'xmlsec1']
372
373     # nettle
374     conf.check_cfg(package="nettle", args='--cflags --libs', uselib_store='NETTLE', mandatory=True)
375
376     # libpng
377     conf.check_cfg(package='libpng', args='--cflags --libs', uselib_store='PNG', mandatory=True)
378
379     # FFmpeg
380     if conf.options.static_ffmpeg:
381         names = ['avformat', 'avfilter', 'avcodec', 'avutil', 'swscale', 'postproc', 'swresample']
382         for name in names:
383             static = subprocess.Popen(shlex.split('pkg-config --static --libs lib%s' % name), stdout=subprocess.PIPE).communicate()[0].decode('utf-8')
384             libs = []
385             stlibs = []
386             include = []
387             libpath = []
388             for s in static.split():
389                 if s.startswith('-L'):
390                     libpath.append(s[2:])
391                 elif s.startswith('-I'):
392                     include.append(s[2:])
393                 elif s.startswith('-l'):
394                     if s[2:] not in names:
395                         libs.append(s[2:])
396                     else:
397                         stlibs.append(s[2:])
398
399             conf.env['LIB_%s' % name.upper()] = libs
400             conf.env['STLIB_%s' % name.upper()] = stlibs
401             conf.env['INCLUDES_%s' % name.upper()] = include
402             conf.env['LIBPATH_%s' % name.upper()] = libpath
403     else:
404         conf.check_cfg(package='libavformat', args='--cflags --libs', uselib_store='AVFORMAT', mandatory=True)
405         conf.check_cfg(package='libavfilter', args='--cflags --libs', uselib_store='AVFILTER', mandatory=True)
406         conf.check_cfg(package='libavcodec', args='--cflags --libs', uselib_store='AVCODEC', mandatory=True)
407         conf.check_cfg(package='libavutil', args='--cflags --libs', uselib_store='AVUTIL', mandatory=True)
408         conf.check_cfg(package='libswscale', args='--cflags --libs', uselib_store='SWSCALE', mandatory=True)
409         conf.check_cfg(package='libpostproc', args='--cflags --libs', uselib_store='POSTPROC', mandatory=True)
410         conf.check_cfg(package='libswresample', args='--cflags --libs', uselib_store='SWRESAMPLE', mandatory=True)
411
412     # Check to see if we have our version of FFmpeg that allows us to get at EBUR128 results
413     conf.check_cxx(fragment="""
414                             extern "C" {\n
415                             #include <libavfilter/f_ebur128.h>\n
416                             }\n
417                             int main () { av_ebur128_get_true_peaks (0); }\n
418                             """,
419                    msg='Checking for EBUR128-patched FFmpeg',
420                    uselib='AVCODEC AVFILTER',
421                    define_name='DCPOMATIC_HAVE_EBUR128_PATCHED_FFMPEG',
422                    mandatory=False)
423
424     # Check to see if we have our AVSubtitleRect has a pict member
425     # Older versions (e.g. that shipped with Ubuntu 16.04) do
426     conf.check_cxx(fragment="""
427                             extern "C" {\n
428                             #include <libavcodec/avcodec.h>\n
429                             }\n
430                             int main () { AVSubtitleRect r; r.pict; }\n
431                             """,
432                    msg='Checking for AVSubtitleRect::pict',
433                    cxxflags='-Wno-unused-result -Wno-unused-value -Wdeprecated-declarations -Werror',
434                    uselib='AVCODEC',
435                    define_name='DCPOMATIC_HAVE_AVSUBTITLERECT_PICT',
436                    mandatory=False)
437
438     # Check to see if we have our AVComponentDescriptor has a depth_minus1 member
439     # Older versions (e.g. that shipped with Ubuntu 16.04) do
440     conf.check_cxx(fragment="""
441                             extern "C" {\n
442                             #include <libavutil/pixdesc.h>\n
443                             }\n
444                             int main () { AVComponentDescriptor d; d.depth_minus1; }\n
445                             """,
446                    msg='Checking for AVComponentDescriptor::depth_minus1',
447                    cxxflags='-Wno-unused-result -Wno-unused-value -Wdeprecated-declarations -Werror',
448                    uselib='AVUTIL',
449                    define_name='DCPOMATIC_HAVE_AVCOMPONENTDESCRIPTOR_DEPTH_MINUS1',
450                    mandatory=False)
451
452     # Hack: the previous two check_cxx calls end up copying their (necessary) cxxflags
453     # to these variables.  We don't want to use these for the actual build, so clean them out.
454     conf.env['CXXFLAGS_AVCODEC'] = []
455     conf.env['CXXFLAGS_AVUTIL'] = []
456
457     if conf.env.TARGET_LINUX:
458         conf.env.LIB_X11 = ['X11']
459
460     # Boost
461     if conf.options.static_boost:
462         conf.env.STLIB_BOOST_THREAD = ['boost_thread']
463         conf.env.STLIB_BOOST_FILESYSTEM = ['boost_filesystem%s' % boost_lib_suffix]
464         conf.env.STLIB_BOOST_DATETIME = ['boost_date_time%s' % boost_lib_suffix, 'boost_system%s' % boost_lib_suffix]
465         conf.env.STLIB_BOOST_SIGNALS2 = ['boost_signals2']
466         conf.env.STLIB_BOOST_SYSTEM = ['boost_system']
467         conf.env.STLIB_BOOST_REGEX = ['boost_regex']
468     else:
469         conf.check_cxx(fragment="""
470                             #include <boost/version.hpp>\n
471                             #if BOOST_VERSION < 104500\n
472                             #error boost too old\n
473                             #endif\n
474                             int main(void) { return 0; }\n
475                             """,
476                        mandatory=True,
477                        msg='Checking for boost library >= 1.45',
478                        okmsg='yes',
479                        errmsg='too old\nPlease install boost version 1.45 or higher.')
480
481         conf.check_cxx(fragment="""
482                             #include <boost/thread.hpp>\n
483                             int main() { boost::thread t (); }\n
484                             """,
485                        msg='Checking for boost threading library',
486                        libpath='/usr/local/lib',
487                        lib=[boost_thread, 'boost_system%s' % boost_lib_suffix],
488                        uselib_store='BOOST_THREAD')
489
490         conf.check_cxx(fragment="""
491                             #include <boost/filesystem.hpp>\n
492                             int main() { boost::filesystem::copy_file ("a", "b"); }\n
493                             """,
494                        msg='Checking for boost filesystem library',
495                        libpath='/usr/local/lib',
496                        lib=['boost_filesystem%s' % boost_lib_suffix, 'boost_system%s' % boost_lib_suffix],
497                        uselib_store='BOOST_FILESYSTEM')
498
499         conf.check_cxx(fragment="""
500                             #include <boost/date_time.hpp>\n
501                             int main() { boost::gregorian::day_clock::local_day(); }\n
502                             """,
503                        msg='Checking for boost datetime library',
504                        libpath='/usr/local/lib',
505                        lib=['boost_date_time%s' % boost_lib_suffix, 'boost_system%s' % boost_lib_suffix],
506                        uselib_store='BOOST_DATETIME')
507
508         conf.check_cxx(fragment="""
509                             #include <boost/signals2.hpp>\n
510                             int main() { boost::signals2::signal<void (int)> x; }\n
511                             """,
512                        msg='Checking for boost signals2 library',
513                        uselib_store='BOOST_SIGNALS2')
514
515         conf.check_cxx(fragment="""
516                             #include <boost/regex.hpp>\n
517                             int main() { boost::regex re ("foo"); }\n
518                             """,
519                        msg='Checking for boost regex library',
520                        lib=['boost_regex%s' % boost_lib_suffix],
521                        uselib_store='BOOST_REGEX')
522
523     # libxml++ requires glibmm and versions of glibmm 2.45.31 and later
524     # must be built with -std=c++11 as they use c++11
525     # features and c++11 is not (yet) the default in gcc.
526     glibmm_version = conf.cmd_and_log(['pkg-config', '--modversion', 'glibmm-2.4'], output=Context.STDOUT, quiet=Context.BOTH)
527     s = glibmm_version.split('.')
528     v = (int(s[0]) << 16) | (int(s[1]) << 8) | int(s[2])
529     if v >= 0x022D1F:
530         conf.env.append_value('CXXFLAGS', '-std=c++11')
531
532     # Other stuff
533
534     conf.find_program('msgfmt', var='MSGFMT')
535     conf.check(header_name='valgrind/memcheck.h', mandatory=False)
536
537     datadir = conf.env.DATADIR
538     if not datadir:
539         datadir = os.path.join(conf.env.PREFIX, 'share')
540
541     conf.define('LOCALEDIR', os.path.join(datadir, 'locale'))
542     conf.define('DATADIR', datadir)
543
544     conf.recurse('src')
545     if not conf.env.DISABLE_TESTS:
546         conf.recurse('test')
547
548     Logs.pprint('YELLOW', '')
549     if conf.env.TARGET_WINDOWS:
550         Logs.pprint('YELLOW', '\t' + 'Target'.ljust(25) + ': Windows')
551     elif conf.env.TARGET_LINUX:
552         Logs.pprint('YELLOW', '\t' + 'Target'.ljust(25) + ': Linux')
553     elif conf.env.TARGET_OSX:
554         Logs.pprint('YELLOW', '\t' + 'Target'.ljust(25) + ': OS X')
555
556     def report(name, variable):
557         linkage = ''
558         if variable:
559             linkage = 'static'
560         else:
561             linkage = 'dynamic'
562         Logs.pprint('YELLOW', '\t%s: %s' % (name.ljust(25), linkage))
563
564     report('DCP-o-matic libraries', conf.options.static_dcpomatic)
565     report('Boost', conf.options.static_boost)
566     report('wxWidgets', conf.options.static_wxwidgets)
567     report('FFmpeg', conf.options.static_ffmpeg)
568     report('libxml++', conf.options.static_xmlpp)
569     report('xmlsec', conf.options.static_xmlsec)
570     report('libssh', conf.options.static_ssh)
571     report('libcxml', conf.options.static_cxml)
572     report('libdcp', conf.options.static_dcp)
573     report('libcurl', conf.options.static_curl)
574
575     Logs.pprint('YELLOW', '')
576
577 def download_supporters(can_fail):
578     r = os.system('curl -m 2 -s -f https://dcpomatic.com/supporters.cc > src/wx/supporters.cc')
579     if (r >> 8) == 0:
580         r = os.system('curl -s -f https://dcpomatic.com/subscribers.cc > src/wx/subscribers.cc')
581     if (r >> 8) != 0:
582         if can_fail:
583             raise Exception("Could not download supporters lists (%d)" % (r >> 8))
584         else:
585             f = open('src/wx/supporters.cc', 'w')
586             print('supported_by.Add(wxT("Debug build - no supporters lists available"));', file=f)
587             f.close()
588             f = open('src/wx/subscribers.cc', 'w')
589             print('subscribers.Add(wxT("Debug build - no subscribers lists available"));', file=f)
590             f.close()
591
592 def build(bld):
593     create_version_cc(VERSION, bld.env.CXXFLAGS)
594     download_supporters(not bld.env.DEBUG)
595
596     bld.recurse('src')
597     bld.recurse('graphics')
598
599     if not bld.env.DISABLE_TESTS:
600         bld.recurse('test')
601     if bld.env.TARGET_WINDOWS:
602         bld.recurse('platform/windows')
603     if bld.env.TARGET_LINUX:
604         bld.recurse('platform/linux')
605     if bld.env.TARGET_OSX:
606         bld.recurse('platform/osx')
607
608     if not bld.env.TARGET_WINDOWS:
609         bld.install_files('${PREFIX}/share/dcpomatic2', 'fonts/LiberationSans-Regular.ttf')
610         bld.install_files('${PREFIX}/share/dcpomatic2', 'fonts/LiberationSans-Italic.ttf')
611         bld.install_files('${PREFIX}/share/dcpomatic2', 'fonts/LiberationSans-Bold.ttf')
612
613     bld.add_post_fun(post)
614
615 def git_revision():
616     if not os.path.exists('.git'):
617         return None
618
619     cmd = "LANG= git log --abbrev HEAD^..HEAD ."
620     output = subprocess.Popen(cmd, shell=True, stderr=subprocess.STDOUT, stdout=subprocess.PIPE).communicate()[0].splitlines()
621     if len(output) == 0:
622         return None
623     o = output[0].decode('utf-8')
624     return o.replace("commit ", "")[0:10]
625
626 def dist(ctx):
627     r = git_revision()
628     if r is not None:
629         f = open('.git_revision', 'w')
630         print(r, file=f)
631         f.close()
632
633     ctx.excl = """
634                TODO core *~ src/wx/*~ src/lib/*~ builds/*~ doc/manual/*~ src/tools/*~ *.pyc .waf* build .git
635                deps alignment hacks sync *.tar.bz2 *.exe .lock* *build-windows doc/manual/pdf doc/manual/html
636                GRSYMS GRTAGS GSYMS GTAGS compile_commands.json
637                """
638
639 def create_version_cc(version, cxx_flags):
640     commit = git_revision()
641     if commit is None and os.path.exists('.git_revision'):
642         f = open('.git_revision', 'r')
643         commit = f.readline().strip()
644
645     if commit is None:
646         commit = 'release'
647
648     try:
649         text =  '#include "version.h"\n'
650         text += 'char const * dcpomatic_git_commit = \"%s\";\n' % commit
651         text += 'char const * dcpomatic_version = \"%s\";\n' % version
652
653         t = ''
654         for f in cxx_flags:
655             f = f.replace('"', '\\"')
656             t += f + ' '
657         text += 'char const * dcpomatic_cxx_flags = \"%s\";\n' % t[:-1]
658
659         print('Writing version information to src/lib/version.cc')
660         o = open('src/lib/version.cc', 'w')
661         o.write(text)
662         o.close()
663     except IOError:
664         print('Could not open src/lib/version.cc for writing\n')
665         sys.exit(-1)
666
667 def post(ctx):
668     if ctx.cmd == 'install' and ctx.env.TARGET_LINUX:
669         ctx.exec_command('/sbin/ldconfig')
670         # I can't find anything which tells me where things have been installed to,
671         # so here's some nasty hacks to guess.
672         debian = os.path.join(ctx.out_dir, '../debian/dcpomatic/usr/bin/dcpomatic2_uuid')
673         prefix = os.path.join(ctx.env['INSTALL_PREFIX'], 'bin/dcpomatic2_uuid')
674         if os.path.exists(debian):
675             os.chmod(debian, 0o4755)
676         if os.path.exists(prefix):
677             os.chmod(prefix, 0o4755)
678
679 def pot(bld):
680     bld.recurse('src')
681
682 def pot_merge(bld):
683     bld.recurse('src')
684
685 def tags(bld):
686     os.system('etags src/lib/*.cc src/lib/*.h src/wx/*.cc src/wx/*.h src/tools/*.cc')
687
688 def cppcheck(bld):
689     os.system('cppcheck --enable=all --quiet .')