mirror of
https://github.com/MarlinFirmware/Marlin.git
synced 2025-01-07 10:11:54 +00:00
338 lines
11 KiB
Bash
Executable file
338 lines
11 KiB
Bash
Executable file
#!/usr/bin/env bash
|
|
#
|
|
# mftest Select a test to apply and build
|
|
# mftest -b [#] Build the auto-detected environment
|
|
# mftest -u [#] Upload the auto-detected environment
|
|
# mftest -tname -n# [-y] Set config options and optionally build a test
|
|
#
|
|
|
|
[[ -d Marlin/src ]] || { echo "Please 'cd' to the Marlin repo root." ; exit 1 ; }
|
|
|
|
perror() { echo -e "$0: \033[0;31m$1 -- $2\033[0m" ; }
|
|
errout() { echo -e "\033[0;31m$1\033[0m" ; }
|
|
bugout() { ((DEBUG)) && echo -e "\033[0;32m$1\033[0m" ; }
|
|
|
|
usage() {
|
|
echo "
|
|
Usage: mftest [-t|--env=<env|index>] [-n|--num=<num>] [-m|--make] [-y|--build=<Y|n>]
|
|
mftest [-a|--autobuild]
|
|
mftest [-r|--rebuild]
|
|
mftest [-s|--silent]
|
|
mftest [-u|--autoupload] [-n|--num=<num>]
|
|
|
|
OPTIONS
|
|
-t --env The environment to apply / run, or the menu index number.
|
|
-n --num The index of the test to run. (In file order.)
|
|
-m --make Use the make / Docker method for the build.
|
|
-y --build Skip 'Do you want to build this test?' and assume YES.
|
|
-h --help Print this help.
|
|
-a --autobuild PIO Build using the MOTHERBOARD environment.
|
|
-u --autoupload PIO Upload using the MOTHERBOARD environment.
|
|
-v --verbose Extra output for debugging.
|
|
-s --silent Silence build output from PlatformIO.
|
|
-d --default Restore to defaults before applying configs.
|
|
|
|
env shortcuts: tree due esp lin lp8|lpc8 lp9|lpc9 m128 m256|mega stm|f1 f4 f7 s6 teensy|t31|t32 t35|t36 t40|t41
|
|
"
|
|
}
|
|
|
|
TESTPATH=buildroot/tests
|
|
|
|
STATE_FILE="./.pio/.mftestrc"
|
|
SED=$(which gsed sed | head -n1)
|
|
|
|
shopt -s extglob nocasematch
|
|
|
|
# Matching patterns
|
|
ISNUM='^[0-9]+$'
|
|
ISRST='^(restore)_'
|
|
ISCMD='^(restore|opt|exec|use|pins|env)_'
|
|
ISEXEC='^exec_'
|
|
ISCONT='\\ *$'
|
|
|
|
# Get environment, test number, etc. from the command
|
|
TESTENV='-'
|
|
CHOICE=0
|
|
DEBUG=0
|
|
|
|
while getopts 'abdhmrsuvyn:t:-:' OFLAG; do
|
|
case "${OFLAG}" in
|
|
a) AUTO_BUILD=1 ; bugout "Auto-Build target..." ;;
|
|
d) DL_DEFAULTS=1 ; bugout "Restore to defaults..." ;;
|
|
h) EXIT_USAGE=1 ;;
|
|
m) USE_MAKE=1 ; bugout "Using make with Docker..." ;;
|
|
n) case "$OPTARG" in
|
|
*[!0-9]*) perror "option requires a number" $OFLAG ; EXIT_USAGE=2 ;;
|
|
*) CHOICE="$OPTARG" ; bugout "Got a number: $CHOICE" ;;
|
|
esac
|
|
;;
|
|
r) REBUILD=1 ; bugout "Rebuilding previous..." ;;
|
|
s) SILENT_FLAG="-s" ;;
|
|
t) TESTENV="$OPTARG" ; bugout "Got a target: $TESTENV" ;;
|
|
u) AUTO_BUILD=2 ; bugout "Auto-Upload target..." ;;
|
|
v) DEBUG=1 ; bugout "Debug ON" ;;
|
|
y) BUILD_YES='Y' ; bugout "Build will initiate..." ;;
|
|
-) IFS="=" read -r ONAM OVAL <<< "$OPTARG"
|
|
case "$ONAM" in
|
|
help) [[ -z "$OVAL" ]] || perror "option can't take value $OVAL" $ONAM ; EXIT_USAGE=1 ;;
|
|
autobuild) AUTO_BUILD=1 ; bugout "Auto-Build target..." ;;
|
|
autoupload) AUTO_BUILD=2 ; bugout "Auto-Upload target..." ;;
|
|
env) case "$OVAL" in
|
|
'') perror "option requires a value" $ONAM ; EXIT_USAGE=2 ;;
|
|
*) TESTENV="$OVAL" ; bugout "Got a target: $TESTENV" ;;
|
|
esac
|
|
;;
|
|
num) case "$OVAL" in
|
|
[0-9]+) CHOICE="$OVAL" ; bugout "Got a number: $CHOICE" ;;
|
|
*) perror "option requires a value" $ONAM ; EXIT_USAGE=2 ;;
|
|
esac
|
|
;;
|
|
rebuild) REBUILD=1 ; bugout "Rebuilding previous..." ;;
|
|
silent) SILENT_FLAG="-s" ;;
|
|
make) USE_MAKE=1 ; bugout "Using make with Docker..." ;;
|
|
debug|verbose) DEBUG=1 ; bugout "Debug ON" ;;
|
|
default) DL_DEFAULTS=1 ; bugout "Restore to defaults..." ;;
|
|
build) case "$OVAL" in
|
|
''|y|yes) BUILD_YES='Y' ;;
|
|
n|no) BUILD_YES='N' ;;
|
|
*) perror "option value must be y, n, yes, or no" $ONAM ; EXIT_USAGE=2 ;;
|
|
esac
|
|
bugout "Build will initiate? ($BUILD_YES)"
|
|
;;
|
|
*) perror "Unknown flag" "$OPTARG" ; EXIT_USAGE=2 ;;
|
|
esac
|
|
;;
|
|
*) EXIT_USAGE=2 ;;
|
|
esac
|
|
done
|
|
|
|
((EXIT_USAGE)) && { usage ; let EXIT_USAGE-- ; exit $EXIT_USAGE ; }
|
|
|
|
if ((REBUILD)); then
|
|
bugout "Rebuilding previous..."
|
|
# Build with the last-built env
|
|
[[ -f "$STATE_FILE" ]] || { errout "No previous (-r) build state found." ; exit 1 ; }
|
|
read TESTENV <"$STATE_FILE"
|
|
pio run $SILENT_FLAG -d . -e $TESTENV
|
|
exit 0
|
|
fi
|
|
|
|
case $TESTENV in
|
|
tree) pio run -d . -e include_tree ; exit 1 ;;
|
|
due) TESTENV='DUE' ;;
|
|
esp) TESTENV='esp32' ;;
|
|
lin*) TESTENV='linux_native' ;;
|
|
lp8|lpc8) TESTENV='LPC1768' ;;
|
|
lp9|lpc9) TESTENV='LPC1769' ;;
|
|
m128) TESTENV='mega1280' ;;
|
|
m256) TESTENV='mega2560' ;;
|
|
mega) TESTENV='mega2560' ;;
|
|
stm) TESTENV='STM32F103RE' ;;
|
|
f1) TESTENV='STM32F103RE' ;;
|
|
f4) TESTENV='STM32F4' ;;
|
|
f7) TESTENV='STM32F7' ;;
|
|
s6) TESTENV='FYSETC_S6' ;;
|
|
teensy) TESTENV='teensy31' ;;
|
|
t31) TESTENV='teensy31' ;;
|
|
t32) TESTENV='teensy31' ;;
|
|
t35) TESTENV='teensy35' ;;
|
|
t36) TESTENV='teensy35' ;;
|
|
t40) TESTENV='teensy41' ;;
|
|
t41) TESTENV='teensy41' ;;
|
|
[1-9]|[1-9][0-9]) TESTNUM=$TESTENV ; TESTENV=- ;;
|
|
esac
|
|
|
|
if ((AUTO_BUILD)); then
|
|
#
|
|
# List environments that apply to the current MOTHERBOARD.
|
|
#
|
|
case $(uname | tr '[:upper:]' '[:lower:]') in
|
|
darwin) SYS='mac' ;;
|
|
*linux) SYS='lin' ;;
|
|
win*) SYS='win' ;;
|
|
msys*) SYS='win' ;;
|
|
cygwin*) SYS='win' ;;
|
|
mingw*) SYS='win' ;;
|
|
*) SYS='uni' ;;
|
|
esac
|
|
echo ; echo -n "Auto " ; ((AUTO_BUILD == 2)) && echo "Upload..." || echo "Build..."
|
|
MB=$( grep -E "^\s*#define MOTHERBOARD" Marlin/Configuration.h | awk '{ print $3 }' | $SED 's/BOARD_//;s/\r//' )
|
|
[[ -z $MB ]] && { echo "Error - Can't read MOTHERBOARD setting." ; exit 1 ; }
|
|
BLINE=$( grep -E "define\s+BOARD_$MB\b" Marlin/src/core/boards.h )
|
|
BNUM=$( $SED -E 's/^.+BOARD_[^ ]+ +([0-9]+).+$/\1/' <<<"$BLINE" )
|
|
BDESC=$( $SED -E 's/^.+\/\/ *(.+)$/\1/' <<<"$BLINE" )
|
|
[[ -z $BNUM ]] && { echo "Error - Can't find BOARD_$MB in core/boards.h." ; exit 1 ; }
|
|
ENVS=( $( grep -EA1 "MB\(.*\b$MB\b.*\)" Marlin/src/pins/pins.h | grep -E "#include.+//.+(env|$SYS):[^ ]+" | grep -oE "(env|$SYS):[^ ]+" | $SED -E "s/(env|$SYS)://" ) )
|
|
[[ -z $ENVS ]] && { errout "Error - Can't find target(s) for $MB ($BNUM)." ; exit 1 ; }
|
|
ECOUNT=${#ENVS[*]}
|
|
|
|
if [[ $ECOUNT == 1 ]]; then
|
|
TARGET=$ENVS
|
|
else
|
|
if [[ $CHOICE == 0 ]]; then
|
|
# List env names and numbers. Get selection.
|
|
echo "Available targets for \"$BDESC\" | $MB ($BNUM):"
|
|
|
|
IND=0 ; for ENV in "${ENVS[@]}"; do let IND++ ; echo " $IND) $ENV" ; done
|
|
|
|
if [[ $ECOUNT > 1 ]]; then
|
|
for (( ; ; ))
|
|
do
|
|
read -p "Select a target for '$MB' (1-$ECOUNT) : " CHOICE
|
|
[[ -z "$CHOICE" ]] && { echo '(canceled)' ; exit 1 ; }
|
|
[[ $CHOICE =~ $ISNUM ]] && ((CHOICE >= 1 && CHOICE <= ECOUNT)) && break
|
|
errout ">>> Invalid environment choice '$CHOICE'."
|
|
done
|
|
echo
|
|
fi
|
|
else
|
|
echo "Detected \"$BDESC\" | $MB ($BNUM)."
|
|
[[ $CHOICE > $ECOUNT ]] && { echo "Environment selection out of range." ; exit 1 ; }
|
|
fi
|
|
TARGET="${ENVS[$CHOICE-1]}"
|
|
echo "Selected $TARGET"
|
|
fi
|
|
|
|
echo "$TARGET" >"$STATE_FILE"
|
|
|
|
if ((AUTO_BUILD == 2)); then
|
|
echo "Uploading environment $TARGET for board $MB ($BNUM)..." ; echo
|
|
pio run $SILENT_FLAG -t upload -e $TARGET
|
|
else
|
|
echo "Building environment $TARGET for board $MB ($BNUM)..." ; echo
|
|
pio run $SILENT_FLAG -e $TARGET
|
|
fi
|
|
exit $?
|
|
fi
|
|
|
|
#
|
|
# List available tests and ask for selection
|
|
#
|
|
|
|
if [[ $TESTENV == '-' ]]; then
|
|
IND=0
|
|
NAMES=()
|
|
MENU=()
|
|
BIGLEN=0
|
|
for FILE in $( ls -1 $TESTPATH/* | sort -f )
|
|
do
|
|
let IND++
|
|
TNAME=${FILE/$TESTPATH\//}
|
|
NAMES+=($TNAME)
|
|
IFS=""
|
|
ITEM=$( printf "%2i) %s" $IND $TNAME )
|
|
MENU+=($ITEM)
|
|
[[ ${#ITEM} -gt $BIGLEN ]] && BIGLEN=${#ITEM}
|
|
done
|
|
|
|
(( BIGLEN += 2 ))
|
|
THIRD=$(( (${#MENU[@]} + 2) / 3 ))
|
|
for ((i = 0; i < $THIRD; i++))
|
|
do
|
|
COL1=$i ; COL2=$(( $i + $THIRD )) ; COL3=$(( $i + 2 * $THIRD ))
|
|
FMT="%-${BIGLEN}s"
|
|
printf "${FMT}${FMT}${FMT}\n" ${MENU[$COL1]} ${MENU[$COL2]} ${MENU[$COL3]}
|
|
done
|
|
|
|
echo
|
|
for (( ; ; ))
|
|
do
|
|
if [[ $TESTNUM -gt 0 ]]; then
|
|
NAMEIND=$TESTNUM
|
|
else
|
|
read -p "Select a test to apply (1-$IND) : " NAMEIND
|
|
fi
|
|
[[ -z $NAMEIND ]] && { errout "(canceled)" ; exit 1 ; }
|
|
TESTENV=${NAMES[$NAMEIND-1]}
|
|
[[ $TESTNUM -gt 0 ]] && { echo "Preselected test $TESTNUM ... ($TESTENV)" ; TESTNUM='' ; }
|
|
[[ $NAMEIND =~ $ISNUM ]] && ((NAMEIND >= 1 && NAMEIND <= IND)) && { TESTENV=${NAMES[$NAMEIND-1]} ; echo ; break ; }
|
|
errout "Invalid selection."
|
|
done
|
|
fi
|
|
|
|
# Get the contents of the test file
|
|
OUT=$( cat $TESTPATH/$TESTENV 2>/dev/null ) || { errout "Can't find test '$TESTENV'." ; exit 1 ; }
|
|
|
|
# Count up the number of tests
|
|
TESTCOUNT=$( awk "/$ISEXEC/{a++}END{print a}" <<<"$OUT" )
|
|
|
|
# User entered a number?
|
|
(( CHOICE && CHOICE > TESTCOUNT )) && { errout "Invalid test selection '$CHOICE' (1-$TESTCOUNT)." ; exit 1 ; }
|
|
|
|
if [[ $CHOICE == 0 ]]; then
|
|
#
|
|
# List test descriptions with numbers and get selection
|
|
#
|
|
echo "Available '$TESTENV' tests:" ; echo "$OUT" | {
|
|
IND=0
|
|
while IFS= read -r LINE
|
|
do
|
|
if [[ $LINE =~ $ISEXEC ]]; then
|
|
DESC=$( "$SED" -E 's/^exec_test \$1 \$2 "([^"]+)".*$/\1/g' <<<"$LINE" )
|
|
(( ++IND < 10 )) && echo -n " "
|
|
echo " $IND) $DESC"
|
|
fi
|
|
done
|
|
}
|
|
CHOICE=1
|
|
if [[ $TESTCOUNT > 1 ]]; then
|
|
for (( ; ; ))
|
|
do
|
|
read -p "Select a '$TESTENV' test (1-$TESTCOUNT) : " CHOICE
|
|
[[ -z "$CHOICE" ]] && { errout "(canceled)" ; exit 1 ; }
|
|
[[ $CHOICE =~ $ISNUM ]] && ((CHOICE >= 1 && CHOICE <= TESTCOUNT)) && break
|
|
errout ">>> Invalid test selection '$CHOICE'."
|
|
done
|
|
fi
|
|
fi
|
|
|
|
#
|
|
# Restore to defaults if requested
|
|
#
|
|
((DL_DEFAULTS)) && use_example_configs
|
|
|
|
#
|
|
# Run the specified test lines
|
|
#
|
|
echo -ne "\033[0;33m"
|
|
echo "$OUT" | {
|
|
IND=0
|
|
GOTX=0
|
|
CMD=""
|
|
while IFS= read -r LINE
|
|
do
|
|
if [[ $LINE =~ $ISCMD || $GOTX == 1 ]]; then
|
|
((!IND)) && let IND++
|
|
if [[ $LINE =~ $ISEXEC ]]; then
|
|
((IND++ > CHOICE)) && break
|
|
else
|
|
((!HEADER)) && {
|
|
HEADER=1
|
|
echo -e "\n#\n# Test $TESTENV ($CHOICE) $DESC\n#"
|
|
}
|
|
((IND == CHOICE)) && {
|
|
GOTX=1
|
|
[[ -n $DL_DEFAULTS && $LINE =~ $ISRST ]] && LINE="use_example_configs"
|
|
[[ $CMD == "" ]] && CMD="$LINE" || CMD=$( echo -e "$CMD$LINE" | $SED -e 's/\\//g' | $SED -E 's/ +/ /g' )
|
|
[[ $LINE =~ $ISCONT ]] || { echo "$CMD" ; eval "$CMD" ; CMD="" ; }
|
|
}
|
|
fi
|
|
fi
|
|
done
|
|
}
|
|
echo -ne "\033[0m"
|
|
|
|
# Make clear it's a TEST
|
|
opt_set CUSTOM_MACHINE_NAME "\"Test $TESTENV ($CHOICE)\""
|
|
|
|
# Build the test too?
|
|
if [[ -z "$BUILD_YES" ]]; then
|
|
echo
|
|
read -p "Build $TESTENV test #$CHOICE (y/N) ? " BUILD_YES
|
|
fi
|
|
|
|
[[ $BUILD_YES == 'Y' || $BUILD_YES == 'Yes' ]] && {
|
|
((USE_MAKE)) && make tests-single-local TEST_TARGET=$TESTENV ONLY_TEST=$CHOICE
|
|
((USE_MAKE)) || pio run $SILENT_FLAG -d . -e $TESTENV
|
|
echo "$TESTENV" >"$STATE_FILE"
|
|
}
|