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