Whamcloud - gitweb
3f4e5cdd6d85b1bcb31cbdba54fae7149b1068cd
[fs/lustre-release.git] / build / lbuild
1 #!/bin/bash
2
3 # vim:expandtab:shiftwidth=4:softtabstop=4:tabstop=4:
4
5 # this is an alternative FD for stdout, to be used especially when we are
6 # taking stdout from a function as it's return value.  i.e. foo=$(bar)
7 # this is a workaround until a version of bash where we can put xtrace
8 # on a specific FD
9 exec 3>&1; STDOUT=3
10
11 #set -x
12 xtrace="+x"
13 if [[ $SHELLOPTS = *xtrace* ]]; then
14     xtrace="-x"
15 fi
16 shopt -s extdebug
17
18 # include the exit_traps library
19 . ${0%/lbuild}/exit_traps.sh
20
21 # our children should die when we do
22 push_exit_trap "kill -INT -$$ || true" kill_children
23
24 # increment this if you have made a change that should force a new kernel
25 # to build built
26 #BUILD_GEN=1
27 #BUILD_GEN=2    # bz19952: remove -lustre tag from kernel RPM names
28 #BUILD_GEN=3    # bz19975: enable the building of src.rpms by default
29 BUILD_GEN=4     # bz22281: use the git hash in the kernel extra version
30
31 TOPDIR=$PWD
32
33 # CVSROOT is inherited from the environment
34 KERNELDIR=
35 LINUX=
36 LUSTRE=
37 RELEASE=false
38 # XXX - some recent hacking has pretty much neutered this option.
39 #       search through this file (and lbuild.old_school -- but that will
40 #       be going away soon) for "-bb" and see how many places
41 #       simply don't account for this option
42 DO_SRC=true
43 DOWNLOAD=true
44 TAG=
45 CANONICAL_TARGET=
46 TARGET=
47 TARGET_ARCH=$(uname -m)
48 TARGET_ARCHS=
49 TARGET_ARCHS_ALL=$TARGET_ARCH
50 [ "$TARGET_ARCH" = "i686" ] && TARGET_ARCHS_ALL="i686 i586 i386"
51 CONFIGURE_FLAGS=
52 EXTERNAL_PATCHES=
53 EXTRA_VERSION=
54 STAGEDIR=
55 TMPDIR=${TMPDIR:-"/var/tmp"}
56 TIMESTAMP=
57 # this is a dir to try reuse old kernel RPMs in (although, it seems to be
58 # unused in any real manner
59 REUSERPM=
60 # this is the dir that should be used to store reuse products
61 REUSEBUILD=
62 # should cached products be used or force rebuilding?
63 USE_BUILD_CACHE=true
64 # what does this do exactly?  does it imply no kernel build?
65 NORPM=false
66 LDISKFSRPM=true
67 SKIPLDISKFSRPM="v1_4_* b1_4"
68 SMPTYPES="smp bigsmp default ''"
69 PATCHLESS=false
70 XEN=false
71 LINUXOBJ=
72 DISTRO=
73 KERNELTREE=
74 # default to not adding -lustre- into the kernel RPM package names
75 KERNEL_LUSTRE_NAMING=false
76
77 # patchless build
78 KERNELRPMSBASE=
79 RPMSMPTYPE=
80
81 # from target file
82 SERIES=
83 BASE_ARCHS=
84 BIGMEM_ARCHS=
85 BOOT_ARCHS=
86 JENSEN_ARCHS=
87 SMP_ARCHS=
88 BIGSMP_ARCHS=
89 PSERIES64_ARCHS=
90 UP_ARCHS=
91
92 # not in the target file any more
93 CONFIG=
94
95 # build the lustre-tests rpm?
96 LUSTRE_TESTS=true
97
98 DATE=$(date)
99
100 USE_DATESTAMP=1
101 RPMBUILD=
102
103 OLD_SCHOOL=false
104
105 export CC=${CC:-gcc}
106
107 # Readlink is not present on some older distributions: emulate it.
108 readlink() {
109     local path=$1 ll
110
111     if [ -L "$path" ]; then
112         ll="$(LC_ALL=C ls -l "$path" 2> /dev/null)" &&
113         echo "${ll/* -> }"
114     else
115         return 1
116     fi
117 }
118
119 cleanup() {
120
121     true
122 }
123
124 error() {
125     local msg="$1"
126
127     [ -n "$msg" ] && echo -e "\n${0##*/}: $msg" >&3
128
129 }
130
131 fatal() {
132
133     cleanup
134     error "$2"
135     exit $1
136
137 }
138
139 usage() {
140     cat <<EOF
141 Usage: ${0##*/} [OPTION]... [-- <lustre configure options>]
142
143   -d CVSROOT
144     Specifies the CVS Root to use when pulling files from CVS.  The
145     environment variable \$CVSROOT is used if this option is not
146     present.
147
148   --external-patches=EXTERNAL_PATCHES
149     Directory similar to lustre/lustre/kernel_patches/ that lbuild should
150     look for seres and config files in before looking in the lustre
151     tree.
152
153   --extraversion=EXTRAVERSION
154     Text to use for the rpm release and kernel extraversion.
155
156   --timestamp=TIMESTAMP
157     Date of building lustre in format YYYYMMDDhhmmss
158
159   --reuserpm=DIR
160     Try to reuse old kernel RPMs from DIR
161
162   --reusebuild=DIR
163     Try to reuse old kernel builds from DIR
164
165   --kernelrpm=DIR
166     Path to distro kernel RPM collection
167
168   --ccache
169     Use ccache
170
171   --norpm
172     Do not build RPMs (compile only mode)
173
174   --patchless
175     Build lustre client only
176
177   --distro=DISTRO
178     Which distro using. Autodetect by default
179
180   --kerneldir=KERNELDIR
181     Directory containing Linux source tarballs referenced by target
182     files.
183
184   --kerneltree=KERNELTREE
185     Directory containing dirs with Linux source tarballs referenced by target
186     files. Dir names in format kernel version ('2.6.9', etc.)
187
188   --linux=LINUX --with-linux=LINUX
189     Directory of Linux kernel sources.  When this option is used, only
190     Lustre modules and userspace are built.
191
192   --lustre=LUSTRE
193     Path to an existing lustre source tarball to use instead of
194     pulling from CVS.
195
196   --nodownload
197     Do not try to download a kernel from downloads.lustre.org
198
199   --nosrc
200     Do not build a .src.rpm, a full kernel patch, or a patched kernel
201     tarball.
202
203   --ldiskfs
204     Do ldiskfs RPM. Now true by default
205
206   --publish
207     Unused.
208
209   --release
210     Specifies that the files generated do not include timestamps, and
211     that this is an official release.
212
213   --src
214     Build a .src.rpm, a full kernel patch, and a patched kernel tarball.
215
216   --stage=DIR
217     Directory used to stage packages for release.  RPMs will be placed
218     more or less in DIR/<target>-<arch>, and the tarball will be
219     placed in DIR.
220
221   --tag=TAG
222     A CVS branch/tag name to build from when pulling from CVS.
223
224   --target=TARGET
225     The name of the target to build.  The available targets are listed
226     below.
227
228   --target-archs=TARGET_ARCHS
229     A (space delimited) list of architectures to build.  By default,
230     all of the archs supported by the TARGET will be built, in
231     addition to a .src.rpm.  This option can limit those, for machines
232     that can only build certain archs or if you only want a certain
233     arch built (for testing, or a one-off kernel).
234
235     Also note that by using a non-"base" arch (eg, i386) only kernels
236     will be built - there will be no lustre-lite-utils package.
237
238   --disable-datestamp
239     Prevents the datestamp flag (-D) from being passed to cvs for
240     checkouts. This is a workaround for a problem encountered when
241     using lbuild with tinderbox.
242
243   --xen
244     Builds a Xen domX kernel.
245
246   --set-value
247     Set's a variable to a given value.
248
249 EOF
250
251 #   list_targets
252
253     fatal "$1" "$2"
254 }
255
256 # canonicalize a relative path to a file
257 canon_filepath() {
258     local PATH="$1"
259
260     if [ ! -f "$PATH" ]; then
261         return 1
262     fi
263
264     local FILE=${PATH##*/}
265     local DIR=${PATH%/*}
266
267     echo $(canon_path "$DIR")/$FILE
268     return 0
269 }
270
271 # canonicalize a relative path to a dir
272 canon_path() {
273     local PATH="$1"
274
275     if [ ! -d "$PATH" ]; then
276         return 1
277     fi
278
279     pushd "$PATH" >/dev/null || return 1
280     local CANONPATH=$PWD
281     popd >/dev/null
282
283     echo "$CANONPATH"
284     return 0
285 }
286
287 check_options() {
288
289     if [ "$LUSTRE" ]; then
290         [ -r "$LUSTRE" ] || \
291             usage 1 "Could not find Lustre source tarball '$LUSTRE'."
292     else
293         [ "$CVSROOT" ] || \
294             usage 1 "Either specify a CVS Root with -d, or a Lustre source tarball with --lustre."
295         [ "$TAG" ] || \
296             usage 1 "A branch/tag name must be specified with --tag when not building from a tarball."
297     fi
298
299     if [ -z "$LINUX" ]; then
300         [ "$KERNELDIR" -o "$KERNELTREE" ] || \
301             usage 1 "A kernel directory must be specified with --kerneldir or --kerneltree."
302
303         [ -d "$KERNELDIR" -o -d "$KERNELTREE" ] || \
304             usage 1 "$KERNELDIR and $KERNELTREE are not a directory."
305
306         if ! $RELEASE; then
307             [ "$TAG" ] || \
308                 usage 1 "When building a snapshot, a tag name must be used."
309         fi
310
311         [ "$TARGET" ] || usage 1 "A target must be specified with --target."
312 #       TARGET_FILE="$TOPDIR/lustre/kernel_patches/targets/$TARGET.target"
313 #       [ -r "$TARGET_FILE" ] || \
314 #               usage 1 "Target '$TARGET' was not found."
315     fi
316
317     case $TARGET in
318         2.6-rhel5)
319             CANONICAL_TARGET="rhel5"
320             ;;
321         2.6-rhel4)
322             CANONICAL_TARGET="rhel-2.6"
323             ;;
324         2.6-suse)
325             CANONICAL_TARGET="sles-2.6"
326             ;;
327         2.6-sles10)
328             CANONICAL_TARGET="sles10-2.6"
329             ;;
330         2.6-sles11)
331             CANONICAL_TARGET="sles11"
332             ;;
333         2.6-oel5)
334             CANONICAL_TARGET="oel5"
335             ;;
336         hp_pnnl-2.4)
337             CANONICAL_TARGET="hp-pnnl-2.4"
338             ;;
339         2.6-vanilla \
340             | suse-2.4.21-2 \
341             | rh-2.4 \
342             | rhel-2.4 \
343             | sles-2.4 \
344             | 2.6-patchless)
345                 CANONICAL_TARGET="$TARGET"
346                 ;;
347     esac
348
349     local timestampnodig=$(echo $TIMESTAMP | sed -e s/[0-9]*//g)
350     [ "$timestampnodig" = "" ] || TIMESTAMP=$(date -d "$DATE" "+%Y%m%d%H%M%S")
351     local timestamplength="${#TIMESTAMP}"
352     if [ $timestamplength -eq 12 ]; then
353         TIMESTAMP="${TIMESTAMP}00"
354     elif [ $timestamplength -ne 14 ]; then
355         TIMESTAMP=$(date -d "$DATE" "+%Y%m%d%H%M%S")
356     fi
357
358     RPMBUILD=$(which rpmbuild 2>/dev/null | head -1)
359     if [ ! "$RPMBUILD" -o "$RPMBUILD" == "" ]; then
360         RPMBUILD=$(which rpm 2>/dev/null | head -1)
361         if [ ! "$RPMBUILD" -o "$RPMBUILD" == "" ]; then
362             usage 1 "Could not find binary for making rpms (tried rpmbuild and rpm)."
363         fi
364     fi
365
366     if [ -n "$CCACHE" ]; then
367         which "$DISTCC" &>/dev/null && export DISTCC RPM_BUILD_NCPUS
368
369         if which "$CCACHE" &>/dev/null; then
370             local ccache=$(which "$CCACHE")
371             local bindir="$TOPDIR/bin"
372
373             if [ ! -d $bindir ]; then
374                 mkdir -p $bindir || fatal 1 "error trying to create $bindir"
375             else
376                 rm ${bindir}/* > /dev/null 2>&1 || true
377             fi
378             ln -s "$ccache" ${bindir}/ccache
379             ln -s "$ccache" ${bindir}/cc
380             ln -s "$ccache" ${bindir}/$CC
381             export PATH=$bindir:$PATH
382             export CCACHE && export CC="ccache $CC"
383             # zero the cache so we can see how effective we are being with it
384             echo -n "ccache "
385             ccache -z
386
387             # get some ccache stats when we are done
388             push_exit_trap '[ -n "$CCACHE" ] && ccache -s' "ccache_summary"
389             # should remove the ccache trap if lbuild is interrupted
390             trap 'echo "Received an INT TERM or HUP signal, terminating."; delete_exit_trap "ccache_summary"; exit 1' INT TERM HUP
391         fi
392     fi
393
394     [ -z "$DISTRO" ] && DISTRO=$(autodetect_distro)
395
396     return 0
397
398 }
399
400 # autodetect used Distro
401 autodetect_distro() {
402
403     local name
404     local version
405
406     if [ -f /etc/SuSE-release ]; then
407         name=sles
408         version=$(grep ^VERSION /etc/SuSE-release)
409         version=${version#*= }
410     elif [ -f /etc/redhat-release ]; then
411         name=$(head -1 /etc/redhat-release)
412         version=$(echo "$distroname" |
413                   sed -e 's/^[^0-9.]*//g' | sed -e 's/[ \.].*//')
414     fi
415     if [ -z "$name" -o -z "$version" ]; then
416         fatal 1 "I don't know how to determine distro type/version.\nEither update autodetect_distro() or use the --distro argument"
417     fi
418
419     echo ${name}${version}
420     return 0
421
422 }
423
424 uniqify() {
425
426     echo $(echo "$*" | xargs -n 1 | sort -u)
427
428 }
429
430 download_srpm() {
431     local target=$1
432     local srpm=$2
433     local force="${3:-false}"
434
435     if $force || [ ! -r "$KERNELDIR/$srpm" ] ||
436        [ ! -s "$KERNELDIR/$srpm" ]; then
437         if $DOWNLOAD; then
438             local location="http://downloads.lustre.org/public/kernels/$target/old"
439             echo "Downloading $location/$srpm..."
440             if ! wget -nv "$location/$srpm" -O "$KERNELDIR/$srpm" 2>&1 ||
441                [ ! -s "$KERNELDIR/$srpm" ]; then
442                 rm -f $KERNELDIR/$srpm
443                 fatal 1 "Could not download target $target's kernel SRPM $srpm from $location."
444             fi
445         else
446             fatal 1 "$srpm not found in directory $KERNELDIR."
447         fi
448     fi
449
450 }
451
452 download_file() {
453     local from="$1"
454     local to="$2"
455     local force="$3"
456
457     local file=${from##*/}
458
459     if [ -d $to ]; then
460         to="$to/$file"
461     fi
462
463     local semaphore="$to-downloading"
464
465     is_downloading() {
466         if [ ! -f $semaphore ]; then
467             return 1
468         fi
469
470         # make sure the download has not been aborted
471         local now=$(date +%s)
472         local file_mtime=$(stat -c %Y "$to")
473         local staleness=$((now - file_mtime))
474         # let's assume an active download will write at least once a minute
475         if [ $staleness -gt 60 ]; then
476             return 1
477         fi
478
479         return 0
480     }
481
482     is_downloaded() {
483         # if the semaphore file exists, the file is either still downloading
484         # or a download was aborted and we cannot trust the target file
485         if [ -f $semaphore ]; then
486             return 1
487         fi
488
489         if ! is_downloading && [ -r "$to" ] && [ -s "$to" ]; then
490             return 0
491         fi
492
493         return 1
494     }
495
496     if $force || ! is_downloaded; then
497         if is_downloading; then
498             echo "Somebody else is downloading $from..."
499             while is_downloading; do
500                 echo "Waiting for $to to finish downloading"
501                 sleep 60
502             done
503             if is_downloaded; then
504                 return 0
505             else
506                 echo "The download we were waiting for seems to have been aborted"
507             fi
508
509         fi
510
511         if $DOWNLOAD; then
512             echo "Downloading $from..."
513             # flag others so they don't try to download also
514             push_exit_trap "rm -f $to $semaphore" "download"
515             touch $semaphore
516             if ! wget -nv "$from" -O "$to" || [ ! -s "$to" ]; then
517                 # the trap will remove the files via the fatal below
518                 fatal 1 "Could not download ${to##*/} from ${from%/*}/."
519             fi
520             rm -f $semaphore
521             delete_exit_trap "download"
522         else
523             fatal 1 "${to##*/} not found in directory ${to%/*}."
524         fi
525     fi
526
527     return 0
528
529 }
530
531 download_ofed() {
532     local force="${1:-false}"
533
534     if [ -z "$OFED_VERSION" -o "$OFED_VERSION" = "inkernel" ]; then
535         return 0
536     fi
537
538     local location="http://www.openfabrics.org/downloads/OFED/ofed-${OFED_VERSION}/"
539
540     if [[ $OFED_VERSION = daily-* ]]; then
541         local Mmv daily
542         OFED_VERSION=${OFED_VERSION/daily-/}
543         Mmv=${OFED_VERSION%%-*}
544         daily=${OFED_VERSION##$Mmv-}
545         location="http://www.openfabrics.org/downloads/OFED/ofed-${Mmv}-daily/"
546         # find the filename for the version for the date specified
547         OFED_VERSION=$(curl -s "$location" | sed -nre "/${daily}-/s/.*href=\"OFED-(${Mmv//./\\.}-${daily}-[0-9]{4,4}).tgz.*$/\1/p" | tail -1)
548         if [ -z "$OFED_VERSION" ]; then
549             fatal 1 "Could not determine the filename of the OFED snapshot for ${daily}"
550         fi
551     fi
552
553     local file="OFED-${OFED_VERSION}.tgz"
554     download_file "$location/$file" "$KERNELTREE" "$force"
555
556 }
557
558 load_target() {
559
560     EXTRA_VERSION_save="$EXTRA_VERSION"
561     for patchesdir in "$EXTERNAL_PATCHES" \
562                       "$TOPDIR/lustre/lustre/kernel_patches"; do
563         TARGET_FILE="$patchesdir/targets/$TARGET.target"
564         [ -r "$TARGET_FILE" ] && break
565     done
566     [ -r "$TARGET_FILE" ] || fatal 1 "Target $TARGET was not found."
567
568     echo "Loading target config file $TARGET.target..."
569
570     # if the caller specified an OFED_VERSION it should override whatever
571     # the target file specifies
572     local env_OFED_VERSION="$OFED_VERSION"
573
574     . "$TARGET_FILE"
575
576     if [ -n "$env_OFED_VERSION" ]; then
577         OFED_VERSION="$env_OFED_VERSION"
578     fi
579
580     # doesn't make any sense to build OFED for xen domX's
581     if $XEN; then
582         OFED_VERSION=""
583     fi
584
585     # XXX - set_rpm_smp_type is an ugly undeterministic hack.  it needs to
586     #       go away and the target just specify the $RPMSMPTYPE
587     [ -z "$RPMSMPTYPE" ] && set_rpm_smp_type
588
589     # CC might have been overwriten in TARGET_FILE
590     if [[ $CC != ccache\ * ]] && which "$CCACHE" &>/dev/null; then
591         export CCACHE && export CC="ccache $CC"
592     fi
593
594     if [ ! "$KERNELTREE" = "" ] && [ -d "$KERNELTREE" ]; then
595         KERNELDIR="$KERNELTREE/${lnxmaj}"
596         [ -d "$KERNELDIR" ] || mkdir "$KERNELDIR"
597     fi
598
599     # verify the series is available
600     if [ "$SERIES" ]; then
601         for series in $SERIES; do
602             for patchesdir in "$EXTERNAL_PATCHES" "$TOPDIR/lustre/lustre/kernel_patches"; do
603                 [ -r "$patchesdir/series/$series" ] && continue 2
604             done
605             fatal 1 "Target $TARGET's series $SERIES could not be found.\nSearched:\n\t$EXTERNAL_PATCHES/series\n\t$TOPDIR/lustre/lustre/kernel_patches/series."
606         done
607     fi
608
609     # set the location of the .config file
610     local XENPOSTFIX=""
611     if $XEN; then
612         XENPOSTFIX="-xen"
613     fi
614
615     if [ -f $TOPDIR/lustre/lustre/kernel_patches/kernel_configs/kernel-$lnxmaj-$TARGET-$TARGET_ARCH.config ]; then
616         CONFIG_FILE="$TOPDIR/lustre/lustre/kernel_patches/kernel_configs/kernel-$lnxmaj-$TARGET$XENPOSTFIX-$TARGET_ARCH${RPMSMPTYPE:+-}${RPMSMPTYPE}.config"
617     fi
618
619     local lnxrelnew=${lnxrel//-/_}
620
621     # remember the EXTRA_VERSION before we diddle it here
622     # XXX - we really should not diddle with any values read in from the
623     #       target file.  if we want to modify a value, we should create
624     #       a new variable.
625     PRISTINE_EXTRA_VERSION=$EXTRA_VERSION
626
627     if ! $PATCHLESS && [ ! -f "$CONFIG_FILE" ]; then
628         fatal 1 "Config file for target $TARGET missing from $TOPDIR/lustre/lustre/kernel_patches/kernel_configs/."
629     fi
630
631     if [ "$EXTRA_VERSION_save" ]; then
632         EXTRA_VERSION="$EXTRA_VERSION_save"
633     elif ! $RELEASE; then
634         # if there is no patch series, then this is not a lustre specific
635         # kernel.  don't make it look like one
636         if $PATCHLESS || [ -n "$SERIES" ]; then
637             EXTRA_VERSION=$(echo $EXTRA_VERSION | sed -e "s/\(.*_lustre\)\..*/\1/")
638 #            EXTRA_VERSION="${EXTRA_VERSION}-${TAG}.${TIMESTAMP}"
639             if ! $PATCHLESS && [ -n "$BUILDID" ]; then
640                 EXTRA_VERSION="${EXTRA_VERSION}.${BUILDID}"
641             fi
642         fi
643     fi
644     # EXTRA_VERSION=${EXTRA_VERSION//-/_}
645
646     ALL_ARCHS="$BASE_ARCHS $BIGMEM_ARCHS $BOOT_ARCHS $JENSEN_ARCHS $SMP_ARCHS $BIGSMP_ARCHS $PSERIES64_ARCHS $UP_ARCHS"
647
648     BUILD_ARCHS=
649     for arch in $(uniqify "$ALL_ARCHS"); do
650         if [ -z "$TARGET_ARCHS" ] ||
651            [[ \ $TARGET_ARCHS\  = *\ $arch\ * ]]; then
652             BUILD_ARCHS="$BUILD_ARCHS $arch"
653         fi
654     done
655     [ "$BUILD_ARCHS" ] || usage 1 "No available target archs to build."
656     echo "Building for: $BUILD_ARCHS"
657 }
658
659 tarflags() {
660     local file="$1"
661
662     case "$file" in
663         '')
664             fatal 1 "tarflags(): File name argument missing."
665             ;;
666         *.tar.gz | *.tgz)
667             echo 'zxf'
668             ;;
669         *.tar.bz2)
670             echo 'jxf'
671             ;;
672         *.tar)
673             echo 'xf'
674             ;;
675         *)
676             fatal 1 "tarflags(): Unrecognized tar extension in file: $1"
677             ;;
678     esac
679
680 }
681
682 untar() {
683     local tarfile="$1"
684     shift
685     local extractfile="$@"
686
687     echo "Untarring ${tarfile##*/}..."
688     tar $(tarflags "$tarfile") "$tarfile" $extractfile
689
690 }
691
692 unpack_ofed() {
693
694     if ! untar "$KERNELTREE/OFED-${OFED_VERSION}.tgz"; then
695         return 1
696     fi
697     [ -d OFED ] || ln -sf OFED-[0-9].[0-9]* OFED
698
699 }
700
701 unpack_lustre() {
702
703     if [ -z "$LUSTRE" ]; then
704         local DATESTAMP=""
705
706         if [ -n "$USE_DATESTAMP" ]; then
707             DATESTAMP="-D '$DATE'"
708         fi
709
710         local DIRNAME="lustre-$TAG-$TIMESTAMP"
711
712         cvs -d "$CVSROOT" -qz3 co $DATESTAMP -d "$DIRNAME" lustre || \
713             fatal 1 "There was an error checking out toplevel Lustre from CVS."
714         pushd "$DIRNAME" > /dev/null
715         ./lustrecvs "$TAG" || \
716             fatal 1 "There was an error checking out Lustre/Portals/Build from CVS."
717         echo "Creating lustre tarball..."
718         sh autogen.sh || fatal 1 "There was an error running autogen.sh."
719         ./configure --enable-dist || \
720             fatal 1 "There was an error running ./configure to create makefiles."
721         make dist || fatal 1 "There was an error running 'make dist'."
722         LUSTRE=$PWD/lustre-*.tar.gz
723         popd > /dev/null
724     fi
725
726     untar "$LUSTRE" || fatal 1 "Error unpacking Lustre tarball"
727     [ -d lustre ] || ln -sf lustre-[0-9].[0-9]* lustre
728
729 }
730
731 do_patch_linux() {
732
733     local do_patch=${1:-true}
734
735     FULL_PATCH="$PWD/lustre-kernel-${TARGET}-${EXTRA_VERSION}.patch"
736     [ -f "$FULL_PATCH" ] && rm -f "$FULL_PATCH"
737     $do_patch && pushd linux >/dev/null
738     for series in $SERIES; do
739         echo -n "Applying series $series:"
740         for patchesdir in "$EXTERNAL_PATCHES" "$TOPDIR/lustre/lustre/kernel_patches"; do
741             [ -r "$patchesdir/series/$series" ] || continue
742             SERIES_FILE="$patchesdir/series/$series"
743             for patch in $(<"$SERIES_FILE"); do
744                 echo -n " $patch"
745                 PATCH_FILE="$patchesdir/patches/$patch"
746                 [ -r "$PATCH_FILE" ] || \
747                     fatal 1 "Patch $patch does not exist in Lustre tree."
748                 cat "$PATCH_FILE" >> "$FULL_PATCH" || {
749                     rm -f $FULL_PATCH
750                     fatal 1 "Error adding patch $patch to full patch."
751                 }
752                 if $do_patch; then
753                     patch -s -p1 < "$PATCH_FILE" 2>&1 || {
754                         rm -f $FULL_PATCH
755                         fatal 1 "Error applying patch $patch."
756                     }
757                 fi
758             done
759             break
760         done
761         echo
762     done
763     $do_patch && popd >/dev/null
764     echo "Full patch has been saved in ${FULL_PATCH##*/}."
765
766 }
767
768 build_lustre() {
769     local linux="$1"
770     local linuxobj="$2"
771
772     cp "$LUSTRE" SOURCES
773
774     pushd lustre >/dev/null
775
776     echo "Building Lustre RPMs for: $BUILD_ARCHS..."
777     local targets arch
778     for arch in $BUILD_ARCHS; do
779         targets="--target $arch $targets"
780     done
781
782     local confoptions=""
783
784     if $PATCHLESS; then
785         confoptions="$confoptions --disable-server"
786     fi
787
788     local rpmbuildopt='-tb'
789     if $NORPM; then
790         rpmbuildopt='-tc'
791         echo NORPM mode. Only compiling.
792     fi
793
794     ( $(skeep_ldiskfs_rpm $TAG) ) || {
795
796         pushd ldiskfs > /dev/null || return 255
797
798         if !  ./configure --enable-dist; then
799             echo "failed to configure in ldiskfs"
800             popd >/dev/null # pushd ldiskfs
801             popd >/dev/null # pushd lustre
802             return 255
803         fi
804
805         if ! make dist 2>&1; then
806             popd >/dev/null # pushd ldiskfs
807             popd >/dev/null # pushd lustre
808             return 255
809         fi
810
811         #cp lustre-ldiskfs*.tar.gz $TOPDIR/SOURCES || \
812         #    fatal 1 "Could not copy lustre-ldiskfs*.tar.gz to $TOPDIR/SOURCES"
813
814         if ! $RPMBUILD $targets $rpmbuildopt lustre-ldiskfs*.tar.gz \
815             --define "configure_args $confoptions ${CONFIGURE_FLAGS}" \
816             --define "kdir $linux" \
817             ${linuxobj:+--define "kobjdir $linuxobj"} \
818             --define "_tmppath /var/tmp" \
819             --define "_topdir $TOPDIR" 2>&1; then
820             popd >/dev/null # pushd ldiskfs
821             popd >/dev/null # pushd lustre
822             return 255
823         fi
824
825         if $DO_SRC; then
826             if ! $RPMBUILD -ts lustre-ldiskfs*.tar.gz \
827                 --define "configure_args $confoptions ${CONFIGURE_FLAGS}" \
828                 --define "kdir $linux" \
829                 ${linuxobj:+--define "kobjdir $linuxobj"} \
830                 --define "_tmppath /var/tmp" \
831                 --define "_topdir $TOPDIR" 2>&1; then
832                 popd >/dev/null # pushd ldiskfs
833                 popd >/dev/null # pushd lustre
834                 return 255
835             fi
836         fi
837         popd >/dev/null # pushd ldiskfs
838
839         # tell lustre where ldiskfs is
840         # XXX - pointing to the RPM BUILD dir is a hack.  we need to flesh
841         #       out the ldiskfs RPM build so that it builds a
842         #       lustre-ldiskfs-devel RPM and install that and point lustre
843         #       to that instead
844         confoptions="$confoptions --with-ldiskfs=$(ls -d $TOPDIR/BUILD/lustre-ldiskfs-*)"
845     }
846
847     # convert the $PATCHLESS boolean to an empty/not-empty boolean
848     # as silly as this seems, it makes the syntax of the rpmbuild command
849     # simpler and not need an eval to deal with the quotes in the quotes
850     local is_patchless=""
851     if $PATCHLESS; then
852         is_patchless="yes"
853     fi
854
855     # ditto for the lustre-tests boolean
856     local lustre_tests=""
857     if ! $LUSTRE_TESTS; then
858         lustre_tests="no"
859     fi
860
861     $RPMBUILD $targets $rpmbuildopt "$LUSTRE" \
862         ${is_patchless:+--define "lustre_name lustre-client"} \
863         ${lustre_tests:+--define "build_lustre_tests 0"} \
864         ${FIND_REQUIRES:+--define "__find_requires $FIND_REQUIRES"} \
865         --define "configure_args $confoptions ${CONFIGURE_FLAGS}" \
866         --define "kdir $linux" \
867         ${linuxobj:+--define "kobjdir $linuxobj"} \
868         --define "_tmppath $TMPDIR" \
869         --define "_topdir $TOPDIR" 2>&1 || \
870         fatal 1 "Error building rpms for $BUILD_ARCHS."
871
872     if $DO_SRC; then
873         if ! $RPMBUILD -ts "$LUSTRE" \
874                  ${is_patchless:+--define "lustre_name lustre-client"} \
875                  ${lustre_tests:+--define "build_lustre_tests 0"} \
876                  ${FIND_REQUIRES:+--define "__find_requires $FIND_REQUIRES"} \
877                  --define "configure_args $confoptions ${CONFIGURE_FLAGS}" \
878                  --define "kdir $linux" \
879                  ${linuxobj:+--define "kobjdir $linuxobj"} \
880                  --define "_tmppath $TMPDIR" \
881                  --define "_topdir $TOPDIR" 2>&1; then
882             popd
883             return 255
884         fi
885     fi
886     popd >/dev/null
887
888 }
889
890 stage() {
891
892     [ "$STAGEDIR" ] || return 0
893
894     for arch in $BUILD_ARCHS; do
895         rpmdir="${STAGEDIR}/${CANONICAL_TARGET}-${arch}"
896         echo "${0##*/}: Copying RPMs into ${rpmdir}"
897         mkdir -p "${rpmdir}"
898         cp -v RPMS/${arch}/*.rpm "${rpmdir}"
899         if [ -d RPMS/noarch ]; then
900             cp -v RPMS/noarch/*.rpm "${rpmdir}"
901         fi
902     done
903
904     cp -v "$LUSTRE" "$STAGEDIR"
905
906 }
907
908 #check if we need to build separate ldiskfs RPM
909 skeep_ldiskfs_rpm() {
910     local tag="$1"
911
912     local skip=false
913
914     if ! $LDISKFSRPM; then
915         skip=true
916     elif $PATCHLESS; then
917         skip=true
918     else
919         for skiptag in $SKIPLDISKFSRPM; do
920             [[ $tag == $skiptag ]] && skip=true && break
921         done
922     fi
923
924     echo $skip
925
926 }
927
928 set_rpm_smp_type() {
929
930     local infact_arch="${TARGET_ARCH}"
931
932     RPMSMPTYPE=""
933     [ "$infact_arch" == "i586" ] && infact_arch="i686"
934
935     local smp_type
936     for smp_type in $SMP_ARCHS; do
937         [ $infact_arch == $smp_type ] && RPMSMPTYPE=smp && break
938     done
939
940     for smp_type in $BIGSMP_ARCHS; do
941         [ $infact_arch == $smp_type ] && RPMSMPTYPE=bigsmp && break
942     done
943
944     for smp_type in $PPC64_ARCHS; do
945         [ $infact_arch == $smp_type ] && RPMSMPTYPE=ppc64 && break
946     done
947
948     for smp_type in $DEFAULT_ARCHS; do
949         [ $infact_arch == $smp_type ] && RPMSMPTYPE=default && break
950     done
951
952 }
953
954 # This function takes a linux source pool and digs out the linux release
955 # from it
956 find_linux_release() {
957     local SRCDIR="$1"
958
959     local LINUXRELEASEHEADER=$SRCDIR/include/linux/version.h
960     if [ -s $SRCDIR/include/linux/utsrelease.h ]; then
961         LINUXRELEASEHEADER=$SRCDIR/include/linux/utsrelease.h
962     fi
963
964     sed -ne 's/#define UTS_RELEASE "\(.*\)"$/\1/p' $LINUXRELEASEHEADER
965
966 }
967
968 # XXX this needs to be re-written as a wrapper around find_rpm
969 #     or just gotten rid of.  :-)
970 find_linux_rpm() {
971     local prefix="$1"
972
973     local pathtorpms="${KERNELRPMSBASE}/${lnxmaj}/${DISTRO}"
974     [ -d $pathtorpms ] || return 255
975
976     local kernelbinaryrpm rpmfile
977     local wanted_kernel="${lnxmaj}${lnxmin}-${lnxrel}"
978
979     local arch ret=1
980     for arch in $TARGET_ARCHS_ALL; do
981         local found_rpm="" rpm
982         for rpm in ${pathtorpms}/${arch}/*.rpm; do
983             if rpm -q --provides -p "$rpm" 2>&3 | grep -q "kernel${prefix} = $wanted_kernel" 2>&3; then
984
985                 found_rpm="$rpm"
986                 ret=0
987                 break
988             fi
989         done
990         [ -f "$found_rpm" ] && break
991     done
992
993     echo "$found_rpm"
994     return $ret
995
996 }
997
998 # unpack kernel(/source/devel) RPM
999 #
1000 # This function and it's setting of $LINUX and $LINUXOBJ is a total hack that
1001 # needs to completely refactored.  It completely ingores that $BUILD_ARCHS may
1002 # contain a list of arches for which rpmbuild commands (including the one for
1003 # lustre itself)
1004 unpack_linux_devel_rpm() {
1005     local kernelrpm="${1}"
1006
1007     [ -f "$kernelrpm" ] || return 255
1008     [ -d $TOPDIR/reused ] || mkdir $TOPDIR/reused || return 255
1009
1010     pushd $TOPDIR/reused &>/dev/null || return 255
1011
1012     if ! rpm2cpio < "$kernelrpm" | cpio -id > /dev/null 2>&1; then
1013         return 255
1014     fi
1015
1016     # call a distro specific hook, if available
1017     if type -p unpack_linux_devel_rpm-$DISTRO; then
1018         if ! unpack_linux_devel_rpm-$DISTRO "$kernelrpm"; then
1019             return 255
1020         fi
1021     fi
1022
1023     popd &>/dev/null
1024
1025     find_linux_devel_paths $TOPDIR/reused
1026
1027     return 0
1028
1029 }
1030
1031 build_kernel_ib() {
1032     local linux="$1"
1033
1034     # build kernel-ib{,-devel}
1035     local K_SRC="K_SRC"
1036     # ofed 1.3 had a bug in the rpm spec
1037     if [ "$OFED_VERSION" = "1.3" ]; then
1038         K_SRC="KSRC"
1039     fi
1040
1041     local OFED_CORE="--with-core-mod --with-ipoib-mod --with-sdp-mod --with-user_mad-mod --with-user_access-mod --with-addr_trans-mod --with-rds-mod --with-qlgc_vnic-mod --with-madeye-mod"
1042     local OFED_HARDWARE="--with-mthca-mod --with-mlx4-mod --with-mlx4_en-mod --with-cxgb3-mod --with-nes-mod"
1043     # some I/B drivers are architecture dependent and kernel-ib's configure
1044     # does not figure it out for us ~sigh~
1045     case "$TARGET_ARCH" in
1046         ppc64)
1047             OFED_HARDWARE="$OFED_HARDWARE --with-ehca-mod"
1048             ;;
1049     esac
1050     # we're no longer shipping the OFED iSCSI
1051     #OFED_ISCSI="--with-srp-mod --with-srp-target-mod"
1052     ## ISER module has no backport support as of OFED 1.5 (i.e. only builds on
1053     ##kernels >= 2.6.30)
1054     #if [[ $OFED_VERSION = 1.[0-4]* ]]; then
1055     #   OFED_ISCSI="$OFED_ISCSI --with-iser-mod"
1056     #fi
1057
1058     # assume we are just rebuilding the SRPM
1059     local BUILD_TYPE=${BUILD_TYPE:-"--rebuild"}
1060     local SOURCE="${TOPDIR}/OFED/SRPMS/ofa_kernel-*.src.rpm"
1061
1062     # but switch to building from the SPEC if we need to apply patches
1063     if ls ${TOPDIR}/lustre/build/patches/ofed/* >/dev/null; then
1064         BUILD_TYPE="-bb"
1065         rpm --define "_topdir ${TOPDIR}" -ivh $SOURCE
1066         SOURCE="${TOPDIR}/SPECS/ofa_kernel.spec"
1067         local file ed_fragment1 ed_fragment2 n=1
1068         for file in $(ls ${TOPDIR}/lustre/build/patches/ofed/*); do
1069             ed_fragment1="$ed_fragment1
1070 Patch$n: ${file%%*/}"
1071             ed_fragment2="$ed_fragment2
1072 %patch$n -p0"
1073             cp $file ${TOPDIR}/SOURCES
1074             let n=$n+1
1075         done
1076
1077         if [ $n -gt 1 ]; then
1078             ed $SOURCE <<EOF
1079 /^Source: /a
1080 $ed_fragment1
1081 .
1082 /^%setup /a
1083 $ed_fragment2
1084 .
1085 wq
1086 EOF
1087         fi
1088     fi
1089
1090     local linuxrelease=$(find_linux_release "$linux")
1091     if ! $RPMBUILD $BUILD_TYPE --define 'build_kernel_ib 1' --define 'build_kernel_ib_devel 1' \
1092                   ${FIND_REQUIRES:+--define "__find_requires $FIND_REQUIRES"} \
1093                   --define "_topdir ${TOPDIR}" --target ${TARGET_ARCH} \
1094                   --define "KVERSION ${linuxrelease}" \
1095                   --define "$K_SRC ${linux}" \
1096                   --define "LIB_MOD_DIR /lib/modules/${linuxrelease}/updates" \
1097                   ${OFA_KERNEL_RELEASE:+--define "_release $OFA_KERNEL_RELEASE"} \
1098                   --define "configure_options --without-quilt $OFED_CORE $OFED_HARDWARE $OFED_ISCSI" \
1099                   ${SOURCE} 2>&1; then
1100         fatal 1 "Error building kernel-ib"
1101     fi
1102
1103 }
1104
1105 store_for_reuse() {
1106     local articles="$1"
1107     local module="$2"
1108     local location="$3"
1109     local signature="$4"
1110     local use_links="$5"
1111
1112     local linkflag=""
1113     if $use_links; then
1114         linkflag="l"
1115     fi
1116
1117     local default_iface=$(/sbin/ip route get 192.1.1.1 | sed -ne 's/.* dev \(.*\)  * src .*/\1/p')
1118     if [ -z "$default_iface" ]; then
1119         fatal 1 "Failed to determine the default route interface"
1120     fi
1121     local unique_id=$(/sbin/ip addr show dev $default_iface | sed -ne '/ inet /s/ *inet \(.*\)\/.*/\1/p' | head -1)
1122     if [ -z "$unique_id" ]; then
1123         fatal 1 "Failed to determine a unique id from interface $default_interface"
1124     fi
1125
1126     local finallocation="$location"/"$signature"/"$module"
1127     location="$location"/"$signature-${unique_id}"/"$module"
1128     mkdir -p "$location"
1129     # the cleanup script removes any directory that doesn't have a
1130     # .lastused, so let's try to prevent that as soon as we can
1131     # this solution still slightly racy with the cleanup script
1132     # but the race is a lot tighter now
1133     touch -t 197001010000 "$location/.lastused"
1134     ## use eval/echo here to make sure shell expansions are performed
1135     #if ! cp -a${linkflag} $(eval echo $articles) "$location"; then
1136     local article
1137     for article in $(eval echo $articles); do
1138         if ! cp -a${linkflag} "$article" "$location"; then
1139             error "Failed to copy \"$article\" to \"$location\" in store_for_reuse()"
1140             # rename the cache location so that it's not cached
1141             # product, but is around for analysis
1142             mv "$location"{,-bad-$(date +%s)} ||
1143                 error "failed to clean up a failed cache attempt" \
1144                       "in \"$location\" -- manual cleanup will be" \
1145                       "necessary"
1146             return 1
1147         fi
1148     done
1149
1150     # flag the cache as complete (i.e. in case lbuild was previously
1151     # interrupted while caching)
1152     touch "$location/.lastused"
1153
1154     # put the temporary location into the final location
1155     # (last one wins)
1156     mkdir -p "${finallocation%/*}"
1157     mv "$location" "$finallocation"
1158     rmdir "${location%/*}"
1159     return 0
1160
1161 }
1162
1163 reuse() {
1164     local module="$1"
1165     local dest="$2"
1166     local use_links="${3:-false}"
1167     local signature="$4"
1168
1169     if [ -n "$REUSEBUILD" ] && [ -d "$REUSEBUILD/$signature/$module" ]; then
1170         if [ ! -f "$REUSEBUILD/$signature/$module/.lastused" ]; then
1171             # the .lastused flag is populated at the end of the caching to
1172             # signal that the caching was completed.  if that flag is not
1173             # there, then the cache is invalid (and should be removed in fact)
1174             mv "$REUSEBUILD/$signature/$module"{,-bad-$(date +%s)} ||
1175                 fatal 1 "failed to clean up a bad cache in location $REUSEBUILD/$signature/$module\" -- manual cleanup will be necessary"
1176             return 1
1177         fi
1178
1179         # so that we know how stale this entry is
1180         touch $REUSEBUILD/$signature/$module/.lastused
1181
1182         if $use_links; then
1183             if ls $REUSEBUILD/$signature/$module/* >/dev/null 2>&1; then
1184                 cp -al $REUSEBUILD/$signature/$module/* $dest/
1185             fi
1186         else
1187             # copying is pretty heavy
1188             # cp -a $REUSEBUILD/$signature/$module/* $dest/
1189             # do some creative symlinking instead
1190             local dir
1191             for dir in BUILD SRPMS SPECS; do
1192                 if ls $REUSEBUILD/$signature/$module/$dir/* >/dev/null 2>&1; then
1193                     ln -s $REUSEBUILD/$signature/$module/$dir/* $dest/$dir
1194                 fi
1195             done
1196             # sources have to be copied by file because we need SOURCES to
1197             # be a dir we can write into
1198 # could overrun ls's arg list here
1199             #ls $REUSEBUILD/$signature/$module/SOURCES/* |
1200             find $REUSEBUILD/$signature/$module/SOURCES/ -type f |
1201                 xargs ln -t $dest/SOURCES -s
1202
1203             # same for RPMS/* dirs
1204 # could overrun ls's arg list here
1205             #ls $REUSEBUILD/$signature/$module/RPMS/$TARGET_ARCH/* |
1206             local dir
1207             for dir in $REUSEBUILD/$signature/$module/RPMS/*; do
1208                 mkdir -p $dest/RPMS/${dir##*/}
1209                 find $dir -type f |
1210                   xargs ln -t $dest/RPMS/${dir##*/} -s
1211             done
1212         fi
1213         return 0
1214     else
1215         return 1
1216     fi
1217 }
1218
1219 basearch() {
1220     local arch="$1"
1221
1222     if [[ $arch = i[3456]86 ]]; then
1223         echo "i386"
1224     else
1225         echo "$arch"
1226     fi
1227
1228 }
1229
1230 #
1231 # in a given directory, find the first rpm matching given requirements
1232 #
1233 find_rpm() {
1234     local dir="$1"
1235     local match_type="$2"
1236     local match="$3"
1237
1238     pushd "$dir" > /dev/null || \
1239         fatal 1 "Unable to chdir to directory \"$dir\" in find_rpm()"
1240
1241     local file
1242     for file in $(ls *.rpm); do
1243         if [ ! -f "$file" ]; then
1244             continue
1245         fi
1246         case "$match_type" in
1247             provides)
1248                 # match is any valid ERE (i.e. given to egrep) match
1249                 if rpm -q --provides -p "$file" 2>&3 | egrep -q "$match"; then
1250                     echo "$file"
1251                     popd >/dev/null
1252                     return 0
1253                 fi
1254                 ;;
1255             *)
1256                 popd >/dev/null
1257                 fatal 1 "Unknown match type \"$match_type\" given to find_rpm()"
1258                 ;;
1259         esac
1260     done
1261
1262     popd >/dev/null
1263     return 1
1264 }
1265
1266 build_kernel_with_srpm() {
1267     local outfd=$1
1268
1269     if [ -z "$outfd" ] || [ $outfd = 1 ]; then
1270         fatal 1 "You must supply a file descriptor to ${FUNCNAME[0]} and it cannot be 1"
1271     fi
1272
1273     # need to generate the patch for this target
1274     do_patch_linux false >&${outfd}    # sets global $FULL_PATCH (yeah, yuck)
1275
1276     # get an md5sum of the kernel patch + config for reuse check
1277     # XXX really, there needs to be a signature and a CONFIG_FILE per arch
1278     #     in BUILD_ARCHS
1279     local release_str
1280     if $RELEASE; then
1281         local release_str="RELEASE=$RELEASE\n"
1282     fi
1283
1284     if $USE_BUILD_CACHE && [ -n "$REUSEBUILD" ]; then
1285         local REUSE_SIGNATURE=$({ echo -en $release_str;
1286                                   echo $BUILD_GEN;
1287                                   cat $CONFIG_FILE $TARGET_FILE $FULL_PATCH; } |
1288                                 md5sum | cut -d" " -f1)
1289         # see if we can link to the reuse pool
1290         # XXX - hrm.  i'm not convinced this doesn't belong in the reuse
1291         #       "library"
1292         local CAN_LINK_FOR_REUSE=false
1293         touch $REUSEBUILD/$$
1294         if cp -al $REUSEBUILD/$$ $TOPDIR/ 2>/dev/null; then
1295             CAN_LINK_FOR_REUSE=true
1296         fi
1297         rm $REUSEBUILD/$$
1298     fi
1299
1300     # the extra version string to use for the kernel (which might be a reused
1301     # kernel, remember)
1302     local kernel_extra_version=""
1303     if ! $USE_BUILD_CACHE || ! reuse kernel "$TOPDIR" "$CAN_LINK_FOR_REUSE" \
1304                                    "$REUSE_SIGNATURE"; then
1305         # nothing cached, build from scratch
1306         if [ ! -r "$KERNELDIR/$KERNEL_SRPM" ]; then
1307             echo "Downloading kernel SRPM"
1308             download_srpm "$CANONICAL_TARGET" "$KERNEL_SRPM" >&${outfd}
1309         fi
1310
1311         if ! rpm -ivh $KERNELDIR/$KERNEL_SRPM \
1312                   --define "_topdir $TOPDIR" >&${outfd} 2>&1; then
1313             # should we clean this up or leave it for analysis?
1314             #rm -rf $RPMTOPDIR
1315             fatal 1 "Error installing kernel SRPM."
1316         fi
1317
1318         # put the Lustre kernel patch into the RPM build tree
1319         cp $FULL_PATCH $TOPDIR/SOURCES/linux-${lnxmaj}-lustre.patch
1320         prepare_and_build_srpm >&${outfd} ||
1321             fatal 1 "failed to prepare_and_build_srpm"
1322
1323         if [ -z "$REUSE_SIGNATURE" ]; then
1324             echo "No reuse signature was caculated so not storing the built kernel" >&${outfd}
1325         else
1326             # store the resulting kernel RPM build tree for future use
1327             echo "Storing the built kernel for future reuse" >&${outfd}
1328             if ! store_for_reuse "$TOPDIR/{SPECS,SOURCES,BUILD,SRPMS,RPMS}" \
1329                                  "kernel" "$REUSEBUILD" "$REUSE_SIGNATURE" \
1330                                  "$CAN_LINK_FOR_REUSE"; then
1331                 error "Failed to store kernel RPMS for reuse"
1332                 echo "unknown"
1333                 return 1
1334             fi
1335         fi
1336     fi  # build reuse
1337
1338     # figure out the EXTRA_VERSION of the kernel we built or are re-using
1339     local KERNEL_RPM
1340     if ! KERNEL_RPM=$(find_rpm "$TOPDIR/RPMS/$TARGET_ARCH/" provides "^kernel ="); then
1341         fatal 1 "Failed to find a kernel RPM in $TOPDIR/RPMS/$TARGET_ARCH/"
1342     fi
1343     kernel_extra_version=$(rpm -q --queryformat "%{RELEASE}" -p $TOPDIR/RPMS/$TARGET_ARCH/$KERNEL_RPM)
1344
1345     # should now have the following RPMs
1346     # $TOPDIR/RPMS/$arch/kernel-lustre-2.6.18-53.1.21.el5_lustre.1.6.5.1.$arch.rpm
1347     # $TOPDIR/RPMS/$arch/kernel-lustre-devel-2.6.18-53.1.21.el5_lustre.1.6.5.1.$arch.rpm
1348     # $TOPDIR/RPMS/$arch/kernel-lustre-headers-2.6.18-53.1.21.el5_lustre.1.6.5.1.$arch.rpm
1349     # $TOPDIR/RPMS/$arch/kernel-lustre-debuginfo-common-2.6.18-53.1.21.el5_lustre.1.6.5.1.$arch.rpm
1350     # $TOPDIR/RPMS/$arch/kernel-lustre-debuginfo-2.6.18-53.1.21.el5_lustre.1.6.5.1.$arch.rpm
1351
1352     echo $kernel_extra_version
1353     return 0
1354
1355 }
1356
1357 # build OFED
1358 # globals used:
1359 #    TOPDIR
1360 #    REUSEBUILD, USE_BUILD_CACHE
1361 #    CONFIGURE_FLAGS
1362
1363 build_ofed() {
1364     local linux="$1"
1365     local ofed_version="$2"
1366
1367     # if an ofed version is given, then it means use OFED proper,
1368     # not any vendor specific "inkernel" version
1369     if [ -z "$ofed_version" ]; then
1370         return 0
1371     fi
1372
1373     if [ "$ofed_version" = "inkernel" ]; then
1374         # see if there is a distro specific override for this and use
1375         # that if it exists
1376         # XXX we need to better integrate a distro specific override with
1377         #     the rest of this function so that all of the reuse cache
1378         #     stuff is leveraged given that 80% of this function is reuse
1379         if type -p build_ofed-${DISTRO}; then
1380             local ofed_location
1381             ofed_location=$(build_ofed-${DISTRO} ${STDOUT})
1382             local rc=${PIPESTATUS[0]}
1383             CONFIGURE_FLAGS="--with-o2ib=${ofed_location} ${CONFIGURE_FLAGS}"
1384             return $rc
1385         else
1386             return 0
1387         fi
1388     fi
1389
1390     # build kernel-ib
1391     if $USE_BUILD_CACHE && [ -n "$REUSEBUILD" ]; then
1392         local REUSE_SIGNATURE=$({ echo "$ofed_version";
1393                                   echo "$(find_linux_release ${linux})";
1394                                   cat "${linux}/include/linux/autoconf.h"; } |
1395                                 md5sum | cut -d" " -f1)
1396         # see if we can link to the reuse pool
1397         # XXX - hrm.  i'm not convinced this doesn't belong in the reuse
1398         #       "library"
1399         local CAN_LINK_FOR_REUSE=false
1400         touch $REUSEBUILD/$$
1401         if cp -al $REUSEBUILD/$$ $TOPDIR/; then
1402             CAN_LINK_FOR_REUSE=true
1403         fi
1404         rm $REUSEBUILD/$$
1405     fi
1406
1407     if ! $USE_BUILD_CACHE || ! reuse ofed "$TOPDIR" "$CAN_LINK_FOR_REUSE" \
1408                                    "$REUSE_SIGNATURE"; then
1409         if [ -n "$REUSE_SIGNATURE" ]; then
1410             # stash away the existing built articles for a moment
1411             mkdir bak
1412             mv {BUILD,{S,}RPMS,S{OURCE,PEC}S} bak
1413             function mv_back {
1414                 pushd bak
1415                 find . | cpio -pudlm ..
1416                 popd
1417                 rm -rf bak
1418             }
1419             create_rpmbuild_dirs
1420         fi
1421         # build it
1422         build_kernel_ib "${linux}"
1423
1424         if [ -z "$REUSE_SIGNATURE" ]; then
1425             echo "No reuse signature was caculated so not storing the built ofed"
1426         else
1427             # store the resulting RPM build tree for future use
1428             echo "Storing the built ofed for future reuse"
1429             if ! store_for_reuse "$TOPDIR/{SPECS,SOURCES,BUILD,SRPMS,RPMS}" \
1430                                  "ofed" "$REUSEBUILD" "$REUSE_SIGNATURE" \
1431                                  "$CAN_LINK_FOR_REUSE"; then
1432                 error "Failed to store OFED RPMS for reuse"
1433                 mv_back
1434                 return 1
1435             fi
1436             # put the stuff we stashed away back
1437             mv_back
1438         fi
1439     fi
1440
1441     pushd "$TOPDIR" >/dev/null
1442     rm -rf kernel-ib-devel
1443     mkdir kernel-ib-devel
1444     cd kernel-ib-devel
1445     # the actual ofed RPMs don't have the -rc$n or -$date string appened that
1446     # might be present on the file
1447     local linuxrelease=$(find_linux_release "$linux")
1448     ofed_version=$(echo $ofed_version |
1449                    sed -re 's/-(20[0-9]{6,6}-[0-9]{4,4}|rc[0-9]*)$//')
1450     local rpm=$(ls $TOPDIR/RPMS/*/kernel-ib-devel-${ofed_version}-${linuxrelease//-/_}.*.rpm)
1451     if ! rpm2cpio < $rpm | cpio -id; then
1452         fatal 1 "could not unpack the kernel-ib-devel rpm."
1453     fi
1454     CONFIGURE_FLAGS="--with-o2ib=$(pwd)/usr/src/ofa_kernel ${CONFIGURE_FLAGS}"
1455     popd >/dev/null
1456
1457 }
1458
1459 build_with_srpm() {
1460
1461     if ! $PATCHLESS; then
1462         local kernel_extra_version
1463         if ! kernel_extra_version=$(build_kernel_with_srpm ${STDOUT}); then
1464             fatal 1 "Failed to build the kernel from it's SRPM"
1465         fi
1466
1467         for arch in $BUILD_ARCHS; do
1468             local kernel_devel_rpm
1469             if ! kernel_devel_rpm=$(find_rpm "$TOPDIR/RPMS/$arch/" provides "^$(devel_kernel_name $KERNEL_LUSTRE_NAMING) ="); then
1470                 fatal 1 "Failed to find a kernel development RPM in $TOPDIR/RPMS/$arch/"
1471             fi
1472
1473             # install the -devel RPM in preparation for modules builds
1474             if ! lnxrel="$kernel_extra_version" unpack_linux_devel_rpm \
1475                            "$TOPDIR/RPMS/$arch/$kernel_devel_rpm"; then
1476                 fatal 1 "Could not find the Linux tree in $TOPDIR/RPMS/$arch/$kernel_devel_rpm"
1477             fi
1478         done
1479     else
1480         # need to find and unpack the vendor's own kernel-devel for patchless
1481         # client build
1482         local kernelrpm
1483         if ! kernelrpm=$(find_linux_rpm "-$DEVEL_KERNEL_TYPE"); then
1484             fatal 1 "Could not find the kernel-$DEVEL_KERNEL_TYPE RPM in ${KERNELRPMSBASE}/${lnxmaj}/${DISTRO}"
1485         fi
1486         if ! lnxrel="$lnxrel" unpack_linux_devel_rpm "$kernelrpm" "-"; then
1487             fatal 1 "Could not find the Linux tree in $kernelrpm"
1488         fi
1489     fi
1490
1491     # ~sigh~  have to make copies of and modify some of the rpm
1492     # infrastructure files so that find-requires can find our unpacked
1493     # kernel-devel artifacts
1494     cp $RPM_HELPERS_DIR/{symset-table,find-requires{,.ksyms}} .
1495     FIND_REQUIRES="$(pwd)/find-requires"
1496     chmod 755 {symset-table,find-requires{,.ksyms}}
1497     local tmp="$(pwd)"
1498     tmp="${tmp//\//\\/}"
1499     ed find-requires <<EOF
1500 1a
1501 set -x
1502 .
1503 /|.*find-requires.ksyms/s/|/| bash -x/
1504 g/ [^ ]*\/\(find-requires\.ksyms\)/s// $tmp\/\1/g
1505 wq
1506 EOF
1507     ed find-requires.ksyms <<EOF
1508 1a
1509 set -x
1510 .
1511 g/\/.*\/\(symset-table\)/s//$tmp\/\1/g
1512 wq
1513 EOF
1514     ed symset-table <<EOF
1515 1a
1516 set -x
1517 .
1518 g/\(\/boot\/\)/s//$tmp\/reused\1/g
1519 g/\(\/usr\/src\/kernels\/\)/s//$tmp\/reused\1/g
1520 wq
1521 EOF
1522
1523     build_ofed "${LINUXOBJ:-$LINUX}" "$OFED_VERSION" ||
1524         fatal 1 "error building OFED"
1525
1526     # now build Lustre
1527     if build_lustre "$LINUX" "$LINUXOBJ"; then
1528         # the build worked.  resolve any symlinked files (i.e. from reuse)
1529         # in RPMS/$arch to real files so that that that huge mess of
1530         # complication known as LTS can copy them yet somewhere else.
1531         # is it any wonder this whole process is so damn so?  anyone ever
1532         # heard of hardlinks?  it's this cool new thing that allows you save
1533         # tons of time and space by creating... well you can go read about
1534         # them if you have not heard about them yet.
1535         # can i say how much the implemenation of all of this really impedes
1536         # RPM reuse?
1537         local dir
1538         for dir in RPMS/*; do
1539             pushd $dir
1540             for file in $(ls); do
1541                 if [ -h $file ]; then
1542                     cp $file foo
1543                     mv foo $file
1544                 fi
1545             done
1546             popd
1547         done
1548         # also, for i?86, make sure all of the RPMs are in RPMS/$TARGET_ARCH
1549         # as that's where LTS expects to find them
1550         for dir in RPMS/*; do
1551             if [ $dir = RPMS/$TARGET_ARCH ]; then
1552                 continue
1553             fi
1554             pushd $dir
1555             local files=$(ls)
1556             if [ -n "$files" ]; then
1557                 cp -al $files ../$TARGET_ARCH
1558             fi
1559             popd
1560         done
1561     else
1562         return 1
1563     fi
1564
1565 }
1566
1567 create_rpmbuild_dirs() {
1568
1569     [ -d RPMS ] || mkdir RPMS
1570     for arch in $BUILD_ARCHS; do
1571         if [[ $arch = i?86 ]]; then
1572             # some stupidity in the sles11 kernel spec requires an RPMS/i386
1573             # even if the target arch is i686
1574             [ -d RPMS/i386 ] || mkdir RPMS/i386
1575         fi
1576         [ -d RPMS/$arch ] || mkdir RPMS/$arch
1577     done
1578     [ -d BUILD ] || mkdir BUILD
1579     [ -d SOURCES ] || mkdir SOURCES
1580     [ -d SPECS ] || mkdir SPECS
1581     [ -d SRPMS ] || mkdir SRPMS
1582
1583 }
1584
1585 new_list() {
1586
1587     echo ""
1588
1589 }
1590
1591 add_list() {
1592     local list="$1"
1593     local item="$2"
1594
1595     echo "$list $item"
1596
1597 }
1598
1599 is_list_member() {
1600     local list="$1"
1601     local item="$2"
1602
1603     [[ $list\  == *\ $item\ * ]]
1604
1605 }
1606
1607 #########################################################################
1608 # Generate a backtrace through the call stack.
1609 #
1610 # Input: None
1611 # Output: None
1612 #########################################################################
1613 backtrace() {
1614     local strip=${1:-1}
1615
1616     local funcname="" sourcefile="" lineno="" n
1617
1618     echo "Call stack: (most recent first)"
1619     for (( n = $strip ; n < ${#FUNCNAME[@]} ; ++n )) ; do
1620         funcname=${FUNCNAME[$n - 1]}
1621         sourcefile=$(basename ${BASH_SOURCE[$n]})
1622         lineno=${BASH_LINENO[$n - 1]}
1623         if [ $n = 1 ]; then
1624             let lineno-=11
1625         fi
1626         # Display function arguments
1627         if [[ ! -z "${BASH_ARGV[@]}" ]]; then
1628             local args newarg j p=0
1629             for (( j = ${BASH_ARGC[$n - 1]}; j > 0; j-- )); do
1630                 newarg=${BASH_ARGV[$j + $p - 1]}
1631                 args="${args:+${args} }'${newarg}'"
1632             done
1633             let p+=${BASH_ARGC[$n - 1]}
1634         fi
1635         echo "  ${funcname} ${args:+${args} }at ${sourcefile}:${lineno}"
1636     done
1637
1638     echo
1639     echo "BEGIN BACKTRACE"
1640
1641     #echo ${BASH_LINENO[*]}
1642     #echo ${BASH_SOURCE[*]}
1643     #echo ${FUNCNAME[*]}
1644     local i=$((${#FUNCNAME[@]} - 1))
1645     while [ $i -ge 0 ]; do
1646         local lineno=${BASH_LINENO[$i]}
1647         if [ $i = 0 ]; then
1648             let lineno-=11
1649         fi
1650         local SOURCELINE="${BASH_SOURCE[$i + 1]}:${lineno}"
1651         # Can't figure out how to get function args from other frames...
1652         local FUNCTION="${FUNCNAME[$i]}()"
1653         echo "$SOURCELINE:$FUNCTION"
1654         i=$((i - 1))
1655     done
1656
1657     echo "END BACKTRACE"
1658
1659     echo $BACKTRACE
1660
1661 }
1662
1663 seen_list=$(new_list)
1664 trap 'set +x;
1665 echo "An unexpected error has occurred at ${BASH_SOURCE[0]##*/}:$((LINENO-1)).
1666 Unfortunately the above line number in the message may or may not be correct,
1667 but details have been send to the lbuild maintainer.  Attempting to continue."; (echo "Untrapped error"
1668 echo
1669 # have we seen this one
1670 echo "checking seen list for ${BASH_SOURCE[0]}:${BASH_LINENO[0]}"
1671
1672 if is_list_member "$seen_list" "${BASH_SOURCE[0]}:${BASH_LINENO[0]}"; then
1673   echo "seen this one already"
1674 else
1675   seen_list=$(add_list "$seen_list" "${BASH_SOURCE[0]}:${BASH_LINENO[0]}")
1676 fi
1677 backtrace
1678 echo
1679 echo "Environment:"
1680 set
1681 ) | mail -s "Untrapped error at ${BASH_SOURCE[0]##*/}:$((LINENO-15)) on $HOSTNAME" brian@sun.com >&2; set $xtrace' ERR
1682 set -E
1683
1684 [ -r ~/.lbuildrc ] && . ~/.lbuildrc
1685
1686 options=$(getopt -o d:D:h -l kerneltree:,distro:,kernelrpm:,reusebuild:,patchless,ldiskfs,ccache,reuse:,norpm,disable-datestamp,external-patches:,timestamp:,extraversion:,kerneldir:,linux:,lustre:,nodownload,nosrc,ofed-version:,publish,release,set-value:,src,stage:,tag:,target:,target-archs:,with-linux:,xen -- "$@")
1687
1688 if [ $? != 0 ]; then
1689     usage 1
1690 fi
1691
1692 eval set -- "$options"
1693
1694 while [ "$1" ]; do
1695     case "$1" in
1696         '')
1697             usage 1
1698             ;;
1699         --ccache)
1700             CCACHE='ccache'
1701             shift
1702             ;;
1703         -d)
1704             CVSROOT=$2
1705             shift 2
1706             ;;
1707         -D)
1708             DATE=$2
1709             shift 2
1710             ;;
1711         --external-patches)
1712             EXTERNAL_PATCHES=$2
1713             shift 2
1714             ;;
1715         --extraversion)
1716             EXTRA_VERSION=$2
1717             shift 2
1718             ;;
1719         --help | -h)
1720             usage 0
1721             ;;
1722         --kerneldir)
1723             KERNELDIR=$2
1724             shift 2
1725             ;;
1726         --kerneltree)
1727             if ! KERNELTREE=$(canon_path "$2"); then
1728                 fatal 1 "Could not determine the canonical location of $2"
1729             fi
1730             shift 2
1731             ;;
1732         --linux | --with-linux)
1733             if ! LINUX=$(canon_path "$2"); then
1734                 fatal 1 "Could not determine the canonical location of $2"
1735             fi
1736             shift 2
1737             ;;
1738         --distro)
1739             DISTRO=$2
1740             shift 2
1741             ;;
1742         --reuserpm)
1743             REUSERPM=$2
1744             shift 2
1745             ;;
1746         --reusebuild)
1747             if ! REUSEBUILD=$(canon_path "$2"); then
1748                 fatal 1 "Could not determine the canonical location of $2"
1749             fi
1750             shift 2
1751             ;;
1752         --norpm)
1753             NORPM=true
1754             shift
1755             ;;
1756         --ldiskfs)
1757             LDISKFSRPM=true
1758             shift
1759             ;;
1760         --patchless)
1761             PATCHLESS=true
1762             shift
1763             ;;
1764         --kernelrpm)
1765             if ! KERNELRPMSBASE=$(canon_path "$2"); then
1766                 fatal 1 "Could not determine the canonical location of $2"
1767             fi
1768             shift 2
1769             ;;
1770         --timestamp)
1771             TIMESTAMP=$2
1772             shift 2
1773             ;;
1774         --lustre)
1775             if ! LUSTRE=$(canon_filepath "$2"); then
1776                 fatal 1 "Could not determine the canonical location of $2"
1777             fi
1778             shift 2
1779             ;;
1780         --nodownload)
1781             DOWNLOAD=false
1782             shift 1
1783             ;;
1784         --nosrc)
1785             DO_SRC=false
1786             shift 1
1787             ;;
1788         --ofed-version)
1789             OFED_VERSION="$2"
1790             shift 2
1791             ;;
1792         --publish)
1793             shift
1794             ;;
1795         --release)
1796             RELEASE=true
1797             shift
1798             ;;
1799         --src)
1800             DO_SRC=true
1801             shift 1
1802             ;;
1803         --stage)
1804             STAGEDIR=$2
1805             shift 2
1806             ;;
1807         --tag)
1808             TAG=$2
1809             shift 2
1810             ;;
1811         --target)
1812             TARGET=$2
1813             shift 2
1814             ;;
1815         --target-archs)
1816             TARGET_ARCHS=$2
1817             shift 2
1818             ;;
1819         --disable-datestamp)
1820             USE_DATESTAMP=
1821             shift
1822             ;;
1823         --xen)
1824             XEN=true
1825             shift
1826             ;;
1827         --set-value)
1828             eval $2
1829             shift 2
1830             ;;
1831         --)
1832             shift
1833             # there are actually some lustre configure flags that we need to
1834             # handle ourselves (but we still give them to configure)
1835             if [[ \ $@\  == *\ --disable-tests\ * ]]; then
1836                 LUSTRE_TESTS=false
1837             fi
1838             CONFIGURE_FLAGS=$@
1839             CONFIGURE_FLAGS="$CONFIGURE_FLAGS --enable-liblustre --enable-liblustre-tests"
1840             break
1841             ;;
1842         *)
1843             usage 1 "Unrecognized option: $1"
1844             ;;
1845     esac
1846 done
1847
1848 check_options
1849
1850 unpack_lustre
1851
1852 # XXX - should we _always_ get the buildid from the META file?  what are the
1853 # other (i.e. non-lustre-tarball use cases of lbuild)?
1854 BUILDID=$(sed -ne '/^BUILDID =/s/.*= *//p' lustre/META)
1855
1856 load_target
1857
1858 if [ -n "$OFED_VERSION" -a "$OFED_VERSION" != "inkernel" ]; then
1859     download_ofed
1860     unpack_ofed || fatal 1 "Error unpacking OFED tarball"
1861 fi
1862
1863 # make sure the RPM build environment is set up
1864 create_rpmbuild_dirs
1865
1866 # if an unpacked kernel source tree was given on the command line
1867 # just build lustre with it (nothing distro kernel specific here)
1868 if [ -n "$LINUX" ]; then
1869     build_ofed "${LINUXOBJ:-$LINUX}" "$OFED_VERSION" ||
1870         fatal 1 "error building OFED"
1871     build_lustre "$LINUX" "$LINUXOBJ"
1872 else
1873     if [ -f "${0%/*}/lbuild-$DISTRO" ]; then
1874         source ${0%/*}/lbuild-$DISTRO
1875
1876         build_with_srpm || fatal 1 "Failed to build_with_srpm"
1877     else
1878         EXTRA_VERSION_DELIMITER=${EXTRA_VERSION_DELIMITER:-"-"}
1879         source ${0%/*}/lbuild.old_school
1880
1881         old_school_download_kernel
1882
1883         build_success=false
1884         if $PATCHLESS; then
1885             patchless_build_sequence && build_success=true
1886         else
1887             [ "$DISTRO" = "sles9" ] && build_sequence_rpm_reuse && build_success=true
1888             if ! $build_success; then
1889                 build_sequence_reuse && build_success=true
1890                 if ! $build_success; then
1891                     build_sequence && build_success=true
1892                     if $build_success; then
1893                         store_for_reuse || echo "Cannot store for future reuse"
1894                     fi
1895                 fi
1896             fi
1897         fi
1898         ( $build_success ) || fatal 1 "Cannot build lustre"
1899     fi
1900 fi
1901
1902 stage