17 EnsureSConsVersion(0, 96)
19 ardour_version = '2.1pre'
24 # Command-line options
27 opts = Options('scache.conf')
29 ('ARCH', 'Set architecture-specific compilation flags by hand (all flags as 1 argument)',''),
30 BoolOption('AUDIOUNITS', 'Compile with Apple\'s AudioUnit library. (experimental)', 0),
31 BoolOption('COREAUDIO', 'Compile with Apple\'s CoreAudio library', 0),
32 BoolOption('DEBUG', 'Set to build with debugging information and no optimizations', 0),
33 PathOption('DESTDIR', 'Set the intermediate install "prefix"', '/'),
34 EnumOption('DIST_TARGET', 'Build target for cross compiling packagers', 'auto', allowed_values=('auto', 'i386', 'i686', 'x86_64', 'powerpc', 'tiger', 'panther', 'none' ), ignorecase=2),
35 BoolOption('DMALLOC', 'Compile and link using the dmalloc library', 0),
36 BoolOption('EXTRA_WARN', 'Compile with -Wextra, -ansi, and -pedantic. Might break compilation. For pedants', 0),
37 BoolOption('FFT_ANALYSIS', 'Include FFT analysis window', 0),
38 BoolOption('FPU_OPTIMIZATION', 'Build runtime checked assembler code', 1),
39 BoolOption('LIBLO', 'Compile with support for liblo library', 1),
40 BoolOption('NLS', 'Set to turn on i18n support', 1),
41 PathOption('PREFIX', 'Set the install "prefix"', '/usr/local'),
42 BoolOption('SURFACES', 'Build support for control surfaces', 1),
43 BoolOption('SYSLIBS', 'USE AT YOUR OWN RISK: CANCELS ALL SUPPORT FROM ARDOUR AUTHORS: Use existing system versions of various libraries instead of internal ones', 0),
44 BoolOption('UNIVERSAL', 'Compile as universal binary. Requires that external libraries are already universal.', 0),
45 BoolOption('VERSIONED', 'Add revision information to ardour/gtk executable name inside the build directory', 0),
46 BoolOption('VST', 'Compile with support for VST', 0),
47 BoolOption('GPROFILE', 'Compile with support for gprofile (Developers only)', 0),
48 BoolOption('TRANZPORT', 'Compile with support for Frontier Designs (if libusb is available)', 1)
51 #----------------------------------------------------------------------
52 # a handy helper that provides a way to merge compile/link information
53 # from multiple different "environments"
54 #----------------------------------------------------------------------
56 class LibraryInfo(Environment):
57 def __init__(self,*args,**kw):
58 Environment.__init__ (self,*args,**kw)
60 def Merge (self,others):
62 self.Append (LIBS = other.get ('LIBS',[]))
63 self.Append (LIBPATH = other.get ('LIBPATH', []))
64 self.Append (CPPPATH = other.get('CPPPATH', []))
65 self.Append (LINKFLAGS = other.get('LINKFLAGS', []))
66 self.Replace(LIBPATH = list(Set(self.get('LIBPATH', []))))
67 self.Replace(CPPPATH = list(Set(self.get('CPPPATH',[]))))
68 #doing LINKFLAGS breaks -framework
69 #doing LIBS break link order dependency
71 def ENV_update(self, src_ENV):
72 for k in src_ENV.keys():
73 if k in self['ENV'].keys() and k in [ 'PATH', 'LD_LIBRARY_PATH',
75 self['ENV'][k]=SCons.Util.AppendPath(self['ENV'][k], src_ENV[k])
77 self['ENV'][k]=src_ENV[k]
79 env = LibraryInfo (options = opts,
81 VERSION = ardour_version,
82 TARBALL='ardour-' + ardour_version + '.tar.bz2',
84 DISTTREE = '#ardour-' + ardour_version,
85 DISTCHECKDIR = '#ardour-' + ardour_version + '/check'
88 env.ENV_update(os.environ)
90 #----------------------------------------------------------------------
92 #----------------------------------------------------------------------
94 # Handy subst-in-file builder
97 def do_subst_in_file(targetfile, sourcefile, dict):
98 """Replace all instances of the keys of dict with their values.
99 For example, if dict is {'%VERSION%': '1.2345', '%BASE%': 'MyProg'},
100 then all instances of %VERSION% in the file will be replaced with 1.2345 etc.
103 f = open(sourcefile, 'rb')
107 raise SCons.Errors.UserError, "Can't read source file %s"%sourcefile
108 for (k,v) in dict.items():
109 contents = re.sub(k, v, contents)
111 f = open(targetfile, 'wb')
115 raise SCons.Errors.UserError, "Can't write target file %s"%targetfile
118 def subst_in_file(target, source, env):
119 if not env.has_key('SUBST_DICT'):
120 raise SCons.Errors.UserError, "SubstInFile requires SUBST_DICT to be set."
121 d = dict(env['SUBST_DICT']) # copy it
122 for (k,v) in d.items():
124 d[k] = env.subst(v())
125 elif SCons.Util.is_String(v):
128 raise SCons.Errors.UserError, "SubstInFile: key %s: %s must be a string or callable"%(k, repr(v))
129 for (t,s) in zip(target, source):
130 return do_subst_in_file(str(t), str(s), d)
132 def subst_in_file_string(target, source, env):
133 """This is what gets printed on the console."""
134 return '\n'.join(['Substituting vars from %s into %s'%(str(s), str(t))
135 for (t,s) in zip(target, source)])
137 def subst_emitter(target, source, env):
138 """Add dependency from substituted SUBST_DICT to target.
139 Returns original target, source tuple unchanged.
141 d = env['SUBST_DICT'].copy() # copy it
142 for (k,v) in d.items():
144 d[k] = env.subst(v())
145 elif SCons.Util.is_String(v):
147 Depends(target, SCons.Node.Python.Value(d))
148 # Depends(target, source) # this doesn't help the install-sapphire-linux.sh problem
149 return target, source
151 subst_action = Action (subst_in_file, subst_in_file_string)
152 env['BUILDERS']['SubstInFile'] = Builder(action=subst_action, emitter=subst_emitter)
155 # internationalization
158 # po_builder: builder function to copy po files to the parent directory while updating them
160 # first source: .po file
161 # second source: .pot file
164 def po_builder(target,source,env):
165 os.spawnvp (os.P_WAIT, 'cp', ['cp', str(source[0]), str(target[0])])
171 print 'Updating ' + str(target[0])
172 return os.spawnvp (os.P_WAIT, 'msgmerge', args)
174 po_bld = Builder (action = po_builder)
175 env.Append(BUILDERS = {'PoBuild' : po_bld})
177 # mo_builder: builder function for (binary) message catalogs (.mo)
179 # first source: .po file
182 def mo_builder(target,source,env):
186 target[0].get_path(),
189 return os.spawnvp (os.P_WAIT, 'msgfmt', args)
191 mo_bld = Builder (action = mo_builder)
192 env.Append(BUILDERS = {'MoBuild' : mo_bld})
194 # pot_builder: builder function for message templates (.pot)
196 # source: list of C/C++ etc. files to extract messages from
199 def pot_builder(target,source,env):
204 '-o', target[0].get_path(),
205 "--default-domain=" + env['PACKAGE'],
206 '--copyright-holder="Paul Davis"' ]
207 args += [ src.get_path() for src in source ]
209 return os.spawnvp (os.P_WAIT, 'xgettext', args)
211 pot_bld = Builder (action = pot_builder)
212 env.Append(BUILDERS = {'PotBuild' : pot_bld})
215 # utility function, not a builder
218 def i18n (buildenv, sources, installenv):
219 domain = buildenv['PACKAGE']
220 potfile = buildenv['POTFILE']
222 installenv.Alias ('potupdate', buildenv.PotBuild (potfile, sources))
224 p_oze = [ os.path.basename (po) for po in glob.glob ('po/*.po') ]
225 languages = [ po.replace ('.po', '') for po in p_oze ]
227 for po_file in p_oze:
228 buildenv.PoBuild(po_file, ['po/'+po_file, potfile])
229 mo_file = po_file.replace (".po", ".mo")
230 installenv.Alias ('install', buildenv.MoBuild (mo_file, po_file))
232 for lang in languages:
233 modir = (os.path.join (install_prefix, 'share/locale/' + lang + '/LC_MESSAGES/'))
234 moname = domain + '.mo'
235 installenv.Alias('install', installenv.InstallAs (os.path.join (modir, moname), lang + '.mo'))
238 def fetch_svn_revision (path):
242 cmd += " | awk '/^Revision:/ { print $2}'"
243 return commands.getoutput (cmd)
245 def create_stored_revision (target = None, source = None, env = None):
246 if os.path.exists('.svn'):
247 rev = fetch_svn_revision ('.');
249 text = "#ifndef __ardour_svn_revision_h__\n"
250 text += "#define __ardour_svn_revision_h__\n"
251 text += "static const char* ardour_svn_revision = \"" + rev + "\";\n";
253 print '============> writing svn revision info to svn_revision.h\n'
254 o = file ('svn_revision.h', 'w')
258 print "Could not open svn_revision.h for writing\n"
261 print "You cannot use \"scons revision\" on without using a checked out"
262 print "copy of the Ardour source code repository"
266 # A generic builder for version.cc files
268 # note: requires that DOMAIN, MAJOR, MINOR, MICRO are set in the construction environment
269 # note: assumes one source files, the header that declares the version variables
272 def version_builder (target, source, env):
274 text = "int " + env['DOMAIN'] + "_major_version = " + str (env['MAJOR']) + ";\n"
275 text += "int " + env['DOMAIN'] + "_minor_version = " + str (env['MINOR']) + ";\n"
276 text += "int " + env['DOMAIN'] + "_micro_version = " + str (env['MICRO']) + ";\n"
279 o = file (target[0].get_path(), 'w')
283 print "Could not open", target[0].get_path(), " for writing\n"
286 text = "#ifndef __" + env['DOMAIN'] + "_version_h__\n"
287 text += "#define __" + env['DOMAIN'] + "_version_h__\n"
288 text += "extern const char* " + env['DOMAIN'] + "_revision;\n"
289 text += "extern int " + env['DOMAIN'] + "_major_version;\n"
290 text += "extern int " + env['DOMAIN'] + "_minor_version;\n"
291 text += "extern int " + env['DOMAIN'] + "_micro_version;\n"
292 text += "#endif /* __" + env['DOMAIN'] + "_version_h__ */\n"
295 o = file (target[1].get_path(), 'w')
299 print "Could not open", target[1].get_path(), " for writing\n"
304 version_bld = Builder (action = version_builder)
305 env.Append (BUILDERS = {'VersionBuild' : version_bld})
308 # a builder that makes a hard link from the 'source' executable to a name with
309 # a "build ID" based on the most recent CVS activity that might be reasonably
310 # related to version activity. this relies on the idea that the SConscript
311 # file that builds the executable is updated with new version info and committed
312 # to the source code repository whenever things change.
315 def versioned_builder(target,source,env):
316 w, r = os.popen2( "LANG= svn info | awk '/^Revision:/ { print $2}'")
318 last_revision = r.readline().strip()
321 if last_revision == "":
322 print "No SVN info found - versioned executable cannot be built"
325 print "The current build ID is " + last_revision
327 tagged_executable = source[0].get_path() + '-' + last_revision
329 if os.path.exists (tagged_executable):
330 print "Replacing existing executable with the same build tag."
331 os.unlink (tagged_executable)
333 return os.link (source[0].get_path(), tagged_executable)
335 verbuild = Builder (action = versioned_builder)
336 env.Append (BUILDERS = {'VersionedExecutable' : verbuild})
339 # source tar file builder
342 def distcopy (target, source, env):
343 treedir = str (target[0])
347 except OSError, (errnum, strerror):
348 if errnum != errno.EEXIST:
349 print 'mkdir ', treedir, ':', strerror
353 # we don't know what characters might be in the file names
354 # so quote them all before passing them to the shell
356 all_files = ([ str(s) for s in source ])
357 cmd += " ".join ([ "'%s'" % quoted for quoted in all_files])
358 cmd += ' | (cd ' + treedir + ' && tar xf -)'
362 def tarballer (target, source, env):
363 cmd = 'tar -jcf ' + str (target[0]) + ' ' + str(source[0]) + " --exclude '*~'"
364 print 'running ', cmd, ' ... '
368 dist_bld = Builder (action = distcopy,
369 target_factory = SCons.Node.FS.default_fs.Entry,
370 source_factory = SCons.Node.FS.default_fs.Entry,
373 tarball_bld = Builder (action = tarballer,
374 target_factory = SCons.Node.FS.default_fs.Entry,
375 source_factory = SCons.Node.FS.default_fs.Entry)
377 env.Append (BUILDERS = {'Distribute' : dist_bld})
378 env.Append (BUILDERS = {'Tarball' : tarball_bld})
381 # Make sure they know what they are doing
385 if os.path.isfile('.personal_use_only'):
386 print "Enabling VST support. Note that distributing a VST-enabled ardour\nis a violation of several different licences.\nBuild with VST=false if you intend to distribute ardour to others."
388 sys.stdout.write ("Are you building Ardour for personal use (rather than distribution to others)? [no]: ")
389 answer = sys.stdin.readline ()
390 answer = answer.rstrip().strip()
391 if answer == "yes" or answer == "y":
392 fh = open('.personal_use_only', 'w')
394 print "OK, VST support will be enabled"
396 print 'You cannot build Ardour with VST support for distribution to others.\nIt is a violation of several different licenses. Build with VST=false.'
399 if os.path.isfile('.personal_use_only'):
400 os.remove('.personal_use_only')
403 #######################
404 # Dependency Checking #
405 #######################
409 'glib-2.0' : '2.10.1',
410 'gthread-2.0' : '2.10.1',
411 'gtk+-2.0' : '2.10.0',
412 'libxml-2.0' : '2.6.0',
413 'samplerate' : '0.1.0',
417 'libgnomecanvas-2.0' : '2.0',
421 def DependenciesRequiredMessage():
422 print 'You do not have the necessary dependencies required to build ardour'
423 print 'Please consult http://ardour.org/building for more information'
425 def CheckPKGConfig(context, version):
426 context.Message( 'Checking for pkg-config version >= %s... ' %version )
427 ret = context.TryAction('pkg-config --atleast-pkgconfig-version=%s' % version)[0]
428 context.Result( ret )
431 def CheckPKGVersion(context, name, version):
432 context.Message( 'Checking for %s... ' % name )
433 ret = context.TryAction('pkg-config --atleast-version=%s %s' %(version,name) )[0]
434 context.Result( ret )
437 conf = Configure(env, custom_tests = { 'CheckPKGConfig' : CheckPKGConfig,
438 'CheckPKGVersion' : CheckPKGVersion })
440 # I think a more recent version is needed on win32
441 min_pkg_config_version = '0.8.0'
443 if not conf.CheckPKGConfig(min_pkg_config_version):
444 print 'pkg-config >= %s not found.' % min_pkg_config_version
447 for pkg, version in deps.iteritems():
448 if not conf.CheckPKGVersion( pkg, version ):
449 print '%s >= %s not found.' %(pkg, version)
450 DependenciesRequiredMessage()
455 # ----------------------------------------------------------------------
456 # Construction environment setup
457 # ----------------------------------------------------------------------
461 libraries['core'] = LibraryInfo (CCFLAGS = '-Ilibs')
463 #libraries['sndfile'] = LibraryInfo()
464 #libraries['sndfile'].ParseConfig('pkg-config --cflags --libs sndfile')
466 libraries['lrdf'] = LibraryInfo()
467 libraries['lrdf'].ParseConfig('pkg-config --cflags --libs lrdf')
469 libraries['raptor'] = LibraryInfo()
470 libraries['raptor'].ParseConfig('pkg-config --cflags --libs raptor')
472 libraries['samplerate'] = LibraryInfo()
473 libraries['samplerate'].ParseConfig('pkg-config --cflags --libs samplerate')
475 if env['FFT_ANALYSIS']:
476 libraries['fftw3f'] = LibraryInfo()
477 libraries['fftw3f'].ParseConfig('pkg-config --cflags --libs fftw3f')
479 # Check for fftw3 header as well as the library
480 conf = Configure (libraries['fftw3f'])
481 if conf.CheckHeader ('fftw3.h') == False:
482 print "FFT Analysis cannot be compiled without the FFTW3 headers, which don't seem to be installed"
484 libraries['fftw3f'] = conf.Finish();
486 libraries['jack'] = LibraryInfo()
487 libraries['jack'].ParseConfig('pkg-config --cflags --libs jack')
489 libraries['xml'] = LibraryInfo()
490 libraries['xml'].ParseConfig('pkg-config --cflags --libs libxml-2.0')
492 libraries['xslt'] = LibraryInfo()
493 libraries['xslt'].ParseConfig('pkg-config --cflags --libs libxslt')
495 libraries['cairo'] = LibraryInfo()
496 libraries['cairo'].ParseConfig ('pkg-config --cflags --libs cairo')
498 libraries['pangocairo'] = LibraryInfo()
499 libraries['pangocairo'].ParseConfig ('pkg-config --cflags --libs pangocairo')
501 libraries['glib2'] = LibraryInfo()
502 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs glib-2.0')
503 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs gobject-2.0')
504 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs gmodule-2.0')
505 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs gthread-2.0')
507 libraries['gtk2'] = LibraryInfo()
508 libraries['gtk2'].ParseConfig ('pkg-config --cflags --libs gtk+-2.0')
510 libraries['gtk2-unix-print'] = LibraryInfo()
511 libraries['gtk2-unix-print'].ParseConfig ('pkg-config --cflags --libs gtk+-unix-print-2.0')
513 libraries['pango'] = LibraryInfo()
514 libraries['pango'].ParseConfig ('pkg-config --cflags --libs pango')
516 libraries['libgnomecanvas2'] = LibraryInfo()
517 libraries['libgnomecanvas2'].ParseConfig ('pkg-config --cflags --libs libgnomecanvas-2.0')
519 #libraries['flowcanvas'] = LibraryInfo(LIBS='flowcanvas', LIBPATH='#/libs/flowcanvas', CPPPATH='#libs/flowcanvas')
521 # The Ardour Control Protocol Library
523 libraries['ardour_cp'] = LibraryInfo (LIBS='ardour_cp', LIBPATH='#libs/surfaces/control_protocol',
524 CPPPATH='#libs/surfaces/control_protocol')
526 # The Ardour backend/engine
528 libraries['ardour'] = LibraryInfo (LIBS='ardour', LIBPATH='#libs/ardour', CPPPATH='#libs/ardour')
529 libraries['midi++2'] = LibraryInfo (LIBS='midi++', LIBPATH='#libs/midi++2', CPPPATH='#libs/midi++2')
530 libraries['pbd'] = LibraryInfo (LIBS='pbd', LIBPATH='#libs/pbd', CPPPATH='#libs/pbd')
531 libraries['gtkmm2ext'] = LibraryInfo (LIBS='gtkmm2ext', LIBPATH='#libs/gtkmm2ext', CPPPATH='#libs/gtkmm2ext')
534 # SCons should really do this for us
536 conf = Configure (env)
538 have_cxx = conf.TryAction (Action (str(env['CXX']) + ' --version'))
540 print "This system has no functional C++ compiler. You cannot build Ardour from source without one."
543 print "Congratulations, you have a functioning C++ compiler."
549 # Compiler flags and other system-dependent stuff
553 if env['GPROFILE'] == 1:
554 debug_flags = [ '-g', '-pg' ]
556 debug_flags = [ '-g' ]
558 # guess at the platform, used to define compiler flags
560 config_guess = os.popen("tools/config.guess").read()[:-1]
566 config = config_guess.split ("-")
568 print "system triple: " + config_guess
571 if env['DIST_TARGET'] == 'auto':
572 if config[config_arch] == 'apple':
573 # The [.] matches to the dot after the major version, "." would match any character
574 if re.search ("darwin[0-7][.]", config[config_kernel]) != None:
575 env['DIST_TARGET'] = 'panther'
577 env['DIST_TARGET'] = 'tiger'
579 if re.search ("x86_64", config[config_cpu]) != None:
580 env['DIST_TARGET'] = 'x86_64'
581 elif re.search("i[0-5]86", config[config_cpu]) != None:
582 env['DIST_TARGET'] = 'i386'
583 elif re.search("powerpc", config[config_cpu]) != None:
584 env['DIST_TARGET'] = 'powerpc'
586 env['DIST_TARGET'] = 'i686'
587 print "\n*******************************"
588 print "detected DIST_TARGET = " + env['DIST_TARGET']
589 print "*******************************\n"
592 if config[config_cpu] == 'powerpc' and env['DIST_TARGET'] != 'none':
594 # Apple/PowerPC optimization options
596 # -mcpu=7450 does not reliably work with gcc 3.*
598 if env['DIST_TARGET'] == 'panther' or env['DIST_TARGET'] == 'tiger':
599 if config[config_arch] == 'apple':
600 ## opt_flags.extend ([ "-mcpu=7450", "-faltivec"])
601 # to support g3s but still have some optimization for above
602 opt_flags.extend ([ "-mcpu=G3", "-mtune=7450"])
604 opt_flags.extend ([ "-mcpu=7400", "-maltivec", "-mabi=altivec"])
606 opt_flags.extend([ "-mcpu=750", "-mmultiple" ])
607 opt_flags.extend (["-mhard-float", "-mpowerpc-gfxopt"])
608 opt_flags.extend (["-Os"])
610 elif ((re.search ("i[0-9]86", config[config_cpu]) != None) or (re.search ("x86_64", config[config_cpu]) != None)) and env['DIST_TARGET'] != 'none':
612 build_host_supports_sse = 0
614 debug_flags.append ("-DARCH_X86")
615 opt_flags.append ("-DARCH_X86")
617 if config[config_kernel] == 'linux' :
619 if env['DIST_TARGET'] != 'i386':
621 flag_line = os.popen ("cat /proc/cpuinfo | grep '^flags'").read()[:-1]
622 x86_flags = flag_line.split (": ")[1:][0].split ()
624 if "mmx" in x86_flags:
625 opt_flags.append ("-mmmx")
626 if "sse" in x86_flags:
627 build_host_supports_sse = 1
628 if "3dnow" in x86_flags:
629 opt_flags.append ("-m3dnow")
631 if config[config_cpu] == "i586":
632 opt_flags.append ("-march=i586")
633 elif config[config_cpu] == "i686":
634 opt_flags.append ("-march=i686")
636 if ((env['DIST_TARGET'] == 'i686') or (env['DIST_TARGET'] == 'x86_64')) and build_host_supports_sse:
637 opt_flags.extend (["-msse", "-mfpmath=sse"])
638 debug_flags.extend (["-msse", "-mfpmath=sse"])
639 # end of processor-specific section
641 # optimization section
642 if env['FPU_OPTIMIZATION']:
643 if env['DIST_TARGET'] == 'tiger':
644 opt_flags.append ("-DBUILD_VECLIB_OPTIMIZATIONS")
645 debug_flags.append ("-DBUILD_VECLIB_OPTIMIZATIONS")
646 libraries['core'].Append(LINKFLAGS= '-framework Accelerate')
647 elif env['DIST_TARGET'] == 'i686' or env['DIST_TARGET'] == 'x86_64':
648 opt_flags.append ("-DBUILD_SSE_OPTIMIZATIONS")
649 debug_flags.append ("-DBUILD_SSE_OPTIMIZATIONS")
650 if env['DIST_TARGET'] == 'x86_64':
651 opt_flags.append ("-DUSE_X86_64_ASM")
652 debug_flags.append ("-DUSE_X86_64_ASM")
653 if build_host_supports_sse != 1:
654 print "\nWarning: you are building Ardour with SSE support even though your system does not support these instructions. (This may not be an error, especially if you are a package maintainer)"
655 # end optimization section
657 # handle x86/x86_64 libdir properly
659 if env['DIST_TARGET'] == 'x86_64':
660 env['LIBDIR']='lib64'
665 # save off guessed arch element in an env
667 env.Append(CONFIG_ARCH=config[config_arch])
671 # ARCH="..." overrides all
674 if env['ARCH'] != '':
675 opt_flags = env['ARCH'].split()
678 # prepend boiler plate optimization flags
683 "-fomit-frame-pointer",
689 if env['DEBUG'] == 1:
690 env.Append(CCFLAGS=" ".join (debug_flags))
691 env.Append(LINKFLAGS=" ".join (debug_flags))
693 env.Append(CCFLAGS=" ".join (opt_flags))
694 env.Append(LINKFLAGS=" ".join (opt_flags))
696 if env['UNIVERSAL'] == 1:
697 env.Append(CCFLAGS="-arch i386 -arch ppc")
698 env.Append(LINKFLAGS="-arch i386 -arch ppc")
704 env.Append(CCFLAGS="-Wall")
705 env.Append(CXXFLAGS="-Woverloaded-virtual")
707 if env['EXTRA_WARN']:
708 env.Append(CCFLAGS="-Wextra -pedantic -ansi")
709 env.Append(CXXFLAGS="-ansi")
710 # env.Append(CFLAGS="-iso")
713 env.Append(CCFLAGS="-DHAVE_LIBLO")
717 # fix scons nitpickiness on APPLE
721 def prep_libcheck(topenv, libinfo):
722 if topenv['DIST_TARGET'] == 'panther' or topenv['DIST_TARGET'] == 'tiger':
723 libinfo.Append(CCFLAGS="-I/opt/local/include", LINKFLAGS="-L/opt/local/lib")
725 prep_libcheck(env, env)
730 libraries['usb'] = LibraryInfo ()
731 prep_libcheck(env, libraries['usb'])
733 conf = Configure (libraries['usb'])
734 if conf.CheckLib ('usb', 'usb_interrupt_write'):
739 libraries['usb'] = conf.Finish ()
744 libraries['flac'] = LibraryInfo ()
745 prep_libcheck(env, libraries['flac'])
746 libraries['flac'].Append(CCFLAGS="-I/usr/local/include", LINKFLAGS="-L/usr/local/lib")
748 conf = Configure (libraries['flac'])
749 if conf.CheckLib ('FLAC', 'FLAC__stream_decoder_new', language='CXX'):
750 conf.env.Append(CCFLAGS='-DHAVE_FLAC')
751 libraries['flac'] = conf.Finish ()
753 # or if that fails...
754 #libraries['flac'] = LibraryInfo (LIBS='FLAC')
756 # boost (we don't link against boost, just use some header files)
758 libraries['boost'] = LibraryInfo ()
759 prep_libcheck(env, libraries['boost'])
760 libraries['boost'].Append(CCFLAGS="-I/usr/local/include", LINKFLAGS="-L/usr/local/lib")
761 conf = Configure (libraries['boost'])
762 if conf.CheckHeader ('boost/shared_ptr.hpp', language='CXX') == False:
763 print "Boost header files do not appear to be installed."
766 libraries['boost'] = conf.Finish ()
772 libraries['lo'] = LibraryInfo ()
773 prep_libcheck(env, libraries['lo'])
775 conf = Configure (libraries['lo'])
776 if conf.CheckLib ('lo', 'lo_server_new') == False:
777 print "liblo does not appear to be installed."
780 libraries['lo'] = conf.Finish ()
785 libraries['dmalloc'] = LibraryInfo ()
786 prep_libcheck(env, libraries['dmalloc'])
789 # look for the threaded version
792 conf = Configure (libraries['dmalloc'])
793 if conf.CheckLib ('dmallocth', 'dmalloc_shutdown'):
794 have_libdmalloc = True
796 have_libdmalloc = False
798 libraries['dmalloc'] = conf.Finish ()
801 # Audio/MIDI library (needed for MIDI, since audio is all handled via JACK)
804 conf = Configure(env)
806 if conf.CheckCHeader('alsa/asoundlib.h'):
807 libraries['sysmidi'] = LibraryInfo (LIBS='asound')
808 env['SYSMIDI'] = 'ALSA Sequencer'
809 subst_dict['%MIDITAG%'] = "seq"
810 subst_dict['%MIDITYPE%'] = "alsa/sequencer"
811 elif conf.CheckCHeader('/System/Library/Frameworks/CoreMIDI.framework/Headers/CoreMIDI.h'):
812 # this line is needed because scons can't handle -framework in ParseConfig() yet.
813 libraries['sysmidi'] = LibraryInfo (LINKFLAGS= '-framework CoreMIDI -framework CoreFoundation -framework CoreAudio -framework CoreServices -framework AudioUnit -framework AudioToolbox -bind_at_load')
814 env['SYSMIDI'] = 'CoreMIDI'
815 subst_dict['%MIDITAG%'] = "ardour"
816 subst_dict['%MIDITYPE%'] = "coremidi"
818 print "It appears you don't have the required MIDI libraries installed. For Linux this means you are missing the development package for ALSA libraries."
827 'sigc++-2.0' : '2.0',
829 'libgnomecanvasmm-2.6' : '2.12.0'
832 conf = Configure(env, custom_tests = { 'CheckPKGConfig' : CheckPKGConfig,
833 'CheckPKGVersion' : CheckPKGVersion })
835 for pkg, version in syslibdeps.iteritems():
836 if not conf.CheckPKGVersion( pkg, version ):
837 print '%s >= %s not found.' %(pkg, version)
838 DependenciesRequiredMessage()
843 libraries['sigc2'] = LibraryInfo()
844 libraries['sigc2'].ParseConfig('pkg-config --cflags --libs sigc++-2.0')
845 libraries['glibmm2'] = LibraryInfo()
846 libraries['glibmm2'].ParseConfig('pkg-config --cflags --libs glibmm-2.4')
847 libraries['cairomm'] = LibraryInfo()
848 libraries['cairomm'].ParseConfig('pkg-config --cflags --libs cairomm-1.0')
849 libraries['gdkmm2'] = LibraryInfo()
850 libraries['gdkmm2'].ParseConfig ('pkg-config --cflags --libs gdkmm-2.4')
851 libraries['gtkmm2'] = LibraryInfo()
852 libraries['gtkmm2'].ParseConfig ('pkg-config --cflags --libs gtkmm-2.4')
853 libraries['atkmm'] = LibraryInfo()
854 libraries['atkmm'].ParseConfig ('pkg-config --cflags --libs atkmm-1.6')
855 libraries['pangomm'] = LibraryInfo()
856 libraries['pangomm'].ParseConfig ('pkg-config --cflags --libs pangomm-1.4')
857 libraries['libgnomecanvasmm'] = LibraryInfo()
858 libraries['libgnomecanvasmm'].ParseConfig ('pkg-config --cflags --libs libgnomecanvasmm-2.6')
861 # cannot use system one for the time being
864 libraries['sndfile-ardour'] = LibraryInfo(LIBS='libsndfile-ardour',
865 LIBPATH='#libs/libsndfile',
866 CPPPATH=['#libs/libsndfile/src'])
868 # libraries['flowcanvas'] = LibraryInfo(LIBS='flowcanvas', LIBPATH='#/libs/flowcanvas', CPPPATH='#libs/flowcanvas')
869 libraries['soundtouch'] = LibraryInfo()
870 libraries['soundtouch'].ParseConfig ('pkg-config --cflags --libs soundtouch-1.0')
871 # Comment the previous line and uncomment this for Debian:
872 #libraries['soundtouch'].ParseConfig ('pkg-config --cflags --libs libSoundTouch')
874 libraries['appleutility'] = LibraryInfo(LIBS='libappleutility',
875 LIBPATH='#libs/appleutility',
876 CPPPATH='#libs/appleutility')
887 # these are unconditionally included but have
888 # tests internally to avoid compilation etc
892 # this is unconditionally included but has
893 # tests internally to avoid compilation etc
894 # if COREAUDIO is not set
906 libraries['sigc2'] = LibraryInfo(LIBS='sigc++2',
907 LIBPATH='#libs/sigc++2',
908 CPPPATH='#libs/sigc++2')
909 libraries['glibmm2'] = LibraryInfo(LIBS='glibmm2',
910 LIBPATH='#libs/glibmm2',
911 CPPPATH='#libs/glibmm2')
912 libraries['cairomm'] = LibraryInfo(LIBS='cairomm',
913 LIBPATH='#libs/cairomm',
914 CPPPATH='#libs/cairomm')
915 libraries['pangomm'] = LibraryInfo(LIBS='pangomm',
916 LIBPATH='#libs/gtkmm2/pango',
917 CPPPATH='#libs/gtkmm2/pango')
918 libraries['atkmm'] = LibraryInfo(LIBS='atkmm',
919 LIBPATH='#libs/gtkmm2/atk',
920 CPPPATH='#libs/gtkmm2/atk')
921 libraries['gdkmm2'] = LibraryInfo(LIBS='gdkmm2',
922 LIBPATH='#libs/gtkmm2/gdk',
923 CPPPATH='#libs/gtkmm2/gdk')
924 libraries['gtkmm2'] = LibraryInfo(LIBS='gtkmm2',
925 LIBPATH="#libs/gtkmm2/gtk",
926 CPPPATH='#libs/gtkmm2/gtk/')
927 libraries['libgnomecanvasmm'] = LibraryInfo(LIBS='libgnomecanvasmm',
928 LIBPATH='#libs/libgnomecanvasmm',
929 CPPPATH='#libs/libgnomecanvasmm')
931 libraries['soundtouch'] = LibraryInfo(LIBS='soundtouch',
932 LIBPATH='#libs/soundtouch',
933 CPPPATH=['#libs', '#libs/soundtouch'])
934 libraries['sndfile-ardour'] = LibraryInfo(LIBS='libsndfile-ardour',
935 LIBPATH='#libs/libsndfile',
936 CPPPATH=['#libs/libsndfile', '#libs/libsndfile/src'])
937 libraries['appleutility'] = LibraryInfo(LIBS='libappleutility',
938 LIBPATH='#libs/appleutility',
939 CPPPATH='#libs/appleutility')
952 # these are unconditionally included but have
953 # tests internally to avoid compilation etc
957 # this is unconditionally included but has
958 # tests internally to avoid compilation etc
959 # if COREAUDIO is not set
970 'libs/libgnomecanvasmm',
978 # * always build the LGPL control protocol lib, since we link against it from libardour
979 # * ditto for generic MIDI
980 # * tranzport checks whether it should build internally, but we need here so that
981 # its included in the tarball
984 surface_subdirs = [ 'libs/surfaces/control_protocol', 'libs/surfaces/generic_midi', 'libs/surfaces/tranzport', 'libs/surfaces/mackie' ]
991 print 'Disabled building Tranzport code because libusb could not be found'
992 if os.access ('libs/surfaces/sony9pin', os.F_OK):
993 surface_subdirs += [ 'libs/surfaces/sony9pin' ]
995 opts.Save('scache.conf', env)
996 Help(opts.GenerateHelpText(env))
998 if os.environ.has_key('PATH'):
999 env.Append(PATH = os.environ['PATH'])
1001 if os.environ.has_key('PKG_CONFIG_PATH'):
1002 env.Append(PKG_CONFIG_PATH = os.environ['PKG_CONFIG_PATH'])
1004 if os.environ.has_key('CC'):
1005 env['CC'] = os.environ['CC']
1007 if os.environ.has_key('CXX'):
1008 env['CXX'] = os.environ['CXX']
1010 if os.environ.has_key('DISTCC_HOSTS'):
1011 env['ENV']['DISTCC_HOSTS'] = os.environ['DISTCC_HOSTS']
1012 env['ENV']['HOME'] = os.environ['HOME']
1014 final_prefix = '$PREFIX'
1017 install_prefix = '$DESTDIR/$PREFIX'
1019 install_prefix = env['PREFIX']
1021 subst_dict['%INSTALL_PREFIX%'] = install_prefix;
1022 subst_dict['%FINAL_PREFIX%'] = final_prefix;
1023 subst_dict['%PREFIX%'] = final_prefix;
1025 if env['PREFIX'] == '/usr':
1026 final_config_prefix = '/etc'
1028 final_config_prefix = env['PREFIX'] + '/etc'
1030 config_prefix = '$DESTDIR' + final_config_prefix
1033 # everybody needs this
1036 env.Merge ([ libraries['core'] ])
1043 conf = Configure (env)
1045 nls_error = 'This system is not configured for internationalized applications. An english-only version will be built:'
1046 print 'Checking for internationalization support ...'
1047 have_gettext = conf.TryAction(Action('xgettext --version'))
1048 if have_gettext[0] != 1:
1049 nls_error += ' No xgettext command.'
1052 print "Found xgettext"
1054 have_msgmerge = conf.TryAction(Action('msgmerge --version'))
1055 if have_msgmerge[0] != 1:
1056 nls_error += ' No msgmerge command.'
1059 print "Found msgmerge"
1061 if not conf.CheckCHeader('libintl.h'):
1062 nls_error += ' No libintl.h.'
1068 print "International version will be built."
1072 env.Append(CCFLAGS="-DENABLE_NLS")
1074 Export('env install_prefix final_prefix config_prefix final_config_prefix libraries i18n ardour_version subst_dict')
1077 # the configuration file may be system dependent
1080 conf = env.Configure ()
1082 if conf.CheckCHeader('/System/Library/Frameworks/CoreAudio.framework/Versions/A/Headers/CoreAudio.h'):
1083 subst_dict['%JACK_INPUT%'] = "coreaudio:Built-in Audio:in"
1084 subst_dict['%JACK_OUTPUT%'] = "coreaudio:Built-in Audio:out"
1086 subst_dict['%JACK_INPUT%'] = "alsa_pcm:playback_"
1087 subst_dict['%JACK_OUTPUT%'] = "alsa_pcm:capture_"
1089 # posix_memalign available
1090 if not conf.CheckFunc('posix_memalign'):
1091 print 'Did not find posix_memalign(), using malloc'
1092 env.Append(CCFLAGS='-DNO_POSIX_MEMALIGN')
1097 # generate the per-user and system rc files from the same source
1099 rcbuild = env.SubstInFile ('ardour.rc','ardour.rc.in', SUBST_DICT = subst_dict)
1100 sysrcbuild = env.SubstInFile ('ardour_system.rc','ardour.rc.in', SUBST_DICT = subst_dict)
1102 # add to the substitution dictionary
1104 subst_dict['%VERSION%'] = ardour_version[0:3]
1105 subst_dict['%EXTRA_VERSION%'] = ardour_version[3:]
1106 subst_dict['%REVISION_STRING%'] = ''
1107 if os.path.exists('.svn'):
1108 subst_dict['%REVISION_STRING%'] = '.' + fetch_svn_revision ('.') + 'svn'
1110 # specbuild = env.SubstInFile ('ardour.spec','ardour.spec.in', SUBST_DICT = subst_dict)
1112 the_revision = env.Command ('frobnicatory_decoy', [], create_stored_revision)
1114 env.Alias('revision', the_revision)
1115 env.Alias('install', env.Install(os.path.join(config_prefix, 'ardour2'), 'ardour_system.rc'))
1116 env.Alias('install', env.Install(os.path.join(config_prefix, 'ardour2'), 'ardour.rc'))
1119 Default (sysrcbuild)
1123 Precious (env['DISTTREE'])
1125 env.Distribute (env['DISTTREE'],
1126 [ 'SConstruct', 'svn_revision.h',
1127 'COPYING', 'PACKAGER_README', 'README',
1129 'tools/config.guess',
1130 'icons/icon/ardour_icon_mac_mask.png',
1131 'icons/icon/ardour_icon_mac.png',
1132 'icons/icon/ardour_icon_tango_16px_blue.png',
1133 'icons/icon/ardour_icon_tango_16px_red.png',
1134 'icons/icon/ardour_icon_tango_22px_blue.png',
1135 'icons/icon/ardour_icon_tango_22px_red.png',
1136 'icons/icon/ardour_icon_tango_32px_blue.png',
1137 'icons/icon/ardour_icon_tango_32px_red.png',
1138 'icons/icon/ardour_icon_tango_48px_blue.png',
1139 'icons/icon/ardour_icon_tango_48px_red.png'
1141 glob.glob ('DOCUMENTATION/AUTHORS*') +
1142 glob.glob ('DOCUMENTATION/CONTRIBUTORS*') +
1143 glob.glob ('DOCUMENTATION/TRANSLATORS*') +
1144 glob.glob ('DOCUMENTATION/BUILD*') +
1145 glob.glob ('DOCUMENTATION/FAQ*') +
1146 glob.glob ('DOCUMENTATION/README*')
1149 srcdist = env.Tarball(env['TARBALL'], [ env['DISTTREE'], the_revision ])
1150 env.Alias ('srctar', srcdist)
1153 # don't leave the distree around
1156 env.AddPreAction (env['DISTTREE'], Action ('rm -rf ' + str (File (env['DISTTREE']))))
1157 env.AddPostAction (srcdist, Action ('rm -rf ' + str (File (env['DISTTREE']))))
1163 for subdir in coredirs:
1164 SConscript (subdir + '/SConscript')
1166 for sublistdir in [ subdirs, gtk_subdirs, surface_subdirs ]:
1167 for subdir in sublistdir:
1168 SConscript (subdir + '/SConscript')
1171 env.Clean ('scrub', [ 'scache.conf', '.sconf_temp', '.sconsign.dblite', 'config.log'])