Allow for parallel builds and saved output
[platform/kernel/u-boot.git] / MAKEALL
1 #!/bin/bash
2 # Tool mainly for U-Boot Quality Assurance: build one or more board
3 # configurations with minimal verbosity, showing only warnings and
4 # errors.
5
6 usage()
7 {
8         # if exiting with 0, write to stdout, else write to stderr
9         local ret=${1:-0}
10         [ "${ret}" -eq 1 ] && exec 1>&2
11         cat <<-EOF
12         Usage: MAKEALL [options] [--] [boards-to-build]
13
14         Options:
15           -a ARCH,   --arch ARCH       Build all boards with arch ARCH
16           -c CPU,    --cpu CPU         Build all boards with cpu CPU
17           -v VENDOR, --vendor VENDOR   Build all boards with vendor VENDOR
18           -s SOC,    --soc SOC         Build all boards with soc SOC
19           -l,        --list            List all targets to be built
20           -m,        --maintainers     List all targets and maintainer email
21           -M,        --mails           List all targets and all affilated emails
22           -h,        --help            This help output
23
24         Selections by these options are logically ANDed; if the same option
25         is used repeatedly, such selections are ORed.  So "-v FOO -v BAR"
26         will select all configurations where the vendor is either FOO or
27         BAR.  Any additional arguments specified on the command line are
28         always build additionally.  See the boards.cfg file for more info.
29
30         If no boards are specified, then the default is "powerpc".
31
32         Environment variables:
33           BUILD_NCPUS      number of parallel make jobs (default: auto)
34           CROSS_COMPILE    cross-compiler toolchain prefix (default: "")
35           MAKEALL_LOGDIR   output all logs to here (default: ./LOG/)
36           BUILD_DIR        output build directory (default: ./)
37           BUILD_NBUILDS    number of parallel targets (default: 1)
38
39         Examples:
40           - build all Power Architecture boards:
41               MAKEALL -a powerpc
42               MAKEALL --arch powerpc
43               MAKEALL powerpc
44           - build all PowerPC boards manufactured by vendor "esd":
45               MAKEALL -a powerpc -v esd
46           - build all PowerPC boards manufactured either by "keymile" or "siemens":
47               MAKEALL -a powerpc -v keymile -v siemens
48           - build all Freescale boards with MPC83xx CPUs, plus all 4xx boards:
49               MAKEALL -c mpc83xx -v freescale 4xx
50         EOF
51         exit ${ret}
52 }
53
54 SHORT_OPTS="ha:c:v:s:lmM"
55 LONG_OPTS="help,arch:,cpu:,vendor:,soc:,list,maintainers,mails"
56
57 # Option processing based on util-linux-2.13/getopt-parse.bash
58
59 # Note that we use `"$@"' to let each command-line parameter expand to a
60 # separate word. The quotes around `$@' are essential!
61 # We need TEMP as the `eval set --' would nuke the return value of
62 # getopt.
63 TEMP=`getopt -o ${SHORT_OPTS} --long ${LONG_OPTS} \
64      -n 'MAKEALL' -- "$@"`
65
66 [ $? != 0 ] && usage 1
67
68 # Note the quotes around `$TEMP': they are essential!
69 eval set -- "$TEMP"
70
71 SELECTED=''
72 ONLY_LIST=''
73 PRINT_MAINTS=''
74 MAINTAINERS_ONLY=''
75
76 while true ; do
77         case "$1" in
78         -a|--arch)
79                 # echo "Option ARCH: argument \`$2'"
80                 if [ "$opt_a" ] ; then
81                         opt_a="${opt_a%)} || \$2 == \"$2\")"
82                 else
83                         opt_a="(\$2 == \"$2\")"
84                 fi
85                 SELECTED='y'
86                 shift 2 ;;
87         -c|--cpu)
88                 # echo "Option CPU: argument \`$2'"
89                 if [ "$opt_c" ] ; then
90                         opt_c="${opt_c%)} || \$3 == \"$2\")"
91                 else
92                         opt_c="(\$3 == \"$2\")"
93                 fi
94                 SELECTED='y'
95                 shift 2 ;;
96         -s|--soc)
97                 # echo "Option SoC: argument \`$2'"
98                 if [ "$opt_s" ] ; then
99                         opt_s="${opt_s%)} || \$6 == \"$2\")"
100                 else
101                         opt_s="(\$6 == \"$2\")"
102                 fi
103                 SELECTED='y'
104                 shift 2 ;;
105         -v|--vendor)
106                 # echo "Option VENDOR: argument \`$2'"
107                 if [ "$opt_v" ] ; then
108                         opt_v="${opt_v%)} || \$5 == \"$2\")"
109                 else
110                         opt_v="(\$5 == \"$2\")"
111                 fi
112                 SELECTED='y'
113                 shift 2 ;;
114         -l|--list)
115                 ONLY_LIST='y'
116                 shift ;;
117         -m|--maintainers)
118                 ONLY_LIST='y'
119                 PRINT_MAINTS='y'
120                 MAINTAINERS_ONLY='y'
121                 shift ;;
122         -M|--mails)
123                 ONLY_LIST='y'
124                 PRINT_MAINTS='y'
125                 shift ;;
126         -h|--help)
127                 usage ;;
128         --)
129                 shift ; break ;;
130         *)
131                 echo "Internal error!" >&2 ; exit 1 ;;
132         esac
133 done
134 # echo "Remaining arguments:"
135 # for arg do echo '--> '"\`$arg'" ; done
136
137 FILTER="\$1 !~ /^#/"
138 [ "$opt_a" ] && FILTER="${FILTER} && $opt_a"
139 [ "$opt_c" ] && FILTER="${FILTER} && $opt_c"
140 [ "$opt_s" ] && FILTER="${FILTER} && $opt_s"
141 [ "$opt_v" ] && FILTER="${FILTER} && $opt_v"
142
143 if [ "$SELECTED" ] ; then
144         SELECTED=$(awk '('"$FILTER"') { print $1 }' boards.cfg)
145
146         # Make sure some boards from boards.cfg are actually found
147         if [ -z "$SELECTED" ] ; then
148                 echo "Error: No boards selected, invalid arguments"
149                 exit 1
150         fi
151 fi
152
153 #########################################################################
154
155 # Print statistics when we exit
156 trap exit 1 2 3 15
157 trap print_stats 0
158
159 # Determine number of CPU cores if no default was set
160 : ${BUILD_NCPUS:="`getconf _NPROCESSORS_ONLN`"}
161
162 if [ "$BUILD_NCPUS" -gt 1 ]
163 then
164         JOBS="-j $((BUILD_NCPUS + 1))"
165 else
166         JOBS=""
167 fi
168
169
170 if [ "${CROSS_COMPILE}" ] ; then
171         MAKE="make CROSS_COMPILE=${CROSS_COMPILE}"
172 else
173         MAKE=make
174 fi
175
176 if [ "${MAKEALL_LOGDIR}" ] ; then
177         LOG_DIR=${MAKEALL_LOGDIR}
178 else
179         LOG_DIR="LOG"
180 fi
181
182 : ${BUILD_NBUILDS:=1}
183 BUILD_MANY=0
184
185 if [ "${BUILD_NBUILDS}" -gt 1 ] ; then
186         BUILD_MANY=1
187         : ${BUILD_DIR:=./build}
188         mkdir -p "${BUILD_DIR}/ERR"
189         find "${BUILD_DIR}/ERR/" -type f -exec rm -f {} +
190 fi
191
192 : ${BUILD_DIR:=.}
193
194 OUTPUT_PREFIX="${BUILD_DIR}"
195
196 [ -d ${LOG_DIR} ] || mkdir "${LOG_DIR}" || exit 1
197 find "${LOG_DIR}/" -type f -exec rm -f {} +
198
199 LIST=""
200
201 # Keep track of the number of builds and errors
202 ERR_CNT=0
203 ERR_LIST=""
204 TOTAL_CNT=0
205 CURRENT_CNT=0
206 OLDEST_IDX=1
207 RC=0
208
209 # Helper funcs for parsing boards.cfg
210 boards_by_field()
211 {
212         awk \
213                 -v field="$1" \
214                 -v select="$2" \
215                 '($1 !~ /^#/ && $field == select) { print $1 }' \
216                 boards.cfg
217 }
218 boards_by_arch() { boards_by_field 2 "$@" ; }
219 boards_by_cpu()  { boards_by_field 3 "$@" ; }
220 boards_by_soc()  { boards_by_field 6 "$@" ; }
221
222 #########################################################################
223 ## MPC5xx Systems
224 #########################################################################
225
226 LIST_5xx="$(boards_by_cpu mpc5xx)"
227
228 #########################################################################
229 ## MPC5xxx Systems
230 #########################################################################
231
232 LIST_5xxx="$(boards_by_cpu mpc5xxx)"
233
234 #########################################################################
235 ## MPC512x Systems
236 #########################################################################
237
238 LIST_512x="$(boards_by_cpu mpc512x)"
239
240 #########################################################################
241 ## MPC8xx Systems
242 #########################################################################
243
244 LIST_8xx="$(boards_by_cpu mpc8xx)"
245
246 #########################################################################
247 ## PPC4xx Systems
248 #########################################################################
249
250 LIST_4xx="$(boards_by_cpu ppc4xx)"
251
252 #########################################################################
253 ## MPC8220 Systems
254 #########################################################################
255
256 LIST_8220="$(boards_by_cpu mpc8220)"
257
258 #########################################################################
259 ## MPC824x Systems
260 #########################################################################
261
262 LIST_824x="$(boards_by_cpu mpc824x)"
263
264 #########################################################################
265 ## MPC8260 Systems (includes 8250, 8255 etc.)
266 #########################################################################
267
268 LIST_8260="$(boards_by_cpu mpc8260)"
269
270 #########################################################################
271 ## MPC83xx Systems (includes 8349, etc.)
272 #########################################################################
273
274 LIST_83xx="$(boards_by_cpu mpc83xx)"
275
276 #########################################################################
277 ## MPC85xx Systems (includes 8540, 8560 etc.)
278 #########################################################################
279
280 LIST_85xx="$(boards_by_cpu mpc85xx)"
281
282 #########################################################################
283 ## MPC86xx Systems
284 #########################################################################
285
286 LIST_86xx="$(boards_by_cpu mpc86xx)"
287
288 #########################################################################
289 ## 74xx/7xx Systems
290 #########################################################################
291
292 LIST_74xx_7xx="$(boards_by_cpu 74xx_7xx)"
293
294 #########################################################################
295 ## PowerPC groups
296 #########################################################################
297
298 LIST_TSEC="             \
299         ${LIST_83xx}    \
300         ${LIST_85xx}    \
301         ${LIST_86xx}    \
302 "
303
304 LIST_powerpc="          \
305         ${LIST_5xx}     \
306         ${LIST_512x}    \
307         ${LIST_5xxx}    \
308         ${LIST_8xx}     \
309         ${LIST_8220}    \
310         ${LIST_824x}    \
311         ${LIST_8260}    \
312         ${LIST_83xx}    \
313         ${LIST_85xx}    \
314         ${LIST_86xx}    \
315         ${LIST_4xx}     \
316         ${LIST_74xx_7xx}\
317 "
318
319 # Alias "ppc" -> "powerpc" to not break compatibility with older scripts
320 # still using "ppc" instead of "powerpc"
321 LIST_ppc="              \
322         ${LIST_powerpc} \
323 "
324
325 #########################################################################
326 ## StrongARM Systems
327 #########################################################################
328
329 LIST_SA="$(boards_by_cpu sa1100)"
330
331 #########################################################################
332 ## ARM9 Systems
333 #########################################################################
334
335 LIST_ARM9="$(boards_by_cpu arm920t)     \
336         $(boards_by_cpu arm926ejs)      \
337         $(boards_by_cpu arm925t)        \
338 "
339
340 #########################################################################
341 ## ARM11 Systems
342 #########################################################################
343 LIST_ARM11="$(boards_by_cpu arm1136)    \
344         imx31_phycore           \
345         imx31_phycore_eet       \
346         mx31pdk                 \
347         smdk6400                \
348 "
349
350 #########################################################################
351 ## ARMV7 Systems
352 #########################################################################
353
354 LIST_ARMV7="$(boards_by_cpu armv7)"
355
356 #########################################################################
357 ## AT91 Systems
358 #########################################################################
359
360 LIST_at91="$(boards_by_soc at91)"
361
362 #########################################################################
363 ## Xscale Systems
364 #########################################################################
365
366 LIST_pxa="$(boards_by_cpu pxa)"
367
368 LIST_ixp="$(boards_by_cpu ixp)
369         pdnb3           \
370         scpu            \
371 "
372
373 #########################################################################
374 ## ARM groups
375 #########################################################################
376
377 LIST_arm="                      \
378         ${LIST_SA}              \
379         ${LIST_ARM9}            \
380         ${LIST_ARM10}           \
381         ${LIST_ARM11}           \
382         ${LIST_ARMV7}   \
383         ${LIST_at91}            \
384         ${LIST_pxa}             \
385         ${LIST_ixp}             \
386 "
387
388 #########################################################################
389 ## MIPS Systems         (default = big endian)
390 #########################################################################
391
392 LIST_mips4kc="          \
393         incaip          \
394         qemu_mips       \
395         vct_platinum    \
396         vct_platinum_small      \
397         vct_platinum_onenand    \
398         vct_platinum_onenand_small      \
399         vct_platinumavc \
400         vct_platinumavc_small   \
401         vct_platinumavc_onenand \
402         vct_platinumavc_onenand_small   \
403         vct_premium     \
404         vct_premium_small       \
405         vct_premium_onenand     \
406         vct_premium_onenand_small       \
407 "
408
409 LIST_au1xx0="           \
410         dbau1000        \
411         dbau1100        \
412         dbau1500        \
413         dbau1550        \
414         gth2            \
415 "
416
417 LIST_mips="             \
418         ${LIST_mips4kc} \
419         ${LIST_mips5kc} \
420         ${LIST_au1xx0}  \
421 "
422
423 #########################################################################
424 ## MIPS Systems         (little endian)
425 #########################################################################
426
427 LIST_xburst_el="        \
428         qi_lb60         \
429 "
430
431 LIST_au1xx0_el="        \
432         dbau1550_el     \
433         pb1000          \
434 "
435 LIST_mips_el="                  \
436         ${LIST_xburst_el}       \
437         ${LIST_au1xx0_el}       \
438 "
439 #########################################################################
440 ## OpenRISC Systems
441 #########################################################################
442
443 LIST_openrisc="$(boards_by_arch openrisc)"
444
445 #########################################################################
446 ## x86 Systems
447 #########################################################################
448
449 LIST_x86="$(boards_by_arch x86)"
450
451 #########################################################################
452 ## Nios-II Systems
453 #########################################################################
454
455 LIST_nios2="$(boards_by_arch nios2)"
456
457 #########################################################################
458 ## MicroBlaze Systems
459 #########################################################################
460
461 LIST_microblaze="$(boards_by_arch microblaze)"
462
463 #########################################################################
464 ## ColdFire Systems
465 #########################################################################
466
467 LIST_m68k="$(boards_by_arch m68k)
468         EB+MCF-EV123            \
469         EB+MCF-EV123_internal   \
470         M52277EVB               \
471         M5235EVB                \
472         M54451EVB               \
473         M54455EVB               \
474 "
475 LIST_coldfire=${LIST_m68k}
476
477 #########################################################################
478 ## AVR32 Systems
479 #########################################################################
480
481 LIST_avr32="$(boards_by_arch avr32)"
482
483 #########################################################################
484 ## Blackfin Systems
485 #########################################################################
486
487 LIST_blackfin="$(boards_by_arch blackfin)"
488
489 #########################################################################
490 ## SH Systems
491 #########################################################################
492
493 LIST_sh2="$(boards_by_cpu sh2)"
494 LIST_sh3="$(boards_by_cpu sh3)"
495 LIST_sh4="$(boards_by_cpu sh4)"
496
497 LIST_sh="$(boards_by_arch sh)"
498
499 #########################################################################
500 ## SPARC Systems
501 #########################################################################
502
503 LIST_sparc="$(boards_by_arch sparc)"
504
505 #########################################################################
506 ## NDS32 Systems
507 #########################################################################
508
509 LIST_nds32="$(boards_by_arch nds32)"
510
511 #-----------------------------------------------------------------------
512
513 get_target_location() {
514         local target=$1
515         local BOARD_NAME=""
516         local CONFIG_NAME=""
517         local board=""
518         local vendor=""
519
520         # Automatic mode
521         local line=`egrep -i "^[[:space:]]*${target}[[:space:]]" boards.cfg`
522
523         if [ -z "${line}" ] ; then echo "" ; return ; fi
524
525         set ${line}
526
527         # add default board name if needed
528         [ $# = 3 ] && set ${line} ${1}
529
530         CONFIG_NAME="${1%_config}"
531
532         [ "${BOARD_NAME}" ] || BOARD_NAME="${1%_config}"
533
534         if [ "$4" = "-" ] ; then
535                 board=${BOARD_NAME}
536         else
537                 board="$4"
538         fi
539
540         [ $# -gt 4 ] && [ "$5" != "-" ] && vendor="$5"
541         [ $# -gt 6 ] && [ "$7" != "-" ] && {
542                 tmp="${7%:*}"
543                 if [ "$tmp" ] ; then
544                         CONFIG_NAME="$tmp"
545                 fi
546         }
547
548         # Assign board directory to BOARDIR variable
549         if [ -z "${vendor}" ] ; then
550             BOARDDIR=${board}
551         else
552             BOARDDIR=${vendor}/${board}
553         fi
554
555         echo "${CONFIG_NAME}:${BOARDDIR}"
556 }
557
558 get_target_maintainers() {
559         local name=`echo $1 | cut -d : -f 1`
560
561         if ! grep -qsi "[[:blank:]]${name}[[:blank:]]" MAINTAINERS ; then
562                 echo ""
563                 return ;
564         fi
565
566         local line=`tac MAINTAINERS | grep -ni "[[:blank:]]${name}[[:blank:]]" | cut -d : -f 1`
567         local mail=`tac MAINTAINERS | tail -n +${line} | \
568                 sed -n ":start /.*@.*/ { b mail } ; n ; b start ; :mail /.*@.*/ { p ; n ; b mail } ; q" | \
569                 sed "s/^.*<//;s/>.*$//"`
570         echo "$mail"
571 }
572
573 list_target() {
574         if [ "$PRINT_MAINTS" != 'y' ] ; then
575                 echo "$1"
576                 return
577         fi
578
579         echo -n "$1:"
580
581         local loc=`get_target_location $1`
582
583         if [ -z "${loc}" ] ; then echo "ERROR" ; return ; fi
584
585         local maintainers_result=`get_target_maintainers ${loc} | tr " " "\n"`
586
587         if [ "$MAINTAINERS_ONLY" != 'y' ] ; then
588
589                 local dir=`echo ${loc} | cut -d ":" -f 2`
590                 local cfg=`echo ${loc} | cut -d ":" -f 1`
591                 local git_result=`git log --format=%aE board/${dir} \
592                                 include/configs/${cfg}.h | grep "@"`
593                 local git_result_recent=`echo ${git_result} | tr " " "\n" | \
594                                                 head -n 3`
595                 local git_result_top=`echo ${git_result} | tr " " "\n" | \
596                         sort | uniq -c | sort -nr | head -n 3 | \
597                         sed "s/^ \+[0-9]\+ \+//"`
598
599                 echo -e "$git_result_recent\n$git_result_top\n$maintainers_result" | \
600                         sort -u | tr "\n" " " | sed "s/ $//" ;
601         else
602                 echo -e "$maintainers_result" | sort -u | tr "\n" " " | \
603                                                 sed "s/ $//" ;
604         fi
605
606         echo ""
607 }
608
609 # Each finished build will have a file called ${donep}${n},
610 # where n is the index of the build. Each build
611 # we've already noted as finished will have ${skipp}${n}.
612 # The code managing the build process will use this information
613 # to ensure that only BUILD_NBUILDS builds are in flight at once
614 donep="${LOG_DIR}/._done_"
615 skipp="${LOG_DIR}/._skip_"
616
617 build_target() {
618         target=$1
619         build_idx=$2
620
621         if [ $BUILD_MANY == 1 ] ; then
622                 output_dir="${OUTPUT_PREFIX}/${target}"
623                 mkdir -p "${output_dir}"
624         else
625                 output_dir="${OUTPUT_PREFIX}"
626         fi
627
628         export BUILD_DIR="${output_dir}"
629
630         if [ "$ONLY_LIST" == 'y' ] ; then
631                 list_target ${target}
632                 return
633         fi
634
635         ${MAKE} distclean >/dev/null
636         ${MAKE} -s ${target}_config
637
638         ${MAKE} ${JOBS} all \
639                 >${LOG_DIR}/$target.MAKELOG 2> ${LOG_DIR}/$target.ERR
640
641         # Check for 'make' errors
642         if [ ${PIPESTATUS[0]} -ne 0 ] ; then
643                 RC=1
644         fi
645
646         if [ $BUILD_MANY == 1 ] ; then
647                 ${MAKE} tidy
648
649                 if [ -s ${LOG_DIR}/${target}.ERR ] ; then
650                         touch ${OUTPUT_PREFIX}/ERR/${target}
651                 else
652                         rm ${LOG_DIR}/${target}.ERR
653                 fi
654         else
655                 if [ -s ${LOG_DIR}/${target}.ERR ] ; then
656                         : $(( ERR_CNT += 1 ))
657                         ERR_LIST="${ERR_LIST} $target"
658                 else
659                         rm ${LOG_DIR}/${target}.ERR
660                 fi
661         fi
662
663         OBJS=${output_dir}/u-boot
664         if [ -e ${output_dir}/spl/u-boot-spl ]; then
665                 OBJS="${OBJS} ${output_dir}/spl/u-boot-spl"
666         fi
667
668         ${CROSS_COMPILE}size ${OBJS} | tee -a ${LOG_DIR}/$target.MAKELOG
669
670         [ -e "${LOG_DIR}/${target}.ERR" ] && cat "${LOG_DIR}/${target}.ERR"
671
672         #echo "Writing ${donep}${build_idx}"
673         touch "${donep}${build_idx}"
674 }
675
676 manage_builds() {
677         search_idx=${OLDEST_IDX}
678         #echo "Searching ${OLDEST_IDX} to ${TOTAL_CNT}"
679         while true; do
680                 if [ -e "${donep}${search_idx}" ] ; then
681         #               echo "Found ${donep}${search_idx}"
682                         : $(( CURRENT_CNT-- ))
683                         [ ${OLDEST_IDX} -eq ${search_idx} ] &&
684                                 : $(( OLDEST_IDX++ ))
685
686                         # Only want to count it once
687                         rm -f "${donep}${search_idx}"
688                         touch "${skipp}${search_idx}"
689                 elif [ -e "${skipp}${search_idx}" ] ; then
690                         [ ${OLDEST_IDX} -eq ${search_idx} ] &&
691                                 : $(( OLDEST_IDX++ ))
692                 fi
693                 #echo "Checking search ${search_idx} vs ${TOTAL_CNT}"
694                 : $(( search_idx++ ))
695                 if [ ${search_idx} -gt ${TOTAL_CNT} ] ; then
696                         #echo "Checking current ${CURRENT_CNT} vs ${BUILD_NBUILDS}"
697                         if [ ${CURRENT_CNT} -ge ${BUILD_NBUILDS} ] ; then
698                                 search_idx=${OLDEST_IDX}
699                                 sleep 1
700                         else
701                                 break
702                         fi
703                 fi
704         done
705 }
706
707 build_targets() {
708         for t in "$@" ; do
709                 # If a LIST_xxx var exists, use it.  But avoid variable
710                 # expansion in the eval when a board name contains certain
711                 # characters that the shell interprets.
712                 case ${t} in
713                         *[-+=]*) list= ;;
714                         *)       list=$(eval echo '${LIST_'$t'}') ;;
715                 esac
716                 if [ -n "${list}" ] ; then
717                         build_targets ${list}
718                 else
719                         : $((TOTAL_CNT += 1))
720                         : $((CURRENT_CNT += 1))
721                         rm -f "${donep}${TOTAL_CNT}"
722                         rm -f "${skipp}${TOTAL_CNT}"
723                         build_target ${t} ${TOTAL_CNT} &
724                 fi
725
726                 # We maintain a running count of all the builds we have done.
727                 # Each finished build will have a file called ${donep}${n},
728                 # where n is the index of the build. Each build
729                 # we've already noted as finished will have ${skipp}${n}.
730                 # We track the current index via TOTAL_CNT, and the oldest
731                 # index. When we exceed the maximum number of parallel builds,
732                 # We look from oldest to current for builds that have completed,
733                 # and update the current count and oldest index as appropriate.
734                 # If we've gone through the entire list, wait a second, and
735                 # reprocess the entire list until we find a build that has
736                 # completed
737                 if [ ${CURRENT_CNT} -ge ${BUILD_NBUILDS} ] ; then
738                         manage_builds
739                 fi
740         done
741 }
742
743 #-----------------------------------------------------------------------
744
745 print_stats() {
746         if [ "$ONLY_LIST" == 'y' ] ; then return ; fi
747
748         rm -f ${donep}* ${skipp}*
749
750         if [ $BUILD_MANY == 1 ] && [ -e "${OUTPUT_PREFIX}/ERR" ] ; then
751                 ERR_LIST=$(ls ${OUTPUT_PREFIX}/ERR/)
752                 ERR_CNT=`ls -1 ${OUTPUT_PREFIX}/ERR/ | wc | awk '{print $1}'`
753         else
754                 ERR_CNT=0
755         fi
756
757         echo ""
758         echo "--------------------- SUMMARY ----------------------------"
759         echo "Boards compiled: ${TOTAL_CNT}"
760         if [ ${ERR_CNT} -gt 0 ] ; then
761                 echo "Boards with warnings or errors: ${ERR_CNT} (${ERR_LIST} )"
762         fi
763         echo "----------------------------------------------------------"
764
765         exit $RC
766 }
767
768 #-----------------------------------------------------------------------
769
770 # Build target groups selected by options, plus any command line args
771 set -- ${SELECTED} "$@"
772 # run PowerPC by default
773 [ $# = 0 ] && set -- powerpc
774 build_targets "$@"
775 wait