Proper fix for mixed mode
[jack2.git] / wscript
blob0621ae7113e24ae99b1e5d1ff8928c81350aa13b
1 #! /usr/bin/python3
2 # encoding: utf-8
3 from __future__ import print_function
5 import os
6 import subprocess
7 import shutil
8 import re
9 import sys
11 from waflib import Logs, Options, Task, Utils
12 from waflib.Build import BuildContext, CleanContext, InstallContext, UninstallContext
14 VERSION='1.9.14'
15 APPNAME='jack'
16 JACK_API_VERSION = '0.1.0'
18 # these variables are mandatory ('/' are converted automatically)
19 top = '.'
20 out = 'build'
22 # lib32 variant name used when building in mixed mode
23 lib32 = 'lib32'
25 def display_feature(conf, msg, build):
26     if build:
27         conf.msg(msg, 'yes', color='GREEN')
28     else:
29         conf.msg(msg, 'no', color='YELLOW')
31 def check_for_celt(conf):
32     found = False
33     for version in ['11', '8', '7', '5']:
34         define = 'HAVE_CELT_API_0_' + version
35         if not found:
36             try:
37                 conf.check_cfg(
38                         package='celt >= 0.%s.0' % version,
39                         args='--cflags --libs')
40                 found = True
41                 conf.define(define, 1)
42                 continue
43             except conf.errors.ConfigurationError:
44                 pass
45         conf.define(define, 0)
47     if not found:
48         raise conf.errors.ConfigurationError
50 def options(opt):
51     # options provided by the modules
52     opt.load('compiler_cxx')
53     opt.load('compiler_c')
54     opt.load('autooptions');
56     opt.load('xcode6')
58     opt.recurse('compat')
60     # install directories
61     opt.add_option('--htmldir', type='string', default=None, help='HTML documentation directory [Default: <prefix>/share/jack-audio-connection-kit/reference/html/')
62     opt.add_option('--libdir', type='string', help='Library directory [Default: <prefix>/lib]')
63     opt.add_option('--libdir32', type='string', help='32bit Library directory [Default: <prefix>/lib32]')
64     opt.add_option('--mandir', type='string', help='Manpage directory [Default: <prefix>/share/man/man1]')
66     # options affecting binaries
67     opt.add_option('--platform', type='string', default=sys.platform, help='Target platform for cross-compiling, e.g. cygwin or win32')
68     opt.add_option('--mixed', action='store_true', default=False, help='Build with 32/64 bits mixed mode')
69     opt.add_option('--debug', action='store_true', default=False, dest='debug', help='Build debuggable binaries')
71     # options affecting general jack functionality
72     opt.add_option('--classic', action='store_true', default=False, help='Force enable standard JACK (jackd) even if D-Bus JACK (jackdbus) is enabled too')
73     opt.add_option('--dbus', action='store_true', default=False, help='Enable D-Bus JACK (jackdbus)')
74     opt.add_option('--autostart', type='string', default='default', help='Autostart method. Possible values: "default", "classic", "dbus", "none"')
75     opt.add_option('--profile', action='store_true', default=False, help='Build with engine profiling')
76     opt.add_option('--clients', default=64, type='int', dest='clients', help='Maximum number of JACK clients')
77     opt.add_option('--ports-per-application', default=768, type='int', dest='application_ports', help='Maximum number of ports per application')
78     opt.add_option('--systemd-unit', action='store_true', default=False, help='Install systemd units.')
80     opt.set_auto_options_define('HAVE_%s')
81     opt.set_auto_options_style('yesno_and_hack')
83     # options with third party dependencies
84     doxygen = opt.add_auto_option(
85             'doxygen',
86             help='Build doxygen documentation',
87             conf_dest='BUILD_DOXYGEN_DOCS',
88             default=False)
89     doxygen.find_program('doxygen')
90     alsa = opt.add_auto_option(
91             'alsa',
92             help='Enable ALSA driver',
93             conf_dest='BUILD_DRIVER_ALSA')
94     alsa.check_cfg(
95             package='alsa >= 1.0.18',
96             args='--cflags --libs')
97     firewire = opt.add_auto_option(
98             'firewire',
99             help='Enable FireWire driver (FFADO)',
100             conf_dest='BUILD_DRIVER_FFADO')
101     firewire.check_cfg(
102             package='libffado >= 1.999.17',
103             args='--cflags --libs')
104     iio = opt.add_auto_option(
105             'iio',
106             help='Enable IIO driver',
107             conf_dest='BUILD_DRIVER_IIO')
108     iio.check_cfg(
109             package='gtkIOStream >= 1.4.0',
110             args='--cflags --libs')
111     iio.check_cfg(
112             package='eigen3 >= 3.1.2',
113             args='--cflags --libs')
114     portaudio = opt.add_auto_option(
115             'portaudio',
116             help='Enable Portaudio driver',
117             conf_dest='BUILD_DRIVER_PORTAUDIO')
118     portaudio.check(header_name='windows.h') # only build portaudio on windows
119     portaudio.check_cfg(
120             package='portaudio-2.0 >= 19',
121             uselib_store='PORTAUDIO',
122             args='--cflags --libs')
123     winmme = opt.add_auto_option(
124             'winmme',
125             help='Enable WinMME driver',
126             conf_dest='BUILD_DRIVER_WINMME')
127     winmme.check(
128             header_name=['windows.h', 'mmsystem.h'],
129             msg='Checking for header mmsystem.h')
131     celt = opt.add_auto_option(
132             'celt',
133             help='Build with CELT')
134     celt.add_function(check_for_celt)
136     # Suffix _PKG to not collide with HAVE_OPUS defined by the option.
137     opus = opt.add_auto_option(
138             'opus',
139             help='Build Opus netjack2')
140     opus.check(header_name='opus/opus_custom.h')
141     opus.check_cfg(
142             package='opus >= 0.9.0',
143             args='--cflags --libs',
144             define_name='HAVE_OPUS_PKG')
146     samplerate = opt.add_auto_option(
147             'samplerate',
148             help='Build with libsamplerate')
149     samplerate.check_cfg(
150             package='samplerate',
151             args='--cflags --libs')
152     sndfile = opt.add_auto_option(
153             'sndfile',
154             help='Build with libsndfile')
155     sndfile.check_cfg(
156             package='sndfile',
157             args='--cflags --libs')
158     readline = opt.add_auto_option(
159             'readline',
160             help='Build with readline')
161     readline.check(lib='readline')
162     readline.check(
163             header_name=['stdio.h', 'readline/readline.h'],
164             msg='Checking for header readline/readline.h')
165     sd = opt.add_auto_option(
166             'systemd',
167             help='Use systemd notify')
168     sd.check(header_name='systemd/sd-daemon.h')
169     sd.check(lib='systemd')
170     db = opt.add_auto_option(
171             'db',
172             help='Use Berkeley DB (metadata)')
173     db.check(header_name='db.h')
174     db.check(lib='db')
176     # dbus options
177     opt.recurse('dbus')
179     # this must be called before the configure phase
180     opt.apply_auto_options_hack()
182 def detect_platform(conf):
183     # GNU/kFreeBSD and GNU/Hurd are treated as Linux
184     platforms = [
185         # ('KEY, 'Human readable name', ['strings', 'to', 'check', 'for'])
186         ('IS_LINUX',   'Linux',   ['gnu0', 'gnukfreebsd', 'linux', 'posix']),
187         ('IS_MACOSX',  'MacOS X', ['darwin']),
188         ('IS_SUN',     'SunOS',   ['sunos']),
189         ('IS_WINDOWS', 'Windows', ['cygwin', 'msys', 'win32'])
190     ]
192     for key,name,strings in platforms:
193         conf.env[key] = False
195     conf.start_msg('Checking platform')
196     platform = Options.options.platform
197     for key,name,strings in platforms:
198         for s in strings:
199             if platform.startswith(s):
200                 conf.env[key] = True
201                 conf.end_msg(name, color='CYAN')
202                 break
205 def configure(conf):
206     conf.load('compiler_cxx')
207     conf.load('compiler_c')
209     detect_platform(conf)
211     if conf.env['IS_WINDOWS']:
212         conf.env.append_unique('CCDEFINES', '_POSIX')
213         conf.env.append_unique('CXXDEFINES', '_POSIX')
215     conf.env.append_unique('CXXFLAGS', '-Wall')
216     conf.env.append_unique('CXXFLAGS', '-std=gnu++11')
217     conf.env.append_unique('CFLAGS', '-Wall')
219     if conf.env['IS_MACOSX']:
220         conf.check(lib='aften', uselib='AFTEN', define_name='AFTEN')
221         conf.check_cxx(
222             fragment=''
223                 + '#include <aften/aften.h>\n'
224                 + 'int\n'
225                 + 'main(void)\n'
226                 + '{\n'
227                 + 'AftenContext fAftenContext;\n'
228                 + 'aften_set_defaults(&fAftenContext);\n'
229                 + 'unsigned char *fb;\n'
230                 + 'float *buf=new float[10];\n'
231                 + 'int res = aften_encode_frame(&fAftenContext, fb, buf, 1);\n'
232                 + '}\n',
233             lib='aften',
234             msg='Checking for aften_encode_frame()',
235             define_name='HAVE_AFTEN_NEW_API',
236             mandatory=False)
238     conf.load('autooptions')
240     conf.recurse('compat')
242     # Check for functions.
243     conf.check(
244             fragment=''
245                 + '#define _GNU_SOURCE\n'
246                 + '#include <poll.h>\n'
247                 + '#include <signal.h>\n'
248                 + '#include <stddef.h>\n'
249                 + 'int\n'
250                 + 'main(void)\n'
251                 + '{\n'
252                 + '   ppoll(NULL, 0, NULL, NULL);\n'
253                 + '}\n',
254             msg='Checking for ppoll',
255             define_name='HAVE_PPOLL',
256             mandatory=False)
258     # Check for backtrace support
259     conf.check(
260         header_name='execinfo.h',
261         define_name='HAVE_EXECINFO_H',
262         mandatory=False)
264     conf.recurse('common')
265     if Options.options.dbus:
266         conf.recurse('dbus')
267         if conf.env['BUILD_JACKDBUS'] != True:
268             conf.fatal('jackdbus was explicitly requested but cannot be built')
269     if conf.env['IS_LINUX']:
270         if Options.options.systemd_unit:
271             conf.recurse('systemd')
272         else:
273             conf.env['SYSTEMD_USER_UNIT_DIR'] = None
276     conf.recurse('example-clients')
278     # test for the availability of ucontext, and how it should be used
279     for t in ['gp_regs', 'uc_regs', 'mc_gregs', 'gregs']:
280         fragment = '#include <ucontext.h>\n'
281         fragment += 'int main() { ucontext_t *ucontext; return (int) ucontext->uc_mcontext.%s[0]; }' % t
282         confvar = 'HAVE_UCONTEXT_%s' % t.upper()
283         conf.check_cc(fragment=fragment, define_name=confvar, mandatory=False,
284                       msg='Checking for ucontext->uc_mcontext.%s' % t)
285         if conf.is_defined(confvar):
286             conf.define('HAVE_UCONTEXT', 1)
288     fragment = '#include <ucontext.h>\n'
289     fragment += 'int main() { return NGREG; }'
290     conf.check_cc(fragment=fragment, define_name='HAVE_NGREG', mandatory=False,
291                   msg='Checking for NGREG')
293     conf.env['LIB_PTHREAD'] = ['pthread']
294     conf.env['LIB_DL'] = ['dl']
295     conf.env['LIB_RT'] = ['rt']
296     conf.env['LIB_M'] = ['m']
297     conf.env['LIB_STDC++'] = ['stdc++']
298     conf.env['JACK_API_VERSION'] = JACK_API_VERSION
299     conf.env['JACK_VERSION'] = VERSION
301     conf.env['BUILD_WITH_PROFILE'] = Options.options.profile
302     conf.env['BUILD_WITH_32_64'] = Options.options.mixed
303     conf.env['BUILD_CLASSIC'] = Options.options.classic
304     conf.env['BUILD_DEBUG'] = Options.options.debug
306     if conf.env['BUILD_JACKDBUS']:
307         conf.env['BUILD_JACKD'] = conf.env['BUILD_CLASSIC']
308     else:
309         conf.env['BUILD_JACKD'] = True
311     conf.env['BINDIR'] = conf.env['PREFIX'] + '/bin'
313     if Options.options.htmldir:
314         conf.env['HTMLDIR'] = Options.options.htmldir
315     else:
316         # set to None here so that the doxygen code can find out the highest
317         # directory to remove upon install
318         conf.env['HTMLDIR'] = None
320     if Options.options.libdir:
321         conf.env['LIBDIR'] = Options.options.libdir
322     else:
323         conf.env['LIBDIR'] = conf.env['PREFIX'] + '/lib'
325     if Options.options.mandir:
326         conf.env['MANDIR'] = Options.options.mandir
327     else:
328         conf.env['MANDIR'] = conf.env['PREFIX'] + '/share/man/man1'
330     if conf.env['BUILD_DEBUG']:
331         conf.env.append_unique('CXXFLAGS', '-g')
332         conf.env.append_unique('CFLAGS', '-g')
333         conf.env.append_unique('LINKFLAGS', '-g')
335     if not Options.options.autostart in ['default', 'classic', 'dbus', 'none']:
336         conf.fatal('Invalid autostart value "' + Options.options.autostart + '"')
338     if Options.options.autostart == 'default':
339         if conf.env['BUILD_JACKD']:
340             conf.env['AUTOSTART_METHOD'] = 'classic'
341         else:
342             conf.env['AUTOSTART_METHOD'] = 'dbus'
343     else:
344         conf.env['AUTOSTART_METHOD'] = Options.options.autostart
346     if conf.env['AUTOSTART_METHOD'] == 'dbus' and not conf.env['BUILD_JACKDBUS']:
347         conf.fatal('D-Bus autostart mode was specified but jackdbus will not be built')
348     if conf.env['AUTOSTART_METHOD'] == 'classic' and not conf.env['BUILD_JACKD']:
349         conf.fatal('Classic autostart mode was specified but jackd will not be built')
351     if conf.env['AUTOSTART_METHOD'] == 'dbus':
352         conf.define('USE_LIBDBUS_AUTOLAUNCH', 1)
353     elif conf.env['AUTOSTART_METHOD'] == 'classic':
354         conf.define('USE_CLASSIC_AUTOLAUNCH', 1)
356     conf.define('CLIENT_NUM', Options.options.clients)
357     conf.define('PORT_NUM_FOR_CLIENT', Options.options.application_ports)
359     if conf.env['IS_WINDOWS']:
360         # we define this in the environment to maintain compatibility with
361         # existing install paths that use ADDON_DIR rather than have to
362         # have special cases for windows each time.
363         conf.env['ADDON_DIR'] = conf.env['BINDIR'] + '/jack'
364         # don't define ADDON_DIR in config.h, use the default 'jack' defined in
365         # windows/JackPlatformPlug_os.h
366     else:
367         conf.env['ADDON_DIR'] = os.path.normpath(os.path.join(conf.env['LIBDIR'], 'jack'))
368         conf.define('ADDON_DIR', conf.env['ADDON_DIR'])
369         conf.define('JACK_LOCATION', os.path.normpath(os.path.join(conf.env['PREFIX'], 'bin')))
371     if not conf.env['IS_WINDOWS']:
372         conf.define('USE_POSIX_SHM', 1)
373     conf.define('JACKMP', 1)
374     if conf.env['BUILD_JACKDBUS']:
375         conf.define('JACK_DBUS', 1)
376     if conf.env['BUILD_WITH_PROFILE']:
377         conf.define('JACK_MONITOR', 1)
378     conf.write_config_header('config.h', remove=False)
380     svnrev = None
381     try:
382         f = open('svnversion.h')
383         data = f.read()
384         m = re.match(r'^#define SVN_VERSION "([^"]*)"$', data)
385         if m != None:
386             svnrev = m.group(1)
387         f.close()
388     except IOError:
389         pass
391     if Options.options.mixed:
392         conf.setenv(lib32, env=conf.env.derive())
393         conf.env.append_unique('CXXFLAGS', '-m32')
394         conf.env.append_unique('CFLAGS', '-m32')
395         conf.env.append_unique('LINKFLAGS', '-m32')
396         if Options.options.libdir32:
397             conf.env['LIBDIR'] = Options.options.libdir32
398         else:
399             conf.env['LIBDIR'] = conf.env['PREFIX'] + '/lib32'
400         # libdb does not work in mixed mode
401         conf.all_envs[lib32]['HAVE_DB'] = 0
402         conf.all_envs[lib32]['HAVE_DB_H'] = 0
403         conf.all_envs[lib32]['LIB_DB'] = []
404         # no need for opus in 32bit mixed mode clients
405         conf.all_envs[lib32]['LIB_OPUS'] = []
406         # someone tell me where this file gets written please..
407         conf.write_config_header('config.h')
409     print()
410     print('==================')
411     version_msg = 'JACK ' + VERSION
412     if svnrev:
413         version_msg += ' exported from r' + svnrev
414     else:
415         version_msg += ' svn revision will checked and eventually updated during build'
416     print(version_msg)
418     conf.msg('Maximum JACK clients', Options.options.clients, color='NORMAL')
419     conf.msg('Maximum ports per application', Options.options.application_ports, color='NORMAL')
421     conf.msg('Install prefix', conf.env['PREFIX'], color='CYAN')
422     conf.msg('Library directory', conf.all_envs['']['LIBDIR'], color='CYAN')
423     if conf.env['BUILD_WITH_32_64']:
424         conf.msg('32-bit library directory', conf.all_envs[lib32]['LIBDIR'], color='CYAN')
425     conf.msg('Drivers directory', conf.env['ADDON_DIR'], color='CYAN')
426     display_feature(conf, 'Build debuggable binaries', conf.env['BUILD_DEBUG'])
428     tool_flags = [
429         ('C compiler flags',   ['CFLAGS', 'CPPFLAGS']),
430         ('C++ compiler flags', ['CXXFLAGS', 'CPPFLAGS']),
431         ('Linker flags',       ['LINKFLAGS', 'LDFLAGS'])
432     ]
433     for name,vars in tool_flags:
434         flags = []
435         for var in vars:
436             flags += conf.all_envs[''][var]
437         conf.msg(name, repr(flags), color='NORMAL')
439     if conf.env['BUILD_WITH_32_64']:
440         conf.msg('32-bit C compiler flags', repr(conf.all_envs[lib32]['CFLAGS']))
441         conf.msg('32-bit C++ compiler flags', repr(conf.all_envs[lib32]['CXXFLAGS']))
442         conf.msg('32-bit linker flags', repr(conf.all_envs[lib32]['LINKFLAGS']))
443     display_feature(conf, 'Build with engine profiling', conf.env['BUILD_WITH_PROFILE'])
444     display_feature(conf, 'Build with 32/64 bits mixed mode', conf.env['BUILD_WITH_32_64'])
446     display_feature(conf, 'Build standard JACK (jackd)', conf.env['BUILD_JACKD'])
447     display_feature(conf, 'Build D-Bus JACK (jackdbus)', conf.env['BUILD_JACKDBUS'])
448     conf.msg('Autostart method', conf.env['AUTOSTART_METHOD'])
450     if conf.env['BUILD_JACKDBUS'] and conf.env['BUILD_JACKD']:
451         print(Logs.colors.RED + 'WARNING !! mixing both jackd and jackdbus may cause issues:' + Logs.colors.NORMAL)
452         print(Logs.colors.RED + 'WARNING !! jackdbus does not use .jackdrc nor qjackctl settings' + Logs.colors.NORMAL)
454     conf.summarize_auto_options()
456     if conf.env['BUILD_JACKDBUS']:
457         conf.msg('D-Bus service install directory', conf.env['DBUS_SERVICES_DIR'], color='CYAN')
459         if conf.env['DBUS_SERVICES_DIR'] != conf.env['DBUS_SERVICES_DIR_REAL']:
460             print()
461             print(Logs.colors.RED + 'WARNING: D-Bus session services directory as reported by pkg-config is')
462             print(Logs.colors.RED + 'WARNING:', end=' ')
463             print(Logs.colors.CYAN + conf.env['DBUS_SERVICES_DIR_REAL'])
464             print(Logs.colors.RED + 'WARNING: but service file will be installed in')
465             print(Logs.colors.RED + 'WARNING:', end=' ')
466             print(Logs.colors.CYAN + conf.env['DBUS_SERVICES_DIR'])
467             print(Logs.colors.RED + 'WARNING: You may need to adjust your D-Bus configuration after installing jackdbus')
468             print('WARNING: You can override dbus service install directory')
469             print('WARNING: with --enable-pkg-config-dbus-service-dir option to this script')
470             print(Logs.colors.NORMAL, end=' ')
471     print()
473 def init(ctx):
474     for y in (BuildContext, CleanContext, InstallContext, UninstallContext):
475         name = y.__name__.replace('Context','').lower()
476         class tmp(y):
477             cmd = name + '_' + lib32
478             variant = lib32
480 def obj_add_includes(bld, obj):
481     if bld.env['BUILD_JACKDBUS']:
482         obj.includes += ['dbus']
484     if bld.env['IS_LINUX']:
485         obj.includes += ['linux', 'posix']
487     if bld.env['IS_MACOSX']:
488         obj.includes += ['macosx', 'posix']
490     if bld.env['IS_SUN']:
491         obj.includes += ['posix', 'solaris']
493     if bld.env['IS_WINDOWS']:
494         obj.includes += ['windows']
496 # FIXME: Is SERVER_SIDE needed?
497 def build_jackd(bld):
498     jackd = bld(
499         features = ['cxx', 'cxxprogram'],
500         defines = ['HAVE_CONFIG_H','SERVER_SIDE'],
501         includes = ['.', 'common', 'common/jack'],
502         target = 'jackd',
503         source = ['common/Jackdmp.cpp'],
504         use = ['serverlib', 'SYSTEMD']
505     )
507     if bld.env['BUILD_JACKDBUS']:
508         jackd.source += ['dbus/audio_reserve.c', 'dbus/reserve.c']
509         jackd.use += ['DBUS-1']
511     if bld.env['IS_LINUX']:
512         jackd.use += ['DL', 'M', 'PTHREAD', 'RT', 'STDC++']
514     if bld.env['IS_MACOSX']:
515         jackd.use += ['DL', 'PTHREAD']
516         jackd.framework = ['CoreFoundation']
518     if bld.env['IS_SUN']:
519         jackd.use += ['DL', 'PTHREAD']
521     obj_add_includes(bld, jackd)
523     return jackd
525 # FIXME: Is SERVER_SIDE needed?
526 def create_driver_obj(bld, **kw):
527     if bld.env['IS_MACOSX'] or bld.env['IS_WINDOWS']:
528         # On MacOSX this is necessary.
529         # I do not know if this is necessary on Windows.
530         # Note added on 2015-12-13 by karllinden.
531         if 'use' in kw:
532             kw['use'] += ['serverlib']
533         else:
534             kw['use'] = ['serverlib']
536     driver = bld(
537         features = ['c', 'cxx', 'cshlib', 'cxxshlib'],
538         defines = ['HAVE_CONFIG_H', 'SERVER_SIDE'],
539         includes = ['.', 'common', 'common/jack'],
540         install_path = '${ADDON_DIR}/',
541         **kw)
543     if bld.env['IS_WINDOWS']:
544         driver.env['cxxshlib_PATTERN'] = 'jack_%s.dll'
545     else:
546         driver.env['cxxshlib_PATTERN'] = 'jack_%s.so'
548     obj_add_includes(bld, driver)
550     return driver
552 def build_drivers(bld):
553     # Non-hardware driver sources. Lexically sorted.
554     dummy_src = [
555         'common/JackDummyDriver.cpp'
556     ]
558     loopback_src = [
559         'common/JackLoopbackDriver.cpp'
560     ]
562     net_src = [
563         'common/JackNetDriver.cpp'
564     ]
566     netone_src = [
567         'common/JackNetOneDriver.cpp',
568         'common/netjack.c',
569         'common/netjack_packet.c'
570     ]
572     proxy_src = [
573         'common/JackProxyDriver.cpp'
574     ]
576     # Hardware driver sources. Lexically sorted.
577     alsa_src = [
578         'common/memops.c',
579         'linux/alsa/JackAlsaDriver.cpp',
580         'linux/alsa/alsa_rawmidi.c',
581         'linux/alsa/alsa_seqmidi.c',
582         'linux/alsa/alsa_midi_jackmp.cpp',
583         'linux/alsa/generic_hw.c',
584         'linux/alsa/hdsp.c',
585         'linux/alsa/alsa_driver.c',
586         'linux/alsa/hammerfall.c',
587         'linux/alsa/ice1712.c'
588     ]
590     alsarawmidi_src = [
591         'linux/alsarawmidi/JackALSARawMidiDriver.cpp',
592         'linux/alsarawmidi/JackALSARawMidiInputPort.cpp',
593         'linux/alsarawmidi/JackALSARawMidiOutputPort.cpp',
594         'linux/alsarawmidi/JackALSARawMidiPort.cpp',
595         'linux/alsarawmidi/JackALSARawMidiReceiveQueue.cpp',
596         'linux/alsarawmidi/JackALSARawMidiSendQueue.cpp',
597         'linux/alsarawmidi/JackALSARawMidiUtil.cpp'
598     ]
600     boomer_src = [
601         'common/memops.c',
602         'solaris/oss/JackBoomerDriver.cpp'
603     ]
605     coreaudio_src = [
606         'macosx/coreaudio/JackCoreAudioDriver.mm',
607         'common/JackAC3Encoder.cpp'
608     ]
610     coremidi_src = [
611         'macosx/coremidi/JackCoreMidiInputPort.mm',
612         'macosx/coremidi/JackCoreMidiOutputPort.mm',
613         'macosx/coremidi/JackCoreMidiPhysicalInputPort.mm',
614         'macosx/coremidi/JackCoreMidiPhysicalOutputPort.mm',
615         'macosx/coremidi/JackCoreMidiVirtualInputPort.mm',
616         'macosx/coremidi/JackCoreMidiVirtualOutputPort.mm',
617         'macosx/coremidi/JackCoreMidiPort.mm',
618         'macosx/coremidi/JackCoreMidiUtil.mm',
619         'macosx/coremidi/JackCoreMidiDriver.mm'
620     ]
622     ffado_src = [
623         'linux/firewire/JackFFADODriver.cpp',
624         'linux/firewire/JackFFADOMidiInputPort.cpp',
625         'linux/firewire/JackFFADOMidiOutputPort.cpp',
626         'linux/firewire/JackFFADOMidiReceiveQueue.cpp',
627         'linux/firewire/JackFFADOMidiSendQueue.cpp'
628     ]
630     iio_driver_src = [
631         'linux/iio/JackIIODriver.cpp'
632     ]
634     oss_src = [
635         'common/memops.c',
636         'solaris/oss/JackOSSDriver.cpp'
637     ]
639     portaudio_src = [
640         'windows/portaudio/JackPortAudioDevices.cpp',
641         'windows/portaudio/JackPortAudioDriver.cpp',
642     ]
644     winmme_src = [
645         'windows/winmme/JackWinMMEDriver.cpp',
646         'windows/winmme/JackWinMMEInputPort.cpp',
647         'windows/winmme/JackWinMMEOutputPort.cpp',
648         'windows/winmme/JackWinMMEPort.cpp',
649     ]
651     # Create non-hardware driver objects. Lexically sorted.
652     create_driver_obj(
653         bld,
654         target = 'dummy',
655         source = dummy_src)
657     create_driver_obj(
658         bld,
659         target = 'loopback',
660         source = loopback_src)
662     create_driver_obj(
663         bld,
664         target = 'net',
665         source = net_src)
667     create_driver_obj(
668         bld,
669         target = 'netone',
670         source = netone_src,
671         use = ['SAMPLERATE', 'CELT'])
673     create_driver_obj(
674         bld,
675         target = 'proxy',
676         source = proxy_src)
678     # Create hardware driver objects. Lexically sorted after the conditional,
679     # e.g. BUILD_DRIVER_ALSA.
680     if bld.env['BUILD_DRIVER_ALSA']:
681         create_driver_obj(
682             bld,
683             target = 'alsa',
684             source = alsa_src,
685             use = ['ALSA'])
686         create_driver_obj(
687             bld,
688             target = 'alsarawmidi',
689             source = alsarawmidi_src,
690             use = ['ALSA'])
692     if bld.env['BUILD_DRIVER_FFADO']:
693         create_driver_obj(
694             bld,
695             target = 'firewire',
696             source = ffado_src,
697             use = ['LIBFFADO'])
699     if bld.env['BUILD_DRIVER_IIO']:
700         create_driver_obj(
701             bld,
702             target = 'iio',
703             source = iio_src,
704             use = ['GTKIOSTREAM', 'EIGEN3'])
706     if bld.env['BUILD_DRIVER_PORTAUDIO']:
707         create_driver_obj(
708             bld,
709             target = 'portaudio',
710             source = portaudio_src,
711             use = ['PORTAUDIO'])
713     if bld.env['BUILD_DRIVER_WINMME']:
714         create_driver_obj(
715             bld,
716             target = 'winmme',
717             source = winmme_src,
718             use = ['WINMME'])
720     if bld.env['IS_MACOSX']:
721         create_driver_obj(
722             bld,
723             target = 'coreaudio',
724             source = coreaudio_src,
725             use = ['AFTEN'],
726             framework = ['AudioUnit', 'CoreAudio', 'CoreServices'])
728         create_driver_obj(
729             bld,
730             target = 'coremidi',
731             source = coremidi_src,
732             use = ['serverlib'], # FIXME: Is this needed?
733             framework = ['AudioUnit', 'CoreMIDI', 'CoreServices', 'Foundation'])
735     if bld.env['IS_SUN']:
736         create_driver_obj(
737             bld,
738             target = 'boomer',
739             source = boomer_src)
740         create_driver_obj(
741             bld,
742             target = 'oss',
743             source = oss_src)
745 def build(bld):
746     if not bld.variant and bld.env['BUILD_WITH_32_64']:
747         Options.commands.append(bld.cmd + '_' + lib32)
749     # process subfolders from here
750     bld.recurse('common')
752     if bld.variant:
753         # only the wscript in common/ knows how to handle variants
754         return
756     bld.recurse('compat')
758     if not os.access('svnversion.h', os.R_OK):
759         def post_run(self):
760             sg = Utils.h_file(self.outputs[0].abspath(self.env))
761             #print sg.encode('hex')
762             Build.bld.node_sigs[self.env.variant()][self.outputs[0].id] = sg
764         script = bld.path.find_resource('svnversion_regenerate.sh')
765         script = script.abspath()
767         bld(
768                 rule = '%s ${TGT}' % script,
769                 name = 'svnversion',
770                 runnable_status = Task.RUN_ME,
771                 before = 'c cxx',
772                 color = 'BLUE',
773                 post_run = post_run,
774                 source = ['svnversion_regenerate.sh'],
775                 target = [bld.path.find_or_declare('svnversion.h')]
776         )
778     if bld.env['BUILD_JACKD']:
779         build_jackd(bld)
781     build_drivers(bld)
783     bld.recurse('example-clients')
784     if bld.env['IS_LINUX']:
785         bld.recurse('man')
786         bld.recurse('systemd')
787     if not bld.env['IS_WINDOWS']:
788         bld.recurse('tests')
789     if bld.env['BUILD_JACKDBUS']:
790         bld.recurse('dbus')
792     if bld.env['BUILD_DOXYGEN_DOCS']:
793         html_build_dir = bld.path.find_or_declare('html').abspath()
795         bld(
796             features = 'subst',
797             source = 'doxyfile.in',
798             target = 'doxyfile',
799             HTML_BUILD_DIR = html_build_dir,
800             SRCDIR = bld.srcnode.abspath(),
801             VERSION = VERSION
802         )
804         # There are two reasons for logging to doxygen.log and using it as
805         # target in the build rule (rather than html_build_dir):
806         # (1) reduce the noise when running the build
807         # (2) waf has a regular file to check for a timestamp. If the directory
808         #     is used instead waf will rebuild the doxygen target (even upon
809         #     install).
810         def doxygen(task):
811             doxyfile = task.inputs[0].abspath()
812             logfile = task.outputs[0].abspath()
813             cmd = '%s %s &> %s' % (task.env['DOXYGEN'][0], doxyfile, logfile)
814             return task.exec_command(cmd)
816         bld(
817             rule = doxygen,
818             source = 'doxyfile',
819             target = 'doxygen.log'
820         )
822         # Determine where to install HTML documentation. Since share_dir is the
823         # highest directory the uninstall routine should remove, there is no
824         # better candidate for share_dir, but the requested HTML directory if
825         # --htmldir is given.
826         if bld.env['HTMLDIR']:
827             html_install_dir = bld.options.destdir + bld.env['HTMLDIR']
828             share_dir = html_install_dir
829         else:
830             share_dir = bld.options.destdir + bld.env['PREFIX'] + '/share/jack-audio-connection-kit'
831             html_install_dir = share_dir + '/reference/html/'
833         if bld.cmd == 'install':
834             if os.path.isdir(html_install_dir):
835                 Logs.pprint('CYAN', 'Removing old doxygen documentation installation...')
836                 shutil.rmtree(html_install_dir)
837                 Logs.pprint('CYAN', 'Removing old doxygen documentation installation done.')
838             Logs.pprint('CYAN', 'Installing doxygen documentation...')
839             shutil.copytree(html_build_dir, html_install_dir)
840             Logs.pprint('CYAN', 'Installing doxygen documentation done.')
841         elif bld.cmd =='uninstall':
842             Logs.pprint('CYAN', 'Uninstalling doxygen documentation...')
843             if os.path.isdir(share_dir):
844                 shutil.rmtree(share_dir)
845             Logs.pprint('CYAN', 'Uninstalling doxygen documentation done.')
846         elif bld.cmd =='clean':
847             if os.access(html_build_dir, os.R_OK):
848                 Logs.pprint('CYAN', 'Removing doxygen generated documentation...')
849                 shutil.rmtree(html_build_dir)
850                 Logs.pprint('CYAN', 'Removing doxygen generated documentation done.')
852 def dist(ctx):
853     # This code blindly assumes it is working in the toplevel source directory.
854     if not os.path.exists('svnversion.h'):
855         os.system('./svnversion_regenerate.sh svnversion.h')
857 from waflib import TaskGen
858 @TaskGen.extension('.mm')
859 def mm_hook(self, node):
860     """Alias .mm files to be compiled the same as .cpp files, gcc will do the right thing."""
861     return self.create_compiled_task('cxx', node)