]> CyberLeo.Net >> Repos - FreeBSD/stable/10.git/blob - release/release.sh
MFC r323275, r324112
[FreeBSD/stable/10.git] / release / release.sh
1 #!/bin/sh
2 #-
3 # Copyright (c) 2013-2017 The FreeBSD Foundation
4 # Copyright (c) 2013 Glen Barber
5 # Copyright (c) 2011 Nathan Whitehorn
6 # All rights reserved.
7 #
8 # Portions of this software were developed by Glen Barber
9 # under sponsorship from the FreeBSD Foundation.
10 #
11 # Redistribution and use in source and binary forms, with or without
12 # modification, are permitted provided that the following conditions
13 # are met:
14 # 1. Redistributions of source code must retain the above copyright
15 #    notice, this list of conditions and the following disclaimer.
16 # 2. Redistributions in binary form must reproduce the above copyright
17 #    notice, this list of conditions and the following disclaimer in the
18 #    documentation and/or other materials provided with the distribution.
19 #
20 # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21 # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 # ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24 # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 # SUCH DAMAGE.
31 #
32 # release.sh: check out source trees, and build release components with
33 #  totally clean, fresh trees.
34 # Based on release/generate-release.sh written by Nathan Whitehorn
35 #
36 # $FreeBSD$
37 #
38
39 export PATH="/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin"
40
41 VERSION=2
42
43 # Prototypes that can be redefined per-chroot or per-target.
44 load_chroot_env() { }
45 load_target_env() { }
46 buildenv_setup() { }
47
48 usage() {
49         echo "Usage: $0 [-c release.conf]"
50         exit 1
51 }
52
53 # env_setup(): Set up the default build environment variables, such as the
54 # CHROOTDIR, VCSCMD, SVNROOT, etc.  This is called before the release.conf
55 # file is sourced, if '-c <release.conf>' is specified.
56 env_setup() {
57         # The directory within which the release will be built.
58         CHROOTDIR="/scratch"
59         RELENGDIR="$(dirname $(realpath ${0}))"
60
61         # The default version control system command to obtain the sources.
62         for _dir in /usr/bin /usr/local/bin; do
63                 for _svn in svn svnlite; do
64                         [ -x "${_dir}/${_svn}" ] && VCSCMD="${_dir}/${_svn}"
65                         [ ! -z "${VCSCMD}" ] && break 2
66                 done
67         done
68         VCSCMD="${VCSCMD} checkout"
69
70         # The default svn checkout server, and svn branches for src/, doc/,
71         # and ports/.
72         SVNROOT="svn://svn.FreeBSD.org/"
73         SRCBRANCH="base/head@rHEAD"
74         DOCBRANCH="doc/head@rHEAD"
75         PORTBRANCH="ports/head@rHEAD"
76
77         # Set for embedded device builds.
78         EMBEDDEDBUILD=
79
80         # Sometimes one needs to checkout src with --force svn option.
81         # If custom kernel configs copied to src tree before checkout, e.g.
82         SRC_FORCE_CHECKOUT=
83
84         # The default make.conf and src.conf to use.  Set to /dev/null
85         # by default to avoid polluting the chroot(8) environment with
86         # non-default settings.
87         MAKE_CONF="/dev/null"
88         SRC_CONF="/dev/null"
89
90         # The number of make(1) jobs, defaults to the number of CPUs available
91         # for buildworld, and half of number of CPUs available for buildkernel.
92         WORLD_FLAGS="-j$(sysctl -n hw.ncpu)"
93         KERNEL_FLAGS="-j$(( $(( $(sysctl -n hw.ncpu) + 1 )) / 2))"
94
95         MAKE_FLAGS="-s"
96
97         # The name of the kernel to build, defaults to GENERIC.
98         KERNEL="GENERIC"
99
100         # Set to non-empty value to disable checkout of doc/ and/or ports/.
101         # Disabling ports/ checkout also forces NODOC to be set.
102         NODOC=
103         NOPORTS=
104
105         # Set to non-empty value to build dvd1.iso as part of the release.
106         WITH_DVD=
107         WITH_COMPRESSED_IMAGES=
108
109         # Set to non-empty value to build virtual machine images as part of
110         # the release.
111         WITH_VMIMAGES=
112         WITH_COMPRESSED_VMIMAGES=
113         XZ_THREADS=0
114
115         # Set to non-empty value to build virtual machine images for various
116         # cloud providers as part of the release.
117         WITH_CLOUDWARE=
118
119         return 0
120 } # env_setup()
121
122 # env_check(): Perform sanity tests on the build environment, such as ensuring
123 # files/directories exist, as well as adding backwards-compatibility hacks if
124 # necessary.  This is called unconditionally, and overrides the defaults set
125 # in env_setup() if '-c <release.conf>' is specified.
126 env_check() {
127         chroot_build_release_cmd="chroot_build_release"
128         # Fix for backwards-compatibility with release.conf that does not have
129         # the trailing '/'.
130         case ${SVNROOT} in
131                 *svn*)
132                         SVNROOT="${SVNROOT}/"
133                         ;;
134                 *)
135                         ;;
136         esac
137
138         # Prefix the branches with the SVNROOT for the full checkout URL.
139         SRCBRANCH="${SVNROOT}${SRCBRANCH}"
140         DOCBRANCH="${SVNROOT}${DOCBRANCH}"
141         PORTBRANCH="${SVNROOT}${PORTBRANCH}"
142
143         if [ -n "${EMBEDDEDBUILD}" ]; then
144                 WITH_DVD=
145                 WITH_COMPRESSED_IMAGES=
146                 NODOC=yes
147                 case ${EMBEDDED_TARGET}:${EMBEDDED_TARGET_ARCH} in
148                         arm:arm*|arm64:aarch64)
149                                 chroot_build_release_cmd="chroot_arm_build_release"
150                                 ;;
151                         *)
152                                 ;;
153                 esac
154         fi
155
156         # If PORTS is set and NODOC is unset, force NODOC=yes because the ports
157         # tree is required to build the documentation set.
158         if [ -n "${NOPORTS}" ] && [ -z "${NODOC}" ]; then
159                 echo "*** NOTICE: Setting NODOC=1 since ports tree is required"
160                 echo "            and NOPORTS is set."
161                 NODOC=yes
162         fi
163
164         # If NOPORTS and/or NODOC are unset, they must not pass to make as
165         # variables.  The release makefile verifies definedness of the
166         # NOPORTS/NODOC variables instead of their values.
167         DOCPORTS=
168         if [ -n "${NOPORTS}" ]; then
169                 DOCPORTS="NOPORTS=yes "
170         fi
171         if [ -n "${NODOC}" ]; then
172                 DOCPORTS="${DOCPORTS}NODOC=yes"
173         fi
174
175         # The aggregated build-time flags based upon variables defined within
176         # this file, unless overridden by release.conf.  In most cases, these
177         # will not need to be changed.
178         CONF_FILES="__MAKE_CONF=${MAKE_CONF} SRCCONF=${SRC_CONF}"
179         if [ -n "${TARGET}" ] && [ -n "${TARGET_ARCH}" ]; then
180                 ARCH_FLAGS="TARGET=${TARGET} TARGET_ARCH=${TARGET_ARCH}"
181         else
182                 ARCH_FLAGS=
183         fi
184         # Force src checkout if configured
185         FORCE_SRC_KEY=
186         if [ -n "${SRC_FORCE_CHECKOUT}" ]; then
187                 FORCE_SRC_KEY="--force"
188         fi
189
190         if [ -z "${CHROOTDIR}" ]; then
191                 echo "Please set CHROOTDIR."
192                 exit 1
193         fi
194
195         if [ $(id -u) -ne 0 ]; then
196                 echo "Needs to be run as root."
197                 exit 1
198         fi
199
200         CHROOT_MAKEENV="${CHROOT_MAKEENV} \
201                 MAKEOBJDIRPREFIX=${CHROOTDIR}/tmp/obj"
202         CHROOT_WMAKEFLAGS="${MAKE_FLAGS} ${WORLD_FLAGS} ${CONF_FILES}"
203         CHROOT_IMAKEFLAGS="${CONF_FILES}"
204         CHROOT_DMAKEFLAGS="${CONF_FILES}"
205         RELEASE_WMAKEFLAGS="${MAKE_FLAGS} ${WORLD_FLAGS} ${ARCH_FLAGS} \
206                 ${CONF_FILES}"
207         RELEASE_KMAKEFLAGS="${MAKE_FLAGS} ${KERNEL_FLAGS} \
208                 KERNCONF=\"${KERNEL}\" ${ARCH_FLAGS} ${CONF_FILES}"
209         RELEASE_RMAKEFLAGS="${ARCH_FLAGS} \
210                 KERNCONF=\"${KERNEL}\" ${CONF_FILES} ${DOCPORTS} \
211                 WITH_DVD=${WITH_DVD} WITH_VMIMAGES=${WITH_VMIMAGES} \
212                 WITH_CLOUDWARE=${WITH_CLOUDWARE} XZ_THREADS=${XZ_THREADS}"
213
214         return 0
215 } # env_check()
216
217 # chroot_setup(): Prepare the build chroot environment for the release build.
218 chroot_setup() {
219         load_chroot_env
220         mkdir -p ${CHROOTDIR}/usr
221
222         if [ -z "${SRC_UPDATE_SKIP}" ]; then
223                 ${VCSCMD} ${FORCE_SRC_KEY} ${SRCBRANCH} ${CHROOTDIR}/usr/src
224         fi
225         if [ -z "${NODOC}" ] && [ -z "${DOC_UPDATE_SKIP}" ]; then
226                 ${VCSCMD} ${DOCBRANCH} ${CHROOTDIR}/usr/doc
227         fi
228         if [ -z "${NOPORTS}" ] && [ -z "${PORTS_UPDATE_SKIP}" ]; then
229                 ${VCSCMD} ${PORTBRANCH} ${CHROOTDIR}/usr/ports
230         fi
231
232         if [ -z "${CHROOTBUILD_SKIP}" ]; then
233                 cd ${CHROOTDIR}/usr/src
234                 env ${CHROOT_MAKEENV} make ${CHROOT_WMAKEFLAGS} buildworld
235                 env ${CHROOT_MAKEENV} make ${CHROOT_IMAKEFLAGS} installworld \
236                         DESTDIR=${CHROOTDIR}
237                 env ${CHROOT_MAKEENV} make ${CHROOT_DMAKEFLAGS} distribution \
238                         DESTDIR=${CHROOTDIR}
239         fi
240
241         return 0
242 } # chroot_setup()
243
244 # extra_chroot_setup(): Prepare anything additional within the build
245 # necessary for the release build.
246 extra_chroot_setup() {
247         mkdir -p ${CHROOTDIR}/dev
248         mount -t devfs devfs ${CHROOTDIR}/dev
249         [ -e /etc/resolv.conf -a ! -e ${CHROOTDIR}/etc/resolv.conf ] && \
250                 cp /etc/resolv.conf ${CHROOTDIR}/etc/resolv.conf
251         # Run ldconfig(8) in the chroot directory so /var/run/ld-elf*.so.hints
252         # is created.  This is needed by ports-mgmt/pkg.
253         eval chroot ${CHROOTDIR} /etc/rc.d/ldconfig forcerestart
254
255         # If MAKE_CONF and/or SRC_CONF are set and not character devices
256         # (/dev/null), copy them to the chroot.
257         if [ -e ${MAKE_CONF} ] && [ ! -c ${MAKE_CONF} ]; then
258                 mkdir -p ${CHROOTDIR}/$(dirname ${MAKE_CONF})
259                 cp ${MAKE_CONF} ${CHROOTDIR}/${MAKE_CONF}
260         fi
261         if [ -e ${SRC_CONF} ] && [ ! -c ${SRC_CONF} ]; then
262                 mkdir -p ${CHROOTDIR}/$(dirname ${SRC_CONF})
263                 cp ${SRC_CONF} ${CHROOTDIR}/${SRC_CONF}
264         fi
265
266         if [ -d ${CHROOTDIR}/usr/ports ]; then
267                 # Trick the ports 'run-autotools-fixup' target to do the right
268                 # thing.
269                 _OSVERSION=$(chroot ${CHROOTDIR} /usr/bin/uname -U)
270                 REVISION=$(chroot ${CHROOTDIR} make -C /usr/src/release -V REVISION)
271                 BRANCH=$(chroot ${CHROOTDIR} make -C /usr/src/release -V BRANCH)
272                 UNAME_r=${REVISION}-${BRANCH}
273                 if [ -d ${CHROOTDIR}/usr/doc ] && [ -z "${NODOC}" ]; then
274                         PBUILD_FLAGS="OSVERSION=${_OSVERSION} BATCH=yes"
275                         PBUILD_FLAGS="${PBUILD_FLAGS} UNAME_r=${UNAME_r}"
276                         PBUILD_FLAGS="${PBUILD_FLAGS} OSREL=${REVISION}"
277                         chroot ${CHROOTDIR} make -C /usr/ports/textproc/docproj \
278                                 ${PBUILD_FLAGS} OPTIONS_UNSET="FOP IGOR" \
279                                 FORCE_PKG_REGISTER=1 \
280                                 install clean distclean
281                 fi
282         fi
283
284         if [ ! -z "${EMBEDDEDPORTS}" ]; then
285                 _OSVERSION=$(chroot ${CHROOTDIR} /usr/bin/uname -U)
286                 REVISION=$(chroot ${CHROOTDIR} make -C /usr/src/release -V REVISION)
287                 BRANCH=$(chroot ${CHROOTDIR} make -C /usr/src/release -V BRANCH)
288                 PBUILD_FLAGS="OSVERSION=${_OSVERSION} BATCH=yes"
289                 PBUILD_FLAGS="${PBUILD_FLAGS} UNAME_r=${UNAME_r}"
290                 PBUILD_FLAGS="${PBUILD_FLAGS} OSREL=${REVISION}"
291                 for _PORT in ${EMBEDDEDPORTS}; do
292                         eval chroot ${CHROOTDIR} make -C /usr/ports/${_PORT} \
293                                 FORCE_PKG_REGISTER=1 ${PBUILD_FLAGS} install clean distclean
294                 done
295         fi
296
297         buildenv_setup
298
299         return 0
300 } # extra_chroot_setup()
301
302 # chroot_build_target(): Build the userland and kernel for the build target.
303 chroot_build_target() {
304         load_target_env
305         if [ ! -z "${EMBEDDEDBUILD}" ]; then
306                 RELEASE_WMAKEFLAGS="${RELEASE_WMAKEFLAGS} \
307                         TARGET=${EMBEDDED_TARGET} \
308                         TARGET_ARCH=${EMBEDDED_TARGET_ARCH}"
309                 RELEASE_KMAKEFLAGS="${RELEASE_KMAKEFLAGS} \
310                         TARGET=${EMBEDDED_TARGET} \
311                         TARGET_ARCH=${EMBEDDED_TARGET_ARCH}"
312         fi
313         eval chroot ${CHROOTDIR} make -C /usr/src ${RELEASE_WMAKEFLAGS} buildworld
314         eval chroot ${CHROOTDIR} make -C /usr/src ${RELEASE_KMAKEFLAGS} buildkernel
315
316         return 0
317 } # chroot_build_target
318
319 # chroot_build_release(): Invoke the 'make release' target.
320 chroot_build_release() {
321         load_target_env
322         if [ ! -z "${WITH_VMIMAGES}" ]; then
323                 if [ -z "${VMFORMATS}" ]; then
324                         VMFORMATS="$(eval chroot ${CHROOTDIR} \
325                                 make -C /usr/src/release -V VMFORMATS)"
326                 fi
327                 if [ -z "${VMSIZE}" ]; then
328                         VMSIZE="$(eval chroot ${CHROOTDIR} \
329                                 make -C /usr/src/release -V VMSIZE)"
330                 fi
331                 RELEASE_RMAKEFLAGS="${RELEASE_RMAKEFLAGS} \
332                         VMFORMATS=\"${VMFORMATS}\" VMSIZE=${VMSIZE}"
333         fi
334         eval chroot ${CHROOTDIR} make -C /usr/src/release \
335                 ${RELEASE_RMAKEFLAGS} release
336         eval chroot ${CHROOTDIR} make -C /usr/src/release \
337                 ${RELEASE_RMAKEFLAGS} install DESTDIR=/R \
338                 WITH_COMPRESSED_IMAGES=${WITH_COMPRESSED_IMAGES} \
339                 WITH_COMPRESSED_VMIMAGES=${WITH_COMPRESSED_VMIMAGES}
340
341         return 0
342 } # chroot_build_release()
343
344 # chroot_arm_build_release(): Create arm SD card image.
345 chroot_arm_build_release() {
346         load_target_env
347         eval chroot ${CHROOTDIR} make -C /usr/src/release obj
348         case ${EMBEDDED_TARGET} in
349                 arm|arm64)
350                         if [ -e "${RELENGDIR}/tools/arm.subr" ]; then
351                                 . "${RELENGDIR}/tools/arm.subr"
352                         fi
353                         ;;
354                 *)
355                         ;;
356         esac
357         [ ! -z "${RELEASECONF}" ] && . "${RELEASECONF}"
358         WORLDDIR="$(eval chroot ${CHROOTDIR} make -C /usr/src/release -V WORLDDIR)"
359         OBJDIR="$(eval chroot ${CHROOTDIR} make -C /usr/src/release -V .OBJDIR)"
360         DESTDIR="${OBJDIR}/${KERNEL}"
361         IMGBASE="${CHROOTDIR}/${OBJDIR}/${KERNEL}.img"
362         OSRELEASE="$(eval chroot ${CHROOTDIR} make -C /usr/src/release \
363                 TARGET=${EMBEDDED_TARGET} TARGET_ARCH=${EMBEDDED_TARGET_ARCH} \
364                 -V OSRELEASE)"
365         chroot ${CHROOTDIR} mkdir -p ${DESTDIR}
366         chroot ${CHROOTDIR} truncate -s ${IMAGE_SIZE} ${IMGBASE##${CHROOTDIR}}
367         export mddev=$(chroot ${CHROOTDIR} \
368                 mdconfig -f ${IMGBASE##${CHROOTDIR}} ${MD_ARGS})
369         arm_create_disk
370         arm_install_base
371         arm_install_uboot
372         mdconfig -d -u ${mddev}
373         chroot ${CHROOTDIR} rmdir ${DESTDIR}
374         mv ${IMGBASE} ${CHROOTDIR}/${OBJDIR}/${OSRELEASE}-${KERNEL}.img
375         chroot ${CHROOTDIR} mkdir -p /R
376         chroot ${CHROOTDIR} cp -p ${OBJDIR}/${OSRELEASE}-${KERNEL}.img \
377                 /R/${OSRELEASE}-${KERNEL}.img
378         chroot ${CHROOTDIR} xz -T ${XZ_THREADS} /R/${OSRELEASE}-${KERNEL}.img
379         cd ${CHROOTDIR}/R && sha512 ${OSRELEASE}* \
380                 > CHECKSUM.SHA512
381         cd ${CHROOTDIR}/R && sha256 ${OSRELEASE}* \
382                 > CHECKSUM.SHA256
383
384         return 0
385 } # chroot_arm_build_release()
386
387 # main(): Start here.
388 main() {
389         set -e # Everything must succeed
390         env_setup
391         while getopts c: opt; do
392                 case ${opt} in
393                         c)
394                                 RELEASECONF="$(realpath ${OPTARG})"
395                                 ;;
396                         \?)
397                                 usage
398                                 ;;
399                 esac
400         done
401         shift $(($OPTIND - 1))
402         if [ ! -z "${RELEASECONF}" ]; then
403                 if [ -e "${RELEASECONF}" ]; then
404                         . ${RELEASECONF}
405                 else
406                         echo "Nonexistent configuration file: ${RELEASECONF}"
407                         echo "Using default build environment."
408                 fi
409         fi
410         env_check
411         trap "umount ${CHROOTDIR}/dev" EXIT # Clean up devfs mount on exit
412         chroot_setup
413         extra_chroot_setup
414         chroot_build_target
415         ${chroot_build_release_cmd}
416
417         return 0
418 } # main()
419
420 main "${@}"