Skip to content

Commit 6298067

Browse files
matetothpalRcColes
authored andcommitted
travis: Add FIH test scripts
Add scripts that can run instruction skip FIH tests on QEMU. Co-authored-by: Raef Coles <raef.coles@arm.com> Change-Id: Ia6da00174115e1dabaf84fdfc0e40476dc1b7a10 Signed-off-by: Mate Toth-Pal <mate.toth-pal@arm.com>
1 parent 5495f20 commit 6298067

File tree

5 files changed

+355
-20
lines changed

5 files changed

+355
-20
lines changed

ci/fih-tests_run.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,4 @@
1616

1717
set -e
1818

19-
docker run mcuboot/fih-test /bin/sh -c '/root/execute_test.sh'
19+
docker run mcuboot/fih-test /bin/sh -c '/root/execute_test.sh $0 $1 $2' 2,4,6,8,10 RELEASE SIGNATURE

ci/fih_test_docker/execute_test.sh

+26-19
Original file line numberDiff line numberDiff line change
@@ -18,43 +18,50 @@ set -e
1818

1919
WORKING_DIRECTORY=/root/work/tfm
2020
MCUBOOT_PATH=$WORKING_DIRECTORY/mcuboot
21-
22-
TFM_DIR=/root/work/tfm/trusted-firmware-m
21+
TFM_DIR=$WORKING_DIRECTORY/trusted-firmware-m
2322
TFM_BUILD_DIR=$TFM_DIR/build
24-
MCUBOOT_AXF=install/outputs/MPS2/AN521/bl2.axf
25-
SIGNED_TFM_BIN=install/outputs/MPS2/AN521/tfm_s_ns_signed.bin
26-
QEMU_LOG_FILE=qemu.log
27-
QEMU_PID_FILE=qemu_pid.txt
23+
24+
SKIP_SIZE=$1
25+
BUILD_TYPE=$2
26+
DAMAGE_TYPE=$3
27+
FIH_LEVEL=$4
2828

2929
source ~/.bashrc
3030

31+
if test -z "$FIH_LEVEL"; then
32+
# Use the default level
33+
CMAKE_FIH_LEVEL=""
34+
else
35+
CMAKE_FIH_LEVEL="-DMCUBOOT_FIH_PROFILE=\"$FIH_LEVEL\""
36+
fi
37+
3138
# build TF-M with MCUBoot
3239
mkdir -p $TFM_BUILD_DIR
3340
cd $TFM_DIR
3441
cmake -B $TFM_BUILD_DIR \
35-
-DCMAKE_BUILD_TYPE=Release \
42+
-DCMAKE_BUILD_TYPE=$BUILD_TYPE \
3643
-DTFM_TOOLCHAIN_FILE=toolchain_GNUARM.cmake \
3744
-DTFM_PLATFORM=mps2/an521 \
3845
-DTEST_NS=ON \
3946
-DTEST_S=ON \
4047
-DTFM_PSA_API=ON \
4148
-DMCUBOOT_PATH=$MCUBOOT_PATH \
4249
-DMCUBOOT_LOG_LEVEL=INFO \
50+
$CMAKE_FIH_LEVEL \
4351
.
4452
cd $TFM_BUILD_DIR
4553
make -j install
4654

47-
# Run MCUBoot and TF-M in QEMU
48-
/usr/bin/qemu-system-arm \
49-
-M mps2-an521 \
50-
-kernel $MCUBOOT_AXF \
51-
-device loader,file=$SIGNED_TFM_BIN,addr=0x10080000 \
52-
-chardev file,id=char0,path=$QEMU_LOG_FILE \
53-
-serial chardev:char0 \
54-
-display none \
55-
-pidfile $QEMU_PID_FILE \
56-
-daemonize
55+
BOOTLOADER_AXF='./install/outputs/MPS2/AN521/bl2.axf'
56+
57+
$MCUBOOT_PATH/ci/fih_test_docker/run_fi_test.sh $BOOTLOADER_AXF $SKIP_SIZE $DAMAGE_TYPE> fih_test_output.yaml
5758

58-
sleep 7
59+
echo ""
60+
echo "test finished with"
61+
echo " - BUILD_TYPE: $BUILD_TYPE"
62+
echo " - FIH_LEVEL: $FIH_LEVEL"
63+
echo " - SKIP_SIZE: $SKIP_SIZE"
64+
echo " - DAMAGE_TYPE: $DAMAGE_TYPE"
5965

