The MAKEALL script cleverly runs make with the appropriate options
to use all of the cores on the system, but your average U-Boot build
can't make much use of more than a few cores.  If you happen to have
a many-core server, your builds will leave most of the system idle.

In order to make full use of such a system, we need to build multiple
targets in parallel, and this requires directing make output into
multiple directories. We add a BUILD_NBUILDS variable, which allows
users to specify how many builds to run in parallel.
When BUILD_NBUILDS is set greater than 1, we redefine BUILD_DIR for
each build to be ${BUILD_DIR}/${target}. Also, we make "./build" the
default BUILD_DIR when BUILD_NBUILDS is greater than 1.

MAKEALL now tracks which builds are still running, and when one
finishes, it starts a new build.

Once each build finishes, we run "make tidy" on its directory, to reduce
the footprint.

As a result, we are left with a build directory with all of the built
targets still there for use, which means anyone who wanted to use
MAKEALL as part of a test harness can now do so.

Signed-off-by: Andy Fleming <aflem...@freescale.com>
---
v2:     - Update to keep BUILD_NBUILDS builds in flight, rather than batching
        - Clean up style things
        - Defer error output until build completion to make output *slightly*
        more readable

Performance tests (highly unscientific):

./MAKEALL 83xx (original)
real    6m27.123s
user    12m59.342s
sys     2m2.528s

BUILD_NBUILDS=2 ./MAKEALL 83xx
real    4m54.854s
user    12m10.336s
sys     1m55.957s

BUILD_NBUILDS=4 ./MAKEALL 83xx
real    3m17.039s
user    11m9.596s
sys     1m40.327s

BUILD_NBUILDS=8 ./MAKEALL 83xx
real    2m44.366s
user    10m38.658s
sys     1m32.425s

Amusingly, this was the best configuration I found so far:

BUILD_NBUILDS=50 BUILD_NCPUS=1 ./MAKEALL 83xx
real    2m1.252s
user    10m49.617s
sys     1m36.176s


 MAKEALL |  115 +++++++++++++++++++++++++++++++++++++++++++++++++++++++-------
 1 files changed, 102 insertions(+), 13 deletions(-)

diff --git a/MAKEALL b/MAKEALL
index 95b7cd3..c14a1da 100755
--- a/MAKEALL
+++ b/MAKEALL
@@ -31,6 +31,7 @@ usage()
          CROSS_COMPILE    cross-compiler toolchain prefix (default: "")
          MAKEALL_LOGDIR   output all logs to here (default: ./LOG/)
          BUILD_DIR        output build directory (default: ./)
+         BUILD_NBUILDS    number of parallel targets (default: 1)
 
        Examples:
          - build all Power Architecture boards:
@@ -160,11 +161,20 @@ else
        LOG_DIR="LOG"
 fi
 
-if [ ! "${BUILD_DIR}" ] ; then
-       BUILD_DIR="."
+: ${BUILD_NBUILDS:=1}
+
+if [ "${BUILD_NBUILDS}" -gt 1 ] ; then
+       : ${BUILD_DIR:=./build}
+       mkdir -p "${BUILD_DIR}/ERR"
+       find "${BUILD_DIR}/ERR/" -type f -exec rm -f {} +
 fi
 
-[ -d ${LOG_DIR} ] || mkdir ${LOG_DIR} || exit 1
+: ${BUILD_DIR:=.}
+
+OUTPUT_PREFIX="${BUILD_DIR}"
+
+[ -d ${LOG_DIR} ] || mkdir "${LOG_DIR}" || exit 1
+find "${LOG_DIR}/" -type f -exec rm -f {} +
 
 LIST=""
 
@@ -172,6 +182,8 @@ LIST=""
 ERR_CNT=0
 ERR_LIST=""
 TOTAL_CNT=0
+CURRENT_COUNT=0
+OLDEST_IDX=1
 RC=0
 
 # Helper funcs for parsing boards.cfg
@@ -485,31 +497,56 @@ LIST_nds32="$(boards_by_arch nds32)"
 
 #-----------------------------------------------------------------------
 
