Home | History | Annotate | Line # | Download | only in src
build.sh revision 1.93
      1  1.67   thorpej #! /usr/bin/env sh
      2  1.93     lukem #	$NetBSD: build.sh,v 1.93 2003/03/04 02:20:28 lukem Exp $
      3  1.84     lukem #
      4  1.84     lukem # Copyright (c) 2001-2003 The NetBSD Foundation, Inc.
      5  1.84     lukem # All rights reserved.
      6  1.84     lukem #
      7  1.84     lukem # This code is derived from software contributed to The NetBSD Foundation
      8  1.84     lukem # by Todd Vierling and Luke Mewburn.
      9  1.84     lukem #
     10  1.84     lukem # Redistribution and use in source and binary forms, with or without
     11  1.84     lukem # modification, are permitted provided that the following conditions
     12  1.84     lukem # are met:
     13  1.84     lukem # 1. Redistributions of source code must retain the above copyright
     14  1.84     lukem #    notice, this list of conditions and the following disclaimer.
     15  1.84     lukem # 2. Redistributions in binary form must reproduce the above copyright
     16  1.84     lukem #    notice, this list of conditions and the following disclaimer in the
     17  1.84     lukem #    documentation and/or other materials provided with the distribution.
     18  1.84     lukem # 3. All advertising materials mentioning features or use of this software
     19  1.84     lukem #    must display the following acknowledgement:
     20  1.84     lukem #        This product includes software developed by the NetBSD
     21  1.84     lukem #        Foundation, Inc. and its contributors.
     22  1.84     lukem # 4. Neither the name of The NetBSD Foundation nor the names of its
     23  1.84     lukem #    contributors may be used to endorse or promote products derived
     24  1.84     lukem #    from this software without specific prior written permission.
     25  1.84     lukem #
     26  1.84     lukem # THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     27  1.84     lukem # ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     28  1.84     lukem # TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     29  1.84     lukem # PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     30  1.84     lukem # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     31  1.84     lukem # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     32  1.84     lukem # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     33  1.84     lukem # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     34  1.84     lukem # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     35  1.84     lukem # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     36  1.84     lukem # POSSIBILITY OF SUCH DAMAGE.
     37  1.84     lukem #
     38   1.1        tv #
     39   1.1        tv # Top level build wrapper, for a system containing no tools.
     40   1.1        tv #
     41   1.1        tv # This script should run on any POSIX-compliant shell.  For systems
     42  1.33        tv # with a strange /bin/sh, "ksh" or "bash" may be an ample alternative.
     43   1.1        tv #
     44  1.67   thorpej # Note, however, that due to the way the interpreter is invoked above,
     45  1.67   thorpej # if a POSIX-compliant shell is the first in the PATH, you won't have
     46  1.67   thorpej # to take any further action.
     47  1.67   thorpej #
     48   1.1        tv 
     49  1.84     lukem progname=${0##*/}
     50  1.84     lukem toppid=$$
     51  1.82     lukem trap "exit 1" 1 2 3 15
     52  1.84     lukem 
     53  1.79     lukem bomb()
     54  1.79     lukem {
     55  1.82     lukem 	cat >&2 <<ERRORMESSAGE
     56  1.82     lukem 
     57  1.82     lukem ERROR: $@
     58  1.82     lukem *** BUILD ABORTED ***
     59  1.82     lukem ERRORMESSAGE
     60  1.82     lukem 	kill $toppid		# in case we were invoked from a subshell
     61   1.1        tv 	exit 1
     62   1.1        tv }
     63  1.78     lukem 
     64  1.84     lukem 
     65  1.84     lukem initdefaults()
     66  1.84     lukem {
     67  1.84     lukem 	cd "$(dirname $0)"
     68  1.84     lukem 	[ -d usr.bin/make ] ||
     69  1.84     lukem 	    bomb "build.sh must be run from the top source level"
     70  1.84     lukem 	[ -f share/mk/bsd.own.mk ] ||
     71  1.84     lukem 	    bomb "src/share/mk is missing; please re-fetch the source tree"
     72  1.84     lukem 
     73  1.84     lukem 	uname_s=$(uname -s 2>/dev/null)
     74  1.84     lukem 	uname_m=$(uname -m 2>/dev/null)
     75  1.84     lukem 
     76  1.84     lukem 	# If $PWD is a valid name of the current directory, POSIX mandates
     77  1.84     lukem 	# that pwd return it by default which causes problems in the
     78  1.84     lukem 	# presence of symlinks.  Unsetting PWD is simpler than changing
     79  1.84     lukem 	# every occurrence of pwd to use -P.
     80  1.84     lukem 	#
     81  1.84     lukem 	# XXX Except that doesn't work on Solaris.
     82  1.84     lukem 	unset PWD
     83  1.84     lukem 	if [ "${uname_s}" = "SunOS" ]; then
     84  1.84     lukem 		TOP=$(pwd -P)
     85  1.84     lukem 	else
     86  1.84     lukem 		TOP=$(pwd)
     87  1.84     lukem 	fi
     88  1.84     lukem 
     89  1.84     lukem 	# Set defaults.
     90  1.84     lukem 	toolprefix=nb
     91  1.84     lukem 	MAKEFLAGS=
     92  1.84     lukem 	makeenv=
     93  1.84     lukem 	makewrapper=
     94  1.84     lukem 	runcmd=
     95  1.84     lukem 	operations=
     96  1.84     lukem 	removedirs=
     97  1.84     lukem 	do_expertmode=false
     98  1.84     lukem 	do_rebuildmake=false
     99  1.84     lukem 	do_removedirs=false
    100  1.84     lukem 
    101  1.84     lukem 	# do_{operation}=true if given operation is requested.
    102  1.84     lukem 	#
    103  1.84     lukem 	do_tools=false
    104  1.84     lukem 	do_obj=false
    105  1.84     lukem 	do_build=false
    106  1.84     lukem 	do_distribution=false
    107  1.84     lukem 	do_release=false
    108  1.84     lukem 	do_kernel=false
    109  1.84     lukem 	do_install=false
    110  1.87     lukem 	do_sets=false
    111  1.84     lukem }
    112  1.29       jmc 
    113  1.79     lukem getarch()
    114  1.79     lukem {
    115   1.1        tv 	# Translate a MACHINE into a default MACHINE_ARCH.
    116  1.84     lukem 	#
    117   1.1        tv 	case $MACHINE in
    118   1.1        tv 
    119  1.84     lukem 	acorn26|acorn32|cats|netwinder|shark|*arm)
    120  1.84     lukem 		MACHINE_ARCH=arm
    121  1.84     lukem 		;;
    122  1.84     lukem 
    123  1.84     lukem 	sun2)
    124  1.84     lukem 		MACHINE_ARCH=m68000
    125  1.84     lukem 		;;
    126  1.84     lukem 
    127  1.84     lukem 	amiga|atari|cesfic|hp300|sun3|*68k)
    128  1.84     lukem 		MACHINE_ARCH=m68k
    129  1.84     lukem 		;;
    130   1.1        tv 
    131  1.84     lukem 	mipsco|newsmips|sbmips|sgimips)
    132  1.84     lukem 		MACHINE_ARCH=mipseb
    133  1.84     lukem 		;;
    134   1.1        tv 
    135  1.84     lukem 	algor|arc|cobalt|evbmips|hpcmips|playstation2|pmax)
    136  1.84     lukem 		MACHINE_ARCH=mipsel
    137  1.84     lukem 		;;
    138   1.1        tv 
    139  1.84     lukem 	pc532)
    140  1.84     lukem 		MACHINE_ARCH=ns32k
    141  1.84     lukem 		;;
    142   1.1        tv 
    143  1.84     lukem 	bebox|prep|sandpoint|*ppc)
    144  1.84     lukem 		MACHINE_ARCH=powerpc
    145  1.84     lukem 		;;
    146   1.1        tv 
    147  1.84     lukem 	evbsh3|mmeye)
    148  1.84     lukem 		MACHINE_ARCH=sh3eb
    149  1.84     lukem 		;;
    150   1.1        tv 
    151  1.84     lukem 	dreamcast|hpcsh)
    152  1.84     lukem 		MACHINE_ARCH=sh3el
    153  1.84     lukem 		;;
    154   1.1        tv 
    155  1.84     lukem 	hp700)
    156  1.84     lukem 		MACHINE_ARCH=hppa
    157  1.84     lukem 		;;
    158   1.1        tv 
    159  1.84     lukem 	evbsh5)
    160  1.84     lukem 		MACHINE_ARCH=sh5el
    161  1.84     lukem 		;;
    162  1.62  fredette 
    163  1.84     lukem 	alpha|i386|sparc|sparc64|vax|x86_64)
    164  1.84     lukem 		MACHINE_ARCH=$MACHINE
    165  1.84     lukem 		;;
    166  1.69   thorpej 
    167  1.84     lukem 	*)
    168  1.84     lukem 		bomb "unknown target MACHINE: $MACHINE"
    169  1.84     lukem 		;;
    170   1.4        tv 
    171   1.1        tv 	esac
    172   1.1        tv }
    173   1.1        tv 
    174  1.79     lukem validatearch()
    175  1.79     lukem {
    176  1.47        tv 	# Ensure that the MACHINE_ARCH exists (and is supported by build.sh).
    177  1.84     lukem 	#
    178  1.47        tv 	case $MACHINE_ARCH in
    179  1.47        tv 
    180  1.84     lukem 	alpha|arm|armeb|hppa|i386|m68000|m68k|mipse[bl]|ns32k|powerpc|sh[35]e[bl]|sparc|sparc64|vax|x86_64)
    181  1.84     lukem 		;;
    182  1.84     lukem 
    183  1.84     lukem 	*)
    184  1.84     lukem 		bomb "unknown target MACHINE_ARCH: $MACHINE_ARCH"
    185  1.84     lukem 		;;
    186  1.84     lukem 
    187  1.47        tv 	esac
    188  1.47        tv }
    189  1.47        tv 
    190  1.79     lukem getmakevar()
    191  1.79     lukem {
    192  1.82     lukem 	[ -x $make ] || bomb "getmakevar $1: $make is not executable"
    193  1.30       jmc 	$make -m ${TOP}/share/mk -s -f- _x_ <<EOF
    194  1.15        tv _x_:
    195  1.15        tv 	echo \${$1}
    196  1.15        tv .include <bsd.prog.mk>
    197  1.70     lukem .include <bsd.kernobj.mk>
    198  1.15        tv EOF
    199  1.15        tv }
    200  1.15        tv 
    201  1.82     lukem safe_getmakevar()
    202  1.82     lukem {
    203  1.84     lukem 	# getmakevar() doesn't work properly if $make hasn't yet been built,
    204  1.84     lukem 	# which can happen when running with the "-n" option.
    205  1.84     lukem 	# safe_getmakevar() deals with this by emitting a literal '$'
    206  1.84     lukem 	# followed by the variable name, instead of trying to find the
    207  1.84     lukem 	# variable's value.
    208  1.84     lukem 	#
    209  1.84     lukem 
    210  1.82     lukem 	if [ -x $make ]; then
    211  1.82     lukem 		getmakevar "$1"
    212  1.82     lukem 	else
    213  1.82     lukem 		echo "\$$1"
    214  1.82     lukem 	fi
    215  1.82     lukem }
    216  1.82     lukem 
    217  1.79     lukem resolvepath()
    218  1.79     lukem {
    219  1.10        tv 	case $OPTARG in
    220  1.84     lukem 	/*)
    221  1.84     lukem 		;;
    222  1.84     lukem 	*)
    223  1.84     lukem 		OPTARG="$TOP/$OPTARG"
    224  1.84     lukem 		;;
    225  1.10        tv 	esac
    226  1.10        tv }
    227  1.10        tv 
    228  1.79     lukem usage()
    229  1.79     lukem {
    230  1.84     lukem 	if [ -n "$*" ]; then
    231  1.84     lukem 		echo ""
    232  1.84     lukem 		echo "${progname}: $*"
    233  1.84     lukem 	fi
    234  1.70     lukem 	cat <<_usage_
    235  1.83     lukem 
    236  1.84     lukem Usage: ${progname} [-EnorUu] [-a arch] [-B buildid] [-D dest] [-j njob] [-M obj]
    237  1.84     lukem 		[-m mach] [-O obj] [-R release] [-T tools] [-V var=[value]]
    238  1.86     lukem 		[-w wrapper]   operation [...]
    239  1.84     lukem 
    240  1.86     lukem  Build operations (all imply "obj" and "tools"):
    241  1.84     lukem     build		Run "make build"
    242  1.89     lukem     distribution	Run "make distribution" (includes DESTDIR/etc/ files)
    243  1.84     lukem     release		Run "make release" (includes kernels & distrib media)
    244  1.84     lukem 
    245  1.84     lukem  Other operations:
    246  1.87     lukem     help		Show this message (and exit)
    247  1.90     enami     makewrapper		Create ${toolprefix}make-\${MACHINE} wrapper and ${toolprefix}make.
    248  1.90     enami 			(Always done)
    249  1.86     lukem     obj			Run "make obj" (default unless -o is used)
    250  1.84     lukem     tools 		Build and install tools
    251  1.84     lukem     kernel=conf		Build kernel with config file \`conf'
    252  1.84     lukem     install=idir	Run "make installworld" to \`idir'
    253  1.84     lukem 			(useful after 'distribution' or 'release')
    254  1.88     lukem     sets		Create distribution sets in RELEASEDIR
    255  1.84     lukem 
    256  1.84     lukem  Options:
    257  1.83     lukem     -a arch	Set MACHINE_ARCH to arch (otherwise deduced from MACHINE)
    258  1.84     lukem     -B buildId	Set BUILDID to buildId
    259  1.83     lukem     -D dest	Set DESTDIR to dest
    260  1.83     lukem     -E		Set "expert" mode; disables some DESTDIR checks
    261  1.83     lukem     -j njob	Run up to njob jobs in parallel; see make(1)
    262  1.83     lukem     -M obj	Set obj root directory to obj (sets MAKEOBJDIRPREFIX)
    263  1.83     lukem     -m mach	Set MACHINE to mach (not required if NetBSD native)
    264  1.83     lukem     -n		Show commands that would be executed, but do not execute them
    265  1.83     lukem     -O obj	Set obj root directory to obj (sets a MAKEOBJDIR pattern)
    266  1.83     lukem     -o		Set MKOBJDIRS=no (do not create objdirs at start of build)
    267  1.84     lukem     -R release	Set RELEASEDIR to release
    268  1.83     lukem     -r		Remove contents of TOOLDIR and DESTDIR before building
    269  1.84     lukem     -T tools	Set TOOLDIR to tools.  If unset, and TOOLDIR is not set in
    270  1.86     lukem 		the environment, ${toolprefix}make will be (re)built unconditionally
    271  1.86     lukem     -U		Set UNPRIVED (build without requiring root privileges)
    272  1.86     lukem     -u		Set UPDATE (do not run "make clean" first)
    273  1.83     lukem     -V v=[val]	Set variable \`v' to \`val'
    274  1.84     lukem     -w wrapper	Create ${toolprefix}make script as wrapper
    275  1.86     lukem 		(Default: \${TOOLDIR}/bin/${toolprefix}make-\${MACHINE})
    276  1.83     lukem 
    277  1.70     lukem _usage_
    278   1.1        tv 	exit 1
    279   1.1        tv }
    280   1.1        tv 
    281  1.84     lukem parseoptions()
    282  1.84     lukem {
    283  1.84     lukem 	opts='a:B:bD:dEhi:j:k:M:m:nO:oR:rT:tUuV:w:'
    284  1.84     lukem 	opt_a=no
    285  1.84     lukem 
    286  1.84     lukem 	if type getopts >/dev/null 2>&1; then
    287  1.84     lukem 		# Use POSIX getopts.
    288  1.84     lukem 		getoptcmd='getopts $opts opt && opt=-$opt'
    289  1.84     lukem 		optargcmd=':'
    290  1.84     lukem 		optremcmd='shift $(($OPTIND -1))'
    291  1.84     lukem 	else
    292  1.84     lukem 		type getopt >/dev/null 2>&1 ||
    293  1.84     lukem 		    bomb "/bin/sh shell is too old; try ksh or bash"
    294  1.84     lukem 
    295  1.84     lukem 		# Use old-style getopt(1) (doesn't handle whitespace in args).
    296  1.84     lukem 		args="$(getopt $opts $*)"
    297  1.84     lukem 		[ $? = 0 ] || usage
    298  1.84     lukem 		set -- $args
    299  1.84     lukem 
    300  1.84     lukem 		getoptcmd='[ $# -gt 0 ] && opt="$1" && shift'
    301  1.84     lukem 		optargcmd='OPTARG="$1"; shift'
    302  1.84     lukem 		optremcmd=':'
    303  1.84     lukem 	fi
    304  1.84     lukem 
    305  1.84     lukem 	# Parse command line options.
    306  1.84     lukem 	#
    307  1.84     lukem 	while eval $getoptcmd; do
    308  1.84     lukem 		case $opt in
    309  1.84     lukem 
    310  1.84     lukem 		-a)
    311  1.84     lukem 			eval $optargcmd
    312  1.84     lukem 			MACHINE_ARCH=$OPTARG
    313  1.84     lukem 			opt_a=yes
    314  1.84     lukem 			;;
    315  1.84     lukem 
    316  1.84     lukem 		-B)
    317  1.84     lukem 			eval $optargcmd
    318  1.84     lukem 			BUILDID=$OPTARG
    319  1.84     lukem 			;;
    320  1.84     lukem 
    321  1.84     lukem 		-b)
    322  1.86     lukem 			usage "'-b' has been replaced by 'makewrapper'"
    323  1.84     lukem 			;;
    324  1.84     lukem 
    325  1.84     lukem 		-D)
    326  1.84     lukem 			eval $optargcmd; resolvepath
    327  1.84     lukem 			DESTDIR="$OPTARG"
    328  1.84     lukem 			export DESTDIR
    329  1.84     lukem 			makeenv="$makeenv DESTDIR"
    330  1.84     lukem 			;;
    331  1.84     lukem 
    332  1.84     lukem 		-d)
    333  1.84     lukem 			usage "'-d' has been replaced by 'distribution'"
    334  1.84     lukem 			;;
    335  1.84     lukem 
    336  1.84     lukem 		-E)
    337  1.84     lukem 			do_expertmode=true
    338  1.84     lukem 			;;
    339  1.84     lukem 
    340  1.84     lukem 		-i)
    341  1.84     lukem 			usage "'-i idir' has been replaced by 'install=idir'"
    342  1.84     lukem 			;;
    343  1.84     lukem 
    344  1.84     lukem 		-j)
    345  1.84     lukem 			eval $optargcmd
    346  1.84     lukem 			parallel="-j $OPTARG"
    347  1.84     lukem 			;;
    348  1.84     lukem 
    349  1.84     lukem 		-k)
    350  1.84     lukem 			usage "'-k conf' has been replaced by 'kernel=conf'"
    351  1.84     lukem 			;;
    352  1.84     lukem 
    353  1.84     lukem 		-M)
    354  1.84     lukem 			eval $optargcmd; resolvepath
    355  1.84     lukem 			MAKEOBJDIRPREFIX="$OPTARG"
    356  1.84     lukem 			export MAKEOBJDIRPREFIX
    357  1.84     lukem 			makeobjdir=$OPTARG
    358  1.84     lukem 			makeenv="$makeenv MAKEOBJDIRPREFIX"
    359  1.84     lukem 			;;
    360  1.84     lukem 
    361  1.84     lukem 			# -m overrides MACHINE_ARCH unless "-a" is specified
    362  1.84     lukem 		-m)
    363  1.84     lukem 			eval $optargcmd
    364  1.84     lukem 			MACHINE=$OPTARG
    365  1.84     lukem 			[ "$opt_a" != "yes" ] && getarch
    366  1.84     lukem 			;;
    367  1.84     lukem 
    368  1.84     lukem 		-n)
    369  1.84     lukem 			runcmd=echo
    370  1.84     lukem 			;;
    371  1.84     lukem 
    372  1.84     lukem 		-O)
    373  1.84     lukem 			eval $optargcmd; resolvepath
    374  1.84     lukem 			MAKEOBJDIR="\${.CURDIR:C,^$TOP,$OPTARG,}"
    375  1.84     lukem 			export MAKEOBJDIR
    376  1.84     lukem 			makeobjdir=$OPTARG
    377  1.84     lukem 			makeenv="$makeenv MAKEOBJDIR"
    378  1.84     lukem 			;;
    379  1.84     lukem 
    380  1.84     lukem 		-o)
    381  1.84     lukem 			MKOBJDIRS=no
    382  1.84     lukem 			;;
    383  1.84     lukem 
    384  1.84     lukem 		-R)
    385  1.84     lukem 			eval $optargcmd; resolvepath
    386  1.84     lukem 			RELEASEDIR=$OPTARG
    387  1.84     lukem 			export RELEASEDIR
    388  1.84     lukem 			makeenv="$makeenv RELEASEDIR"
    389  1.84     lukem 			;;
    390  1.84     lukem 
    391  1.84     lukem 		-r)
    392  1.84     lukem 			do_removedirs=true
    393  1.84     lukem 			do_rebuildmake=true
    394  1.84     lukem 			;;
    395  1.84     lukem 
    396  1.84     lukem 		-T)
    397  1.84     lukem 			eval $optargcmd; resolvepath
    398  1.84     lukem 			TOOLDIR="$OPTARG"
    399  1.84     lukem 			export TOOLDIR
    400  1.84     lukem 			;;
    401  1.84     lukem 
    402  1.84     lukem 		-t)
    403  1.84     lukem 			usage "'-t' has been replaced by 'tools'"
    404  1.84     lukem 			;;
    405  1.16   thorpej 
    406  1.84     lukem 		-U)
    407  1.84     lukem 			UNPRIVED=yes
    408  1.84     lukem 			export UNPRIVED
    409  1.84     lukem 			makeenv="$makeenv UNPRIVED"
    410  1.84     lukem 			;;
    411  1.44     lukem 
    412  1.84     lukem 		-u)
    413  1.84     lukem 			UPDATE=yes
    414  1.84     lukem 			export UPDATE
    415  1.84     lukem 			makeenv="$makeenv UPDATE"
    416  1.84     lukem 			;;
    417  1.15        tv 
    418  1.84     lukem 		-V)
    419  1.84     lukem 			eval $optargcmd
    420  1.84     lukem 			case "${OPTARG}" in
    421  1.80     lukem 		    # XXX: consider restricting which variables can be changed?
    422  1.84     lukem 			[a-zA-Z_][a-zA-Z_0-9]*=*)
    423  1.84     lukem 				var=${OPTARG%%=*}
    424  1.84     lukem 				value=${OPTARG#*=}
    425  1.84     lukem 				eval "${var}=\"${value}\"; export ${var}"
    426  1.84     lukem 				makeenv="$makeenv ${var}"
    427  1.84     lukem 				;;
    428  1.84     lukem 			*)
    429  1.84     lukem 				usage "-V argument must be of the form 'var=[value]'"
    430  1.84     lukem 				;;
    431  1.84     lukem 			esac
    432  1.84     lukem 			;;
    433  1.84     lukem 
    434  1.84     lukem 		-w)
    435  1.84     lukem 			eval $optargcmd; resolvepath
    436  1.84     lukem 			makewrapper="$OPTARG"
    437  1.84     lukem 			;;
    438  1.84     lukem 
    439  1.84     lukem 		--)
    440  1.84     lukem 			break
    441  1.84     lukem 			;;
    442  1.84     lukem 
    443  1.84     lukem 		-'?'|-h)
    444  1.84     lukem 			usage
    445  1.84     lukem 			;;
    446  1.84     lukem 
    447  1.84     lukem 		esac
    448  1.84     lukem 	done
    449  1.84     lukem 
    450  1.84     lukem 	# Validate operations.
    451  1.84     lukem 	#
    452  1.84     lukem 	eval $optremcmd
    453  1.84     lukem 	while [ $# -gt 0 ]; do
    454  1.84     lukem 		op=$1; shift
    455  1.84     lukem 		operations="$operations $op"
    456  1.84     lukem 
    457  1.84     lukem 		case "$op" in
    458  1.84     lukem 
    459  1.87     lukem 		help)
    460  1.87     lukem 			usage
    461  1.87     lukem 			;;
    462  1.87     lukem 
    463  1.87     lukem 		makewrapper|obj|tools|build|distribution|release|sets)
    464  1.80     lukem 			;;
    465  1.84     lukem 
    466  1.84     lukem 		kernel=*)
    467  1.84     lukem 			arg=${op#*=}
    468  1.84     lukem 			op=${op%%=*}
    469  1.86     lukem 			if [ -z "${arg}" ]; then
    470  1.84     lukem 				bomb "Must supply a kernel name with \`kernel=...'"
    471  1.84     lukem 			fi
    472  1.84     lukem 			;;
    473  1.84     lukem 
    474  1.84     lukem 		install=*)
    475  1.84     lukem 			arg=${op#*=}
    476  1.84     lukem 			op=${op%%=*}
    477  1.86     lukem 			if [ -z "${arg}" ]; then
    478  1.84     lukem 				bomb "Must supply a directory with \`install=...'"
    479  1.84     lukem 			fi
    480  1.84     lukem 			;;
    481  1.84     lukem 
    482  1.80     lukem 		*)
    483  1.84     lukem 			usage "Unknown operation \`${op}'"
    484  1.84     lukem 			;;
    485  1.84     lukem 
    486  1.80     lukem 		esac
    487  1.84     lukem 		eval do_$op=true
    488  1.84     lukem 	done
    489  1.86     lukem 	if [ -z "${operations}" ]; then
    490  1.86     lukem 		usage "Missing operation to perform."
    491  1.86     lukem 	fi
    492  1.84     lukem 
    493  1.84     lukem 	# Set up MACHINE*.  On a NetBSD host, these are allowed to be unset.
    494  1.84     lukem 	#
    495  1.84     lukem 	if [ -z "$MACHINE" ]; then
    496  1.84     lukem 		if [ "${uname_s}" != "NetBSD" ]; then
    497  1.84     lukem 			bomb "MACHINE must be set, or -m must be used, for cross builds."
    498  1.84     lukem 		fi
    499  1.84     lukem 		MACHINE=${uname_m}
    500  1.84     lukem 	fi
    501  1.84     lukem 	[ -n "$MACHINE_ARCH" ] || getarch
    502  1.84     lukem 	validatearch
    503  1.84     lukem 
    504  1.84     lukem 	# Set up default make(1) environment.
    505  1.84     lukem 	#
    506  1.84     lukem 	makeenv="$makeenv TOOLDIR MACHINE MACHINE_ARCH MAKEFLAGS"
    507  1.84     lukem 	if [ ! -z "$BUILDID" ]; then
    508  1.84     lukem 		makeenv="$makeenv BUILDID"
    509  1.84     lukem 	fi
    510  1.84     lukem 	MAKEFLAGS="-m $TOP/share/mk $MAKEFLAGS MKOBJDIRS=${MKOBJDIRS-yes}"
    511  1.84     lukem 	export MAKEFLAGS MACHINE MACHINE_ARCH
    512  1.84     lukem }
    513  1.84     lukem 
    514  1.84     lukem rebuildmake()
    515  1.84     lukem {
    516  1.84     lukem 	# Test make source file timestamps against installed ${toolprefix}make
    517  1.84     lukem 	# binary, if TOOLDIR is pre-set.
    518  1.84     lukem 	#
    519  1.84     lukem 	# Note that we do NOT try to grovel "mk.conf" here to find out if
    520  1.84     lukem 	# TOOLDIR is set there, because it can contain make variable
    521  1.84     lukem 	# expansions and other stuff only parseable *after* we have a working
    522  1.84     lukem 	# ${toolprefix}make.  So this logic can only work if the user has
    523  1.84     lukem 	# pre-set TOOLDIR in the environment or used the -T option to build.sh.
    524  1.84     lukem 	#
    525  1.84     lukem 	make="${TOOLDIR-nonexistent}/bin/${toolprefix}make"
    526  1.84     lukem 	if [ -x $make ]; then
    527  1.84     lukem 		for f in usr.bin/make/*.[ch] usr.bin/make/lst.lib/*.[ch]; do
    528  1.84     lukem 			if [ $f -nt $make ]; then
    529  1.84     lukem 				do_rebuildmake=true
    530  1.84     lukem 				break
    531  1.84     lukem 			fi
    532  1.84     lukem 		done
    533  1.84     lukem 	else
    534  1.84     lukem 		do_rebuildmake=true
    535  1.84     lukem 	fi
    536  1.84     lukem 
    537  1.84     lukem 	# Build bootstrap ${toolprefix}make if needed.
    538  1.84     lukem 	if $do_rebuildmake; then
    539  1.84     lukem 		$runcmd echo "===> Bootstrapping ${toolprefix}make"
    540  1.84     lukem 		tmpdir="${TMPDIR-/tmp}/nbbuild$$"
    541  1.84     lukem 
    542  1.84     lukem 		$runcmd mkdir "$tmpdir" || bomb "cannot mkdir: $tmpdir"
    543  1.84     lukem 		trap "cd /; rm -r -f \"$tmpdir\"" 0
    544  1.84     lukem 		$runcmd cd "$tmpdir"
    545  1.84     lukem 
    546  1.84     lukem 		$runcmd env CC="${HOST_CC-cc}" CPPFLAGS="${HOST_CPPFLAGS}" \
    547  1.84     lukem 			CFLAGS="${HOST_CFLAGS--O}" LDFLAGS="${HOST_LDFLAGS}" \
    548  1.84     lukem 			"$TOP/tools/make/configure" ||
    549  1.84     lukem 		    bomb "configure of ${toolprefix}make failed"
    550  1.84     lukem 		$runcmd sh buildmake.sh ||
    551  1.84     lukem 		    bomb "build of ${toolprefix}make failed"
    552  1.84     lukem 
    553  1.84     lukem 		make="$tmpdir/${toolprefix}make"
    554  1.84     lukem 		$runcmd cd "$TOP"
    555  1.84     lukem 		$runcmd rm -f usr.bin/make/*.o usr.bin/make/lst.lib/*.o
    556  1.84     lukem 	fi
    557  1.84     lukem }
    558  1.84     lukem 
    559  1.84     lukem validatemakeparams()
    560  1.84     lukem {
    561  1.84     lukem 	if [ "$runcmd" = "echo" ]; then
    562  1.84     lukem 		TOOLCHAIN_MISSING=no
    563  1.84     lukem 		EXTERNAL_TOOLCHAIN=""
    564  1.84     lukem 	else
    565  1.84     lukem 		TOOLCHAIN_MISSING=$(getmakevar TOOLCHAIN_MISSING)
    566  1.84     lukem 		EXTERNAL_TOOLCHAIN=$(getmakevar EXTERNAL_TOOLCHAIN)
    567  1.84     lukem 	fi
    568  1.84     lukem 	if [ "${TOOLCHAIN_MISSING}" = "yes" ] && \
    569  1.86     lukem 	   [ -z "${EXTERNAL_TOOLCHAIN}" ]; then
    570  1.84     lukem 		$runcmd echo "ERROR: build.sh (in-tree cross-toolchain) is not yet available for"
    571  1.84     lukem 		$runcmd echo "	MACHINE:      ${MACHINE}"
    572  1.84     lukem 		$runcmd echo "	MACHINE_ARCH: ${MACHINE_ARCH}"
    573  1.84     lukem 		$runcmd echo ""
    574  1.84     lukem 		$runcmd echo "All builds for this platform should be done via a traditional make"
    575  1.84     lukem 		$runcmd echo "If you wish to use an external cross-toolchain, set"
    576  1.84     lukem 		$runcmd echo "	EXTERNAL_TOOLCHAIN=<path to toolchain root>"
    577  1.84     lukem 		$runcmd echo "in either the environment or mk.conf and rerun"
    578  1.84     lukem 		$runcmd echo "	$progname $*"
    579  1.84     lukem 		exit 1
    580  1.84     lukem 	fi
    581  1.84     lukem 
    582  1.84     lukem 	# If TOOLDIR isn't already set, make objdirs in "tools" in case the
    583  1.84     lukem 	# default setting from <bsd.own.mk> is used.
    584  1.84     lukem 	#
    585  1.84     lukem 	if [ -z "$TOOLDIR" ] && [ "$MKOBJDIRS" != "no" ]; then
    586  1.84     lukem 		$runcmd cd tools
    587  1.84     lukem 		$runcmd $make -m ${TOP}/share/mk obj NOSUBDIR= ||
    588  1.84     lukem 		    bomb "make obj failed in tools"
    589  1.84     lukem 		$runcmd cd "$TOP"
    590  1.84     lukem 	fi
    591  1.80     lukem 
    592  1.84     lukem 	# If setting -M or -O to root an obj dir make sure the base directory
    593  1.84     lukem 	# is made before continuing as bsd.own.mk will need this to pick up
    594  1.84     lukem 	# _SRC_TOP_OBJ_
    595  1.84     lukem 	#
    596  1.84     lukem 	if [ "$MKOBJDIRS" != "no" ] && [ ! -z "$makeobjdir" ]; then
    597  1.84     lukem 		$runcmd mkdir -p "$makeobjdir"
    598  1.84     lukem 	fi
    599  1.12        tv 
    600  1.84     lukem 	# Find DESTDIR and TOOLDIR.
    601  1.84     lukem 	#
    602  1.84     lukem 	DESTDIR=$(safe_getmakevar DESTDIR)
    603  1.84     lukem 	$runcmd echo "===> DESTDIR path: $DESTDIR"
    604  1.84     lukem 	TOOLDIR=$(safe_getmakevar TOOLDIR)
    605  1.84     lukem 	$runcmd echo "===> TOOLDIR path: $TOOLDIR"
    606  1.84     lukem 	export DESTDIR TOOLDIR
    607  1.84     lukem 
    608  1.84     lukem 	# Check validity of TOOLDIR and DESTDIR.
    609  1.84     lukem 	#
    610  1.84     lukem 	if [ -z "$TOOLDIR" ] || [ "$TOOLDIR" = "/" ]; then
    611  1.84     lukem 		bomb "TOOLDIR '$TOOLDIR' invalid"
    612  1.84     lukem 	fi
    613  1.84     lukem 	removedirs="$TOOLDIR"
    614  1.15        tv 
    615  1.84     lukem 	if [ -z "$DESTDIR" ] || [ "$DESTDIR" = "/" ]; then
    616  1.84     lukem 		if $do_build || $do_distribution || $do_release; then
    617  1.84     lukem 			if ! $do_build || \
    618  1.84     lukem 			   [ "${uname_s}" != "NetBSD" ] || \
    619  1.84     lukem 			   [ "${uname_m}" != "$MACHINE" ]; then
    620  1.84     lukem 				bomb "DESTDIR must != / for cross builds, or ${progname} 'distribution' or 'release'."
    621  1.84     lukem 			fi
    622  1.84     lukem 			if ! $do_expertmode; then
    623  1.84     lukem 				bomb "DESTDIR must != / for non -E (expert) builds"
    624  1.84     lukem 			fi
    625  1.84     lukem 			$runcmd echo "===> WARNING: Building to /, in expert mode."
    626  1.84     lukem 			$runcmd echo "===>          This may cause your system to break!  Reasons include:"
    627  1.84     lukem 			$runcmd echo "===>             - your kernel is not up to date"
    628  1.84     lukem 			$runcmd echo "===>             - the libraries or toolchain have changed"
    629  1.84     lukem 			$runcmd echo "===>          YOU HAVE BEEN WARNED!"
    630   1.1        tv 		fi
    631  1.84     lukem 	else
    632  1.84     lukem 		removedirs="$removedirs $DESTDIR"
    633  1.84     lukem 	fi
    634  1.86     lukem 	if $do_build || $do_distribution || $do_release; then
    635  1.86     lukem 		if ! $do_expertmode && \
    636  1.86     lukem 		    [ $(id -u 2>/dev/null) -ne 0 ] &&\
    637  1.86     lukem 		    [ -z "$UNPRIVED" ] ; then
    638  1.86     lukem 			bomb "-U or -E must be set for build as an unprivileged user."
    639  1.86     lukem 		fi
    640  1.86     lukem         fi
    641  1.84     lukem }
    642  1.30       jmc 
    643  1.15        tv 
    644  1.84     lukem createmakewrapper()
    645  1.84     lukem {
    646  1.84     lukem 	# Remove the target directories.
    647  1.84     lukem 	#
    648  1.84     lukem 	if $do_removedirs; then
    649  1.84     lukem 		for f in $removedirs; do
    650  1.84     lukem 			$runcmd echo "===> Removing $f"
    651  1.84     lukem 			$runcmd rm -r -f $f
    652  1.84     lukem 		done
    653  1.84     lukem 	fi
    654  1.15        tv 
    655  1.84     lukem 	# Recreate $TOOLDIR.
    656  1.84     lukem 	#
    657  1.84     lukem 	$runcmd mkdir -p "$TOOLDIR/bin" || bomb "mkdir of '$TOOLDIR/bin' failed"
    658  1.84     lukem 
    659  1.84     lukem 	# Install ${toolprefix}make if it was built.
    660  1.84     lukem 	#
    661  1.84     lukem 	if $do_rebuildmake; then
    662  1.84     lukem 		$runcmd rm -f "$TOOLDIR/bin/${toolprefix}make"
    663  1.84     lukem 		$runcmd cp $make "$TOOLDIR/bin/${toolprefix}make" ||
    664  1.84     lukem 		    bomb "failed to install \$TOOLDIR/bin/${toolprefix}make"
    665  1.84     lukem 		make="$TOOLDIR/bin/${toolprefix}make"
    666  1.84     lukem 		$runcmd echo "===> Created ${make}"
    667  1.84     lukem 		$runcmd rm -r -f "$tmpdir"
    668  1.84     lukem 		trap 0
    669  1.84     lukem 	fi
    670  1.15        tv 
    671  1.84     lukem 	# Build a ${toolprefix}make wrapper script, usable by hand as
    672  1.84     lukem 	# well as by build.sh.
    673  1.84     lukem 	#
    674  1.84     lukem 	if [ -z "$makewrapper" ]; then
    675  1.84     lukem 		makewrapper="$TOOLDIR/bin/${toolprefix}make-$MACHINE"
    676  1.84     lukem 		if [ ! -z "$BUILDID" ]; then
    677  1.84     lukem 			makewrapper="$makewrapper-$BUILDID"
    678  1.84     lukem 		fi
    679  1.52   thorpej 	fi
    680   1.4        tv 
    681  1.84     lukem 	$runcmd rm -f "$makewrapper"
    682  1.84     lukem 	if [ "$runcmd" = "echo" ]; then
    683  1.84     lukem 		echo 'cat <<EOF >'$makewrapper
    684  1.84     lukem 		makewrapout=
    685  1.84     lukem 	else
    686  1.84     lukem 		makewrapout=">>\$makewrapper"
    687  1.84     lukem 	fi
    688  1.18        tv 
    689  1.84     lukem 	eval cat <<EOF $makewrapout
    690   1.4        tv #! /bin/sh
    691   1.4        tv # Set proper variables to allow easy "make" building of a NetBSD subtree.
    692  1.93     lukem # Generated from:  \$NetBSD: build.sh,v 1.93 2003/03/04 02:20:28 lukem Exp $
    693   1.4        tv #
    694  1.18        tv 
    695  1.18        tv EOF
    696  1.84     lukem 	for f in $makeenv; do
    697  1.84     lukem 		eval echo "$f=\'\$$(echo $f)\'\;\ export\ $f" $makewrapout
    698  1.84     lukem 	done
    699  1.84     lukem 	eval echo "USETOOLS=yes\; export USETOOLS" $makewrapout
    700  1.18        tv 
    701  1.84     lukem 	eval cat <<EOF $makewrapout
    702  1.18        tv 
    703  1.84     lukem exec "\$TOOLDIR/bin/${toolprefix}make" \${1+"\$@"}
    704   1.4        tv EOF
    705  1.84     lukem 	[ "$runcmd" = "echo" ] && echo EOF
    706  1.84     lukem 	$runcmd chmod +x "$makewrapper"
    707  1.84     lukem 	$runcmd echo "===> Updated ${makewrapper}"
    708  1.84     lukem }
    709  1.84     lukem 
    710  1.84     lukem buildtools()
    711  1.84     lukem {
    712  1.84     lukem 	if [ "$MKOBJDIRS" != "no" ]; then
    713  1.84     lukem 		$runcmd "$makewrapper" $parallel obj-tools ||
    714  1.84     lukem 		    bomb "failed to make obj-tools"
    715  1.84     lukem 	fi
    716  1.84     lukem 	$runcmd cd tools
    717  1.86     lukem 	if [ -z "$UPDATE" ]; then
    718  1.84     lukem 		cleandir=cleandir
    719  1.84     lukem 	else
    720  1.84     lukem 		cleandir=
    721  1.84     lukem 	fi
    722  1.84     lukem 	$runcmd "$makewrapper" ${cleandir} dependall install ||
    723  1.84     lukem 	    bomb "failed to make tools"
    724  1.84     lukem }
    725   1.4        tv 
    726  1.84     lukem buildkernel()
    727  1.84     lukem {
    728  1.84     lukem 	kernconf="$1"
    729  1.84     lukem 	if ! $do_tools; then
    730  1.84     lukem 		# Building tools every time we build a kernel is clearly
    731  1.84     lukem 		# unnecessary.  We could try to figure out whether rebuilding
    732  1.84     lukem 		# the tools is necessary this time, but it doesn't seem worth
    733  1.84     lukem 		# the trouble.  Instead, we say it's the user's responsibility
    734  1.84     lukem 		# to rebuild the tools if necessary.
    735  1.84     lukem 		#
    736  1.84     lukem 		$runcmd echo "===> Building kernel without building new tools"
    737  1.84     lukem 	fi
    738  1.84     lukem 	$runcmd echo "===> Building kernel ${kernconf}"
    739  1.84     lukem 	if [ "$MKOBJDIRS" != "no" ] && [ ! -z "$makeobjdir" ]; then
    740  1.84     lukem 		# The correct value of KERNOBJDIR might
    741  1.84     lukem 		# depend on a prior "make obj" in
    742  1.84     lukem 		# ${KERNSRCDIR}/${KERNARCHDIR}/compile.
    743  1.84     lukem 		#
    744  1.84     lukem 		KERNSRCDIR="$(safe_getmakevar KERNSRCDIR)"
    745  1.84     lukem 		KERNARCHDIR="$(safe_getmakevar KERNARCHDIR)"
    746  1.84     lukem 		$runcmd cd "${KERNSRCDIR}/${KERNARCHDIR}/compile"
    747  1.84     lukem 		$runcmd "$makewrapper" obj ||
    748  1.84     lukem 		    bomb "failed to make obj in ${KERNSRCDIR}/${KERNARCHDIR}/compile"
    749  1.84     lukem 		$runcmd cd "$TOP"
    750  1.84     lukem 	fi
    751  1.84     lukem 	KERNCONFDIR="$(safe_getmakevar KERNCONFDIR)"
    752  1.84     lukem 	KERNOBJDIR="$(safe_getmakevar KERNOBJDIR)"
    753  1.84     lukem 	case "${kernconf}" in
    754  1.84     lukem 	*/*)
    755  1.84     lukem 		kernconfpath="${kernconf}"
    756  1.84     lukem 		kernconfbase="$(basename "${kernconf}")"
    757  1.84     lukem 		;;
    758  1.84     lukem 	*)
    759  1.84     lukem 		kernconfpath="${KERNCONFDIR}/${kernconf}"
    760  1.84     lukem 		kernconfbase="${kernconf}"
    761  1.84     lukem 		;;
    762  1.84     lukem 	esac
    763  1.84     lukem 	kernbuilddir="${KERNOBJDIR}/${kernconfbase}"
    764  1.84     lukem 	$runcmd echo "===> Kernel build directory: ${kernbuilddir}"
    765  1.84     lukem 	$runcmd mkdir -p "${kernbuilddir}" ||
    766  1.84     lukem 	    bomb "cannot mkdir: ${kernbuilddir}"
    767  1.86     lukem 	if [ -z "$UPDATE" ]; then
    768  1.84     lukem 		$runcmd cd "${kernbuilddir}"
    769  1.84     lukem 		$runcmd "$makewrapper" cleandir ||
    770  1.84     lukem 		    bomb "make cleandir failed in ${kernbuilddir}"
    771  1.84     lukem 		$runcmd cd "$TOP"
    772  1.16   thorpej 	fi
    773  1.84     lukem 	$runcmd "${TOOLDIR}/bin/${toolprefix}config" -b "${kernbuilddir}" \
    774  1.84     lukem 		-s "${TOP}/sys" "${kernconfpath}" ||
    775  1.84     lukem 	    bomb "${toolprefix}config failed for ${kernconf}"
    776  1.84     lukem 	$runcmd cd "${kernbuilddir}"
    777  1.84     lukem 	$runcmd "$makewrapper" depend ||
    778  1.84     lukem 	    bomb "make depend failed in ${kernbuilddir}"
    779  1.84     lukem 	$runcmd "$makewrapper" $parallel all ||
    780  1.84     lukem 	    bomb "make all failed in ${kernbuilddir}"
    781  1.91     lukem 
    782  1.91     lukem 	if [ "$runcmd" != "echo" ]; then
    783  1.91     lukem 		echo "===> New kernels built:"
    784  1.91     lukem 		kernlist=$(awk '$1 == "config" { print $2 }' ${kernconfpath})
    785  1.91     lukem 		for kern in ${kernlist:-netbsd}; do
    786  1.91     lukem 			[ -f "${kernbuilddir}/${kern}" ] && \
    787  1.91     lukem 			    echo "  ${kernbuilddir}/${kern}"
    788  1.91     lukem 		done
    789  1.91     lukem 		echo "."
    790  1.91     lukem 	fi
    791  1.84     lukem }
    792  1.84     lukem 
    793  1.84     lukem installworld()
    794  1.84     lukem {
    795  1.84     lukem 	dir="$1"
    796  1.84     lukem 	${runcmd} "$makewrapper" INSTALLWORLDDIR="${dir}" installworld ||
    797  1.84     lukem 	    bomb "failed to make installworld to ${dir}"
    798  1.84     lukem }
    799  1.84     lukem 
    800  1.84     lukem 
    801  1.84     lukem main()
    802  1.84     lukem {
    803  1.84     lukem 	initdefaults
    804  1.84     lukem 	parseoptions "$@"
    805  1.93     lukem 
    806  1.93     lukem 	build_start=$(date)
    807  1.93     lukem 	echo "===> ${progname} command: $0 $@"
    808  1.93     lukem 	echo "===> ${progname} started: $build_start"
    809  1.93     lukem 
    810  1.84     lukem 	rebuildmake
    811  1.84     lukem 	validatemakeparams
    812  1.84     lukem 	createmakewrapper
    813  1.84     lukem 
    814  1.84     lukem 	# Perform the operations.
    815  1.84     lukem 	#
    816  1.84     lukem 	for op in $operations; do
    817  1.84     lukem 		case "$op" in
    818  1.86     lukem 
    819  1.86     lukem 		makewrapper)
    820  1.86     lukem 			# no-op
    821  1.86     lukem 			;;
    822  1.84     lukem 
    823  1.84     lukem 		tools)
    824  1.84     lukem 			buildtools
    825  1.84     lukem 			;;
    826  1.84     lukem 
    827  1.87     lukem 		obj|build|distribution|release|sets)
    828  1.84     lukem 			${runcmd} "$makewrapper" $parallel $op ||
    829  1.84     lukem 			    bomb "failed to make $op"
    830  1.84     lukem 			;;
    831  1.84     lukem 
    832  1.84     lukem 		kernel=*)
    833  1.84     lukem 			arg=${op#*=}
    834  1.84     lukem 			buildkernel "${arg}"
    835  1.84     lukem 			;;
    836  1.84     lukem 
    837  1.84     lukem 		install=*)
    838  1.84     lukem 			arg=${op#*=}
    839  1.85     lukem 			if [ "${arg}" = "/" ] && \
    840  1.85     lukem 			    (	[ "${uname_s}" != "NetBSD" ] || \
    841  1.85     lukem 				[ "${uname_m}" != "$MACHINE" ] ); then
    842  1.85     lukem 				bomb "'${op}' must != / for cross builds."
    843  1.85     lukem 			fi
    844  1.84     lukem 			installworld "${arg}"
    845  1.70     lukem 			;;
    846  1.84     lukem 
    847  1.70     lukem 		*)
    848  1.84     lukem 			bomb "Unknown operation \`${op}'"
    849  1.70     lukem 			;;
    850  1.84     lukem 
    851  1.70     lukem 		esac
    852  1.84     lukem 	done
    853  1.93     lukem 
    854  1.93     lukem 	echo "===> ${progname} started: $build_start"
    855  1.93     lukem 	echo "===> ${progname} ended:   $(date)"
    856  1.84     lukem }
    857  1.84     lukem 
    858  1.84     lukem main "$@"
    859