mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-03 21:48:15 +00:00 
			
		
		
		
	There are a few make options such as BUILD_TAG which can be provided when building U-Boot. Provide a way for buildman to pass these flags to make also. The flags should be in a [make-flags] section and arranged by target name (the 'target' column in boards.cfg. See the README for more details. Signed-off-by: Simon Glass <sjg@chromium.org>
		
			
				
	
	
		
			1431 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1431 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# Copyright (c) 2013 The Chromium OS Authors.
 | 
						|
#
 | 
						|
# Bloat-o-meter code used here Copyright 2004 Matt Mackall <mpm@selenic.com>
 | 
						|
#
 | 
						|
# SPDX-License-Identifier:	GPL-2.0+
 | 
						|
#
 | 
						|
 | 
						|
import collections
 | 
						|
import errno
 | 
						|
from datetime import datetime, timedelta
 | 
						|
import glob
 | 
						|
import os
 | 
						|
import re
 | 
						|
import Queue
 | 
						|
import shutil
 | 
						|
import string
 | 
						|
import sys
 | 
						|
import threading
 | 
						|
import time
 | 
						|
 | 
						|
import command
 | 
						|
import gitutil
 | 
						|
import terminal
 | 
						|
import toolchain
 | 
						|
 | 
						|
 | 
						|
"""
 | 
						|
Theory of Operation
 | 
						|
 | 
						|
Please see README for user documentation, and you should be familiar with
 | 
						|
that before trying to make sense of this.
 | 
						|
 | 
						|
Buildman works by keeping the machine as busy as possible, building different
 | 
						|
commits for different boards on multiple CPUs at once.
 | 
						|
 | 
						|
The source repo (self.git_dir) contains all the commits to be built. Each
 | 
						|
thread works on a single board at a time. It checks out the first commit,
 | 
						|
configures it for that board, then builds it. Then it checks out the next
 | 
						|
commit and builds it (typically without re-configuring). When it runs out
 | 
						|
of commits, it gets another job from the builder and starts again with that
 | 
						|
board.
 | 
						|
 | 
						|
Clearly the builder threads could work either way - they could check out a
 | 
						|
commit and then built it for all boards. Using separate directories for each
 | 
						|
commit/board pair they could leave their build product around afterwards
 | 
						|
also.
 | 
						|
 | 
						|
The intent behind building a single board for multiple commits, is to make
 | 
						|
use of incremental builds. Since each commit is built incrementally from
 | 
						|
the previous one, builds are faster. Reconfiguring for a different board
 | 
						|
removes all intermediate object files.
 | 
						|
 | 
						|
Many threads can be working at once, but each has its own working directory.
 | 
						|
When a thread finishes a build, it puts the output files into a result
 | 
						|
directory.
 | 
						|
 | 
						|
The base directory used by buildman is normally '../<branch>', i.e.
 | 
						|
a directory higher than the source repository and named after the branch
 | 
						|
being built.
 | 
						|
 | 
						|
Within the base directory, we have one subdirectory for each commit. Within
 | 
						|
that is one subdirectory for each board. Within that is the build output for
 | 
						|
that commit/board combination.
 | 
						|
 | 
						|
Buildman also create working directories for each thread, in a .bm-work/
 | 
						|
subdirectory in the base dir.
 | 
						|
 | 
						|
As an example, say we are building branch 'us-net' for boards 'sandbox' and
 | 
						|
'seaboard', and say that us-net has two commits. We will have directories
 | 
						|
like this:
 | 
						|
 | 
						|
us-net/             base directory
 | 
						|
    01_of_02_g4ed4ebc_net--Add-tftp-speed-/
 | 
						|
        sandbox/
 | 
						|
            u-boot.bin
 | 
						|
        seaboard/
 | 
						|
            u-boot.bin
 | 
						|
    02_of_02_g4ed4ebc_net--Check-tftp-comp/
 | 
						|
        sandbox/
 | 
						|
            u-boot.bin
 | 
						|
        seaboard/
 | 
						|
            u-boot.bin
 | 
						|
    .bm-work/
 | 
						|
        00/         working directory for thread 0 (contains source checkout)
 | 
						|
            build/  build output
 | 
						|
        01/         working directory for thread 1
 | 
						|
            build/  build output
 | 
						|
        ...
 | 
						|
u-boot/             source directory
 | 
						|
    .git/           repository
 | 
						|
"""
 | 
						|
 | 
						|
# Possible build outcomes
 | 
						|
OUTCOME_OK, OUTCOME_WARNING, OUTCOME_ERROR, OUTCOME_UNKNOWN = range(4)
 | 
						|
 | 
						|
# Translate a commit subject into a valid filename
 | 
						|
trans_valid_chars = string.maketrans("/: ", "---")
 | 
						|
 | 
						|
 | 
						|
def Mkdir(dirname):
 | 
						|
    """Make a directory if it doesn't already exist.
 | 
						|
 | 
						|
    Args:
 | 
						|
        dirname: Directory to create
 | 
						|
    """
 | 
						|
    try:
 | 
						|
        os.mkdir(dirname)
 | 
						|
    except OSError as err:
 | 
						|
        if err.errno == errno.EEXIST:
 | 
						|
            pass
 | 
						|
        else:
 | 
						|
            raise
 | 
						|
 | 
						|
class BuilderJob:
 | 
						|
    """Holds information about a job to be performed by a thread
 | 
						|
 | 
						|
    Members:
 | 
						|
        board: Board object to build
 | 
						|
        commits: List of commit options to build.
 | 
						|
    """
 | 
						|
    def __init__(self):
 | 
						|
        self.board = None
 | 
						|
        self.commits = []
 | 
						|
 | 
						|
 | 
						|
class ResultThread(threading.Thread):
 | 
						|
    """This thread processes results from builder threads.
 | 
						|
 | 
						|
    It simply passes the results on to the builder. There is only one
 | 
						|
    result thread, and this helps to serialise the build output.
 | 
						|
    """
 | 
						|
    def __init__(self, builder):
 | 
						|
        """Set up a new result thread
 | 
						|
 | 
						|
        Args:
 | 
						|
            builder: Builder which will be sent each result
 | 
						|
        """
 | 
						|
        threading.Thread.__init__(self)
 | 
						|
        self.builder = builder
 | 
						|
 | 
						|
    def run(self):
 | 
						|
        """Called to start up the result thread.
 | 
						|
 | 
						|
        We collect the next result job and pass it on to the build.
 | 
						|
        """
 | 
						|
        while True:
 | 
						|
            result = self.builder.out_queue.get()
 | 
						|
            self.builder.ProcessResult(result)
 | 
						|
            self.builder.out_queue.task_done()
 | 
						|
 | 
						|
 | 
						|
class BuilderThread(threading.Thread):
 | 
						|
    """This thread builds U-Boot for a particular board.
 | 
						|
 | 
						|
    An input queue provides each new job. We run 'make' to build U-Boot
 | 
						|
    and then pass the results on to the output queue.
 | 
						|
 | 
						|
    Members:
 | 
						|
        builder: The builder which contains information we might need
 | 
						|
        thread_num: Our thread number (0-n-1), used to decide on a
 | 
						|
                temporary directory
 | 
						|
    """
 | 
						|
    def __init__(self, builder, thread_num):
 | 
						|
        """Set up a new builder thread"""
 | 
						|
        threading.Thread.__init__(self)
 | 
						|
        self.builder = builder
 | 
						|
        self.thread_num = thread_num
 | 
						|
 | 
						|
    def Make(self, commit, brd, stage, cwd, *args, **kwargs):
 | 
						|
        """Run 'make' on a particular commit and board.
 | 
						|
 | 
						|
        The source code will already be checked out, so the 'commit'
 | 
						|
        argument is only for information.
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit: Commit object that is being built
 | 
						|
            brd: Board object that is being built
 | 
						|
            stage: Stage of the build. Valid stages are:
 | 
						|
                        distclean - can be called to clean source
 | 
						|
                        config - called to configure for a board
 | 
						|
                        build - the main make invocation - it does the build
 | 
						|
            args: A list of arguments to pass to 'make'
 | 
						|
            kwargs: A list of keyword arguments to pass to command.RunPipe()
 | 
						|
 | 
						|
        Returns:
 | 
						|
            CommandResult object
 | 
						|
        """
 | 
						|
        return self.builder.do_make(commit, brd, stage, cwd, *args,
 | 
						|
                **kwargs)
 | 
						|
 | 
						|
    def RunCommit(self, commit_upto, brd, work_dir, do_config, force_build):
 | 
						|
        """Build a particular commit.
 | 
						|
 | 
						|
        If the build is already done, and we are not forcing a build, we skip
 | 
						|
        the build and just return the previously-saved results.
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to build (0...n-1)
 | 
						|
            brd: Board object to build
 | 
						|
            work_dir: Directory to which the source will be checked out
 | 
						|
            do_config: True to run a make <board>_config on the source
 | 
						|
            force_build: Force a build even if one was previously done
 | 
						|
 | 
						|
        Returns:
 | 
						|
            tuple containing:
 | 
						|
                - CommandResult object containing the results of the build
 | 
						|
                - boolean indicating whether 'make config' is still needed
 | 
						|
        """
 | 
						|
        # Create a default result - it will be overwritte by the call to
 | 
						|
        # self.Make() below, in the event that we do a build.
 | 
						|
        result = command.CommandResult()
 | 
						|
        result.return_code = 0
 | 
						|
        out_dir = os.path.join(work_dir, 'build')
 | 
						|
 | 
						|
        # Check if the job was already completed last time
 | 
						|
        done_file = self.builder.GetDoneFile(commit_upto, brd.target)
 | 
						|
        result.already_done = os.path.exists(done_file)
 | 
						|
        if result.already_done and not force_build:
 | 
						|
            # Get the return code from that build and use it
 | 
						|
            with open(done_file, 'r') as fd:
 | 
						|
                result.return_code = int(fd.readline())
 | 
						|
            err_file = self.builder.GetErrFile(commit_upto, brd.target)
 | 
						|
            if os.path.exists(err_file) and os.stat(err_file).st_size:
 | 
						|
                result.stderr = 'bad'
 | 
						|
        else:
 | 
						|
            # We are going to have to build it. First, get a toolchain
 | 
						|
            if not self.toolchain:
 | 
						|
                try:
 | 
						|
                    self.toolchain = self.builder.toolchains.Select(brd.arch)
 | 
						|
                except ValueError as err:
 | 
						|
                    result.return_code = 10
 | 
						|
                    result.stdout = ''
 | 
						|
                    result.stderr = str(err)
 | 
						|
                    # TODO(sjg@chromium.org): This gets swallowed, but needs
 | 
						|
                    # to be reported.
 | 
						|
 | 
						|
            if self.toolchain:
 | 
						|
                # Checkout the right commit
 | 
						|
                if commit_upto is not None:
 | 
						|
                    commit = self.builder.commits[commit_upto]
 | 
						|
                    if self.builder.checkout:
 | 
						|
                        git_dir = os.path.join(work_dir, '.git')
 | 
						|
                        gitutil.Checkout(commit.hash, git_dir, work_dir,
 | 
						|
                                         force=True)
 | 
						|
                else:
 | 
						|
                    commit = self.builder.commit # Ick, fix this for BuildCommits()
 | 
						|
 | 
						|
                # Set up the environment and command line
 | 
						|
                env = self.toolchain.MakeEnvironment()
 | 
						|
                Mkdir(out_dir)
 | 
						|
                args = ['O=build', '-s']
 | 
						|
                if self.builder.num_jobs is not None:
 | 
						|
                    args.extend(['-j', str(self.builder.num_jobs)])
 | 
						|
                config_args = ['%s_config' % brd.target]
 | 
						|
                config_out = ''
 | 
						|
                args.extend(self.builder.toolchains.GetMakeArguments(brd))
 | 
						|
 | 
						|
                # If we need to reconfigure, do that now
 | 
						|
                if do_config:
 | 
						|
                    result = self.Make(commit, brd, 'distclean', work_dir,
 | 
						|
                            'distclean', *args, env=env)
 | 
						|
                    result = self.Make(commit, brd, 'config', work_dir,
 | 
						|
                            *(args + config_args), env=env)
 | 
						|
                    config_out = result.combined
 | 
						|
                    do_config = False   # No need to configure next time
 | 
						|
                if result.return_code == 0:
 | 
						|
                    result = self.Make(commit, brd, 'build', work_dir, *args,
 | 
						|
                            env=env)
 | 
						|
                    result.stdout = config_out + result.stdout
 | 
						|
            else:
 | 
						|
                result.return_code = 1
 | 
						|
                result.stderr = 'No tool chain for %s\n' % brd.arch
 | 
						|
            result.already_done = False
 | 
						|
 | 
						|
        result.toolchain = self.toolchain
 | 
						|
        result.brd = brd
 | 
						|
        result.commit_upto = commit_upto
 | 
						|
        result.out_dir = out_dir
 | 
						|
        return result, do_config
 | 
						|
 | 
						|
    def _WriteResult(self, result, keep_outputs):
 | 
						|
        """Write a built result to the output directory.
 | 
						|
 | 
						|
        Args:
 | 
						|
            result: CommandResult object containing result to write
 | 
						|
            keep_outputs: True to store the output binaries, False
 | 
						|
                to delete them
 | 
						|
        """
 | 
						|
        # Fatal error
 | 
						|
        if result.return_code < 0:
 | 
						|
            return
 | 
						|
 | 
						|
        # Aborted?
 | 
						|
        if result.stderr and 'No child processes' in result.stderr:
 | 
						|
            return
 | 
						|
 | 
						|
        if result.already_done:
 | 
						|
            return
 | 
						|
 | 
						|
        # Write the output and stderr
 | 
						|
        output_dir = self.builder._GetOutputDir(result.commit_upto)
 | 
						|
        Mkdir(output_dir)
 | 
						|
        build_dir = self.builder.GetBuildDir(result.commit_upto,
 | 
						|
                result.brd.target)
 | 
						|
        Mkdir(build_dir)
 | 
						|
 | 
						|
        outfile = os.path.join(build_dir, 'log')
 | 
						|
        with open(outfile, 'w') as fd:
 | 
						|
            if result.stdout:
 | 
						|
                fd.write(result.stdout)
 | 
						|
 | 
						|
        errfile = self.builder.GetErrFile(result.commit_upto,
 | 
						|
                result.brd.target)
 | 
						|
        if result.stderr:
 | 
						|
            with open(errfile, 'w') as fd:
 | 
						|
                fd.write(result.stderr)
 | 
						|
        elif os.path.exists(errfile):
 | 
						|
            os.remove(errfile)
 | 
						|
 | 
						|
        if result.toolchain:
 | 
						|
            # Write the build result and toolchain information.
 | 
						|
            done_file = self.builder.GetDoneFile(result.commit_upto,
 | 
						|
                    result.brd.target)
 | 
						|
            with open(done_file, 'w') as fd:
 | 
						|
                fd.write('%s' % result.return_code)
 | 
						|
            with open(os.path.join(build_dir, 'toolchain'), 'w') as fd:
 | 
						|
                print >>fd, 'gcc', result.toolchain.gcc
 | 
						|
                print >>fd, 'path', result.toolchain.path
 | 
						|
                print >>fd, 'cross', result.toolchain.cross
 | 
						|
                print >>fd, 'arch', result.toolchain.arch
 | 
						|
                fd.write('%s' % result.return_code)
 | 
						|
 | 
						|
            with open(os.path.join(build_dir, 'toolchain'), 'w') as fd:
 | 
						|
                print >>fd, 'gcc', result.toolchain.gcc
 | 
						|
                print >>fd, 'path', result.toolchain.path
 | 
						|
 | 
						|
            # Write out the image and function size information and an objdump
 | 
						|
            env = result.toolchain.MakeEnvironment()
 | 
						|
            lines = []
 | 
						|
            for fname in ['u-boot', 'spl/u-boot-spl']:
 | 
						|
                cmd = ['%snm' % self.toolchain.cross, '--size-sort', fname]
 | 
						|
                nm_result = command.RunPipe([cmd], capture=True,
 | 
						|
                        capture_stderr=True, cwd=result.out_dir,
 | 
						|
                        raise_on_error=False, env=env)
 | 
						|
                if nm_result.stdout:
 | 
						|
                    nm = self.builder.GetFuncSizesFile(result.commit_upto,
 | 
						|
                                    result.brd.target, fname)
 | 
						|
                    with open(nm, 'w') as fd:
 | 
						|
                        print >>fd, nm_result.stdout,
 | 
						|
 | 
						|
                cmd = ['%sobjdump' % self.toolchain.cross, '-h', fname]
 | 
						|
                dump_result = command.RunPipe([cmd], capture=True,
 | 
						|
                        capture_stderr=True, cwd=result.out_dir,
 | 
						|
                        raise_on_error=False, env=env)
 | 
						|
                rodata_size = ''
 | 
						|
                if dump_result.stdout:
 | 
						|
                    objdump = self.builder.GetObjdumpFile(result.commit_upto,
 | 
						|
                                    result.brd.target, fname)
 | 
						|
                    with open(objdump, 'w') as fd:
 | 
						|
                        print >>fd, dump_result.stdout,
 | 
						|
                    for line in dump_result.stdout.splitlines():
 | 
						|
                        fields = line.split()
 | 
						|
                        if len(fields) > 5 and fields[1] == '.rodata':
 | 
						|
                            rodata_size = fields[2]
 | 
						|
 | 
						|
                cmd = ['%ssize' % self.toolchain.cross, fname]
 | 
						|
                size_result = command.RunPipe([cmd], capture=True,
 | 
						|
                        capture_stderr=True, cwd=result.out_dir,
 | 
						|
                        raise_on_error=False, env=env)
 | 
						|
                if size_result.stdout:
 | 
						|
                    lines.append(size_result.stdout.splitlines()[1] + ' ' +
 | 
						|
                                 rodata_size)
 | 
						|
 | 
						|
            # Write out the image sizes file. This is similar to the output
 | 
						|
            # of binutil's 'size' utility, but it omits the header line and
 | 
						|
            # adds an additional hex value at the end of each line for the
 | 
						|
            # rodata size
 | 
						|
            if len(lines):
 | 
						|
                sizes = self.builder.GetSizesFile(result.commit_upto,
 | 
						|
                                result.brd.target)
 | 
						|
                with open(sizes, 'w') as fd:
 | 
						|
                    print >>fd, '\n'.join(lines)
 | 
						|
 | 
						|
        # Now write the actual build output
 | 
						|
        if keep_outputs:
 | 
						|
            patterns = ['u-boot', '*.bin', 'u-boot.dtb', '*.map',
 | 
						|
                        'include/autoconf.mk', 'spl/u-boot-spl',
 | 
						|
                        'spl/u-boot-spl.bin']
 | 
						|
            for pattern in patterns:
 | 
						|
                file_list = glob.glob(os.path.join(result.out_dir, pattern))
 | 
						|
                for fname in file_list:
 | 
						|
                    shutil.copy(fname, build_dir)
 | 
						|
 | 
						|
 | 
						|
    def RunJob(self, job):
 | 
						|
        """Run a single job
 | 
						|
 | 
						|
        A job consists of a building a list of commits for a particular board.
 | 
						|
 | 
						|
        Args:
 | 
						|
            job: Job to build
 | 
						|
        """
 | 
						|
        brd = job.board
 | 
						|
        work_dir = self.builder.GetThreadDir(self.thread_num)
 | 
						|
        self.toolchain = None
 | 
						|
        if job.commits:
 | 
						|
            # Run 'make board_config' on the first commit
 | 
						|
            do_config = True
 | 
						|
            commit_upto  = 0
 | 
						|
            force_build = False
 | 
						|
            for commit_upto in range(0, len(job.commits), job.step):
 | 
						|
                result, request_config = self.RunCommit(commit_upto, brd,
 | 
						|
                        work_dir, do_config,
 | 
						|
                        force_build or self.builder.force_build)
 | 
						|
                failed = result.return_code or result.stderr
 | 
						|
                if failed and not do_config:
 | 
						|
                    # If our incremental build failed, try building again
 | 
						|
                    # with a reconfig.
 | 
						|
                    if self.builder.force_config_on_failure:
 | 
						|
                        result, request_config = self.RunCommit(commit_upto,
 | 
						|
                            brd, work_dir, True, True)
 | 
						|
                do_config = request_config
 | 
						|
 | 
						|
                # If we built that commit, then config is done. But if we got
 | 
						|
                # an warning, reconfig next time to force it to build the same
 | 
						|
                # files that created warnings this time. Otherwise an
 | 
						|
                # incremental build may not build the same file, and we will
 | 
						|
                # think that the warning has gone away.
 | 
						|
                # We could avoid this by using -Werror everywhere...
 | 
						|
                # For errors, the problem doesn't happen, since presumably
 | 
						|
                # the build stopped and didn't generate output, so will retry
 | 
						|
                # that file next time. So we could detect warnings and deal
 | 
						|
                # with them specially here. For now, we just reconfigure if
 | 
						|
                # anything goes work.
 | 
						|
                # Of course this is substantially slower if there are build
 | 
						|
                # errors/warnings (e.g. 2-3x slower even if only 10% of builds
 | 
						|
                # have problems).
 | 
						|
                if (failed and not result.already_done and not do_config and
 | 
						|
                        self.builder.force_config_on_failure):
 | 
						|
                    # If this build failed, try the next one with a
 | 
						|
                    # reconfigure.
 | 
						|
                    # Sometimes if the board_config.h file changes it can mess
 | 
						|
                    # with dependencies, and we get:
 | 
						|
                    # make: *** No rule to make target `include/autoconf.mk',
 | 
						|
                    #     needed by `depend'.
 | 
						|
                    do_config = True
 | 
						|
                    force_build = True
 | 
						|
                else:
 | 
						|
                    force_build = False
 | 
						|
                    if self.builder.force_config_on_failure:
 | 
						|
                        if failed:
 | 
						|
                            do_config = True
 | 
						|
                    result.commit_upto = commit_upto
 | 
						|
                    if result.return_code < 0:
 | 
						|
                        raise ValueError('Interrupt')
 | 
						|
 | 
						|
                # We have the build results, so output the result
 | 
						|
                self._WriteResult(result, job.keep_outputs)
 | 
						|
                self.builder.out_queue.put(result)
 | 
						|
        else:
 | 
						|
            # Just build the currently checked-out build
 | 
						|
            result = self.RunCommit(None, True)
 | 
						|
            result.commit_upto = self.builder.upto
 | 
						|
            self.builder.out_queue.put(result)
 | 
						|
 | 
						|
    def run(self):
 | 
						|
        """Our thread's run function
 | 
						|
 | 
						|
        This thread picks a job from the queue, runs it, and then goes to the
 | 
						|
        next job.
 | 
						|
        """
 | 
						|
        alive = True
 | 
						|
        while True:
 | 
						|
            job = self.builder.queue.get()
 | 
						|
            try:
 | 
						|
                if self.builder.active and alive:
 | 
						|
                    self.RunJob(job)
 | 
						|
            except Exception as err:
 | 
						|
                alive = False
 | 
						|
                print err
 | 
						|
            self.builder.queue.task_done()
 | 
						|
 | 
						|
 | 
						|
class Builder:
 | 
						|
    """Class for building U-Boot for a particular commit.
 | 
						|
 | 
						|
    Public members: (many should ->private)
 | 
						|
        active: True if the builder is active and has not been stopped
 | 
						|
        already_done: Number of builds already completed
 | 
						|
        base_dir: Base directory to use for builder
 | 
						|
        checkout: True to check out source, False to skip that step.
 | 
						|
            This is used for testing.
 | 
						|
        col: terminal.Color() object
 | 
						|
        count: Number of commits to build
 | 
						|
        do_make: Method to call to invoke Make
 | 
						|
        fail: Number of builds that failed due to error
 | 
						|
        force_build: Force building even if a build already exists
 | 
						|
        force_config_on_failure: If a commit fails for a board, disable
 | 
						|
            incremental building for the next commit we build for that
 | 
						|
            board, so that we will see all warnings/errors again.
 | 
						|
        git_dir: Git directory containing source repository
 | 
						|
        last_line_len: Length of the last line we printed (used for erasing
 | 
						|
            it with new progress information)
 | 
						|
        num_jobs: Number of jobs to run at once (passed to make as -j)
 | 
						|
        num_threads: Number of builder threads to run
 | 
						|
        out_queue: Queue of results to process
 | 
						|
        re_make_err: Compiled regular expression for ignore_lines
 | 
						|
        queue: Queue of jobs to run
 | 
						|
        threads: List of active threads
 | 
						|
        toolchains: Toolchains object to use for building
 | 
						|
        upto: Current commit number we are building (0.count-1)
 | 
						|
        warned: Number of builds that produced at least one warning
 | 
						|
 | 
						|
    Private members:
 | 
						|
        _base_board_dict: Last-summarised Dict of boards
 | 
						|
        _base_err_lines: Last-summarised list of errors
 | 
						|
        _build_period_us: Time taken for a single build (float object).
 | 
						|
        _complete_delay: Expected delay until completion (timedelta)
 | 
						|
        _next_delay_update: Next time we plan to display a progress update
 | 
						|
                (datatime)
 | 
						|
        _show_unknown: Show unknown boards (those not built) in summary
 | 
						|
        _timestamps: List of timestamps for the completion of the last
 | 
						|
            last _timestamp_count builds. Each is a datetime object.
 | 
						|
        _timestamp_count: Number of timestamps to keep in our list.
 | 
						|
        _working_dir: Base working directory containing all threads
 | 
						|
    """
 | 
						|
    class Outcome:
 | 
						|
        """Records a build outcome for a single make invocation
 | 
						|
 | 
						|
        Public Members:
 | 
						|
            rc: Outcome value (OUTCOME_...)
 | 
						|
            err_lines: List of error lines or [] if none
 | 
						|
            sizes: Dictionary of image size information, keyed by filename
 | 
						|
                - Each value is itself a dictionary containing
 | 
						|
                    values for 'text', 'data' and 'bss', being the integer
 | 
						|
                    size in bytes of each section.
 | 
						|
            func_sizes: Dictionary keyed by filename - e.g. 'u-boot'. Each
 | 
						|
                    value is itself a dictionary:
 | 
						|
                        key: function name
 | 
						|
                        value: Size of function in bytes
 | 
						|
        """
 | 
						|
        def __init__(self, rc, err_lines, sizes, func_sizes):
 | 
						|
            self.rc = rc
 | 
						|
            self.err_lines = err_lines
 | 
						|
            self.sizes = sizes
 | 
						|
            self.func_sizes = func_sizes
 | 
						|
 | 
						|
    def __init__(self, toolchains, base_dir, git_dir, num_threads, num_jobs,
 | 
						|
                 checkout=True, show_unknown=True, step=1):
 | 
						|
        """Create a new Builder object
 | 
						|
 | 
						|
        Args:
 | 
						|
            toolchains: Toolchains object to use for building
 | 
						|
            base_dir: Base directory to use for builder
 | 
						|
            git_dir: Git directory containing source repository
 | 
						|
            num_threads: Number of builder threads to run
 | 
						|
            num_jobs: Number of jobs to run at once (passed to make as -j)
 | 
						|
            checkout: True to check out source, False to skip that step.
 | 
						|
                This is used for testing.
 | 
						|
            show_unknown: Show unknown boards (those not built) in summary
 | 
						|
            step: 1 to process every commit, n to process every nth commit
 | 
						|
        """
 | 
						|
        self.toolchains = toolchains
 | 
						|
        self.base_dir = base_dir
 | 
						|
        self._working_dir = os.path.join(base_dir, '.bm-work')
 | 
						|
        self.threads = []
 | 
						|
        self.active = True
 | 
						|
        self.do_make = self.Make
 | 
						|
        self.checkout = checkout
 | 
						|
        self.num_threads = num_threads
 | 
						|
        self.num_jobs = num_jobs
 | 
						|
        self.already_done = 0
 | 
						|
        self.force_build = False
 | 
						|
        self.git_dir = git_dir
 | 
						|
        self._show_unknown = show_unknown
 | 
						|
        self._timestamp_count = 10
 | 
						|
        self._build_period_us = None
 | 
						|
        self._complete_delay = None
 | 
						|
        self._next_delay_update = datetime.now()
 | 
						|
        self.force_config_on_failure = True
 | 
						|
        self._step = step
 | 
						|
 | 
						|
        self.col = terminal.Color()
 | 
						|
 | 
						|
        self.queue = Queue.Queue()
 | 
						|
        self.out_queue = Queue.Queue()
 | 
						|
        for i in range(self.num_threads):
 | 
						|
            t = BuilderThread(self, i)
 | 
						|
            t.setDaemon(True)
 | 
						|
            t.start()
 | 
						|
            self.threads.append(t)
 | 
						|
 | 
						|
        self.last_line_len = 0
 | 
						|
        t = ResultThread(self)
 | 
						|
        t.setDaemon(True)
 | 
						|
        t.start()
 | 
						|
        self.threads.append(t)
 | 
						|
 | 
						|
        ignore_lines = ['(make.*Waiting for unfinished)', '(Segmentation fault)']
 | 
						|
        self.re_make_err = re.compile('|'.join(ignore_lines))
 | 
						|
 | 
						|
    def __del__(self):
 | 
						|
        """Get rid of all threads created by the builder"""
 | 
						|
        for t in self.threads:
 | 
						|
            del t
 | 
						|
 | 
						|
    def _AddTimestamp(self):
 | 
						|
        """Add a new timestamp to the list and record the build period.
 | 
						|
 | 
						|
        The build period is the length of time taken to perform a single
 | 
						|
        build (one board, one commit).
 | 
						|
        """
 | 
						|
        now = datetime.now()
 | 
						|
        self._timestamps.append(now)
 | 
						|
        count = len(self._timestamps)
 | 
						|
        delta = self._timestamps[-1] - self._timestamps[0]
 | 
						|
        seconds = delta.total_seconds()
 | 
						|
 | 
						|
        # If we have enough data, estimate build period (time taken for a
 | 
						|
        # single build) and therefore completion time.
 | 
						|
        if count > 1 and self._next_delay_update < now:
 | 
						|
            self._next_delay_update = now + timedelta(seconds=2)
 | 
						|
            if seconds > 0:
 | 
						|
                self._build_period = float(seconds) / count
 | 
						|
                todo = self.count - self.upto
 | 
						|
                self._complete_delay = timedelta(microseconds=
 | 
						|
                        self._build_period * todo * 1000000)
 | 
						|
                # Round it
 | 
						|
                self._complete_delay -= timedelta(
 | 
						|
                        microseconds=self._complete_delay.microseconds)
 | 
						|
 | 
						|
        if seconds > 60:
 | 
						|
            self._timestamps.popleft()
 | 
						|
            count -= 1
 | 
						|
 | 
						|
    def ClearLine(self, length):
 | 
						|
        """Clear any characters on the current line
 | 
						|
 | 
						|
        Make way for a new line of length 'length', by outputting enough
 | 
						|
        spaces to clear out the old line. Then remember the new length for
 | 
						|
        next time.
 | 
						|
 | 
						|
        Args:
 | 
						|
            length: Length of new line, in characters
 | 
						|
        """
 | 
						|
        if length < self.last_line_len:
 | 
						|
            print ' ' * (self.last_line_len - length),
 | 
						|
            print '\r',
 | 
						|
        self.last_line_len = length
 | 
						|
        sys.stdout.flush()
 | 
						|
 | 
						|
    def SelectCommit(self, commit, checkout=True):
 | 
						|
        """Checkout the selected commit for this build
 | 
						|
        """
 | 
						|
        self.commit = commit
 | 
						|
        if checkout and self.checkout:
 | 
						|
            gitutil.Checkout(commit.hash)
 | 
						|
 | 
						|
    def Make(self, commit, brd, stage, cwd, *args, **kwargs):
 | 
						|
        """Run make
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit: Commit object that is being built
 | 
						|
            brd: Board object that is being built
 | 
						|
            stage: Stage that we are at (distclean, config, build)
 | 
						|
            cwd: Directory where make should be run
 | 
						|
            args: Arguments to pass to make
 | 
						|
            kwargs: Arguments to pass to command.RunPipe()
 | 
						|
        """
 | 
						|
        cmd = ['make'] + list(args)
 | 
						|
        result = command.RunPipe([cmd], capture=True, capture_stderr=True,
 | 
						|
                cwd=cwd, raise_on_error=False, **kwargs)
 | 
						|
        return result
 | 
						|
 | 
						|
    def ProcessResult(self, result):
 | 
						|
        """Process the result of a build, showing progress information
 | 
						|
 | 
						|
        Args:
 | 
						|
            result: A CommandResult object
 | 
						|
        """
 | 
						|
        col = terminal.Color()
 | 
						|
        if result:
 | 
						|
            target = result.brd.target
 | 
						|
 | 
						|
            if result.return_code < 0:
 | 
						|
                self.active = False
 | 
						|
                command.StopAll()
 | 
						|
                return
 | 
						|
 | 
						|
            self.upto += 1
 | 
						|
            if result.return_code != 0:
 | 
						|
                self.fail += 1
 | 
						|
            elif result.stderr:
 | 
						|
                self.warned += 1
 | 
						|
            if result.already_done:
 | 
						|
                self.already_done += 1
 | 
						|
        else:
 | 
						|
            target = '(starting)'
 | 
						|
 | 
						|
        # Display separate counts for ok, warned and fail
 | 
						|
        ok = self.upto - self.warned - self.fail
 | 
						|
        line = '\r' + self.col.Color(self.col.GREEN, '%5d' % ok)
 | 
						|
        line += self.col.Color(self.col.YELLOW, '%5d' % self.warned)
 | 
						|
        line += self.col.Color(self.col.RED, '%5d' % self.fail)
 | 
						|
 | 
						|
        name = ' /%-5d  ' % self.count
 | 
						|
 | 
						|
        # Add our current completion time estimate
 | 
						|
        self._AddTimestamp()
 | 
						|
        if self._complete_delay:
 | 
						|
            name += '%s  : ' % self._complete_delay
 | 
						|
        # When building all boards for a commit, we can print a commit
 | 
						|
        # progress message.
 | 
						|
        if result and result.commit_upto is None:
 | 
						|
            name += 'commit %2d/%-3d' % (self.commit_upto + 1,
 | 
						|
                    self.commit_count)
 | 
						|
 | 
						|
        name += target
 | 
						|
        print line + name,
 | 
						|
        length = 13 + len(name)
 | 
						|
        self.ClearLine(length)
 | 
						|
 | 
						|
    def _GetOutputDir(self, commit_upto):
 | 
						|
        """Get the name of the output directory for a commit number
 | 
						|
 | 
						|
        The output directory is typically .../<branch>/<commit>.
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
        """
 | 
						|
        commit = self.commits[commit_upto]
 | 
						|
        subject = commit.subject.translate(trans_valid_chars)
 | 
						|
        commit_dir = ('%02d_of_%02d_g%s_%s' % (commit_upto + 1,
 | 
						|
                self.commit_count, commit.hash, subject[:20]))
 | 
						|
        output_dir = os.path.join(self.base_dir, commit_dir)
 | 
						|
        return output_dir
 | 
						|
 | 
						|
    def GetBuildDir(self, commit_upto, target):
 | 
						|
        """Get the name of the build directory for a commit number
 | 
						|
 | 
						|
        The build directory is typically .../<branch>/<commit>/<target>.
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
            target: Target name
 | 
						|
        """
 | 
						|
        output_dir = self._GetOutputDir(commit_upto)
 | 
						|
        return os.path.join(output_dir, target)
 | 
						|
 | 
						|
    def GetDoneFile(self, commit_upto, target):
 | 
						|
        """Get the name of the done file for a commit number
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
            target: Target name
 | 
						|
        """
 | 
						|
        return os.path.join(self.GetBuildDir(commit_upto, target), 'done')
 | 
						|
 | 
						|
    def GetSizesFile(self, commit_upto, target):
 | 
						|
        """Get the name of the sizes file for a commit number
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
            target: Target name
 | 
						|
        """
 | 
						|
        return os.path.join(self.GetBuildDir(commit_upto, target), 'sizes')
 | 
						|
 | 
						|
    def GetFuncSizesFile(self, commit_upto, target, elf_fname):
 | 
						|
        """Get the name of the funcsizes file for a commit number and ELF file
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
            target: Target name
 | 
						|
            elf_fname: Filename of elf image
 | 
						|
        """
 | 
						|
        return os.path.join(self.GetBuildDir(commit_upto, target),
 | 
						|
                            '%s.sizes' % elf_fname.replace('/', '-'))
 | 
						|
 | 
						|
    def GetObjdumpFile(self, commit_upto, target, elf_fname):
 | 
						|
        """Get the name of the objdump file for a commit number and ELF file
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
            target: Target name
 | 
						|
            elf_fname: Filename of elf image
 | 
						|
        """
 | 
						|
        return os.path.join(self.GetBuildDir(commit_upto, target),
 | 
						|
                            '%s.objdump' % elf_fname.replace('/', '-'))
 | 
						|
 | 
						|
    def GetErrFile(self, commit_upto, target):
 | 
						|
        """Get the name of the err file for a commit number
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to use (0..self.count-1)
 | 
						|
            target: Target name
 | 
						|
        """
 | 
						|
        output_dir = self.GetBuildDir(commit_upto, target)
 | 
						|
        return os.path.join(output_dir, 'err')
 | 
						|
 | 
						|
    def FilterErrors(self, lines):
 | 
						|
        """Filter out errors in which we have no interest
 | 
						|
 | 
						|
        We should probably use map().
 | 
						|
 | 
						|
        Args:
 | 
						|
            lines: List of error lines, each a string
 | 
						|
        Returns:
 | 
						|
            New list with only interesting lines included
 | 
						|
        """
 | 
						|
        out_lines = []
 | 
						|
        for line in lines:
 | 
						|
            if not self.re_make_err.search(line):
 | 
						|
                out_lines.append(line)
 | 
						|
        return out_lines
 | 
						|
 | 
						|
    def ReadFuncSizes(self, fname, fd):
 | 
						|
        """Read function sizes from the output of 'nm'
 | 
						|
 | 
						|
        Args:
 | 
						|
            fd: File containing data to read
 | 
						|
            fname: Filename we are reading from (just for errors)
 | 
						|
 | 
						|
        Returns:
 | 
						|
            Dictionary containing size of each function in bytes, indexed by
 | 
						|
            function name.
 | 
						|
        """
 | 
						|
        sym = {}
 | 
						|
        for line in fd.readlines():
 | 
						|
            try:
 | 
						|
                size, type, name = line[:-1].split()
 | 
						|
            except:
 | 
						|
                print "Invalid line in file '%s': '%s'" % (fname, line[:-1])
 | 
						|
                continue
 | 
						|
            if type in 'tTdDbB':
 | 
						|
                # function names begin with '.' on 64-bit powerpc
 | 
						|
                if '.' in name[1:]:
 | 
						|
                    name = 'static.' + name.split('.')[0]
 | 
						|
                sym[name] = sym.get(name, 0) + int(size, 16)
 | 
						|
        return sym
 | 
						|
 | 
						|
    def GetBuildOutcome(self, commit_upto, target, read_func_sizes):
 | 
						|
        """Work out the outcome of a build.
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit_upto: Commit number to check (0..n-1)
 | 
						|
            target: Target board to check
 | 
						|
            read_func_sizes: True to read function size information
 | 
						|
 | 
						|
        Returns:
 | 
						|
            Outcome object
 | 
						|
        """
 | 
						|
        done_file = self.GetDoneFile(commit_upto, target)
 | 
						|
        sizes_file = self.GetSizesFile(commit_upto, target)
 | 
						|
        sizes = {}
 | 
						|
        func_sizes = {}
 | 
						|
        if os.path.exists(done_file):
 | 
						|
            with open(done_file, 'r') as fd:
 | 
						|
                return_code = int(fd.readline())
 | 
						|
                err_lines = []
 | 
						|
                err_file = self.GetErrFile(commit_upto, target)
 | 
						|
                if os.path.exists(err_file):
 | 
						|
                    with open(err_file, 'r') as fd:
 | 
						|
                        err_lines = self.FilterErrors(fd.readlines())
 | 
						|
 | 
						|
                # Decide whether the build was ok, failed or created warnings
 | 
						|
                if return_code:
 | 
						|
                    rc = OUTCOME_ERROR
 | 
						|
                elif len(err_lines):
 | 
						|
                    rc = OUTCOME_WARNING
 | 
						|
                else:
 | 
						|
                    rc = OUTCOME_OK
 | 
						|
 | 
						|
                # Convert size information to our simple format
 | 
						|
                if os.path.exists(sizes_file):
 | 
						|
                    with open(sizes_file, 'r') as fd:
 | 
						|
                        for line in fd.readlines():
 | 
						|
                            values = line.split()
 | 
						|
                            rodata = 0
 | 
						|
                            if len(values) > 6:
 | 
						|
                                rodata = int(values[6], 16)
 | 
						|
                            size_dict = {
 | 
						|
                                'all' : int(values[0]) + int(values[1]) +
 | 
						|
                                        int(values[2]),
 | 
						|
                                'text' : int(values[0]) - rodata,
 | 
						|
                                'data' : int(values[1]),
 | 
						|
                                'bss' : int(values[2]),
 | 
						|
                                'rodata' : rodata,
 | 
						|
                            }
 | 
						|
                            sizes[values[5]] = size_dict
 | 
						|
 | 
						|
            if read_func_sizes:
 | 
						|
                pattern = self.GetFuncSizesFile(commit_upto, target, '*')
 | 
						|
                for fname in glob.glob(pattern):
 | 
						|
                    with open(fname, 'r') as fd:
 | 
						|
                        dict_name = os.path.basename(fname).replace('.sizes',
 | 
						|
                                                                    '')
 | 
						|
                        func_sizes[dict_name] = self.ReadFuncSizes(fname, fd)
 | 
						|
 | 
						|
            return Builder.Outcome(rc, err_lines, sizes, func_sizes)
 | 
						|
 | 
						|
        return Builder.Outcome(OUTCOME_UNKNOWN, [], {}, {})
 | 
						|
 | 
						|
    def GetResultSummary(self, boards_selected, commit_upto, read_func_sizes):
 | 
						|
        """Calculate a summary of the results of building a commit.
 | 
						|
 | 
						|
        Args:
 | 
						|
            board_selected: Dict containing boards to summarise
 | 
						|
            commit_upto: Commit number to summarize (0..self.count-1)
 | 
						|
            read_func_sizes: True to read function size information
 | 
						|
 | 
						|
        Returns:
 | 
						|
            Tuple:
 | 
						|
                Dict containing boards which passed building this commit.
 | 
						|
                    keyed by board.target
 | 
						|
                List containing a summary of error/warning lines
 | 
						|
        """
 | 
						|
        board_dict = {}
 | 
						|
        err_lines_summary = []
 | 
						|
 | 
						|
        for board in boards_selected.itervalues():
 | 
						|
            outcome = self.GetBuildOutcome(commit_upto, board.target,
 | 
						|
                                           read_func_sizes)
 | 
						|
            board_dict[board.target] = outcome
 | 
						|
            for err in outcome.err_lines:
 | 
						|
                if err and not err.rstrip() in err_lines_summary:
 | 
						|
                    err_lines_summary.append(err.rstrip())
 | 
						|
        return board_dict, err_lines_summary
 | 
						|
 | 
						|
    def AddOutcome(self, board_dict, arch_list, changes, char, color):
 | 
						|
        """Add an output to our list of outcomes for each architecture
 | 
						|
 | 
						|
        This simple function adds failing boards (changes) to the
 | 
						|
        relevant architecture string, so we can print the results out
 | 
						|
        sorted by architecture.
 | 
						|
 | 
						|
        Args:
 | 
						|
             board_dict: Dict containing all boards
 | 
						|
             arch_list: Dict keyed by arch name. Value is a string containing
 | 
						|
                    a list of board names which failed for that arch.
 | 
						|
             changes: List of boards to add to arch_list
 | 
						|
             color: terminal.Colour object
 | 
						|
        """
 | 
						|
        done_arch = {}
 | 
						|
        for target in changes:
 | 
						|
            if target in board_dict:
 | 
						|
                arch = board_dict[target].arch
 | 
						|
            else:
 | 
						|
                arch = 'unknown'
 | 
						|
            str = self.col.Color(color, ' ' + target)
 | 
						|
            if not arch in done_arch:
 | 
						|
                str = self.col.Color(color, char) + '  ' + str
 | 
						|
                done_arch[arch] = True
 | 
						|
            if not arch in arch_list:
 | 
						|
                arch_list[arch] = str
 | 
						|
            else:
 | 
						|
                arch_list[arch] += str
 | 
						|
 | 
						|
 | 
						|
    def ColourNum(self, num):
 | 
						|
        color = self.col.RED if num > 0 else self.col.GREEN
 | 
						|
        if num == 0:
 | 
						|
            return '0'
 | 
						|
        return self.col.Color(color, str(num))
 | 
						|
 | 
						|
    def ResetResultSummary(self, board_selected):
 | 
						|
        """Reset the results summary ready for use.
 | 
						|
 | 
						|
        Set up the base board list to be all those selected, and set the
 | 
						|
        error lines to empty.
 | 
						|
 | 
						|
        Following this, calls to PrintResultSummary() will use this
 | 
						|
        information to work out what has changed.
 | 
						|
 | 
						|
        Args:
 | 
						|
            board_selected: Dict containing boards to summarise, keyed by
 | 
						|
                board.target
 | 
						|
        """
 | 
						|
        self._base_board_dict = {}
 | 
						|
        for board in board_selected:
 | 
						|
            self._base_board_dict[board] = Builder.Outcome(0, [], [], {})
 | 
						|
        self._base_err_lines = []
 | 
						|
 | 
						|
    def PrintFuncSizeDetail(self, fname, old, new):
 | 
						|
        grow, shrink, add, remove, up, down = 0, 0, 0, 0, 0, 0
 | 
						|
        delta, common = [], {}
 | 
						|
 | 
						|
        for a in old:
 | 
						|
            if a in new:
 | 
						|
                common[a] = 1
 | 
						|
 | 
						|
        for name in old:
 | 
						|
            if name not in common:
 | 
						|
                remove += 1
 | 
						|
                down += old[name]
 | 
						|
                delta.append([-old[name], name])
 | 
						|
 | 
						|
        for name in new:
 | 
						|
            if name not in common:
 | 
						|
                add += 1
 | 
						|
                up += new[name]
 | 
						|
                delta.append([new[name], name])
 | 
						|
 | 
						|
        for name in common:
 | 
						|
                diff = new.get(name, 0) - old.get(name, 0)
 | 
						|
                if diff > 0:
 | 
						|
                    grow, up = grow + 1, up + diff
 | 
						|
                elif diff < 0:
 | 
						|
                    shrink, down = shrink + 1, down - diff
 | 
						|
                delta.append([diff, name])
 | 
						|
 | 
						|
        delta.sort()
 | 
						|
        delta.reverse()
 | 
						|
 | 
						|
        args = [add, -remove, grow, -shrink, up, -down, up - down]
 | 
						|
        if max(args) == 0:
 | 
						|
            return
 | 
						|
        args = [self.ColourNum(x) for x in args]
 | 
						|
        indent = ' ' * 15
 | 
						|
        print ('%s%s: add: %s/%s, grow: %s/%s bytes: %s/%s (%s)' %
 | 
						|
               tuple([indent, self.col.Color(self.col.YELLOW, fname)] + args))
 | 
						|
        print '%s  %-38s %7s %7s %+7s' % (indent, 'function', 'old', 'new',
 | 
						|
                                        'delta')
 | 
						|
        for diff, name in delta:
 | 
						|
            if diff:
 | 
						|
                color = self.col.RED if diff > 0 else self.col.GREEN
 | 
						|
                msg = '%s  %-38s %7s %7s %+7d' % (indent, name,
 | 
						|
                        old.get(name, '-'), new.get(name,'-'), diff)
 | 
						|
                print self.col.Color(color, msg)
 | 
						|
 | 
						|
 | 
						|
    def PrintSizeDetail(self, target_list, show_bloat):
 | 
						|
        """Show details size information for each board
 | 
						|
 | 
						|
        Args:
 | 
						|
            target_list: List of targets, each a dict containing:
 | 
						|
                    'target': Target name
 | 
						|
                    'total_diff': Total difference in bytes across all areas
 | 
						|
                    <part_name>: Difference for that part
 | 
						|
            show_bloat: Show detail for each function
 | 
						|
        """
 | 
						|
        targets_by_diff = sorted(target_list, reverse=True,
 | 
						|
        key=lambda x: x['_total_diff'])
 | 
						|
        for result in targets_by_diff:
 | 
						|
            printed_target = False
 | 
						|
            for name in sorted(result):
 | 
						|
                diff = result[name]
 | 
						|
                if name.startswith('_'):
 | 
						|
                    continue
 | 
						|
                if diff != 0:
 | 
						|
                    color = self.col.RED if diff > 0 else self.col.GREEN
 | 
						|
                msg = ' %s %+d' % (name, diff)
 | 
						|
                if not printed_target:
 | 
						|
                    print '%10s  %-15s:' % ('', result['_target']),
 | 
						|
                    printed_target = True
 | 
						|
                print self.col.Color(color, msg),
 | 
						|
            if printed_target:
 | 
						|
                print
 | 
						|
                if show_bloat:
 | 
						|
                    target = result['_target']
 | 
						|
                    outcome = result['_outcome']
 | 
						|
                    base_outcome = self._base_board_dict[target]
 | 
						|
                    for fname in outcome.func_sizes:
 | 
						|
                        self.PrintFuncSizeDetail(fname,
 | 
						|
                                                 base_outcome.func_sizes[fname],
 | 
						|
                                                 outcome.func_sizes[fname])
 | 
						|
 | 
						|
 | 
						|
    def PrintSizeSummary(self, board_selected, board_dict, show_detail,
 | 
						|
                         show_bloat):
 | 
						|
        """Print a summary of image sizes broken down by section.
 | 
						|
 | 
						|
        The summary takes the form of one line per architecture. The
 | 
						|
        line contains deltas for each of the sections (+ means the section
 | 
						|
        got bigger, - means smaller). The nunmbers are the average number
 | 
						|
        of bytes that a board in this section increased by.
 | 
						|
 | 
						|
        For example:
 | 
						|
           powerpc: (622 boards)   text -0.0
 | 
						|
          arm: (285 boards)   text -0.0
 | 
						|
          nds32: (3 boards)   text -8.0
 | 
						|
 | 
						|
        Args:
 | 
						|
            board_selected: Dict containing boards to summarise, keyed by
 | 
						|
                board.target
 | 
						|
            board_dict: Dict containing boards for which we built this
 | 
						|
                commit, keyed by board.target. The value is an Outcome object.
 | 
						|
            show_detail: Show detail for each board
 | 
						|
            show_bloat: Show detail for each function
 | 
						|
        """
 | 
						|
        arch_list = {}
 | 
						|
        arch_count = {}
 | 
						|
 | 
						|
        # Calculate changes in size for different image parts
 | 
						|
        # The previous sizes are in Board.sizes, for each board
 | 
						|
        for target in board_dict:
 | 
						|
            if target not in board_selected:
 | 
						|
                continue
 | 
						|
            base_sizes = self._base_board_dict[target].sizes
 | 
						|
            outcome = board_dict[target]
 | 
						|
            sizes = outcome.sizes
 | 
						|
 | 
						|
            # Loop through the list of images, creating a dict of size
 | 
						|
            # changes for each image/part. We end up with something like
 | 
						|
            # {'target' : 'snapper9g45, 'data' : 5, 'u-boot-spl:text' : -4}
 | 
						|
            # which means that U-Boot data increased by 5 bytes and SPL
 | 
						|
            # text decreased by 4.
 | 
						|
            err = {'_target' : target}
 | 
						|
            for image in sizes:
 | 
						|
                if image in base_sizes:
 | 
						|
                    base_image = base_sizes[image]
 | 
						|
                    # Loop through the text, data, bss parts
 | 
						|
                    for part in sorted(sizes[image]):
 | 
						|
                        diff = sizes[image][part] - base_image[part]
 | 
						|
                        col = None
 | 
						|
                        if diff:
 | 
						|
                            if image == 'u-boot':
 | 
						|
                                name = part
 | 
						|
                            else:
 | 
						|
                                name = image + ':' + part
 | 
						|
                            err[name] = diff
 | 
						|
            arch = board_selected[target].arch
 | 
						|
            if not arch in arch_count:
 | 
						|
                arch_count[arch] = 1
 | 
						|
            else:
 | 
						|
                arch_count[arch] += 1
 | 
						|
            if not sizes:
 | 
						|
                pass    # Only add to our list when we have some stats
 | 
						|
            elif not arch in arch_list:
 | 
						|
                arch_list[arch] = [err]
 | 
						|
            else:
 | 
						|
                arch_list[arch].append(err)
 | 
						|
 | 
						|
        # We now have a list of image size changes sorted by arch
 | 
						|
        # Print out a summary of these
 | 
						|
        for arch, target_list in arch_list.iteritems():
 | 
						|
            # Get total difference for each type
 | 
						|
            totals = {}
 | 
						|
            for result in target_list:
 | 
						|
                total = 0
 | 
						|
                for name, diff in result.iteritems():
 | 
						|
                    if name.startswith('_'):
 | 
						|
                        continue
 | 
						|
                    total += diff
 | 
						|
                    if name in totals:
 | 
						|
                        totals[name] += diff
 | 
						|
                    else:
 | 
						|
                        totals[name] = diff
 | 
						|
                result['_total_diff'] = total
 | 
						|
                result['_outcome'] = board_dict[result['_target']]
 | 
						|
 | 
						|
            count = len(target_list)
 | 
						|
            printed_arch = False
 | 
						|
            for name in sorted(totals):
 | 
						|
                diff = totals[name]
 | 
						|
                if diff:
 | 
						|
                    # Display the average difference in this name for this
 | 
						|
                    # architecture
 | 
						|
                    avg_diff = float(diff) / count
 | 
						|
                    color = self.col.RED if avg_diff > 0 else self.col.GREEN
 | 
						|
                    msg = ' %s %+1.1f' % (name, avg_diff)
 | 
						|
                    if not printed_arch:
 | 
						|
                        print '%10s: (for %d/%d boards)' % (arch, count,
 | 
						|
                                arch_count[arch]),
 | 
						|
                        printed_arch = True
 | 
						|
                    print self.col.Color(color, msg),
 | 
						|
 | 
						|
            if printed_arch:
 | 
						|
                print
 | 
						|
                if show_detail:
 | 
						|
                    self.PrintSizeDetail(target_list, show_bloat)
 | 
						|
 | 
						|
 | 
						|
    def PrintResultSummary(self, board_selected, board_dict, err_lines,
 | 
						|
                           show_sizes, show_detail, show_bloat):
 | 
						|
        """Compare results with the base results and display delta.
 | 
						|
 | 
						|
        Only boards mentioned in board_selected will be considered. This
 | 
						|
        function is intended to be called repeatedly with the results of
 | 
						|
        each commit. It therefore shows a 'diff' between what it saw in
 | 
						|
        the last call and what it sees now.
 | 
						|
 | 
						|
        Args:
 | 
						|
            board_selected: Dict containing boards to summarise, keyed by
 | 
						|
                board.target
 | 
						|
            board_dict: Dict containing boards for which we built this
 | 
						|
                commit, keyed by board.target. The value is an Outcome object.
 | 
						|
            err_lines: A list of errors for this commit, or [] if there is
 | 
						|
                none, or we don't want to print errors
 | 
						|
            show_sizes: Show image size deltas
 | 
						|
            show_detail: Show detail for each board
 | 
						|
            show_bloat: Show detail for each function
 | 
						|
        """
 | 
						|
        better = []     # List of boards fixed since last commit
 | 
						|
        worse = []      # List of new broken boards since last commit
 | 
						|
        new = []        # List of boards that didn't exist last time
 | 
						|
        unknown = []    # List of boards that were not built
 | 
						|
 | 
						|
        for target in board_dict:
 | 
						|
            if target not in board_selected:
 | 
						|
                continue
 | 
						|
 | 
						|
            # If the board was built last time, add its outcome to a list
 | 
						|
            if target in self._base_board_dict:
 | 
						|
                base_outcome = self._base_board_dict[target].rc
 | 
						|
                outcome = board_dict[target]
 | 
						|
                if outcome.rc == OUTCOME_UNKNOWN:
 | 
						|
                    unknown.append(target)
 | 
						|
                elif outcome.rc < base_outcome:
 | 
						|
                    better.append(target)
 | 
						|
                elif outcome.rc > base_outcome:
 | 
						|
                    worse.append(target)
 | 
						|
            else:
 | 
						|
                new.append(target)
 | 
						|
 | 
						|
        # Get a list of errors that have appeared, and disappeared
 | 
						|
        better_err = []
 | 
						|
        worse_err = []
 | 
						|
        for line in err_lines:
 | 
						|
            if line not in self._base_err_lines:
 | 
						|
                worse_err.append('+' + line)
 | 
						|
        for line in self._base_err_lines:
 | 
						|
            if line not in err_lines:
 | 
						|
                better_err.append('-' + line)
 | 
						|
 | 
						|
        # Display results by arch
 | 
						|
        if better or worse or unknown or new or worse_err or better_err:
 | 
						|
            arch_list = {}
 | 
						|
            self.AddOutcome(board_selected, arch_list, better, '',
 | 
						|
                    self.col.GREEN)
 | 
						|
            self.AddOutcome(board_selected, arch_list, worse, '+',
 | 
						|
                    self.col.RED)
 | 
						|
            self.AddOutcome(board_selected, arch_list, new, '*', self.col.BLUE)
 | 
						|
            if self._show_unknown:
 | 
						|
                self.AddOutcome(board_selected, arch_list, unknown, '?',
 | 
						|
                        self.col.MAGENTA)
 | 
						|
            for arch, target_list in arch_list.iteritems():
 | 
						|
                print '%10s: %s' % (arch, target_list)
 | 
						|
            if better_err:
 | 
						|
                print self.col.Color(self.col.GREEN, '\n'.join(better_err))
 | 
						|
            if worse_err:
 | 
						|
                print self.col.Color(self.col.RED, '\n'.join(worse_err))
 | 
						|
 | 
						|
        if show_sizes:
 | 
						|
            self.PrintSizeSummary(board_selected, board_dict, show_detail,
 | 
						|
                                  show_bloat)
 | 
						|
 | 
						|
        # Save our updated information for the next call to this function
 | 
						|
        self._base_board_dict = board_dict
 | 
						|
        self._base_err_lines = err_lines
 | 
						|
 | 
						|
        # Get a list of boards that did not get built, if needed
 | 
						|
        not_built = []
 | 
						|
        for board in board_selected:
 | 
						|
            if not board in board_dict:
 | 
						|
                not_built.append(board)
 | 
						|
        if not_built:
 | 
						|
            print "Boards not built (%d): %s" % (len(not_built),
 | 
						|
                    ', '.join(not_built))
 | 
						|
 | 
						|
 | 
						|
    def ShowSummary(self, commits, board_selected, show_errors, show_sizes,
 | 
						|
                    show_detail, show_bloat):
 | 
						|
        """Show a build summary for U-Boot for a given board list.
 | 
						|
 | 
						|
        Reset the result summary, then repeatedly call GetResultSummary on
 | 
						|
        each commit's results, then display the differences we see.
 | 
						|
 | 
						|
        Args:
 | 
						|
            commit: Commit objects to summarise
 | 
						|
            board_selected: Dict containing boards to summarise
 | 
						|
            show_errors: Show errors that occured
 | 
						|
            show_sizes: Show size deltas
 | 
						|
            show_detail: Show detail for each board
 | 
						|
            show_bloat: Show detail for each function
 | 
						|
        """
 | 
						|
        self.commit_count = len(commits)
 | 
						|
        self.commits = commits
 | 
						|
        self.ResetResultSummary(board_selected)
 | 
						|
 | 
						|
        for commit_upto in range(0, self.commit_count, self._step):
 | 
						|
            board_dict, err_lines = self.GetResultSummary(board_selected,
 | 
						|
                    commit_upto, read_func_sizes=show_bloat)
 | 
						|
            msg = '%02d: %s' % (commit_upto + 1, commits[commit_upto].subject)
 | 
						|
            print self.col.Color(self.col.BLUE, msg)
 | 
						|
            self.PrintResultSummary(board_selected, board_dict,
 | 
						|
                    err_lines if show_errors else [], show_sizes, show_detail,
 | 
						|
                    show_bloat)
 | 
						|
 | 
						|
 | 
						|
    def SetupBuild(self, board_selected, commits):
 | 
						|
        """Set up ready to start a build.
 | 
						|
 | 
						|
        Args:
 | 
						|
            board_selected: Selected boards to build
 | 
						|
            commits: Selected commits to build
 | 
						|
        """
 | 
						|
        # First work out how many commits we will build
 | 
						|
        count = (len(commits) + self._step - 1) / self._step
 | 
						|
        self.count = len(board_selected) * count
 | 
						|
        self.upto = self.warned = self.fail = 0
 | 
						|
        self._timestamps = collections.deque()
 | 
						|
 | 
						|
    def BuildBoardsForCommit(self, board_selected, keep_outputs):
 | 
						|
        """Build all boards for a single commit"""
 | 
						|
        self.SetupBuild(board_selected)
 | 
						|
        self.count = len(board_selected)
 | 
						|
        for brd in board_selected.itervalues():
 | 
						|
            job = BuilderJob()
 | 
						|
            job.board = brd
 | 
						|
            job.commits = None
 | 
						|
            job.keep_outputs = keep_outputs
 | 
						|
            self.queue.put(brd)
 | 
						|
 | 
						|
        self.queue.join()
 | 
						|
        self.out_queue.join()
 | 
						|
        print
 | 
						|
        self.ClearLine(0)
 | 
						|
 | 
						|
    def BuildCommits(self, commits, board_selected, show_errors, keep_outputs):
 | 
						|
        """Build all boards for all commits (non-incremental)"""
 | 
						|
        self.commit_count = len(commits)
 | 
						|
 | 
						|
        self.ResetResultSummary(board_selected)
 | 
						|
        for self.commit_upto in range(self.commit_count):
 | 
						|
            self.SelectCommit(commits[self.commit_upto])
 | 
						|
            self.SelectOutputDir()
 | 
						|
            Mkdir(self.output_dir)
 | 
						|
 | 
						|
            self.BuildBoardsForCommit(board_selected, keep_outputs)
 | 
						|
            board_dict, err_lines = self.GetResultSummary()
 | 
						|
            self.PrintResultSummary(board_selected, board_dict,
 | 
						|
                err_lines if show_errors else [])
 | 
						|
 | 
						|
        if self.already_done:
 | 
						|
            print '%d builds already done' % self.already_done
 | 
						|
 | 
						|
    def GetThreadDir(self, thread_num):
 | 
						|
        """Get the directory path to the working dir for a thread.
 | 
						|
 | 
						|
        Args:
 | 
						|
            thread_num: Number of thread to check.
 | 
						|
        """
 | 
						|
        return os.path.join(self._working_dir, '%02d' % thread_num)
 | 
						|
 | 
						|
    def _PrepareThread(self, thread_num):
 | 
						|
        """Prepare the working directory for a thread.
 | 
						|
 | 
						|
        This clones or fetches the repo into the thread's work directory.
 | 
						|
 | 
						|
        Args:
 | 
						|
            thread_num: Thread number (0, 1, ...)
 | 
						|
        """
 | 
						|
        thread_dir = self.GetThreadDir(thread_num)
 | 
						|
        Mkdir(thread_dir)
 | 
						|
        git_dir = os.path.join(thread_dir, '.git')
 | 
						|
 | 
						|
        # Clone the repo if it doesn't already exist
 | 
						|
        # TODO(sjg@chromium): Perhaps some git hackery to symlink instead, so
 | 
						|
        # we have a private index but uses the origin repo's contents?
 | 
						|
        if self.git_dir:
 | 
						|
            src_dir = os.path.abspath(self.git_dir)
 | 
						|
            if os.path.exists(git_dir):
 | 
						|
                gitutil.Fetch(git_dir, thread_dir)
 | 
						|
            else:
 | 
						|
                print 'Cloning repo for thread %d' % thread_num
 | 
						|
                gitutil.Clone(src_dir, thread_dir)
 | 
						|
 | 
						|
    def _PrepareWorkingSpace(self, max_threads):
 | 
						|
        """Prepare the working directory for use.
 | 
						|
 | 
						|
        Set up the git repo for each thread.
 | 
						|
 | 
						|
        Args:
 | 
						|
            max_threads: Maximum number of threads we expect to need.
 | 
						|
        """
 | 
						|
        Mkdir(self._working_dir)
 | 
						|
        for thread in range(max_threads):
 | 
						|
            self._PrepareThread(thread)
 | 
						|
 | 
						|
    def _PrepareOutputSpace(self):
 | 
						|
        """Get the output directories ready to receive files.
 | 
						|
 | 
						|
        We delete any output directories which look like ones we need to
 | 
						|
        create. Having left over directories is confusing when the user wants
 | 
						|
        to check the output manually.
 | 
						|
        """
 | 
						|
        dir_list = []
 | 
						|
        for commit_upto in range(self.commit_count):
 | 
						|
            dir_list.append(self._GetOutputDir(commit_upto))
 | 
						|
 | 
						|
        for dirname in glob.glob(os.path.join(self.base_dir, '*')):
 | 
						|
            if dirname not in dir_list:
 | 
						|
                shutil.rmtree(dirname)
 | 
						|
 | 
						|
    def BuildBoards(self, commits, board_selected, show_errors, keep_outputs):
 | 
						|
        """Build all commits for a list of boards
 | 
						|
 | 
						|
        Args:
 | 
						|
            commits: List of commits to be build, each a Commit object
 | 
						|
            boards_selected: Dict of selected boards, key is target name,
 | 
						|
                    value is Board object
 | 
						|
            show_errors: True to show summarised error/warning info
 | 
						|
            keep_outputs: True to save build output files
 | 
						|
        """
 | 
						|
        self.commit_count = len(commits)
 | 
						|
        self.commits = commits
 | 
						|
 | 
						|
        self.ResetResultSummary(board_selected)
 | 
						|
        Mkdir(self.base_dir)
 | 
						|
        self._PrepareWorkingSpace(min(self.num_threads, len(board_selected)))
 | 
						|
        self._PrepareOutputSpace()
 | 
						|
        self.SetupBuild(board_selected, commits)
 | 
						|
        self.ProcessResult(None)
 | 
						|
 | 
						|
        # Create jobs to build all commits for each board
 | 
						|
        for brd in board_selected.itervalues():
 | 
						|
            job = BuilderJob()
 | 
						|
            job.board = brd
 | 
						|
            job.commits = commits
 | 
						|
            job.keep_outputs = keep_outputs
 | 
						|
            job.step = self._step
 | 
						|
            self.queue.put(job)
 | 
						|
 | 
						|
        # Wait until all jobs are started
 | 
						|
        self.queue.join()
 | 
						|
 | 
						|
        # Wait until we have processed all output
 | 
						|
        self.out_queue.join()
 | 
						|
        print
 | 
						|
        self.ClearLine(0)
 |