60-
cat $QEMU_LOG_FILE
66+
# TODO: Create human readable output
67+
cat fih_test_output.yaml
+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
#!/bin/bash
2+
3+
# Copyright (c) 2020 Arm Limited
4+
#
5+
# Licensed under the Apache License, Version 2.0 (the "License");
6+
# you may not use this file except in compliance with the License.
7+
# You may obtain a copy of the License at
8+
#
9+
# http://www.apache.org/licenses/LICENSE-2.0
10+
#
11+
# Unless required by applicable law or agreed to in writing, software
12+
# distributed under the License is distributed on an "AS IS" BASIS,
13+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
# See the License for the specific language governing permissions and
15+
# limitations under the License.
16+
17+
OBJDUMP=arm-none-eabi-objdump
18+
GDB=gdb-multiarch
19+
20+
# Check if the ELF file specified is compatible
21+
if test $# -eq 0 || ! file $1 | grep "ELF" | grep "ARM" | grep "32" &>/dev/null; then
22+
echo "Incompatible file: $1" 1>&2
23+
exit 1
24+
fi
25+
26+
# Extract the full path
27+
AXF_PATH=$(realpath $1)
28+
#Dump all objects that have a name containing FIH_LABEL
29+
ADDRESSES=$($OBJDUMP $AXF_PATH -t | grep "FIH_LABEL")
30+
# strip all data except "address, label_name"
31+
ADDRESSES=$(echo "$ADDRESSES" | sed "s/\([[:xdigit:]]*\).*\(FIH_LABEL_FIH_CALL_[a-zA-Z]*\)_.*/0x\1, \2/g")
32+
# Sort by address in ascending order
33+
ADDRESSES=$(echo "$ADDRESSES" | sort)
34+
# In the case that there is a START followed by another START take the first one
35+
ADDRESSES=$(echo "$ADDRESSES" | sed "N;s/\(.*START.*\)\n\(.*START.*\)/\1/;P;D")
36+
# Same for END except take the second one
37+
ADDRESSES=$(echo "$ADDRESSES" | sed "N;s/\(.*END.*\)\n\(.*END.*\)/\2/;P;D")
38+
39+
# Output in CSV format with a label
40+
echo "Address, Type"
41+
echo "$ADDRESSES"

ci/fih_test_docker/fi_tester_gdb.sh