+DONE_PREFIX="${LOG_DIR}/._build_"
+SKIP_PREFIX="${LOG_DIR}/._skip_"
+
 build_target() {
        target=$1
+       build_idx=$2
+
+       if [ "$BUILD_NBUILDS" -gt 1 ] ; then
+               output_dir="${OUTPUT_PREFIX}/${target}"
+               mkdir -p "${output_dir}"
+       else
+               output_dir="${OUTPUT_PREFIX}"
+       fi
+
+       export BUILD_DIR="${output_dir}"
 
        ${MAKE} distclean >/dev/null
        ${MAKE} -s ${target}_config
 
-       ${MAKE} ${JOBS} all 2>&1 >${LOG_DIR}/$target.MAKELOG \
-                               | tee ${LOG_DIR}/$target.ERR
+       ${MAKE} ${JOBS} all \
+               >${LOG_DIR}/$target.MAKELOG 2> ${LOG_DIR}/$target.ERR
 
        # Check for 'make' errors
        if [ ${PIPESTATUS[0]} -ne 0 ] ; then
                RC=1
        fi
 
-       if [ -s ${LOG_DIR}/$target.ERR ] ; then
-               ERR_CNT=$((ERR_CNT + 1))
-               ERR_LIST="${ERR_LIST} $target"
+       if [ "$BUILD_NBUILDS" -gt 1 ] ; then
+               ${MAKE} tidy
+
+               if [ -s ${LOG_DIR}/${target}.ERR ] ; then
+                       touch ${OUTPUT_PREFIX}/ERR/${target}
+               else
+                       rm ${LOG_DIR}/${target}.ERR
+               fi
        else
-               rm ${LOG_DIR}/$target.ERR
+               if [ -s ${LOG_DIR}/${target}.ERR ] ; then
+                       : $(( ERR_CNT += 1 ))
+                       ERR_LIST="${ERR_LIST} $target"
+               else
+                       rm ${LOG_DIR}/${target}.ERR
+               fi
        fi
 
-       TOTAL_CNT=$((TOTAL_CNT + 1))
-
-       ${CROSS_COMPILE}size ${BUILD_DIR}/u-boot \
+       ${CROSS_COMPILE}size ${output_dir}/u-boot \
                                | tee -a ${LOG_DIR}/$target.MAKELOG
+
+       [ -e "${LOG_DIR}/${target}.ERR" ] && cat "${LOG_DIR}/${target}.ERR"
+
+       touch "${DONE_PREFIX}${build_idx}"
 }
 build_targets() {
        for t in "$@" ; do
@@ -523,7 +560,49 @@ build_targets() {
                if [ -n "${list}" ] ; then
                        build_targets ${list}
                else
-                       build_target ${t}
+                       : $((TOTAL_CNT += 1))
+                       : $((CURRENT_COUNT += 1))
+                       rm -f "${DONE_PREFIX}${TOTAL_CNT}"
+                       rm -f "${SKIP_PREFIX}${TOTAL_CNT}"
+                       build_target ${t} ${TOTAL_CNT} &
+               fi
+
+               # We maintain a running count of all the builds we have done.
+               # Each finished build will have a file called 
${DONE_PREFIX}${n},
+               # where n is the index of the build. Each build
+               # we've already noted as finished will have ${SKIP_PREFIX}${n}.
+               # We track the current index via TOTAL_CNT, and the oldest
+               # index. When we exceed the maximum number of parallel builds,
+               # We look from oldest to current for builds that have completed,
+               # and update the current count and oldest index as appropriate.
+               # If we've gone through the entire list, wait a second, and
+               # reprocess the entire list until we find a build that has
+               # completed
+               if [ ${CURRENT_COUNT} -ge ${BUILD_NBUILDS} ] ; then
+                       search_idx=${OLDEST_IDX}
+                       while true; do
+                               if [ -e "${DONE_PREFIX}${search_idx}" ] ; then
+                                       : $(( CURRENT_COUNT-- ))
+                                       [ ${OLDEST_IDX} -eq ${search_idx} ] &&
+                                               : $(( OLDEST_IDX++ ))
+
+                                       # Only want to count it once
+                                       rm -f "${DONE_PREFIX}${search_idx}"
+                                       touch "${SKIP_PREFIX}${search_idx}"
+                               elif [ -e "${SKIP_PREFIX}${search_idx}" ] ; then
+                                       [ ${OLDEST_IDX} -eq ${search_idx} ] &&
+                                               : $(( OLDEST_IDX++ ))
+                               fi
+                               if [ ${search_idx} -ge ${TOTAL_CNT} ] ; then
+                                       if [ ${CURRENT_COUNT} -ge 
${BUILD_NBUILDS} ] ; then
+                                               search_idx=${OLDEST_IDX}
+                                               sleep 1
+                                       else
+                                               break
+                                       fi
+                               fi
+                               : $(( search_idx++ ))
+                       done
                fi
        done
 }
@@ -531,6 +610,15 @@ build_targets() {
 #-----------------------------------------------------------------------
 
 print_stats() {
+       for ((build_num=1; ${build_num} <= ${TOTAL_CNT} ; build_num++)) ; do
+               rm -f "${DONE_PREFIX}${build_num}"
+               rm -f "${SKIP_PREFIX}${build_num}"
+       done
+
+       if [ "$BUILD_NBUILDS" -gt 1 ] ; then
+               ERR_LIST=$(ls ${OUTPUT_PREFIX}/ERR/)
+               ERR_CNT=`ls -1 ${OUTPUT_PREFIX}/ERR/ | wc | awk '{print $1}'`
+       fi
        echo ""
        echo "--------------------- SUMMARY ----------------------------"
        echo "Boards compiled: ${TOTAL_CNT}"
@@ -549,3 +637,4 @@ set -- ${SELECTED} "$@"
 # run PowerPC by default
 [ $# = 0 ] && set -- powerpc
 build_targets "$@"
+wait
-- 
1.7.3.4


_______________________________________________
U-Boot mailing list
U-Boot@lists.denx.de
http://lists.denx.de/mailman/listinfo/u-boot

Reply via email to