define ARCH_X86 even on x86_64 (untested!); giso's patch for control surface track...
[ardour.git] / SConstruct
1 # -*- python -*-
2
3 #
4 # and there we have it, or do we?
5 #
6
7 import os
8 import os.path
9 import sys
10 import re
11 import shutil
12 import glob
13 import errno
14 import time
15 import platform
16 import string
17 import commands
18 from sets import Set
19 import SCons.Node.FS
20
21 SConsignFile()
22 EnsureSConsVersion(0, 96)
23
24 ardour_version = '2.7.1'
25
26 subst_dict = { }
27
28 #
29 # Command-line options
30 #
31
32 opts = Options('scache.conf')
33 opts.AddOptions(
34     ('ARCH', 'Set architecture-specific compilation flags by hand (all flags as 1 argument)',''),
35     ('WINDOWS_KEY', 'Set X Modifier (Mod1,Mod2,Mod3,Mod4,Mod5) for "Windows" key', 'Mod4'),
36     BoolOption('AUDIOUNITS', 'Compile with Apple\'s AudioUnit library. (experimental)', 0),
37     BoolOption('COREAUDIO', 'Compile with Apple\'s CoreAudio library', 0),
38     BoolOption('GTKOSX', 'Compile for use with GTK-OSX, not GTK-X11', 0),
39     BoolOption('OLDFONTS', 'Old school font sizes', 0),
40     BoolOption('DEBUG', 'Set to build with debugging information and no optimizations', 0),
41     BoolOption('STL_DEBUG', 'Set to build with Standard Template Library Debugging', 0),
42     PathOption('DESTDIR', 'Set the intermediate install "prefix"', '/'),
43     EnumOption('DIST_TARGET', 'Build target for cross compiling packagers', 'auto', allowed_values=('auto', 'i386', 'i686', 'x86_64', 'powerpc', 'tiger', 'panther', 'leopard', 'none' ), ignorecase=2),
44     BoolOption('DMALLOC', 'Compile and link using the dmalloc library', 0),
45     BoolOption('EXTRA_WARN', 'Compile with -Wextra, -ansi, and -pedantic.  Might break compilation.  For pedants', 0),
46     BoolOption('FFT_ANALYSIS', 'Include FFT analysis window', 1),
47     BoolOption('FREESOUND', 'Include Freesound database lookup', 0),
48     BoolOption('FPU_OPTIMIZATION', 'Build runtime checked assembler code', 1),
49     BoolOption('LIBLO', 'Compile with support for liblo library', 1),
50     BoolOption('NLS', 'Set to turn on i18n support', 1),
51     PathOption('PREFIX', 'Set the install "prefix"', '/usr/local'),
52     BoolOption('SURFACES', 'Build support for control surfaces', 1),
53     BoolOption('WIIMOTE', 'Build the wiimote control surface', 0),
54     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),
55     BoolOption('UNIVERSAL', 'Compile as universal binary.  Requires that external libraries are already universal.', 0),
56     BoolOption('VERSIONED', 'Add revision information to ardour/gtk executable name inside the build directory', 0),
57     BoolOption('VST', 'Compile with support for VST', 0),
58     BoolOption('LV2', 'Compile with support for LV2 (if slv2 is available)', 1),
59     BoolOption('GPROFILE', 'Compile with support for gprofile (Developers only)', 0),
60     BoolOption('FREEDESKTOP', 'Install MIME type, icons and .desktop file as per the freedesktop.org spec (requires xdg-utils and shared-mime-info). "scons uninstall" removes associations in desktop database', 0),
61     BoolOption('TRANZPORT', 'Compile with support for Frontier Designs (if libusb is available)', 1),
62     BoolOption('AUBIO', "Use Paul Brossier's aubio library for feature detection (if available)", 1)
63     
64 )
65
66 #----------------------------------------------------------------------
67 # a handy helper that provides a way to merge compile/link information
68 # from multiple different "environments"
69 #----------------------------------------------------------------------
70 #
71 class LibraryInfo(Environment):
72     def __init__(self,*args,**kw):
73         Environment.__init__ (self,*args,**kw)
74     
75     def Merge (self,others):
76         for other in others:
77             self.Append (LIBS = other.get ('LIBS',[]))
78             self.Append (LIBPATH = other.get ('LIBPATH', []))
79             self.Append (CPPPATH = other.get('CPPPATH', []))
80             self.Append (LINKFLAGS = other.get('LINKFLAGS', []))
81             self.Append (CCFLAGS = other.get('CCFLAGS', []))
82         self.Replace(LIBPATH = list(Set(self.get('LIBPATH', []))))
83         self.Replace(CPPPATH = list(Set(self.get('CPPPATH',[]))))
84         #doing LINKFLAGS breaks -framework
85         #doing LIBS break link order dependency
86     
87     def ENV_update(self, src_ENV):
88         for k in src_ENV.keys():
89             if k in self['ENV'].keys() and k in [ 'PATH', 'LD_LIBRARY_PATH',
90                                                   'LIB', 'INCLUDE' ]:
91                 self['ENV'][k]=SCons.Util.AppendPath(self['ENV'][k], src_ENV[k])
92             else:
93                 self['ENV'][k]=src_ENV[k]
94
95 env = LibraryInfo (options = opts,
96                    CPPPATH = [ '.' ],
97                    VERSION = ardour_version,
98                    TARBALL='ardour-' + ardour_version + '.tar.bz2',
99                    DISTFILES = [ ],
100                    DISTTREE  = '#ardour-' + ardour_version,
101                    DISTCHECKDIR = '#ardour-' + ardour_version + '/check'
102                    )
103
104 env.ENV_update(os.environ)
105
106 #----------------------------------------------------------------------
107 # Builders
108 #----------------------------------------------------------------------
109
110 # Handy subst-in-file builder
111 #
112
113 def do_subst_in_file(targetfile, sourcefile, dict):
114     """Replace all instances of the keys of dict with their values.
115     For example, if dict is {'%VERSION%': '1.2345', '%BASE%': 'MyProg'},
116     then all instances of %VERSION% in the file will be replaced with 1.2345 etc.
117     """
118     try:
119         f = open(sourcefile, 'rb')
120         contents = f.read()
121         f.close()
122     except:
123         raise SCons.Errors.UserError, "Can't read source file %s"%sourcefile
124     for (k,v) in dict.items():
125         contents = re.sub(k, v, contents)
126     try:
127         f = open(targetfile, 'wb')
128         f.write(contents)
129         f.close()
130     except:
131         raise SCons.Errors.UserError, "Can't write target file %s"%targetfile
132     return 0 # success
133
134 def subst_in_file(target, source, env):
135     if not env.has_key('SUBST_DICT'):
136         raise SCons.Errors.UserError, "SubstInFile requires SUBST_DICT to be set."
137     d = dict(env['SUBST_DICT']) # copy it
138     for (k,v) in d.items():
139         if callable(v):
140             d[k] = env.subst(v())
141         elif SCons.Util.is_String(v):
142             d[k]=env.subst(v)
143         else:
144             raise SCons.Errors.UserError, "SubstInFile: key %s: %s must be a string or callable"%(k, repr(v))
145     for (t,s) in zip(target, source):
146         return do_subst_in_file(str(t), str(s), d)
147
148 def subst_in_file_string(target, source, env):
149     """This is what gets printed on the console."""
150     return '\n'.join(['Substituting vars from %s into %s'%(str(s), str(t))
151                       for (t,s) in zip(target, source)])
152
153 def subst_emitter(target, source, env):
154     """Add dependency from substituted SUBST_DICT to target.
155     Returns original target, source tuple unchanged.
156     """
157     d = env['SUBST_DICT'].copy() # copy it
158     for (k,v) in d.items():
159         if callable(v):
160             d[k] = env.subst(v())
161         elif SCons.Util.is_String(v):
162             d[k]=env.subst(v)
163     Depends(target, SCons.Node.Python.Value(d))
164     # Depends(target, source) # this doesn't help the install-sapphire-linux.sh problem
165     return target, source
166
167 subst_action = Action (subst_in_file, subst_in_file_string)
168 env['BUILDERS']['SubstInFile'] = Builder(action=subst_action, emitter=subst_emitter)
169
170 #
171 # internationalization
172 #
173
174 # po_builder: builder function to copy po files to the parent directory while updating them
175 #
176 # first source:  .po file
177 # second source: .pot file
178 #
179
180 def po_builder(target,source,env):
181     os.spawnvp (os.P_WAIT, 'cp', ['cp', str(source[0]), str(target[0])])
182     args = [ 'msgmerge',
183              '--update',
184              str(target[0]),
185              str(source[1])
186              ]
187     print 'Updating ' + str(target[0])
188     return os.spawnvp (os.P_WAIT, 'msgmerge', args)
189
190 po_bld = Builder (action = po_builder)
191 env.Append(BUILDERS = {'PoBuild' : po_bld})
192
193 # mo_builder: builder function for (binary) message catalogs (.mo)
194 #
195 # first source:  .po file
196 #
197
198 def mo_builder(target,source,env):
199     args = [ 'msgfmt',
200              '-c',
201              '-o',
202              target[0].get_path(),
203              source[0].get_path()
204              ]
205     return os.spawnvp (os.P_WAIT, 'msgfmt', args)
206
207 mo_bld = Builder (action = mo_builder)
208 env.Append(BUILDERS = {'MoBuild' : mo_bld})
209
210 # pot_builder: builder function for message templates (.pot)
211 #
212 # source: list of C/C++ etc. files to extract messages from
213 #
214
215 def pot_builder(target,source,env):
216     args = [ 'xgettext',
217              '--keyword=_',
218              '--keyword=N_',
219              '--from-code=UTF-8',
220              '-o', target[0].get_path(),
221              "--default-domain=" + env['PACKAGE'],
222              '--copyright-holder="Paul Davis"' ]
223     args += [ src.get_path() for src in source ]
224     
225     return os.spawnvp (os.P_WAIT, 'xgettext', args)
226
227 pot_bld = Builder (action = pot_builder)
228 env.Append(BUILDERS = {'PotBuild' : pot_bld})
229
230 #
231 # utility function, not a builder
232 #
233
234 def i18n (buildenv, sources, installenv):
235     domain = buildenv['PACKAGE']
236     potfile = buildenv['POTFILE']
237     
238     installenv.Alias ('potupdate', buildenv.PotBuild (potfile, sources))
239     
240     p_oze = [ os.path.basename (po) for po in glob.glob ('po/*.po') ]
241     languages = [ po.replace ('.po', '') for po in p_oze ]
242     
243     for po_file in p_oze:
244         buildenv.PoBuild(po_file, ['po/'+po_file, potfile])
245         mo_file = po_file.replace (".po", ".mo")
246         installenv.Alias ('install', buildenv.MoBuild (mo_file, po_file))
247         installenv.Alias ('msgupdate', buildenv.MoBuild (mo_file, po_file))
248     
249     for lang in languages:
250         modir = (os.path.join (install_prefix, 'share/locale/' + lang + '/LC_MESSAGES/'))
251         moname = domain + '.mo'
252         installenv.Alias('install', installenv.InstallAs (os.path.join (modir, moname), lang + '.mo'))
253
254
255 def fetch_svn_revision (path):
256     cmd = "LANG= "
257     cmd += "svn info "
258     cmd += path
259     cmd += " | awk '/^Revision:/ { print $2}'"
260     return commands.getoutput (cmd)
261
262 def create_stored_revision (target = None, source = None, env = None):
263     if os.path.exists('.svn'):    
264         rev = fetch_svn_revision ('.');
265         try:
266             text  = "#include <ardour/svn_revision.h>\n"
267             text += "namespace ARDOUR {\n";
268             text += "extern const char* svn_revision = \"" + rev + "\";\n";
269             text += "}\n";
270             print '============> writing svn revision info to libs/ardour/svn_revision.cc\n'
271             o = file ('libs/ardour/svn_revision.cc', 'w')
272             o.write (text)
273             o.close ()
274         except IOError:
275             print "Could not open libs/ardour/svn_revision.cc for writing\n"
276             sys.exit (-1)
277     else:
278         if not os.path.exists('libs/ardour/ardour/svn_revision.h'):    
279             print "This release of ardour is missing libs/ardour/ardour/svn_revision.h. Blame the packager."
280             sys.exit (-1)
281
282 #
283 # A generic builder for version.cc files
284 #
285 # note: requires that DOMAIN, MAJOR, MINOR, MICRO are set in the construction environment
286 # note: assumes one source files, the header that declares the version variables
287 #
288
289 def version_builder (target, source, env):
290
291     text  = "int " + env['DOMAIN'] + "_major_version = " + str (env['MAJOR']) + ";\n"
292     text += "int " + env['DOMAIN'] + "_minor_version = " + str (env['MINOR']) + ";\n"
293     text += "int " + env['DOMAIN'] + "_micro_version = " + str (env['MICRO']) + ";\n"
294     
295     try:
296         o = file (target[0].get_path(), 'w')
297         o.write (text)
298         o.close ()
299     except IOError:
300         print "Could not open", target[0].get_path(), " for writing\n"
301         sys.exit (-1)
302
303     text  = "#ifndef __" + env['DOMAIN'] + "_version_h__\n"
304     text += "#define __" + env['DOMAIN'] + "_version_h__\n"
305     text += "extern const char* " + env['DOMAIN'] + "_revision;\n"
306     text += "extern int " + env['DOMAIN'] + "_major_version;\n"
307     text += "extern int " + env['DOMAIN'] + "_minor_version;\n"
308     text += "extern int " + env['DOMAIN'] + "_micro_version;\n"
309     text += "#endif /* __" + env['DOMAIN'] + "_version_h__ */\n"
310     
311     try:
312         o = file (target[1].get_path(), 'w')
313         o.write (text)
314         o.close ()
315     except IOError:
316         print "Could not open", target[1].get_path(), " for writing\n"
317         sys.exit (-1)
318         
319     return None
320
321 version_bld = Builder (action = version_builder)
322 env.Append (BUILDERS = {'VersionBuild' : version_bld})
323
324 #
325 # a builder that makes a hard link from the 'source' executable to a name with
326 # a "build ID" based on the most recent CVS activity that might be reasonably
327 # related to version activity. this relies on the idea that the SConscript
328 # file that builds the executable is updated with new version info and committed
329 # to the source code repository whenever things change.
330 #
331
332 def versioned_builder(target,source,env):
333     w, r = os.popen2( "LANG= svn info | awk '/^Revision:/ { print $2}'")
334     
335     last_revision = r.readline().strip()
336     w.close()
337     r.close()
338     if last_revision == "":
339         print "No SVN info found - versioned executable cannot be built"
340         return -1
341     
342     print "The current build ID is " + last_revision
343     
344     tagged_executable = source[0].get_path() + '-' + last_revision
345     
346     if os.path.exists (tagged_executable):
347         print "Replacing existing executable with the same build tag."
348         os.unlink (tagged_executable)
349     
350     return os.link (source[0].get_path(), tagged_executable)
351
352 verbuild = Builder (action = versioned_builder)
353 env.Append (BUILDERS = {'VersionedExecutable' : verbuild})
354
355 #
356 # source tar file builder
357 #
358
359 def distcopy (target, source, env):
360     treedir = str (target[0])
361     
362     try:
363         os.mkdir (treedir)
364     except OSError, (errnum, strerror):
365         if errnum != errno.EEXIST:
366             print 'mkdir ', treedir, ':', strerror
367     
368     cmd = 'tar cf - '
369     #
370     # we don't know what characters might be in the file names
371     # so quote them all before passing them to the shell
372     #
373     all_files = ([ str(s) for s in source ])
374     cmd += " ".join ([ "'%s'" % quoted for quoted in all_files])
375     cmd += ' | (cd ' + treedir + ' && tar xf -)'
376     p = os.popen (cmd)
377     return p.close ()
378
379 def tarballer (target, source, env):
380     cmd = 'tar -jcf ' + str (target[0]) +  ' ' + str(source[0]) + "  --exclude '*~'" + " --exclude .svn --exclude '.svn/*'"
381     print 'running ', cmd, ' ... '
382     p = os.popen (cmd)
383     return p.close ()
384
385 dist_bld = Builder (action = distcopy,
386                     target_factory = SCons.Node.FS.default_fs.Entry,
387                     source_factory = SCons.Node.FS.default_fs.Entry,
388                     multi = 1)
389
390 tarball_bld = Builder (action = tarballer,
391                        target_factory = SCons.Node.FS.default_fs.Entry,
392                        source_factory = SCons.Node.FS.default_fs.Entry)
393
394 env.Append (BUILDERS = {'Distribute' : dist_bld})
395 env.Append (BUILDERS = {'Tarball' : tarball_bld})
396
397 #
398 # Make sure they know what they are doing
399 #
400
401 if env['VST']:
402     if os.path.isfile('.personal_use_only'):
403         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."
404     else:
405         sys.stdout.write ("Are you building Ardour for personal use (rather than distribution to others)? [no]: ")
406         answer = sys.stdin.readline ()
407         answer = answer.rstrip().strip()
408         if answer == "yes" or answer == "y":
409             fh = open('.personal_use_only', 'w')
410             fh.close()
411             print "OK, VST support will be enabled"
412         else:
413             print 'You cannot build Ardour with VST support for distribution to others.\nIt is a violation of several different licenses. Build with VST=false.'
414             sys.exit (-1);
415 else:
416     if os.path.isfile('.personal_use_only'):
417         os.remove('.personal_use_only')
418
419 ####################
420 # push environment
421 ####################
422
423 def pushEnvironment(context):
424     if os.environ.has_key('PATH'):
425         context.Append(PATH = os.environ['PATH'])
426         
427     if os.environ.has_key('PKG_CONFIG_PATH'):
428         context.Append(PKG_CONFIG_PATH = os.environ['PKG_CONFIG_PATH'])
429             
430     if os.environ.has_key('CC'):
431         context['CC'] = os.environ['CC']
432                 
433     if os.environ.has_key('CXX'):
434         context['CXX'] = os.environ['CXX']
435
436     if os.environ.has_key('DISTCC_HOSTS'):
437         context['ENV']['DISTCC_HOSTS'] = os.environ['DISTCC_HOSTS']
438         context['ENV']['HOME'] = os.environ['HOME']
439
440 pushEnvironment (env)
441
442 #######################
443 # Dependency Checking #
444 #######################
445
446 deps = \
447 {
448         'glib-2.0'             : '2.10.1',
449         'gthread-2.0'          : '2.10.1',
450         'gtk+-2.0'             : '2.8.1',
451         'libxml-2.0'           : '2.6.0',
452         'samplerate'           : '0.1.0',
453         'raptor'               : '1.4.2',
454         'lrdf'                 : '0.4.0',
455         'jack'                 : '0.109.0',
456         'libgnomecanvas-2.0'   : '2.0',
457 }
458
459 def DependenciesRequiredMessage():
460         print 'You do not have the necessary dependencies required to build ardour'
461         print 'Please consult http://ardour.org/building for more information'
462
463 def CheckPKGConfig(context, version):
464     context.Message( 'Checking for pkg-config version >= %s... ' %version )
465     ret = context.TryAction('pkg-config --atleast-pkgconfig-version=%s' % version)[0]
466     context.Result( ret )
467     return ret
468
469 def CheckPKGVersion(context, name, version):
470     context.Message( 'Checking for %s... ' % name )
471     ret = context.TryAction('pkg-config --atleast-version=%s %s' %(version,name) )[0]
472     context.Result( ret )
473     return ret
474
475 def CheckPKGExists(context, name):
476     context.Message ('Checking for %s...' % name)
477     ret = context.TryAction('pkg-config --exists %s' % name)[0]
478     context.Result (ret)
479     return ret
480
481 conf = Configure(env, custom_tests = { 'CheckPKGConfig' : CheckPKGConfig,
482                                        'CheckPKGVersion' : CheckPKGVersion })
483
484 # I think a more recent version is needed on win32
485 min_pkg_config_version = '0.8.0'
486
487 if not conf.CheckPKGConfig(min_pkg_config_version):
488      print 'pkg-config >= %s not found.' % min_pkg_config_version
489      Exit(1)
490
491 for pkg, version in deps.iteritems():
492         if not conf.CheckPKGVersion( pkg, version ):
493                 print '%s >= %s not found.' %(pkg, version)
494                 DependenciesRequiredMessage()
495                 Exit(1)
496
497 env = conf.Finish()
498
499 # ----------------------------------------------------------------------
500 # Construction environment setup
501 # ----------------------------------------------------------------------
502
503 libraries = { }
504
505 libraries['core'] = LibraryInfo (CCFLAGS = '-Ilibs')
506
507 #libraries['sndfile'] = LibraryInfo()
508 #libraries['sndfile'].ParseConfig('pkg-config --cflags --libs sndfile')
509
510 libraries['lrdf'] = LibraryInfo()
511 libraries['lrdf'].ParseConfig('pkg-config --cflags --libs lrdf')
512
513 libraries['raptor'] = LibraryInfo()
514 libraries['raptor'].ParseConfig('pkg-config --cflags --libs raptor')
515
516 libraries['samplerate'] = LibraryInfo()
517 libraries['samplerate'].ParseConfig('pkg-config --cflags --libs samplerate')
518
519 conf = env.Configure (custom_tests = { 'CheckPKGExists' : CheckPKGExists } )
520
521 if conf.CheckPKGExists ('fftw3f'):
522     libraries['fftw3f'] = LibraryInfo()
523     libraries['fftw3f'].ParseConfig('pkg-config --cflags --libs fftw3f')
524
525 if conf.CheckPKGExists ('fftw3'):
526     libraries['fftw3'] = LibraryInfo()
527     libraries['fftw3'].ParseConfig('pkg-config --cflags --libs fftw3')
528
529 if conf.CheckPKGExists ('aubio'):
530     libraries['aubio'] = LibraryInfo()
531     libraries['aubio'].ParseConfig('pkg-config --cflags --libs aubio')
532     env['AUBIO'] = 1
533 else:
534     env['AUBIO'] = 0
535
536 env = conf.Finish ()
537
538 if env['FFT_ANALYSIS']:
539         #
540         # Check for fftw3 header as well as the library
541         #
542
543         conf = Configure(libraries['fftw3'])
544
545         if conf.CheckHeader ('fftw3.h') == False:
546             print ('Ardour cannot be compiled without the FFTW3 headers, which do not seem to be installed')
547             sys.exit (1)            
548         conf.Finish()
549
550 if env['FREESOUND']:
551         #
552         # Check for curl header as well as the library
553         #
554
555         libraries['curl'] = LibraryInfo()
556
557         conf = Configure(libraries['curl'])
558
559         if conf.CheckHeader ('curl/curl.h') == False:
560                 print ('Ardour cannot be compiled without the curl headers, which do not seem to be installed')
561                 sys.exit (1)            
562         else:
563                 libraries['curl'].ParseConfig('pkg-config --cflags --libs libcurl')
564         conf.Finish()
565 else:
566         print 'FREESOUND support is not enabled.  Build with \'scons FREESOUND=1\' to enable.'
567
568 if env['LV2']:
569         conf = env.Configure(custom_tests = { 'CheckPKGVersion' : CheckPKGVersion})
570         
571         if conf.CheckPKGVersion('slv2', '0.6.1'):
572                 libraries['slv2'] = LibraryInfo()
573                 libraries['slv2'].ParseConfig('pkg-config --cflags --libs slv2')
574                 env.Append (CCFLAGS="-DHAVE_LV2")
575         else:
576                 print 'LV2 support is not enabled (SLV2 not found or older than 0.6.0)'
577                 env['LV2'] = 0
578         conf.Finish()
579 else:
580         print 'LV2 support is not enabled.  Build with \'scons LV2=1\' to enable.'
581
582 if not env['WIIMOTE']:
583         print 'WIIMOTE not enabled. Build with \'scons WIIMOTE=1\' to enable support.'
584
585 libraries['jack'] = LibraryInfo()
586 libraries['jack'].ParseConfig('pkg-config --cflags --libs jack')
587
588 libraries['xml'] = LibraryInfo()
589 libraries['xml'].ParseConfig('pkg-config --cflags --libs libxml-2.0')
590
591 libraries['xslt'] = LibraryInfo()
592 libraries['xslt'].ParseConfig('pkg-config --cflags --libs libxslt')
593
594 libraries['glib2'] = LibraryInfo()
595 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs glib-2.0')
596 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs gobject-2.0')
597 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs gmodule-2.0')
598 libraries['glib2'].ParseConfig ('pkg-config --cflags --libs gthread-2.0')
599
600 libraries['freetype2'] = LibraryInfo()
601 libraries['freetype2'].ParseConfig ('pkg-config --cflags --libs freetype2')
602
603 libraries['gtk2'] = LibraryInfo()
604 libraries['gtk2'].ParseConfig ('pkg-config --cflags --libs gtk+-2.0')
605
606 libraries['pango'] = LibraryInfo()
607 libraries['pango'].ParseConfig ('pkg-config --cflags --libs pango')
608
609 libraries['libgnomecanvas2'] = LibraryInfo()
610 libraries['libgnomecanvas2'].ParseConfig ('pkg-config --cflags --libs libgnomecanvas-2.0')
611
612 #libraries['flowcanvas'] = LibraryInfo(LIBS='flowcanvas', LIBPATH='#/libs/flowcanvas', CPPPATH='#libs/flowcanvas')
613
614 # The Ardour Control Protocol Library
615
616 libraries['ardour_cp'] = LibraryInfo (LIBS='ardour_cp', LIBPATH='#libs/surfaces/control_protocol',
617                                       CPPPATH='#libs/surfaces/control_protocol')
618
619 # The Ardour backend/engine
620
621 libraries['ardour'] = LibraryInfo (LIBS='ardour', LIBPATH='#libs/ardour', CPPPATH='#libs/ardour')
622 libraries['midi++2'] = LibraryInfo (LIBS='midi++', LIBPATH='#libs/midi++2', CPPPATH='#libs/midi++2')
623 libraries['pbd']    = LibraryInfo (LIBS='pbd', LIBPATH='#libs/pbd', CPPPATH='#libs/pbd')
624 libraries['gtkmm2ext'] = LibraryInfo (LIBS='gtkmm2ext', LIBPATH='#libs/gtkmm2ext', CPPPATH='#libs/gtkmm2ext')
625
626
627 # SCons should really do this for us
628
629 conf = env.Configure ()
630
631 have_cxx = conf.TryAction (Action (str(env['CXX']) + ' --version'))
632 if have_cxx[0] != 1:
633     print "This system has no functional C++ compiler. You cannot build Ardour from source without one."
634     sys.exit (1)
635 else:
636     print "Congratulations, you have a functioning C++ compiler."
637
638 env = conf.Finish()
639
640
641 #
642 # Compiler flags and other system-dependent stuff
643 #
644
645 opt_flags = []
646 if env['GPROFILE'] == 1:
647     debug_flags = [ '-g', '-pg' ]
648 else:
649     debug_flags = [ '-g' ]
650
651 # guess at the platform, used to define compiler flags
652
653 config_guess = os.popen("tools/config.guess").read()[:-1]
654
655 config_cpu = 0
656 config_arch = 1
657 config_kernel = 2
658 config_os = 3
659 config = config_guess.split ("-")
660
661 print "system triple: " + config_guess
662
663 # Autodetect
664 if env['DIST_TARGET'] == 'auto':
665     if config[config_arch] == 'apple':
666         # The [.] matches to the dot after the major version, "." would match any character
667         if re.search ("darwin[0-7][.]", config[config_kernel]) != None:
668             env['DIST_TARGET'] = 'panther'
669         if re.search ("darwin8[.]", config[config_kernel]) != None:
670             env['DIST_TARGET'] = 'tiger'
671         else:
672             env['DIST_TARGET'] = 'leopard'
673     else:
674         if re.search ("x86_64", config[config_cpu]) != None:
675             env['DIST_TARGET'] = 'x86_64'
676         elif re.search("i[0-5]86", config[config_cpu]) != None:
677             env['DIST_TARGET'] = 'i386'
678         elif re.search("powerpc", config[config_cpu]) != None:
679             env['DIST_TARGET'] = 'powerpc'
680         else:
681             env['DIST_TARGET'] = 'i686'
682     print "\n*******************************"
683     print "detected DIST_TARGET = " + env['DIST_TARGET']
684     print "*******************************\n"
685
686
687 if config[config_cpu] == 'powerpc' and env['DIST_TARGET'] != 'none':
688     #
689     # Apple/PowerPC optimization options
690     #
691     # -mcpu=7450 does not reliably work with gcc 3.*
692     #
693     if env['DIST_TARGET'] == 'panther' or env['DIST_TARGET'] == 'tiger':
694         if config[config_arch] == 'apple':
695             ## opt_flags.extend ([ "-mcpu=7450", "-faltivec"])
696             # to support g3s but still have some optimization for above
697             opt_flags.extend ([ "-mcpu=G3", "-mtune=7450"])
698         else:
699             opt_flags.extend ([ "-mcpu=7400", "-maltivec", "-mabi=altivec"])
700     else:
701         opt_flags.extend([ "-mcpu=750", "-mmultiple" ])
702     opt_flags.extend (["-mhard-float", "-mpowerpc-gfxopt"])
703     opt_flags.extend (["-Os"])
704
705 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':
706     
707     build_host_supports_sse = 0
708     
709     #
710     # ARCH_X86 means anything in the x86 family from i386 to x86_64
711     # USE_X86_64_ASM is used to distingush 32 and 64 bit assembler
712     #
713
714     if (re.search ("(i[0-9]86|x86_64)", config[config_cpu]) != None):
715         debug_flags.append ("-DARCH_X86")
716         opt_flags.append ("-DARCH_X86")
717     
718     if config[config_kernel] == 'linux' :
719         
720         if env['DIST_TARGET'] != 'i386':
721             
722             flag_line = os.popen ("cat /proc/cpuinfo | grep '^flags'").read()[:-1]
723             x86_flags = flag_line.split (": ")[1:][0].split ()
724             
725             if "mmx" in x86_flags:
726                 opt_flags.append ("-mmmx")
727             if "sse" in x86_flags:
728                 build_host_supports_sse = 1
729             if "3dnow" in x86_flags:
730                 opt_flags.append ("-m3dnow")
731             
732             if config[config_cpu] == "i586":
733                 opt_flags.append ("-march=i586")
734             elif config[config_cpu] == "i686":
735                 opt_flags.append ("-march=i686")
736     
737     if ((env['DIST_TARGET'] == 'i686') or (env['DIST_TARGET'] == 'x86_64')) and build_host_supports_sse:
738         opt_flags.extend (["-msse", "-mfpmath=sse", "-DUSE_XMMINTRIN"])
739         debug_flags.extend (["-msse", "-mfpmath=sse", "-DUSE_XMMINTRIN"])
740 # end of processor-specific section
741
742 # optimization section
743 if env['FPU_OPTIMIZATION']:
744     if env['DIST_TARGET'] == 'tiger' or env['DIST_TARGET'] == 'leopard':
745         opt_flags.append ("-DBUILD_VECLIB_OPTIMIZATIONS");
746         debug_flags.append ("-DBUILD_VECLIB_OPTIMIZATIONS");
747         libraries['core'].Append(LINKFLAGS= '-framework Accelerate')
748     elif env['DIST_TARGET'] == 'i686' or env['DIST_TARGET'] == 'x86_64':
749         opt_flags.append ("-DBUILD_SSE_OPTIMIZATIONS")
750         debug_flags.append ("-DBUILD_SSE_OPTIMIZATIONS")
751         if env['DIST_TARGET'] == 'x86_64':
752             opt_flags.append ("-DUSE_X86_64_ASM")
753             debug_flags.append ("-DUSE_X86_64_ASM")
754         if build_host_supports_sse != 1:
755             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)"
756 # end optimization section
757
758 # handle x86/x86_64 libdir properly
759
760 if env['DIST_TARGET'] == 'x86_64':
761     env['LIBDIR']='lib64'
762 else:
763     env['LIBDIR']='lib'
764
765 #
766 # a single way to test if we're on OS X
767 #
768
769 if env['DIST_TARGET'] in ['panther', 'tiger', 'leopard' ]:
770     env['IS_OSX'] = 1
771     # force tiger or later, to avoid issues on PPC which defaults
772     # back to 10.1 if we don't tell it otherwise.
773     env.Append (CCFLAGS="-DMAC_OS_X_VERSION_MIN_REQUIRED=1040")
774
775     if env['DIST_TARGET'] == 'leopard':
776         # need this to really build against the 10.4 SDK when building on leopard
777         # ideally this would be configurable, but lets just do that later when we need it
778         env.Append(CCFLAGS="-mmacosx-version-min=10.4 -isysroot /Developer/SDKs/MacOSX10.4u.sdk")
779         env.Append(LINKFLAGS="-mmacosx-version-min=10.4 -isysroot /Developer/SDKs/MacOSX10.4u.sdk")
780
781 else:
782     env['IS_OSX'] = 0
783
784 #
785 # save off guessed arch element in an env
786 #
787 env.Append(CONFIG_ARCH=config[config_arch])
788
789
790 #
791 # ARCH="..." overrides all
792 #
793
794 if env['ARCH'] != '':
795     opt_flags = env['ARCH'].split()
796
797 #
798 # prepend boiler plate optimization flags
799 #
800
801 opt_flags[:0] = [
802     "-O3",
803     "-fomit-frame-pointer",
804     "-ffast-math",
805     "-fstrength-reduce",
806     "-pipe"
807     ]
808
809 if env['DEBUG'] == 1:
810     env.Append(CCFLAGS=" ".join (debug_flags))
811     env.Append(LINKFLAGS=" ".join (debug_flags))
812 else:
813     env.Append(CCFLAGS=" ".join (opt_flags))
814     env.Append(LINKFLAGS=" ".join (opt_flags))
815
816 if env['STL_DEBUG'] == 1:
817     env.Append(CXXFLAGS="-D_GLIBCXX_DEBUG")
818
819 if env['UNIVERSAL'] == 1:
820     env.Append(CCFLAGS="-arch i386 -arch ppc")
821     env.Append(LINKFLAGS="-arch i386 -arch ppc")
822
823
824 #
825 # warnings flags
826 #
827
828 env.Append(CCFLAGS="-Wall")
829 env.Append(CXXFLAGS="-Woverloaded-virtual")
830
831 if env['EXTRA_WARN']:
832     env.Append(CCFLAGS="-Wextra -pedantic -ansi")
833     env.Append(CXXFLAGS="-ansi")
834 #    env.Append(CFLAGS="-iso")
835
836 if env['LIBLO']:
837     env.Append(CCFLAGS="-DHAVE_LIBLO")
838
839
840 #
841 # fix scons nitpickiness on APPLE
842 #
843
844
845 def prep_libcheck(topenv, libinfo):
846     if topenv['IS_OSX']:
847         #
848         # rationale: GTK-Quartz uses jhbuild and installs to ~/gtk/inst by default.
849         # All libraries needed should be built against this location
850         if topenv['GTKOSX']:
851             GTKROOT = os.path.expanduser ('~/gtk/inst')
852             libinfo.Append(CPPPATH= GTKROOT + "/include", LIBPATH= GTKROOT + "/lib")
853             libinfo.Append(CXXFLAGS="-I" + GTKROOT + "/include", LINKFLAGS="-L" + GTKROOT + "/lib")
854
855
856                 
857
858 prep_libcheck(env, env)
859
860
861 #
862 # these are part of the Ardour source tree because they are C++
863
864
865 libraries['vamp'] = LibraryInfo (LIBS='vampsdk',
866                                  LIBPATH='#libs/vamp-sdk',
867                                  CPPPATH='#libs/vamp-sdk')
868 libraries['vamphost'] = LibraryInfo (LIBS='vamphostsdk',
869                                  LIBPATH='#libs/vamp-sdk',
870                                  CPPPATH='#libs/vamp-sdk')
871
872 env['RUBBERBAND'] = False
873
874 conf = Configure (env)
875
876 if conf.CheckHeader ('fftw3.h'):
877     env['RUBBERBAND'] = True
878     libraries['rubberband'] = LibraryInfo (LIBS='rubberband',
879                                            LIBPATH='#libs/rubberband',
880                                            CPPPATH='#libs/rubberband',
881                                            CCFLAGS='-DUSE_RUBBERBAND')
882 else:
883     print ""
884     print "-------------------------------------------------------------------------"
885     print "You do not have the FFTW single-precision development package installed."
886     print "This prevents Ardour from using the Rubberband library for timestretching"
887     print "and pitchshifting. It will fall back on SoundTouch for timestretch, and "
888     print "pitchshifting will not be available."
889     print "-------------------------------------------------------------------------"
890     print ""
891
892 conf.Finish()
893
894 #
895 # Check for libusb
896
897 libraries['usb'] = LibraryInfo ()
898 prep_libcheck(env, libraries['usb'])
899
900 conf = Configure (libraries['usb'])
901 if conf.CheckLib ('usb', 'usb_interrupt_write'):
902     have_libusb = True
903 else:
904     have_libusb = False
905
906 # check for linux/input.h while we're at it for powermate
907 if conf.CheckHeader('linux/input.h'):
908     have_linux_input = True
909 else:
910     have_linux_input = False
911
912 libraries['usb'] = conf.Finish ()
913
914 #
915 # Check for wiimote dependencies
916
917 if env['WIIMOTE']:
918     wiimoteConf = env.Configure ( )
919     if not wiimoteConf.CheckHeader('cwiid.h'):
920         print 'WIIMOTE configured but you are missing libcwiid!'
921         sys.exit(1)
922     if not wiimoteConf.CheckHeader('bluetooth/bluetooth.h'):
923         print 'WIIMOTE configured but you are missing the libbluetooth headers which you need to compile wiimote support!'
924         sys.exit(1)
925     wiimoteConf.Finish()
926
927
928 #
929 # Check for FLAC
930
931 libraries['flac'] = LibraryInfo ()
932 prep_libcheck(env, libraries['flac'])
933 libraries['flac'].Append(CPPPATH="/usr/local/include", LIBPATH="/usr/local/lib")
934
935 #
936 # june 1st 2007: look for a function that is in FLAC 1.1.2 and not in later versions
937 #                since the version of libsndfile we have internally does not support
938 #                the new API that libFLAC has adopted
939 #
940
941 conf = Configure (libraries['flac'])
942 if conf.CheckLib ('FLAC', 'FLAC__seekable_stream_decoder_init', language='CXX'):
943     conf.env.Append(CCFLAGS='-DHAVE_FLAC')
944     use_flac = True
945 else:
946     use_flac = False
947     
948 libraries['flac'] = conf.Finish ()
949
950 # or if that fails...
951 #libraries['flac']    = LibraryInfo (LIBS='FLAC')
952
953 # boost (we don't link against boost, just use some header files)
954
955 libraries['boost'] = LibraryInfo ()
956 prep_libcheck(env, libraries['boost'])
957 libraries['boost'].Append(CPPPATH="/usr/local/include", LIBPATH="/usr/local/lib")
958 conf = Configure (libraries['boost'])
959 if conf.CheckHeader ('boost/shared_ptr.hpp', language='CXX') == False:
960         print "Boost header files do not appear to be installed. You also might be running a buggy version of scons. Try scons 0.97 if you can."
961         sys.exit (1)
962     
963 libraries['boost'] = conf.Finish ()
964
965 #
966 # Check for liblo
967
968 if env['LIBLO']:
969     libraries['lo'] = LibraryInfo ()
970     prep_libcheck(env, libraries['lo'])
971
972     conf = Configure (libraries['lo'])
973     if conf.CheckLib ('lo', 'lo_server_new') == False:
974         print "liblo does not appear to be installed."
975         sys.exit (1)
976     
977     libraries['lo'] = conf.Finish ()
978
979 #
980 # Check for dmalloc
981
982 libraries['dmalloc'] = LibraryInfo ()
983 prep_libcheck(env, libraries['dmalloc'])
984
985 #
986 # look for the threaded version
987 #
988
989 conf = Configure (libraries['dmalloc'])
990 if conf.CheckLib ('dmallocth', 'dmalloc_shutdown'):
991     have_libdmalloc = True
992 else:
993     have_libdmalloc = False
994
995 libraries['dmalloc'] = conf.Finish ()
996
997 #
998 # ensure FREEDESKTOP target is doable..
999 #
1000
1001 conf = env.Configure ()
1002 if env['FREEDESKTOP']:
1003         have_update_mime_database = conf.TryAction (Action ('update-mime-database -v'))
1004         if have_update_mime_database[0] != 1:
1005                 print "Warning. You have no update-mime-database command in your PATH. FREEDESKTOP is now disabled."
1006                 env['FREEDESKTOP'] = 0
1007         have_gtk_update_icon_cache = conf.TryAction (Action ('gtk-update-icon-cache -?'))
1008         if have_gtk_update_icon_cache[0] != 1:
1009                 print "Warning. You have no gtk-update-icon-cache command in your PATH. FREEDESKTOP is now disabled."
1010                 env['FREEDESKTOP'] = 0
1011         have_update_desktop_database = conf.TryAction (Action ('update-desktop-database -?'))
1012         if have_update_desktop_database[0] != 1:
1013                 print "Warning. You have no update-desktop-database command in your PATH. FREEDESKTOP is now disabled."
1014                 env['FREEDESKTOP'] = 0
1015 env = conf.Finish()
1016
1017 #
1018 # Audio/MIDI library (needed for MIDI, since audio is all handled via JACK)
1019 #
1020
1021 conf = Configure(env)
1022
1023 if conf.CheckCHeader('alsa/asoundlib.h'):
1024     libraries['sysmidi'] = LibraryInfo (LIBS='asound')
1025     env['SYSMIDI'] = 'ALSA Sequencer'
1026     subst_dict['%MIDITAG%'] = "seq"
1027     subst_dict['%MIDITYPE%'] = "alsa/sequencer"
1028 elif conf.CheckCHeader('/System/Library/Frameworks/CoreMIDI.framework/Headers/CoreMIDI.h'):
1029     # this line is needed because scons can't handle -framework in ParseConfig() yet.
1030     if env['GTKOSX']:
1031         # We need Carbon as well as the rest
1032         libraries['sysmidi'] = LibraryInfo (
1033                 LINKFLAGS = ' -framework CoreMIDI -framework CoreFoundation -framework CoreAudio -framework CoreServices -framework AudioUnit -framework AudioToolbox -framework Carbon -bind_at_load' )
1034     else:
1035         libraries['sysmidi'] = LibraryInfo (
1036                 LINKFLAGS = ' -framework CoreMIDI -framework CoreFoundation -framework CoreAudio -framework CoreServices -framework AudioUnit -framework AudioToolbox -bind_at_load' )
1037     env['SYSMIDI'] = 'CoreMIDI'
1038     subst_dict['%MIDITAG%'] = "ardour"
1039     subst_dict['%MIDITYPE%'] = "coremidi"
1040 else:
1041     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."
1042     sys.exit (1)
1043
1044 env = conf.Finish()
1045
1046 if env['GTKOSX']:
1047     clearlooks_version = 'libs/clearlooks-newer'
1048 else:
1049     clearlooks_version = 'libs/clearlooks-older'
1050
1051 if env['SYSLIBS']:
1052
1053     syslibdeps = \
1054     {
1055         'sigc++-2.0'           : '2.0',
1056         'gtkmm-2.4'            : '2.8',
1057         'libgnomecanvasmm-2.6' : '2.12.0'
1058     }
1059
1060     conf = Configure(env, custom_tests = { 'CheckPKGConfig' : CheckPKGConfig,
1061                     'CheckPKGVersion' : CheckPKGVersion })
1062
1063     for pkg, version in syslibdeps.iteritems():
1064         if not conf.CheckPKGVersion( pkg, version ):
1065             print '%s >= %s not found.' %(pkg, version)
1066             DependenciesRequiredMessage()
1067             Exit(1)
1068     
1069     env = conf.Finish()
1070     
1071     libraries['sigc2'] = LibraryInfo()
1072     libraries['sigc2'].ParseConfig('pkg-config --cflags --libs sigc++-2.0')
1073     libraries['glibmm2'] = LibraryInfo()
1074     libraries['glibmm2'].ParseConfig('pkg-config --cflags --libs glibmm-2.4')
1075     libraries['cairomm'] = LibraryInfo()
1076     libraries['cairomm'].ParseConfig('pkg-config --cflags --libs cairomm-1.0')
1077     libraries['gdkmm2'] = LibraryInfo()
1078     libraries['gdkmm2'].ParseConfig ('pkg-config --cflags --libs gdkmm-2.4')
1079     libraries['gtkmm2'] = LibraryInfo()
1080     libraries['gtkmm2'].ParseConfig ('pkg-config --cflags --libs gtkmm-2.4')
1081     libraries['atkmm'] = LibraryInfo()
1082     libraries['atkmm'].ParseConfig ('pkg-config --cflags --libs atkmm-1.6')
1083     libraries['pangomm'] = LibraryInfo()
1084     libraries['pangomm'].ParseConfig ('pkg-config --cflags --libs pangomm-1.4')
1085     libraries['libgnomecanvasmm'] = LibraryInfo()
1086     libraries['libgnomecanvasmm'].ParseConfig ('pkg-config --cflags --libs libgnomecanvasmm-2.6')
1087
1088 #
1089 # cannot use system one for the time being
1090 #
1091     
1092     libraries['sndfile-ardour'] = LibraryInfo(LIBS='libsndfile-ardour',
1093                                     LIBPATH='#libs/libsndfile',
1094                                     CPPPATH=['#libs/libsndfile/src'])
1095
1096 #    libraries['libglademm'] = LibraryInfo()
1097 #    libraries['libglademm'].ParseConfig ('pkg-config --cflags --libs libglademm-2.4')
1098
1099 #    libraries['flowcanvas'] = LibraryInfo(LIBS='flowcanvas', LIBPATH='#/libs/flowcanvas', CPPPATH='#libs/flowcanvas')
1100     libraries['soundtouch'] = LibraryInfo()
1101     libraries['soundtouch'].ParseConfig ('pkg-config --cflags --libs soundtouch-1.0')
1102     # Comment the previous line and uncomment this for old versions of Debian:
1103     #libraries['soundtouch'].ParseConfig ('pkg-config --cflags --libs libSoundTouch')
1104
1105     libraries['appleutility'] = LibraryInfo(LIBS='libappleutility',
1106                                             LIBPATH='#libs/appleutility',
1107                                             CPPPATH='#libs/appleutility')
1108     
1109     coredirs = [
1110         'templates',
1111         'manual'
1112     ]
1113     
1114     subdirs = [
1115         'libs/libsndfile',
1116         'libs/pbd',
1117         'libs/midi++2',
1118         'libs/ardour',
1119         'libs/vamp-sdk',
1120         'libs/vamp-plugins/',
1121     # these are unconditionally included but have
1122     # tests internally to avoid compilation etc
1123     # if VST is not set
1124         'libs/fst',
1125         'vst',
1126     # this is unconditionally included but has
1127     # tests internally to avoid compilation etc
1128     # if COREAUDIO is not set
1129         'libs/appleutility'
1130         ]
1131     
1132     gtk_subdirs = [
1133 #        'libs/flowcanvas',
1134         'libs/gtkmm2ext',
1135         'gtk2_ardour',
1136         clearlooks_version
1137         ]
1138
1139 else:
1140     libraries['sigc2'] = LibraryInfo(LIBS='sigc++2',
1141                                     LIBPATH='#libs/sigc++2',
1142                                     CPPPATH='#libs/sigc++2')
1143     libraries['glibmm2'] = LibraryInfo(LIBS='glibmm2',
1144                                     LIBPATH='#libs/glibmm2',
1145                                     CPPPATH='#libs/glibmm2')
1146     libraries['pangomm'] = LibraryInfo(LIBS='pangomm',
1147                                     LIBPATH='#libs/gtkmm2/pango',
1148                                     CPPPATH='#libs/gtkmm2/pango')
1149     libraries['cairomm'] = LibraryInfo(LIBS='cairomm',
1150                                     LIBPATH='#libs/cairomm',
1151                                     CPPPATH='#libs/cairomm')
1152     libraries['atkmm'] = LibraryInfo(LIBS='atkmm',
1153                                      LIBPATH='#libs/gtkmm2/atk',
1154                                      CPPPATH='#libs/gtkmm2/atk')
1155     libraries['gdkmm2'] = LibraryInfo(LIBS='gdkmm2',
1156                                       LIBPATH='#libs/gtkmm2/gdk',
1157                                       CPPPATH='#libs/gtkmm2/gdk')
1158     libraries['gtkmm2'] = LibraryInfo(LIBS='gtkmm2',
1159                                      LIBPATH="#libs/gtkmm2/gtk",
1160                                      CPPPATH='#libs/gtkmm2/gtk/')
1161     libraries['libgnomecanvasmm'] = LibraryInfo(LIBS='libgnomecanvasmm',
1162                                                 LIBPATH='#libs/libgnomecanvasmm',
1163                                                 CPPPATH='#libs/libgnomecanvasmm')
1164     
1165     libraries['soundtouch'] = LibraryInfo(LIBS='soundtouch',
1166                                           LIBPATH='#libs/soundtouch',
1167                                           CPPPATH=['#libs', '#libs/soundtouch'])
1168     libraries['sndfile-ardour'] = LibraryInfo(LIBS='libsndfile-ardour',
1169                                     LIBPATH='#libs/libsndfile',
1170                                     CPPPATH=['#libs/libsndfile', '#libs/libsndfile/src'])
1171 #    libraries['libglademm'] = LibraryInfo(LIBS='libglademm',
1172 #                                          LIBPATH='#libs/libglademm',
1173 #                                          CPPPATH='#libs/libglademm')
1174     libraries['appleutility'] = LibraryInfo(LIBS='libappleutility',
1175                                             LIBPATH='#libs/appleutility',
1176                                             CPPPATH='#libs/appleutility')
1177
1178     coredirs = [
1179         'templates',
1180         'manual'
1181     ]
1182     
1183     subdirs = [
1184         'libs/sigc++2',
1185         'libs/libsndfile',
1186         'libs/pbd',
1187         'libs/midi++2',
1188         'libs/ardour',
1189         'libs/vamp-sdk',
1190         'libs/vamp-plugins/',
1191     # these are unconditionally included but have
1192     # tests internally to avoid compilation etc
1193     # if VST is not set
1194         'libs/fst',
1195         'vst',
1196     # this is unconditionally included but has
1197     # tests internally to avoid compilation etc
1198     # if COREAUDIO is not set
1199         'libs/appleutility'
1200         ]
1201     
1202     gtk_subdirs = [
1203         'libs/glibmm2',
1204         'libs/gtkmm2/pango',
1205         'libs/gtkmm2/atk',
1206         'libs/gtkmm2/gdk',
1207         'libs/gtkmm2/gtk',
1208         'libs/libgnomecanvasmm',
1209         'libs/gtkmm2ext',
1210         'gtk2_ardour',
1211         clearlooks_version
1212         ]
1213
1214 #
1215 # * always build the LGPL control protocol lib, since we link against it from libardour
1216 # * ditto for generic MIDI
1217 # * tranzport & wiimote check whether they should build internally, but we need them here
1218 #   so that they are included in the tarball
1219 #
1220
1221 surface_subdirs = [ 'libs/surfaces/control_protocol',
1222                     'libs/surfaces/generic_midi',
1223                     'libs/surfaces/tranzport',
1224                     'libs/surfaces/mackie',
1225                     'libs/surfaces/powermate',
1226                     'libs/surfaces/wiimote'
1227                     ]
1228
1229 if env['SURFACES']:
1230     if have_libusb:
1231         env['TRANZPORT'] = 1
1232     else:
1233         env['TRANZPORT'] = 0
1234         print 'Disabled building Tranzport code because libusb could not be found'
1235
1236     if have_linux_input:
1237         env['POWERMATE'] = 1
1238     else:
1239         env['POWERMATE'] = 0
1240         print 'Disabled building Powermate code because linux/input.h could not be found'
1241
1242     if os.access ('libs/surfaces/sony9pin', os.F_OK):
1243         surface_subdirs += [ 'libs/surfaces/sony9pin' ]
1244 else:
1245     env['POWERMATE'] = 0
1246     env['TRANZPORT'] = 0
1247
1248 #
1249 # timestretch libraries
1250 #
1251
1252 timefx_subdirs = ['libs/soundtouch']
1253 if env['RUBBERBAND']:
1254     timefx_subdirs += ['libs/rubberband']
1255
1256 opts.Save('scache.conf', env)
1257 Help(opts.GenerateHelpText(env))
1258
1259 final_prefix = '$PREFIX'
1260
1261 if env['DESTDIR'] :
1262     install_prefix = '$DESTDIR/$PREFIX'
1263 else:
1264     install_prefix = env['PREFIX']
1265
1266 subst_dict['%INSTALL_PREFIX%'] = install_prefix;
1267 subst_dict['%FINAL_PREFIX%'] = final_prefix;
1268 subst_dict['%PREFIX%'] = final_prefix;
1269
1270 if env['PREFIX'] == '/usr':
1271     final_config_prefix = '/etc'
1272 else:
1273     final_config_prefix = env['PREFIX'] + '/etc'
1274
1275 config_prefix = '$DESTDIR' + final_config_prefix
1276
1277 #
1278 # everybody needs this
1279 #
1280
1281 env.Merge ([ libraries['core'] ])
1282
1283
1284 #
1285 # i18n support
1286 #
1287
1288 conf = Configure (env)
1289 if env['NLS']:
1290     nls_error = 'This system is not configured for internationalized applications.  An english-only version will be built:'
1291     print 'Checking for internationalization support ...'
1292     have_gettext = conf.TryAction(Action('xgettext --version'))
1293     if have_gettext[0] != 1:
1294         nls_error += ' No xgettext command.'
1295         env['NLS'] = 0
1296     else:
1297         print "Found xgettext"
1298     
1299     have_msgmerge = conf.TryAction(Action('msgmerge --version'))
1300     if have_msgmerge[0] != 1:
1301         nls_error += ' No msgmerge command.'
1302         env['NLS'] = 0
1303     else:
1304         print "Found msgmerge"
1305     
1306     if not conf.CheckCHeader('libintl.h'):
1307         nls_error += ' No libintl.h.'
1308         env['NLS'] = 0
1309         
1310     if env['NLS'] == 0:
1311         print nls_error
1312     else:
1313         print "International version will be built."
1314 env = conf.Finish()
1315
1316 if env['NLS'] == 1:
1317     env.Append(CCFLAGS="-DENABLE_NLS")
1318
1319 Export('env install_prefix final_prefix config_prefix final_config_prefix libraries i18n ardour_version subst_dict use_flac')
1320
1321 #
1322 # the configuration file may be system dependent
1323 #
1324
1325 conf = env.Configure ()
1326
1327 if conf.CheckCHeader('/System/Library/Frameworks/CoreAudio.framework/Versions/A/Headers/CoreAudio.h'):
1328     subst_dict['%JACK_INPUT%'] = "coreaudio:Built-in Audio:in"
1329     subst_dict['%JACK_OUTPUT%'] = "coreaudio:Built-in Audio:out"
1330 else:
1331     subst_dict['%JACK_INPUT%'] = "alsa_pcm:playback_"
1332     subst_dict['%JACK_OUTPUT%'] = "alsa_pcm:capture_"
1333
1334 # posix_memalign available
1335 if not conf.CheckFunc('posix_memalign'):
1336     print 'Did not find posix_memalign(), using malloc'
1337     env.Append(CCFLAGS='-DNO_POSIX_MEMALIGN')
1338
1339
1340 env = conf.Finish()
1341
1342 # Which GTK tooltips API
1343
1344 gtktestenv = env.Clone ()
1345 gtktestenv.Merge ([
1346         libraries['gtk2']
1347         ])
1348
1349 conf = gtktestenv.Configure ()
1350
1351 if conf.CheckFunc('gtk_widget_set_tooltip_text'):
1352     env.Append (CXXFLAGS='-DGTK_NEW_TOOLTIP_API')
1353
1354 conf.Finish ()
1355
1356
1357 # generate the per-user and system rc files from the same source
1358
1359 sysrcbuild = env.SubstInFile ('ardour_system.rc','ardour.rc.in', SUBST_DICT = subst_dict)
1360
1361 # add to the substitution dictionary
1362
1363 subst_dict['%VERSION%'] = ardour_version[0:3]
1364 subst_dict['%EXTRA_VERSION%'] = ardour_version[3:]
1365 subst_dict['%REVISION_STRING%'] = ''
1366 if os.path.exists('.svn'):
1367     subst_dict['%REVISION_STRING%'] = '.' + fetch_svn_revision ('.') + 'svn'
1368
1369 # specbuild = env.SubstInFile ('ardour.spec','ardour.spec.in', SUBST_DICT = subst_dict)
1370
1371 the_revision = env.Command ('frobnicatory_decoy', [], create_stored_revision)
1372 remove_ardour = env.Command ('frobnicatory_decoy2', [],
1373                              [ Delete ('$PREFIX/etc/ardour2'),
1374                                Delete ('$PREFIX/lib/ardour2'),
1375                                Delete ('$PREFIX/bin/ardour2'),
1376                                Delete ('$PREFIX/share/ardour2')])
1377
1378 env.Alias('revision', the_revision)
1379 env.Alias('install', env.Install(os.path.join(config_prefix, 'ardour2'), 'ardour_system.rc'))
1380 env.Alias('uninstall', remove_ardour)
1381
1382 Default (sysrcbuild)
1383
1384 # source tarball
1385
1386 Precious (env['DISTTREE'])
1387
1388 env.Distribute (env['DISTTREE'],
1389                [ 'SConstruct', 
1390                   'COPYING', 'PACKAGER_README', 'README',
1391                   'ardour.rc.in',
1392                   'tools/config.guess',
1393                   'icons/icon/ardour_icon_mac_mask.png',
1394                   'icons/icon/ardour_icon_mac.png',
1395                   'icons/icon/ardour_icon_tango_16px_blue.png',
1396                   'icons/icon/ardour_icon_tango_16px_red.png',
1397                   'icons/icon/ardour_icon_tango_22px_blue.png',
1398                   'icons/icon/ardour_icon_tango_22px_red.png',
1399                   'icons/icon/ardour_icon_tango_32px_blue.png',
1400                   'icons/icon/ardour_icon_tango_32px_red.png',
1401                   'icons/icon/ardour_icon_tango_48px_blue.png',
1402                   'icons/icon/ardour_icon_tango_48px_red.png'
1403                   ] +
1404                 glob.glob ('ardour.1*') +
1405                 glob.glob ('libs/clearlooks-newer/*.c') +
1406                 glob.glob ('libs/clearlooks-newer/*.h') +
1407                 glob.glob ('libs/clearlooks-newer/SConscript')
1408                 )
1409
1410 srcdist = env.Tarball(env['TARBALL'], [ env['DISTTREE'], the_revision ])
1411 env.Alias ('srctar', srcdist)
1412
1413 #
1414 # don't leave the distree around
1415 #
1416
1417 env.AddPreAction (env['DISTTREE'], Action ('rm -rf ' + str (File (env['DISTTREE']))))
1418 env.AddPostAction (srcdist, Action ('rm -rf ' + str (File (env['DISTTREE']))))
1419
1420 #
1421 # Update revision info before going into subdirs
1422 #
1423
1424 create_stored_revision()
1425
1426 #
1427 # the subdirs
1428 #
1429
1430 for subdir in coredirs:
1431     SConscript (subdir + '/SConscript')
1432
1433 for sublistdir in [ subdirs, timefx_subdirs, gtk_subdirs, surface_subdirs ]:
1434     for subdir in sublistdir:
1435         SConscript (subdir + '/SConscript')
1436
1437 # cleanup
1438 env.Clean ('scrub', [ 'scache.conf', '.sconf_temp', '.sconsign.dblite', 'config.log'])
1439