+199
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,199 @@
1+
#!/bin/bash
2+
3+
# Copyright (c) 2020 Arm Limited
4+
#
5+
# Licensed under the Apache License, Version 2.0 (the "License");
6+
# you may not use this file except in compliance with the License.
7+
# You may obtain a copy of the License at
8+
#
9+
# http://www.apache.org/licenses/LICENSE-2.0
10+
#
11+
# Unless required by applicable law or agreed to in writing, software
12+
# distributed under the License is distributed on an "AS IS" BASIS,
13+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
# See the License for the specific language governing permissions and
15+
# limitations under the License.
16+
17+
function skip_instruction {
18+
19+
local SKIP_ADDRESS=$1
20+
local SKIP_SIZE=$2
21+
22+
# Parse the ASM instruction from the address using gdb
23+
INSTR=$($GDB $AXF_FILE --batch -ex "disassemble $SKIP_ADDRESS" | grep "^ *$SKIP_ADDRESS" | sed "s/.*:[ \t]*\(.*\)$/\1/g")
24+
# Parse the C line from the address using gdb
25+
LINE=$($GDB $AXF_FILE --batch -ex "info line *$SKIP_ADDRESS" | sed "s/Line \([0-9]*\).*\"\(.*\)\".*/\2:\1/g")
26+
27+
# Sometimes an address is in the middle of a 4 byte instruction. In that case
28+
# don't run the test
29+
if test "$INSTR" == ""; then
30+
return
31+
fi
32+
33+
# Print out the meta-info about the test, in YAML
34+
echo "- skip_test:"
35+
echo " addr: $SKIP_ADDRESS"
36+
echo " asm: \"$INSTR\""
37+
echo " line: \"$LINE\""
38+
echo " skip: $SKIP_SIZE"
39+
# echo -ne "$SKIP_ADDRESS | $INSTR...\t"
40+
41+
cat >commands.gdb <<EOF
42+
target remote localhost: 1234
43+
b *$SKIP_ADDRESS
44+
continue&
45+
eval "shell sleep 0.5"
46+
interrupt
47+
if \$pc == $SKIP_ADDRESS
48+
echo "Stopped at breakpoint"
49+
else
50+
echo "Failed to stop at breakpoint"
51+
end
52+
echo "PC before increase:"
53+
print \$pc
54+
set \$pc += $SKIP_SIZE
55+
echo "PC after increase:"
56+
print \$pc
57+
detach
58+
eval "shell sleep 0.5"
59+
EOF
60+
61+
echo -n '.' 1>&2
62+
63+
# start qemu, dump the serial output to $QEMU_LOG_FILE
64+
QEMU_LOG_FILE=qemu.log
65+
QEMU_PID_FILE=qemu_pid.txt
66+
rm -f $QEMU_PID_FILE $QEMU_LOG_FILE
67+
/usr/bin/qemu-system-arm \
68+
-M mps2-an521 \
69+
-s -S \
70+
-kernel $IMAGE_DIR/bl2.axf \
71+
-device loader,file=$IMAGE_DIR/tfm_s_ns_signed.bin,addr=0x10080000 \
72+
-chardev file,id=char0,path=$QEMU_LOG_FILE \
73+
-serial chardev:char0 \
74+
-display none \
75+
-pidfile $QEMU_PID_FILE \
76+
-daemonize
77+
78+
# start qemu, skip the instruction, and continue execution
79+
$GDB < ./commands.gdb &>gdb_out.txt
80+
81+
# kill qemu
82+
kill -9 `cat $QEMU_PID_FILE`
83+
84+
# If "Secure image initializing" is seen the TFM booted, which means that a skip
85+
# managed to defeat the signature check. Write out whether the image booted or
86+
# not to the log in YAML
87+
if cat $QEMU_LOG_FILE | grep -i "Starting bootloader" &>/dev/null; then
88+
# bootloader started successfully
89+
if cat gdb_out.txt | grep -i "Stopped at breakpoint" &>/dev/null; then
90+
# The target was stopped at the desired address
91+
if cat $QEMU_LOG_FILE | grep -i "Secure image initializing" &>/dev/null; then
92+
echo " test_exec_ok: True"
93+
echo " skipped: True"
94+
echo " boot: True"
95+
96+
#print the address that was skipped, and some context to the console
97+
echo "" 1>&2
98+
echo "Boot success: address: $SKIP_ADDRESS skipped: $SKIP_SIZE" 1>&2
99+
arm-none-eabi-objdump -d $IMAGE_DIR/bl2.axf --start-address=$SKIP_ADDRESS -S | tail -n +7 | head -n 14 1>&2
100+
echo "" 1>&2
101+
echo "" 1>&2
102+
else
103+
LAST_LINE=`tail -n 1 $QEMU_LOG_FILE | tr -dc '[:print:]'`
104+
echo " test_exec_ok: True"
105+
echo " skipped: True"
106+
echo " boot: False"
107+
echo " last_line: '$LAST_LINE' "
108+
fi
109+
else
110+
# The target was not stopped at the desired address.
111+
# The most probable reason is that the instruction for that address is
112+
# on a call path that is not taken in this run (e.g. error handling)
113+
if cat $QEMU_LOG_FILE | grep -i "Secure image initializing" &>/dev/null; then
114+
# The image booted, although it shouldn't happen as the test is to
115+
# be run with a corrupt image.
116+
echo " test_exec_ok: False"
117+
echo " test_exec_fail_reason: \"No instructions were skipped (e.g. branch was not executed), but booted successfully\""
118+
else
119+
# the execution didn't stop at the address (e.g. the instruction
120+
# is on a branch that is not taken)
121+
echo " test_exec_ok: True"
122+
echo " skipped: False"
123+
fi
124+
fi
125+
else
126+
# failed before the first printout
127+
echo " test_exec_ok: True"
128+
echo " skipped: True"
129+
echo " boot: False"
130+
echo " last_line: 'N/A' "
131+
fi
132+
}
133+
134+
# Inform how the script is used
135+
usage() {
136+
echo "$0 <image_dir> <start_addr> [<end_addr>] [(-s | --skip) <skip_len>]"
137+
}
138+
139+
#defaults
140+
SKIP=2
141+
BIN_DIR=$(pwd)/install/outputs/MPS2/AN521
142+
AXF_FILE=$BIN_DIR/bl2.axf
143+
GDB=gdb-multiarch
144+
BOOTLOADER=true
145+
146+
# Parse arguments
147+
while [[ $# -gt 0 ]]; do
148+
case $1 in
149+
-s|--skip)
150+
SKIP="$2"
151+
shift
152+
shift
153+
;;
154+
-h|--help)
155+
usage
156+
exit 0
157+
;;
158+
*)
159+
if test -z "$IMAGE_DIR"; then
160+
IMAGE_DIR=$1
161+
elif test -z "$START"; then
162+
START=$1
163+
elif test -z "$END"; then
164+
END=$1
165+
else
166+
usage
167+
exit 1
168+
fi
169+
shift
170+
;;
171+
esac
172+
done
173+
174+
# Check that image directory, start and end address have been supplied
175+
if test -z "$IMAGE_DIR"; then
176+
usage
177+
exit 2
178+
fi
179+
180+
if test -z "$START"; then
181+
usage
182+
exit 2
183+
fi
184+
185+
if test -z "$END"; then
186+
END=$START
187+
fi
188+
189+
if test -z "$SKIP"; then
190+
SKIP='2'
191+
fi
192+
193+
# Create the start-end address range (step 2)
194+
ADDRS=$(printf '0x%x\n' $(seq "$START" 2 "$END"))
195+
196+
# For each address run the skip_instruction function on it
197+
for ADDR in $ADDRS; do
198+
skip_instruction $ADDR $SKIP
199+
done

0 commit comments

Comments
 (0)