1*4882a593Smuzhiyun#!/bin/sh 2*4882a593Smuzhiyun# Copyright (c) 2020 Wind River Systems, Inc. 3*4882a593Smuzhiyun# 4*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0-only 5*4882a593Smuzhiyun# 6*4882a593Smuzhiyun# buildall-qemu: a tool for automating build testing of recipes 7*4882a593Smuzhiyun# TODO: Add support for selecting which qemu architectures to build 8*4882a593Smuzhiyun# TODO: Add support for queueing up multiple recipe builds 9*4882a593Smuzhiyun# TODO: Add more logging options (e.g. local.conf info, bitbake env info) 10*4882a593Smuzhiyun 11*4882a593Smuzhiyunusage () 12*4882a593Smuzhiyun{ 13*4882a593Smuzhiyun base=$(basename "$0") 14*4882a593Smuzhiyun echo "Usage: $base [options] [recipename/target]" 15*4882a593Smuzhiyun echo "Executes a build of a given target for selected LIBCs. With no options, default to both libc and musl." 16*4882a593Smuzhiyun echo "Options:" 17*4882a593Smuzhiyun echo "-l, --libc Specify one of \"glibc\" or \"musl\"" 18*4882a593Smuzhiyun} 19*4882a593Smuzhiyun 20*4882a593Smuzhiyun 21*4882a593Smuzhiyunbuildall () 22*4882a593Smuzhiyun{ 23*4882a593Smuzhiyun # Get path to oe-core directory. Since oe-init-build-env prepends $PATH with 24*4882a593Smuzhiyun # the path to the scripts directory, get it from there 25*4882a593Smuzhiyun SCRIPTS_PATH="$(echo "$PATH" | cut -d ":" -f 1)" 26*4882a593Smuzhiyun OE_CORE_PATH=$(echo "$SCRIPTS_PATH" | sed 's|\(.*\)/.*|\1|') 27*4882a593Smuzhiyun 28*4882a593Smuzhiyun # Get target list and host machine information 29*4882a593Smuzhiyun TARGET_LIST=$(find "$OE_CORE_PATH"/meta/conf/machine -maxdepth 1 -type f | grep qemu | sed 's|.*/||' | sed -e 's/\.conf//') 30*4882a593Smuzhiyun 31*4882a593Smuzhiyun # Set LIBC value to use for the builds based on options provided by the user 32*4882a593Smuzhiyun if [ -n "$2" ] 33*4882a593Smuzhiyun then 34*4882a593Smuzhiyun LIBC_LIST="$2" 35*4882a593Smuzhiyun echo "$LIBC_LIST" 36*4882a593Smuzhiyun else 37*4882a593Smuzhiyun LIBC_LIST="glibc musl" 38*4882a593Smuzhiyun echo "$LIBC_LIST" 39*4882a593Smuzhiyun fi 40*4882a593Smuzhiyun 41*4882a593Smuzhiyun START_TIME=$(date "+%Y-%m-%d_%H:%M:%S") 42*4882a593Smuzhiyun LOG_FILE="$1-buildall.log" 43*4882a593Smuzhiyun OS_INFO=$(grep "PRETTY_NAME=" /etc/os-release | awk -F "=" '{print $2}' | sed -e 's/^"//' -e 's/"$//') 44*4882a593Smuzhiyun 45*4882a593Smuzhiyun # Append an existing log file for this build with .old if one exists 46*4882a593Smuzhiyun if [ -f "${LOG_FILE}" ] 47*4882a593Smuzhiyun then 48*4882a593Smuzhiyun mv "${LOG_FILE}" "${LOG_FILE}.old" 49*4882a593Smuzhiyun else 50*4882a593Smuzhiyun touch "${LOG_FILE}" 51*4882a593Smuzhiyun fi 52*4882a593Smuzhiyun 53*4882a593Smuzhiyun # Fill the log file with build and host info 54*4882a593Smuzhiyun echo "BUILDALL-QEMU LOG FOR $1" >> "${LOG_FILE}" 55*4882a593Smuzhiyun echo "START TIME: ${START_TIME}" >> "${LOG_FILE}" 56*4882a593Smuzhiyun echo "HOSTNAME: $(uname -n)" >> "${LOG_FILE}" 57*4882a593Smuzhiyun echo "HOST OS: ${OS_INFO}" >> "${LOG_FILE}" 58*4882a593Smuzhiyun echo "HOST KERNEL: $(uname -r)" >> "${LOG_FILE}" 59*4882a593Smuzhiyun echo "===============" >> "${LOG_FILE}" 60*4882a593Smuzhiyun echo "BUILD RESULTS:" >> "${LOG_FILE}" 61*4882a593Smuzhiyun 62*4882a593Smuzhiyun # start the builds for each MACHINE and TCLIBC 63*4882a593Smuzhiyun for j in ${LIBC_LIST} 64*4882a593Smuzhiyun do 65*4882a593Smuzhiyun echo "[$j]" >> "${LOG_FILE}" 66*4882a593Smuzhiyun for i in ${TARGET_LIST} 67*4882a593Smuzhiyun do 68*4882a593Smuzhiyun echo "$i" "$j"; \ 69*4882a593Smuzhiyun TCLIBC=$j MACHINE=$i bitbake "$1" && echo "PASS: $i" >> "${LOG_FILE}" || echo "FAIL: $i" >> "${LOG_FILE}" 70*4882a593Smuzhiyun done 71*4882a593Smuzhiyun done 72*4882a593Smuzhiyun 73*4882a593Smuzhiyun # Get pass/fail totals and add them to the end of the log 74*4882a593Smuzhiyun PASSED=$(grep "PASS:" "${LOG_FILE}" | wc -l) 75*4882a593Smuzhiyun FAILED=$(grep "FAIL:" "${LOG_FILE}" | wc -l) 76*4882a593Smuzhiyun 77*4882a593Smuzhiyun echo "===============" >> "${LOG_FILE}" 78*4882a593Smuzhiyun echo "PASSED: ${PASSED}" >> "${LOG_FILE}" 79*4882a593Smuzhiyun echo "FAILED: ${FAILED}" >> "${LOG_FILE}" 80*4882a593Smuzhiyun} 81*4882a593Smuzhiyun 82*4882a593Smuzhiyun 83*4882a593Smuzhiyun# fail entire script if any command fails 84*4882a593Smuzhiyunset -e 85*4882a593Smuzhiyun 86*4882a593Smuzhiyun# print usage and exit if not enough args given 87*4882a593Smuzhiyun[ $# -eq 0 ] && usage && exit 1 88*4882a593Smuzhiyun 89*4882a593Smuzhiyun# handle arguments 90*4882a593SmuzhiyunRECIPE= 91*4882a593Smuzhiyunwhile [ $# -gt 0 ] 92*4882a593Smuzhiyundo 93*4882a593Smuzhiyun arg=$1 94*4882a593Smuzhiyun case $arg in 95*4882a593Smuzhiyun -l|--libc) 96*4882a593Smuzhiyun if [ "$2" = "glibc" ] || [ "$2" = "musl" ] 97*4882a593Smuzhiyun then 98*4882a593Smuzhiyun LIBC_LIST="$2" 99*4882a593Smuzhiyun else 100*4882a593Smuzhiyun echo "Unrecognized libc option." 101*4882a593Smuzhiyun usage && exit 1 102*4882a593Smuzhiyun fi 103*4882a593Smuzhiyun shift 104*4882a593Smuzhiyun shift 105*4882a593Smuzhiyun ;; 106*4882a593Smuzhiyun *) 107*4882a593Smuzhiyun RECIPE="$1" 108*4882a593Smuzhiyun shift 109*4882a593Smuzhiyun ;; 110*4882a593Smuzhiyun esac 111*4882a593Smuzhiyundone 112*4882a593Smuzhiyun 113*4882a593Smuzhiyunset -- "$RECIPE" 114*4882a593Smuzhiyun 115*4882a593Smuzhiyun# run buildall for the given recipe and LIBC 116*4882a593Smuzhiyunif [ -n "$1" ] 117*4882a593Smuzhiyunthen 118*4882a593Smuzhiyun buildall "$1" "$LIBC_LIST" 119*4882a593Smuzhiyunfi 120*4882a593Smuzhiyun 121