#!/bin/ksh -p # # CDDL HEADER START # # The contents of this file are subject to the terms of the # Common Development and Distribution License (the "License"). # You may not use this file except in compliance with the License. # # You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE # or http://www.opensolaris.org/os/licensing. # See the License for the specific language governing permissions # and limitations under the License. # # When distributing Covered Code, include this CDDL HEADER in each # file and include the License file at usr/src/OPENSOLARIS.LICENSE. # If applicable, add the following below this CDDL HEADER, with the # fields enclosed by brackets "[]" replaced with your own identifying # information: Portions Copyright [yyyy] [name of copyright owner] # # CDDL HEADER END # # # Copyright 2008 Sun Microsystems, Inc. All rights reserved. # Use is subject to license terms. # # ident "%Z%%M% %I% %E% SMI" # # Based on the nightly script from the integration folks, # Mostly modified and owned by mike_s. # Changes also by kjc, dmk. # # BRINGOVER_WS may be specified in the env file. # The default is the old behavior of CLONE_WS # # -i on the command line, means fast options, so when it's on the # command line (only), lint and check builds are skipped no matter what # the setting of their individual flags are in NIGHTLY_OPTIONS. # # LINTDIRS can be set in the env file, format is a list of: # # /dirname-to-run-lint-on flag # # Where flag is: y - enable lint noise diff output # n - disable lint noise diff output # # For example: LINTDIRS="$SRC/uts n $SRC/stand y $SRC/psm y" # # -A flag in NIGHTLY_OPTIONS checks ABI diffs in .so files # This option requires a couple of scripts. # # OPTHOME and TEAMWARE may be set in the environment to override /opt # and /opt/teamware defaults. # # # The CDPATH variable causes ksh's `cd' builtin to emit messages to stdout # under certain circumstances, which can really screw things up; unset it. # unset CDPATH # Get the absolute path of the nightly script that the user invoked. This # may be a relative path, and we need to do this before changing directory. nightly_path=`whence $0` # # Keep track of where we found nightly so we can invoke the matching # which_scm script. If that doesn't work, don't go guessing, just rely # on the $PATH settings, which will generally give us either /opt/onbld # or the user's workspace. # WHICH_SCM=$(dirname $nightly_path)/which_scm if [[ ! -x $WHICH_SCM ]]; then WHICH_SCM=which_scm fi # # Print the tag string used to identify a build (e.g., "DEBUG # open-only") # usage: tagstring debug-part open-part # tagstring() { debug_part=$1 open_part=$2 if [ -n "$open_part" ]; then echo "$debug_part $open_part" else echo "$debug_part" fi } # # Function to do a DEBUG and non-DEBUG build. Needed because we might # need to do another for the source build, and since we only deliver DEBUG or # non-DEBUG packages. # # usage: normal_build [-O] # -O OpenSolaris delivery build. Put the proto area and # (eventually) packages in -open directories. Use skeleton # closed binaries. Don't generate archives--that needs to be # done later, after we've generated the closed binaries. Also # skip the package build (until 6414822 is fixed). # normal_build() { typeset orig_p_FLAG="$p_FLAG" typeset orig_a_FLAG="$a_FLAG" suffix="" open_only="" while getopts O FLAG $*; do case $FLAG in O) suffix="-open" open_only="open-only" p_FLAG=n a_FLAG=n ;; esac done # non-DEBUG build begins if [ "$F_FLAG" = "n" ]; then set_non_debug_build_flags mytag=`tagstring "non-DEBUG" "$open_only"` build "$mytag" "$suffix-nd" "$MULTI_PROTO" if [ "$build_ok" = "y" -a "$X_FLAG" = "y" -a \ "$p_FLAG" = "y" ]; then copy_ihv_pkgs non-DEBUG -nd fi else echo "\n==== No non-DEBUG $open_only build ====\n" >> "$LOGFILE" fi # non-DEBUG build ends # DEBUG build begins if [ "$D_FLAG" = "y" ]; then set_debug_build_flags mytag=`tagstring "DEBUG" "$open_only"` build "$mytag" "$suffix" "$MULTI_PROTO" if [ "$build_ok" = "y" -a "$X_FLAG" = "y" -a \ "$p_FLAG" = "y" ]; then copy_ihv_pkgs DEBUG "" fi else echo "\n==== No DEBUG $open_only build ====\n" >> "$LOGFILE" fi # DEBUG build ends p_FLAG="$orig_p_FLAG" a_FLAG="$orig_a_FLAG" } # # usage: run_hook HOOKNAME ARGS... # # If variable "$HOOKNAME" is defined, insert a section header into # our logs and then run the command with ARGS # run_hook() { HOOKNAME=$1 eval HOOKCMD=\$$HOOKNAME shift if [ -n "$HOOKCMD" ]; then ( echo "\n==== Running $HOOKNAME command: $HOOKCMD ====\n" ( $HOOKCMD "$@" 2>&1 ) if [ "$?" -ne 0 ]; then # Let exit status propagate up touch $TMPDIR/abort fi ) | tee -a $mail_msg_file >> $LOGFILE if [ -f $TMPDIR/abort ]; then build_ok=n echo "\nAborting at request of $HOOKNAME" | tee -a $mail_msg_file >> $LOGFILE exit 1 fi fi } # # usage: filelist DESTDIR PATTERN # filelist() { DEST=$1 PATTERN=$2 cd ${DEST} OBJFILES=${ORIG_SRC}/xmod/obj_files if [ ! -f ${OBJFILES} ]; then return; fi for i in `grep -v '^#' ${OBJFILES} | \ grep ${PATTERN} | cut -d: -f2 | tr -d ' \t'` do # wildcard expansion for j in $i do if [ -f "$j" ]; then echo $j fi if [ -d "$j" ]; then echo $j fi done done | sort | uniq } # function to save off binaries after a full build for later # restoration save_binaries() { # save off list of binaries echo "\n==== Saving binaries from build at `date` ====\n" | \ tee -a $mail_msg_file >> $LOGFILE rm -f ${BINARCHIVE} cd ${CODEMGR_WS} filelist ${CODEMGR_WS} '^preserve:' >> $LOGFILE filelist ${CODEMGR_WS} '^preserve:' | \ cpio -ocB 2>/dev/null | compress \ > ${BINARCHIVE} } # delete files # usage: hybridize_files DESTDIR MAKE_TARGET hybridize_files() { DEST=$1 MAKETARG=$2 echo "\n==== Hybridizing files at `date` ====\n" | \ tee -a $mail_msg_file >> $LOGFILE for i in `filelist ${DEST} '^delete:'` do echo "removing ${i}." | tee -a $mail_msg_file >> $LOGFILE rm -rf "${i}" done for i in `filelist ${DEST} '^hybridize:' ` do echo "hybridizing ${i}." | tee -a $mail_msg_file >> $LOGFILE rm -f ${i}+ sed -e "/^# HYBRID DELETE START/,/^# HYBRID DELETE END/d" \ < ${i} > ${i}+ mv ${i}+ ${i} done } # restore binaries into the proper source tree. # usage: restore_binaries DESTDIR MAKE_TARGET restore_binaries() { DEST=$1 MAKETARG=$2 echo "\n==== Restoring binaries to ${MAKETARG} at `date` ====\n" | \ tee -a $mail_msg_file >> $LOGFILE cd ${DEST} zcat ${BINARCHIVE} | \ cpio -idmucvB 2>/dev/null | tee -a $mail_msg_file >> ${LOGFILE} } # rename files we save binaries of # usage: rename_files DESTDIR MAKE_TARGET rename_files() { DEST=$1 MAKETARG=$2 echo "\n==== Renaming source files in ${MAKETARG} at `date` ====\n" | \ tee -a $mail_msg_file >> $LOGFILE for i in `filelist ${DEST} '^rename:'` do echo ${i} | tee -a $mail_msg_file >> ${LOGFILE} rm -f ${i}.export mv ${i} ${i}.export done } # # Copy some or all of the source tree. # # Returns 0 for success, non-zero for failure. # # usage: copy_source CODEMGR_WS DESTDIR LABEL SRCROOT # copy_source() { WS=$1 DEST=$2 label=$3 srcroot=$4 printf "\n==== Creating %s source from %s (%s) ====\n\n" \ "$DEST" "$WS" "$label" | tee -a $mail_msg_file >> $LOGFILE printf "cleaning out %s\n" "$DEST." >> $LOGFILE rm -rf "$DEST" >> $LOGFILE 2>&1 printf "creating %s\n" "$DEST." >> $LOGFILE mkdir -p "$DEST" 2>> $LOGFILE if (( $? != 0 )) ; then printf "failed to create %s\n" "$DEST" | tee -a $mail_msg_file >> $LOGFILE build_ok=n return 1 fi cd "$WS" printf "populating %s\n" "$DEST." >> $LOGFILE case "$SCM_TYPE" in teamware) find $srcroot -name 's\.*' -a -type f -print | \ sed -e 's,SCCS\/s.,,' | \ grep -v '/\.del-*' | \ cpio -pd $DEST >>$LOGFILE 2>&1 if (( $? != 0 )) ; then printf "cpio failed for %s\n" "$DEST" | tee -a $mail_msg_file >> $LOGFILE build_ok=n return 1 fi ;; mercurial) copy_source_mercurial $DEST $srcroot if (( $? != 0 )) ; then build_ok=n return 1 fi ;; *) build_ok=n echo "Tree copy is not supported for workspace type" \ "$SCM_TYPE" | tee -a $mail_msg_file >> $LOGFILE return 1 ;; esac return 0 } # # Mercurial-specific copy code for copy_source(). Handles the # combined open and closed trees. # # Returns 0 for success, non-zero for failure. # # usage: copy_source_mercurial destdir srcroot # function copy_source_mercurial { typeset dest=$1 typeset srcroot=$2 typeset open_top closed_top case $srcroot in usr) open_top=usr if [[ "$CLOSED_IS_PRESENT" = yes ]]; then closed_top=usr/closed fi ;; usr/closed*) if [[ "$CLOSED_IS_PRESENT" = no ]]; then printf "can't copy %s: closed tree not present.\n" \ "$srcroot" | tee -a $mail_msg_file >> $LOGFILE return 1 fi closed_top="$srcroot" ;; *) open_top="$srcroot" ;; esac if [[ -n "$open_top" ]]; then hg locate -I "$open_top" | cpio -pd "$dest" >>$LOGFILE 2>&1 if (( $? != 0 )) ; then printf "cpio failed for %s\n" "$dest" | tee -a $mail_msg_file >> $LOGFILE return 1 fi fi if [[ -n "$closed_top" ]]; then mkdir -p "$dest/usr/closed" || return 1 if [[ "$closed_top" = usr/closed ]]; then (cd usr/closed; hg locate | cpio -pd "$dest/usr/closed") >>$LOGFILE 2>&1 if (( $? != 0 )) ; then printf "cpio failed for %s/usr/closed\n" \ "$dest" | tee -a $mail_msg_file >> $LOGFILE return 1 fi else # copy subtree of usr/closed closed_top=${closed_top#usr/closed/} (cd usr/closed; hg locate -I "$closed_top" | cpio -pd "$dest/usr/closed") >>$LOGFILE 2>&1 if (( $? != 0 )) ; then printf "cpio failed for %s/usr/closed/%s\n" \ "$dest" "$closed_top" | tee -a $mail_msg_file >> $LOGFILE return 1 fi fi fi return 0 } # # function to create (but not build) the export/crypt source tree. # usage: set_up_source_build CODEMGR_WS DESTDIR MAKE_TARGET # Sets SRC to the modified source tree, for use by the caller when it # builds the tree. # set_up_source_build() { WS=$1 DEST=$2 MAKETARG=$3 copy_source $WS $DEST $MAKETARG usr if (( $? != 0 )); then echo "\nCould not copy source tree for source build." | tee -a $mail_msg_file >> $LOGFILE build_ok=n return fi SRC=${DEST}/usr/src cd $SRC rm -f ${MAKETARG}.out echo "making ${MAKETARG} in ${SRC}." >> $LOGFILE /bin/time $MAKE -e ${MAKETARG} 2>&1 | \ tee -a $SRC/${MAKETARG}.out >> $LOGFILE echo "\n==== ${MAKETARG} build errors ====\n" >> $mail_msg_file egrep ":" $SRC/${MAKETARG}.out | \ egrep -e "(^${MAKE}:|[ ]error[: \n])" | \ egrep -v "Ignoring unknown host" | \ egrep -v "warning" >> $mail_msg_file echo "clearing state files." >> $LOGFILE find . -name '.make*' -exec rm -f {} \; cd ${DEST} if [ "${MAKETARG}" = "CRYPT_SRC" ]; then rm -f ${CODEMGR_WS}/crypt_files.cpio.Z echo "\n==== xmod/cry_files that don't exist ====\n" | \ tee -a $mail_msg_file >> $LOGFILE CRYPT_FILES=${WS}/usr/src/xmod/cry_files for i in `cat ${CRYPT_FILES}` do # make sure the files exist if [ -f "$i" ]; then continue fi if [ -d "$i" ]; then continue fi echo "$i" | tee -a $mail_msg_file >> $LOGFILE done find `cat ${CRYPT_FILES}` -print 2>/dev/null | \ cpio -ocB 2>/dev/null | \ compress > ${CODEMGR_WS}/crypt_files.cpio.Z fi if [ "${MAKETARG}" = "EXPORT_SRC" ]; then # rename first, since we might restore a file # of the same name (mapfiles) rename_files ${EXPORT_SRC} EXPORT_SRC if [ "$SH_FLAG" = "y" ]; then hybridize_files ${EXPORT_SRC} EXPORT_SRC fi fi # save the cleartext echo "\n==== Creating ${MAKETARG}.cpio.Z ====\n" | \ tee -a $mail_msg_file >> $LOGFILE cd ${DEST} rm -f ${MAKETARG}.cpio.Z find usr -depth -print | \ grep -v usr/src/${MAKETARG}.out | \ cpio -ocB 2>/dev/null | \ compress > ${CODEMGR_WS}/${MAKETARG}.cpio.Z if [ "${MAKETARG}" = "EXPORT_SRC" ]; then restore_binaries ${EXPORT_SRC} EXPORT_SRC fi if [ "${MAKETARG}" = "CRYPT_SRC" ]; then restore_binaries ${CRYPT_SRC} CRYPT_SRC fi } # Return library search directive as function of given root. myldlibs() { echo "-L$1/lib -L$1/usr/lib" } # Return header search directive as function of given root. myheaders() { echo "-I$1/usr/include" } # # Wrapper over commands that generate BFU archives. The entire # command output gets written to LOGFILE, and any unexpected messages # are written to the mail message. Returns with the status of the # original command. # makebfu_filt() { typeset tmplog typeset errors typeset cmd integer cmd_stat cmd="$1" shift tmplog="$TMPDIR/$cmd.out" errors="$TMPDIR/$cmd-errors" $cmd $* > "$tmplog" 2>&1 cmd_stat=$? cat "$tmplog" >> "$LOGFILE" grep -v "^Creating .* archive:" "$tmplog" | grep -v "^Making" | \ grep -v "^$" | sort -u > "$errors" if [[ -s "$errors" ]]; then echo "\n==== cpio archives build errors ($LABEL) ====\n" \ >> "$mail_msg_file" cat "$errors" >> "$mail_msg_file" fi rm -f "$tmplog" "$errors" return $cmd_stat } # # Function to do the build, including cpio archive and package generation. # usage: build LABEL SUFFIX MULTIPROTO # - LABEL is used to tag build output. # - SUFFIX is used to distinguish files (e.g., debug vs non-debug). # - If MULTIPROTO is "yes", it means to name the proto area according to # SUFFIX. Otherwise ("no"), (re)use the standard proto area. # build() { LABEL=$1 SUFFIX=$2 MULTIPROTO=$3 INSTALLOG=install${SUFFIX}-${MACH} NOISE=noise${SUFFIX}-${MACH} CPIODIR=${CPIODIR_ORIG}${SUFFIX} PKGARCHIVE=${PKGARCHIVE_ORIG}${SUFFIX} if [ "$SPARC_RM_PKGARCHIVE_ORIG" ]; then SPARC_RM_PKGARCHIVE=${SPARC_RM_PKGARCHIVE_ORIG}${SUFFIX} fi ORIGROOT=$ROOT [ $MULTIPROTO = no ] || export ROOT=$ROOT$SUFFIX export ENVLDLIBS1=`myldlibs $ROOT` export ENVCPPFLAGS1=`myheaders $ROOT` this_build_ok=y # # Build OS-Networking source # echo "\n==== Building OS-Net source at `date` ($LABEL) ====\n" \ >> $LOGFILE rm -f $SRC/${INSTALLOG}.out cd $SRC /bin/time $MAKE -e install 2>&1 | \ tee -a $SRC/${INSTALLOG}.out >> $LOGFILE if [[ "$SCM_TYPE" = teamware ]]; then echo "\n==== SCCS Noise ($LABEL) ====\n" >> $mail_msg_file egrep 'sccs(check:| *get)' $SRC/${INSTALLOG}.out >> \ $mail_msg_file fi echo "\n==== Build errors ($LABEL) ====\n" >> $mail_msg_file egrep ":" $SRC/${INSTALLOG}.out | egrep -e "(^${MAKE}:|[ ]error[: \n])" | \ egrep -v "Ignoring unknown host" | \ egrep -v "cc .* -o error " | \ egrep -v "warning" >> $mail_msg_file if [ "$?" = "0" ]; then build_ok=n this_build_ok=n fi grep "bootblock image is .* bytes too big" $SRC/${INSTALLOG}.out \ >> $mail_msg_file if [ "$?" = "0" ]; then build_ok=n this_build_ok=n fi if [ "$W_FLAG" = "n" ]; then echo "\n==== Build warnings ($LABEL) ====\n" >>$mail_msg_file egrep -i warning: $SRC/${INSTALLOG}.out \ | egrep -v '^tic:' \ | egrep -v "symbol \`timezone' has differing types:" \ | egrep -v "parameter set to" \ | egrep -v "Ignoring unknown host" \ | egrep -v "redefining segment flags attribute for" \ >> $mail_msg_file fi echo "\n==== Ended OS-Net source build at `date` ($LABEL) ====\n" \ >> $LOGFILE echo "\n==== Elapsed build time ($LABEL) ====\n" >>$mail_msg_file tail -3 $SRC/${INSTALLOG}.out >>$mail_msg_file if [ "$i_FLAG" = "n" -a "$W_FLAG" = "n" ]; then rm -f $SRC/${NOISE}.ref if [ -f $SRC/${NOISE}.out ]; then mv $SRC/${NOISE}.out $SRC/${NOISE}.ref fi grep : $SRC/${INSTALLOG}.out \ | egrep -v '^/' \ | egrep -v '^(Start|Finish|real|user|sys|./bld_awk)' \ | egrep -v '^tic:' \ | egrep -v '^mcs' \ | egrep -v '^LD_LIBRARY_PATH=' \ | egrep -v 'ar: creating' \ | egrep -v 'ar: writing' \ | egrep -v 'conflicts:' \ | egrep -v ':saved created' \ | egrep -v '^stty.*c:' \ | egrep -v '^mfgname.c:' \ | egrep -v '^uname-i.c:' \ | egrep -v '^volumes.c:' \ | egrep -v '^lint library construction:' \ | egrep -v 'tsort: INFORM:' \ | egrep -v 'stripalign:' \ | egrep -v 'chars, width' \ | egrep -v "symbol \`timezone' has differing types:" \ | egrep -v 'PSTAMP' \ | egrep -v '|%WHOANDWHERE%|' \ | egrep -v '^Manifying' \ | egrep -v 'Ignoring unknown host' \ | egrep -v 'Processing method:' \ | egrep -v '^Writing' \ | egrep -v 'spellin1:' \ | egrep -v '^adding:' \ | egrep -v "^echo 'msgid" \ | egrep -v '^echo ' \ | egrep -v '\.c:$' \ | egrep -v '^Adding file:' \ | egrep -v 'CLASSPATH=' \ | egrep -v '\/var\/mail\/:saved' \ | egrep -v -- '-DUTS_VERSION=' \ | egrep -v '^Running Mkbootstrap' \ | egrep -v '^Applet length read:' \ | egrep -v 'bytes written:' \ | egrep -v '^File:SolarisAuthApplet.bin' \ | egrep -v -i 'jibversion' \ | egrep -v '^Output size:' \ | egrep -v '^Solo size statistics:' \ | egrep -v '^Using ROM API Version' \ | egrep -v '^Zero Signature length:' \ | egrep -v '^Note \(probably harmless\):' \ | egrep -v '::' \ | egrep -v -- '-xcache' \ | egrep -v '^\+' \ | egrep -v '^cc1: note: -fwritable-strings' \ | egrep -v 'svccfg-native -s svc:/' \ | sort | uniq >$SRC/${NOISE}.out if [ ! -f $SRC/${NOISE}.ref ]; then cp $SRC/${NOISE}.out $SRC/${NOISE}.ref fi echo "\n==== Build noise differences ($LABEL) ====\n" \ >>$mail_msg_file diff $SRC/${NOISE}.ref $SRC/${NOISE}.out >>$mail_msg_file fi # # Re-sign selected binaries using signing server # (gatekeeper builds only) # if [ -n "$CODESIGN_USER" -a "$this_build_ok" = "y" ]; then echo "\n==== Signing proto area at `date` ====\n" >> $LOGFILE signing_file="${TMPDIR}/signing" rm -f ${signing_file} export CODESIGN_USER signproto $SRC/tools/codesign/creds 2>&1 | \ tee -a ${signing_file} >> $LOGFILE echo "\n==== Finished signing proto area at `date` ====\n" \ >> $LOGFILE echo "\n==== Crypto module signing errors ($LABEL) ====\n" \ >> $mail_msg_file egrep 'WARNING|ERROR' ${signing_file} >> $mail_msg_file if (( $? == 0 )) ; then build_ok=n this_build_ok=n fi fi # # Create cpio archives for preintegration testing (PIT) # if [ "$a_FLAG" = "y" -a "$this_build_ok" = "y" ]; then echo "\n==== Creating $LABEL cpio archives at `date` ====\n" \ >> $LOGFILE makebfu_filt makebfu # hack for test folks if [ -z "`echo $PARENT_WS|egrep '^\/ws\/'`" ]; then X=/net/`uname -n`${CPIODIR} else X=${CPIODIR} fi echo "Archive_directory: ${X}" >${TMPDIR}/f cp ${TMPDIR}/f $(dirname $(dirname ${CPIODIR}))/.${MACH}_wgtrun rm -f ${TMPDIR}/f else echo "\n==== Not creating $LABEL cpio archives ====\n" \ >> $LOGFILE fi # # Building Packages # if [ "$p_FLAG" = "y" -a "$this_build_ok" = "y" ]; then echo "\n==== Creating $LABEL packages at `date` ====\n" \ >> $LOGFILE rm -f $SRC/pkgdefs/${INSTALLOG}.out echo "Clearing out $PKGARCHIVE ..." >> $LOGFILE rm -rf $PKGARCHIVE mkdir -p $PKGARCHIVE # # Optional build of sparc realmode on i386 # if [ "$MACH" = "i386" ] && [ "${SPARC_RM_PKGARCHIVE}" ]; then echo "Clearing out ${SPARC_RM_PKGARCHIVE} ..." \ >> $LOGFILE rm -rf ${SPARC_RM_PKGARCHIVE} mkdir -p ${SPARC_RM_PKGARCHIVE} fi cd $SRC/pkgdefs $MAKE -e install 2>&1 | \ tee -a $SRC/pkgdefs/${INSTALLOG}.out >> $LOGFILE echo "\n==== Package build errors ($LABEL) ====\n" \ >> $mail_msg_file egrep "${MAKE}|ERROR|WARNING" $SRC/pkgdefs/${INSTALLOG}.out | \ grep ':' | \ grep -v PSTAMP | \ egrep -v "Ignoring unknown host" \ >> $mail_msg_file else echo "\n==== Not creating $LABEL packages ====\n" >> $LOGFILE fi ROOT=$ORIGROOT } # Usage: dolint /dir y|n # Arg. 2 is a flag to turn on/off the lint diff output dolint() { if [ ! -d "$1" ]; then echo "dolint error: $1 is not a directory" exit 1 fi if [ "$2" != "y" -a "$2" != "n" ]; then echo "dolint internal error: $2 should be 'y' or 'n'" exit 1 fi lintdir=$1 dodiff=$2 base=`basename $lintdir` LINTOUT=$lintdir/lint-${MACH}.out LINTNOISE=$lintdir/lint-noise-${MACH} export ENVLDLIBS1=`myldlibs $ROOT` export ENVCPPFLAGS1=`myheaders $ROOT` set_debug_build_flags # # '$MAKE lint' in $lintdir # echo "\n==== Begin '$MAKE lint' of $base at `date` ====\n" >> $LOGFILE # remove old lint.out rm -f $lintdir/lint.out $lintdir/lint-noise.out if [ -f $lintdir/lint-noise.ref ]; then mv $lintdir/lint-noise.ref ${LINTNOISE}.ref fi rm -f $LINTOUT cd $lintdir # # Remove all .ln files to ensure a full reference file # rm -f Nothing_to_remove \ `find . \( -name SCCS -o -name .hg -o -name .svn \) \ -prune -o -type f -name '*.ln' -print ` /bin/time $MAKE -ek lint 2>&1 | \ tee -a $LINTOUT >> $LOGFILE echo "\n==== '$MAKE lint' of $base ERRORS ====\n" >> $mail_msg_file grep "$MAKE:" $LINTOUT | egrep -v "Ignoring unknown host" \ >> $mail_msg_file echo "\n==== Ended '$MAKE lint' of $base at `date` ====\n" >> $LOGFILE echo "\n==== Elapsed time of '$MAKE lint' of $base ====\n" \ >>$mail_msg_file tail -3 $LINTOUT >>$mail_msg_file rm -f ${LINTNOISE}.ref if [ -f ${LINTNOISE}.out ]; then mv ${LINTNOISE}.out ${LINTNOISE}.ref fi grep : $LINTOUT | \ egrep -v '^(real|user|sys)' | egrep -v '(library construction)' | \ egrep -v ': global crosschecks' | \ egrep -v 'Ignoring unknown host' | \ egrep -v '\.c:$' | \ sort | uniq > ${LINTNOISE}.out if [ ! -f ${LINTNOISE}.ref ]; then cp ${LINTNOISE}.out ${LINTNOISE}.ref fi if [ "$dodiff" != "n" ]; then echo "\n==== lint warnings $base ====\n" \ >>$mail_msg_file # should be none, though there are a few that were filtered out # above egrep -i '(warning|lint):' ${LINTNOISE}.out \ | sort | uniq >> $mail_msg_file echo "\n==== lint noise differences $base ====\n" \ >> $mail_msg_file diff ${LINTNOISE}.ref ${LINTNOISE}.out \ >> $mail_msg_file fi } # Install proto area from IHV build copy_ihv_proto() { echo "\n==== Installing IHV proto area ====\n" \ >> $LOGFILE if [ -d "$IA32_IHV_ROOT" ]; then if [ ! -d "$ROOT" ]; then echo "mkdir -p $ROOT" >> $LOGFILE mkdir -p $ROOT fi echo "copying $IA32_IHV_ROOT to $ROOT\n" >> $LOGFILE cd $IA32_IHV_ROOT tar -cf - . | (cd $ROOT; umask 0; tar xpf - ) 2>&1 >> $LOGFILE else echo "$IA32_IHV_ROOT: not found" >> $LOGFILE fi if [ "$MULTI_PROTO" = yes ]; then if [ ! -d "$ROOT-nd" ]; then echo "mkdir -p $ROOT-nd" >> $LOGFILE mkdir -p $ROOT-nd fi # If there's a non-debug version of the IHV proto area, # copy it, but copy something if there's not. if [ -d "$IA32_IHV_ROOT-nd" ]; then echo "copying $IA32_IHV_ROOT-nd to $ROOT-nd\n" >> $LOGFILE cd $IA32_IHV_ROOT-nd elif [ -d "$IA32_IHV_ROOT" ]; then echo "copying $IA32_IHV_ROOT to $ROOT-nd\n" >> $LOGFILE cd $IA32_IHV_ROOT else echo "$IA32_IHV_ROOT{-nd,}: not found" >> $LOGFILE return fi tar -cf - . | (cd $ROOT-nd; umask 0; tar xpf - ) 2>&1 >> $LOGFILE fi } # Install IHV packages in PKGARCHIVE # usage: copy_ihv_pkgs LABEL SUFFIX copy_ihv_pkgs() { LABEL=$1 SUFFIX=$2 # always use non-DEBUG IHV packages IA32_IHV_PKGS=${IA32_IHV_PKGS_ORIG}-nd PKGARCHIVE=${PKGARCHIVE_ORIG}${SUFFIX} echo "\n==== Installing IHV packages from $IA32_IHV_PKGS ($LABEL) ====\n" \ >> $LOGFILE if [ -d "$IA32_IHV_PKGS" ]; then cd $IA32_IHV_PKGS tar -cf - * | \ (cd $PKGARCHIVE; umask 0; tar xpf - ) 2>&1 >> $LOGFILE else echo "$IA32_IHV_PKGS: not found" >> $LOGFILE fi echo "\n==== Installing IHV packages from $IA32_IHV_BINARY_PKGS ($LABEL) ====\n" \ >> $LOGFILE if [ -d "$IA32_IHV_BINARY_PKGS" ]; then cd $IA32_IHV_BINARY_PKGS tar -cf - * | \ (cd $PKGARCHIVE; umask 0; tar xpf - ) 2>&1 >> $LOGFILE else echo "$IA32_IHV_BINARY_PKGS: not found" >> $LOGFILE fi } # # Build and install the onbld tools. # # usage: build_tools DESTROOT # # returns non-zero status if the build was successful. # build_tools() { DESTROOT=$1 INSTALLOG=install-${MACH} echo "\n==== Building tools at `date` ====\n" \ >> $LOGFILE rm -f ${TOOLS}/${INSTALLOG}.out cd ${TOOLS} /bin/time $MAKE ROOT=${DESTROOT} -e install 2>&1 | \ tee -a ${TOOLS}/${INSTALLOG}.out >> $LOGFILE echo "\n==== Tools build errors ====\n" >> $mail_msg_file egrep ":" ${TOOLS}/${INSTALLOG}.out | egrep -e "(${MAKE}:|[ ]error[: \n])" | \ egrep -v "Ignoring unknown host" | \ egrep -v warning >> $mail_msg_file return $? } # # Set up to use locally installed tools. # # usage: use_tools TOOLSROOT # use_tools() { TOOLSROOT=$1 STABS=${TOOLSROOT}/opt/onbld/bin/${MACH}/stabs export STABS CTFSTABS=${TOOLSROOT}/opt/onbld/bin/${MACH}/ctfstabs export CTFSTABS GENOFFSETS=${TOOLSROOT}/opt/onbld/bin/genoffsets export GENOFFSETS CTFCONVERT=${TOOLSROOT}/opt/onbld/bin/${MACH}/ctfconvert export CTFCONVERT CTFMERGE=${TOOLSROOT}/opt/onbld/bin/${MACH}/ctfmerge export CTFMERGE CTFCVTPTBL=${TOOLSROOT}/opt/onbld/bin/ctfcvtptbl export CTFCVTPTBL CTFFINDMOD=${TOOLSROOT}/opt/onbld/bin/ctffindmod export CTFFINDMOD if [ "$VERIFY_ELFSIGN" = "y" ]; then ELFSIGN=${TOOLSROOT}/opt/onbld/bin/elfsigncmp else ELFSIGN=${TOOLSROOT}/opt/onbld/bin/${MACH}/elfsign fi export ELFSIGN PATH="${TOOLSROOT}/opt/onbld/bin/${MACH}:${PATH}" PATH="${TOOLSROOT}/opt/onbld/bin:${PATH}" export PATH echo "\n==== New environment settings. ====\n" >> $LOGFILE echo "STABS=${STABS}" >> $LOGFILE echo "CTFSTABS=${CTFSTABS}" >> $LOGFILE echo "CTFCONVERT=${CTFCONVERT}" >> $LOGFILE echo "CTFMERGE=${CTFMERGE}" >> $LOGFILE echo "CTFCVTPTBL=${CTFCVTPTBL}" >> $LOGFILE echo "CTFFINDMOD=${CTFFINDMOD}" >> $LOGFILE echo "ELFSIGN=${ELFSIGN}" >> $LOGFILE echo "PATH=${PATH}" >> $LOGFILE } staffer() { if [ $ISUSER -ne 0 ]; then "$@" else arg="\"$1\"" shift for i do arg="$arg \"$i\"" done eval su $STAFFER -c \'$arg\' fi } # # Verify that the closed tree is present if it needs to be. # Sets CLOSED_IS_PRESENT for future use. # check_closed_tree() { if [ -z "$CLOSED_IS_PRESENT" ]; then if [ -d $SRC/../closed ]; then CLOSED_IS_PRESENT="yes" else CLOSED_IS_PRESENT="no" fi export CLOSED_IS_PRESENT fi if [[ "$CLOSED_IS_PRESENT" = no && ! -d "$ON_CLOSED_BINS" ]]; then # # If it's an old (pre-split) tree or an empty # workspace, don't complain. # if grep -s CLOSED_BUILD $SRC/Makefile.master > /dev/null; then echo "If the closed sources are not present," \ "ON_CLOSED_BINS" echo "must point to the closed binaries tree." build_ok=n exit 1 fi fi } obsolete_build() { echo "WARNING: Obsolete $1 build requested; request will be ignored" } # # wrapper over wsdiff. # usage: do_wsdiff LABEL OLDPROTO NEWPROTO # do_wsdiff() { label=$1 oldproto=$2 newproto=$3 echo "\n==== Objects that differ since last build ($label) ====\n" | \ tee -a $LOGFILE >> $mail_msg_file wsdiff="wsdiff" [ "$t_FLAG" = y ] && wsdiff="wsdiff -t" $wsdiff -r ${TMPDIR}/wsdiff.results $oldproto $newproto 2>&1 | \ tee -a $LOGFILE >> $mail_msg_file } # # Functions for setting build flags (debug/non-debug). Keep them # together. # set_non_debug_build_flags() { export INTERNAL_RELEASE_BUILD ; INTERNAL_RELEASE_BUILD= export RELEASE_BUILD ; RELEASE_BUILD= unset EXTRA_OPTIONS unset EXTRA_CFLAGS } set_debug_build_flags() { export INTERNAL_RELEASE_BUILD ; INTERNAL_RELEASE_BUILD= unset RELEASE_BUILD unset EXTRA_OPTIONS unset EXTRA_CFLAGS } MACH=`uname -p` if [ "$OPTHOME" = "" ]; then OPTHOME=/opt export OPTHOME fi if [ "$TEAMWARE" = "" ]; then TEAMWARE=$OPTHOME/teamware export TEAMWARE fi USAGE='Usage: nightly [-in] [-V VERS ] [ -S E|D|H|O ] Where: -i Fast incremental options (no clobber, lint, check) -n Do not do a bringover -V VERS set the build version string to VERS -S Build a variant of the source product E - build exportable source D - build domestic source (exportable + crypt) H - build hybrid source (binaries + deleted source) O - build (only) open source file in Bourne shell syntax that sets and exports variables that configure the operation of this script and many of the scripts this one calls. If does not exist, it will be looked for in $OPTHOME/onbld/env. non-DEBUG is the default build type. Build options can be set in the NIGHTLY_OPTIONS variable in the as follows: -A check for ABI differences in .so files -C check for cstyle/hdrchk errors -D do a build with DEBUG on -F do _not_ do a non-DEBUG build -G gate keeper default group of options (-au) -I integration engineer default group of options (-ampu) -M do not run pmodes (safe file permission checker) -N do not run protocmp -O generate OpenSolaris deliverables -R default group of options for building a release (-mp) -U update proto area in the parent -V VERS set the build version string to VERS -X copy x86 IHV proto area -a create cpio archives -f find unreferenced files -i do an incremental build (no "make clobber") -l do "make lint" in $LINTDIRS (default: $SRC y) -m send mail to $MAILTO at end of build -n do not do a bringover -o build using root privileges to set OWNER/GROUP (old style) -p create packages -r check ELF runtime attributes in the proto area -t build and use the tools in $SRC/tools -u update proto_list_$MACH and friends in the parent workspace; when used with -f, also build an unrefmaster.out in the parent -w report on differences between previous and current proto areas -z compress cpio archives with gzip -W Do not report warnings (freeware gate ONLY) -S Build a variant of the source product E - build exportable source D - build domestic source (exportable + crypt) H - build hybrid source (binaries + deleted source) O - build (only) open source ' # # -x less public handling of xmod source for the source product # # A log file will be generated under the name $LOGFILE # for partially completed build and log.`date '+%F'` # in the same directory for fully completed builds. # # default values for low-level FLAGS; G I R are group FLAGS A_FLAG=n a_FLAG=n C_FLAG=n D_FLAG=n F_FLAG=n f_FLAG=n i_FLAG=n; i_CMD_LINE_FLAG=n l_FLAG=n M_FLAG=n m_FLAG=n N_FLAG=n n_FLAG=n O_FLAG=n o_FLAG=n P_FLAG=n p_FLAG=n r_FLAG=n T_FLAG=n t_FLAG=y U_FLAG=n u_FLAG=n V_FLAG=n W_FLAG=n w_FLAG=n X_FLAG=n z_FLAG=n SD_FLAG=n SE_FLAG=n SH_FLAG=n SO_FLAG=n # XMOD_OPT= # build_ok=y is_source_build() { [ "$SE_FLAG" = "y" -o "$SD_FLAG" = "y" -o \ "$SH_FLAG" = "y" -o "$SO_FLAG" = "y" ] return $? } # # examine arguments # # # single function for setting -S flag and doing error checking. # usage: set_S_flag # where is the source build type ("E", "D", ...). # set_S_flag() { if is_source_build; then echo "Can only build one source variant at a time." exit 1 fi if [ "$1" = "E" ]; then SE_FLAG=y elif [ "$1" = "D" ]; then SD_FLAG=y elif [ "$1" = "H" ]; then SH_FLAG=y elif [ "$1" = "O" ]; then SO_FLAG=y else echo "$USAGE" exit 1 fi } OPTIND=1 while getopts inS:tV: FLAG do case $FLAG in i ) i_FLAG=y; i_CMD_LINE_FLAG=y ;; n ) n_FLAG=y ;; S ) set_S_flag $OPTARG ;; +t ) t_FLAG=n ;; V ) V_FLAG=y V_ARG="$OPTARG" ;; \? ) echo "$USAGE" exit 1 ;; esac done # correct argument count after options shift `expr $OPTIND - 1` # test that the path to the environment-setting file was given if [ $# -ne 1 ]; then echo "$USAGE" exit 1 fi # check if user is running nightly as root # ISUSER is set non-zero if an ordinary user runs nightly, or is zero # when root invokes nightly. /usr/bin/id | grep '^uid=0(' >/dev/null 2>&1 ISUSER=$?; export ISUSER # # force locale to C LC_COLLATE=C; export LC_COLLATE LC_CTYPE=C; export LC_CTYPE LC_MESSAGES=C; export LC_MESSAGES LC_MONETARY=C; export LC_MONETARY LC_NUMERIC=C; export LC_NUMERIC LC_TIME=C; export LC_TIME # clear environment variables we know to be bad for the build unset LD_OPTIONS unset LD_AUDIT LD_AUDIT_32 LD_AUDIT_64 unset LD_BIND_NOW LD_BIND_NOW_32 LD_BIND_NOW_64 unset LD_BREADTH LD_BREADTH_32 LD_BREADTH_64 unset LD_CONFIG LD_CONFIG_32 LD_CONFIG_64 unset LD_DEBUG LD_DEBUG_32 LD_DEBUG_64 unset LD_DEMANGLE LD_DEMANGLE_32 LD_DEMANGLE_64 unset LD_FLAGS LD_FLAGS_32 LD_FLAGS_64 unset LD_LIBRARY_PATH LD_LIBRARY_PATH_32 LD_LIBRARY_PATH_64 unset LD_LOADFLTR LD_LOADFLTR_32 LD_LOADFLTR_64 unset LD_NOAUDIT LD_NOAUDIT_32 LD_NOAUDIT_64 unset LD_NOAUXFLTR LD_NOAUXFLTR_32 LD_NOAUXFLTR_64 unset LD_NOCONFIG LD_NOCONFIG_32 LD_NOCONFIG_64 unset LD_NODIRCONFIG LD_NODIRCONFIG_32 LD_NODIRCONFIG_64 unset LD_NODIRECT LD_NODIRECT_32 LD_NODIRECT_64 unset LD_NOLAZYLOAD LD_NOLAZYLOAD_32 LD_NOLAZYLOAD_64 unset LD_NOOBJALTER LD_NOOBJALTER_32 LD_NOOBJALTER_64 unset LD_NOVERSION LD_NOVERSION_32 LD_NOVERSION_64 unset LD_ORIGIN LD_ORIGIN_32 LD_ORIGIN_64 unset LD_PRELOAD LD_PRELOAD_32 LD_PRELOAD_64 unset LD_PROFILE LD_PROFILE_32 LD_PROFILE_64 unset CONFIG unset GROUP unset OWNER unset REMOTE unset ENV unset ARCH unset CLASSPATH unset NAME # # Setup environmental variables # if [ -f /etc/nightly.conf ]; then . /etc/nightly.conf fi if [ -f $1 ]; then if [[ $1 = */* ]]; then . $1 else . ./$1 fi else if [ -f $OPTHOME/onbld/env/$1 ]; then . $OPTHOME/onbld/env/$1 else echo "Cannot find env file as either $1 or $OPTHOME/onbld/env/$1" exit 1 fi fi # contents of stdenv.sh inserted after next line: # STDENV_START # STDENV_END # # place ourselves in a new task, respecting BUILD_PROJECT if set. # if [ -z "$BUILD_PROJECT" ]; then /usr/bin/newtask -c $$ else /usr/bin/newtask -c $$ -p $BUILD_PROJECT fi ps -o taskid= -p $$ | read build_taskid ps -o project= -p $$ | read build_project # # See if NIGHTLY_OPTIONS is set # if [ "$NIGHTLY_OPTIONS" = "" ]; then NIGHTLY_OPTIONS="-aBm" fi # # If BRINGOVER_WS was not specified, let it default to CLONE_WS # if [ "$BRINGOVER_WS" = "" ]; then BRINGOVER_WS=$CLONE_WS fi # # If CLOSED_BRINGOVER_WS was not specified, let it default to CLOSED_CLONE_WS # if [ "$CLOSED_BRINGOVER_WS" = "" ]; then CLOSED_BRINGOVER_WS=$CLOSED_CLONE_WS fi # # If BRINGOVER_FILES was not specified, default to usr # if [ "$BRINGOVER_FILES" = "" ]; then BRINGOVER_FILES="usr" fi # # If the closed sources are not present, the closed binaries must be # present for the build to succeed. If there's no pointer to the # closed binaries, flag that now, rather than forcing the user to wait # a couple hours (or more) to find out. # orig_closed_is_present="$CLOSED_IS_PRESENT" check_closed_tree # # Note: changes to the option letters here should also be applied to the # bldenv script. `d' is listed for backward compatibility. # NIGHTLY_OPTIONS=-${NIGHTLY_OPTIONS#-} OPTIND=1 while getopts AaBCDdFfGIilMmNnOoPpRrS:TtUuWwXxz FLAG $NIGHTLY_OPTIONS do case $FLAG in A ) A_FLAG=y ;; a ) a_FLAG=y ;; B ) D_FLAG=y ;; # old version of D C ) C_FLAG=y ;; D ) D_FLAG=y ;; F ) F_FLAG=y ;; f ) f_FLAG=y ;; G ) a_FLAG=y u_FLAG=y ;; I ) a_FLAG=y m_FLAG=y p_FLAG=y u_FLAG=y ;; i ) i_FLAG=y ;; l ) l_FLAG=y ;; M ) M_FLAG=y ;; m ) m_FLAG=y ;; N ) N_FLAG=y ;; n ) n_FLAG=y ;; O ) O_FLAG=y ;; o ) o_FLAG=y ;; P ) P_FLAG=y ;; # obsolete p ) p_FLAG=y ;; R ) m_FLAG=y p_FLAG=y ;; r ) r_FLAG=y ;; S ) set_S_flag $OPTARG ;; T ) T_FLAG=y ;; # obsolete +t ) t_FLAG=n ;; U ) if [ -z "${PARENT_ROOT}" ]; then echo "PARENT_ROOT must be set if the U flag is" \ "present in NIGHTLY_OPTIONS." exit 1 fi U_FLAG=y NIGHTLY_PARENT_ROOT=$PARENT_ROOT ;; u ) u_FLAG=y ;; W ) W_FLAG=y ;; w ) w_FLAG=y ;; X ) # now that we no longer need realmode builds, just # copy IHV packages. only meaningful on x86. if [ "$MACH" = "i386" ]; then X_FLAG=y fi ;; x ) XMOD_OPT="-x" ;; z ) z_FLAG=y ;; \? ) echo "$USAGE" exit 1 ;; esac done if [ $ISUSER -ne 0 ]; then if [ "$o_FLAG" = "y" ]; then echo "Old-style build requires root permission." exit 1 fi # Set default value for STAFFER, if needed. if [ -z "$STAFFER" -o "$STAFFER" = "nobody" ]; then STAFFER=`/usr/xpg4/bin/id -un` export STAFFER fi fi if [ -z "$MAILTO" -o "$MAILTO" = "nobody" ]; then MAILTO=$STAFFER export MAILTO fi PATH="$OPTHOME/onbld/bin:$OPTHOME/onbld/bin/${MACH}:/usr/ccs/bin" PATH="$PATH:$OPTHOME/SUNWspro/bin:$TEAMWARE/bin:/usr/bin:/usr/sbin:/usr/ucb" PATH="$PATH:/usr/openwin/bin:/usr/sfw/bin:/opt/sfw/bin:." export PATH # roots of source trees, both relative to $SRC and absolute. relsrcdirs="." if [[ -d $SRC/../closed && "$CLOSED_IS_PRESENT" != no ]]; then relsrcdirs="$relsrcdirs ../closed" fi abssrcdirs="" for d in $relsrcdirs; do abssrcdirs="$abssrcdirs $SRC/$d" done unset CH if [ "$o_FLAG" = "y" ]; then # root invoked old-style build -- make sure it works as it always has # by exporting 'CH'. The current Makefile.master doesn't use this, but # the old ones still do. PROTOCMPTERSE="protocmp.terse" CH= export CH else PROTOCMPTERSE="protocmp.terse -gu" fi POUND_SIGN="#" # we export POUND_SIGN to speed up the build process -- prevents evaluation of # the Makefile.master definitions. export o_FLAG X_FLAG POUND_SIGN maketype="distributed" MAKE=dmake # get the dmake version string alone DMAKE_VERSION=$( $MAKE -v ) DMAKE_VERSION=${DMAKE_VERSION#*: } # focus in on just the dotted version number alone DMAKE_MAJOR=$( echo $DMAKE_VERSION | \ sed -e 's/.*\<\([^.]*\.[^ ]*\).*$/\1/' ) # extract the second (or final) integer DMAKE_MINOR=${DMAKE_MAJOR#*.} DMAKE_MINOR=${DMAKE_MINOR%%.*} # extract the first integer DMAKE_MAJOR=${DMAKE_MAJOR%%.*} CHECK_DMAKE=${CHECK_DMAKE:-y} # x86 was built on the 12th, sparc on the 13th. if [ "$CHECK_DMAKE" = "y" -a \ "$DMAKE_VERSION" != "Sun Distributed Make 7.3 2003/03/12" -a \ "$DMAKE_VERSION" != "Sun Distributed Make 7.3 2003/03/13" -a \( \ "$DMAKE_MAJOR" -lt 7 -o \ "$DMAKE_MAJOR" -eq 7 -a "$DMAKE_MINOR" -lt 4 \) ]; then if [ -z "$DMAKE_VERSION" ]; then echo "$MAKE is missing." exit 1 fi echo `whence $MAKE`" version is:" echo " ${DMAKE_VERSION}" cat <>$LOGFILE 2>&1 if [ -s $build_noise_file ]; then echo "\n==== Nightly build noise ====\n" | tee -a $LOGFILE >>$mail_msg_file cat $build_noise_file >>$LOGFILE cat $build_noise_file >>$mail_msg_file echo | tee -a $LOGFILE >>$mail_msg_file fi rm -f $build_noise_file case "$build_ok" in y) state=Completed ;; i) state=Interrupted ;; *) state=Failed ;; esac NIGHTLY_STATUS=$state export NIGHTLY_STATUS run_hook POST_NIGHTLY $state run_hook SYS_POST_NIGHTLY $state cat $build_time_file $build_environ_file $mail_msg_file \ > ${LLOG}/mail_msg if [ "$m_FLAG" = "y" ]; then cat ${LLOG}/mail_msg | /usr/bin/mailx -s \ "Nightly ${MACH} Build of `basename ${CODEMGR_WS}` ${state}." \ ${MAILTO} fi if [ "$u_FLAG" = "y" -a "$build_ok" = "y" ]; then staffer cp ${LLOG}/mail_msg $PARENT_WS/usr/src/mail_msg-${MACH} staffer cp $LOGFILE $PARENT_WS/usr/src/nightly-${MACH}.log fi mv $LOGFILE $LLOG } # # Remove the locks and temporary files on any exit # cleanup() { logshuffle [ -z "$lockfile" ] || staffer rm -f $lockfile [ -z "$atloglockfile" ] || rm -f $atloglockfile [ -z "$ulockfile" ] || staffer rm -f $ulockfile [ -z "$Ulockfile" ] || rm -f $Ulockfile set -- $newdirlist while [ $# -gt 0 ]; do ISUSER=$1 staffer rmdir $2 shift; shift done rm -rf $TMPDIR } cleanup_signal() { build_ok=i # this will trigger cleanup(), above. exit 1 } trap cleanup 0 trap cleanup_signal 1 2 3 15 # # Generic lock file processing -- make sure that the lock file doesn't # exist. If it does, it should name the build host and PID. If it # doesn't, then make sure we can create it. Clean up locks that are # known to be stale (assumes host name is unique among build systems # for the workspace). # create_lock() { lockf=$1 lockvar=$2 ldir=`dirname $lockf` [ -d $ldir ] || newdir $ldir || exit 1 eval $lockvar=$lockf while ! staffer ln -s $hostname.$STAFFER.$$ $lockf 2> /dev/null; do basews=`basename $CODEMGR_WS` ls -l $lockf | nawk '{print $NF}' | IFS=. read host user pid if [ "$host" != "$hostname" ]; then echo "$MACH build of $basews apparently" \ "already started by $user on $host as $pid." exit 1 elif kill -s 0 $pid 2>/dev/null; then echo "$MACH build of $basews already started" \ "by $user as $pid." exit 1 else # stale lock; clear it out and try again rm -f $lockf fi done } # # Return the list of interesting proto areas, depending on the current # options. # allprotos() { roots="$ROOT" if [ $O_FLAG = y ]; then # OpenSolaris deliveries require separate proto areas. [ $D_FLAG = y ] && roots="$roots $ROOT-open" [ $F_FLAG = n ] && roots="$roots $ROOT-open-nd" fi if [[ $D_FLAG = y && $F_FLAG = n ]]; then [ $MULTI_PROTO = yes ] && roots="$roots $ROOT-nd" fi echo $roots } # Ensure no other instance of this script is running on this host. # LOCKNAME can be set in , and is by default, but is not # required due to the use of $ATLOG below. if [ -n "$LOCKNAME" ]; then create_lock /tmp/$LOCKNAME "lockfile" fi # # Create from one, two, or three other locks: # $ATLOG/nightly.lock # - protects against multiple builds in same workspace # $PARENT_WS/usr/src/nightly.$MACH.lock # - protects against multiple 'u' copy-backs # $NIGHTLY_PARENT_ROOT/nightly.lock # - protects against multiple 'U' copy-backs # # Overriding ISUSER to 1 causes the lock to be created as root if the # script is run as root. The default is to create it as $STAFFER. ISUSER=1 create_lock $ATLOG/nightly.lock "atloglockfile" if [ "$u_FLAG" = "y" ]; then create_lock $PARENT_WS/usr/src/nightly.$MACH.lock "ulockfile" fi if [ "$U_FLAG" = "y" ]; then # NIGHTLY_PARENT_ROOT is written as root if script invoked as root. ISUSER=1 create_lock $NIGHTLY_PARENT_ROOT/nightly.lock "Ulockfile" fi # Locks have been taken, so we're doing a build and we're committed to # the directories we may have created so far. newdirlist= # # Create mail_msg_file # mail_msg_file="${TMPDIR}/mail_msg" touch $mail_msg_file build_time_file="${TMPDIR}/build_time" build_environ_file="${TMPDIR}/build_environ" touch $build_environ_file # # Move old LOGFILE aside # ATLOG directory already made by 'create_lock' above # if [ -f $LOGFILE ]; then mv -f $LOGFILE ${LOGFILE}- fi # # Build OsNet source # START_DATE=`date` SECONDS=0 echo "\n==== Nightly $maketype build started: $START_DATE ====" \ | tee -a $LOGFILE > $build_time_file echo "\nBuild project: $build_project\nBuild taskid: $build_taskid" | \ tee -a $mail_msg_file >> $LOGFILE # make sure we log only to the nightly build file build_noise_file="${TMPDIR}/build_noise" exec $build_noise_file 2>&1 run_hook SYS_PRE_NIGHTLY run_hook PRE_NIGHTLY echo "\n==== list of environment variables ====\n" >> $LOGFILE env >> $LOGFILE echo "\n==== Nightly argument issues ====\n" | tee -a $mail_msg_file >> $LOGFILE if [ "$P_FLAG" = "y" ]; then obsolete_build GPROF | tee -a $mail_msg_file >> $LOGFILE fi if [ "$T_FLAG" = "y" ]; then obsolete_build TRACE | tee -a $mail_msg_file >> $LOGFILE fi if is_source_build; then if [ "$i_FLAG" = "y" -o "$i_CMD_LINE_FLAG" = "y" ]; then echo "WARNING: the -S flags do not support incremental" \ "builds; forcing clobber\n" | tee -a $mail_msg_file >> $LOGFILE i_FLAG=n i_CMD_LINE_FLAG=n fi if [ "$N_FLAG" = "n" ]; then echo "WARNING: the -S flags do not support protocmp;" \ "protocmp disabled\n" | \ tee -a $mail_msg_file >> $LOGFILE N_FLAG=y fi if [ "$l_FLAG" = "y" ]; then echo "WARNING: the -S flags do not support lint;" \ "lint disabled\n" | tee -a $mail_msg_file >> $LOGFILE l_FLAG=n fi if [ "$C_FLAG" = "y" ]; then echo "WARNING: the -S flags do not support cstyle;" \ "cstyle check disabled\n" | tee -a $mail_msg_file >> $LOGFILE C_FLAG=n fi else if [ "$N_FLAG" = "y" ]; then if [ "$p_FLAG" = "y" ]; then cat <> $LOGFILE WARNING: the p option (create packages) is set, but so is the N option (do not run protocmp); this is dangerous; you should unset the N option EOF else cat <> $LOGFILE Warning: the N option (do not run protocmp) is set; it probably shouldn't be EOF fi echo "" | tee -a $mail_msg_file >> $LOGFILE fi fi if [ "$O_FLAG" = "y" -a "$a_FLAG" = "n" ]; then echo "WARNING: OpenSolaris deliveries (-O) require archives;" \ "enabling the -a flag." | tee -a $mail_msg_file >> $LOGFILE a_FLAG=y fi if [ "$a_FLAG" = "y" -a "$D_FLAG" = "n" -a "$F_FLAG" = "y" ]; then echo "WARNING: Neither DEBUG nor non-DEBUG build requested, but the" \ "'a' option was set." | tee -a $mail_msg_file >> $LOGFILE fi if [ "$D_FLAG" = "n" -a "$l_FLAG" = "y" ]; then # # In the past we just complained but went ahead with the lint # pass, even though the proto area was built non-debug. It's # unlikely that non-debug headers will make a difference, but # rather than assuming it's a safe combination, force the user # to specify a debug build. # echo "WARNING: DEBUG build not requested; disabling lint.\n" \ | tee -a $mail_msg_file >> $LOGFILE l_FLAG=n fi if [ "$f_FLAG" = "y" ]; then if [ "$i_FLAG" = "y" ]; then echo "WARNING: the -f flag cannot be used during incremental" \ "builds; ignoring -f\n" | tee -a $mail_msg_file >> $LOGFILE f_FLAG=n fi if [ "$p_FLAG" != "y" -o "$l_FLAG" != "y" ]; then echo "WARNING: the -f flag requires -l and -p; ignoring -f\n" | \ tee -a $mail_msg_file >> $LOGFILE f_FLAG=n fi fi if [ "$w_FLAG" = "y" -a ! -d $ROOT ]; then echo "WARNING: -w specified, but $ROOT does not exist;" \ "ignoring -w\n" | tee -a $mail_msg_file >> $LOGFILE w_FLAG=n fi if [ "$t_FLAG" = "n" ]; then # # We're not doing a tools build, so make sure elfsign(1) is # new enough to safely sign non-crypto binaries. We test # debugging output from elfsign to detect the old version. # newelfsigntest=`SUNW_CRYPTO_DEBUG=stderr /usr/bin/elfsign verify \ -e /usr/lib/security/pkcs11_softtoken.so.1 2>&1 \ | egrep algorithmOID` if [ -z "$newelfsigntest" ]; then echo "WARNING: /usr/bin/elfsign out of date;" \ "will only sign crypto modules\n" | \ tee -a $mail_msg_file >> $LOGFILE export ELFSIGN_OBJECT=true elif [ "$VERIFY_ELFSIGN" = "y" ]; then echo "WARNING: VERIFY_ELFSIGN=y requires" \ "the -t flag; ignoring VERIFY_ELFSIGN\n" | \ tee -a $mail_msg_file >> $LOGFILE fi fi [ "$O_FLAG" = y ] && MULTI_PROTO=yes case $MULTI_PROTO in yes|no) ;; *) echo "WARNING: MULTI_PROTO is \"$MULTI_PROTO\"; " \ "should be \"yes\" or \"no\"." | tee -a $mail_msg_file >> $LOGFILE echo "Setting MULTI_PROTO to \"no\".\n" | \ tee -a $mail_msg_file >> $LOGFILE export MULTI_PROTO=no ;; esac echo "\n==== Build version ====\n" | tee -a $mail_msg_file >> $LOGFILE echo $VERSION | tee -a $mail_msg_file >> $LOGFILE # Save the current proto area if we're comparing against the last build if [ "$w_FLAG" = "y" -a -d "$ROOT" ]; then if [ -d "$ROOT.prev" ]; then rm -rf $ROOT.prev fi mv $ROOT $ROOT.prev fi # Same for non-DEBUG proto area if [ "$w_FLAG" = "y" -a "$MULTI_PROTO" = yes -a -d "$ROOT-nd" ]; then if [ -d "$ROOT-nd.prev" ]; then rm -rf $ROOT-nd.prev fi mv $ROOT-nd $ROOT-nd.prev fi # # Decide whether to clobber # if [ "$i_FLAG" = "n" -a -d "$SRC" ]; then echo "\n==== Make clobber at `date` ====\n" >> $LOGFILE cd $SRC # remove old clobber file rm -f $SRC/clobber.out rm -f $SRC/clobber-${MACH}.out # Remove all .make.state* files, just in case we are restarting # the build after having interrupted a previous 'make clobber'. find . \( -name SCCS -o -name .hg -o -name .svn \ -o -name 'interfaces.*' \) -prune \ -o -name '.make.*' -print | xargs rm -f $MAKE -ek clobber 2>&1 | tee -a $SRC/clobber-${MACH}.out >> $LOGFILE echo "\n==== Make clobber ERRORS ====\n" >> $mail_msg_file grep "$MAKE:" $SRC/clobber-${MACH}.out | egrep -v "Ignoring unknown host" \ >> $mail_msg_file if [[ "$t_FLAG" = "y" || "$O_FLAG" = "y" ]]; then echo "\n==== Make tools clobber at `date` ====\n" >> $LOGFILE cd ${TOOLS} rm -f ${TOOLS}/clobber-${MACH}.out $MAKE -ek clobber 2>&1 | \ tee -a ${TOOLS}/clobber-${MACH}.out >> $LOGFILE echo "\n==== Make tools clobber ERRORS ====\n" \ >> $mail_msg_file grep "$MAKE:" ${TOOLS}/clobber-${MACH}.out \ >> $mail_msg_file rm -rf ${TOOLS_PROTO} mkdir -p ${TOOLS_PROTO} fi rm -rf `allprotos` # Get back to a clean workspace as much as possible to catch # problems that only occur on fresh workspaces. # Remove all .make.state* files, libraries, and .o's that may # have been omitted from clobber. A couple of libraries are # under source code control, so leave them alone. # We should probably blow away temporary directories too. cd $SRC find $relsrcdirs \( -name SCCS -o -name .hg -o -name .svn \ -o -name 'interfaces.*' \) -prune -o \ \( -name '.make.*' -o -name 'lib*.a' -o -name 'lib*.so*' -o \ -name '*.o' \) -print | \ grep -v 'tools/ctf/dwarf/.*/libdwarf' | xargs rm -f else echo "\n==== No clobber at `date` ====\n" >> $LOGFILE fi type bringover_teamware > /dev/null 2>&1 || bringover_teamware() { # sleep on the parent workspace's lock while egrep -s write $BRINGOVER_WS/Codemgr_wsdata/locks do sleep 120 done if [[ -z $BRINGOVER ]]; then BRINGOVER=$TEAMWARE/bin/bringover fi staffer $BRINGOVER -c "nightly update" -p $BRINGOVER_WS \ -w $CODEMGR_WS $BRINGOVER_FILES < /dev/null 2>&1 || touch $TMPDIR/bringover_failed staffer bringovercheck $CODEMGR_WS >$TMPDIR/bringovercheck.out 2>&1 if [ -s $TMPDIR/bringovercheck.out ]; then echo "\n==== POST-BRINGOVER CLEANUP NOISE ====\n" cat $TMPDIR/bringovercheck.out fi } type bringover_mercurial > /dev/null 2>&1 || bringover_mercurial() { typeset -x PATH=$PATH # If the repository doesn't exist yet, then we want to populate it. # We only initially populate the closed repository if the open # repository doesn't also exist (e.g.: don't bringover closed if # only usr/src exists) if [[ ! -d $CODEMGR_WS/.hg ]]; then staffer hg init $CODEMGR_WS # If the user set CLOSED_BRINGOVER_WS, then we'll want to # initialise the closed repository if [[ -n $CLOSED_BRINGOVER_WS ]]; then staffer mkdir -p $CODEMGR_WS/usr staffer hg init $CODEMGR_WS/usr/closed CLOSED_IS_PRESENT="yes" fi fi typeset -x HGMERGE="/bin/false" # Do the bringover, along with an update. If a merge is necessary, # this will not fail properly: # http://www.selenic.com/mercurial/bts/issue186 # so we grep through the output for signs of a merge. # # Note that pulling even non-conflicting changes on top of local # changes (committed or not) requires a merge. staffer hg --cwd $CODEMGR_WS pull -u $BRINGOVER_WS 2>&1 | \ tee $TMPDIR/pull_open.out hgstatus=$? if [ $hgstatus -ne 0 ]; then touch $TMPDIR/bringover_failed return fi # We never want to clone usr/closed by default. If the usr/closed # repository exists already and CLOSED_BRINGOVER_WS is set, then # we update (pull -u). if [[ -d $SRC/../closed && -n $CLOSED_BRINGOVER_WS ]]; then staffer hg --cwd $SRC/../closed pull -u \ $CLOSED_BRINGOVER_WS 2>&1 | tee -a $TMPDIR/pull_closed.out hgstatus=$? if [ $hgstatus -ne 0 ]; then touch $TMPDIR/bringover_failed return fi fi # If there were uncommitted changes, the pull -u would have # attempted a merge. In the likely event that it failed, fail the # bringover. if grep "^merging.*failed" $TMPDIR/pull_open.out > /dev/null 2>&1 || \ grep "^merging.*failed" $TMPDIR/pull_closed.out >/dev/null 2>&1; then touch $TMPDIR/bringover_failed return fi # If heads were added, then a merge is required. We attempt the # merge, in case the branches touch different files (something that # teamware wouldn't balk at). If the merge fails, fail the # bringover, leaving the conflicts unresolved. # # Note that even if the merge succeeds, the result of the merge # will need to be committed, and not doing so will block any # further pulls. if grep "not updating" $TMPDIR/pull_open.out > /dev/null 2>&1; then hg --cwd $CODEMGR_WS merge 2>&1 | tee $TMPDIR/merge.out if grep "^merging.*failed" $TMPDIR/merge.out >/dev/null 2>&1; then touch $TMPDIR/bringover_failed fi fi if grep "not updating" $TMPDIR/pull_closed.out > /dev/null 2>&1; then hg --cwd $SRC/../closed merge 2>&1 | tee $TMPDIR/merge.out if grep "^merging.*failed" $TMPDIR/merge.out >/dev/null 2>&1; then touch $TMPDIR/bringover_failed fi fi } type bringover_subversion > /dev/null 2>&1 || bringover_subversion() { typeset -x PATH=$PATH if [[ ! -d $CODEMGR_WS/.svn ]]; then staffer svn checkout $BRINGOVER_WS $CODEMGR_WS || touch $TMPDIR/bringover_failed else typeset root root=$(staffer svn info $CODEMGR_WS | nawk '/^Repository Root:/ {print $NF}') if [[ $root != $BRINGOVER_WS ]]; then # We fail here because there's no way to update # from a named repo. cat <<-EOF \$BRINGOVER_WS doesn't match repository root: \$BRINGOVER_WS: $BRINGOVER_WS Repository root: $root EOF touch $TMPDIR/bringover_failed else # If a conflict happens, svn still exits 0. staffer svn update $CODEMGR_WS | tee $TMPDIR/pull.out || touch $TMPDIR/bringover_failed if grep "^C" $TMPDIR/pull.out > /dev/null 2>&1; then touch $TMPDIR/bringover_failed fi fi fi } type bringover_none > /dev/null 2>&1 || bringover_none() { echo "Couldn't figure out what kind of SCM to use for $BRINGOVER_WS." touch $TMPDIR/bringover_failed } # Parse the URL. # The other way to deal with empty components is to echo a string that can # be eval'ed by the caller to associate values (possibly empty) with # variables. In that case, passing in a printf string would let the caller # choose the variable names. parse_url() { typeset url method host port path url=$1 method=${url%%://*} host=${url#$method://} path=${host#*/} host=${host%%/*} if [[ $host == *:* ]]; then port=${host#*:} host=${host%:*} fi # method can never be empty. host can only be empty if method is # file, and that implies it's localhost. path can default to / if # it's otherwise empty, leaving port as the only component without # a default, so it has to go last. echo $method ${host:-localhost} ${path:-/} $port } http_get() { typeset url method host port path url=$1 if [[ -n $http_proxy ]]; then parse_url $http_proxy | read method host path port echo "GET $url HTTP/1.0\r\n" | mconnect -p ${port:-8080} $host else parse_url $url | read method host path port echo "GET $path HTTP/1.0\r\n" | mconnect -p ${port:-80} $host fi } # Echo a string name for the workspace SCM (teamware, hg, or svn). function wstype { typeset rtype ltype env CODEMGR_WS=$BRINGOVER_WS $WHICH_SCM 2>/dev/null | read rtype junk if [[ -z "$rtype" || "$rtype" == unknown ]]; then # Probe BRINGOVER_WS to determine its type if [[ $BRINGOVER_WS == svn*://* ]]; then rtype="subversion" elif [[ $BRINGOVER_WS == file://* ]] && egrep -s "This is a Subversion repository" \ ${BRINGOVER_WS#file://}/README.txt 2> /dev/null; then rtype="subversion" elif [[ $BRINGOVER_WS == ssh://* ]]; then rtype="mercurial" elif svn info $BRINGOVER_WS > /dev/null 2>&1; then rtype="subversion" elif [[ $BRINGOVER_WS == http://* ]] && \ http_get "$BRINGOVER_WS/?cmd=heads" | \ egrep -s "application/mercurial" 2> /dev/null; then rtype="mercurial" else rtype="none" fi fi # Make sure that the repository is one we support. If not, then # call it "none." case "$rtype" in none|subversion|teamware|mercurial) ;; *) rtype=none ;; esac # Probe CODEMGR_WS to determine its type if [[ -d $CODEMGR_WS ]]; then $WHICH_SCM | read ltype junk || exit 1 # This occurs when we have an empty build directory and nightly # itself creates the directory. It's not an error, as long as # we'll be doing a bringover. if [[ "$ltype" == unknown ]]; then ltype=$rtype fi fi # If BRINGOVER_WS and CODEMGR_WS don't match, doing a bringover # will be problematic! if [[ -n $ltype && $rtype != $ltype ]]; then echo "none" else echo $rtype fi } SCM_TYPE=$(wstype) export SCM_TYPE # # Decide whether to bringover to the codemgr workspace # if [ "$n_FLAG" = "n" ]; then run_hook PRE_BRINGOVER echo "\n==== bringover to $CODEMGR_WS at `date` ====\n" >> $LOGFILE echo "\n==== BRINGOVER LOG ====\n" >> $mail_msg_file eval "bringover_$SCM_TYPE" 2>&1 | tee -a $mail_msg_file >> $LOGFILE if [ -f $TMPDIR/bringover_failed ]; then rm -f $TMPDIR/bringover_failed build_ok=n echo "trouble with bringover, quitting at `date`." | tee -a $mail_msg_file >> $LOGFILE exit 1 fi run_hook POST_BRINGOVER # # Possible transition from pre-split workspace to split # workspace. See if the bringover changed anything. # CLOSED_IS_PRESENT="$orig_closed_is_present" check_closed_tree else echo "\n==== No bringover to $CODEMGR_WS ====\n" >> $LOGFILE fi echo "\n==== Build environment ====\n" | tee -a $build_environ_file >> $LOGFILE # System whence uname | tee -a $build_environ_file >> $LOGFILE uname -a 2>&1 | tee -a $build_environ_file >> $LOGFILE echo | tee -a $build_environ_file >> $LOGFILE # nightly echo "$0 $@" | tee -a $build_environ_file >> $LOGFILE if [[ $nightly_path = "/opt/onbld/bin/nightly" ]] && pkginfo SUNWonbld > /dev/null 2>&1 ; then pkginfo -l SUNWonbld | egrep "PKGINST:|VERSION:|PSTAMP:" else ls -l "$nightly_path" fi | tee -a $build_environ_file >> $LOGFILE echo | tee -a $build_environ_file >> $LOGFILE # make whence $MAKE | tee -a $build_environ_file >> $LOGFILE $MAKE -v | tee -a $build_environ_file >> $LOGFILE echo "number of concurrent jobs = $DMAKE_MAX_JOBS" | tee -a $build_environ_file >> $LOGFILE # # Report the compiler versions. # if [[ ! -f $SRC/Makefile ]]; then build_ok=n echo "\nUnable to find \"Makefile\" in $SRC." | \ tee -a $build_environ_file >> $LOGFILE exit 1 fi ( cd $SRC for target in cc-version cc64-version java-version; do echo # # Put statefile somewhere we know we can write to rather than trip # over a read-only $srcroot. # rm -f $TMPDIR/make-state export SRC if $MAKE -K $TMPDIR/make-state -e $target 2>/dev/null; then continue fi touch $TMPDIR/nocompiler done echo ) | tee -a $build_environ_file >> $LOGFILE if [ -f $TMPDIR/nocompiler ]; then rm -f $TMPDIR/nocompiler build_ok=n echo "Aborting due to missing compiler." | tee -a $build_environ_file >> $LOGFILE exit 1 fi # as whence as | tee -a $build_environ_file >> $LOGFILE as -V 2>&1 | head -1 | tee -a $build_environ_file >> $LOGFILE echo | tee -a $build_environ_file >> $LOGFILE # Check that we're running a capable link-editor whence ld | tee -a $build_environ_file >> $LOGFILE LDVER=`ld -V 2>&1` echo $LDVER | tee -a $build_environ_file >> $LOGFILE LDVER=`echo $LDVER | sed -e "s/.*-1\.//" -e "s/:.*//"` if [ `expr $LDVER \< 422` -eq 1 ]; then echo "The link-editor needs to be at version 422 or higher to build" | \ tee -a $build_environ_file >> $LOGFILE echo "the latest stuff. Hope your build works." | \ tee -a $build_environ_file >> $LOGFILE fi # # Build and use the workspace's tools if requested # if [[ "$t_FLAG" = "y" || "$O_FLAG" = y ]]; then set_non_debug_build_flags build_tools ${TOOLS_PROTO} if [[ $? -ne 0 && "$t_FLAG" = y ]]; then use_tools $TOOLS_PROTO export ONBLD_TOOLS=${ONBLD_TOOLS:=${TOOLS_PROTO}/opt/onbld} fi fi # # copy ihv proto area in addition to the build itself # if [ "$X_FLAG" = "y" ]; then copy_ihv_proto fi if [ "$i_FLAG" = "y" -a "$SH_FLAG" = "y" ]; then echo "\n==== NOT Building base OS-Net source ====\n" | \ tee -a $LOGFILE >> $mail_msg_file else # timestamp the start of the normal build; the findunref tool uses it. touch $SRC/.build.tstamp normal_build fi # # Generate the THIRDPARTYLICENSE files if needed. This is done before # findunref to help identify license files that need to be added to # the list. # if [ "$O_FLAG" = y -a "$build_ok" = y ]; then echo "\n==== Generating THIRDPARTYLICENSE files ====\n" | \ tee -a $mail_msg_file >> $LOGFILE mktpl usr/src/tools/opensolaris/license-list >>$LOGFILE 2>&1 if (( $? != 0 )) ; then echo "Couldn't create THIRDPARTYLICENSE files" | tee -a $mail_msg_file >> $LOGFILE fi fi # # If OpenSolaris deliverables were requested, do the open-only build # now, so that it happens at roughly the same point as the source # product builds. This lets us take advantage of checks that come # later (e.g., the core file check). # if [ "$O_FLAG" = y -a "$build_ok" = y ]; then # # Generate skeleton (minimal) closed binaries for open-only # build. There's no need to distinguish debug from non-debug # binaries, but it simplifies file management to have separate # trees. # echo "\n==== Generating skeleton closed binaries for" \ "open-only build ====\n" | \ tee -a $LOGFILE >> $mail_msg_file rm -rf $CODEMGR_WS/closed.skel if [ "$D_FLAG" = y ]; then mkclosed $MACH $ROOT $CODEMGR_WS/closed.skel/root_$MACH \ >>$LOGFILE 2>&1 if (( $? != 0 )) ; then echo "Couldn't create skeleton DEBUG closed binaries." | tee -a $mail_msg_file >> $LOGFILE fi fi if [ "$F_FLAG" = n ]; then mkclosed $MACH $ROOT-nd $CODEMGR_WS/closed.skel/root_$MACH-nd \ >>$LOGFILE 2>&1 if (( $? != 0 )) ; then echo "Couldn't create skeleton non-DEBUG closed binaries." | tee -a $mail_msg_file >> $LOGFILE fi fi ORIG_CLOSED_IS_PRESENT=$CLOSED_IS_PRESENT export CLOSED_IS_PRESENT=no ORIG_ON_CLOSED_BINS="$ON_CLOSED_BINS" export ON_CLOSED_BINS=$CODEMGR_WS/closed.skel normal_build -O ON_CLOSED_BINS=$ORIG_ON_CLOSED_BINS CLOSED_IS_PRESENT=$ORIG_CLOSED_IS_PRESENT fi ORIG_SRC=$SRC BINARCHIVE=${CODEMGR_WS}/bin-${MACH}.cpio.Z if [ "$SE_FLAG" = "y" -o "$SD_FLAG" = "y" -o "$SH_FLAG" = "y" ]; then save_binaries fi # EXPORT_SRC comes after CRYPT_SRC since a domestic build will need # $SRC pointing to the export_source usr/src. if [ "$SE_FLAG" = "y" -o "$SD_FLAG" = "y" -o "$SH_FLAG" = "y" ]; then if [ "$SD_FLAG" = "y" -a $build_ok = y ]; then set_up_source_build ${CODEMGR_WS} ${CRYPT_SRC} CRYPT_SRC fi if [ $build_ok = y ]; then set_up_source_build ${CODEMGR_WS} ${EXPORT_SRC} EXPORT_SRC fi fi if [ "$SD_FLAG" = "y" -a $build_ok = y ]; then # drop the crypt files in place. cd ${EXPORT_SRC} echo "\nextracting crypt_files.cpio.Z onto export_source.\n" \ >> ${LOGFILE} zcat ${CODEMGR_WS}/crypt_files.cpio.Z | \ cpio -idmucvB 2>/dev/null >> ${LOGFILE} if [ "$?" = "0" ]; then echo "\n==== DOMESTIC extraction succeeded ====\n" \ >> $mail_msg_file else echo "\n==== DOMESTIC extraction failed ====\n" \ >> $mail_msg_file fi fi if [ "$SO_FLAG" = "y" -a $build_ok = y ]; then # # Copy the open sources into their own tree, set up the closed # binaries, and set up the environment. The build looks for # the closed binaries in a location that depends on whether # it's a DEBUG build, so we might need to make two copies. # # If copy_source fails, it will have already generated an # error message and set build_ok=n, so we don't need to worry # about that here. # copy_source $CODEMGR_WS $OPEN_SRCDIR OPEN_SOURCE usr/src fi if [ "$SO_FLAG" = "y" -a $build_ok = y ]; then SRC=$OPEN_SRCDIR/usr/src # Try not to clobber any user-provided closed binaries. export ON_CLOSED_BINS=$CODEMGR_WS/closed$$ echo "\n==== Copying skeleton closed binaries to" \ "$ON_CLOSED_BINS ====\n" | \ tee -a $mail_msg_file >> $LOGFILE if [ "$D_FLAG" = y ]; then mkclosed $MACH $ROOT $ON_CLOSED_BINS/root_$MACH >>$LOGFILE 2>&1 if (( $? != 0 )) ; then build_ok=n echo "Couldn't create DEBUG closed binaries." | tee -a $mail_msg_file >> $LOGFILE fi fi if [ "$F_FLAG" = n ]; then root=$ROOT [ "$MULTI_PROTO" = yes ] && root=$ROOT-nd mkclosed $MACH $root $ON_CLOSED_BINS/root_$MACH-nd \ >>$LOGFILE 2>&1 if (( $? != 0 )) ; then build_ok=n echo "Couldn't create non-DEBUG closed binaries." | tee -a $mail_msg_file >> $LOGFILE fi fi export CLOSED_IS_PRESENT=no fi if is_source_build && [ $build_ok = y ] ; then # remove proto area(s) here, since we don't clobber rm -rf `allprotos` if [ "$t_FLAG" = "y" ]; then set_non_debug_build_flags ORIG_TOOLS=$TOOLS # # SRC was set earlier to point to the source build # source tree (e.g., $EXPORT_SRC). # TOOLS=${SRC}/tools build_tools ${SRC}/tools/proto TOOLS=$ORIG_TOOLS fi export EXPORT_RELEASE_BUILD ; EXPORT_RELEASE_BUILD=# normal_build fi if [[ "$SO_FLAG" = "y" && "$build_ok" = "y" ]]; then rm -rf $ON_CLOSED_BINS fi # # There are several checks that need to look at the proto area, but # they only need to look at one, and they don't care whether it's # DEBUG or non-DEBUG. # if [[ "$MULTI_PROTO" = yes && "$D_FLAG" = n ]]; then checkroot=$ROOT-nd else checkroot=$ROOT fi if [ "$build_ok" = "y" ]; then echo "\n==== Creating protolist system file at `date` ====" \ >> $LOGFILE protolist $checkroot > $ATLOG/proto_list_${MACH} echo "==== protolist system file created at `date` ====\n" \ >> $LOGFILE if [ "$N_FLAG" != "y" ]; then echo "\n==== Impact on packages ====\n" >> $mail_msg_file # If there is a reference proto list, compare the build's proto # list with the reference to see changes in proto areas. # Use the current exception list. exc=etc/exception_list_$MACH if [ -f $SRC/pkgdefs/$exc ]; then ELIST="-e $SRC/pkgdefs/$exc" fi if [ "$X_FLAG" = "y" -a -f $IA32_IHV_WS/usr/src/pkgdefs/$exc ]; then ELIST="$ELIST -e $IA32_IHV_WS/usr/src/pkgdefs/$exc" fi if [ -f "$REF_PROTO_LIST" ]; then # For builds that copy the IHV proto area (-X), add the # IHV proto list to the reference list if the reference # was built without -X. # # For builds that don't copy the IHV proto area, add the # IHV proto list to the build's proto list if the # reference was built with -X. # # Use the presence of the first file entry of the cached # IHV proto list in the reference list to determine # whether it was build with -X or not. IHV_REF_PROTO_LIST=$SRC/pkgdefs/etc/proto_list_ihv_$MACH grepfor=$(nawk '$1 == "f" { print $2; exit }' \ $IHV_REF_PROTO_LIST 2> /dev/null) if [ $? = 0 -a -n "$grepfor" ]; then if [ "$X_FLAG" = "y" ]; then grep -w "$grepfor" \ $REF_PROTO_LIST > /dev/null if [ ! "$?" = "0" ]; then REF_IHV_PROTO="-d $IHV_REF_PROTO_LIST" fi else grep -w "$grepfor" \ $REF_PROTO_LIST > /dev/null if [ "$?" = "0" ]; then IHV_PROTO_LIST="$IHV_REF_PROTO_LIST" fi fi fi $PROTOCMPTERSE \ "Files in yesterday's proto area, but not today's:" \ "Files in today's proto area, but not yesterday's:" \ "Files that changed between yesterday and today:" \ ${ELIST} \ -d $REF_PROTO_LIST \ $REF_IHV_PROTO \ $ATLOG/proto_list_${MACH} \ $IHV_PROTO_LIST \ >> $mail_msg_file fi # Compare the build's proto list with current package # definitions to audit the quality of package definitions # and makefile install targets. Use the current exception list. PKGDEFS_LIST="" for d in $abssrcdirs; do if [ -d $d/pkgdefs ]; then PKGDEFS_LIST="$PKGDEFS_LIST -d $d/pkgdefs" fi done if [ "$X_FLAG" = "y" -a -d $IA32_IHV_WS/usr/src/pkgdefs ]; then PKGDEFS_LIST="$PKGDEFS_LIST -d $IA32_IHV_WS/usr/src/pkgdefs" fi $PROTOCMPTERSE \ "Files missing from the proto area:" \ "Files missing from packages:" \ "Inconsistencies between pkgdefs and proto area:" \ ${ELIST} \ ${PKGDEFS_LIST} \ $ATLOG/proto_list_${MACH} \ >> $mail_msg_file fi fi if [ "$u_FLAG" = "y" -a "$build_ok" = "y" ]; then staffer cp $ATLOG/proto_list_${MACH} \ $PARENT_WS/usr/src/proto_list_${MACH} fi # Update parent proto area if necessary. This is done now # so that the proto area has either DEBUG or non-DEBUG kernels. # Note that this clears out the lock file, so we can dispense with # the variable now. if [ "$U_FLAG" = "y" -a "$build_ok" = "y" ]; then echo "\n==== Copying proto area to $NIGHTLY_PARENT_ROOT ====\n" | \ tee -a $LOGFILE >> $mail_msg_file # The rm -rf command below produces predictable errors if # nightly is invoked from the parent's $ROOT/opt/onbld/bin, # and that directory is accessed via NFS. This is because # deleted-but-still-open files don't actually disappear as # expected, but rather turn into .nfsXXXX junk files, leaving # the directory non-empty. Since this is a not-unusual usage # pattern, and we still want to catch other errors here, we # take the unusal step of moving aside 'nightly' from that # directory (if we're using it). mypath=${nightly_path##*/root_$MACH/} if [ "$mypath" = $nightly_path ]; then mypath=opt/onbld/bin/${nightly_path##*/} fi if [ $nightly_path -ef $PARENT_WS/proto/root_$MACH/$mypath ]; then mv -f $nightly_path $PARENT_WS/proto/root_$MACH fi rm -rf $PARENT_WS/proto/root_$MACH/* unset Ulockfile mkdir -p $NIGHTLY_PARENT_ROOT if [[ "$MULTI_PROTO" = no || "$D_FLAG" = y ]]; then cd $ROOT ( tar cf - . | ( cd $NIGHTLY_PARENT_ROOT; umask 0; tar xpf - ) ) 2>&1 | tee -a $mail_msg_file >> $LOGFILE fi if [[ "$MULTI_PROTO" = yes && "$F_FLAG" = n ]]; then mkdir -p $NIGHTLY_PARENT_ROOT-nd cd $ROOT-nd ( tar cf - . | ( cd $NIGHTLY_PARENT_ROOT-nd; umask 0; tar xpf - ) ) 2>&1 | tee -a $mail_msg_file >> $LOGFILE fi fi # # do shared library interface verification # if [ "$A_FLAG" = "y" -a "$build_ok" = "y" ]; then echo "\n==== Check versioning and ABI information ====\n" | \ tee -a $LOGFILE >> $mail_msg_file rm -rf $SRC/interfaces.ref if [ -d $SRC/interfaces.out ]; then mv $SRC/interfaces.out $SRC/interfaces.ref fi rm -rf $SRC/interfaces.out mkdir -p $SRC/interfaces.out intf_check -V -m -o -b $SRC/tools/abi/etc \ -d $SRC/interfaces.out $checkroot 2>&1 | sort \ > $SRC/interfaces.out/log # report any ERROR found in log file fgrep 'ERROR' $SRC/interfaces.out/log | sed 's/^ERROR: //' | \ tee -a $LOGFILE >> $mail_msg_file if [ ! -d $SRC/interfaces.ref ] ; then mkdir -p $SRC/interfaces.ref if [ -d $SRC/interfaces.out ]; then cp -r $SRC/interfaces.out/* $SRC/interfaces.ref fi fi echo "\n==== Diff versioning warnings (since last build) ====\n" | \ tee -a $LOGFILE >> $mail_msg_file out_vers=`grep ^VERSION $SRC/interfaces.out/log`; ref_vers=`grep ^VERSION $SRC/interfaces.ref/log`; # Report any differences in WARNING messages between last # and current build. if [ "$out_vers" = "$ref_vers" ]; then diff $SRC/interfaces.ref/log $SRC/interfaces.out/log | \ fgrep 'WARNING' | sed 's/WARNING: //' | \ tee -a $LOGFILE >> $mail_msg_file fi fi if [ "$r_FLAG" = "y" -a "$build_ok" = "y" ]; then echo "\n==== Check ELF runtime attributes ====\n" | \ tee -a $LOGFILE >> $mail_msg_file LDDUSAGE="^ldd: does not support -e" LDDWRONG="wrong class" CRLERROR="^crle:" CRLECONF="^crle: configuration file:" RUNTIMEREF=$SRC/runtime-${MACH}.ref RUNTIMEOUT=$SRC/runtime-${MACH}.out rm -f $RUNTIMEREF if [ -f $RUNTIMEOUT ]; then egrep -v "$LDDUSAGE|$LDDWRONG|$CRLERROR|$CRLECONF" \ $RUNTIMEOUT > $RUNTIMEREF fi # If we're doing a debug build the proto area will be left with # debuggable objects, thus don't assert -s. if [ "$D_FLAG" = "y" ]; then rtime_sflag="" else rtime_sflag="-s" fi check_rtime -d $checkroot -i -m -o $rtime_sflag $checkroot 2>&1 | \ egrep -v ": unreferenced object=$checkroot/.*/lib(w|intl|thread|pthread).so" | \ egrep -v ": unused object=$checkroot/.*/lib(w|intl|thread|pthread).so" | \ sort > $RUNTIMEOUT # Determine any processing errors that will affect the final output # and display these first. grep -l "$LDDUSAGE" $RUNTIMEOUT > /dev/null if (( $? == 0 )) ; then echo "WARNING: ldd(1) does not support -e. The version of ldd(1)" | \ tee -a $LOGFILE >> $mail_msg_file echo "on your system is old - 4390308 (s81_30) is required.\n" | \ tee -a $LOGFILE >> $mail_msg_file fi grep -l "$LDDWRONG" $RUNTIMEOUT > /dev/null if (( $? == 0 )) ; then echo "WARNING: wrong class message detected. ldd(1) was unable" | \ tee -a $LOGFILE >> $mail_msg_file echo "to execute an object, thus it could not be checked fully." | \ tee -a $LOGFILE >> $mail_msg_file echo "Perhaps a 64-bit object was encountered on a 32-bit system," | \ tee -a $LOGFILE >> $mail_msg_file echo "or an i386 object was encountered on a sparc system?\n" | \ tee -a $LOGFILE >> $mail_msg_file fi grep -l "$CRLECONF" $RUNTIMEOUT > /dev/null if (( $? == 0 )) ; then echo "WARNING: creation of an alternative dependency cache failed." | \ tee -a $LOGFILE >> $mail_msg_file echo "Dependencies will bind to the base system libraries.\n" | \ tee -a $LOGFILE >> $mail_msg_file grep "$CRLECONF" $RUNTIMEOUT | \ tee -a $LOGFILE >> $mail_msg_file grep "$CRLERROR" $RUNTIMEOUT | grep -v "$CRLECONF" | \ tee -a $LOGFILE >> $mail_msg_file echo "\n" | tee -a $LOGFILE >> $mail_msg_file fi egrep '' $RUNTIMEOUT | \ tee -a $LOGFILE >> $mail_msg_file # NEEDED= and RPATH= are informational; report anything else that we # haven't already. egrep -v "NEEDED=|RPATH=|$LDDUSAGE|$LDDWRONG|$CRLERROR|$CRLECONF" \ $RUNTIMEOUT | tee -a $LOGFILE >> $mail_msg_file # probably should compare against a 'known ok runpaths' list if [ ! -f $RUNTIMEREF ]; then egrep -v "$LDDUSAGE|$LDDWRONG|$CRLERROR|$CRLECONF" \ $RUNTIMEOUT > $RUNTIMEREF fi echo "\n==== Diff ELF runtime attributes (since last build) ====\n" \ >> $mail_msg_file egrep -v "$LDDUSAGE|$LDDWRONG|$CRLERROR|$CRLECONF" $RUNTIMEOUT | \ diff $RUNTIMEREF - >> $mail_msg_file fi # DEBUG lint of kernel begins if [ "$i_CMD_LINE_FLAG" = "n" -a "$l_FLAG" = "y" ]; then if [ "$LINTDIRS" = "" ]; then # LINTDIRS="$SRC/uts y $SRC/stand y $SRC/psm y" LINTDIRS="$SRC y" fi set $LINTDIRS while [ $# -gt 0 ]; do dolint $1 $2; shift; shift done else echo "\n==== No '$MAKE lint' ====\n" >> $LOGFILE fi # "make check" begins if [ "$i_CMD_LINE_FLAG" = "n" -a "$C_FLAG" = "y" ]; then # remove old check.out rm -f $SRC/check.out rm -f $SRC/check-${MACH}.out cd $SRC $MAKE -ek check 2>&1 | tee -a $SRC/check-${MACH}.out >> $LOGFILE echo "\n==== cstyle/hdrchk errors ====\n" >> $mail_msg_file grep ":" $SRC/check-${MACH}.out | egrep -v "Ignoring unknown host" | \ sort | uniq >> $mail_msg_file else echo "\n==== No '$MAKE check' ====\n" >> $LOGFILE fi echo "\n==== Find core files ====\n" | \ tee -a $LOGFILE >> $mail_msg_file find $abssrcdirs -name core -a -type f -exec file {} \; | \ tee -a $LOGFILE >> $mail_msg_file if [ "$f_FLAG" = "y" -a "$build_ok" = "y" ]; then echo "\n==== Diff unreferenced files (since last build) ====\n" \ | tee -a $LOGFILE >>$mail_msg_file rm -f $SRC/unref-${MACH}.ref if [ -f $SRC/unref-${MACH}.out ]; then mv $SRC/unref-${MACH}.out $SRC/unref-${MACH}.ref fi # # For any SCM other than teamware, we want to disable the # managed-by-SCCS test in findunref # findunref_all="" if [ "$SCM_TYPE" != teamware ]; then findunref_all="-a" fi findunref $findunref_all -t $SRC/.build.tstamp $SRC/.. \ ${TOOLS}/findunref/exception_list 2>> $mail_msg_file | \ sort | sed -e s=^./src/=./= -e s=^./closed/=../closed/= \ > $SRC/unref-${MACH}.out if [ ! -f $SRC/unref-${MACH}.ref ]; then cp $SRC/unref-${MACH}.out $SRC/unref-${MACH}.ref fi diff $SRC/unref-${MACH}.ref $SRC/unref-${MACH}.out >>$mail_msg_file fi # # Generate the OpenSolaris deliverables if requested. Some of these # steps need to come after findunref and are commented below. # if [ "$O_FLAG" = y -a "$build_ok" = y ]; then echo "\n==== Generating OpenSolaris tarballs ====\n" | \ tee -a $mail_msg_file >> $LOGFILE cd $CODEMGR_WS # # This step grovels through the pkgdefs proto* files, so it # must come after findunref. # echo "Generating closed binaries tarball(s)..." >> $LOGFILE closed_basename=on-closed-bins if [ "$D_FLAG" = y ]; then bindrop "$ROOT" "$ROOT-open" "$closed_basename" \ >>"$LOGFILE" 2>&1 if (( $? != 0 )) ; then echo "Couldn't create DEBUG closed binaries." | tee -a $mail_msg_file >> $LOGFILE fi fi if [ "$F_FLAG" = n ]; then bindrop -n "$ROOT-nd" "$ROOT-open-nd" "$closed_basename-nd" \ >>"$LOGFILE" 2>&1 if (( $? != 0 )) ; then echo "Couldn't create non-DEBUG closed binaries." | tee -a $mail_msg_file >> $LOGFILE fi fi echo "Generating SUNWonbld tarball..." >> $LOGFILE PKGARCHIVE=$PKGARCHIVE_ORIG onblddrop >> $LOGFILE 2>&1 if (( $? != 0 )) ; then echo "Couldn't create SUNWonbld tarball." | tee -a $mail_msg_file >> $LOGFILE fi echo "Generating README.opensolaris..." >> $LOGFILE cat $SRC/tools/opensolaris/README.opensolaris.tmpl | \ mkreadme_osol $CODEMGR_WS/README.opensolaris >> $LOGFILE 2>&1 if (( $? != 0 )) ; then echo "Couldn't create README.opensolaris." | tee -a $mail_msg_file >> $LOGFILE fi # This step walks the source tree, so it must come after # findunref. It depends on README.opensolaris. echo "Generating source tarball..." >> $LOGFILE sdrop >>$LOGFILE 2>&1 if (( $? != 0 )) ; then echo "Couldn't create source tarball." | tee -a "$mail_msg_file" >> "$LOGFILE" fi # This step depends on the closed binaries tarballs. echo "Generating BFU tarball(s)..." >> $LOGFILE if [ "$D_FLAG" = y ]; then makebfu_filt bfudrop "$ROOT-open" \ "$closed_basename.$MACH.tar.bz2" nightly-osol if (( $? != 0 )) ; then echo "Couldn't create DEBUG archives tarball." | tee -a $mail_msg_file >> $LOGFILE fi fi if [ "$F_FLAG" = n ]; then makebfu_filt bfudrop -n "$ROOT-open-nd" \ "$closed_basename-nd.$MACH.tar.bz2" nightly-osol-nd if (( $? != 0 )) ; then echo "Couldn't create non-DEBUG archives tarball." | tee -a $mail_msg_file >> $LOGFILE fi fi fi # Verify that the usual lists of files, such as exception lists, # contain only valid references to files. If the build has failed, # then don't check the proto area. CHECK_PATHS=${CHECK_PATHS:-y} if [ "$CHECK_PATHS" = y -a "$N_FLAG" != y ]; then echo "\n==== Check lists of files ====\n" | tee -a $LOGFILE \ >>$mail_msg_file arg=-b [ "$build_ok" = y ] && arg= checkpaths $arg $checkroot 2>&1 | tee -a $LOGFILE >>$mail_msg_file fi if [ "$M_FLAG" != "y" -a "$build_ok" = y ]; then echo "\n==== Impact on file permissions ====\n" \ >> $mail_msg_file # # Get pkginfo files from usr/src/pkgdefs # pmodes -qvdP \ `for d in $abssrcdirs; do if [ -d "$d/pkgdefs" ] then find $d/pkgdefs -name pkginfo.tmpl -print -o -name .del\* -prune fi done | sed -e 's:/pkginfo.tmpl$::' | sort -u ` >> $mail_msg_file fi if [ "$w_FLAG" = "y" -a "$build_ok" = "y" ]; then if [[ "$MULTI_PROTO" = no || "$D_FLAG" = y ]]; then do_wsdiff DEBUG $ROOT.prev $ROOT fi if [[ "$MULTI_PROTO" = yes && "$F_FLAG" = n ]]; then do_wsdiff non-DEBUG $ROOT-nd.prev $ROOT-nd fi fi END_DATE=`date` echo "==== Nightly $maketype build completed: $END_DATE ====" | \ tee -a $LOGFILE >> $build_time_file typeset -Z2 minutes typeset -Z2 seconds elapsed_time=$SECONDS ((hours = elapsed_time / 3600 )) ((minutes = elapsed_time / 60 % 60)) ((seconds = elapsed_time % 60)) echo "\n==== Total build time ====" | \ tee -a $LOGFILE >> $build_time_file echo "\nreal ${hours}:${minutes}:${seconds}" | \ tee -a $LOGFILE >> $build_time_file if [ "$u_FLAG" = "y" -a "$f_FLAG" = "y" -a "$build_ok" = "y" ]; then staffer cp ${SRC}/unref-${MACH}.out $PARENT_WS/usr/src/ # # Produce a master list of unreferenced files -- ideally, we'd # generate the master just once after all of the nightlies # have finished, but there's no simple way to know when that # will be. Instead, we assume that we're the last nightly to # finish and merge all of the unref-${MACH}.out files in # $PARENT_WS/usr/src/. If we are in fact the final ${MACH} to # finish, then this file will be the authoritative master # list. Otherwise, another ${MACH}'s nightly will eventually # overwrite ours with its own master, but in the meantime our # temporary "master" will be no worse than any older master # which was already on the parent. # set -- $PARENT_WS/usr/src/unref-*.out cp "$1" ${TMPDIR}/unref.merge shift for unreffile; do comm -12 ${TMPDIR}/unref.merge "$unreffile" > ${TMPDIR}/unref.$$ mv ${TMPDIR}/unref.$$ ${TMPDIR}/unref.merge done staffer cp ${TMPDIR}/unref.merge $PARENT_WS/usr/src/unrefmaster.out fi # # All done save for the sweeping up. # (whichever exit we hit here will trigger the "cleanup" trap which # optionally sends mail on completion). # if [ "$build_ok" = "y" ]; then exit 0 fi exit 1