1 #!/bin/ksh -p 2 # 3 # CDDL HEADER START 4 # 5 # The contents of this file are subject to the terms of the 6 # Common Development and Distribution License (the "License"). 7 # You may not use this file except in compliance with the License. 8 # 9 # You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10 # or http://www.opensolaris.org/os/licensing. 11 # See the License for the specific language governing permissions 12 # and limitations under the License. 13 # 14 # When distributing Covered Code, include this CDDL HEADER in each 15 # file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16 # If applicable, add the following below this CDDL HEADER, with the 17 # fields enclosed by brackets "[]" replaced with your own identifying 18 # information: Portions Copyright [yyyy] [name of copyright owner] 19 # 20 # CDDL HEADER END 21 # 22 23 # 24 # Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. 25 # Copyright 2008, 2010, Richard Lowe 26 # Copyright 2011 Nexenta Systems, Inc. All rights reserved. 27 # Copyright 2012 Joshua M. Clulow <josh@sysmgr.org> 28 # 29 # Based on the nightly script from the integration folks, 30 # Mostly modified and owned by mike_s. 31 # Changes also by kjc, dmk. 32 # 33 # BRINGOVER_WS may be specified in the env file. 34 # The default is the old behavior of CLONE_WS 35 # 36 # -i on the command line, means fast options, so when it's on the 37 # command line (only), lint and check builds are skipped no matter what 38 # the setting of their individual flags are in NIGHTLY_OPTIONS. 39 # 40 # LINTDIRS can be set in the env file, format is a list of: 41 # 42 # /dirname-to-run-lint-on flag 43 # 44 # Where flag is: y - enable lint noise diff output 45 # n - disable lint noise diff output 46 # 47 # For example: LINTDIRS="$SRC/uts n $SRC/stand y $SRC/psm y" 48 # 49 # OPTHOME may be set in the environment to override /opt 50 # 51 52 # 53 # The CDPATH variable causes ksh's `cd' builtin to emit messages to stdout 54 # under certain circumstances, which can really screw things up; unset it. 55 # 56 unset CDPATH 57 58 # Get the absolute path of the nightly script that the user invoked. This 59 # may be a relative path, and we need to do this before changing directory. 60 nightly_path=`whence $0` 61 62 # 63 # Keep track of where we found nightly so we can invoke the matching 64 # which_scm script. If that doesn't work, don't go guessing, just rely 65 # on the $PATH settings, which will generally give us either /opt/onbld 66 # or the user's workspace. 67 # 68 WHICH_SCM=$(dirname $nightly_path)/which_scm 69 if [[ ! -x $WHICH_SCM ]]; then 70 WHICH_SCM=which_scm 71 fi 72 73 function fatal_error 74 { 75 print -u2 "nightly: $*" 76 exit 1 77 } 78 79 # 80 # Function to do a DEBUG and non-DEBUG build. Needed because we might 81 # need to do another for the source build, and since we only deliver DEBUG or 82 # non-DEBUG packages. 83 # 84 # usage: normal_build 85 # 86 function normal_build { 87 88 typeset orig_p_FLAG="$p_FLAG" 89 typeset crypto_signer="$CODESIGN_USER" 90 91 suffix="" 92 93 # non-DEBUG build begins 94 95 if [ "$F_FLAG" = "n" ]; then 96 set_non_debug_build_flags 97 CODESIGN_USER="$crypto_signer" \ 98 build "non-DEBUG" "$suffix-nd" "-nd" "$MULTI_PROTO" 99 else 100 echo "\n==== No non-DEBUG $open_only build ====\n" >> "$LOGFILE" 101 fi 102 103 # non-DEBUG build ends 104 105 # DEBUG build begins 106 107 if [ "$D_FLAG" = "y" ]; then 108 set_debug_build_flags 109 CODESIGN_USER="$crypto_signer" \ 110 build "DEBUG" "$suffix" "" "$MULTI_PROTO" 111 else 112 echo "\n==== No DEBUG $open_only build ====\n" >> "$LOGFILE" 113 fi 114 115 # DEBUG build ends 116 117 p_FLAG="$orig_p_FLAG" 118 } 119 120 # 121 # usage: run_hook HOOKNAME ARGS... 122 # 123 # If variable "$HOOKNAME" is defined, insert a section header into 124 # our logs and then run the command with ARGS 125 # 126 function run_hook { 127 HOOKNAME=$1 128 eval HOOKCMD=\$$HOOKNAME 129 shift 130 131 if [ -n "$HOOKCMD" ]; then 132 ( 133 echo "\n==== Running $HOOKNAME command: $HOOKCMD ====\n" 134 ( $HOOKCMD "$@" 2>&1 ) 135 if [ "$?" -ne 0 ]; then 136 # Let exit status propagate up 137 touch $TMPDIR/abort 138 fi 139 ) | tee -a $mail_msg_file >> $LOGFILE 140 141 if [ -f $TMPDIR/abort ]; then 142 build_ok=n 143 echo "\nAborting at request of $HOOKNAME" | 144 tee -a $mail_msg_file >> $LOGFILE 145 exit 1 146 fi 147 fi 148 } 149 150 # Return library search directive as function of given root. 151 function myldlibs { 152 echo "-L$1/lib -L$1/usr/lib" 153 } 154 155 # Return header search directive as function of given root. 156 function myheaders { 157 echo "-I$1/usr/include" 158 } 159 160 # 161 # Function to do the build, including package generation. 162 # usage: build LABEL SUFFIX ND MULTIPROTO 163 # - LABEL is used to tag build output. 164 # - SUFFIX is used to distinguish files (e.g., DEBUG vs non-DEBUG, 165 # open-only vs full tree). 166 # - ND is "-nd" (non-DEBUG builds) or "" (DEBUG builds). 167 # - If MULTIPROTO is "yes", it means to name the proto area according to 168 # SUFFIX. Otherwise ("no"), (re)use the standard proto area. 169 # 170 function build { 171 LABEL=$1 172 SUFFIX=$2 173 ND=$3 174 MULTIPROTO=$4 175 INSTALLOG=install${SUFFIX}-${MACH} 176 NOISE=noise${SUFFIX}-${MACH} 177 PKGARCHIVE=${PKGARCHIVE_ORIG}${SUFFIX} 178 179 ORIGROOT=$ROOT 180 [ $MULTIPROTO = no ] || export ROOT=$ROOT$SUFFIX 181 182 export ENVLDLIBS1=`myldlibs $ROOT` 183 export ENVCPPFLAGS1=`myheaders $ROOT` 184 185 this_build_ok=y 186 # 187 # Build OS-Networking source 188 # 189 echo "\n==== Building OS-Net source at `date` ($LABEL) ====\n" \ 190 >> $LOGFILE 191 192 rm -f $SRC/${INSTALLOG}.out 193 cd $SRC 194 /bin/time $MAKE -e install 2>&1 | \ 195 tee -a $SRC/${INSTALLOG}.out >> $LOGFILE 196 197 echo "\n==== Build errors ($LABEL) ====\n" >> $mail_msg_file 198 egrep ":" $SRC/${INSTALLOG}.out | 199 egrep -e "(^${MAKE}:|[ ]error[: \n])" | \ 200 egrep -v "Ignoring unknown host" | \ 201 egrep -v "cc .* -o error " | \ 202 egrep -v "warning" | tee $TMPDIR/build_errs${SUFFIX} \ 203 >> $mail_msg_file 204 if [[ -s $TMPDIR/build_errs${SUFFIX} ]]; then 205 build_ok=n 206 this_build_ok=n 207 fi 208 grep "bootblock image is .* bytes too big" $SRC/${INSTALLOG}.out \ 209 >> $mail_msg_file 210 if [ "$?" = "0" ]; then 211 build_ok=n 212 this_build_ok=n 213 fi 214 215 echo "\n==== Build warnings ($LABEL) ====\n" >>$mail_msg_file 216 egrep -i warning: $SRC/${INSTALLOG}.out \ 217 | egrep -v '^tic:' \ 218 | egrep -v "symbol (\`|')timezone' has differing types:" \ 219 | egrep -v "parameter <PSTAMP> set to" \ 220 | egrep -v "Ignoring unknown host" \ 221 | egrep -v "redefining segment flags attribute for" \ 222 | tee $TMPDIR/build_warnings${SUFFIX} >> $mail_msg_file 223 if [[ -s $TMPDIR/build_warnings${SUFFIX} ]]; then 224 build_ok=n 225 this_build_ok=n 226 fi 227 228 echo "\n==== Ended OS-Net source build at `date` ($LABEL) ====\n" \ 229 >> $LOGFILE 230 231 echo "\n==== Elapsed build time ($LABEL) ====\n" >>$mail_msg_file 232 tail -3 $SRC/${INSTALLOG}.out >>$mail_msg_file 233 234 if [ "$i_FLAG" = "n" ]; then 235 rm -f $SRC/${NOISE}.ref 236 if [ -f $SRC/${NOISE}.out ]; then 237 mv $SRC/${NOISE}.out $SRC/${NOISE}.ref 238 fi 239 grep : $SRC/${INSTALLOG}.out \ 240 | egrep -v '^/' \ 241 | egrep -v '^(Start|Finish|real|user|sys|./bld_awk)' \ 242 | egrep -v '^tic:' \ 243 | egrep -v '^mcs' \ 244 | egrep -v '^LD_LIBRARY_PATH=' \ 245 | egrep -v 'ar: creating' \ 246 | egrep -v 'ar: writing' \ 247 | egrep -v 'conflicts:' \ 248 | egrep -v ':saved created' \ 249 | egrep -v '^stty.*c:' \ 250 | egrep -v '^mfgname.c:' \ 251 | egrep -v '^uname-i.c:' \ 252 | egrep -v '^volumes.c:' \ 253 | egrep -v '^lint library construction:' \ 254 | egrep -v 'tsort: INFORM:' \ 255 | egrep -v 'stripalign:' \ 256 | egrep -v 'chars, width' \ 257 | egrep -v "symbol (\`|')timezone' has differing types:" \ 258 | egrep -v 'PSTAMP' \ 259 | egrep -v '|%WHOANDWHERE%|' \ 260 | egrep -v '^Manifying' \ 261 | egrep -v 'Ignoring unknown host' \ 262 | egrep -v 'Processing method:' \ 263 | egrep -v '^Writing' \ 264 | egrep -v 'spellin1:' \ 265 | egrep -v '^adding:' \ 266 | egrep -v "^echo 'msgid" \ 267 | egrep -v '^echo ' \ 268 | egrep -v '\.c:$' \ 269 | egrep -v '^Adding file:' \ 270 | egrep -v 'CLASSPATH=' \ 271 | egrep -v '\/var\/mail\/:saved' \ 272 | egrep -v -- '-DUTS_VERSION=' \ 273 | egrep -v '^Running Mkbootstrap' \ 274 | egrep -v '^Applet length read:' \ 275 | egrep -v 'bytes written:' \ 276 | egrep -v '^File:SolarisAuthApplet.bin' \ 277 | egrep -v -i 'jibversion' \ 278 | egrep -v '^Output size:' \ 279 | egrep -v '^Solo size statistics:' \ 280 | egrep -v '^Using ROM API Version' \ 281 | egrep -v '^Zero Signature length:' \ 282 | egrep -v '^Note \(probably harmless\):' \ 283 | egrep -v '::' \ 284 | egrep -v -- '-xcache' \ 285 | egrep -v '^\+' \ 286 | egrep -v '^cc1: note: -fwritable-strings' \ 287 | egrep -v 'svccfg-native -s svc:/' \ 288 | sort | uniq >$SRC/${NOISE}.out 289 if [ ! -f $SRC/${NOISE}.ref ]; then 290 cp $SRC/${NOISE}.out $SRC/${NOISE}.ref 291 fi 292 echo "\n==== Build noise differences ($LABEL) ====\n" \ 293 >>$mail_msg_file 294 diff $SRC/${NOISE}.ref $SRC/${NOISE}.out >>$mail_msg_file 295 fi 296 297 # 298 # Re-sign selected binaries using signing server 299 # (gatekeeper builds only) 300 # 301 if [ -n "$CODESIGN_USER" -a "$this_build_ok" = "y" ]; then 302 echo "\n==== Signing proto area at `date` ====\n" >> $LOGFILE 303 signing_file="${TMPDIR}/signing" 304 rm -f ${signing_file} 305 export CODESIGN_USER 306 signproto $SRC/tools/codesign/creds 2>&1 | \ 307 tee -a ${signing_file} >> $LOGFILE 308 echo "\n==== Finished signing proto area at `date` ====\n" \ 309 >> $LOGFILE 310 echo "\n==== Crypto module signing errors ($LABEL) ====\n" \ 311 >> $mail_msg_file 312 egrep 'WARNING|ERROR' ${signing_file} >> $mail_msg_file 313 if (( $? == 0 )) ; then 314 build_ok=n 315 this_build_ok=n 316 fi 317 fi 318 319 # 320 # Building Packages 321 # 322 if [ "$p_FLAG" = "y" -a "$this_build_ok" = "y" ]; then 323 if [ -d $SRC/pkg ]; then 324 echo "\n==== Creating $LABEL packages at `date` ====\n" \ 325 >> $LOGFILE 326 echo "Clearing out $PKGARCHIVE ..." >> $LOGFILE 327 rm -rf $PKGARCHIVE >> "$LOGFILE" 2>&1 328 mkdir -p $PKGARCHIVE >> "$LOGFILE" 2>&1 329 330 rm -f $SRC/pkg/${INSTALLOG}.out 331 cd $SRC/pkg 332 /bin/time $MAKE -e install 2>&1 | \ 333 tee -a $SRC/pkg/${INSTALLOG}.out >> $LOGFILE 334 335 echo "\n==== package build errors ($LABEL) ====\n" \ 336 >> $mail_msg_file 337 338 egrep "${MAKE}|ERROR|WARNING" $SRC/pkg/${INSTALLOG}.out | \ 339 grep ':' | \ 340 grep -v PSTAMP | \ 341 egrep -v "Ignoring unknown host" | \ 342 tee $TMPDIR/package >> $mail_msg_file 343 if [[ -s $TMPDIR/package ]]; then 344 build_extras_ok=n 345 this_build_ok=n 346 fi 347 else 348 # 349 # Handle it gracefully if -p was set but there so 350 # no pkg directory. 351 # 352 echo "\n==== No $LABEL packages to build ====\n" \ 353 >> $LOGFILE 354 fi 355 else 356 echo "\n==== Not creating $LABEL packages ====\n" >> $LOGFILE 357 fi 358 359 ROOT=$ORIGROOT 360 } 361 362 # Usage: dolint /dir y|n 363 # Arg. 2 is a flag to turn on/off the lint diff output 364 function dolint { 365 if [ ! -d "$1" ]; then 366 echo "dolint error: $1 is not a directory" 367 exit 1 368 fi 369 370 if [ "$2" != "y" -a "$2" != "n" ]; then 371 echo "dolint internal error: $2 should be 'y' or 'n'" 372 exit 1 373 fi 374 375 lintdir=$1 376 dodiff=$2 377 base=`basename $lintdir` 378 LINTOUT=$lintdir/lint-${MACH}.out 379 LINTNOISE=$lintdir/lint-noise-${MACH} 380 export ENVLDLIBS1=`myldlibs $ROOT` 381 export ENVCPPFLAGS1=`myheaders $ROOT` 382 383 set_debug_build_flags 384 385 # 386 # '$MAKE lint' in $lintdir 387 # 388 echo "\n==== Begin '$MAKE lint' of $base at `date` ====\n" >> $LOGFILE 389 390 # remove old lint.out 391 rm -f $lintdir/lint.out $lintdir/lint-noise.out 392 if [ -f $lintdir/lint-noise.ref ]; then 393 mv $lintdir/lint-noise.ref ${LINTNOISE}.ref 394 fi 395 396 rm -f $LINTOUT 397 cd $lintdir 398 # 399 # Remove all .ln files to ensure a full reference file 400 # 401 rm -f Nothing_to_remove \ 402 `find . \( -name SCCS -o -name .hg -o -name .svn -o -name .git \) \ 403 -prune -o -type f -name '*.ln' -print ` 404 405 /bin/time $MAKE -ek lint 2>&1 | \ 406 tee -a $LINTOUT >> $LOGFILE 407 408 echo "\n==== '$MAKE lint' of $base ERRORS ====\n" >> $mail_msg_file 409 410 grep "$MAKE:" $LINTOUT | 411 egrep -v "Ignoring unknown host" | \ 412 tee $TMPDIR/lint_errs >> $mail_msg_file 413 if [[ -s $TMPDIR/lint_errs ]]; then 414 build_extras_ok=n 415 fi 416 417 echo "\n==== Ended '$MAKE lint' of $base at `date` ====\n" >> $LOGFILE 418 419 echo "\n==== Elapsed time of '$MAKE lint' of $base ====\n" \ 420 >>$mail_msg_file 421 tail -3 $LINTOUT >>$mail_msg_file 422 423 rm -f ${LINTNOISE}.ref 424 if [ -f ${LINTNOISE}.out ]; then 425 mv ${LINTNOISE}.out ${LINTNOISE}.ref 426 fi 427 grep : $LINTOUT | \ 428 egrep -v '^(real|user|sys)' | 429 egrep -v '(library construction)' | \ 430 egrep -v ': global crosschecks' | \ 431 egrep -v 'Ignoring unknown host' | \ 432 egrep -v '\.c:$' | \ 433 sort | uniq > ${LINTNOISE}.out 434 if [ ! -f ${LINTNOISE}.ref ]; then 435 cp ${LINTNOISE}.out ${LINTNOISE}.ref 436 fi 437 438 if [ "$dodiff" != "n" ]; then 439 echo "\n==== lint warnings $base ====\n" \ 440 >>$mail_msg_file 441 # should be none, though there are a few that were filtered out 442 # above 443 egrep -i '(warning|lint):' ${LINTNOISE}.out \ 444 | sort | uniq | tee $TMPDIR/lint_warns >> $mail_msg_file 445 if [[ -s $TMPDIR/lint_warns ]]; then 446 build_extras_ok=n 447 fi 448 echo "\n==== lint noise differences $base ====\n" \ 449 >> $mail_msg_file 450 diff ${LINTNOISE}.ref ${LINTNOISE}.out \ 451 >> $mail_msg_file 452 fi 453 } 454 455 # 456 # Build and install the onbld tools. 457 # 458 # usage: build_tools DESTROOT 459 # 460 # returns non-zero status if the build was successful. 461 # 462 function build_tools { 463 DESTROOT=$1 464 465 INSTALLOG=install-${MACH} 466 467 echo "\n==== Building tools at `date` ====\n" \ 468 >> $LOGFILE 469 470 rm -f ${TOOLS}/${INSTALLOG}.out 471 cd ${TOOLS} 472 /bin/time $MAKE TOOLS_PROTO=${DESTROOT} -e install 2>&1 | \ 473 tee -a ${TOOLS}/${INSTALLOG}.out >> $LOGFILE 474 475 echo "\n==== Tools build errors ====\n" >> $mail_msg_file 476 477 egrep ":" ${TOOLS}/${INSTALLOG}.out | 478 egrep -e "(${MAKE}:|[ ]error[: \n])" | \ 479 egrep -v "Ignoring unknown host" | \ 480 egrep -v warning | tee $TMPDIR/tools_errors >> $mail_msg_file 481 482 if [[ -s $TMPDIR/tools_errors ]]; then 483 return 1 484 fi 485 return 0 486 } 487 488 # 489 # Set up to use locally installed tools. 490 # 491 # usage: use_tools TOOLSROOT 492 # 493 function use_tools { 494 TOOLSROOT=$1 495 496 # 497 # If we're not building ON workspace, then the TOOLSROOT 498 # settings here are clearly ignored by the workspace 499 # makefiles, prepending nonexistent directories to PATH is 500 # harmless, and we clearly do not wish to override 501 # ONBLD_TOOLS. 502 # 503 # If we're building an ON workspace, then the prepended PATH 504 # elements should supercede the preexisting ONBLD_TOOLS paths, 505 # and we want to override ONBLD_TOOLS to catch the tools that 506 # don't have specific path env vars here. 507 # 508 # So the only conditional behavior is overriding ONBLD_TOOLS, 509 # and we check for "an ON workspace" by looking for 510 # ${TOOLSROOT}/opt/onbld. 511 # 512 513 STABS=${TOOLSROOT}/opt/onbld/bin/${MACH}/stabs 514 export STABS 515 CTFSTABS=${TOOLSROOT}/opt/onbld/bin/${MACH}/ctfstabs 516 export CTFSTABS 517 GENOFFSETS=${TOOLSROOT}/opt/onbld/bin/genoffsets 518 export GENOFFSETS 519 520 CTFCONVERT=${TOOLSROOT}/opt/onbld/bin/${MACH}/ctfconvert 521 export CTFCONVERT 522 CTFMERGE=${TOOLSROOT}/opt/onbld/bin/${MACH}/ctfmerge 523 export CTFMERGE 524 525 CTFCVTPTBL=${TOOLSROOT}/opt/onbld/bin/ctfcvtptbl 526 export CTFCVTPTBL 527 CTFFINDMOD=${TOOLSROOT}/opt/onbld/bin/ctffindmod 528 export CTFFINDMOD 529 530 if [ "$VERIFY_ELFSIGN" = "y" ]; then 531 ELFSIGN=${TOOLSROOT}/opt/onbld/bin/elfsigncmp 532 else 533 ELFSIGN=${TOOLSROOT}/opt/onbld/bin/${MACH}/elfsign 534 fi 535 export ELFSIGN 536 537 PATH="${TOOLSROOT}/opt/onbld/bin/${MACH}:${PATH}" 538 PATH="${TOOLSROOT}/opt/onbld/bin:${PATH}" 539 export PATH 540 541 if [ -d "${TOOLSROOT}/opt/onbld" ]; then 542 ONBLD_TOOLS=${TOOLSROOT}/opt/onbld 543 export ONBLD_TOOLS 544 fi 545 546 echo "\n==== New environment settings. ====\n" >> $LOGFILE 547 echo "STABS=${STABS}" >> $LOGFILE 548 echo "CTFSTABS=${CTFSTABS}" >> $LOGFILE 549 echo "CTFCONVERT=${CTFCONVERT}" >> $LOGFILE 550 echo "CTFMERGE=${CTFMERGE}" >> $LOGFILE 551 echo "CTFCVTPTBL=${CTFCVTPTBL}" >> $LOGFILE 552 echo "CTFFINDMOD=${CTFFINDMOD}" >> $LOGFILE 553 echo "ELFSIGN=${ELFSIGN}" >> $LOGFILE 554 echo "PATH=${PATH}" >> $LOGFILE 555 echo "ONBLD_TOOLS=${ONBLD_TOOLS}" >> $LOGFILE 556 } 557 558 function staffer { 559 if [ $ISUSER -ne 0 ]; then 560 "$@" 561 else 562 arg="\"$1\"" 563 shift 564 for i 565 do 566 arg="$arg \"$i\"" 567 done 568 eval su $STAFFER -c \'$arg\' 569 fi 570 } 571 572 # 573 # Verify that the closed bins are present 574 # 575 function check_closed_bins { 576 if [[ ! -d "$ON_CLOSED_BINS" ]]; then 577 echo "ON_CLOSED_BINS must point to the closed binaries tree." 578 build_ok=n 579 exit 1 580 fi 581 } 582 583 # 584 # wrapper over wsdiff. 585 # usage: do_wsdiff LABEL OLDPROTO NEWPROTO 586 # 587 function do_wsdiff { 588 label=$1 589 oldproto=$2 590 newproto=$3 591 592 wsdiff="wsdiff" 593 [ "$t_FLAG" = y ] && wsdiff="wsdiff -t" 594 595 echo "\n==== Getting object changes since last build at `date`" \ 596 "($label) ====\n" | tee -a $LOGFILE >> $mail_msg_file 597 $wsdiff -s -r ${TMPDIR}/wsdiff.results $oldproto $newproto 2>&1 | \ 598 tee -a $LOGFILE >> $mail_msg_file 599 echo "\n==== Object changes determined at `date` ($label) ====\n" | \ 600 tee -a $LOGFILE >> $mail_msg_file 601 } 602 603 # 604 # Functions for setting build flags (DEBUG/non-DEBUG). Keep them 605 # together. 606 # 607 608 function set_non_debug_build_flags { 609 export RELEASE_BUILD ; RELEASE_BUILD= 610 unset EXTRA_OPTIONS 611 unset EXTRA_CFLAGS 612 } 613 614 function set_debug_build_flags { 615 unset RELEASE_BUILD 616 unset EXTRA_OPTIONS 617 unset EXTRA_CFLAGS 618 } 619 620 621 MACH=`uname -p` 622 623 if [ "$OPTHOME" = "" ]; then 624 OPTHOME=/opt 625 export OPTHOME 626 fi 627 628 USAGE='Usage: nightly [-in] [+t] [-V VERS ] <env_file> 629 630 Where: 631 -i Fast incremental options (no clobber, lint, check) 632 -n Do not do a bringover 633 +t Use the build tools in $ONBLD_TOOLS/bin 634 -V VERS set the build version string to VERS 635 636 <env_file> file in Bourne shell syntax that sets and exports 637 variables that configure the operation of this script and many of 638 the scripts this one calls. If <env_file> does not exist, 639 it will be looked for in $OPTHOME/onbld/env. 640 641 non-DEBUG is the default build type. Build options can be set in the 642 NIGHTLY_OPTIONS variable in the <env_file> as follows: 643 644 -A check for ABI differences in .so files 645 -C check for cstyle/hdrchk errors 646 -D do a build with DEBUG on 647 -F do _not_ do a non-DEBUG build 648 -G gate keeper default group of options (-au) 649 -I integration engineer default group of options (-ampu) 650 -M do not run pmodes (safe file permission checker) 651 -N do not run protocmp 652 -R default group of options for building a release (-mp) 653 -U update proto area in the parent 654 -V VERS set the build version string to VERS 655 -f find unreferenced files 656 -i do an incremental build (no "make clobber") 657 -l do "make lint" in $LINTDIRS (default: $SRC y) 658 -m send mail to $MAILTO at end of build 659 -n do not do a bringover 660 -p create packages 661 -r check ELF runtime attributes in the proto area 662 -t build and use the tools in $SRC/tools (default setting) 663 +t Use the build tools in $ONBLD_TOOLS/bin 664 -u update proto_list_$MACH and friends in the parent workspace; 665 when used with -f, also build an unrefmaster.out in the parent 666 -w report on differences between previous and current proto areas 667 ' 668 # 669 # A log file will be generated under the name $LOGFILE 670 # for partially completed build and log.`date '+%F'` 671 # in the same directory for fully completed builds. 672 # 673 674 # default values for low-level FLAGS; G I R are group FLAGS 675 A_FLAG=n 676 C_FLAG=n 677 D_FLAG=n 678 F_FLAG=n 679 f_FLAG=n 680 i_FLAG=n; i_CMD_LINE_FLAG=n 681 l_FLAG=n 682 M_FLAG=n 683 m_FLAG=n 684 N_FLAG=n 685 n_FLAG=n 686 p_FLAG=n 687 r_FLAG=n 688 t_FLAG=y 689 U_FLAG=n 690 u_FLAG=n 691 V_FLAG=n 692 w_FLAG=n 693 W_FLAG=n 694 # 695 build_ok=y 696 build_extras_ok=y 697 698 # 699 # examine arguments 700 # 701 702 OPTIND=1 703 while getopts +intV:W FLAG 704 do 705 case $FLAG in 706 i ) i_FLAG=y; i_CMD_LINE_FLAG=y 707 ;; 708 n ) n_FLAG=y 709 ;; 710 +t ) t_FLAG=n 711 ;; 712 V ) V_FLAG=y 713 V_ARG="$OPTARG" 714 ;; 715 W ) W_FLAG=y 716 ;; 717 \? ) echo "$USAGE" 718 exit 1 719 ;; 720 esac 721 done 722 723 # correct argument count after options 724 shift `expr $OPTIND - 1` 725 726 # test that the path to the environment-setting file was given 727 if [ $# -ne 1 ]; then 728 echo "$USAGE" 729 exit 1 730 fi 731 732 # check if user is running nightly as root 733 # ISUSER is set non-zero if an ordinary user runs nightly, or is zero 734 # when root invokes nightly. 735 /usr/bin/id | grep '^uid=0(' >/dev/null 2>&1 736 ISUSER=$?; export ISUSER 737 738 # 739 # force locale to C 740 LANG=C; export LANG 741 LC_ALL=C; export LC_ALL 742 LC_COLLATE=C; export LC_COLLATE 743 LC_CTYPE=C; export LC_CTYPE 744 LC_MESSAGES=C; export LC_MESSAGES 745 LC_MONETARY=C; export LC_MONETARY 746 LC_NUMERIC=C; export LC_NUMERIC 747 LC_TIME=C; export LC_TIME 748 749 # clear environment variables we know to be bad for the build 750 unset LD_OPTIONS 751 unset LD_AUDIT LD_AUDIT_32 LD_AUDIT_64 752 unset LD_BIND_NOW LD_BIND_NOW_32 LD_BIND_NOW_64 753 unset LD_BREADTH LD_BREADTH_32 LD_BREADTH_64 754 unset LD_CONFIG LD_CONFIG_32 LD_CONFIG_64 755 unset LD_DEBUG LD_DEBUG_32 LD_DEBUG_64 756 unset LD_DEMANGLE LD_DEMANGLE_32 LD_DEMANGLE_64 757 unset LD_FLAGS LD_FLAGS_32 LD_FLAGS_64 758 unset LD_LIBRARY_PATH LD_LIBRARY_PATH_32 LD_LIBRARY_PATH_64 759 unset LD_LOADFLTR LD_LOADFLTR_32 LD_LOADFLTR_64 760 unset LD_NOAUDIT LD_NOAUDIT_32 LD_NOAUDIT_64 761 unset LD_NOAUXFLTR LD_NOAUXFLTR_32 LD_NOAUXFLTR_64 762 unset LD_NOCONFIG LD_NOCONFIG_32 LD_NOCONFIG_64 763 unset LD_NODIRCONFIG LD_NODIRCONFIG_32 LD_NODIRCONFIG_64 764 unset LD_NODIRECT LD_NODIRECT_32 LD_NODIRECT_64 765 unset LD_NOLAZYLOAD LD_NOLAZYLOAD_32 LD_NOLAZYLOAD_64 766 unset LD_NOOBJALTER LD_NOOBJALTER_32 LD_NOOBJALTER_64 767 unset LD_NOVERSION LD_NOVERSION_32 LD_NOVERSION_64 768 unset LD_ORIGIN LD_ORIGIN_32 LD_ORIGIN_64 769 unset LD_PRELOAD LD_PRELOAD_32 LD_PRELOAD_64 770 unset LD_PROFILE LD_PROFILE_32 LD_PROFILE_64 771 772 unset CONFIG 773 unset GROUP 774 unset OWNER 775 unset REMOTE 776 unset ENV 777 unset ARCH 778 unset CLASSPATH 779 unset NAME 780 781 # 782 # To get ONBLD_TOOLS from the environment, it must come from the env file. 783 # If it comes interactively, it is generally TOOLS_PROTO, which will be 784 # clobbered before the compiler version checks, which will therefore fail. 785 # 786 unset ONBLD_TOOLS 787 788 # 789 # Setup environmental variables 790 # 791 if [ -f /etc/nightly.conf ]; then 792 . /etc/nightly.conf 793 fi 794 795 if [ -f $1 ]; then 796 if [[ $1 = */* ]]; then 797 . $1 798 else 799 . ./$1 800 fi 801 else 802 if [ -f $OPTHOME/onbld/env/$1 ]; then 803 . $OPTHOME/onbld/env/$1 804 else 805 echo "Cannot find env file as either $1 or $OPTHOME/onbld/env/$1" 806 exit 1 807 fi 808 fi 809 810 # contents of stdenv.sh inserted after next line: 811 # STDENV_START 812 # STDENV_END 813 814 # Check if we have sufficient data to continue... 815 [[ -v CODEMGR_WS ]] || fatal_error "Error: Variable CODEMGR_WS not set." 816 if [[ "${NIGHTLY_OPTIONS}" == ~(F)n ]] ; then 817 # Check if the gate data are valid if we don't do a "bringover" below 818 [[ -d "${CODEMGR_WS}" ]] || \ 819 fatal_error "Error: ${CODEMGR_WS} is not a directory." 820 [[ -f "${CODEMGR_WS}/usr/src/Makefile" ]] || \ 821 fatal_error "Error: ${CODEMGR_WS}/usr/src/Makefile not found." 822 fi 823 824 # 825 # place ourselves in a new task, respecting BUILD_PROJECT if set. 826 # 827 if [ -z "$BUILD_PROJECT" ]; then 828 /usr/bin/newtask -c $$ 829 else 830 /usr/bin/newtask -c $$ -p $BUILD_PROJECT 831 fi 832 833 ps -o taskid= -p $$ | read build_taskid 834 ps -o project= -p $$ | read build_project 835 836 # 837 # See if NIGHTLY_OPTIONS is set 838 # 839 if [ "$NIGHTLY_OPTIONS" = "" ]; then 840 NIGHTLY_OPTIONS="-aBm" 841 fi 842 843 # 844 # If BRINGOVER_WS was not specified, let it default to CLONE_WS 845 # 846 if [ "$BRINGOVER_WS" = "" ]; then 847 BRINGOVER_WS=$CLONE_WS 848 fi 849 850 # 851 # If BRINGOVER_FILES was not specified, default to usr 852 # 853 if [ "$BRINGOVER_FILES" = "" ]; then 854 BRINGOVER_FILES="usr" 855 fi 856 857 check_closed_bins 858 859 # 860 # Note: changes to the option letters here should also be applied to the 861 # bldenv script. `d' is listed for backward compatibility. 862 # 863 NIGHTLY_OPTIONS=-${NIGHTLY_OPTIONS#-} 864 OPTIND=1 865 while getopts +ABCDdFfGIilMmNnpRrtUuwW FLAG $NIGHTLY_OPTIONS 866 do 867 case $FLAG in 868 A ) A_FLAG=y 869 ;; 870 B ) D_FLAG=y 871 ;; # old version of D 872 C ) C_FLAG=y 873 ;; 874 D ) D_FLAG=y 875 ;; 876 F ) F_FLAG=y 877 ;; 878 f ) f_FLAG=y 879 ;; 880 G ) u_FLAG=y 881 ;; 882 I ) m_FLAG=y 883 p_FLAG=y 884 u_FLAG=y 885 ;; 886 i ) i_FLAG=y 887 ;; 888 l ) l_FLAG=y 889 ;; 890 M ) M_FLAG=y 891 ;; 892 m ) m_FLAG=y 893 ;; 894 N ) N_FLAG=y 895 ;; 896 n ) n_FLAG=y 897 ;; 898 p ) p_FLAG=y 899 ;; 900 R ) m_FLAG=y 901 p_FLAG=y 902 ;; 903 r ) r_FLAG=y 904 ;; 905 +t ) t_FLAG=n 906 ;; 907 U ) if [ -z "${PARENT_ROOT}" ]; then 908 echo "PARENT_ROOT must be set if the U flag is" \ 909 "present in NIGHTLY_OPTIONS." 910 exit 1 911 fi 912 NIGHTLY_PARENT_ROOT=$PARENT_ROOT 913 if [ -n "${PARENT_TOOLS_ROOT}" ]; then 914 NIGHTLY_PARENT_TOOLS_ROOT=$PARENT_TOOLS_ROOT 915 fi 916 U_FLAG=y 917 ;; 918 u ) u_FLAG=y 919 ;; 920 w ) w_FLAG=y 921 ;; 922 W ) W_FLAG=y 923 ;; 924 \? ) echo "$USAGE" 925 exit 1 926 ;; 927 esac 928 done 929 930 if [ $ISUSER -ne 0 ]; then 931 # Set default value for STAFFER, if needed. 932 if [ -z "$STAFFER" -o "$STAFFER" = "nobody" ]; then 933 STAFFER=`/usr/xpg4/bin/id -un` 934 export STAFFER 935 fi 936 fi 937 938 if [ -z "$MAILTO" -o "$MAILTO" = "nobody" ]; then 939 MAILTO=$STAFFER 940 export MAILTO 941 fi 942 943 PATH="$OPTHOME/onbld/bin:$OPTHOME/onbld/bin/${MACH}:/usr/ccs/bin" 944 PATH="$PATH:$OPTHOME/SUNWspro/bin:/usr/bin:/usr/sbin:/usr/ucb" 945 PATH="$PATH:/usr/openwin/bin:/usr/sfw/bin:/opt/sfw/bin:." 946 export PATH 947 948 # roots of source trees, both relative to $SRC and absolute. 949 relsrcdirs="." 950 abssrcdirs="$SRC" 951 952 PROTOCMPTERSE="protocmp.terse -gu" 953 POUND_SIGN="#" 954 # have we set RELEASE_DATE in our env file? 955 if [ -z "$RELEASE_DATE" ]; then 956 RELEASE_DATE=$(LC_ALL=C date +"%B %Y") 957 fi 958 BUILD_DATE=$(LC_ALL=C date +%Y-%b-%d) 959 BASEWSDIR=$(basename $CODEMGR_WS) 960 DEV_CM="\"@(#)SunOS Internal Development: $LOGNAME $BUILD_DATE [$BASEWSDIR]\"" 961 962 # we export POUND_SIGN, RELEASE_DATE and DEV_CM to speed up the build process 963 # by avoiding repeated shell invocations to evaluate Makefile.master 964 # definitions. 965 export POUND_SIGN RELEASE_DATE DEV_CM 966 967 maketype="distributed" 968 if [[ -z "$MAKE" ]]; then 969 MAKE=dmake 970 elif [[ ! -x "$MAKE" ]]; then 971 echo "\$MAKE is set to garbage in the environment" 972 exit 1 973 fi 974 # get the dmake version string alone 975 DMAKE_VERSION=$( $MAKE -v ) 976 DMAKE_VERSION=${DMAKE_VERSION#*: } 977 # focus in on just the dotted version number alone 978 DMAKE_MAJOR=$( echo $DMAKE_VERSION | \ 979 sed -e 's/.*\<\([^.]*\.[^ ]*\).*$/\1/' ) 980 # extract the second (or final) integer 981 DMAKE_MINOR=${DMAKE_MAJOR#*.} 982 DMAKE_MINOR=${DMAKE_MINOR%%.*} 983 # extract the first integer 984 DMAKE_MAJOR=${DMAKE_MAJOR%%.*} 985 CHECK_DMAKE=${CHECK_DMAKE:-y} 986 # x86 was built on the 12th, sparc on the 13th. 987 if [ "$CHECK_DMAKE" = "y" -a \ 988 "$DMAKE_VERSION" != "Sun Distributed Make 7.3 2003/03/12" -a \ 989 "$DMAKE_VERSION" != "Sun Distributed Make 7.3 2003/03/13" -a \( \ 990 "$DMAKE_MAJOR" -lt 7 -o \ 991 "$DMAKE_MAJOR" -eq 7 -a "$DMAKE_MINOR" -lt 4 \) ]; then 992 if [ -z "$DMAKE_VERSION" ]; then 993 echo "$MAKE is missing." 994 exit 1 995 fi 996 echo `whence $MAKE`" version is:" 997 echo " ${DMAKE_VERSION}" 998 cat <<EOF 999 1000 This version may not be safe for use, if you really want to use this version 1001 anyway add the following to your environment to disable this check: 1002 1003 CHECK_DMAKE=n 1004 EOF 1005 exit 1 1006 fi 1007 export PATH 1008 export MAKE 1009 1010 if [ "${SUNWSPRO}" != "" ]; then 1011 PATH="${SUNWSPRO}/bin:$PATH" 1012 export PATH 1013 fi 1014 1015 hostname=$(uname -n) 1016 if [[ $DMAKE_MAX_JOBS != +([0-9]) || $DMAKE_MAX_JOBS -eq 0 ]] 1017 then 1018 maxjobs= 1019 if [[ -f $HOME/.make.machines ]] 1020 then 1021 # Note: there is a hard tab and space character in the []s 1022 # below. 1023 egrep -i "^[ ]*$hostname[ \.]" \ 1024 $HOME/.make.machines | read host jobs 1025 maxjobs=${jobs##*=} 1026 fi 1027 1028 if [[ $maxjobs != +([0-9]) || $maxjobs -eq 0 ]] 1029 then 1030 # default 1031 maxjobs=4 1032 fi 1033 1034 export DMAKE_MAX_JOBS=$maxjobs 1035 fi 1036 1037 DMAKE_MODE=parallel; 1038 export DMAKE_MODE 1039 1040 if [ -z "${ROOT}" ]; then 1041 echo "ROOT must be set." 1042 exit 1 1043 fi 1044 1045 # 1046 # if -V flag was given, reset VERSION to V_ARG 1047 # 1048 if [ "$V_FLAG" = "y" ]; then 1049 VERSION=$V_ARG 1050 fi 1051 1052 TMPDIR="/tmp/nightly.tmpdir.$$" 1053 export TMPDIR 1054 rm -rf ${TMPDIR} 1055 mkdir -p $TMPDIR || exit 1 1056 chmod 777 $TMPDIR 1057 1058 # 1059 # Keep elfsign's use of pkcs11_softtoken from looking in the user home 1060 # directory, which doesn't always work. Needed until all build machines 1061 # have the fix for 6271754 1062 # 1063 SOFTTOKEN_DIR=$TMPDIR 1064 export SOFTTOKEN_DIR 1065 1066 # 1067 # Tools should only be built non-DEBUG. Keep track of the tools proto 1068 # area path relative to $TOOLS, because the latter changes in an 1069 # export build. 1070 # 1071 # TOOLS_PROTO is included below for builds other than usr/src/tools 1072 # that look for this location. For usr/src/tools, this will be 1073 # overridden on the $MAKE command line in build_tools(). 1074 # 1075 TOOLS=${SRC}/tools 1076 TOOLS_PROTO_REL=proto/root_${MACH}-nd 1077 TOOLS_PROTO=${TOOLS}/${TOOLS_PROTO_REL}; export TOOLS_PROTO 1078 1079 unset CFLAGS LD_LIBRARY_PATH LDFLAGS 1080 1081 # create directories that are automatically removed if the nightly script 1082 # fails to start correctly 1083 function newdir { 1084 dir=$1 1085 toadd= 1086 while [ ! -d $dir ]; do 1087 toadd="$dir $toadd" 1088 dir=`dirname $dir` 1089 done 1090 torm= 1091 newlist= 1092 for dir in $toadd; do 1093 if staffer mkdir $dir; then 1094 newlist="$ISUSER $dir $newlist" 1095 torm="$dir $torm" 1096 else 1097 [ -z "$torm" ] || staffer rmdir $torm 1098 return 1 1099 fi 1100 done 1101 newdirlist="$newlist $newdirlist" 1102 return 0 1103 } 1104 newdirlist= 1105 1106 [ -d $CODEMGR_WS ] || newdir $CODEMGR_WS || exit 1 1107 1108 # since this script assumes the build is from full source, it nullifies 1109 # variables likely to have been set by a "ws" script; nullification 1110 # confines the search space for headers and libraries to the proto area 1111 # built from this immediate source. 1112 ENVLDLIBS1= 1113 ENVLDLIBS2= 1114 ENVLDLIBS3= 1115 ENVCPPFLAGS1= 1116 ENVCPPFLAGS2= 1117 ENVCPPFLAGS3= 1118 ENVCPPFLAGS4= 1119 PARENT_ROOT= 1120 1121 export ENVLDLIBS3 ENVCPPFLAGS1 ENVCPPFLAGS2 ENVCPPFLAGS3 ENVCPPFLAGS4 \ 1122 ENVLDLIBS1 ENVLDLIBS2 PARENT_ROOT 1123 1124 PKGARCHIVE_ORIG=$PKGARCHIVE 1125 1126 # 1127 # Juggle the logs and optionally send mail on completion. 1128 # 1129 1130 function logshuffle { 1131 LLOG="$ATLOG/log.`date '+%F.%H:%M'`" 1132 if [ -f $LLOG -o -d $LLOG ]; then 1133 LLOG=$LLOG.$$ 1134 fi 1135 mkdir $LLOG 1136 export LLOG 1137 1138 if [ "$build_ok" = "y" ]; then 1139 mv $ATLOG/proto_list_${MACH} $LLOG 1140 1141 if [ -f $ATLOG/proto_list_tools_${MACH} ]; then 1142 mv $ATLOG/proto_list_tools_${MACH} $LLOG 1143 fi 1144 1145 if [ -f $TMPDIR/wsdiff.results ]; then 1146 mv $TMPDIR/wsdiff.results $LLOG 1147 fi 1148 1149 if [ -f $TMPDIR/wsdiff-nd.results ]; then 1150 mv $TMPDIR/wsdiff-nd.results $LLOG 1151 fi 1152 fi 1153 1154 # 1155 # Now that we're about to send mail, it's time to check the noise 1156 # file. In the event that an error occurs beyond this point, it will 1157 # be recorded in the nightly.log file, but nowhere else. This would 1158 # include only errors that cause the copying of the noise log to fail 1159 # or the mail itself not to be sent. 1160 # 1161 1162 exec >>$LOGFILE 2>&1 1163 if [ -s $build_noise_file ]; then 1164 echo "\n==== Nightly build noise ====\n" | 1165 tee -a $LOGFILE >>$mail_msg_file 1166 cat $build_noise_file >>$LOGFILE 1167 cat $build_noise_file >>$mail_msg_file 1168 echo | tee -a $LOGFILE >>$mail_msg_file 1169 fi 1170 rm -f $build_noise_file 1171 1172 case "$build_ok" in 1173 y) 1174 state=Completed 1175 ;; 1176 i) 1177 state=Interrupted 1178 ;; 1179 *) 1180 state=Failed 1181 ;; 1182 esac 1183 1184 if [[ $state != "Interrupted" && $build_extras_ok != "y" ]]; then 1185 state=Failed 1186 fi 1187 1188 NIGHTLY_STATUS=$state 1189 export NIGHTLY_STATUS 1190 1191 run_hook POST_NIGHTLY $state 1192 run_hook SYS_POST_NIGHTLY $state 1193 1194 # 1195 # mailx(1) sets From: based on the -r flag 1196 # if it is given. 1197 # 1198 mailx_r= 1199 if [[ -n "${MAILFROM}" ]]; then 1200 mailx_r="-r ${MAILFROM}" 1201 fi 1202 1203 cat $build_time_file $build_environ_file $mail_msg_file \ 1204 > ${LLOG}/mail_msg 1205 if [ "$m_FLAG" = "y" ]; then 1206 cat ${LLOG}/mail_msg | /usr/bin/mailx ${mailx_r} -s \ 1207 "Nightly ${MACH} Build of `basename ${CODEMGR_WS}` ${state}." \ 1208 ${MAILTO} 1209 fi 1210 1211 if [ "$u_FLAG" = "y" -a "$build_ok" = "y" ]; then 1212 staffer cp ${LLOG}/mail_msg $PARENT_WS/usr/src/mail_msg-${MACH} 1213 staffer cp $LOGFILE $PARENT_WS/usr/src/nightly-${MACH}.log 1214 fi 1215 1216 mv $LOGFILE $LLOG 1217 } 1218 1219 # 1220 # Remove the locks and temporary files on any exit 1221 # 1222 function cleanup { 1223 logshuffle 1224 1225 [ -z "$lockfile" ] || staffer rm -f $lockfile 1226 [ -z "$atloglockfile" ] || rm -f $atloglockfile 1227 [ -z "$ulockfile" ] || staffer rm -f $ulockfile 1228 [ -z "$Ulockfile" ] || rm -f $Ulockfile 1229 1230 set -- $newdirlist 1231 while [ $# -gt 0 ]; do 1232 ISUSER=$1 staffer rmdir $2 1233 shift; shift 1234 done 1235 rm -rf $TMPDIR 1236 } 1237 1238 function cleanup_signal { 1239 build_ok=i 1240 # this will trigger cleanup(), above. 1241 exit 1 1242 } 1243 1244 trap cleanup 0 1245 trap cleanup_signal 1 2 3 15 1246 1247 # 1248 # Generic lock file processing -- make sure that the lock file doesn't 1249 # exist. If it does, it should name the build host and PID. If it 1250 # doesn't, then make sure we can create it. Clean up locks that are 1251 # known to be stale (assumes host name is unique among build systems 1252 # for the workspace). 1253 # 1254 function create_lock { 1255 lockf=$1 1256 lockvar=$2 1257 1258 ldir=`dirname $lockf` 1259 [ -d $ldir ] || newdir $ldir || exit 1 1260 eval $lockvar=$lockf 1261 1262 while ! staffer ln -s $hostname.$STAFFER.$$ $lockf 2> /dev/null; do 1263 basews=`basename $CODEMGR_WS` 1264 ls -l $lockf | nawk '{print $NF}' | IFS=. read host user pid 1265 if [ "$host" != "$hostname" ]; then 1266 echo "$MACH build of $basews apparently" \ 1267 "already started by $user on $host as $pid." 1268 exit 1 1269 elif kill -s 0 $pid 2>/dev/null; then 1270 echo "$MACH build of $basews already started" \ 1271 "by $user as $pid." 1272 exit 1 1273 else 1274 # stale lock; clear it out and try again 1275 rm -f $lockf 1276 fi 1277 done 1278 } 1279 1280 # 1281 # Return the list of interesting proto areas, depending on the current 1282 # options. 1283 # 1284 function allprotos { 1285 typeset roots="$ROOT" 1286 1287 if [[ "$F_FLAG" = n && "$MULTI_PROTO" = yes ]]; then 1288 roots="$roots $ROOT-nd" 1289 fi 1290 1291 echo $roots 1292 } 1293 1294 # Ensure no other instance of this script is running on this host. 1295 # LOCKNAME can be set in <env_file>, and is by default, but is not 1296 # required due to the use of $ATLOG below. 1297 if [ -n "$LOCKNAME" ]; then 1298 create_lock /tmp/$LOCKNAME "lockfile" 1299 fi 1300 # 1301 # Create from one, two, or three other locks: 1302 # $ATLOG/nightly.lock 1303 # - protects against multiple builds in same workspace 1304 # $PARENT_WS/usr/src/nightly.$MACH.lock 1305 # - protects against multiple 'u' copy-backs 1306 # $NIGHTLY_PARENT_ROOT/nightly.lock 1307 # - protects against multiple 'U' copy-backs 1308 # 1309 # Overriding ISUSER to 1 causes the lock to be created as root if the 1310 # script is run as root. The default is to create it as $STAFFER. 1311 ISUSER=1 create_lock $ATLOG/nightly.lock "atloglockfile" 1312 if [ "$u_FLAG" = "y" ]; then 1313 create_lock $PARENT_WS/usr/src/nightly.$MACH.lock "ulockfile" 1314 fi 1315 if [ "$U_FLAG" = "y" ]; then 1316 # NIGHTLY_PARENT_ROOT is written as root if script invoked as root. 1317 ISUSER=1 create_lock $NIGHTLY_PARENT_ROOT/nightly.lock "Ulockfile" 1318 fi 1319 1320 # Locks have been taken, so we're doing a build and we're committed to 1321 # the directories we may have created so far. 1322 newdirlist= 1323 1324 # 1325 # Create mail_msg_file 1326 # 1327 mail_msg_file="${TMPDIR}/mail_msg" 1328 touch $mail_msg_file 1329 build_time_file="${TMPDIR}/build_time" 1330 build_environ_file="${TMPDIR}/build_environ" 1331 touch $build_environ_file 1332 # 1333 # Move old LOGFILE aside 1334 # ATLOG directory already made by 'create_lock' above 1335 # 1336 if [ -f $LOGFILE ]; then 1337 mv -f $LOGFILE ${LOGFILE}- 1338 fi 1339 # 1340 # Build OsNet source 1341 # 1342 START_DATE=`date` 1343 SECONDS=0 1344 echo "\n==== Nightly $maketype build started: $START_DATE ====" \ 1345 | tee -a $LOGFILE > $build_time_file 1346 1347 echo "\nBuild project: $build_project\nBuild taskid: $build_taskid" | \ 1348 tee -a $mail_msg_file >> $LOGFILE 1349 1350 # make sure we log only to the nightly build file 1351 build_noise_file="${TMPDIR}/build_noise" 1352 exec </dev/null >$build_noise_file 2>&1 1353 1354 run_hook SYS_PRE_NIGHTLY 1355 run_hook PRE_NIGHTLY 1356 1357 echo "\n==== list of environment variables ====\n" >> $LOGFILE 1358 env >> $LOGFILE 1359 1360 echo "\n==== Nightly argument issues ====\n" | tee -a $mail_msg_file >> $LOGFILE 1361 1362 if [ "$N_FLAG" = "y" ]; then 1363 if [ "$p_FLAG" = "y" ]; then 1364 cat <<EOF | tee -a $mail_msg_file >> $LOGFILE 1365 WARNING: the p option (create packages) is set, but so is the N option (do 1366 not run protocmp); this is dangerous; you should unset the N option 1367 EOF 1368 else 1369 cat <<EOF | tee -a $mail_msg_file >> $LOGFILE 1370 Warning: the N option (do not run protocmp) is set; it probably shouldn't be 1371 EOF 1372 fi 1373 echo "" | tee -a $mail_msg_file >> $LOGFILE 1374 fi 1375 1376 if [ "$D_FLAG" = "n" -a "$l_FLAG" = "y" ]; then 1377 # 1378 # In the past we just complained but went ahead with the lint 1379 # pass, even though the proto area was built non-DEBUG. It's 1380 # unlikely that non-DEBUG headers will make a difference, but 1381 # rather than assuming it's a safe combination, force the user 1382 # to specify a DEBUG build. 1383 # 1384 echo "WARNING: DEBUG build not requested; disabling lint.\n" \ 1385 | tee -a $mail_msg_file >> $LOGFILE 1386 l_FLAG=n 1387 fi 1388 1389 if [ "$f_FLAG" = "y" ]; then 1390 if [ "$i_FLAG" = "y" ]; then 1391 echo "WARNING: the -f flag cannot be used during incremental" \ 1392 "builds; ignoring -f\n" | tee -a $mail_msg_file >> $LOGFILE 1393 f_FLAG=n 1394 fi 1395 if [ "${l_FLAG}${p_FLAG}" != "yy" ]; then 1396 echo "WARNING: the -f flag requires -l, and -p;" \ 1397 "ignoring -f\n" | tee -a $mail_msg_file >> $LOGFILE 1398 f_FLAG=n 1399 fi 1400 fi 1401 1402 if [ "$w_FLAG" = "y" -a ! -d $ROOT ]; then 1403 echo "WARNING: -w specified, but $ROOT does not exist;" \ 1404 "ignoring -w\n" | tee -a $mail_msg_file >> $LOGFILE 1405 w_FLAG=n 1406 fi 1407 1408 if [ "$t_FLAG" = "n" ]; then 1409 # 1410 # We're not doing a tools build, so make sure elfsign(1) is 1411 # new enough to safely sign non-crypto binaries. We test 1412 # debugging output from elfsign to detect the old version. 1413 # 1414 newelfsigntest=`SUNW_CRYPTO_DEBUG=stderr /usr/bin/elfsign verify \ 1415 -e /usr/lib/security/pkcs11_softtoken.so.1 2>&1 \ 1416 | egrep algorithmOID` 1417 if [ -z "$newelfsigntest" ]; then 1418 echo "WARNING: /usr/bin/elfsign out of date;" \ 1419 "will only sign crypto modules\n" | \ 1420 tee -a $mail_msg_file >> $LOGFILE 1421 export ELFSIGN_OBJECT=true 1422 elif [ "$VERIFY_ELFSIGN" = "y" ]; then 1423 echo "WARNING: VERIFY_ELFSIGN=y requires" \ 1424 "the -t flag; ignoring VERIFY_ELFSIGN\n" | \ 1425 tee -a $mail_msg_file >> $LOGFILE 1426 fi 1427 fi 1428 1429 case $MULTI_PROTO in 1430 yes|no) ;; 1431 *) 1432 echo "WARNING: MULTI_PROTO is \"$MULTI_PROTO\"; " \ 1433 "should be \"yes\" or \"no\"." | tee -a $mail_msg_file >> $LOGFILE 1434 echo "Setting MULTI_PROTO to \"no\".\n" | \ 1435 tee -a $mail_msg_file >> $LOGFILE 1436 export MULTI_PROTO=no 1437 ;; 1438 esac 1439 1440 echo "\n==== Build version ====\n" | tee -a $mail_msg_file >> $LOGFILE 1441 echo $VERSION | tee -a $mail_msg_file >> $LOGFILE 1442 1443 # Save the current proto area if we're comparing against the last build 1444 if [ "$w_FLAG" = "y" -a -d "$ROOT" ]; then 1445 if [ -d "$ROOT.prev" ]; then 1446 rm -rf $ROOT.prev 1447 fi 1448 mv $ROOT $ROOT.prev 1449 fi 1450 1451 # Same for non-DEBUG proto area 1452 if [ "$w_FLAG" = "y" -a "$MULTI_PROTO" = yes -a -d "$ROOT-nd" ]; then 1453 if [ -d "$ROOT-nd.prev" ]; then 1454 rm -rf $ROOT-nd.prev 1455 fi 1456 mv $ROOT-nd $ROOT-nd.prev 1457 fi 1458 1459 # 1460 # Echo the SCM type of the parent workspace, this can't just be which_scm 1461 # as that does not know how to identify various network repositories. 1462 # 1463 function parent_wstype { 1464 typeset scm_type junk 1465 1466 CODEMGR_WS="$BRINGOVER_WS" "$WHICH_SCM" 2>/dev/null \ 1467 | read scm_type junk 1468 if [[ -z "$scm_type" || "$scm_type" == unknown ]]; then 1469 # Probe BRINGOVER_WS to determine its type 1470 if [[ $BRINGOVER_WS == ssh://* ]]; then 1471 scm_type="mercurial" 1472 elif [[ $BRINGOVER_WS == http://* ]] && \ 1473 wget -q -O- --save-headers "$BRINGOVER_WS/?cmd=heads" | \ 1474 egrep -s "application/mercurial" 2> /dev/null; then 1475 scm_type="mercurial" 1476 else 1477 scm_type="none" 1478 fi 1479 fi 1480 1481 # fold both unsupported and unrecognized results into "none" 1482 case "$scm_type" in 1483 mercurial) 1484 ;; 1485 *) scm_type=none 1486 ;; 1487 esac 1488 1489 echo $scm_type 1490 } 1491 1492 # Echo the SCM types of $CODEMGR_WS and $BRINGOVER_WS 1493 function child_wstype { 1494 typeset scm_type junk 1495 1496 # Probe CODEMGR_WS to determine its type 1497 if [[ -d $CODEMGR_WS ]]; then 1498 $WHICH_SCM | read scm_type junk || exit 1 1499 fi 1500 1501 case "$scm_type" in 1502 none|git|mercurial) 1503 ;; 1504 *) scm_type=none 1505 ;; 1506 esac 1507 1508 echo $scm_type 1509 } 1510 1511 SCM_TYPE=$(child_wstype) 1512 1513 # 1514 # Decide whether to clobber 1515 # 1516 if [ "$i_FLAG" = "n" -a -d "$SRC" ]; then 1517 echo "\n==== Make clobber at `date` ====\n" >> $LOGFILE 1518 1519 cd $SRC 1520 # remove old clobber file 1521 rm -f $SRC/clobber.out 1522 rm -f $SRC/clobber-${MACH}.out 1523 1524 # Remove all .make.state* files, just in case we are restarting 1525 # the build after having interrupted a previous 'make clobber'. 1526 find . \( -name SCCS -o -name .hg -o -name .svn -o -name .git \ 1527 -o -name 'interfaces.*' \) -prune \ 1528 -o -name '.make.*' -print | xargs rm -f 1529 1530 $MAKE -ek clobber 2>&1 | tee -a $SRC/clobber-${MACH}.out >> $LOGFILE 1531 echo "\n==== Make clobber ERRORS ====\n" >> $mail_msg_file 1532 grep "$MAKE:" $SRC/clobber-${MACH}.out | 1533 egrep -v "Ignoring unknown host" | \ 1534 tee $TMPDIR/clobber_errs >> $mail_msg_file 1535 1536 if [[ -s $TMPDIR/clobber_errs ]]; then 1537 build_extras_ok=n 1538 fi 1539 1540 if [[ "$t_FLAG" = "y" ]]; then 1541 echo "\n==== Make tools clobber at `date` ====\n" >> $LOGFILE 1542 cd ${TOOLS} 1543 rm -f ${TOOLS}/clobber-${MACH}.out 1544 $MAKE TOOLS_PROTO=$TOOLS_PROTO -ek clobber 2>&1 | \ 1545 tee -a ${TOOLS}/clobber-${MACH}.out >> $LOGFILE 1546 echo "\n==== Make tools clobber ERRORS ====\n" \ 1547 >> $mail_msg_file 1548 grep "$MAKE:" ${TOOLS}/clobber-${MACH}.out \ 1549 >> $mail_msg_file 1550 if (( $? == 0 )); then 1551 build_extras_ok=n 1552 fi 1553 rm -rf ${TOOLS_PROTO} 1554 mkdir -p ${TOOLS_PROTO} 1555 fi 1556 1557 typeset roots=$(allprotos) 1558 echo "\n\nClearing $roots" >> "$LOGFILE" 1559 rm -rf $roots 1560 1561 # Get back to a clean workspace as much as possible to catch 1562 # problems that only occur on fresh workspaces. 1563 # Remove all .make.state* files, libraries, and .o's that may 1564 # have been omitted from clobber. A couple of libraries are 1565 # under source code control, so leave them alone. 1566 # We should probably blow away temporary directories too. 1567 cd $SRC 1568 find $relsrcdirs \( -name SCCS -o -name .hg -o -name .svn \ 1569 -o -name .git -o -name 'interfaces.*' \) -prune -o \ 1570 \( -name '.make.*' -o -name 'lib*.a' -o -name 'lib*.so*' -o \ 1571 -name '*.o' \) -print | \ 1572 grep -v 'tools/ctf/dwarf/.*/libdwarf' | xargs rm -f 1573 else 1574 echo "\n==== No clobber at `date` ====\n" >> $LOGFILE 1575 fi 1576 1577 type bringover_mercurial > /dev/null 2>&1 || function bringover_mercurial { 1578 typeset -x PATH=$PATH 1579 1580 # If the repository doesn't exist yet, then we want to populate it. 1581 if [[ ! -d $CODEMGR_WS/.hg ]]; then 1582 staffer hg init $CODEMGR_WS 1583 staffer echo "[paths]" > $CODEMGR_WS/.hg/hgrc 1584 staffer echo "default=$BRINGOVER_WS" >> $CODEMGR_WS/.hg/hgrc 1585 touch $TMPDIR/new_repository 1586 fi 1587 1588 typeset -x HGMERGE="/bin/false" 1589 1590 # 1591 # If the user has changes, regardless of whether those changes are 1592 # committed, and regardless of whether those changes conflict, then 1593 # we'll attempt to merge them either implicitly (uncommitted) or 1594 # explicitly (committed). 1595 # 1596 # These are the messages we'll use to help clarify mercurial output 1597 # in those cases. 1598 # 1599 typeset mergefailmsg="\ 1600 ***\n\ 1601 *** nightly was unable to automatically merge your changes. You should\n\ 1602 *** redo the full merge manually, following the steps outlined by mercurial\n\ 1603 *** above, then restart nightly.\n\ 1604 ***\n" 1605 typeset mergepassmsg="\ 1606 ***\n\ 1607 *** nightly successfully merged your changes. This means that your working\n\ 1608 *** directory has been updated, but those changes are not yet committed.\n\ 1609 *** After nightly completes, you should validate the results of the merge,\n\ 1610 *** then use hg commit manually.\n\ 1611 ***\n" 1612 1613 # 1614 # For each repository in turn: 1615 # 1616 # 1. Do the pull. If this fails, dump the output and bail out. 1617 # 1618 # 2. If the pull resulted in an extra head, do an explicit merge. 1619 # If this fails, dump the output and bail out. 1620 # 1621 # Because we can't rely on Mercurial to exit with a failure code 1622 # when a merge fails (Mercurial issue #186), we must grep the 1623 # output of pull/merge to check for attempted and/or failed merges. 1624 # 1625 # 3. If a merge failed, set the message and fail the bringover. 1626 # 1627 # 4. Otherwise, if a merge succeeded, set the message 1628 # 1629 # 5. Dump the output, and any message from step 3 or 4. 1630 # 1631 1632 typeset HG_SOURCE=$BRINGOVER_WS 1633 if [ ! -f $TMPDIR/new_repository ]; then 1634 HG_SOURCE=$TMPDIR/open_bundle.hg 1635 staffer hg --cwd $CODEMGR_WS incoming --bundle $HG_SOURCE \ 1636 -v $BRINGOVER_WS > $TMPDIR/incoming_open.out 1637 1638 # 1639 # If there are no incoming changesets, then incoming will 1640 # fail, and there will be no bundle file. Reset the source, 1641 # to allow the remaining logic to complete with no false 1642 # negatives. (Unlike incoming, pull will return success 1643 # for the no-change case.) 1644 # 1645 if (( $? != 0 )); then 1646 HG_SOURCE=$BRINGOVER_WS 1647 fi 1648 fi 1649 1650 staffer hg --cwd $CODEMGR_WS pull -u $HG_SOURCE \ 1651 > $TMPDIR/pull_open.out 2>&1 1652 if (( $? != 0 )); then 1653 printf "%s: pull failed as follows:\n\n" "$CODEMGR_WS" 1654 cat $TMPDIR/pull_open.out 1655 if grep "^merging.*failed" $TMPDIR/pull_open.out > /dev/null 2>&1; then 1656 printf "$mergefailmsg" 1657 fi 1658 touch $TMPDIR/bringover_failed 1659 return 1660 fi 1661 1662 if grep "not updating" $TMPDIR/pull_open.out > /dev/null 2>&1; then 1663 staffer hg --cwd $CODEMGR_WS merge \ 1664 >> $TMPDIR/pull_open.out 2>&1 1665 if (( $? != 0 )); then 1666 printf "%s: merge failed as follows:\n\n" \ 1667 "$CODEMGR_WS" 1668 cat $TMPDIR/pull_open.out 1669 if grep "^merging.*failed" $TMPDIR/pull_open.out \ 1670 > /dev/null 2>&1; then 1671 printf "$mergefailmsg" 1672 fi 1673 touch $TMPDIR/bringover_failed 1674 return 1675 fi 1676 fi 1677 1678 printf "updated %s with the following results:\n" "$CODEMGR_WS" 1679 cat $TMPDIR/pull_open.out 1680 if grep "^merging" $TMPDIR/pull_open.out >/dev/null 2>&1; then 1681 printf "$mergepassmsg" 1682 fi 1683 printf "\n" 1684 1685 # 1686 # Per-changeset output is neither useful nor manageable for a 1687 # newly-created repository. 1688 # 1689 if [ -f $TMPDIR/new_repository ]; then 1690 return 1691 fi 1692 1693 printf "\nadded the following changesets to open repository:\n" 1694 cat $TMPDIR/incoming_open.out 1695 } 1696 1697 type bringover_none > /dev/null 2>&1 || function bringover_none { 1698 echo "Couldn't figure out what kind of SCM to use for $BRINGOVER_WS." 1699 touch $TMPDIR/bringover_failed 1700 } 1701 1702 # 1703 # Decide whether to bringover to the codemgr workspace 1704 # 1705 if [ "$n_FLAG" = "n" ]; then 1706 PARENT_SCM_TYPE=$(parent_wstype) 1707 1708 if [[ $SCM_TYPE != none && $SCM_TYPE != $PARENT_SCM_TYPE ]]; then 1709 echo "cannot bringover from $PARENT_SCM_TYPE to $SCM_TYPE, " \ 1710 "quitting at `date`." | tee -a $mail_msg_file >> $LOGFILE 1711 exit 1 1712 fi 1713 1714 run_hook PRE_BRINGOVER 1715 1716 echo "\n==== bringover to $CODEMGR_WS at `date` ====\n" >> $LOGFILE 1717 echo "\n==== BRINGOVER LOG ====\n" >> $mail_msg_file 1718 1719 eval "bringover_${PARENT_SCM_TYPE}" 2>&1 | 1720 tee -a $mail_msg_file >> $LOGFILE 1721 1722 if [ -f $TMPDIR/bringover_failed ]; then 1723 rm -f $TMPDIR/bringover_failed 1724 build_ok=n 1725 echo "trouble with bringover, quitting at `date`." | 1726 tee -a $mail_msg_file >> $LOGFILE 1727 exit 1 1728 fi 1729 1730 # 1731 # It's possible that we used the bringover above to create 1732 # $CODEMGR_WS. If so, then SCM_TYPE was previously "none," 1733 # but should now be the same as $BRINGOVER_WS. 1734 # 1735 [[ $SCM_TYPE = none ]] && SCM_TYPE=$PARENT_SCM_TYPE 1736 1737 run_hook POST_BRINGOVER 1738 1739 check_closed_bins 1740 1741 else 1742 echo "\n==== No bringover to $CODEMGR_WS ====\n" >> $LOGFILE 1743 fi 1744 1745 # Safeguards 1746 [[ -v CODEMGR_WS ]] || fatal_error "Error: Variable CODEMGR_WS not set." 1747 [[ -d "${CODEMGR_WS}" ]] || fatal_error "Error: ${CODEMGR_WS} is not a directory." 1748 [[ -f "${CODEMGR_WS}/usr/src/Makefile" ]] || fatal_error "Error: ${CODEMGR_WS}/usr/src/Makefile not found." 1749 1750 echo "\n==== Build environment ====\n" | tee -a $build_environ_file >> $LOGFILE 1751 1752 # System 1753 whence uname | tee -a $build_environ_file >> $LOGFILE 1754 uname -a 2>&1 | tee -a $build_environ_file >> $LOGFILE 1755 echo | tee -a $build_environ_file >> $LOGFILE 1756 1757 # make 1758 whence $MAKE | tee -a $build_environ_file >> $LOGFILE 1759 $MAKE -v | tee -a $build_environ_file >> $LOGFILE 1760 echo "number of concurrent jobs = $DMAKE_MAX_JOBS" | 1761 tee -a $build_environ_file >> $LOGFILE 1762 1763 # 1764 # Report the compiler versions. 1765 # 1766 1767 if [[ ! -f $SRC/Makefile ]]; then 1768 build_ok=n 1769 echo "\nUnable to find \"Makefile\" in $SRC." | \ 1770 tee -a $build_environ_file >> $LOGFILE 1771 exit 1 1772 fi 1773 1774 ( cd $SRC 1775 for target in cc-version cc64-version java-version; do 1776 echo 1777 # 1778 # Put statefile somewhere we know we can write to rather than trip 1779 # over a read-only $srcroot. 1780 # 1781 rm -f $TMPDIR/make-state 1782 export SRC 1783 if $MAKE -K $TMPDIR/make-state -e $target 2>/dev/null; then 1784 continue 1785 fi 1786 touch $TMPDIR/nocompiler 1787 done 1788 echo 1789 ) | tee -a $build_environ_file >> $LOGFILE 1790 1791 if [ -f $TMPDIR/nocompiler ]; then 1792 rm -f $TMPDIR/nocompiler 1793 build_ok=n 1794 echo "Aborting due to missing compiler." | 1795 tee -a $build_environ_file >> $LOGFILE 1796 exit 1 1797 fi 1798 1799 # as 1800 whence as | tee -a $build_environ_file >> $LOGFILE 1801 as -V 2>&1 | head -1 | tee -a $build_environ_file >> $LOGFILE 1802 echo | tee -a $build_environ_file >> $LOGFILE 1803 1804 # Check that we're running a capable link-editor 1805 whence ld | tee -a $build_environ_file >> $LOGFILE 1806 LDVER=`ld -V 2>&1` 1807 echo $LDVER | tee -a $build_environ_file >> $LOGFILE 1808 LDVER=`echo $LDVER | sed -e "s/.*-1\.\([0-9]*\).*/\1/"` 1809 if [ `expr $LDVER \< 422` -eq 1 ]; then 1810 echo "The link-editor needs to be at version 422 or higher to build" | \ 1811 tee -a $build_environ_file >> $LOGFILE 1812 echo "the latest stuff. Hope your build works." | \ 1813 tee -a $build_environ_file >> $LOGFILE 1814 fi 1815 1816 # 1817 # Build and use the workspace's tools if requested 1818 # 1819 if [[ "$t_FLAG" = "y" ]]; then 1820 set_non_debug_build_flags 1821 1822 build_tools ${TOOLS_PROTO} 1823 if (( $? != 0 )); then 1824 build_ok=n 1825 else 1826 use_tools $TOOLS_PROTO 1827 fi 1828 fi 1829 1830 # timestamp the start of the normal build; the findunref tool uses it. 1831 touch $SRC/.build.tstamp 1832 1833 normal_build 1834 1835 ORIG_SRC=$SRC 1836 BINARCHIVE=${CODEMGR_WS}/bin-${MACH}.cpio.Z 1837 1838 1839 # 1840 # There are several checks that need to look at the proto area, but 1841 # they only need to look at one, and they don't care whether it's 1842 # DEBUG or non-DEBUG. 1843 # 1844 if [[ "$MULTI_PROTO" = yes && "$D_FLAG" = n ]]; then 1845 checkroot=$ROOT-nd 1846 else 1847 checkroot=$ROOT 1848 fi 1849 1850 if [ "$build_ok" = "y" ]; then 1851 echo "\n==== Creating protolist system file at `date` ====" \ 1852 >> $LOGFILE 1853 protolist $checkroot > $ATLOG/proto_list_${MACH} 1854 echo "==== protolist system file created at `date` ====\n" \ 1855 >> $LOGFILE 1856 1857 if [ "$N_FLAG" != "y" ]; then 1858 1859 E1= 1860 f1= 1861 for f in $f1; do 1862 if [ -f "$f" ]; then 1863 E1="$E1 -e $f" 1864 fi 1865 done 1866 1867 E2= 1868 f2= 1869 if [ -d "$SRC/pkg" ]; then 1870 f2="$f2 exceptions/packaging" 1871 fi 1872 1873 for f in $f2; do 1874 if [ -f "$f" ]; then 1875 E2="$E2 -e $f" 1876 fi 1877 done 1878 fi 1879 1880 if [ "$N_FLAG" != "y" -a -d $SRC/pkg ]; then 1881 echo "\n==== Validating manifests against proto area ====\n" \ 1882 >> $mail_msg_file 1883 ( cd $SRC/pkg ; $MAKE -e protocmp ROOT="$checkroot" ) | \ 1884 tee $TMPDIR/protocmp_noise >> $mail_msg_file 1885 if [[ -s $TMPDIR/protocmp_noise ]]; then 1886 build_extras_ok=n 1887 fi 1888 fi 1889 1890 if [ "$N_FLAG" != "y" -a -f "$REF_PROTO_LIST" ]; then 1891 echo "\n==== Impact on proto area ====\n" >> $mail_msg_file 1892 if [ -n "$E2" ]; then 1893 ELIST=$E2 1894 else 1895 ELIST=$E1 1896 fi 1897 $PROTOCMPTERSE \ 1898 "Files in yesterday's proto area, but not today's:" \ 1899 "Files in today's proto area, but not yesterday's:" \ 1900 "Files that changed between yesterday and today:" \ 1901 ${ELIST} \ 1902 -d $REF_PROTO_LIST \ 1903 $ATLOG/proto_list_${MACH} \ 1904 >> $mail_msg_file 1905 fi 1906 fi 1907 1908 if [[ "$u_FLAG" == "y" && "$build_ok" == "y" && \ 1909 "$build_extras_ok" == "y" ]]; then 1910 staffer cp $ATLOG/proto_list_${MACH} \ 1911 $PARENT_WS/usr/src/proto_list_${MACH} 1912 fi 1913 1914 # Update parent proto area if necessary. This is done now 1915 # so that the proto area has either DEBUG or non-DEBUG kernels. 1916 # Note that this clears out the lock file, so we can dispense with 1917 # the variable now. 1918 if [ "$U_FLAG" = "y" -a "$build_ok" = "y" ]; then 1919 echo "\n==== Copying proto area to $NIGHTLY_PARENT_ROOT ====\n" | \ 1920 tee -a $LOGFILE >> $mail_msg_file 1921 rm -rf $NIGHTLY_PARENT_ROOT/* 1922 unset Ulockfile 1923 mkdir -p $NIGHTLY_PARENT_ROOT 1924 if [[ "$MULTI_PROTO" = no || "$D_FLAG" = y ]]; then 1925 ( cd $ROOT; tar cf - . | 1926 ( cd $NIGHTLY_PARENT_ROOT; umask 0; tar xpf - ) ) 2>&1 | 1927 tee -a $mail_msg_file >> $LOGFILE 1928 fi 1929 if [[ "$MULTI_PROTO" = yes && "$F_FLAG" = n ]]; then 1930 rm -rf $NIGHTLY_PARENT_ROOT-nd/* 1931 mkdir -p $NIGHTLY_PARENT_ROOT-nd 1932 cd $ROOT-nd 1933 ( tar cf - . | 1934 ( cd $NIGHTLY_PARENT_ROOT-nd; umask 0; tar xpf - ) ) 2>&1 | 1935 tee -a $mail_msg_file >> $LOGFILE 1936 fi 1937 if [ -n "${NIGHTLY_PARENT_TOOLS_ROOT}" ]; then 1938 echo "\n==== Copying tools proto area to $NIGHTLY_PARENT_TOOLS_ROOT ====\n" | \ 1939 tee -a $LOGFILE >> $mail_msg_file 1940 rm -rf $NIGHTLY_PARENT_TOOLS_ROOT/* 1941 mkdir -p $NIGHTLY_PARENT_TOOLS_ROOT 1942 if [[ "$MULTI_PROTO" = no || "$D_FLAG" = y ]]; then 1943 ( cd $TOOLS_PROTO; tar cf - . | 1944 ( cd $NIGHTLY_PARENT_TOOLS_ROOT; 1945 umask 0; tar xpf - ) ) 2>&1 | 1946 tee -a $mail_msg_file >> $LOGFILE 1947 fi 1948 fi 1949 fi 1950 1951 # 1952 # ELF verification: ABI (-A) and runtime (-r) checks 1953 # 1954 if [[ ($build_ok = y) && (($A_FLAG = y) || ($r_FLAG = y)) ]]; then 1955 # Directory ELF-data.$MACH holds the files produced by these tests. 1956 elf_ddir=$SRC/ELF-data.$MACH 1957 1958 # If there is a previous ELF-data backup directory, remove it. Then, 1959 # rotate current ELF-data directory into its place and create a new 1960 # empty directory 1961 rm -rf $elf_ddir.ref 1962 if [[ -d $elf_ddir ]]; then 1963 mv $elf_ddir $elf_ddir.ref 1964 fi 1965 mkdir -p $elf_ddir 1966 1967 # Call find_elf to produce a list of the ELF objects in the proto area. 1968 # This list is passed to check_rtime and interface_check, preventing 1969 # them from separately calling find_elf to do the same work twice. 1970 find_elf -fr $checkroot > $elf_ddir/object_list 1971 1972 if [[ $A_FLAG = y ]]; then 1973 echo "\n==== Check versioning and ABI information ====\n" | \ 1974 tee -a $LOGFILE >> $mail_msg_file 1975 1976 # Produce interface description for the proto. Report errors. 1977 interface_check -o -w $elf_ddir -f object_list \ 1978 -i interface -E interface.err 1979 if [[ -s $elf_ddir/interface.err ]]; then 1980 tee -a $LOGFILE < $elf_ddir/interface.err \ 1981 >> $mail_msg_file 1982 build_extras_ok=n 1983 fi 1984 1985 # If ELF_DATA_BASELINE_DIR is defined, compare the new interface 1986 # description file to that from the baseline gate. Issue a 1987 # warning if the baseline is not present, and keep going. 1988 if [[ "$ELF_DATA_BASELINE_DIR" != '' ]]; then 1989 base_ifile="$ELF_DATA_BASELINE_DIR/interface" 1990 1991 echo "\n==== Compare versioning and ABI information" \ 1992 "to baseline ====\n" | \ 1993 tee -a $LOGFILE >> $mail_msg_file 1994 echo "Baseline: $base_ifile\n" >> $LOGFILE 1995 1996 if [[ -f $base_ifile ]]; then 1997 interface_cmp -d -o $base_ifile \ 1998 $elf_ddir/interface > $elf_ddir/interface.cmp 1999 if [[ -s $elf_ddir/interface.cmp ]]; then 2000 echo | tee -a $LOGFILE >> $mail_msg_file 2001 tee -a $LOGFILE < \ 2002 $elf_ddir/interface.cmp \ 2003 >> $mail_msg_file 2004 build_extras_ok=n 2005 fi 2006 else 2007 echo "baseline not available. comparison" \ 2008 "skipped" | \ 2009 tee -a $LOGFILE >> $mail_msg_file 2010 fi 2011 2012 fi 2013 fi 2014 2015 if [[ $r_FLAG = y ]]; then 2016 echo "\n==== Check ELF runtime attributes ====\n" | \ 2017 tee -a $LOGFILE >> $mail_msg_file 2018 2019 # If we're doing a DEBUG build the proto area will be left 2020 # with debuggable objects, thus don't assert -s. 2021 if [[ $D_FLAG = y ]]; then 2022 rtime_sflag="" 2023 else 2024 rtime_sflag="-s" 2025 fi 2026 check_rtime -i -m -v $rtime_sflag -o -w $elf_ddir \ 2027 -D object_list -f object_list -E runtime.err \ 2028 -I runtime.attr.raw 2029 if (( $? != 0 )); then 2030 build_extras_ok=n 2031 fi 2032 2033 # check_rtime -I output needs to be sorted in order to 2034 # compare it to that from previous builds. 2035 sort $elf_ddir/runtime.attr.raw > $elf_ddir/runtime.attr 2036 rm $elf_ddir/runtime.attr.raw 2037 2038 # Report errors 2039 if [[ -s $elf_ddir/runtime.err ]]; then 2040 tee -a $LOGFILE < $elf_ddir/runtime.err \ 2041 >> $mail_msg_file 2042 build_extras_ok=n 2043 fi 2044 2045 # If there is an ELF-data directory from a previous build, 2046 # then diff the attr files. These files contain information 2047 # about dependencies, versioning, and runpaths. There is some 2048 # overlap with the ABI checking done above, but this also 2049 # flushes out non-ABI interface differences along with the 2050 # other information. 2051 echo "\n==== Diff ELF runtime attributes" \ 2052 "(since last build) ====\n" | \ 2053 tee -a $LOGFILE >> $mail_msg_file >> $mail_msg_file 2054 2055 if [[ -f $elf_ddir.ref/runtime.attr ]]; then 2056 diff $elf_ddir.ref/runtime.attr \ 2057 $elf_ddir/runtime.attr \ 2058 >> $mail_msg_file 2059 fi 2060 fi 2061 2062 # If -u set, copy contents of ELF-data.$MACH to the parent workspace. 2063 if [[ "$u_FLAG" = "y" ]]; then 2064 p_elf_ddir=$PARENT_WS/usr/src/ELF-data.$MACH 2065 2066 # If parent lacks the ELF-data.$MACH directory, create it 2067 if [[ ! -d $p_elf_ddir ]]; then 2068 staffer mkdir -p $p_elf_ddir 2069 fi 2070 2071 # These files are used asynchronously by other builds for ABI 2072 # verification, as above for the -A option. As such, we require 2073 # the file replacement to be atomic. Copy the data to a temp 2074 # file in the same filesystem and then rename into place. 2075 ( 2076 cd $elf_ddir 2077 for elf_dfile in *; do 2078 staffer cp $elf_dfile \ 2079 ${p_elf_ddir}/${elf_dfile}.new 2080 staffer mv -f ${p_elf_ddir}/${elf_dfile}.new \ 2081 ${p_elf_ddir}/${elf_dfile} 2082 done 2083 ) 2084 fi 2085 fi 2086 2087 # DEBUG lint of kernel begins 2088 2089 if [ "$i_CMD_LINE_FLAG" = "n" -a "$l_FLAG" = "y" ]; then 2090 if [ "$LINTDIRS" = "" ]; then 2091 # LINTDIRS="$SRC/uts y $SRC/stand y $SRC/psm y" 2092 LINTDIRS="$SRC y" 2093 fi 2094 set $LINTDIRS 2095 while [ $# -gt 0 ]; do 2096 dolint $1 $2; shift; shift 2097 done 2098 else 2099 echo "\n==== No '$MAKE lint' ====\n" >> $LOGFILE 2100 fi 2101 2102 # "make check" begins 2103 2104 if [ "$i_CMD_LINE_FLAG" = "n" -a "$C_FLAG" = "y" ]; then 2105 # remove old check.out 2106 rm -f $SRC/check.out 2107 2108 rm -f $SRC/check-${MACH}.out 2109 cd $SRC 2110 $MAKE -ek check ROOT="$checkroot" 2>&1 | tee -a $SRC/check-${MACH}.out \ 2111 >> $LOGFILE 2112 echo "\n==== cstyle/hdrchk errors ====\n" >> $mail_msg_file 2113 2114 grep ":" $SRC/check-${MACH}.out | 2115 egrep -v "Ignoring unknown host" | \ 2116 sort | uniq | tee $TMPDIR/check_errors >> $mail_msg_file 2117 2118 if [[ -s $TMPDIR/check_errors ]]; then 2119 build_extras_ok=n 2120 fi 2121 else 2122 echo "\n==== No '$MAKE check' ====\n" >> $LOGFILE 2123 fi 2124 2125 echo "\n==== Find core files ====\n" | \ 2126 tee -a $LOGFILE >> $mail_msg_file 2127 2128 find $abssrcdirs -name core -a -type f -exec file {} \; | \ 2129 tee -a $LOGFILE >> $mail_msg_file 2130 2131 if [ "$f_FLAG" = "y" -a "$build_ok" = "y" ]; then 2132 echo "\n==== Diff unreferenced files (since last build) ====\n" \ 2133 | tee -a $LOGFILE >>$mail_msg_file 2134 rm -f $SRC/unref-${MACH}.ref 2135 if [ -f $SRC/unref-${MACH}.out ]; then 2136 mv $SRC/unref-${MACH}.out $SRC/unref-${MACH}.ref 2137 fi 2138 2139 findunref -S $SCM_TYPE -t $SRC/.build.tstamp -s usr $CODEMGR_WS \ 2140 ${TOOLS}/findunref/exception_list 2>> $mail_msg_file | \ 2141 sort > $SRC/unref-${MACH}.out 2142 2143 if [ ! -f $SRC/unref-${MACH}.ref ]; then 2144 cp $SRC/unref-${MACH}.out $SRC/unref-${MACH}.ref 2145 fi 2146 2147 diff $SRC/unref-${MACH}.ref $SRC/unref-${MACH}.out >>$mail_msg_file 2148 fi 2149 2150 # Verify that the usual lists of files, such as exception lists, 2151 # contain only valid references to files. If the build has failed, 2152 # then don't check the proto area. 2153 CHECK_PATHS=${CHECK_PATHS:-y} 2154 if [ "$CHECK_PATHS" = y -a "$N_FLAG" != y ]; then 2155 echo "\n==== Check lists of files ====\n" | tee -a $LOGFILE \ 2156 >>$mail_msg_file 2157 arg=-b 2158 [ "$build_ok" = y ] && arg= 2159 checkpaths $arg $checkroot > $SRC/checkpaths.out 2>&1 2160 if [[ -s $SRC/checkpaths.out ]]; then 2161 tee -a $LOGFILE < $SRC/checkpaths.out >> $mail_msg_file 2162 build_extras_ok=n 2163 fi 2164 fi 2165 2166 if [ "$M_FLAG" != "y" -a "$build_ok" = y ]; then 2167 echo "\n==== Impact on file permissions ====\n" \ 2168 >> $mail_msg_file 2169 2170 abspkg= 2171 for d in $abssrcdirs; do 2172 if [ -d "$d/pkg" ]; then 2173 abspkg="$abspkg $d" 2174 fi 2175 done 2176 2177 if [ -n "$abspkg" ]; then 2178 for d in "$abspkg"; do 2179 ( cd $d/pkg ; $MAKE -e pmodes ) >> $mail_msg_file 2180 done 2181 fi 2182 fi 2183 2184 if [ "$w_FLAG" = "y" -a "$build_ok" = "y" ]; then 2185 if [[ "$MULTI_PROTO" = no || "$D_FLAG" = y ]]; then 2186 do_wsdiff DEBUG $ROOT.prev $ROOT 2187 fi 2188 2189 if [[ "$MULTI_PROTO" = yes && "$F_FLAG" = n ]]; then 2190 do_wsdiff non-DEBUG $ROOT-nd.prev $ROOT-nd 2191 fi 2192 fi 2193 2194 END_DATE=`date` 2195 echo "==== Nightly $maketype build completed: $END_DATE ====" | \ 2196 tee -a $LOGFILE >> $build_time_file 2197 2198 typeset -i10 hours 2199 typeset -Z2 minutes 2200 typeset -Z2 seconds 2201 2202 elapsed_time=$SECONDS 2203 ((hours = elapsed_time / 3600 )) 2204 ((minutes = elapsed_time / 60 % 60)) 2205 ((seconds = elapsed_time % 60)) 2206 2207 echo "\n==== Total build time ====" | \ 2208 tee -a $LOGFILE >> $build_time_file 2209 echo "\nreal ${hours}:${minutes}:${seconds}" | \ 2210 tee -a $LOGFILE >> $build_time_file 2211 2212 if [ "$u_FLAG" = "y" -a "$f_FLAG" = "y" -a "$build_ok" = "y" ]; then 2213 staffer cp ${SRC}/unref-${MACH}.out $PARENT_WS/usr/src/ 2214 2215 # 2216 # Produce a master list of unreferenced files -- ideally, we'd 2217 # generate the master just once after all of the nightlies 2218 # have finished, but there's no simple way to know when that 2219 # will be. Instead, we assume that we're the last nightly to 2220 # finish and merge all of the unref-${MACH}.out files in 2221 # $PARENT_WS/usr/src/. If we are in fact the final ${MACH} to 2222 # finish, then this file will be the authoritative master 2223 # list. Otherwise, another ${MACH}'s nightly will eventually 2224 # overwrite ours with its own master, but in the meantime our 2225 # temporary "master" will be no worse than any older master 2226 # which was already on the parent. 2227 # 2228 2229 set -- $PARENT_WS/usr/src/unref-*.out 2230 cp "$1" ${TMPDIR}/unref.merge 2231 shift 2232 2233 for unreffile; do 2234 comm -12 ${TMPDIR}/unref.merge "$unreffile" > ${TMPDIR}/unref.$$ 2235 mv ${TMPDIR}/unref.$$ ${TMPDIR}/unref.merge 2236 done 2237 2238 staffer cp ${TMPDIR}/unref.merge $PARENT_WS/usr/src/unrefmaster.out 2239 fi 2240 2241 # 2242 # All done save for the sweeping up. 2243 # (whichever exit we hit here will trigger the "cleanup" trap which 2244 # optionally sends mail on completion). 2245 # 2246 if [[ "$build_ok" == "y" ]]; then 2247 if [[ "$W_FLAG" == "y" || "$build_extras_ok" == "y" ]]; then 2248 exit 0 2249 fi 2250 fi 2251 2252 exit 1