summaryrefslogtreecommitdiff
path: root/acprep
diff options
context:
space:
mode:
Diffstat (limited to 'acprep')
-rwxr-xr-xacprep1696
1 files changed, 1567 insertions, 129 deletions
diff --git a/acprep b/acprep
index 07a457f9..20be2db4 100755
--- a/acprep
+++ b/acprep
@@ -1,132 +1,1570 @@
-#!/bin/sh
+#!/usr/bin/env python
# acprep, version 3.0
#
-# This script configures my ledger source tree on my Mac OS/X machine.
-# This is not necessary, however, since I keep all the files necessary
-# for building checked in to the source tree. Users can just type
-# './configure && make'. This script simply sets up the compiler and
-# linker flags for all the various build permutations I use for testing
-# and profiling.
-
-cmd=$(which glibtoolize 2>&1)
-if [ -x "$cmd" ]; then
- export LIBTOOLIZE="$cmd"
-fi
-autoreconf --force --install
-
-
-INCDIRS="-I/sw/include"
-INCDIRS="$INCDIRS -I/usr/local/include"
-INCDIRS="$INCDIRS -I/usr/local/include/boost-1_34"
-
-LIBDIRS="-L/sw/lib"
-LIBDIRS="$LIBDIRS -L/usr/local/lib"
-
-PYTHON_HOME="/Library/Frameworks/Python.framework/Versions/2.5"
-
-
-SYSTEM=`uname -s`
-
-if [ $SYSTEM = Linux ]; then
- CXXFLAGS="-pthread"
-elif [ $SYSTEM = Solaris ]; then
- CXXFLAGS="-pthreads"
-elif [ $SYSTEM = Darwin ]; then
- #CXXFLAGS="-arch i386 -arch ppc -isysroot /Developer/SDKs/MacOSX10.4u.sdk"
- CXXFLAGS="$CXXFLAGS -Wno-long-double"
- #LIBDIRS="$LIBDIRS -arch i386 -arch ppc -isysroot /Developer/SDKs/MacOSX10.4u.sdk"
-else
- CXXFLAGS=""
-fi
-
-# Building the command-line tool as a shared library is a luxury,
-# since there are no clients except a GUI tool which might use it (and
-# that is built again anyway by Xcode).
-SWITCHES=""
-CPPFLAGS="$INCDIRS"
-CXXFLAGS="-pipe"
-LDFLAGS="$LIBDIRS"
-LOCAL=false
-
-# Warning flags
-CXXFLAGS="$CXXFLAGS -Wall -ansi"
-#CXXFLAGS="$CXXFLAGS -Wextra"
-#CXXFLAGS="$CXXFLAGS -Weffc++"
-#CXXFLAGS="$CXXFLAGS -Wcast-align"
-#CXXFLAGS="$CXXFLAGS -Wcast-qual"
-#CXXFLAGS="$CXXFLAGS -Wconversion"
-#CXXFLAGS="$CXXFLAGS -Wfloat-equal"
-#CXXFLAGS="$CXXFLAGS -Wmissing-field-initializers"
-#CXXFLAGS="$CXXFLAGS -Wno-endif-labels"
-#CXXFLAGS="$CXXFLAGS -Wold-style-cast"
-#CXXFLAGS="$CXXFLAGS -Woverloaded-virtual"
-#CXXFLAGS="$CXXFLAGS -Wshorten-64-to-32"
-#CXXFLAGS="$CXXFLAGS -Wsign-compare"
-#CXXFLAGS="$CXXFLAGS -Wsign-promo"
-#CXXFLAGS="$CXXFLAGS -Wstrict-null-sentinel"
-#CXXFLAGS="$CXXFLAGS -Wwrite-strings"
-
-
-while [ -n "$1" ]; do
- case "$1" in
- --devel)
- SWITCHES="$SWITCHES --disable-shared --enable-pch"
- ;;
-
- --debug)
- SWITCHES="$SWITCHES --enable-debug"
- #CPPFLAGS="$CPPFLAGS -D_GLIBCXX_DEBUG=1"
- CXXFLAGS="$CXXFLAGS -g" ;;
-
- --boost)
- shift 1
- SWITCHES="$SWITCHES --with-boost-suffix=$1"
- ;;
-
- --gcov)
- CXXFLAGS="$CXXFLAGS -fprofile-arcs -ftest-coverage" ;;
-
- --gprof)
- CXXFLAGS="$CXXFLAGS -g -pg" ;;
-
- --python)
- if [ -d "$PYTHON_HOME" ]; then
- SWITCHES="$SWITCHES --enable-python"
- CPPFLAGS="$CPPFLAGS -I$PYTHON_HOME/include/python2.5"
- LDFLAGS="$LDFLAGS -L$PYTHON_HOME/lib/python2.5/config"
- fi ;;
-
- --pic)
- CXXFLAGS="$CXXFLAGS -fPIC" ;;
-
- --opt)
- CXXFLAGS="$CXXFLAGS -fomit-frame-pointer -O3" ;;
-
- --local)
- LOCAL=true ;;
-
- *)
- break ;;
- esac
- shift 1
-done
-
-
-HERE="$PWD"
-
-if [ "$LOCAL" = "false" -a -d "$HOME/Products" ]; then
- version=""
- if [ -x pending/version ]; then
- version="-$(pending/version)"
- fi
- projdir="$HOME/Products/$(basename $HERE)$version"
- if [ ! -d "$projdir" ]; then
- mkdir -p "$projdir"
- fi
- cd "$projdir" || (echo "Cannot change to $projdir"; exit 1)
-fi
-
-"$HERE/configure" --srcdir="$HERE" \
- CPPFLAGS="$CPPFLAGS" CXXFLAGS="$CXXFLAGS $local_cxxflags" \
- LDFLAGS="$LDFLAGS" LIBS="$LIBS" $SWITCHES "$@"
+# This script configures my ledger source tree on my Mac OS/X machine. This
+# is not necessary, however, since I keep all the files necessary for building
+# checked in to the source tree. Users can just type './configure && make'.
+# This script simply sets up the compiler and linker flags for all the various
+# build permutations I use for testing and profiling.
+
+import inspect
+import logging
+import logging.handlers
+import optparse
+import os
+import re
+import shutil
+import string
+import sys
+import time
+import tempfile
+import datetime
+
+try:
+ import hashlib
+except:
+ import md5
+
+from os.path import *
+from stat import *
+from subprocess import Popen, PIPE, call
+
+LEVELS = {'DEBUG': logging.DEBUG,
+ 'INFO': logging.INFO,
+ 'WARNING': logging.WARNING,
+ 'ERROR': logging.ERROR,
+ 'CRITICAL': logging.CRITICAL}
+
+
+class CommandLineApp(object):
+ "Base class for building command line applications."
+
+ force_exit = True # If true, always ends run() with sys.exit()
+ log_handler = None
+ darwin_gcc = False
+
+ options = {
+ 'debug': False,
+ 'verbose': False,
+ 'logfile': False,
+ 'loglevel': False
+ }
+
+ def __init__(self):
+ "Initialize CommandLineApp."
+ # Create the logger
+ self.log = logging.getLogger(os.path.basename(sys.argv[0]))
+ ch = logging.StreamHandler()
+ formatter = logging.Formatter("%(name)s: %(levelname)s: %(message)s")
+ ch.setFormatter(formatter)
+ self.log.addHandler(ch)
+ self.log_handler = ch
+
+ # Setup the options parser
+ usage = 'usage: %prog [OPTIONS...] [ARGS...]'
+ op = self.option_parser = optparse.OptionParser(usage = usage,
+ conflict_handler = 'resolve')
+ op.add_option('', '--debug',
+ action='store_true', dest='debug',
+ default=False, help='show debug messages and pass exceptions')
+ op.add_option('-v', '--verbose',
+ action='store_true', dest='verbose',
+ default=False, help='show informational messages')
+ op.add_option('-q', '--quiet',
+ action='store_true', dest='quiet',
+ default=False, help='do not show log messages on console')
+ op.add_option('', '--log', metavar='FILE',
+ type='string', action='store', dest='logfile',
+ default=False, help='append logging data to FILE')
+ op.add_option('', '--loglevel', metavar='LEVEL',
+ type='string', action='store', dest='loglevel',
+ default=False, help='set log level: DEBUG, INFO, WARNING, ERROR, CRITICAL')
+ return
+
+ def main(self, *args):
+ """Main body of your application.
+
+ This is the main portion of the app, and is run after all of the
+ arguments are processed. Override this method to implment the primary
+ processing section of your application."""
+ pass
+
+ def handleInterrupt(self):
+ """Called when the program is interrupted via Control-C or SIGINT.
+ Returns exit code."""
+ self.log.error('Canceled by user.')
+ return 1
+
+ def handleMainException(self):
+ "Invoked when there is an error in the main() method."
+ if not self.options.debug:
+ self.log.exception('Caught exception')
+ return 1
+
+ ## INTERNALS (Subclasses should not need to override these methods)
+
+ def run(self):
+ """Entry point.
+
+ Process options and execute callback functions as needed. This method
+ should not need to be overridden, if the main() method is defined."""
+ # Process the options supported and given
+ self.options, main_args = self.option_parser.parse_args()
+
+ if self.options.logfile:
+ fh = logging.handlers.RotatingFileHandler(self.options.logfile,
+ maxBytes = (1024 * 1024),
+ backupCount = 5)
+ formatter = logging.Formatter("%(asctime)s - %(levelname)s: %(message)s")
+ fh.setFormatter(formatter)
+ self.log.addHandler(fh)
+
+ if self.options.quiet:
+ self.log.removeHandler(self.log_handler)
+ ch = logging.handlers.SysLogHandler()
+ formatter = logging.Formatter("%(name)s: %(levelname)s: %(message)s")
+ ch.setFormatter(formatter)
+ self.log.addHandler(ch)
+ self.log_handler = ch
+
+ if self.options.loglevel:
+ self.log.setLevel(LEVELS[self.options.loglevel])
+ elif self.options.debug:
+ self.log.setLevel(logging.DEBUG)
+ elif self.options.verbose:
+ self.log.setLevel(logging.INFO)
+
+ exit_code = 0
+ try:
+ # We could just call main() and catch a TypeError, but that would
+ # not let us differentiate between application errors and a case
+ # where the user has not passed us enough arguments. So, we check
+ # the argument count ourself.
+ argspec = inspect.getargspec(self.main)
+ expected_arg_count = len(argspec[0]) - 1
+
+ if len(main_args) >= expected_arg_count:
+ exit_code = self.main(*main_args)
+ else:
+ self.log.debug('Incorrect argument count (expected %d, got %d)' %
+ (expected_arg_count, len(main_args)))
+ self.option_parser.print_help()
+ exit_code = 1
+
+ except KeyboardInterrupt:
+ exit_code = self.handleInterrupt()
+
+ except SystemExit, msg:
+ exit_code = msg.args[0]
+
+ except Exception:
+ exit_code = self.handleMainException()
+ if self.options.debug:
+ raise
+
+ if self.force_exit:
+ sys.exit(exit_code)
+ return exit_code
+
+
+class PrepareBuild(CommandLineApp):
+ #########################################################################
+ # Initialization routines #
+ #########################################################################
+
+ def initialize(self):
+ self.log.debug('Initializing all state variables')
+
+ self.should_clean = False
+ self.configured = False
+ self.current_ver = None
+ #self.current_flavor = 'default'
+ self.current_flavor = 'debug'
+ self.products_dir = None
+ self.build_dir = self.source_dir
+ self.configure_args = ['--with-included-gettext', '--enable-python']
+ self.sys_include_dirs = []
+ self.sys_library_dirs = []
+
+ self.CPPFLAGS = []
+ self.CFLAGS = []
+ self.CXXFLAGS = []
+ self.LDFLAGS = []
+
+ self.envvars = {
+ 'PYTHON_HOME': '/usr',
+ 'PYTHON_VERSION': '2.6',
+ 'BOOST_SUFFIX': None,
+ 'BOOST_HOME': '/usr',
+ 'LEDGER_PRODUCTS': None,
+ 'CC': 'gcc',
+ 'CPPFLAGS': '',
+ 'CFLAGS': '',
+ 'CXX': 'g++',
+ 'CXXFLAGS': '',
+ 'LD': 'g++',
+ 'LDFLAGS': ''
+ }
+
+ for varname in self.envvars.keys():
+ if os.environ.has_key(varname):
+ self.envvars[varname] = os.environ[varname]
+
+ if varname.endswith('FLAGS'):
+ self.__dict__[varname] = string.split(os.environ[varname])
+ self.envvars[varname] = ''
+
+ # If ~/Products/ or build/ exists, use them instead of the source tree
+ # for building
+ products = self.default_products_directory()
+ if (exists(products) and isdir(products)) or \
+ (exists('build') and isdir('build')):
+ self.build_dir = None
+
+ def __init__(self):
+ CommandLineApp.__init__(self)
+ self.log.setLevel(logging.INFO)
+
+ op = self.option_parser
+ op.add_option('-j', '--jobs', metavar='N',
+ type='int', action='store', dest='jobs',
+ default=1, help='Allow N make jobs at once')
+ op.add_option('', '--boost', metavar='SUFFIX',
+ action="callback", type="string",
+ callback=self.option_boost,
+ help='Set Boost library suffix (ex: "--boost=-mt")')
+ op.add_option('', '--force', action="callback",
+ callback=self.option_force,
+ help="Perform every action, without checking")
+ op.add_option('', '--glibcxx-debug', action='store_true',
+ dest='use_glibcxx_debug', default=False,
+ help='Define _GLIBCXX_DEBUG=1 during compilation')
+ op.add_option('', '--help', action="callback",
+ callback=self.option_help,
+ help='Show this help text')
+ op.add_option('', '--local', action="callback",
+ callback=self.option_local,
+ help='Build directly within the source tree (default)')
+ op.add_option('', '--no-pch', action="callback",
+ callback=self.option_no_pch,
+ help='Do not use pre-compiled headers')
+ op.add_option('', '--no-patch', action='store_true', dest='no_patch',
+ default=False,
+ help='Do not patch the Makefile for prettier output')
+ op.add_option('', '--no-git', action='store_true', dest='no_git',
+ default=False,
+ help='Do not call out to Git; useful for offline builds')
+ op.add_option('', '--output', metavar='DIR', action="callback",
+ callback=self.option_output,
+ help='Build in the specified directory')
+ op.add_option('', '--pch', action="callback",
+ callback=self.option_pch,
+ help='Enable use of pre-compiled headers')
+ op.add_option('', '--pic', action="callback",
+ callback=self.option_pic,
+ help='Compile with explicit PIC support')
+ op.add_option('', '--products', metavar='DIR', action="callback",
+ callback=self.option_products,
+ help='Collect all build products in this directory')
+ op.add_option('', '--trees', action="callback",
+ callback=self.option_trees,
+ help='Use separate build trees for each flavor')
+ op.add_option('', '--release', action="callback",
+ callback=self.option_release,
+ help='Setup for doing a faster, once-only build')
+ op.add_option('', '--warn', action="callback",
+ callback=self.option_warn,
+ help='Enable full warning flags')
+
+ self.force = False
+ self.no_pch = False
+ self.source_dir = os.getcwd()
+
+ self.initialize()
+
+ def main(self, *args):
+ if args and args[0] in ['default', 'debug', 'opt', 'gcov', 'gprof']:
+ self.current_flavor = args[0]
+ args = args[1:]
+
+ if args:
+ cmd = args[0]
+ if not PrepareBuild.__dict__.has_key('phase_' + cmd):
+ cmd = 'config'
+ else:
+ args = args[1:]
+ else:
+ cmd = 'config'
+
+ self.log.info('Invoking primary phase: ' + cmd)
+ PrepareBuild.__dict__['phase_' + cmd](self, *args)
+
+ #########################################################################
+ # General utility code #
+ #########################################################################
+
+ def execute(self, *args):
+ try:
+ self.log.debug('Executing command: ' + string.join(args, ' '))
+
+ retcode = call(args, shell=False)
+ if retcode < 0:
+ self.log.error("Child was terminated by signal", -retcode)
+ sys.exit(1)
+ elif retcode != 0:
+ self.log.error("Execution failed: " + string.join(args, ' '))
+ sys.exit(1)
+ except OSError, e:
+ self.log.error("Execution failed:", e)
+ sys.exit(1)
+
+ def get_stdout(self, *args):
+ try:
+ self.log.debug('Executing command: ' + string.join(args, ' '))
+
+ proc = Popen(args, shell=False, stdout=PIPE)
+ stdout = proc.stdout.read()
+ retcode = proc.wait()
+ if retcode < 0:
+ self.log.error("Child was terminated by signal",
+ -retcode)
+ sys.exit(1)
+ elif retcode != 0:
+ self.log.error("Execution failed: " + string.join(args, ' '))
+ sys.exit(1)
+ return stdout[:-1]
+ except OSError, e:
+ self.log.error("Execution failed:", e)
+ sys.exit(1)
+
+ def isnewer(self, file1, file2):
+ "Check if file1 is newer than file2."
+ if not exists(file2):
+ return True
+ return os.stat(file1)[ST_MTIME] > os.stat(file2)[ST_MTIME]
+
+ #########################################################################
+ # Determine information about the surroundings #
+ #########################################################################
+
+ def default_products_directory(self):
+ if self.envvars['LEDGER_PRODUCTS']:
+ return self.envvars['LEDGER_PRODUCTS']
+ else:
+ return join(os.environ['HOME'], "Products")
+
+ def products_directory(self):
+ if not self.products_dir:
+ products = self.default_products_directory()
+
+ if not exists(products) or not isdir(products):
+ products = join(self.source_dir, 'build')
+
+ products = join(products, basename(self.source_dir))
+
+ self.products_dir = products
+
+ return self.products_dir
+
+ def build_directory(self):
+ if not self.build_dir:
+ self.build_dir = join(self.products_directory(),
+ self.current_flavor)
+ return self.build_dir
+
+ def ensure(self, dirname):
+ if not exists(dirname):
+ self.log.info('Making directory: ' + dirname)
+ os.makedirs(dirname)
+ elif not isdir(dirname):
+ self.log.error('Directory is not a directory: ' + dirname)
+ sys.exit(1)
+ return dirname
+
+ def git_working_tree(self):
+ return exists('.git') and isdir('.git') and not self.options.no_git
+
+ def current_version(self):
+ if not self.current_ver:
+ if self.git_working_tree():
+ #date = self.get_stdout('git', 'log', '--format=%ci', '-1', 'HEAD')
+ #date = re.sub(" [-+][0-9][0-9][0-9][0-9]$", "", date)
+ #when = datetime.datetime.strptime(date, "%Y-%m-%d %H:%M:%S")
+ #self.current_ver = when.strftime("%Y%m%d_%H%M%S")
+ #commit = self.get_stdout('git', 'log', '--format=%h', 'HEAD^..HEAD')
+ #self.current_ver += "_" + commit
+ tag = self.get_stdout('git', 'describe', '--all', '--long')
+ self.current_ver = re.sub('heads/', '', tag)
+ else:
+ self.current_ver = "no-git"
+ return self.current_ver
+
+ def need_to_prepare_autotools(self):
+ if self.force:
+ return 'because it was forced'
+ elif self.isnewer('acprep', 'configure'):
+ self.should_clean = True
+ return 'because acprep is newer than configure'
+ elif self.isnewer('acprep', 'Makefile.in'):
+ self.should_clean = True
+ return 'because acprep is newer than Makefile.in'
+ elif self.isnewer('configure.ac', 'configure'):
+ return 'because confgure.ac is newer than configure'
+ elif self.isnewer('Makefile.am', 'Makefile.in'):
+ return 'because Makefile.am is newer than Makefile.in'
+ return False
+
+ def phase_products(self, *args):
+ self.log.info('Executing phase: products')
+ print self.products_directory()
+
+ def phase_info(self, *args):
+ self.log.info('Executing phase: info')
+
+ environ, conf_args = self.configure_environment()
+
+ self.log.info("Current version => " + self.current_version())
+ self.log.info("Current flavor => " + self.current_flavor)
+ self.log.info("Source directory => " + self.source_dir)
+ self.log.info("Need to run autogen.sh => " +
+ str(self.need_to_prepare_autotools()))
+ self.log.info("Products directory => " + self.products_directory())
+ self.log.info("Build directory => " + self.build_directory())
+ self.log.info("Need to run configure => " +
+ str(self.need_to_run_configure()))
+ self.log.info("Use _GLIBCXX_DEBUG => " +
+ str(self.options.use_glibcxx_debug))
+ self.log.info("Use pre-compiled headers => " +
+ str('--enable-pch' in conf_args))
+
+ self.log.debug('Configure environment =>')
+
+ keys = environ.keys()
+ keys.sort()
+ for key in keys:
+ if key in ['PATH', 'CC', 'LD', 'CXX'] or \
+ key.endswith('FLAGS'):
+ self.log.debug(' %s=%s' % (key, environ[key]))
+
+ self.log.debug('Configure arguments =>')
+
+ for arg in conf_args + list(args):
+ self.log.debug(' %s' % arg)
+
+ def phase_sloc(self, *args):
+ self.log.info('Executing phase: sloc')
+ self.execute('sloccount', 'src', 'python', 'lisp', 'test')
+
+ #########################################################################
+ # Configure source tree using autogen #
+ #########################################################################
+
+ def phase_gettext(self, *args):
+ self.log.info('Executing phase: gettext')
+
+ # configure the template files
+ assert exists('po') and isdir('po')
+ if not exists(join('po', 'Makevars')):
+ assert exists(join('po', 'Makevars.template'))
+ self.log.info('Moving po/Makevars.template -> po/Makevars')
+ os.rename(join('po', 'Makevars.template'),
+ join('po', 'Makevars'))
+
+ POTFILES_in = open('po/POTFILES.in', 'w')
+ for filename in (f for f in os.listdir(join(self.source_dir, 'src'))
+ if re.search('\.(cc|h)', f)):
+ POTFILES_in.write(join('src', filename))
+ POTFILES_in.write('\n')
+ POTFILES_in.close()
+
+ def phase_version(self, *args):
+ self.log.info('Executing phase: version')
+ version_m4 = open('version.m4', 'w')
+ version_m4.write("m4_define([VERSION_NUMBER], [%s])\n" %
+ self.current_version())
+ version_m4.close()
+
+ def phase_autogen(self, *args):
+ self.log.info('Executing phase: autogen')
+
+ if not exists('autogen.sh') or \
+ self.isnewer('tools/autogen.sh', 'autogen.sh'):
+ shutil.copyfile('tools/autogen.sh', 'autogen.sh')
+
+ self.execute('sh', 'tools/autogen.sh')
+
+ def phase_aclocal(self, *args):
+ self.log.info('Executing phase: aclocal')
+ self.execute('aclocal', '-I', 'm4')
+
+ def phase_autoconf(self, *args):
+ self.log.info('Executing phase: autoconf')
+
+ if not exists('configure.ac') or \
+ self.isnewer('tools/configure.ac', 'configure.ac'):
+ shutil.copyfile('tools/configure.ac', 'configure.ac')
+
+ if not exists('Makefile.am') or \
+ self.isnewer('tools/Makefile.am', 'Makefile.am'):
+ shutil.copyfile('tools/Makefile.am', 'Makefile.am')
+
+ reason = self.need_to_prepare_autotools()
+ if reason:
+ self.log.info('autogen.sh must be run ' + reason)
+ self.phase_version()
+ self.phase_autogen()
+ self.phase_gettext()
+ self.phase_aclocal()
+ else:
+ self.log.debug('autogen.sh does not need to be run')
+
+ #########################################################################
+ # Update local files with the latest information #
+ #########################################################################
+
+ def phase_submodule(self, *args):
+ self.log.info('Executing phase: submodule')
+ if self.git_working_tree():
+ self.execute('git', 'submodule', 'init')
+ self.execute('git', 'submodule', 'update')
+
+ def phase_pull(self, *args):
+ self.log.info('Executing phase: pull')
+ if self.git_working_tree():
+ self.execute('git', 'pull')
+ self.phase_submodule()
+
+ #########################################################################
+ # Automatic installation of build dependencies #
+ #########################################################################
+
+ def phase_dependencies(self, *args):
+ self.log.info('Executing phase: dependencies')
+
+ self.log.info("Installing Ledger's build dependencies ...")
+
+ system = self.get_stdout('uname', '-s')
+
+ if system == 'Darwin':
+ if exists('/opt/local/bin/port'):
+ self.log.info('Looks like you are using MacPorts on OS X')
+ packages = [
+ 'sudo', 'port', 'install', '-f',
+ 'automake',
+ 'autoconf',
+ 'libtool',
+ 'python26', '+universal',
+ 'libiconv', '+universal',
+ 'zlib', '+universal',
+ 'gmp' ,'+universal',
+ 'mpfr', '+universal',
+ 'ncurses', '+universal',
+ 'ncursesw', '+universal',
+ 'gettext' ,'+universal',
+ 'libedit' ,'+universal',
+ 'boost-jam',
+ 'boost', '+icu+python26+st+universal',
+ 'cppunit', '+universal',
+ #'texlive',
+ #'doxygen',
+ #'graphviz',
+ 'texinfo',
+ 'lcov',
+ 'sloccount'
+ ]
+ self.log.info('Executing: ' + string.join(packages, ' '))
+ self.execute(*packages)
+ elif exists('/sw/bin/fink'):
+ self.log.info('Looks like you are using Fink on OS X')
+ self.log.error("I don't know the package names for Fink yet!")
+
+ elif system == 'Linux':
+ if exists('/etc/issue'):
+ issue = open('/etc/issue')
+ if issue.readline().startswith('Ubuntu'):
+ release = open('/etc/lsb-release')
+ info = release.read()
+ release.close()
+ if re.search('karmic', info):
+ self.log.info('Looks like you are using APT on Ubuntu Karmic')
+ packages = [
+ 'sudo', 'apt-get', 'install',
+ 'build-essential',
+ 'libtool',
+ 'autoconf',
+ 'automake',
+ 'zlib1g-dev',
+ 'libbz2-dev',
+ 'python-dev',
+ 'libgmp3-dev',
+ 'libmpfr-dev',
+ 'bjam',
+ 'gettext',
+ 'cvs',
+ 'libboost1.40-dev',
+ 'libboost-regex1.40-dev',
+ 'libboost-date-time1.40-dev',
+ 'libboost-filesystem1.40-dev',
+ 'libboost-python1.40-dev',
+ 'libedit-dev',
+ 'libcppunit-dev',
+ #'texlive-full',
+ #'doxygen',
+ #'graphviz',
+ 'texinfo',
+ 'lcov',
+ 'sloccount'
+ ]
+ else:
+ self.log.info('Looks like you are using APT on Ubuntu Hardy')
+ packages = [
+ 'sudo', 'apt-get', 'install',
+ 'build-essential',
+ 'libtool',
+ 'autoconf',
+ 'automake',
+ 'zlib1g-dev',
+ 'libbz2-dev',
+ 'python-dev',
+ 'bjam',
+ 'cvs',
+ 'gettext',
+ 'libgmp3-dev',
+ 'libmpfr-dev',
+ 'libboost1.35-dev',
+ 'libboost-python1.35-dev',
+ 'libboost-regex1.35-dev',
+ 'libboost-date-time1.35-dev',
+ 'libboost-filesystem1.35-dev',
+ 'libedit-dev',
+ 'libcppunit-dev',
+ #'texlive-full',
+ #'doxygen',
+ #'graphviz',
+ 'texinfo',
+ 'lcov',
+ 'sloccount'
+ ]
+ self.log.info('Executing: ' + string.join(packages, ' '))
+ self.execute(*packages)
+
+ if exists('/etc/redhat-release'):
+ release = open('/etc/redhat-release')
+ if release.readline().startswith('CentOS'):
+ self.log.info('Looks like you are using YUM on CentOS')
+ packages = [
+ 'sudo', 'yum', 'install',
+ 'gcc',
+ 'gcc-c++',
+ 'compat-gcc-*',
+ 'make',
+ 'libtool',
+ 'autoconf',
+ 'automake',
+ 'zlib-devel',
+ 'bzip2-devel',
+ 'python-devel',
+ 'bboost-devel',
+ 'gmp-devel',
+ 'gettext-devel',
+ #'mpfr-devel'
+ 'libedit-devel',
+ 'cppunit-devel',
+ #'texlive-full',
+ #'doxygen',
+ #'graphviz',
+ 'texinfo',
+ #'lcov',
+ #'sloccount'
+ ]
+ self.log.info('Executing: ' + string.join(packages, ' '))
+ self.execute(*packages)
+
+ def phase_buildlibs(self, *args):
+ self.log.info('Executing phase: buildlibs')
+
+ try:
+ os.chdir('lib')
+
+ environ, conf_args = self.configure_environment()
+
+ boost = 'boost_1_41_0'
+ tarball = boost + '.7z'
+
+ if not exists(boost):
+ if not exists(tarball):
+ self.log.info('Downloading Boost source tarball ...')
+ self.execute('curl', '-L', '-o', tarball,
+ 'http://downloads.sourceforge.net/boost/' +
+ boost + '.7z?use_mirror=ufpr')
+
+ if not exists(tarball):
+ self.log.error('Failed to locate the Boost source tarball')
+ sys.exit(1)
+
+ fd = open(tarball)
+ try:
+ csum = hashlib.md5()
+ except:
+ csum = md5.md5()
+ csum.update(fd.read())
+ fd.close()
+ digest = csum.hexdigest()
+
+ if digest != 'b74ee2f0f46cef601544dd4ac2d7dec4':
+ self.log.error('Boost source tarball fails to match checksum')
+ sys.exit(1)
+
+ self.log.info('Extracting Boost source tarball ...')
+ self.execute('7za', 'x', tarball)
+
+ if not exists(boost):
+ self.log.error('Failed to locate the Boost sources')
+ sys.exit(1)
+
+ if not exists('cppunit') and self.git_working_tree():
+ self.execute('git', 'clone', 'git://github.com/jwiegley/cppunit.git')
+
+ if not exists('cppunit'):
+ self.log.error('Failed to locate the CppUnit sources')
+ sys.exit(1)
+
+ self.execute('make',
+ 'BOOST_SOURCE=%s' % boost,
+ 'CC=%s' % environ['CC'],
+ 'CXX=%s' % environ['CXX'],
+ 'LD=%s' % environ['LD'],
+ 'build-all')
+ finally:
+ os.chdir(self.source_dir)
+
+ #########################################################################
+ # Determine the system's basic configuration #
+ #########################################################################
+
+ def locate_boost_in_dir(self, path):
+ if exists(path) and isdir(path):
+ entries = os.listdir(path)
+ entries.sort()
+ for entry in entries:
+ if re.search('boost_regex', entry):
+ self.log.info('Found a Boost library: ' + entry)
+
+ match = re.match('libboost_regex([^.]*)\.(a|so|dylib)', entry)
+ if match:
+ suffix = match.group(1)
+ self.log.info('Found Boost suffix => ' + suffix)
+ self.envvars['BOOST_HOME'] = dirname(path)
+ return suffix
+ else:
+ self.log.debug('The directory "%s" is not valid, skipping' %
+ path)
+ return None
+
+ def inform_boost_location(self, text, suffix):
+ self.log.info('Boost %s here:' % text)
+ self.log.info('BOOST_HOME => ' + self.envvars['BOOST_HOME'])
+ self.log.info('BOOST_SUFFIX => ' + suffix)
+
+ def locate_boost(self):
+ if self.envvars['BOOST_SUFFIX']:
+ self.log.info(("Not looking for Boost, since " +
+ "a suffix of '%s' was given") %
+ self.envvars['BOOST_SUFFIX'])
+ else:
+ suffix = None
+ for path in ['/usr/local/lib', '/opt/local/lib',
+ '/sw/lib', '/usr/lib']:
+ self.log.info('Looking for Boost in %s...' % path)
+ suffix = self.locate_boost_in_dir(path)
+ if suffix is not None:
+ self.inform_boost_location('was found', suffix)
+ break
+ if suffix is None:
+ self.log.error("Boost could not be found.")
+ self.envvars['BOOST_SUFFIX'] = suffix
+ return self.envvars['BOOST_SUFFIX']
+
+ def setup_system_directories(self):
+ boost_suffix = self.locate_boost()
+
+ # Each of these becomes '-isystem <name>'
+ for path in ['/usr/local/include',
+ '%s/include/boost' %
+ self.envvars['BOOST_HOME'],
+ '%s/include' % self.envvars['BOOST_HOME'],
+ '%s/include/python%s' %
+ (self.envvars['PYTHON_HOME'],
+ self.envvars['PYTHON_VERSION'].strip()),
+ '/opt/local/include',
+ '/sw/include']:
+ if exists(path) and isdir(path) and \
+ path != '/usr/include':
+ self.log.info('Noticing include directory => ' + path)
+ self.sys_include_dirs.append(path)
+
+ # Each of these becomes '-L<name>'
+ for path in ['/usr/local/lib',
+ '%s/lib' % self.envvars['PYTHON_HOME'],
+ '%s/lib/python%s/config'
+ % (self.envvars['PYTHON_HOME'],
+ self.envvars['PYTHON_VERSION'].strip()),
+ '/opt/local/lib',
+ '/sw/lib']:
+ if exists(path) and isdir(path):
+ self.log.info('Noticing library directory => ' + path)
+ self.sys_library_dirs.append(path)
+
+ def setup_for_johnw(self):
+ if self.current_flavor == 'debug' or self.current_flavor == 'gcov':
+ if exists('/usr/local/stow/cppunit/include'):
+ self.sys_include_dirs.insert(0, '/usr/local/stow/cppunit/include')
+ self.sys_library_dirs.insert(0, '/usr/local/stow/cppunit/lib')
+
+ if exists('/usr/local/stow/icu/include'):
+ self.sys_include_dirs.insert(0, '/usr/local/stow/icu/include')
+ self.sys_library_dirs.insert(0, '/usr/local/stow/icu/lib')
+
+ self.CPPFLAGS.append('-D_GLIBCXX_FULLY_DYNAMIC_STRING=1')
+ self.configure_args.append('--disable-shared')
+
+ self.options.use_glibcxx_debug = True
+ else:
+ self.CXXFLAGS.append('-march=nocona')
+ self.CXXFLAGS.append('-msse3')
+
+ self.configure_args.append('--enable-doxygen')
+ self.configure_args.append('--enable-python')
+
+ def setup_for_system(self):
+ self.setup_system_directories()
+
+ system = self.get_stdout('uname', '-s')
+
+ self.log.info('System type is => ' + system)
+
+ # These options are global defaults at the moment
+ #self.option_warn()
+ if not self.no_pch:
+ self.option_pch()
+
+ if system == 'Linux':
+ arch = self.get_stdout('uname', '-m')
+ if arch == 'x86_64':
+ if '--disable-shared' in self.configure_args:
+ self.configure_args.remove('--disable-shared')
+ self.configure_args.append('--disable-static')
+ self.CXXFLAGS.append('-pthread')
+
+ elif system == 'Solaris':
+ self.CXXFLAGS.append('-pthread')
+
+ elif system == 'Darwin':
+ if exists('/Users/johnw/Dropbox/Accounts/ledger.dat'):
+ self.setup_for_johnw()
+
+ self.locate_darwin_libraries()
+
+ if (self.current_flavor == 'opt' or \
+ self.current_flavor == 'default') and \
+ exists('/usr/bin/g++-4.2'):
+ self.envvars['CC'] = '/usr/bin/gcc-4.2'
+ self.envvars['CXX'] = '/usr/bin/g++-4.2'
+ self.envvars['LD'] = '/usr/bin/g++-4.2'
+ self.darwin_gcc = True
+ elif exists('/opt/local/bin/g++-mp-4.4'):
+ self.envvars['CC'] = '/opt/local/bin/gcc-mp-4.4'
+ self.envvars['CXX'] = '/opt/local/bin/g++-mp-4.4'
+ self.envvars['LD'] = '/opt/local/bin/g++-mp-4.4'
+ elif exists('/opt/local/bin/g++-mp-4.3'):
+ self.envvars['CC'] = '/opt/local/bin/gcc-mp-4.3'
+ self.envvars['CXX'] = '/opt/local/bin/g++-mp-4.3'
+ self.envvars['LD'] = '/opt/local/bin/g++-mp-4.3'
+ elif exists('/usr/bin/g++-4.2'):
+ self.envvars['CC'] = '/usr/bin/gcc-4.2'
+ self.envvars['CXX'] = '/usr/bin/g++-4.2'
+ self.envvars['LD'] = '/usr/bin/g++-4.2'
+ self.darwin_gcc = True
+ else:
+ # g++ 4.0.1 cannot use PCH headers on OS X 10.5
+ self.option_no_pch()
+
+ if '--enable-pch' not in self.configure_args and \
+ (exists('/opt/local/bin/ccache') or \
+ exists('/usr/local/bin/ccache')):
+ self.envvars['CC'] = 'ccache ' + self.envvars['CC']
+ self.envvars['CXX'] = 'ccache ' + self.envvars['CXX']
+ self.envvars['LD'] = 'ccache ' + self.envvars['LD']
+
+ def setup_flags(self):
+ for path in self.sys_include_dirs:
+ self.CPPFLAGS.append('-isystem')
+ self.CPPFLAGS.append(path)
+
+ self.CXXFLAGS.append('-pipe')
+
+ for path in self.sys_library_dirs:
+ self.LDFLAGS.append('-L' + path)
+
+ def setup_flavor(self):
+ self.setup_for_system()
+
+ if not PrepareBuild.__dict__.has_key('setup_flavor_' +
+ self.current_flavor):
+ self.log.error('Unknown build flavor "%s"' % self.current_flavor)
+ sys.exit(1)
+
+ self.log.info('Setting up build flavor => ' + self.current_flavor)
+ PrepareBuild.__dict__['setup_flavor_' + self.current_flavor](self)
+
+ self.setup_flags()
+
+ def escape_string(self, data):
+ return re.sub('(["\\\\])', '\\\\\\1', data)
+
+ def finalize_config(self):
+ self.setup_flavor()
+
+ for var in ('CPPFLAGS', 'CFLAGS', 'CXXFLAGS', 'LDFLAGS'):
+ value = self.__dict__[var]
+ if value:
+ first = not self.envvars[var]
+ for member in value:
+ #escaped = self.escape_string(member)
+ #if member != escaped:
+ # member = escaped
+ if first:
+ first = False
+ else:
+ self.envvars[var] += ' '
+ self.envvars[var] += member
+ self.log.debug('Final value of %s: %s' %
+ (var, self.envvars[var]))
+
+ elif self.envvars.has_key(var):
+ del self.envvars[var]
+
+ #########################################################################
+ # Options that can modify any build flavor #
+ #########################################################################
+
+ def option_pch(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --pch')
+
+ self.configure_args.append('--enable-pch')
+
+ self.CXXFLAGS.append('-fpch-deps')
+ self.CXXFLAGS.append('-Wconversion')
+ #self.CXXFLAGS.append('-Wold-style-cast')
+
+ system = self.get_stdout('uname', '-s')
+
+ def option_no_pch(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --no-pch')
+
+ self.no_pch = True
+
+ if '--enable-pch' in self.configure_args:
+ self.configure_args.remove('--enable-pch')
+
+ if '-Wconversion' in self.configure_args:
+ self.CXXFLAGS.remove('-Wconversion')
+ #if '-Wold-style-cast' in self.configure_args:
+ # self.CXXFLAGS.remove('-Wold-style-cast')
+
+ system = self.get_stdout('uname', '-s')
+
+ if system == "Darwin":
+ self.envvars['CC'] = 'gcc'
+ self.envvars['CXX'] = 'g++'
+ self.envvars['LD'] = 'g++'
+
+ def option_force(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --force')
+ self.force = True
+
+ def option_warn(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --warn')
+ self.CXXFLAGS.append('-ansi')
+ self.CXXFLAGS.append('-pedantic')
+ self.CXXFLAGS.append('-pedantic-errors')
+ self.CXXFLAGS.append('-Wall')
+ self.CXXFLAGS.append('-Winvalid-pch')
+ self.CXXFLAGS.append('-Wextra')
+ self.CXXFLAGS.append('-Wcast-align')
+ self.CXXFLAGS.append('-Wcast-qual')
+ self.CXXFLAGS.append('-Wfloat-equal')
+ self.CXXFLAGS.append('-Wmissing-field-initializers')
+ self.CXXFLAGS.append('-Wno-endif-labels')
+ self.CXXFLAGS.append('-Woverloaded-virtual')
+ self.CXXFLAGS.append('-Wsign-compare')
+ self.CXXFLAGS.append('-Wsign-promo')
+ self.CXXFLAGS.append('-Wstrict-null-sentinel')
+ self.CXXFLAGS.append('-Wwrite-strings')
+ self.CXXFLAGS.append('-Wno-old-style-cast')
+ self.CXXFLAGS.append('-Wno-deprecated')
+ self.CXXFLAGS.append('-Wno-strict-aliasing')
+ self.CXXFLAGS.append('-Werror')
+
+ def option_boost(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --boost')
+ self.envvars['BOOST_SUFFIX'] = value
+
+ def option_pic(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --pic')
+ self.CXXFLAGS.append('-fPIC')
+
+ def option_output(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --output')
+ self.build_dir = value
+
+ def option_products(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --products')
+ self.products_dir = value
+
+ def option_local(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --local')
+ self.build_dir = self.source_dir
+
+ def option_trees(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --trees')
+ self.build_dir = None
+
+ def option_release(self, option=None, opt_str=None, value=None, parser=None):
+ self.log.debug('Saw option --release')
+ if '--disable-shared' in self.configure_args:
+ self.configure_args.remove('--disable-shared')
+ self.configure_args.append('--disable-dependency-tracking')
+
+ def option_help(self, option=None, opt_str=None, value=None, parser=None):
+ self.phase_help()
+
+ #########################################################################
+ # The various build flavors #
+ #########################################################################
+
+ def locate_darwin_libraries(self):
+ if self.current_flavor == 'debug' or self.current_flavor == 'gcov':
+ self.log.debug('We are using GLIBCXX_DEBUG, so setting up flags')
+
+ if self.options.use_glibcxx_debug:
+ self.CPPFLAGS.append('-D_GLIBCXX_DEBUG=1')
+
+ if exists('/usr/local/lib/libboost_regex-xgcc44-sd-1_41.a'):
+ self.envvars['BOOST_HOME'] = '/usr/local'
+ self.envvars['BOOST_SUFFIX'] = '-xgcc44-sd-1_41'
+ self.sys_include_dirs.append('/usr/local/include/boost-1_41')
+ self.inform_boost_location('is really located',
+ self.envvars['BOOST_SUFFIX'])
+
+ elif exists('/usr/local/lib/libboost_regex-xgcc44-d-1_41.a'):
+ self.envvars['BOOST_HOME'] = '/usr/local'
+ self.envvars['BOOST_SUFFIX'] = '-xgcc44-d-1_41'
+ self.sys_include_dirs.append('/usr/local/include/boost-1_41')
+ self.inform_boost_location('is really located',
+ self.envvars['BOOST_SUFFIX'])
+
+ elif exists('/opt/local/lib/libboost_regex-d.a'):
+ self.envvars['BOOST_HOME'] = '/opt/local'
+ self.envvars['BOOST_SUFFIX'] = '-d'
+ self.sys_include_dirs.append('/opt/local/include/boost')
+ self.inform_boost_location('is really located',
+ self.envvars['BOOST_SUFFIX'])
+
+ else:
+ if exists('/opt/local/lib/libboost_regex.a'):
+ self.envvars['BOOST_HOME'] = '/opt/local'
+ self.envvars['BOOST_SUFFIX'] = ''
+ self.sys_include_dirs.append('/opt/local/include/boost')
+ self.inform_boost_location('is really located',
+ self.envvars['BOOST_SUFFIX'])
+
+ elif exists('/usr/local/lib/libboost_regex-xgcc44-s-1_41.a'):
+ self.envvars['BOOST_HOME'] = '/usr/local'
+ self.envvars['BOOST_SUFFIX'] = '-xgcc44-s-1_41'
+ self.sys_include_dirs.append('/usr/local/include/boost-1_41')
+ self.inform_boost_location('is really located',
+ self.envvars['BOOST_SUFFIX'])
+
+ elif exists('/usr/local/lib/libboost_regex-xgcc44-1_41.a'):
+ self.envvars['BOOST_HOME'] = '/usr/local'
+ self.envvars['BOOST_SUFFIX'] = '-xgcc44-1_41'
+ self.sys_include_dirs.append('/usr/local/include/boost-1_41')
+ self.inform_boost_location('is really located',
+ self.envvars['BOOST_SUFFIX'])
+
+ def setup_flavor_default(self):
+ pass
+
+ def setup_flavor_debug(self):
+ self.configure_args.append('--enable-debug')
+
+ self.CXXFLAGS.append('-g')
+ self.LDFLAGS.append('-g')
+
+ def setup_flavor_opt(self):
+ if self.darwin_gcc:
+ self.option_no_pch()
+ if '--disable-shared' in self.configure_args:
+ self.configure_args.remove('--disable-shared')
+ self.configure_args.append('--disable-dependency-tracking')
+ for i in ['-fast']:
+ self.CXXFLAGS.append(i)
+ self.CFLAGS.append(i)
+ self.LDFLAGS.append(i)
+ for i in ['-arch', 'i386', '-arch', 'x86_64']:
+ self.CXXFLAGS.append(i)
+ self.CFLAGS.append(i)
+ self.LDFLAGS.append(i)
+ else:
+ for i in ['-O3', '-fomit-frame-pointer']:
+ self.CXXFLAGS.append(i)
+ self.CFLAGS.append(i)
+ self.LDFLAGS.append(i)
+
+ def setup_flavor_gcov(self):
+ self.CXXFLAGS.append('-g')
+ self.CXXFLAGS.append('-fprofile-arcs')
+ self.CXXFLAGS.append('-ftest-coverage')
+ self.LDFLAGS.append('-g')
+
+ def setup_flavor_gprof(self):
+ self.CXXFLAGS.append('-g')
+ self.CXXFLAGS.append('-pg')
+ self.LDFLAGS.append('-g')
+ self.LDFLAGS.append('-pg')
+
+ #########################################################################
+ # Prettify the output from automake, by rewriting the Makefile #
+ #########################################################################
+
+ def phase_patch(self, *args):
+ """Alter the Makefile so that it's not nearly so verbose.
+
+ This makes errors and warnings much easier to spot."""
+ self.log.info('Executing phase: patch')
+
+ if exists('Makefile'):
+ self.log.debug('Patching generated Makefile')
+ Makefile = open('Makefile')
+ Makefile_new = open('Makefile.new', 'w')
+ for line in Makefile.readlines():
+ line = re.sub('^\t(\$\((LIBTOOL|CXX)\).*?\.(cc|cpp))$',
+ '\t@echo " " CXX \$@;\\1 > /dev/null', line)
+ line = re.sub('^\tmv -f', '\t@mv -f', line)
+ line = re.sub('^\t\$\(am__mv\)', '\t@$(am__mv)', line)
+ line = re.sub('^\t(\$\((.*?)LINK\).*)',
+ '\t@echo " LD " \$@;\\1 > /dev/null', line)
+ Makefile_new.write(line)
+ Makefile_new.close()
+ Makefile.close()
+
+ os.remove('Makefile')
+ os.rename('Makefile.new', 'Makefile')
+
+ stamp = open('.timestamp', 'w')
+ stamp.write('timestamp')
+ stamp.close()
+
+ #########################################################################
+ # Configure build tree using autoconf #
+ #########################################################################
+
+ def configure_environment(self):
+ self.finalize_config()
+
+ environ = dict(os.environ)
+ for key, value in self.envvars.items():
+ if value:
+ environ[key] = value
+
+ environ['PATH'] = ('%s/bin:%s' %
+ (environ['PYTHON_HOME'], environ['PATH']))
+
+ if self.build_directory() == self.source_dir:
+ conf_args = ['sh', 'configure']
+ else:
+ conf_args = ['sh', join(self.source_dir, 'configure'),
+ '--srcdir', self.source_dir]
+
+ for var in ('CC', 'CPPFLAGS', 'CFLAGS', 'CXX', 'CXXFLAGS',
+ 'LD', 'LDFLAGS'):
+ if self.envvars.has_key(var) and self.envvars[var] and \
+ (var.endswith('FLAGS') or exists(self.envvars[var])):
+ conf_args.append('%s=%s' % (var, self.envvars[var]))
+
+ if environ.has_key('BOOST_SUFFIX') and environ['BOOST_SUFFIX']:
+ conf_args.append('--with-boost-suffix=%s' %
+ environ['BOOST_SUFFIX'])
+
+ return (environ, conf_args + self.configure_args)
+
+ def need_to_run_configure(self):
+ Makefile = join(self.build_directory(), 'Makefile')
+ if self.force:
+ return 'because it was forced'
+ elif not exists(Makefile):
+ return 'because Makefile does not exist'
+ elif self.isnewer(join(self.source_dir, 'configure'), Makefile):
+ return 'because configure is newer than Makefile'
+ elif self.isnewer(join(self.source_dir, 'Makefile.in'), Makefile):
+ return 'because Makefile.in is newer than Makefile'
+ return False
+
+ def phase_configure(self, *args):
+ self.log.info('Executing phase: configure')
+
+ self.configured = True
+
+ build_dir = self.ensure(self.build_directory())
+ try:
+ os.chdir(build_dir)
+
+ reason = self.need_to_run_configure()
+ if reason:
+ self.log.info('./configure must be run ' + reason)
+ self.log.debug('Source => ' + self.source_dir)
+ self.log.debug('Build => ' + build_dir)
+
+ environ, conf_args = self.configure_environment()
+ for arg in args:
+ if arg: conf_args.append(arg)
+
+ self.log.debug('configure env => ' + str(environ))
+ self.log.debug('configure args => ' + str(conf_args))
+
+ configure = Popen(conf_args, shell=False, env=environ)
+ retcode = configure.wait()
+ if retcode < 0:
+ self.log.error("Child was terminated by signal", -retcode)
+ sys.exit(1)
+ elif retcode != 0:
+ self.log.error("Execution failed: " + string.join(conf_args, ' '))
+ sys.exit(1)
+
+ if not self.options.no_patch:
+ self.phase_patch()
+
+ # Wipe the pre-compiled header, if there is one
+ pch = join(self.build_directory(), 'system.hh.gch')
+ if exists(pch):
+ os.remove(pch)
+ else:
+ if not self.options.no_patch and \
+ self.isnewer('Makefile', '.timestamp'):
+ self.phase_patch()
+
+ self.log.debug('configure does not need to be run')
+
+ finally:
+ os.chdir(self.source_dir)
+
+ def phase_config(self, *args):
+ self.log.info('Executing phase: config')
+ self.phase_submodule()
+ self.phase_autoconf()
+ self.phase_configure(*args)
+ if self.should_clean:
+ self.phase_clean()
+
+ #########################################################################
+ # Builds products from the sources #
+ #########################################################################
+
+ def phase_make(self, *args):
+ self.log.info('Executing phase: make')
+
+ config_args = []
+ make_args = []
+
+ for arg in args:
+ if arg.startswith('--'):
+ config_args.append(arg)
+ else:
+ make_args.append(arg)
+
+ if self.options.jobs > 1:
+ make_args.append('-j%d' % self.options.jobs)
+
+ self.log.debug('Configure arguments => ' + str(config_args))
+ self.log.debug('Makefile arguments => ' + str(make_args))
+
+ if not self.configured:
+ self.phase_config(*config_args)
+
+ build_dir = self.ensure(self.build_directory())
+ try:
+ self.log.debug('Changing directory to ' + build_dir)
+ os.chdir(build_dir)
+
+ self.execute(*(['make'] + make_args))
+ finally:
+ os.chdir(self.source_dir)
+
+ def phase_update(self, *args):
+ self.log.info('Executing phase: update')
+ self.phase_pull()
+ self.phase_make(*args)
+
+ #########################################################################
+ # Build directory cleaning phases #
+ #########################################################################
+
+ def phase_clean(self, *args):
+ self.log.info('Executing phase: clean')
+ self.phase_make('clean')
+
+ def phase_distclean(self, *args):
+ self.log.info('Executing phase: distclean')
+ self.phase_make('distclean')
+
+ def phase_gitclean(self, *args):
+ self.log.info('Executing phase: gitclean')
+ if self.git_working_tree():
+ self.execute('git', 'clean', '-dfx')
+
+ #########################################################################
+ # Packaging phases #
+ #########################################################################
+
+ def translate_file(self, path, dest):
+ dest_file = join(dest, basename(path))
+
+ self.log.debug("Translating file %s -> %s" % (path, dest_file))
+
+ if not exists(dest_file):
+ shutil.copyfile(path, dest_file)
+ os.chmod(dest_file, 0755)
+
+ for line in self.get_stdout('otool', '-L', dest_file).split('\n'):
+ match = re.search('/opt/local/lib/(.+?)\.dylib', line)
+ if not match:
+ continue
+
+ lib = match.group(0)
+ base = basename(lib)
+
+ if lib != path:
+ self.translate_file(lib, dest)
+
+ self.execute('install_name_tool', '-change', lib,
+ '@loader_path/' + base, dest_file)
+
+ def phase_bindmg(self, *args):
+ self.log.info('Executing phase: bindmg')
+
+ self.phase_make()
+
+ binary = join(self.build_directory(), 'ledger')
+ if not exists(binary):
+ self.log.error('Failed to build Ledger: ' + binary)
+ sys.exit(1)
+
+ cwd = os.getcwd()
+ tempdir = tempfile.mkdtemp()
+ try:
+ name = 'ledger-%s' % self.current_version()
+ dest = join(tempdir, name)
+
+ os.makedirs(dest)
+ self.translate_file(binary, dest)
+
+ self.execute('hdiutil', 'create', '-srcfolder', dest,
+ '-ov', join(cwd, name + '.dmg'))
+ finally:
+ os.chdir(cwd)
+ shutil.rmtree(tempdir)
+
+ def phase_upload(self, *args):
+ self.log.info('Executing phase: upload')
+
+ self.phase_bindmg()
+
+ dmg = 'ledger-%s.dmg' % self.current_version()
+ self.execute('zip', '%s.zip' % dmg, dmg)
+ dmg = '%s.zip' % dmg
+
+ self.execute('ssh', 'jw', 'rm', '-f', '/srv/ftp/pub/ledger/ledger-*.dmg.zip')
+ self.execute('scp', dmg, 'jw:/srv/ftp/pub/ledger')
+ self.execute('ssh', 'jw',
+ '(cd /srv/ftp/pub/ledger; ln -sf %s ledger-current.dmg.zip)' %
+ basename(dmg))
+
+ #########################################################################
+ # Other build phases #
+ #########################################################################
+
+ def configure_flavor(self, flavor, reset=True):
+ self.initialize() # reset everything
+ self.build_dir = None # use the build/ tree
+ self.current_flavor = flavor
+
+ if reset and exists(self.build_directory()) and \
+ isdir(self.build_directory()):
+ self.option_release()
+
+ self.log.info('=== Wiping build directory %s ===' %
+ self.build_directory())
+ shutil.rmtree(self.build_directory())
+
+ def phase_distcheck(self, *args):
+ self.log.info('Executing phase: distcheck')
+
+ self.configure_flavor('default', False)
+
+ environ, conf_args = self.configure_environment()
+
+ configure_args = []
+
+ skip_next = False
+ for arg in conf_args[2:]: # skip "sh configure"
+ if arg == '--srcdir':
+ skip_next = True
+ elif skip_next:
+ skip_next = False
+ else:
+ configure_args.append('"' + self.escape_string(arg) + '"')
+
+ make_args = ['DISTCHECK_CONFIGURE_FLAGS=%s' %
+ string.join(configure_args, ' '), 'distcheck']
+
+ self.log.debug('make_args for distcheck => ' + str(make_args))
+
+ self.phase_make(*make_args)
+
+ def phase_rsync(self, *args):
+ self.log.info('Executing phase: rsync')
+
+ source_copy_dir = join(self.ensure(self.products_directory()),
+ 'ledger-proof')
+
+ self.execute('rsync', '-a', '--delete',
+ '--exclude=.git/', '--exclude=b/',
+ '%s/' % self.source_dir, '%s/' % source_copy_dir)
+
+ self.source_dir = source_copy_dir
+
+ def phase_proof(self, *args):
+ self.log.info('Executing phase: proof')
+
+ self.log.info('=== Copying source tree ===')
+ self.phase_rsync()
+ self.phase_makeall(reset=True, *args)
+
+ self.configure_flavor('opt')
+ self.log.info('=== Testing opt ===')
+ self.phase_make('fullcheck')
+
+ self.configure_flavor('gcov')
+ self.log.info('=== Testing gcov ===')
+ self.phase_make('check')
+
+ self.configure_flavor('debug')
+ self.log.info('=== Testing debug ===')
+ self.phase_make('fullcheck')
+
+ self.configure_flavor('default')
+ self.log.info('=== Testing default ===')
+ self.phase_make('fullcheck')
+
+ self.log.info('=== Building final distcheck ===')
+ self.phase_distcheck()
+
+ def phase_makeall(self, reset=False, *args):
+ self.log.info('Executing phase: makeall')
+
+ self.configure_flavor('opt', reset)
+
+ system_hh_gch = join(self.source_dir, 'src', 'system.hh.gch')
+ if exists(system_hh_gch):
+ os.remove(system_hh_gch)
+
+ self.log.info('=== Building opt ===')
+ self.phase_make(*args)
+
+ self.configure_flavor('gcov', reset)
+
+ system_hh_gch = join(self.source_dir, 'src', 'system.hh.gch')
+ if exists(system_hh_gch):
+ os.remove(system_hh_gch)
+
+ self.log.info('=== Building gcov ===')
+ self.phase_make(*args)
+
+ system_hh_gch = join(self.source_dir, 'src', 'system.hh.gch')
+ if exists(system_hh_gch):
+ os.remove(system_hh_gch)
+
+ self.log.info('=== Building default ===')
+ self.phase_make(*args)
+
+ self.configure_flavor('debug', reset)
+
+ system_hh_gch = join(self.source_dir, 'src', 'system.hh.gch')
+ if exists(system_hh_gch):
+ os.remove(system_hh_gch)
+
+ self.log.info('=== Building debug ===')
+ self.phase_make(*args)
+
+ self.configure_flavor('default', reset)
+
+ #########################################################################
+ # Help #
+ #########################################################################
+
+ def phase_help(self, *args):
+ self.option_parser.print_help()
+
+ print """
+Of the optional ARGS, the first is an optional build FLAVOR, with the default
+being 'debug':
+
+ default Regular autoconf settings
+ debug Debugging and --verify support (default)
+ opt Full optimizations
+ gcov Coverage analysis
+ gprof Code profiling (for OS X, just use: 'shark -i ledger ...')
+
+Next is the optional build PHASE, with 'config' being the default:
+
+ clean Runs 'make clean' in the build directory
+ config Configure the environment for building
+ dependencies Automatically install all necessary build dependencies
+ distcheck Properly does 'make distcheck', carrying all flags
+ distclean Runs 'make distclean' in the build directory
+ gitclean Runs 'git clean -dfx', which *really* cleans things
+ help Displays this help text
+ info Show information about the build environment
+ make Do a make in the build directory
+ proof Proves Ledger by building and testing every flavor
+ pull Pulls the latest, and updates local config if need be
+ update Does it all, updates your environment and re-make's
+
+There are many other build phases, though most are not of interest to the
+typical user:
+
+ aclocal Runs aclocal -I m4
+ autoconf Prepare the autoconf subsystem
+ autogen Runs autogen.sh
+ configure Runs just ./configure
+ do_all Runs makeall followed by proof
+ gettext Initialize gettext support
+ makeall Build every flavor there is
+ patch Patches the automake Makefile to be less verbose
+ products Report the products directory path
+ rsync Rsync a copy of the source tree into Products
+ sloc Report total Source Lines Of Code
+ submodule Updates Git submodules (better to use 'pull')
+ version Output current HEAD version to version.m4
+
+NOTE: If you wish to pass options to configure or make, add "--" followed by
+your options. Here are some real-world examples:
+
+ ./acprep
+ ./acprep opt -- make -j3
+ ./acprep -- --enable-doxygen"""
+ sys.exit(0)
+
+PrepareBuild().run()