#!/bin/sh # # Copyright (C) 2016 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # # Rebuild the mingw64 cross-toolchain from scratch # # See --help for usage example. PROGNAME=$(basename $0) PROGDIR=$(dirname $0) PROGDIR=$(cd $PROGDIR && pwd) ANDROID_BUILD_TOP=$(realpath $PROGDIR/../..) TOOLCHAIN_DIR=$(realpath $ANDROID_BUILD_TOP/toolchain) HELP= VERBOSE=2 # This will be reset later. LOG_FILE=/dev/null panic () { 1>&2 echo "Error: $@" exit 1 } fail_panic () { if [ $? != 0 ]; then panic "$@" fi } var_value () { eval echo \"$1\" } var_append () { local _varname=$1 local _varval=$(var_value $_varname) shift if [ -z "$_varval" ]; then eval $_varname=\"$*\" else eval $_varname=\$$_varname\" $*\" fi } run () { if [ "$VERBOSE" -gt 0 ]; then echo "COMMAND: >>>> $@" >> $LOG_FILE fi if [ "$VERBOSE" -gt 1 ]; then echo "COMMAND: >>>> $@" fi if [ "$VERBOSE" -gt 1 ]; then "$@" else "$@" > /dev/null 2>&1 fi } log () { if [ "$LOG_FILE" ]; then echo "$@" >> $LOG_FILE fi if [ "$VERBOSE" -gt 0 ]; then echo "$@" fi } NUM_CORES=$(grep -c -e '^processor' /proc/cpuinfo) JOBS=$(( $NUM_CORES * 2 )) GMP_VERSION=5.0.5 MPFR_VERSION=3.1.1 MPC_VERSION=1.0.1 BINUTILS_VERSION=2.25 GCC_VERSION=4.8.3 MINGW_W64_VERSION=v5.0.0 TARGET_ARCH=x86_64 TARGET_MULTILIBS=true # not empty to enable multilib CLEANUP= OPT_GCC_VERSION= for opt; do optarg=`expr "x$opt" : 'x[^=]*=\(.*\)'` case $opt in -h|-?|--help) HELP=true;; --verbose) VERBOSE=$(( $VERBOSE + 1 ));; --quiet) VERBOSE=$(( $VERBOSE - 1 ));; -j*|--jobs=*) JOBS=$optarg;; --target-arch=*) TARGET_ARCH=$optarg;; --no-multilib) TARGET_MULTILIBS="";; --gcc-version=*) OPT_GCC_VERSION=$optarg;; --cleanup) CLEANUP=true;; -*) panic "Unknown option '$opt', see --help for list of valid ones.";; *) panic "This script doesn't take any parameter, see --help for details.";; esac done if [ "$HELP" ]; then cat <<EOF Usage: $PROGNAME [options] This program is used to rebuild a mingw64 cross-toolchain from scratch. All toolchain binaries can generate both Win32 and Win64 executables. Use -m32 or -m64 at compile/link time to select a specific target. Valid options: -h|-?|--help Print this message." --verbose Increase verbosity." --quiet Decrease verbosity." --jobs=<num> Run <num> build tasks in parallel [$JOBS]." -j<num> Same as --jobs=<num>." --no-multilib Disable multilib toolchain build." --target-arch=<arch> Select default target architecture [$TARGET_ARCH]." --gcc-version=<version> Select alternative GCC version [$GCC_VERSION]" EOF exit 0 fi if [ "$OPT_GCC_VERSION" ]; then GCC_VERSION=$OPT_GCC_VERSION fi GCC_SRC_DIR=$TOOLCHAIN_DIR/gcc/gcc-$GCC_VERSION if [ ! -d "$GCC_SRC_DIR" ]; then panic "Missing GCC source directory: $GCC_SRC_DIR" fi GCC_MAJOR_MINOR=$(echo "$GCC_VERSION" | cut -d. -f1-2) # Top level out directory. OUT_DIR=$ANDROID_BUILD_TOP/out # Name of the directory inside the package. PACKAGE_DIR=x86_64-w64-mingw32-$GCC_MAJOR_MINOR # Directory to isolate the install package from any similarly named directories. OUTER_INSTALL_DIR=$OUT_DIR/install # Actual install path. INSTALL_DIR=$OUTER_INSTALL_DIR/$PACKAGE_DIR # Install directory for build dependencies that are not in the final package # (gmp and whatnot). SUPPORT_DIR=$INSTALL_DIR # For the final artifacts. Will be archived on the build server. if [ -z "$DIST_DIR" ]; then DIST_DIR=$OUT_DIR/dist fi BUILD_TAG64=x86_64-linux-gnu BUILD_TAG32=i686-linux-gnu # We don't want debug executables BUILD_CFLAGS="-O2 -fomit-frame-pointer -s" BUILD_LDFLAGS="" rm -rf $OUT_DIR mkdir -p $OUT_DIR mkdir -p $INSTALL_DIR mkdir -p $DIST_DIR mkdir -p $SUPPORT_DIR LOG_FILE=$OUT_DIR/build.log rm -f $LOG_FILE && touch $LOG_FILE if [ "$VERBOSE" -eq 1 ]; then echo "To follow build, use in another terminal: tail -F $LOG_FILE" fi case $TARGET_ARCH in x86_64) TARGET_BITS=64;; i686) TARGET_BITS=32;; *) panic "Invalid --target parameter. Valid values are: x86_64 i686";; esac TARGET_TAG=$TARGET_ARCH-w64-mingw32 log "Target arch: $TARGET_TAG" log "Target bits: $TARGET_BITS" HOST_ARCH=x86_64 HOST_BITS=64 HOST_TAG=$HOST_ARCH-linux-gnu log "Host arch: $HOST_TAG" # Copy this script cp $0 $INSTALL_DIR/ && echo "This file has been automatically generated on $(date) with the following command:" > $INSTALL_DIR/README && echo "$PROGNAME $@" >> $INSTALL_DIR/README && echo "" >> $INSTALL_DIR/README && echo "The MD5 hashes for the original sources packages are:" >> $INSTALL_DIR/README fail_panic "Could not copy script to installation directory." PREFIX_FOR_TARGET=$INSTALL_DIR/$TARGET_TAG WITH_WIDL=$INSTALL_DIR/bin MINGW_W64_SRC=$TOOLCHAIN_DIR/mingw/mingw-w64-$MINGW_W64_VERSION setup_host_build_env () { local BINPREFIX=$ANDROID_BUILD_TOP/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.11-4.8/bin/x86_64-linux- CC=${BINPREFIX}gcc CXX=${BINPREFIX}g++ LD=${BINPREFIX}ld AS=${BINPREFIX}as AR=${BINPREFIX}ar RANLIB=${BINPREFIX}ranlib STRIP=${BINPREFIX}strip export CC CXX LD AS AR RANLIB STRIP export CFLAGS="$BUILD_CFLAGS" export CXXFLAGS="$BUILD_CFLAGS" export LDFLAGS="$BUILD_LDFLAGS" } setup_mingw_build_env () { local BINPREFIX=$INSTALL_DIR/bin/x86_64-w64-mingw32- CC=${BINPREFIX}gcc CXX=${BINPREFIX}g++ LD=${BINPREFIX}ld AS=${BINPREFIX}as AR=${BINPREFIX}ar RANLIB=${BINPREFIX}ranlib RC=${BINPREFIX}windres STRIP=${BINPREFIX}strip export CC CXX LD AS AR RANLIB RC STRIP } setup_install_env () { export PATH=$INSTALL_DIR/bin:$PATH } build_binutils_package () { local PKGNAME=$1 shift ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && setup_host_build_env && log "$PKGNAME: Configuring" && run $TOOLCHAIN_DIR/binutils/$PKGNAME/configure "$@" fail_panic "Can't configure $PKGNAME !!" log "$PKGNAME: Building" && run make -j$JOBS MAKEINFO=true fail_panic "Can't build $PKGNAME !!" log "$PKGNAME: Installing" && run make install MAKEINFO=true fail_panic "Can't install $PKGNAME" ) || exit 1 } # The GCC build in Android is insane and stores gmp and friends as tarballs and # extracts them as a part of the build step (in the meta-configure of all # places). No one understands how any of that mess works, so just deal with # extracting them here... EXTRACTED_PACKAGES=$OUT_DIR/packages mkdir -p $EXTRACTED_PACKAGES fail_panic "Could not create directory for packages." log "gmp-$GMP_VERSION: Extracting" && tar xf $TOOLCHAIN_DIR/gmp/gmp-$GMP_VERSION.tar.bz2 -C $EXTRACTED_PACKAGES log "mpfr-$MPFR_VERSION: Extracting" && tar xf $TOOLCHAIN_DIR/mpfr/mpfr-$MPFR_VERSION.tar.bz2 -C $EXTRACTED_PACKAGES log "mpc-$MPC_VERSION: Extracting" && tar xf $TOOLCHAIN_DIR/mpc/mpc-$MPC_VERSION.tar.gz -C $EXTRACTED_PACKAGES build_host_package () { local PKGNAME=$1 shift ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && setup_host_build_env && log "$PKGNAME: Configuring" && run $EXTRACTED_PACKAGES/$PKGNAME/configure "$@" fail_panic "Can't configure $PKGNAME !!" log "$PKGNAME: Building" && run make -j$JOBS fail_panic "Can't build $PKGNAME !!" log "$PKGNAME: Installing" && run make install fail_panic "Can't install $PKGNAME" ) || exit 1 } export ABI=$HOST_BITS SUPPORT_INSTALL= BASE_HOST_OPTIONS="--prefix=$SUPPORT_DIR --disable-shared" build_host_package gmp-$GMP_VERSION $BASE_HOST_OPTIONS var_append BASE_HOST_OPTIONS "--with-gmp=$SUPPORT_DIR" build_host_package mpfr-$MPFR_VERSION $BASE_HOST_OPTIONS var_append BASE_HOST_OPTIONS "--with-mpfr=$SUPPORT_DIR" build_host_package mpc-$MPC_VERSION $BASE_HOST_OPTIONS var_append BASE_HOST_OPTIONS "--with-mpc=$SUPPORT_DIR" BINUTILS_CONFIGURE_OPTIONS=$BASE_HOST_OPTIONS var_append BINUTILS_CONFIGURE_OPTIONS "--target=$TARGET_TAG --disable-nls" if [ "$TARGET_MULTILIBS" ]; then var_append BINUTILS_CONFIGURE_OPTIONS "--enable-targets=x86_64-w64-mingw32,i686-w64-mingw32" fi var_append BINUTILS_CONFIGURE_OPTIONS "--with-sysroot=$INSTALL_DIR" var_append BINUTILS_CONFIGURE_OPTIONS "--enable-lto --enable-plugin --enable-gold" build_binutils_package binutils-$BINUTILS_VERSION $BINUTILS_CONFIGURE_OPTIONS build_mingw_tools () { local PKGNAME=$1 ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && log "$PKGNAME: Configuring" && run $MINGW_W64_SRC/mingw-w64-tools/widl/configure --prefix=$INSTALL_DIR --target=$TARGET_TAG --includedir=$OUT_DIR/install/x86_64-w64-mingw32-4.8/x86_64-w64-mingw32/ fail_panic "Can't configure mingw-64-tools" log "$PKGNAME: Installing" && run make install -j$JOBS ) || exit 1 } build_mingw_pthreads_lib () { local PKGNAME=$1 local CONFIGURE_EXTRA_ARGS=$2 ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && setup_mingw_build_env && log "$PKGNAME (32-bit): Configuring" && run $MINGW_W64_SRC/mingw-w64-libraries/winpthreads/configure --prefix=$PREFIX_FOR_TARGET --target=$TARGET_TAG --host=$TARGET_TAG $CONFIGURE_EXTRA_ARGS && fail_panic "Can't configure $PKGNAME" ) || exit 1 # run it once so fakelib/libgcc.a is created and make subsequently fails # while looking for libpthread.a. Copy libgcc.a to libpthread.a and # retry. cd $OUT_DIR/$PKGNAME && run make install -j$JOBS ( cd $OUT_DIR/$PKGNAME cp fakelib/libgcc.a fakelib/libpthread.a && log "$PKGNAME: Installing" && run make install -j$JOBS ) || exit 1 } build_mingw_pthreads () { local PKGNAME=$1 ( CFLAGS="$CFLAGS -m32" CXXFLAGS="$CXXFLAGS -m32" LDFLAGS="-m32" RCFLAGS="-F pe-i386" export CFLAGS CXXFLAGS LDFLAGS RCFLAGS build_mingw_pthreads_lib $PKGNAME-32 "--build=$BUILD_TAG32 --libdir=$PREFIX_FOR_TARGET/lib32" (run cp $PREFIX_FOR_TARGET/bin/libwinpthread-1.dll $PREFIX_FOR_TARGET/lib32) || exit 1 ) build_mingw_pthreads_lib $PKGNAME-64 "--build=$BUILD_TAG64" } # Install the right mingw64 headers into the sysroot build_mingw_headers () { local PKGNAME=$1 ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && log "$PKGNAME: Configuring" && run $MINGW_W64_SRC/mingw-w64-headers/configure --prefix=$PREFIX_FOR_TARGET --host=$TARGET_TAG \ --build=$HOST_TAG --enable-sdk=all --enable-secure-api fail_panic "Can't configure mingw-64-headers" run make log "$PKGNAME: Installing" && run make install -j$JOBS && run cd $INSTALL_DIR && run ln -s $TARGET_TAG mingw && run cd $INSTALL_DIR/mingw && run ln -s lib lib$TARGET_BITS fail_panic "Can't install mingw-64-headers" ) || exit 1 } # Slightly different from build_host_package because we need to call # 'make all-gcc' and 'make install-gcc' as a special case. # build_core_gcc () { local PKGNAME=$1 shift ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && setup_host_build_env && log "core-$PKGNAME: Configuring" && run $TOOLCHAIN_DIR/gcc/$PKGNAME/configure "$@" fail_panic "Can't configure $PKGNAME !!" log "core-$PKGNAME: Building" && run make -j$JOBS all-gcc fail_panic "Can't build $PKGNAME !!" log "core-$PKGNAME: Installing" && run make -j$JOBS install-gcc fail_panic "Can't install $PKGNAME" ) || exit 1 } # Build and install the C runtime files needed by the toolchain build_mingw_crt () { local PKGNAME=$1 shift ( mkdir -p $OUT_DIR/$PKGNAME && cd $OUT_DIR/$PKGNAME && export PATH=$INSTALL_DIR/bin:$PATH log "$PKGNAME: Configuring" && run $MINGW_W64_SRC/mingw-w64-crt/configure "$@" fail_panic "Can't configure $PKGNAME !!" log "$PKGNAME: Building" && run make -j$JOBS fail_panic "Can't build $PKGNAME !!" log "$PKGNAME: Installing" && run make -j$JOBS install fail_panic "Can't install $PKGNAME" ) || exit 1 } build_libgcc () { local PKGNAME=$1 shift ( # No configure step here! We're resuming work that was started # in build_core_gcc. cd $OUT_DIR/$PKGNAME && setup_host_build_env && log "libgcc-$PKGNAME: Building" && run make -j$JOBS fail_panic "Can't build libgcc-$PKGNAME !!" log "libgcc-$PKGNAME: Installing" && run make -j$JOBS install fail_panic "Can't install libgcc-$PKGNAME" ) || exit 1 } GCC_CONFIGURE_OPTIONS=$BASE_HOST_OPTIONS var_append GCC_CONFIGURE_OPTIONS "--target=$TARGET_TAG" if [ "$TARGET_MULTILIBS" ]; then var_append GCC_CONFIGURE_OPTIONS "--enable-targets=all" fi var_append GCC_CONFIGURE_OPTIONS "--enable-languages=c,c++" var_append GCC_CONFIGURE_OPTIONS "--with-sysroot=$INSTALL_DIR" var_append GCC_CONFIGURE_OPTIONS "--enable-threads=posix" build_mingw_tools mingw-w64-tools build_mingw_headers mingw-w64-headers build_core_gcc gcc-$GCC_VERSION $GCC_CONFIGURE_OPTIONS CRT_CONFIGURE_OPTIONS="--host=$TARGET_TAG --with-sysroot=$INSTALL_DIR --prefix=$PREFIX_FOR_TARGET" if [ "$TARGET_MULTILIBS" ]; then var_append CRT_CONFIGURE_OPTIONS "--enable-lib32" fi build_mingw_crt mingw-w64-crt $CRT_CONFIGURE_OPTIONS # Build winpthreads build_mingw_pthreads mingw-w64-pthreads build_libgcc gcc-$GCC_VERSION # Let's generate the licenses/ directory LICENSE_DIRS="$SRC_DIR" var_append LICENSE_DIRS "$TOOLCHAIN_DIR/binutils/binutils-$BINUTILS_VERSION" var_append LICENSE_DIRS "$GCC_SRC_DIR" var_append LICENSE_DIRS "$EXTRACTED_PACKAGES" echo > $INSTALL_DIR/NOTICE for LICENSE in $(find $LICENSE_DIRS -name "COPYING*"); do cat $SRC_DIR/$LICENSE >> $INSTALL_DIR/NOTICE done touch $INSTALL_DIR/MODULE_LICENSE_GPL # The build server generates a repo.prop file that contains the current SHAs of # each project. REPO_PROP_PATH=$INSTALL_DIR/repo.prop if [ -f $DIST_DIR/repo.prop ]; then cp $DIST_DIR/repo.prop $REPO_PROP_PATH else # Generate our own if we're building locally. # The pushd/popd is to ensure that we're at least somewhere within our # source tree. There aren't any assumptions made about our CWD. pushd $ANDROID_BUILD_TOP repo forall \ -c 'echo $REPO_PROJECT $(git rev-parse HEAD)' > $REPO_PROP_PATH popd fi PACKAGE_NAME=$DIST_DIR/$TARGET_TAG-linux-x86_64.tar.bz2 log "Packaging $TARGET_TAG toolchain to $PACKAGE_NAME" run tar cjf $PACKAGE_NAME -C $OUTER_INSTALL_DIR $PACKAGE_DIR/ fail_panic "Could not package $TARGET_TAG toolchain!" log "Done. See $DIST_DIR:" ls -l $PACKAGE_NAME exit 0