diff options
Diffstat (limited to 'build_tc.py')
-rwxr-xr-x | build_tc.py | 337 |
1 files changed, 337 insertions, 0 deletions
diff --git a/build_tc.py b/build_tc.py new file mode 100755 index 00000000..55fc5b70 --- /dev/null +++ b/build_tc.py @@ -0,0 +1,337 @@ +#!/usr/bin/python2 +# +# Copyright 2010 The Chromium OS Authors. All rights reserved. +# Use of this source code is governed by a BSD-style license that can be +# found in the LICENSE file. +"""Script to build the ChromeOS toolchain. + +This script sets up the toolchain if you give it the gcctools directory. +""" + +from __future__ import print_function + +__author__ = 'asharif@google.com (Ahmad Sharif)' + +import argparse +import getpass +import os +import sys +import tempfile + +import tc_enter_chroot +from cros_utils import command_executer +from cros_utils import constants +from cros_utils import misc + + +class ToolchainPart(object): + """Class to hold the toolchain pieces.""" + + def __init__(self, + name, + source_path, + chromeos_root, + board, + incremental, + build_env, + gcc_enable_ccache=False): + self._name = name + self._source_path = misc.CanonicalizePath(source_path) + self._chromeos_root = chromeos_root + self._board = board + self._ctarget = misc.GetCtargetFromBoard(self._board, self._chromeos_root) + self._gcc_libs_dest = misc.GetGccLibsDestForBoard(self._board, + self._chromeos_root) + self.tag = '%s-%s' % (name, self._ctarget) + self._ce = command_executer.GetCommandExecuter() + self._mask_file = os.path.join(self._chromeos_root, 'chroot', + 'etc/portage/package.mask/cross-%s' % + self._ctarget) + self._new_mask_file = None + + self._chroot_source_path = os.path.join(constants.MOUNTED_TOOLCHAIN_ROOT, + self._name).lstrip('/') + self._incremental = incremental + self._build_env = build_env + self._gcc_enable_ccache = gcc_enable_ccache + + def RunSetupBoardIfNecessary(self): + cross_symlink = os.path.join(self._chromeos_root, 'chroot', + 'usr/local/bin/emerge-%s' % self._board) + if not os.path.exists(cross_symlink): + command = ('%s/setup_board --board=%s' % + (misc.CHROMEOS_SCRIPTS_DIR, self._board)) + self._ce.ChrootRunCommand(self._chromeos_root, command) + + def Build(self): + rv = 1 + try: + self.UninstallTool() + self.MoveMaskFile() + self.MountSources(False) + self.RemoveCompiledFile() + rv = self.BuildTool() + finally: + self.UnMoveMaskFile() + return rv + + def RemoveCompiledFile(self): + compiled_file = os.path.join(self._chromeos_root, 'chroot', + 'var/tmp/portage/cross-%s' % self._ctarget, + '%s-9999' % self._name, '.compiled') + command = 'rm -f %s' % compiled_file + self._ce.RunCommand(command) + + def MountSources(self, unmount_source): + mount_points = [] + mounted_source_path = os.path.join(self._chromeos_root, 'chroot', + self._chroot_source_path) + src_mp = tc_enter_chroot.MountPoint(self._source_path, mounted_source_path, + getpass.getuser(), 'ro') + mount_points.append(src_mp) + + build_suffix = 'build-%s' % self._ctarget + build_dir = '%s-%s' % (self._source_path, build_suffix) + + if not self._incremental and os.path.exists(build_dir): + command = 'rm -rf %s/*' % build_dir + self._ce.RunCommand(command) + + # Create a -build directory for the objects. + command = 'mkdir -p %s' % build_dir + self._ce.RunCommand(command) + + mounted_build_dir = os.path.join(self._chromeos_root, 'chroot', '%s-%s' % + (self._chroot_source_path, build_suffix)) + build_mp = tc_enter_chroot.MountPoint(build_dir, mounted_build_dir, + getpass.getuser()) + mount_points.append(build_mp) + + if unmount_source: + unmount_statuses = [mp.UnMount() == 0 for mp in mount_points] + assert all(unmount_statuses), 'Could not unmount all mount points!' + else: + mount_statuses = [mp.DoMount() == 0 for mp in mount_points] + + if not all(mount_statuses): + mounted = [mp + for mp, status in zip(mount_points, mount_statuses) + if status] + unmount_statuses = [mp.UnMount() == 0 for mp in mounted] + assert all(unmount_statuses), 'Could not unmount all mount points!' + + def UninstallTool(self): + command = 'sudo CLEAN_DELAY=0 emerge -C cross-%s/%s' % (self._ctarget, + self._name) + self._ce.ChrootRunCommand(self._chromeos_root, command) + + def BuildTool(self): + env = self._build_env + # FEATURES=buildpkg adds minutes of time so we disable it. + # TODO(shenhan): keep '-sandbox' for a while for compatibility, then remove + # it after a while. + features = ('nostrip userpriv userfetch -usersandbox -sandbox noclean ' + '-buildpkg') + env['FEATURES'] = features + + if self._incremental: + env['FEATURES'] += ' keepwork' + + if 'USE' in env: + env['USE'] += ' multislot mounted_%s' % self._name + else: + env['USE'] = 'multislot mounted_%s' % self._name + + # Disable ccache in our compilers. cache may be problematic for us. + # It ignores compiler environments settings and it is not clear if + # the cache hit algorithm verifies all the compiler binaries or + # just the driver. + if self._name == 'gcc' and not self._gcc_enable_ccache: + env['USE'] += ' -wrapper_ccache' + + env['%s_SOURCE_PATH' % self._name.upper()] = ( + os.path.join('/', self._chroot_source_path)) + env['ACCEPT_KEYWORDS'] = '~*' + env_string = ' '.join(["%s=\"%s\"" % var for var in env.items()]) + command = 'emerge =cross-%s/%s-9999' % (self._ctarget, self._name) + full_command = 'sudo %s %s' % (env_string, command) + rv = self._ce.ChrootRunCommand(self._chromeos_root, full_command) + if rv != 0: + return rv + if self._name == 'gcc': + command = ('sudo cp -r /usr/lib/gcc/%s %s' % + (self._ctarget, self._gcc_libs_dest)) + rv = self._ce.ChrootRunCommand(self._chromeos_root, command) + return rv + + def MoveMaskFile(self): + self._new_mask_file = None + if os.path.isfile(self._mask_file): + self._new_mask_file = tempfile.mktemp() + command = 'sudo mv %s %s' % (self._mask_file, self._new_mask_file) + self._ce.RunCommand(command) + + def UnMoveMaskFile(self): + if self._new_mask_file: + command = 'sudo mv %s %s' % (self._new_mask_file, self._mask_file) + self._ce.RunCommand(command) + + +def Main(argv): + """The main function.""" + # Common initializations + parser = argparse.ArgumentParser() + parser.add_argument('-c', + '--chromeos_root', + dest='chromeos_root', + default='../../', + help=('ChromeOS root checkout directory' + ' uses ../.. if none given.')) + parser.add_argument('-g', + '--gcc_dir', + dest='gcc_dir', + help='The directory where gcc resides.') + parser.add_argument('--binutils_dir', + dest='binutils_dir', + help='The directory where binutils resides.') + parser.add_argument('-x', + '--gdb_dir', + dest='gdb_dir', + help='The directory where gdb resides.') + parser.add_argument('-b', + '--board', + dest='board', + default='x86-alex', + help='The target board.') + parser.add_argument('-n', + '--noincremental', + dest='noincremental', + default=False, + action='store_true', + help='Use FEATURES=keepwork to do incremental builds.') + parser.add_argument('--cflags', + dest='cflags', + default='', + help='Build a compiler with specified CFLAGS') + parser.add_argument('--cxxflags', + dest='cxxflags', + default='', + help='Build a compiler with specified CXXFLAGS') + parser.add_argument('--cflags_for_target', + dest='cflags_for_target', + default='', + help='Build the target libraries with specified flags') + parser.add_argument('--cxxflags_for_target', + dest='cxxflags_for_target', + default='', + help='Build the target libraries with specified flags') + parser.add_argument('--ldflags', + dest='ldflags', + default='', + help='Build a compiler with specified LDFLAGS') + parser.add_argument('-d', + '--debug', + dest='debug', + default=False, + action='store_true', + help='Build a compiler with -g3 -O0 appended to both' + ' CFLAGS and CXXFLAGS.') + parser.add_argument('-m', + '--mount_only', + dest='mount_only', + default=False, + action='store_true', + help='Just mount the tool directories.') + parser.add_argument('-u', + '--unmount_only', + dest='unmount_only', + default=False, + action='store_true', + help='Just unmount the tool directories.') + parser.add_argument('--extra_use_flags', + dest='extra_use_flags', + default='', + help='Extra flag for USE, to be passed to the ebuild. ' + "('multislot' and 'mounted_<tool>' are always passed.)") + parser.add_argument('--gcc_enable_ccache', + dest='gcc_enable_ccache', + default=False, + action='store_true', + help='Enable ccache for the gcc invocations') + + options = parser.parse_args(argv) + + chromeos_root = misc.CanonicalizePath(options.chromeos_root) + if options.gcc_dir: + gcc_dir = misc.CanonicalizePath(options.gcc_dir) + assert gcc_dir and os.path.isdir(gcc_dir), 'gcc_dir does not exist!' + if options.binutils_dir: + binutils_dir = misc.CanonicalizePath(options.binutils_dir) + assert os.path.isdir(binutils_dir), 'binutils_dir does not exist!' + if options.gdb_dir: + gdb_dir = misc.CanonicalizePath(options.gdb_dir) + assert os.path.isdir(gdb_dir), 'gdb_dir does not exist!' + if options.unmount_only: + options.mount_only = False + elif options.mount_only: + options.unmount_only = False + build_env = {} + if options.cflags: + build_env['CFLAGS'] = '`portageq envvar CFLAGS` ' + options.cflags + if options.cxxflags: + build_env['CXXFLAGS'] = '`portageq envvar CXXFLAGS` ' + options.cxxflags + if options.cflags_for_target: + build_env['CFLAGS_FOR_TARGET'] = options.cflags_for_target + if options.cxxflags_for_target: + build_env['CXXFLAGS_FOR_TARGET'] = options.cxxflags_for_target + if options.ldflags: + build_env['LDFLAGS'] = options.ldflags + if options.debug: + debug_flags = '-g3 -O0' + if 'CFLAGS' in build_env: + build_env['CFLAGS'] += ' %s' % (debug_flags) + else: + build_env['CFLAGS'] = debug_flags + if 'CXXFLAGS' in build_env: + build_env['CXXFLAGS'] += ' %s' % (debug_flags) + else: + build_env['CXXFLAGS'] = debug_flags + if options.extra_use_flags: + build_env['USE'] = options.extra_use_flags + + # Create toolchain parts + toolchain_parts = {} + for board in options.board.split(','): + if options.gcc_dir: + tp = ToolchainPart('gcc', gcc_dir, chromeos_root, board, + not options.noincremental, build_env, + options.gcc_enable_ccache) + toolchain_parts[tp.tag] = tp + tp.RunSetupBoardIfNecessary() + if options.binutils_dir: + tp = ToolchainPart('binutils', binutils_dir, chromeos_root, board, + not options.noincremental, build_env) + toolchain_parts[tp.tag] = tp + tp.RunSetupBoardIfNecessary() + if options.gdb_dir: + tp = ToolchainPart('gdb', gdb_dir, chromeos_root, board, + not options.noincremental, build_env) + toolchain_parts[tp.tag] = tp + tp.RunSetupBoardIfNecessary() + + rv = 0 + try: + for tag in toolchain_parts: + tp = toolchain_parts[tag] + if options.mount_only or options.unmount_only: + tp.MountSources(options.unmount_only) + else: + rv = rv + tp.Build() + finally: + print('Exiting...') + return rv + + +if __name__ == '__main__': + retval = Main(sys.argv[1:]) + sys.exit(retval) |