bb90fb55d0
board_status.sh was originally written for use cases where the DUT is remote, i.e. accessed via serial port or SSH. This lead to some issues when attempting to run the script on the DUT itself. This patch attempts to handle the local use case more gracefully. sudo is used when running the cbmem command, and the '-c' option can be used to set cbmem path in case it's not in the default path used by sudo. Change-Id: I62957678ccae65fc46fd6ddf5ae92983d36cffad Signed-off-by: David Hendricks <david.hendricks@gmail.com> Reviewed-on: https://review.coreboot.org/21566 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: Martin Roth <martinroth@google.com>
474 lines
12 KiB
Bash
Executable file
474 lines
12 KiB
Bash
Executable file
#!/bin/sh
|
|
#
|
|
# This file is part of the coreboot project.
|
|
#
|
|
# Copyright (C) 2013 Google Inc.
|
|
# Copyright (C) 2014 Sage Electronic Engineering, LLC.
|
|
#
|
|
|
|
EXIT_SUCCESS=0
|
|
EXIT_FAILURE=1
|
|
|
|
# Stuff from command-line switches
|
|
COREBOOT_IMAGE="build/coreboot.rom"
|
|
REMOTE_HOST=""
|
|
REMOTE_PORT_OPTION=""
|
|
CLOBBER_OUTPUT=0
|
|
UPLOAD_RESULTS=0
|
|
SERIAL_PORT_SPEED=115200
|
|
|
|
# Used to specify whether a command should always be run locally or
|
|
# if command should be run remoteley when a remote host is specified.
|
|
LOCAL=0
|
|
REMOTE=1
|
|
FATAL=0
|
|
NONFATAL=1
|
|
|
|
# Used if cbmem is not in default $PATH, e.g. not installed or when using `sudo`
|
|
CBMEM_PATH=""
|
|
|
|
# test a command
|
|
#
|
|
# $1: 0 ($LOCAL) to run command locally,
|
|
# 1 ($REMOTE) to run remotely if remote host defined
|
|
# $2: command to test
|
|
# $3: 0 ($FATAL) Exit with an error if the command fails
|
|
# 1 ($NONFATAL) Don't exit on command test failure
|
|
test_cmd()
|
|
{
|
|
local rc
|
|
|
|
if [ -e "$2" ]; then
|
|
return
|
|
fi
|
|
|
|
if [ "$1" -eq "$REMOTE" ] && [ -n "$REMOTE_HOST" ]; then
|
|
ssh $REMOTE_PORT_OPTION root@${REMOTE_HOST} command -v "$2" > /dev/null
|
|
rc=$?
|
|
else
|
|
command -v "$2" >/dev/null
|
|
rc=$?
|
|
fi
|
|
|
|
if [ $rc -eq 0 ]; then
|
|
return 0
|
|
fi
|
|
|
|
if [ "$3" = "1" ]; then
|
|
return 1
|
|
fi
|
|
|
|
echo "$2 not found"
|
|
exit $EXIT_FAILURE
|
|
}
|
|
|
|
_cmd()
|
|
{
|
|
if [ -e "$2" ]; then
|
|
return $EXIT_FAILURE
|
|
fi
|
|
|
|
if [ -n "$3" ]; then
|
|
pipe_location="${3}"
|
|
else
|
|
pipe_location="/dev/null"
|
|
fi
|
|
|
|
if [ "$1" -eq "$REMOTE" ] && [ -n "$REMOTE_HOST" ]; then
|
|
ssh $REMOTE_PORT_OPTION "root@${REMOTE_HOST}" "$2" > "$pipe_location" 2>&1
|
|
else
|
|
$2 > "$pipe_location" 2>&1
|
|
fi
|
|
|
|
return $?
|
|
}
|
|
|
|
# run a command
|
|
#
|
|
# $1: 0 ($LOCAL) to run command locally,
|
|
# 1 ($REMOTE) to run remotely if remote host defined
|
|
# $2: command
|
|
# $3: filename to direct output of command into
|
|
cmd()
|
|
{
|
|
_cmd $1 "$2" "$3"
|
|
|
|
if [ $? -eq 0 ]; then
|
|
return
|
|
fi
|
|
|
|
echo "Failed to run \"$2\", aborting"
|
|
rm -f "$3" # don't leave an empty file
|
|
exit $EXIT_FAILURE
|
|
}
|
|
|
|
# run a command where failure is considered to be non-fatal
|
|
#
|
|
# $1: 0 ($LOCAL) to run command locally,
|
|
# 1 ($REMOTE) to run remotely if remote host defined
|
|
# $2: command
|
|
# $3: filename to direct output of command into
|
|
cmd_nonfatal()
|
|
{
|
|
_cmd $1 "$2" "$3"
|
|
|
|
if [ $? -eq 0 ]; then
|
|
return
|
|
fi
|
|
|
|
echo "Failed to run \"$2\", ignoring"
|
|
rm -f "$3" # don't leave an empty file
|
|
}
|
|
|
|
# read from a serial port device
|
|
#
|
|
# $1: serial device to read from
|
|
# $2: serial port speed
|
|
# $3: filename to direct output of command into
|
|
get_serial_bootlog () {
|
|
|
|
local TTY=$1
|
|
local SPEED=$2
|
|
local FILENAME=$3
|
|
|
|
if [ ! -c "$TTY" ]; then
|
|
echo "$TTY is not a valid serial device"
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
|
|
# make the text more noticible
|
|
test_cmd $LOCAL "tput" $NONFATAL
|
|
tput_not_available=$?
|
|
if [ $tput_not_available -eq 0 ]; then
|
|
tput bold
|
|
tput setaf 10 # set bright green
|
|
fi
|
|
|
|
echo
|
|
echo "Waiting to receive boot log from $TTY"
|
|
echo "Press [Enter] when the boot is complete."
|
|
echo
|
|
|
|
if [ $tput_not_available -eq 0 ]; then
|
|
tput sgr0
|
|
fi
|
|
|
|
# set up the serial port
|
|
stty -F $TTY $SPEED cs8 -cstopb -parenb clocal
|
|
|
|
# read from the serial port - user must press enter when complete
|
|
test_cmd $LOCAL "tee"
|
|
while read LINE; do
|
|
echo "$LINE" | tee -a "$FILENAME"
|
|
done < "$SERIAL_DEVICE" &
|
|
PID=$!
|
|
|
|
read foo
|
|
kill "$PID" 2>/dev/null
|
|
|
|
echo "Finished reading boot log."
|
|
}
|
|
|
|
show_help() {
|
|
echo "Usage:
|
|
${0} <option>
|
|
|
|
Options
|
|
-c, --cbmem
|
|
Path to cbmem on device under test (DUT).
|
|
-C, --clobber
|
|
Clobber temporary output when finished. Useful for debugging.
|
|
-h, --help
|
|
Show this message.
|
|
-i, --image <image>
|
|
Path to coreboot image (Default is $COREBOOT_IMAGE).
|
|
-r, --remote-host <host>
|
|
Obtain machine information from remote host (using ssh).
|
|
-s, --serial-device </dev/xxx>
|
|
Obtain boot log via serial device.
|
|
-S, --serial-speed <speed>
|
|
Set the port speed for the serial device (Default is $SERIAL_PORT_SPEED).
|
|
-u, --upload-results
|
|
Upload results to coreboot.org.
|
|
|
|
Long options:
|
|
--ssh-port <port>
|
|
Use a specific SSH port.
|
|
"
|
|
}
|
|
|
|
getopt -T
|
|
if [ $? -ne 4 ]; then
|
|
echo "GNU-compatible getopt(1) required."
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
|
|
LONGOPTS="cbmem:,clobber,help,image:,remote-host:,upload-results"
|
|
LONGOPTS="${LONGOPTS},serial-device:,serial-speed:"
|
|
LONGOPTS="${LONGOPTS},ssh-port:"
|
|
|
|
ARGS=$(getopt -o c:Chi:r:s:S:u -l "$LONGOPTS" -n "$0" -- "$@");
|
|
if [ $? != 0 ] ; then echo "Terminating..." >&2 ; exit 1 ; fi
|
|
eval set -- "$ARGS"
|
|
while true ; do
|
|
case "$1" in
|
|
# generic options
|
|
-c|--cbmem)
|
|
shift
|
|
CBMEM_PATH="$1"
|
|
;;
|
|
-C|--clobber)
|
|
CLOBBER_OUTPUT=1
|
|
;;
|
|
-h|--help)
|
|
show_help
|
|
exit $EXIT_SUCCESS
|
|
;;
|
|
-i|--image)
|
|
shift
|
|
COREBOOT_IMAGE="$1"
|
|
;;
|
|
-r|--remote-host)
|
|
shift
|
|
REMOTE_HOST="$1"
|
|
;;
|
|
-u|--upload-results)
|
|
UPLOAD_RESULTS=1
|
|
;;
|
|
|
|
# serial port options
|
|
-s|--serial-device)
|
|
shift
|
|
SERIAL_DEVICE="$1"
|
|
;;
|
|
-S|--serial-speed)
|
|
shift
|
|
SERIAL_PORT_SPEED="$1"
|
|
;;
|
|
|
|
# ssh options
|
|
--ssh-port)
|
|
shift
|
|
REMOTE_PORT_OPTION="-p $1"
|
|
;;
|
|
|
|
# error handling
|
|
--)
|
|
shift
|
|
if [ -n "$*" ]; then
|
|
echo "Non-option parameters detected: '$*'"
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
break
|
|
;;
|
|
*)
|
|
echo "error processing options at '$1'"
|
|
exit $EXIT_FAILURE
|
|
esac
|
|
shift
|
|
done
|
|
|
|
grep -rH 'coreboot.org' .git/config >/dev/null 2>&1
|
|
if [ $? -ne 0 ]; then
|
|
echo "Script must be run from root of coreboot directory"
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
|
|
if [ ! -e "$COREBOOT_IMAGE" ]; then
|
|
echo "board_status needs $COREBOOT_IMAGE, but it does not exist."
|
|
echo "Use \"-i IMAGE_FILE\" to select a different image, or \"--help\" for more options."
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
|
|
# Results will be placed in a temporary location until we're ready to upload.
|
|
# If the user does not wish to upload, results will remain in /tmp.
|
|
tmpdir=$(mktemp -d --tmpdir coreboot_board_status.XXXXXXXX)
|
|
|
|
# Obtain coreboot config by running cbfstool on the ROM image. cbfstool may
|
|
# already exist in build/ or util/cbfstool/, but if not then we'll build it
|
|
# now and clean it when we're done.
|
|
cbfstool_cmd="build/cbfstool"
|
|
do_clean_cbfstool=0
|
|
if [ ! -x $cbfstool_cmd ]; then
|
|
cbfstool_cmd="util/cbfstool/cbfstool"
|
|
if [ -e $cbfstool_cmd ]; then
|
|
if test ! -x $cbfstool_cmd; then
|
|
echo "Cannot execute $cbfstool_cmd."
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
else
|
|
make -C util/cbfstool/
|
|
do_clean_cbfstool=1
|
|
fi
|
|
fi
|
|
test_cmd $LOCAL "$cbfstool_cmd"
|
|
|
|
tmpcfg=$(mktemp coreboot_config.XXXXXX)
|
|
echo "Extracting config.txt from $COREBOOT_IMAGE"
|
|
$cbfstool_cmd "$COREBOOT_IMAGE" extract -n config -f "${tmpdir}/config.txt" >/dev/null 2>&1
|
|
mv "${tmpdir}/config.txt" "${tmpdir}/config.short.txt"
|
|
cp "${tmpdir}/config.short.txt" "${tmpcfg}"
|
|
yes "" | make "DOTCONFIG=${tmpcfg}" oldconfig 2>/dev/null >/dev/null
|
|
mv "${tmpcfg}" "${tmpdir}/config.txt"
|
|
rm -f "${tmpcfg}.old"
|
|
$cbfstool_cmd "$COREBOOT_IMAGE" print > "${tmpdir}/cbfs.txt"
|
|
rom_contents=$($cbfstool_cmd "$COREBOOT_IMAGE" print 2>&1)
|
|
if [ -n "$(echo $rom_contents | grep payload_config)" ]; then
|
|
echo "Extracting payload_config from $COREBOOT_IMAGE"
|
|
$cbfstool_cmd "$COREBOOT_IMAGE" extract -n payload_config -f "${tmpdir}/payload_config.txt" >/dev/null 2>&1
|
|
fi
|
|
if [ -n "$(echo $rom_contents | grep payload_version)" ]; then
|
|
echo "Extracting payload_version from $COREBOOT_IMAGE"
|
|
$cbfstool_cmd "$COREBOOT_IMAGE" extract -n payload_version -f "${tmpdir}/payload_version.txt" >/dev/null 2>&1
|
|
fi
|
|
md5sum -b "$COREBOOT_IMAGE" > "${tmpdir}/rom_checksum.txt"
|
|
|
|
if test $do_clean_cbfstool -eq 1; then
|
|
make -C util/cbfstool clean
|
|
fi
|
|
|
|
# Obtain board and revision info to form the directory structure:
|
|
# <vendor>/<board>/<revision>/<timestamp>
|
|
mainboard_dir="$(grep CONFIG_MAINBOARD_DIR "${tmpdir}/config.txt" | awk -F '"' '{ print $2 }')"
|
|
vendor=$(echo "$mainboard_dir" | awk -F '/' '{ print $1 }')
|
|
mainboard=$(echo "$mainboard_dir" | awk -F '/' '{ print $2 }')
|
|
|
|
getrevision="util/board_status/getrevision.sh"
|
|
test_cmd $LOCAL $getrevision
|
|
tagged_version=$($getrevision -T)
|
|
timestamp=$($getrevision -t)
|
|
|
|
results="${vendor}/${mainboard}/${tagged_version}/${timestamp}"
|
|
|
|
echo "Temporarily placing output in ${tmpdir}/${results}"
|
|
mkdir -p "${tmpdir}/${results}"
|
|
|
|
mv "${tmpdir}/config.txt" "${tmpdir}/${results}"
|
|
test -f "${tmpdir}/payload_config.txt" && mv "${tmpdir}/payload_config.txt" "${tmpdir}/${results}"
|
|
test -f "${tmpdir}/payload_version.txt" && mv "${tmpdir}/payload_version.txt" "${tmpdir}/${results}"
|
|
mv "${tmpdir}/config.short.txt" "${tmpdir}/${results}"
|
|
mv "${tmpdir}/cbfs.txt" "${tmpdir}/${results}"
|
|
mv "${tmpdir}/rom_checksum.txt" "${tmpdir}/${results}"
|
|
|
|
touch "${tmpdir}/${results}/revision.txt"
|
|
printf "Local revision: %s\n" "$($getrevision -l)" >> "${tmpdir}/${results}/revision.txt"
|
|
printf "Tagged revision: %s\n" "${tagged_version}" >> "${tmpdir}/${results}/revision.txt"
|
|
printf "Upstream revision: %s\n" "$($getrevision -u)" >> "${tmpdir}/${results}/revision.txt"
|
|
printf "Upstream URL: %s\n" "$($getrevision -U)" >> "${tmpdir}/${results}/revision.txt"
|
|
printf "Timestamp: %s\n" "$timestamp" >> "${tmpdir}/${results}/revision.txt"
|
|
|
|
if [ -n "$CBMEM_PATH" ]; then
|
|
cbmem_cmd="$CBMEM_PATH"
|
|
else
|
|
cbmem_cmd="cbmem"
|
|
fi
|
|
|
|
if [ -n "$SERIAL_DEVICE" ]; then
|
|
get_serial_bootlog "$SERIAL_DEVICE" "$SERIAL_PORT_SPEED" "${tmpdir}/${results}/coreboot_console.txt"
|
|
elif [ -n "$REMOTE_HOST" ]; then
|
|
echo "Verifying that CBMEM is available on remote device"
|
|
test_cmd $REMOTE "$cbmem_cmd"
|
|
echo "Getting coreboot boot log"
|
|
cmd $REMOTE "$cbmem_cmd -c" "${tmpdir}/${results}/coreboot_console.txt"
|
|
echo "Getting timestamp data"
|
|
cmd_nonfatal $REMOTE "$cbmem_cmd -t" "${tmpdir}/${results}/coreboot_timestamps.txt"
|
|
|
|
echo "Getting remote dmesg"
|
|
cmd $REMOTE dmesg "${tmpdir}/${results}/kernel_log.txt"
|
|
else
|
|
echo "Verifying that CBMEM is available"
|
|
if [ $(id -u) -ne 0 ]; then
|
|
sudo command -v "$cbmem_cmd" >/dev/null
|
|
if [ $? -ne 0 ]; then
|
|
echo "Failed to run $cbmem_cmd using sudo. Check \$PATH or use -c" \
|
|
"to specify path to cbmem binary."
|
|
exit $EXIT_FAILURE
|
|
else
|
|
cbmem_cmd="sudo $cbmem_cmd"
|
|
fi
|
|
else
|
|
test_cmd $LOCAL "$cbmem_cmd"
|
|
fi
|
|
|
|
echo "Getting coreboot boot log"
|
|
cmd $LOCAL "$cbmem_cmd -c" "${tmpdir}/${results}/coreboot_console.txt"
|
|
echo "Getting timestamp data"
|
|
cmd_nonfatal $LOCAL "$cbmem_cmd -t" "${tmpdir}/${results}/coreboot_timestamps.txt"
|
|
|
|
echo "Getting local dmesg"
|
|
cmd $LOCAL dmesg "${tmpdir}/${results}/kernel_log.txt"
|
|
fi
|
|
|
|
#
|
|
# Finish up.
|
|
#
|
|
coreboot_dir=$(pwd)
|
|
if [ $UPLOAD_RESULTS -eq 1 ]; then
|
|
# extract username from ssh://<username>@review.coreboot.org/blah
|
|
bsrepo=$(git config --get remote.origin.url | sed "s,\(.*\)/coreboot,\1/board-status,")
|
|
|
|
cd "util/board_status/"
|
|
if [ ! -e "board-status" ]; then
|
|
# FIXME: the board-status directory might get big over time.
|
|
# Is there a way we can push the results without fetching the
|
|
# whole repo?
|
|
git clone "$bsrepo"
|
|
if [ $? -ne 0 ]; then
|
|
echo "Error cloning board-status repo, aborting."
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
fi
|
|
|
|
cd "board-status"
|
|
|
|
echo "Checking for duplicate results"
|
|
# get any updates to board-status
|
|
git pull
|
|
|
|
echo "${tagged_version}" | grep dirty >/dev/null 2>&1
|
|
clean_version=$?
|
|
existing_results=$(git ls-files "${mainboard_dir}/${tagged_version}")
|
|
|
|
# reject duplicate results of non-dirty versions
|
|
if [ "${clean_version}" -eq 1 ] && [ -n "${existing_results}" ] ; then
|
|
echo "Result is a duplicate, aborting"
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
|
|
echo "Copying results to $(pwd)/${results}"
|
|
|
|
# Note: Result directory should be unique due to the timestamp.
|
|
cp -R "${tmpdir}/${vendor}" .
|
|
|
|
echo "Uploading results"
|
|
git add "${vendor}"
|
|
git commit -a -m "${mainboard_dir}/${tagged_version}/${timestamp}"
|
|
count=0
|
|
until git push origin master || test $count -eq 3; do
|
|
git pull --rebase
|
|
count=$((count + 1))
|
|
done
|
|
|
|
# Results have been uploaded so it's pointless to keep the
|
|
# temporary files around.
|
|
rm -rf "${tmpdir}"
|
|
if test $count -eq 3; then
|
|
echo "Error uploading to board-status repo, aborting."
|
|
exit $EXIT_FAILURE
|
|
fi
|
|
fi
|
|
cd "$coreboot_dir"
|
|
|
|
if [ $CLOBBER_OUTPUT -eq 1 ]; then
|
|
rm -rf "${tmpdir}"
|
|
else
|
|
if [ $UPLOAD_RESULTS -eq 1 ]; then
|
|
echo
|
|
echo "output files are in $(dirname $0)/board-status/${mainboard_dir}/${tagged_version}/${timestamp}"
|
|
else
|
|
echo
|
|
echo "output files are in ${tmpdir}/${results}"
|
|
fi
|
|
fi
|
|
|
|
exit $EXIT_SUCCESS
|