|
- #! /usr/bin/env python
- # encoding: utf-8
- from __future__ import print_function
-
- import os
- import subprocess
- g_maxlen = 40
- import shutil
- import re
- import sys
-
- from waflib import Logs, Options, Task, Utils
- from waflib.Build import BuildContext, CleanContext, InstallContext, UninstallContext
-
- VERSION='1.9.11'
- APPNAME='jack'
- JACK_API_VERSION = '0.1.0'
-
- # these variables are mandatory ('/' are converted automatically)
- top = '.'
- out = 'build'
-
- # lib32 variant name used when building in mixed mode
- lib32 = 'lib32'
-
- auto_options = []
-
- def display_msg(msg, status = None, color = None):
- sr = msg
- global g_maxlen
- g_maxlen = max(g_maxlen, len(msg))
- if status:
- Logs.pprint('NORMAL', "%s :" % msg.ljust(g_maxlen), sep=' ')
- Logs.pprint(color, status)
- else:
- print("%s" % msg.ljust(g_maxlen))
-
- def display_feature(msg, build):
- if build:
- display_msg(msg, "yes", 'GREEN')
- else:
- display_msg(msg, "no", 'YELLOW')
-
- # This function prints an error without stopping waf. The reason waf should not
- # be stopped is to be able to list all missing dependencies in one chunk.
- def print_error(msg):
- print(Logs.colors.RED + msg + Logs.colors.NORMAL)
-
- class AutoOption:
- """
- This class is the foundation for the auto options. It adds an option
- --foo=no|yes to the list of options and deals with all logic and checks for
- these options.
-
- Each option can have different dependencies that will be checked. If all
- dependencies are available and the user has not done any request the option
- will be enabled. If the user has requested to enable the option the class
- ensures that all dependencies are available and prints an error message
- otherwise. If the user disables the option, i.e. --foo=no, no checks are
- made.
-
- For each option it is possible to add packages that are required for the
- option using the add_package function. For dependency programs add_program
- should be used. For libraries (without pkg-config support) the add_library
- function should be used. For headers the add_header function exists. If
- there is another type of requirement or dependency the check hook (an
- external function called when configuring) can be used.
-
- When all checks have been made and the class has made a decision the result
- is saved in conf.env['NAME'] where 'NAME' by default is the uppercase of the
- name argument to __init__, but it can be changed with the conf_dest argument
- to __init__.
-
- The class will define a preprocessor symbol with the result. The default
- name is HAVE_NAME, but it can be changed using the define argument to
- __init__.
- """
-
- def __init__(self, opt, name, help, conf_dest=None, define=None):
- # check hook to call upon configuration
- self.check_hook = None
- self.check_hook_error = None
- self.check_hook_found = True
-
- # required libraries
- self.libs = [] # elements on the form [lib,uselib_store]
- self.libs_not_found = [] # elements on the form lib
-
- # required headers
- self.headers = []
- self.headers_not_found = []
-
- # required packages (checked with pkg-config)
- self.packages = [] # elements on the form [package,uselib_store,atleast_version]
- self.packages_not_found = [] # elements on the form [package,atleast_version]
-
- # required programs
- self.programs = [] # elements on the form [program,var]
- self.programs_not_found = [] # elements on the form program
-
- # the result of the configuration (should the option be enabled or not?)
- self.result = False
-
- self.help = help
- self.option = '--' + name
- self.dest = 'auto_option_' + name
- if conf_dest:
- self.conf_dest = conf_dest
- else:
- self.conf_dest = name.upper()
- if not define:
- self.define = 'HAVE_' + name.upper()
- else:
- self.define = define
- opt.add_option(self.option, type='string', default='auto', dest=self.dest, help=self.help+' (enabled by default if possible)', metavar='no|yes')
-
- def add_library(self, library, uselib_store=None):
- """
- Add a required library that should be checked during configuration. The
- library will be checked using the conf.check_cc function. If the
- uselib_store arugment is not given it defaults to LIBRARY (the uppercase
- of the library argument). The uselib_store argument will be passed to
- check_cc which means LIB_LIBRARY, CFLAGS_LIBRARY and DEFINES_LIBRARY,
- etc. will be defined if the option is enabled.
- """
- if not uselib_store:
- uselib_store = library.upper().replace('-', '_')
- self.libs.append([library, uselib_store])
-
- def add_header(self, header):
- """
- Add a required header that should be checked during configuration. The
- header will be checked using the conf.check_cc function which means
- HAVE_HEADER_H will be defined if found.
- """
- self.headers.append(header)
-
- def add_package(self, package, uselib_store=None, atleast_version=None):
- """
- Add a required package that should be checked using pkg-config during
- configuration. The package will be checked using the conf.check_cfg
- function and the uselib_store and atleast_version will be passed to
- check_cfg. If uselib_store is None it defaults to PACKAGE (uppercase of
- the package argument) with hyphens and dots replaced with underscores.
- If atleast_version is None it defaults to '0'.
- """
- if not uselib_store:
- uselib_store = package.upper().replace('-', '_').replace('.', '_')
- if not atleast_version:
- atleast_version = '0'
- self.packages.append([package, uselib_store, atleast_version])
-
- def add_program(self, program, var=None):
- """
- Add a required program that should be checked during configuration. If
- var is not given it defaults to PROGRAM (the uppercase of the program
- argument). If the option is enabled the program is saved as a list (?!)
- in conf.env['PROGRAM'].
- """
- if not var:
- var = program.upper().replace('-', '_')
- self.programs.append([program, var])
-
- def set_check_hook(self, check_hook, check_hook_error):
- """
- Set the check hook and the corresponding error printing function to the
- configure step. The check_hook argument is a function that should return
- True if the extra prerequisites were found and False if not. The
- check_hook_error argument is an error printing function that should
- print an error message telling the user that --foo was explicitly
- requested but cannot be built since the extra prerequisites were not
- found. Both function should take a single argument that is the waf
- configuration context.
- """
- self.check_hook = check_hook
- self.check_hook_error = check_hook_error
-
- def _check(self, conf):
- """
- This is an internal function that runs all necessary configure checks.
- It checks all dependencies (even if some dependency was not found) so
- that the user can install all missing dependencies in one go, instead
- of playing the infamous hit-configure-hit-configure game.
-
- This function returns True if all dependencies were found and False if
- not.
- """
- all_found = True
-
- # Use-variables that should be used when checking libraries, headers and
- # programs. The list will be populated when looking for packages.
- use = []
-
- # check for packages
- for package,uselib_store,atleast_version in self.packages:
- try:
- conf.check_cfg(package=package, uselib_store=uselib_store, atleast_version=atleast_version, args='--cflags --libs')
- use.append(uselib_store)
- except conf.errors.ConfigurationError:
- all_found = False
- self.packages_not_found.append([package,atleast_version])
-
- # check for libraries
- for lib,uselib_store in self.libs:
- try:
- conf.check_cc(lib=lib, uselib_store=uselib_store, use=use)
- except conf.errors.ConfigurationError:
- all_found = False
- self.libs_not_found.append(lib)
-
- # check for headers
- for header in self.headers:
- try:
- conf.check_cc(header_name=header, use=use)
- except conf.errors.ConfigurationError:
- all_found = False
- self.headers_not_found.append(header)
-
- # check for programs
- for program,var in self.programs:
- try:
- conf.find_program(program, var=var, use=use)
- except conf.errors.ConfigurationError:
- all_found = False
- self.programs_not_found.append(program)
-
- # call hook (if specified)
- if self.check_hook:
- self.check_hook_found = self.check_hook(conf)
- if not self.check_hook_found:
- all_found = False
-
- return all_found
-
- def _configure_error(self, conf):
- """
- This is an internal function that prints errors for each missing
- dependency. The error messages tell the user that this option required
- some dependency, but it cannot be found.
- """
-
- for lib in self.libs_not_found:
- print_error('%s requires the %s library, but it cannot be found.' % (self.option, lib))
-
- for header in self.headers_not_found:
- print_error('%s requires the %s header, but it cannot be found.' % (self.option, header))
-
- for package,atleast_version in self.packages_not_found:
- string = package
- if atleast_version:
- string += ' >= ' + atleast_version
- print_error('%s requires the package %s, but it cannot be found.' % (self.option, string))
-
- for program in self.programs_not_found:
- print_error('%s requires the %s program, but it cannot be found.' % (self.option, program))
-
- if not self.check_hook_found:
- self.check_hook_error(conf)
-
- def configure(self, conf):
- """
- This function configures the option examining the argument given too
- --foo (where foo is this option). This function sets self.result to the
- result of the configuration; True if the option should be enabled or
- False if not. If not all dependencies were found self.result will shall
- be False. conf.env['NAME'] will be set to the same value aswell as a
- preprocessor symbol will be defined according to the result.
-
- If --foo[=yes] was given, but some dependency was not found an error
- message is printed (foreach missing dependency).
-
- This function returns True on success and False on error.
- """
- argument = getattr(Options.options, self.dest)
- if argument == 'no':
- self.result = False
- retvalue = True
- elif argument == 'yes':
- if self._check(conf):
- self.result = True
- retvalue = True
- else:
- self.result = False
- retvalue = False
- self._configure_error(conf)
- elif argument == 'auto':
- self.result = self._check(conf)
- retvalue = True
- else:
- print_error('Invalid argument "' + argument + '" to ' + self.option)
- self.result = False
- retvalue = False
-
- conf.env[self.conf_dest] = self.result
- if self.result:
- conf.define(self.define, 1)
- else:
- conf.define(self.define, 0)
- return retvalue
-
- def display_message(self):
- """
- This function displays a result message with the help text and the
- result of the configuration.
- """
- display_feature(self.help, self.result)
-
- # This function adds an option to the list of auto options and returns the newly
- # created option.
- def add_auto_option(opt, name, help, conf_dest=None, define=None):
- option = AutoOption(opt, name, help, conf_dest=conf_dest, define=define)
- auto_options.append(option)
- return option
-
- # This function applies a hack that for each auto option --foo=no|yes replaces
- # any occurence --foo in argv with --foo=yes, in effect interpreting --foo as
- # --foo=yes. The function has to be called before waf issues the option parser,
- # i.e. before the configure phase.
- def auto_options_argv_hack():
- for option in auto_options:
- for x in range(1, len(sys.argv)):
- if sys.argv[x] == option.option:
- sys.argv[x] += '=yes'
-
- # This function configures all auto options. It stops waf and prints an error
- # message if there were unsatisfied requirements.
- def configure_auto_options(conf):
- ok = True
- for option in auto_options:
- if not option.configure(conf):
- ok = False
- if not ok:
- conf.fatal('There were unsatisfied requirements.')
-
- # This function displays all options and the configuration results.
- def display_auto_options_messages():
- for option in auto_options:
- option.display_message()
-
- def check_for_celt(conf):
- found = False
- for version in ['11', '8', '7', '5']:
- define = 'HAVE_CELT_API_0_' + version
- if not found:
- try:
- conf.check_cfg(package='celt', atleast_version='0.' + version + '.0', args='--cflags --libs')
- found = True
- conf.define(define, 1)
- continue
- except conf.errors.ConfigurationError:
- pass
- conf.define(define, 0)
- return found
-
- def check_for_celt_error(conf):
- print_error('--celt requires the package celt, but it could not be found.')
-
- # The readline/readline.h header does not work if stdio.h is not included
- # before. Thus a fragment with both stdio.h and readline/readline.h need to be
- # test-compiled to find out whether readline is available.
- def check_for_readline(conf):
- try:
- conf.check_cc(fragment='''
- #include <stdio.h>
- #include <readline/readline.h>
- int main(void) { return 0; }''',
- execute=False,
- msg='Checking for header readline/readline.h')
- return True
- except conf.errors.ConfigurationError:
- return False
-
- def check_for_readline_error(conf):
- print_error('--readline requires the readline/readline.h header, but it cannot be found.')
-
- def check_for_mmsystem(conf):
- try:
- conf.check_cc(fragment='''
- #include <windows.h>
- #include <mmsystem.h>
- int main(void) { return 0; }''',
- execute=False,
- msg='Checking for header mmsystem.h')
- return True
- except conf.errors.ConfigurationError:
- return False
-
- def check_for_mmsystem_error(conf):
- print_error('--winmme requires the mmsystem.h header, but it cannot be found.')
-
- def options(opt):
- # options provided by the modules
- opt.load('compiler_cxx')
- opt.load('compiler_c')
-
- opt.load('xcode')
- opt.load('xcode6')
-
- # install directories
- opt.add_option('--htmldir', type='string', default=None, help="HTML documentation directory [Default: <prefix>/share/jack-audio-connection-kit/reference/html/")
- opt.add_option('--libdir', type='string', help="Library directory [Default: <prefix>/lib]")
- opt.add_option('--libdir32', type='string', help="32bit Library directory [Default: <prefix>/lib32]")
- opt.add_option('--mandir', type='string', help="Manpage directory [Default: <prefix>/share/man/man1]")
-
- # options affecting binaries
- opt.add_option('--dist-target', type='string', default='auto', help='Specify the target for cross-compiling [auto,mingw]')
- opt.add_option('--mixed', action='store_true', default=False, help='Build with 32/64 bits mixed mode')
- opt.add_option('--debug', action='store_true', default=False, dest='debug', help='Build debuggable binaries')
-
- # options affecting general jack functionality
- opt.add_option('--classic', action='store_true', default=False, help='Force enable standard JACK (jackd) even if D-Bus JACK (jackdbus) is enabled too')
- opt.add_option('--dbus', action='store_true', default=False, help='Enable D-Bus JACK (jackdbus)')
- opt.add_option('--autostart', type='string', default="default", help='Autostart method. Possible values: "default", "classic", "dbus", "none"')
- opt.add_option('--profile', action='store_true', default=False, help='Build with engine profiling')
- opt.add_option('--clients', default=64, type="int", dest="clients", help='Maximum number of JACK clients')
- opt.add_option('--ports-per-application', default=768, type="int", dest="application_ports", help='Maximum number of ports per application')
-
- # options with third party dependencies
- doxygen = add_auto_option(opt, 'doxygen', help='Build doxygen documentation', conf_dest='BUILD_DOXYGEN_DOCS')
- doxygen.add_program('doxygen')
- alsa = add_auto_option(opt, 'alsa', help='Enable ALSA driver', conf_dest='BUILD_DRIVER_ALSA')
- alsa.add_package('alsa', atleast_version='1.0.18')
- firewire = add_auto_option(opt, 'firewire', help='Enable FireWire driver (FFADO)', conf_dest='BUILD_DRIVER_FFADO')
- firewire.add_package('libffado', atleast_version='1.999.17')
- freebob = add_auto_option(opt, 'freebob', help='Enable FreeBob driver')
- freebob.add_package('libfreebob', atleast_version='1.0.0')
- iio = add_auto_option(opt, 'iio', help='Enable IIO driver', conf_dest='BUILD_DRIVER_IIO')
- iio.add_package('gtkIOStream', atleast_version='1.4.0')
- iio.add_package('eigen3', atleast_version='3.1.2')
- portaudio = add_auto_option(opt, 'portaudio', help='Enable Portaudio driver', conf_dest='BUILD_DRIVER_PORTAUDIO')
- portaudio.add_header('windows.h') # only build portaudio on windows
- portaudio.add_package('portaudio-2.0', uselib_store='PORTAUDIO', atleast_version='19')
- winmme = add_auto_option(opt, 'winmme', help='Enable WinMME driver', conf_dest='BUILD_DRIVER_WINMME')
- winmme.set_check_hook(check_for_mmsystem, check_for_mmsystem_error)
-
- celt = add_auto_option(opt, 'celt', help='Build with CELT')
- celt.set_check_hook(check_for_celt, check_for_celt_error)
- opus = add_auto_option(opt, 'opus', help='Build Opus netjack2')
- opus.add_header('opus/opus_custom.h')
- opus.add_package('opus', atleast_version='0.9.0')
- samplerate = add_auto_option(opt, 'samplerate', help='Build with libsamplerate')
- samplerate.add_package('samplerate')
- sndfile = add_auto_option(opt, 'sndfile', help='Build with libsndfile')
- sndfile.add_package('sndfile')
- readline = add_auto_option(opt, 'readline', help='Build with readline')
- readline.add_library('readline')
- readline.set_check_hook(check_for_readline, check_for_readline_error)
-
- # dbus options
- opt.recurse('dbus')
-
- # this must be called before the configure phase
- auto_options_argv_hack()
-
- def configure(conf):
- conf.load('compiler_cxx')
- conf.load('compiler_c')
- if Options.options.dist_target == 'auto':
- platform = sys.platform
- conf.env['IS_MACOSX'] = platform == 'darwin'
- conf.env['IS_LINUX'] = platform == 'linux' or platform == 'linux2' or platform == 'linux3' or platform == 'posix'
- conf.env['IS_SUN'] = platform == 'sunos'
- # GNU/kFreeBSD and GNU/Hurd are treated as Linux
- if platform.startswith('gnu0') or platform.startswith('gnukfreebsd'):
- conf.env['IS_LINUX'] = True
- elif Options.options.dist_target == 'mingw':
- conf.env['IS_WINDOWS'] = True
-
- if conf.env['IS_LINUX']:
- Logs.pprint('CYAN', "Linux detected")
-
- if conf.env['IS_MACOSX']:
- Logs.pprint('CYAN', "MacOS X detected")
-
- if conf.env['IS_SUN']:
- Logs.pprint('CYAN', "SunOS detected")
-
- if conf.env['IS_WINDOWS']:
- Logs.pprint('CYAN', "Windows detected")
-
- if conf.env['IS_WINDOWS']:
- conf.env.append_unique('CCDEFINES', '_POSIX')
- conf.env.append_unique('CXXDEFINES', '_POSIX')
-
- conf.env.append_unique('CXXFLAGS', '-Wall')
- conf.env.append_unique('CFLAGS', '-Wall')
-
- # configure all auto options
- configure_auto_options(conf)
-
- # Check for functions.
- conf.check_cc(
- function_name='ppoll',
- header_name=['poll.h', 'signal.h'],
- defines=['_GNU_SOURCE'],
- mandatory=False)
-
- conf.recurse('common')
- if Options.options.dbus:
- conf.recurse('dbus')
- if conf.env['BUILD_JACKDBUS'] != True:
- conf.fatal('jackdbus was explicitly requested but cannot be built')
-
- conf.recurse('example-clients')
-
- conf.env['LIB_PTHREAD'] = ['pthread']
- conf.env['LIB_DL'] = ['dl']
- conf.env['LIB_RT'] = ['rt']
- conf.env['LIB_M'] = ['m']
- conf.env['LIB_STDC++'] = ['stdc++']
- conf.env['JACK_API_VERSION'] = JACK_API_VERSION
- conf.env['JACK_VERSION'] = VERSION
-
- conf.env['BUILD_WITH_PROFILE'] = Options.options.profile
- conf.env['BUILD_WITH_32_64'] = Options.options.mixed
- conf.env['BUILD_CLASSIC'] = Options.options.classic
- conf.env['BUILD_DEBUG'] = Options.options.debug
-
- if conf.env['BUILD_JACKDBUS']:
- conf.env['BUILD_JACKD'] = conf.env['BUILD_CLASSIC']
- else:
- conf.env['BUILD_JACKD'] = True
-
- conf.env['BINDIR'] = conf.env['PREFIX'] + '/bin'
-
- if Options.options.htmldir:
- conf.env['HTMLDIR'] = Options.options.htmldir
- else:
- # set to None here so that the doxygen code can find out the highest
- # directory to remove upon install
- conf.env['HTMLDIR'] = None
-
- if Options.options.libdir:
- conf.env['LIBDIR'] = Options.options.libdir
- else:
- conf.env['LIBDIR'] = conf.env['PREFIX'] + '/lib'
-
- if Options.options.mandir:
- conf.env['MANDIR'] = Options.options.mandir
- else:
- conf.env['MANDIR'] = conf.env['PREFIX'] + '/share/man/man1'
-
- if conf.env['BUILD_DEBUG']:
- conf.env.append_unique('CXXFLAGS', '-g')
- conf.env.append_unique('CFLAGS', '-g')
- conf.env.append_unique('LINKFLAGS', '-g')
-
- if not Options.options.autostart in ["default", "classic", "dbus", "none"]:
- conf.fatal("Invalid autostart value \"" + Options.options.autostart + "\"")
-
- if Options.options.autostart == "default":
- if conf.env['BUILD_JACKDBUS'] == True and conf.env['BUILD_JACKD'] == False:
- conf.env['AUTOSTART_METHOD'] = "dbus"
- else:
- conf.env['AUTOSTART_METHOD'] = "classic"
- else:
- conf.env['AUTOSTART_METHOD'] = Options.options.autostart
-
- if conf.env['AUTOSTART_METHOD'] == "dbus" and not conf.env['BUILD_JACKDBUS']:
- conf.fatal("D-Bus autostart mode was specified but jackdbus will not be built")
- if conf.env['AUTOSTART_METHOD'] == "classic" and not conf.env['BUILD_JACKD']:
- conf.fatal("Classic autostart mode was specified but jackd will not be built")
-
- if conf.env['AUTOSTART_METHOD'] == "dbus":
- conf.define('USE_LIBDBUS_AUTOLAUNCH', 1)
- elif conf.env['AUTOSTART_METHOD'] == "classic":
- conf.define('USE_CLASSIC_AUTOLAUNCH', 1)
-
- conf.define('CLIENT_NUM', Options.options.clients)
- conf.define('PORT_NUM_FOR_CLIENT', Options.options.application_ports)
-
- if conf.env['IS_WINDOWS']:
- # we define this in the environment to maintain compatability with
- # existing install paths that use ADDON_DIR rather than have to
- # have special cases for windows each time.
- conf.env['ADDON_DIR'] = conf.env['BINDIR'] + '/jack'
- # don't define ADDON_DIR in config.h, use the default 'jack' defined in
- # windows/JackPlatformPlug_os.h
- else:
- conf.env['ADDON_DIR'] = os.path.normpath(os.path.join(conf.env['LIBDIR'], 'jack'))
- conf.define('ADDON_DIR', conf.env['ADDON_DIR'])
- conf.define('JACK_LOCATION', os.path.normpath(os.path.join(conf.env['PREFIX'], 'bin')))
-
- if not conf.env['IS_WINDOWS']:
- conf.define('USE_POSIX_SHM', 1)
- conf.define('JACKMP', 1)
- if conf.env['BUILD_JACKDBUS'] == True:
- conf.define('JACK_DBUS', 1)
- if conf.env['BUILD_WITH_PROFILE'] == True:
- conf.define('JACK_MONITOR', 1)
- conf.write_config_header('config.h', remove=False)
-
- svnrev = None
- try:
- f = open('svnversion.h')
- data = f.read()
- m = re.match(r'^#define SVN_VERSION "([^"]*)"$', data)
- if m != None:
- svnrev = m.group(1)
- f.close()
- except IOError:
- pass
-
- if Options.options.mixed == True:
- conf.setenv(lib32, env=conf.env.derive())
- conf.env.append_unique('CXXFLAGS', '-m32')
- conf.env.append_unique('CFLAGS', '-m32')
- conf.env.append_unique('LINKFLAGS', '-m32')
- if Options.options.libdir32:
- conf.env['LIBDIR'] = Options.options.libdir32
- else:
- conf.env['LIBDIR'] = conf.env['PREFIX'] + '/lib32'
- conf.write_config_header('config.h')
-
- print()
- display_msg("==================")
- version_msg = "JACK " + VERSION
- if svnrev:
- version_msg += " exported from r" + svnrev
- else:
- version_msg += " svn revision will checked and eventually updated during build"
- print(version_msg)
-
- print("Build with a maximum of %d JACK clients" % Options.options.clients)
- print("Build with a maximum of %d ports per application" % Options.options.application_ports)
-
- display_msg("Install prefix", conf.env['PREFIX'], 'CYAN')
- display_msg("Library directory", conf.all_envs[""]['LIBDIR'], 'CYAN')
- if conf.env['BUILD_WITH_32_64'] == True:
- display_msg("32-bit library directory", conf.all_envs[lib32]['LIBDIR'], 'CYAN')
- display_msg("Drivers directory", conf.env['ADDON_DIR'], 'CYAN')
- display_feature('Build debuggable binaries', conf.env['BUILD_DEBUG'])
- display_msg('C compiler flags', repr(conf.all_envs[""]['CFLAGS']))
- display_msg('C++ compiler flags', repr(conf.all_envs[""]['CXXFLAGS']))
- display_msg('Linker flags', repr(conf.all_envs[""]['LINKFLAGS']))
- if conf.env['BUILD_WITH_32_64'] == True:
- display_msg('32-bit C compiler flags', repr(conf.all_envs[lib32]['CFLAGS']))
- display_msg('32-bit C++ compiler flags', repr(conf.all_envs[lib32]['CXXFLAGS']))
- display_msg('32-bit linker flags', repr(conf.all_envs[lib32]['LINKFLAGS']))
- display_feature('Build with engine profiling', conf.env['BUILD_WITH_PROFILE'])
- display_feature('Build with 32/64 bits mixed mode', conf.env['BUILD_WITH_32_64'])
-
- display_feature('Build standard JACK (jackd)', conf.env['BUILD_JACKD'])
- display_feature('Build D-Bus JACK (jackdbus)', conf.env['BUILD_JACKDBUS'])
- display_msg('Autostart method', conf.env['AUTOSTART_METHOD'])
-
- if conf.env['BUILD_JACKDBUS'] and conf.env['BUILD_JACKD']:
- print(Logs.colors.RED + 'WARNING !! mixing both jackd and jackdbus may cause issues:' + Logs.colors.NORMAL)
- print(Logs.colors.RED + 'WARNING !! jackdbus does not use .jackdrc nor qjackctl settings' + Logs.colors.NORMAL)
-
- # display configuration result messages for auto options
- display_auto_options_messages()
-
- if conf.env['BUILD_JACKDBUS'] == True:
- display_msg('D-Bus service install directory', conf.env['DBUS_SERVICES_DIR'], 'CYAN')
- #display_msg('Settings persistence', xxx)
-
- if conf.env['DBUS_SERVICES_DIR'] != conf.env['DBUS_SERVICES_DIR_REAL']:
- print()
- print(Logs.colors.RED + "WARNING: D-Bus session services directory as reported by pkg-config is")
- print(Logs.colors.RED + "WARNING:", end=' ')
- print(Logs.colors.CYAN + conf.env['DBUS_SERVICES_DIR_REAL'])
- print(Logs.colors.RED + 'WARNING: but service file will be installed in')
- print(Logs.colors.RED + "WARNING:", end=' ')
- print(Logs.colors.CYAN + conf.env['DBUS_SERVICES_DIR'])
- print(Logs.colors.RED + 'WARNING: You may need to adjust your D-Bus configuration after installing jackdbus')
- print('WARNING: You can override dbus service install directory')
- print('WARNING: with --enable-pkg-config-dbus-service-dir option to this script')
- print(Logs.colors.NORMAL, end=' ')
- print()
-
- def init(ctx):
- for y in (BuildContext, CleanContext, InstallContext, UninstallContext):
- name = y.__name__.replace('Context','').lower()
- class tmp(y):
- cmd = name + '_' + lib32
- variant = lib32
-
- def obj_add_includes(bld, obj):
- if bld.env['BUILD_JACKDBUS']:
- obj.includes += ['dbus']
-
- if bld.env['IS_LINUX']:
- obj.includes += ['linux', 'posix']
-
- if bld.env['IS_MACOSX']:
- obj.includes += ['macosx', 'posix']
-
- if bld.env['IS_SUN']:
- obj.includes += ['posix', 'solaris']
-
- if bld.env['IS_WINDOWS']:
- obj.includes += ['windows']
-
- # FIXME: Is SERVER_SIDE needed?
- def build_jackd(bld):
- jackd = bld(
- features = ['cxx', 'cxxprogram'],
- defines = ['HAVE_CONFIG_H','SERVER_SIDE'],
- includes = ['.', 'common', 'common/jack'],
- target = 'jackd',
- source = ['common/Jackdmp.cpp'],
- use = ['serverlib'])
-
- if bld.env['BUILD_JACKDBUS']:
- jackd.source += ['dbus/audio_reserve.c', 'dbus/reserve.c']
- jackd.use += ['DBUS-1']
-
- if bld.env['IS_LINUX']:
- jackd.use += ['DL', 'M', 'PTHREAD', 'RT', 'STDC++']
-
- if bld.env['IS_MACOSX']:
- bld.framework = ['CoreFoundation']
- jackd.use += ['DL', 'PTHREAD']
-
- if bld.env['IS_SUN']:
- jackd.use += ['DL', 'PTHREAD']
-
- obj_add_includes(bld, jackd)
-
- return jackd
-
- # FIXME: Is SERVER_SIDE needed?
- def create_driver_obj(bld, **kw):
- if bld.env['IS_MACOSX'] or bld.env['IS_WINDOWS']:
- # On MacOSX this is necessary.
- # I do not know if this is necessary on Windows.
- # Note added on 2015-12-13 by lilrc.
- if 'use' in kw:
- kw['use'] += ['serverlib']
- else:
- kw['use'] = ['serverlib']
-
- driver = bld(
- features = ['c', 'cshlib', 'cxx', 'cxxshlib'],
- defines = ['HAVE_CONFIG_H', 'SERVER_SIDE'],
- includes = ['.', 'common', 'common/jack'],
- install_path = '${ADDON_DIR}/',
- **kw)
-
- if bld.env['IS_WINDOWS']:
- driver.env['cxxshlib_PATTERN'] = 'jack_%s.dll'
- else:
- driver.env['cxxshlib_PATTERN'] = 'jack_%s.so'
-
- obj_add_includes(bld, driver)
-
- return driver
-
- def build_drivers(bld):
- # Non-hardware driver sources. Lexically sorted.
- dummy_src = [
- 'common/JackDummyDriver.cpp'
- ]
-
- loopback_src = [
- 'common/JackLoopbackDriver.cpp'
- ]
-
- net_src = [
- 'common/JackNetDriver.cpp'
- ]
-
- netone_src = [
- 'common/JackNetOneDriver.cpp',
- 'common/netjack.c',
- 'common/netjack_packet.c'
- ]
-
- proxy_src = [
- 'common/JackProxyDriver.cpp'
- ]
-
- # Hardware driver sources. Lexically sorted.
- alsa_src = [
- 'common/memops.c',
- 'linux/alsa/JackAlsaDriver.cpp',
- 'linux/alsa/alsa_rawmidi.c',
- 'linux/alsa/alsa_seqmidi.c',
- 'linux/alsa/alsa_midi_jackmp.cpp',
- 'linux/alsa/generic_hw.c',
- 'linux/alsa/hdsp.c',
- 'linux/alsa/alsa_driver.c',
- 'linux/alsa/hammerfall.c',
- 'linux/alsa/ice1712.c'
- ]
-
- alsarawmidi_src = [
- 'linux/alsarawmidi/JackALSARawMidiDriver.cpp',
- 'linux/alsarawmidi/JackALSARawMidiInputPort.cpp',
- 'linux/alsarawmidi/JackALSARawMidiOutputPort.cpp',
- 'linux/alsarawmidi/JackALSARawMidiPort.cpp',
- 'linux/alsarawmidi/JackALSARawMidiReceiveQueue.cpp',
- 'linux/alsarawmidi/JackALSARawMidiSendQueue.cpp',
- 'linux/alsarawmidi/JackALSARawMidiUtil.cpp'
- ]
-
- boomer_src = [
- 'common/memops.c',
- 'solaris/oss/JackBoomerDriver.cpp'
- ]
-
- coreaudio_src = [
- 'macosx/coreaudio/JackCoreAudioDriver.cpp'
- ]
-
- coremidi_src = [
- 'macosx/coremidi/JackCoreMidiInputPort.cpp',
- 'macosx/coremidi/JackCoreMidiOutputPort.cpp',
- 'macosx/coremidi/JackCoreMidiPhysicalInputPort.cpp',
- 'macosx/coremidi/JackCoreMidiPhysicalOutputPort.cpp',
- 'macosx/coremidi/JackCoreMidiVirtualInputPort.cpp',
- 'macosx/coremidi/JackCoreMidiVirtualOutputPort.cpp',
- 'macosx/coremidi/JackCoreMidiPort.cpp',
- 'macosx/coremidi/JackCoreMidiUtil.cpp',
- 'macosx/coremidi/JackCoreMidiDriver.cpp'
- ]
-
- ffado_src = [
- 'linux/firewire/JackFFADODriver.cpp',
- 'linux/firewire/JackFFADOMidiInputPort.cpp',
- 'linux/firewire/JackFFADOMidiOutputPort.cpp',
- 'linux/firewire/JackFFADOMidiReceiveQueue.cpp',
- 'linux/firewire/JackFFADOMidiSendQueue.cpp'
- ]
-
- freebob_src = [
- 'linux/freebob/JackFreebobDriver.cpp'
- ]
-
- iio_driver_src = [
- 'linux/iio/JackIIODriver.cpp'
- ]
-
- oss_src = [
- 'common/memops.c',
- 'solaris/oss/JackOSSDriver.cpp'
- ]
-
- portaudio_src = [
- 'windows/portaudio/JackPortAudioDevices.cpp',
- 'windows/portaudio/JackPortAudioDriver.cpp',
- ]
-
- winmme_driver_src = [
- 'windows/winmme/JackWinMMEDriver.cpp',
- 'windows/winmme/JackWinMMEInputPort.cpp',
- 'windows/winmme/JackWinMMEOutputPort.cpp',
- 'windows/winmme/JackWinMMEPort.cpp',
- ]
-
- # Create non-hardware driver objects. Lexically sorted.
- create_driver_obj(
- bld,
- target = 'dummy',
- source = dummy_src)
-
- create_driver_obj(
- bld,
- target = 'loopback',
- source = loopback_src)
-
- create_driver_obj(
- bld,
- target = 'net',
- source = net_src)
-
- create_driver_obj(
- bld,
- target = 'netone',
- source = netone_src,
- use = ['SAMPLERATE', 'CELT'])
-
- create_driver_obj(
- bld,
- target = 'proxy',
- source = proxy_src)
-
- # Create hardware driver objects. Lexically sorted after the conditional,
- # e.g. BUILD_DRIVER_ALSA.
- if bld.env['BUILD_DRIVER_ALSA']:
- create_driver_obj(
- bld,
- target = 'alsa',
- source = alsa_src,
- use = ['ALSA'])
- create_driver_obj(
- bld,
- target = 'alsarawmidi',
- source = alsarawmidi_src,
- use = ['ALSA'])
-
- if bld.env['BUILD_DRIVER_FREEBOB']:
- create_driver_obj(
- bld,
- target = 'freebob',
- source = freebob_src,
- use = ['LIBFREEBOB'])
-
- if bld.env['BUILD_DRIVER_FFADO']:
- create_driver_obj(
- bld,
- target = 'firewire',
- source = ffado_src,
- use = ['LIBFFADO'])
-
- if bld.env['BUILD_DRIVER_IIO']:
- create_driver_obj(
- bld,
- target = 'iio',
- source = iio_src,
- use = ['GTKIOSTREAM', 'EIGEN3'])
-
- if bld.env['BUILD_DRIVER_PORTAUDIO']:
- create_driver_obj(
- bld,
- target = 'portaudio',
- source = portaudio_src,
- use = ['PORTAUDIO'])
-
- if bld.env['BUILD_DRIVER_WINMME']:
- create_driver_obj(
- bld,
- target = 'winmme',
- source = winmme_src,
- use = ['WINMME'])
-
- if bld.env['IS_MACOSX']:
- create_driver_obj(
- bld,
- target = 'coreaudio',
- source = coreaudio_src,
- framework = ['AudioUnit', 'CoreAudio', 'CoreServices'])
-
- create_driver_obj(
- bld,
- target = 'coremidi',
- source = coremidi_src,
- framework = ['AudioUnit', 'CoreMIDI', 'CoreServices'])
-
- if bld.env['IS_SUN']:
- create_driver_obj(
- bld,
- target = 'boomer',
- source = boomer_src)
- create_driver_obj(
- bld,
- target = 'oss',
- source = oss_src)
-
- def build(bld):
- if not bld.variant:
- out2 = out
- else:
- out2 = out + "/" + bld.variant
- print("make[1]: Entering directory `" + os.getcwd() + "/" + out2 + "'")
-
- if not bld.variant:
- if bld.env['BUILD_WITH_32_64'] == True:
- Options.commands.append(bld.cmd + '_' + lib32)
-
- # process subfolders from here
- bld.recurse('common')
-
- if bld.variant:
- # only the wscript in common/ knows how to handle variants
- return
-
- if not os.access('svnversion.h', os.R_OK):
- def post_run(self):
- sg = Utils.h_file(self.outputs[0].abspath(self.env))
- #print sg.encode('hex')
- Build.bld.node_sigs[self.env.variant()][self.outputs[0].id] = sg
-
- script = bld.path.find_resource('svnversion_regenerate.sh')
- script = script.abspath()
-
- bld(
- rule = '%s ${TGT}' % script,
- name = 'svnversion',
- runnable_status = Task.RUN_ME,
- before = 'c cxx',
- color = 'BLUE',
- post_run = post_run,
- source = ['svnversion_regenerate.sh'],
- target = [bld.path.find_or_declare('svnversion.h')]
- )
-
- if bld.env['BUILD_JACKD']:
- build_jackd(bld)
-
- build_drivers(bld)
-
- if bld.env['IS_LINUX']:
- bld.recurse('example-clients')
- bld.recurse('tests')
- bld.recurse('man')
- if bld.env['BUILD_JACKDBUS'] == True:
- bld.recurse('dbus')
-
- if bld.env['IS_MACOSX']:
- bld.recurse('example-clients')
- bld.recurse('tests')
- if bld.env['BUILD_JACKDBUS'] == True:
- bld.recurse('dbus')
-
- if bld.env['IS_SUN']:
- bld.recurse('example-clients')
- bld.recurse('tests')
- if bld.env['BUILD_JACKDBUS'] == True:
- bld.recurse('dbus')
-
- if bld.env['IS_WINDOWS']:
- bld.recurse('example-clients')
- #bld.recurse('tests')
-
- if bld.env['BUILD_DOXYGEN_DOCS'] == True:
- html_build_dir = bld.path.find_or_declare('html').abspath()
-
- bld(
- features = 'subst',
- source = 'doxyfile.in',
- target = 'doxyfile',
- HTML_BUILD_DIR = html_build_dir,
- SRCDIR = bld.srcnode.abspath(),
- VERSION = VERSION
- )
-
- # There are two reasons for logging to doxygen.log and using it as
- # target in the build rule (rather than html_build_dir):
- # (1) reduce the noise when running the build
- # (2) waf has a regular file to check for a timestamp. If the directory
- # is used instead waf will rebuild the doxygen target (even upon
- # install).
- def doxygen(task):
- doxyfile = task.inputs[0].abspath()
- logfile = task.outputs[0].abspath()
- cmd = '%s %s &> %s' % (task.env['DOXYGEN'][0], doxyfile, logfile)
- return task.exec_command(cmd)
-
- bld(
- rule = doxygen,
- source = 'doxyfile',
- target = 'doxygen.log'
- )
-
- # Determine where to install HTML documentation. Since share_dir is the
- # highest directory the uninstall routine should remove, there is no
- # better candidate for share_dir, but the requested HTML directory if
- # --htmldir is given.
- if bld.env['HTMLDIR']:
- html_install_dir = bld.options.destdir + bld.env['HTMLDIR']
- share_dir = html_install_dir
- else:
- share_dir = bld.options.destdir + bld.env['PREFIX'] + '/share/jack-audio-connection-kit'
- html_install_dir = share_dir + '/reference/html/'
-
- if bld.cmd == 'install':
- if os.path.isdir(html_install_dir):
- Logs.pprint('CYAN', "Removing old doxygen documentation installation...")
- shutil.rmtree(html_install_dir)
- Logs.pprint('CYAN', "Removing old doxygen documentation installation done.")
- Logs.pprint('CYAN', "Installing doxygen documentation...")
- shutil.copytree(html_build_dir, html_install_dir)
- Logs.pprint('CYAN', "Installing doxygen documentation done.")
- elif bld.cmd =='uninstall':
- Logs.pprint('CYAN', "Uninstalling doxygen documentation...")
- if os.path.isdir(share_dir):
- shutil.rmtree(share_dir)
- Logs.pprint('CYAN', "Uninstalling doxygen documentation done.")
- elif bld.cmd =='clean':
- if os.access(html_build_dir, os.R_OK):
- Logs.pprint('CYAN', "Removing doxygen generated documentation...")
- shutil.rmtree(html_build_dir)
- Logs.pprint('CYAN', "Removing doxygen generated documentation done.")
-
- def dist(ctx):
- # This code blindly assumes it is working in the toplevel source directory.
- if not os.path.exists('svnversion.h'):
- os.system('./svnversion_regenerate.sh svnversion.h')
|