xref: /qemu/tests/qemu-iotests/check (revision 138ca49a)
1#!/usr/bin/env bash
2#
3# Copyright (C) 2009 Red Hat, Inc.
4# Copyright (c) 2000-2002,2006 Silicon Graphics, Inc.  All Rights Reserved.
5#
6# This program is free software; you can redistribute it and/or
7# modify it under the terms of the GNU General Public License as
8# published by the Free Software Foundation.
9#
10# This program is distributed in the hope that it would be useful,
11# but WITHOUT ANY WARRANTY; without even the implied warranty of
12# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13# GNU General Public License for more details.
14#
15# You should have received a copy of the GNU General Public License
16# along with this program.  If not, see <http://www.gnu.org/licenses/>.
17#
18#
19# Control script for QA
20#
21
22status=0
23needwrap=true
24try=0
25n_bad=0
26bad=""
27notrun=""
28casenotrun=""
29interrupt=true
30makecheck=false
31
32_init_error()
33{
34    echo "check: $1" >&2
35    exit 1
36}
37
38if [ -L "$0" ]
39then
40    # called from the build tree
41    source_iotests=$(dirname "$(readlink "$0")")
42    if [ -z "$source_iotests" ]
43    then
44        _init_error "failed to obtain source tree name from check symlink"
45    fi
46    source_iotests=$(cd "$source_iotests"; pwd) || _init_error "failed to enter source tree"
47    build_iotests=$(cd "$(dirname "$0")"; pwd)
48else
49    # called from the source tree
50    source_iotests=$PWD
51    # this may be an in-tree build (note that in the following code we may not
52    # assume that it truly is and have to test whether the build results
53    # actually exist)
54    build_iotests=$PWD
55fi
56
57build_root="$build_iotests/../.."
58
59# we need common.env
60if ! . "$build_iotests/common.env"
61then
62    _init_error "failed to source common.env (make sure the qemu-iotests are run from tests/qemu-iotests in the build tree)"
63fi
64
65# we need common.config
66if ! . "$source_iotests/common.config"
67then
68    _init_error "failed to source common.config"
69fi
70
71_full_imgfmt_details()
72{
73    if [ -n "$IMGOPTS" ]; then
74        echo "$IMGFMT ($IMGOPTS)"
75    else
76        echo "$IMGFMT"
77    fi
78}
79
80_full_platform_details()
81{
82    os=$(uname -s)
83    host=$(hostname -s)
84    kernel=$(uname -r)
85    platform=$(uname -m)
86    echo "$os/$platform $host $kernel"
87}
88
89_full_env_details()
90{
91    cat <<EOF
92QEMU          -- "$QEMU_PROG" $QEMU_OPTIONS
93QEMU_IMG      -- "$QEMU_IMG_PROG" $QEMU_IMG_OPTIONS
94QEMU_IO       -- "$QEMU_IO_PROG" $QEMU_IO_OPTIONS
95QEMU_NBD      -- "$QEMU_NBD_PROG" $QEMU_NBD_OPTIONS
96IMGFMT        -- $FULL_IMGFMT_DETAILS
97IMGPROTO      -- $IMGPROTO
98PLATFORM      -- $FULL_HOST_DETAILS
99TEST_DIR      -- $TEST_DIR
100SOCK_DIR      -- $SOCK_DIR
101SOCKET_SCM_HELPER -- $SOCKET_SCM_HELPER
102
103EOF
104}
105
106# $1 = prog to look for
107set_prog_path()
108{
109    p=$(command -v $1 2> /dev/null)
110    if [ -n "$p" -a -x "$p" ]; then
111        type -p "$p"
112    else
113        return 1
114    fi
115}
116
117if [ -z "$TEST_DIR" ]; then
118        TEST_DIR=$PWD/scratch
119fi
120mkdir -p "$TEST_DIR" || _init_error 'Failed to create TEST_DIR'
121
122tmp_sock_dir=false
123if [ -z "$SOCK_DIR" ]; then
124    SOCK_DIR=$(mktemp -d)
125    tmp_sock_dir=true
126fi
127mkdir -p "$SOCK_DIR" || _init_error 'Failed to create SOCK_DIR'
128
129diff="diff -u"
130verbose=false
131debug=false
132group=false
133xgroup=false
134imgopts=false
135showme=false
136sortme=false
137expunge=true
138have_test_arg=false
139cachemode=false
140aiomode=false
141
142tmp="${TEST_DIR}"/$$
143rm -f $tmp.list $tmp.tmp $tmp.sed
144
145export IMGFMT=raw
146export IMGFMT_GENERIC=true
147export IMGPROTO=file
148export IMGOPTS=""
149export CACHEMODE="writeback"
150export AIOMODE="threads"
151export QEMU_IO_OPTIONS=""
152export QEMU_IO_OPTIONS_NO_FMT=""
153export CACHEMODE_IS_DEFAULT=true
154export VALGRIND_QEMU=
155export IMGKEYSECRET=
156export IMGOPTSSYNTAX=false
157
158# Save current tty settings, since an aborting qemu call may leave things
159# screwed up
160STTY_RESTORE=
161if test -t 0; then
162    STTY_RESTORE=$(stty -g)
163fi
164
165for r
166do
167
168    if $group
169    then
170        # arg after -g
171        group_list=$(sed -n <"$source_iotests/group" -e 's/$/ /' -e "/^[0-9][0-9][0-9].* $r /"'{
172s/ .*//p
173}')
174        if [ -z "$group_list" ]
175        then
176            echo "Group \"$r\" is empty or not defined?"
177            exit 1
178        fi
179        [ ! -s $tmp.list ] && touch $tmp.list
180        for t in $group_list
181        do
182            if grep -s "^$t\$" $tmp.list >/dev/null
183            then
184                :
185            else
186                echo "$t" >>$tmp.list
187            fi
188        done
189        group=false
190        continue
191
192    elif $xgroup
193    then
194        # arg after -x
195        # Populate $tmp.list with all tests
196        awk '/^[0-9]{3,}/ {print $1}' "${source_iotests}/group" > $tmp.list 2>/dev/null
197        group_list=$(sed -n <"$source_iotests/group" -e 's/$/ /' -e "/^[0-9][0-9][0-9].* $r /"'{
198s/ .*//p
199}')
200        if [ -z "$group_list" ]
201        then
202            echo "Group \"$r\" is empty or not defined?"
203            exit 1
204        fi
205        numsed=0
206        rm -f $tmp.sed
207        for t in $group_list
208        do
209            if [ $numsed -gt 100 ]
210            then
211                sed -f $tmp.sed <$tmp.list >$tmp.tmp
212                mv $tmp.tmp $tmp.list
213                numsed=0
214                rm -f $tmp.sed
215            fi
216            echo "/^$t\$/d" >>$tmp.sed
217            numsed=$(expr $numsed + 1)
218        done
219        sed -f $tmp.sed <$tmp.list >$tmp.tmp
220        mv $tmp.tmp $tmp.list
221        xgroup=false
222        continue
223
224    elif $imgopts
225    then
226        IMGOPTS="$r"
227        imgopts=false
228        continue
229    elif $cachemode
230    then
231        CACHEMODE="$r"
232        CACHEMODE_IS_DEFAULT=false
233        cachemode=false
234        continue
235    elif $aiomode
236    then
237        AIOMODE="$r"
238        aiomode=false
239        continue
240    fi
241
242    xpand=true
243    case "$r"
244    in
245
246        -\? | -h | --help)        # usage
247            echo "Usage: $0 [options] [testlist]"'
248
249common options
250    -v                  verbose
251    -d                  debug
252
253image format options
254    -raw                test raw (default)
255    -bochs              test bochs
256    -cloop              test cloop
257    -parallels          test parallels
258    -qcow               test qcow
259    -qcow2              test qcow2
260    -qed                test qed
261    -vdi                test vdi
262    -vpc                test vpc
263    -vhdx               test vhdx
264    -vmdk               test vmdk
265    -luks               test luks
266    -dmg                test dmg
267
268image protocol options
269    -file               test file (default)
270    -rbd                test rbd
271    -sheepdog           test sheepdog
272    -nbd                test nbd
273    -fuse               test fuse
274    -ssh                test ssh
275    -nfs                test nfs
276
277other options
278    -xdiff              graphical mode diff
279    -nocache            use O_DIRECT on backing file
280    -misalign           misalign memory allocations
281    -n                  show me, do not run tests
282    -o options          -o options to pass to qemu-img create/convert
283    -c mode             cache mode
284    -i mode             AIO mode
285    -makecheck          pretty print output for make check
286
287testlist options
288    -g group[,group...]        include tests from these groups
289    -x group[,group...]        exclude tests from these groups
290    NNN                        include test NNN
291    NNN-NNN                    include test range (eg. 012-021)
292'
293            exit 0
294            ;;
295
296        -raw)
297            IMGFMT=raw
298            xpand=false
299            ;;
300
301        -bochs)
302            IMGFMT=bochs
303            IMGFMT_GENERIC=false
304            xpand=false
305            ;;
306
307        -cloop)
308            IMGFMT=cloop
309            IMGFMT_GENERIC=false
310            xpand=false
311            ;;
312
313        -parallels)
314            IMGFMT=parallels
315            xpand=false
316            ;;
317
318        -qcow)
319            IMGFMT=qcow
320            xpand=false
321            ;;
322
323        -qcow2)
324            IMGFMT=qcow2
325            xpand=false
326            ;;
327
328        -luks)
329            IMGOPTSSYNTAX=true
330            IMGFMT=luks
331            IMGKEYSECRET=123456
332            xpand=false
333            ;;
334
335        -dmg)
336            IMGFMT=dmg
337            IMGFMT_GENERIC=false
338            xpand=false
339            ;;
340
341        -qed)
342            IMGFMT=qed
343            xpand=false
344            ;;
345
346        -vdi)
347            IMGFMT=vdi
348            xpand=false
349            ;;
350
351        -vmdk)
352            IMGFMT=vmdk
353            xpand=false
354            ;;
355
356        -vpc)
357            IMGFMT=vpc
358            xpand=false
359            ;;
360
361        -vhdx)
362            IMGFMT=vhdx
363            xpand=false
364            ;;
365
366        -file)
367            IMGPROTO=file
368            xpand=false
369            ;;
370
371        -rbd)
372            IMGPROTO=rbd
373            xpand=false
374            ;;
375
376        -sheepdog)
377            IMGPROTO=sheepdog
378            xpand=false
379            ;;
380
381        -nbd)
382            IMGPROTO=nbd
383            xpand=false
384            ;;
385
386        -fuse)
387            IMGPROTO=fuse
388            xpand=false
389            ;;
390
391        -ssh)
392            IMGPROTO=ssh
393            xpand=false
394            ;;
395
396        -nfs)
397            IMGPROTO=nfs
398            xpand=false
399            ;;
400
401        -nocache)
402            CACHEMODE="none"
403            CACHEMODE_IS_DEFAULT=false
404            xpand=false
405            ;;
406
407        -misalign)
408            QEMU_IO_OPTIONS="$QEMU_IO_OPTIONS --misalign"
409            xpand=false
410            ;;
411
412        -valgrind)
413            VALGRIND_QEMU='y'
414            xpand=false
415            ;;
416
417        -g)        # -g group ... pick from group file
418            group=true
419            xpand=false
420            ;;
421
422        -xdiff)        # graphical diff mode
423            xpand=false
424
425            if [ ! -z "$DISPLAY" ]
426            then
427                command -v xdiff >/dev/null 2>&1 && diff=xdiff
428                command -v gdiff >/dev/null 2>&1 && diff=gdiff
429                command -v tkdiff >/dev/null 2>&1 && diff=tkdiff
430                command -v xxdiff >/dev/null 2>&1 && diff=xxdiff
431            fi
432            ;;
433        -makecheck)   # makecheck friendly output
434            makecheck=true
435            xpand=false
436            ;;
437        -n)        # show me, don't do it
438            showme=true
439            xpand=false
440            ;;
441        -o)
442            imgopts=true
443            xpand=false
444            ;;
445        -c)
446            cachemode=true
447            xpand=false
448            ;;
449        -i)
450            aiomode=true
451            xpand=false
452            ;;
453        -T)        # deprecated timestamp option
454            xpand=false
455            ;;
456        -v)
457            verbose=true
458            xpand=false
459            ;;
460        -d)
461            debug=true
462            xpand=false
463            ;;
464        -x)        # -x group ... exclude from group file
465            xgroup=true
466            xpand=false
467            ;;
468        '[0-9][0-9][0-9] [0-9][0-9][0-9][0-9]')
469            echo "No tests?"
470            status=1
471            exit $status
472            ;;
473
474        [0-9]*-[0-9]*)
475            eval $(echo $r | sed -e 's/^/start=/' -e 's/-/ end=/')
476            ;;
477
478        [0-9]*-)
479            eval $(echo $r | sed -e 's/^/start=/' -e 's/-//')
480            end=$(echo [0-9][0-9][0-9] [0-9][0-9][0-9][0-9] | sed -e 's/\[0-9]//g' -e 's/  *$//' -e 's/.* //')
481            if [ -z "$end" ]
482            then
483                echo "No tests in range \"$r\"?"
484                status=1
485                exit $status
486            fi
487            ;;
488
489        *)
490            start=$r
491            end=$r
492            ;;
493
494    esac
495
496    # get rid of leading 0s as can be interpreted as octal
497    start=$(echo $start | sed 's/^0*//')
498    end=$(echo $end | sed 's/^0*//')
499
500    if $xpand
501    then
502        have_test_arg=true
503        awk </dev/null '
504BEGIN        { for (t='$start'; t<='$end'; t++) printf "%03d\n",t }' \
505        | while read id
506        do
507            if grep -s "^$id\( \|\$\)" "$source_iotests/group" >/dev/null
508            then
509                # in group file ... OK
510                echo $id >>$tmp.list
511            else
512                if [ -f expunged ] && $expunge && egrep "^$id([         ]|\$)" expunged >/dev/null
513                then
514                    # expunged ... will be reported, but not run, later
515                    echo $id >>$tmp.list
516                else
517                    # oops
518                    if [ "$start" == "$end" -a "$id" == "$end" ]
519                    then
520                        echo "$id - unknown test"
521                        exit 1
522                    else
523                        echo "$id - unknown test, ignored"
524                    fi
525                fi
526            fi
527        done || exit 1
528    fi
529
530done
531
532# Set qemu-io cache mode with $CACHEMODE we have
533QEMU_IO_OPTIONS="$QEMU_IO_OPTIONS --cache $CACHEMODE"
534# Set qemu-io aio mode with $AIOMODE we have
535QEMU_IO_OPTIONS="$QEMU_IO_OPTIONS --aio $AIOMODE"
536
537QEMU_IO_OPTIONS_NO_FMT="$QEMU_IO_OPTIONS"
538if [ "$IMGOPTSSYNTAX" != "true" ]; then
539    QEMU_IO_OPTIONS="$QEMU_IO_OPTIONS -f $IMGFMT"
540fi
541
542# Set default options for qemu-img create -o if they were not specified
543if [ "$IMGFMT" == "qcow2" ] && ! (echo "$IMGOPTS" | grep "compat=" > /dev/null); then
544    IMGOPTS=$(_optstr_add "$IMGOPTS" "compat=1.1")
545fi
546if [ "$IMGFMT" == "luks" ] && ! (echo "$IMGOPTS" | grep "iter-time=" > /dev/null); then
547    IMGOPTS=$(_optstr_add "$IMGOPTS" "iter-time=10")
548fi
549if [ "$IMGFMT" == "vmdk" ] && ! (echo "$IMGOPTS" | grep "zeroed_grain=" > /dev/null); then
550    IMGOPTS=$(_optstr_add "$IMGOPTS" "zeroed_grain=on")
551fi
552
553if [ -z "$SAMPLE_IMG_DIR" ]; then
554        SAMPLE_IMG_DIR="$source_iotests/sample_images"
555fi
556
557export TEST_DIR
558export SOCK_DIR
559export SAMPLE_IMG_DIR
560
561if [ -s $tmp.list ]
562then
563    # found some valid test numbers ... this is good
564    :
565else
566    if $have_test_arg
567    then
568        # had test numbers, but none in group file ... do nothing
569        touch $tmp.list
570    else
571        # no test numbers, do everything from group file
572        sed -n -e '/^[0-9][0-9][0-9]*/s/^\([0-9]*\).*/\1/p' <"$source_iotests/group" >$tmp.list
573    fi
574fi
575
576# should be sort -n, but this did not work for Linux when this
577# was ported from IRIX
578#
579list=$(sort $tmp.list)
580rm -f $tmp.list $tmp.tmp $tmp.sed
581
582if [ -z "$QEMU_PROG" ]
583then
584    if [ -x "$build_iotests/qemu" ]; then
585        export QEMU_PROG="$build_iotests/qemu"
586    elif [ -x "$build_root/qemu-system-${qemu_arch}" ]; then
587        export QEMU_PROG="$build_root/qemu-system-${qemu_arch}"
588    else
589        pushd "$build_root" > /dev/null
590        for binary in qemu-system-*
591        do
592            if [ -x "$binary" ]
593            then
594                export QEMU_PROG="$build_root/$binary"
595                break
596            fi
597        done
598        popd > /dev/null
599        [ "$QEMU_PROG" = "" ] && _init_error "qemu not found"
600    fi
601fi
602export QEMU_PROG="$(type -p "$QEMU_PROG")"
603
604export QEMU_OPTIONS="-nodefaults -display none -accel qtest"
605case "$QEMU_PROG" in
606    *qemu-system-arm|*qemu-system-aarch64)
607        export QEMU_OPTIONS="$QEMU_OPTIONS -machine virt"
608        ;;
609    *qemu-system-avr)
610        export QEMU_OPTIONS="$QEMU_OPTIONS -machine mega2560"
611        ;;
612    *qemu-system-rx)
613        export QEMU_OPTIONS="$QEMU_OPTIONS -machine gdbsim-r5f562n8"
614        ;;
615    *qemu-system-tricore)
616        export QEMU_OPTIONS="-$QEMU_OPTIONS -machine tricore_testboard"
617        ;;
618esac
619
620if [ -z "$QEMU_IMG_PROG" ]; then
621    if [ -x "$build_iotests/qemu-img" ]; then
622        export QEMU_IMG_PROG="$build_iotests/qemu-img"
623    elif [ -x "$build_root/qemu-img" ]; then
624        export QEMU_IMG_PROG="$build_root/qemu-img"
625    else
626        _init_error "qemu-img not found"
627    fi
628fi
629export QEMU_IMG_PROG="$(type -p "$QEMU_IMG_PROG")"
630
631if [ -z "$QEMU_IO_PROG" ]; then
632    if [ -x "$build_iotests/qemu-io" ]; then
633        export QEMU_IO_PROG="$build_iotests/qemu-io"
634    elif [ -x "$build_root/qemu-io" ]; then
635        export QEMU_IO_PROG="$build_root/qemu-io"
636    else
637        _init_error "qemu-io not found"
638    fi
639fi
640export QEMU_IO_PROG="$(type -p "$QEMU_IO_PROG")"
641
642if [ -z $QEMU_NBD_PROG ]; then
643    if [ -x "$build_iotests/qemu-nbd" ]; then
644        export QEMU_NBD_PROG="$build_iotests/qemu-nbd"
645    elif [ -x "$build_root/qemu-nbd" ]; then
646        export QEMU_NBD_PROG="$build_root/qemu-nbd"
647    else
648        _init_error "qemu-nbd not found"
649    fi
650fi
651export QEMU_NBD_PROG="$(type -p "$QEMU_NBD_PROG")"
652
653if [ -z "$QSD_PROG" ]; then
654    if [ -x "$build_iotests/qemu-storage-daemon" ]; then
655        export QSD_PROG="$build_iotests/qemu-storage-daemon"
656    elif [ -x "$build_root/storage-daemon/qemu-storage-daemon" ]; then
657        export QSD_PROG="$build_root/storage-daemon/qemu-storage-daemon"
658    else
659        _init_error "qemu-storage-daemon not found"
660    fi
661fi
662export QSD_PROG="$(type -p "$QSD_PROG")"
663
664if [ -x "$build_iotests/socket_scm_helper" ]
665then
666    export SOCKET_SCM_HELPER="$build_iotests/socket_scm_helper"
667fi
668
669python_usable=false
670if $PYTHON -c 'import sys; sys.exit(0 if sys.version_info >= (3,6) else 1)'
671then
672    # Our python framework also requires virtio-blk
673    if "$QEMU_PROG" -M none -device help | grep -q virtio-blk >/dev/null 2>&1
674    then
675        python_usable=true
676    else
677        python_unusable_because="Missing virtio-blk in QEMU binary"
678    fi
679else
680    python_unusable_because="Unsupported Python version"
681fi
682
683default_machine=$($QEMU_PROG -machine help | sed -n '/(default)/ s/ .*//p')
684default_alias_machine=$($QEMU_PROG -machine help | \
685   sed -n "/(alias of $default_machine)/ { s/ .*//p; q; }")
686if [[ "$default_alias_machine" ]]; then
687    default_machine="$default_alias_machine"
688fi
689
690export QEMU_DEFAULT_MACHINE="$default_machine"
691
692TIMESTAMP_FILE=check.time-$IMGPROTO-$IMGFMT
693
694_wallclock()
695{
696    date "+%H %M %S" | awk '{ print $1*3600 + $2*60 + $3 }'
697}
698
699_wrapup()
700{
701    if $showme
702    then
703        :
704    elif $needwrap
705    then
706        if [ -f $TIMESTAMP_FILE -a -f $tmp.time ]
707        then
708            cat $TIMESTAMP_FILE $tmp.time \
709            | awk '
710        { t[$1] = $2 }
711END        { if (NR > 0) {
712            for (i in t) print i " " t[i]
713          }
714        }' \
715            | sort -n >$tmp.out
716            mv $tmp.out $TIMESTAMP_FILE
717        fi
718
719        if [ -f $tmp.expunged ]
720        then
721            notrun=$(wc -l <$tmp.expunged | sed -e 's/  *//g')
722            try=$(expr $try - $notrun)
723            list=$(echo "$list" | sed -f $tmp.expunged)
724        fi
725
726        echo "" >>check.log
727        date >>check.log
728        echo $list | fmt | sed -e 's/^/    /' >>check.log
729        $interrupt && echo "Interrupted!" >>check.log
730
731        if [ ! -z "$notrun" ]
732        then
733            echo "Not run:$notrun"
734            echo "Not run:$notrun" >>check.log
735        fi
736        if [ ! -z "$casenotrun" ]
737        then
738            echo "Some cases not run in:$casenotrun"
739            echo "Some cases not run in:$casenotrun" >>check.log
740        fi
741        if [ ! -z "$n_bad" -a $n_bad != 0 ]
742        then
743            echo "Failures:$bad"
744            echo "Failed $n_bad of $try iotests"
745            echo "Failures:$bad" | fmt >>check.log
746            echo "Failed $n_bad of $try iotests" >>check.log
747        else
748            echo "Passed all $try iotests"
749            echo "Passed all $try iotests" >>check.log
750        fi
751        needwrap=false
752    fi
753
754    if test -n "$STTY_RESTORE"; then
755        stty $STTY_RESTORE
756    fi
757    rm -f "${TEST_DIR}"/*.out "${TEST_DIR}"/*.err "${TEST_DIR}"/*.time
758    rm -f "${TEST_DIR}"/check.pid "${TEST_DIR}"/check.sts
759    rm -f $tmp.*
760
761    if $tmp_sock_dir
762    then
763        rm -rf "$SOCK_DIR"
764    fi
765}
766
767trap "_wrapup; exit \$status" 0 1 2 3 15
768
769# Report the test start and results. For makecheck we want to pretty
770# print the whole report at the end of the execution.
771# args: $seq, $starttime, $lasttime
772_report_test_start()
773{
774    if ! $makecheck; then
775        if [ -n "$3" ]; then
776            local lasttime=" (last: $3s)"
777        fi
778        printf "%-8s %-10s [%s]            %4s%-14s\r" "$1" "..." "$2" "..." "$lasttime"
779    fi
780}
781# args:$seq $status $starttime $lasttime $thistime $details
782_report_test_result()
783{
784    local status lasttime thistime
785    if $makecheck; then
786        if [ -n "$2" ] && [ "$2" != "pass" ]; then
787            status=" [$2]"
788        fi
789        printf "  TEST    iotest-$IMGFMT: %s%s\n" "$1" "$status"
790        return
791    fi
792
793    if [ -n "$4" ]; then
794        lasttime=" (last: $4s)"
795    fi
796    if [ -n "$5" ]; then
797        thistime=" $5s"
798    fi
799    case "$2" in
800        "pass")     status=$(printf "\e[32m%-10s\e[0m" "$2") ;;
801        "fail")     status=$(printf "\e[1m\e[31m%-10s\e[0m" "$2") ;;
802        "not run")  status=$(printf "\e[33m%-10s\e[0m" "$2") ;;
803        *)          status=$(printf "%-10s" "$2") ;;
804    esac
805
806    printf "%-8s %s [%s] [%s] %4s%-14s %s\n" "$1" "$status" "$3" "$(date '+%T')" "$thistime" "$lasttime" "$6"
807}
808
809[ -f $TIMESTAMP_FILE ] || touch $TIMESTAMP_FILE
810
811FULL_IMGFMT_DETAILS=$(_full_imgfmt_details)
812FULL_HOST_DETAILS=$(_full_platform_details)
813
814if ! $makecheck; then
815    _full_env_details
816fi
817
818seq="check"
819
820[ -n "$TESTS_REMAINING_LOG" ] && echo $list > $TESTS_REMAINING_LOG
821
822for seq in $list
823do
824    err=false       # error flag
825    printdiff=false # show diff to reference output?
826    status=""       # test result summary
827    results=""      # test result details
828    thistime=""     # time the test took
829
830    if [ -n "$TESTS_REMAINING_LOG" ] ; then
831        sed -e "s/$seq//" -e 's/  / /' -e 's/^ *//' $TESTS_REMAINING_LOG > $TESTS_REMAINING_LOG.tmp
832        mv $TESTS_REMAINING_LOG.tmp $TESTS_REMAINING_LOG
833        sync
834    fi
835
836    lasttime=$(sed -n -e "/^$seq /s/.* //p" <$TIMESTAMP_FILE)
837    starttime=$(date "+%T")
838    _report_test_start $seq $starttime $lasttime
839
840    if $showme
841    then
842        status="not run"
843    elif [ -f expunged ] && $expunge && egrep "^$seq([         ]|\$)" expunged >/dev/null
844    then
845        status="not run"
846        results="expunged"
847        rm -f $seq.out.bad
848        echo "/^$seq\$/d" >>$tmp.expunged
849    elif [ ! -f "$source_iotests/$seq" ]
850    then
851        status="not run"
852        results="no such test?"
853        echo "/^$seq\$/d" >>$tmp.expunged
854    else
855        # really going to try and run this one
856        #
857        rm -f $seq.out.bad
858        rm -f core $seq.notrun
859        rm -f $seq.casenotrun
860
861        start=$(_wallclock)
862
863        if [ "$(head -n 1 "$source_iotests/$seq")" == "#!/usr/bin/env python3" ]; then
864            if $python_usable; then
865                run_command="$PYTHON $seq"
866            else
867                run_command="false"
868                echo "$python_unusable_because" > $seq.notrun
869            fi
870        else
871            run_command="./$seq"
872        fi
873        export OUTPUT_DIR=$PWD
874        if $debug; then
875            (cd "$source_iotests";
876            MALLOC_PERTURB_=${MALLOC_PERTURB_:-$(($RANDOM % 255 + 1))} \
877                    $run_command -d 2>&1 | tee $tmp.out)
878        else
879            (cd "$source_iotests";
880            MALLOC_PERTURB_=${MALLOC_PERTURB_:-$(($RANDOM % 255 + 1))} \
881                    $run_command >$tmp.out 2>&1)
882        fi
883        sts=$?
884        stop=$(_wallclock)
885
886        if [ -f core ]
887        then
888            mv core $seq.core
889            status="fail"
890            results="[dumped core] $seq.core"
891            err=true
892        fi
893
894        if [ -f $seq.notrun ]
895        then
896            # overwrites timestamp output
897            status="not run"
898            results="$(cat $seq.notrun)"
899        else
900            if [ $sts -ne 0 ]
901            then
902                status="fail"
903                results=$(printf %s "[failed, exit status $sts]")
904                err=true
905            fi
906
907            reference="$source_iotests/$seq.out"
908            reference_machine="$source_iotests/$seq.$QEMU_DEFAULT_MACHINE.out"
909            if [ -f "$reference_machine" ]; then
910                reference="$reference_machine"
911            fi
912
913            reference_format="$source_iotests/$seq.out.$IMGFMT"
914            if [ -f "$reference_format" ]; then
915                reference="$reference_format"
916            fi
917
918            if [ "$CACHEMODE" = "none" ]; then
919                [ -f "$source_iotests/$seq.out.nocache" ] && reference="$source_iotests/$seq.out.nocache"
920            fi
921
922            if [ ! -f "$reference" ]
923            then
924                status="fail"
925                results="no qualified output"
926                err=true
927            else
928                if diff -w "$reference" $tmp.out >/dev/null 2>&1
929                then
930                    if ! $err; then
931                        status="pass"
932                        thistime=$(expr $stop - $start)
933                        echo "$seq $thistime" >>$tmp.time
934                    fi
935                else
936                    mv $tmp.out $seq.out.bad
937                    status="fail"
938                    results="output mismatch (see $seq.out.bad)"
939                    printdiff=true
940                    err=true
941                fi
942            fi
943        fi
944        if [ -f $seq.casenotrun ]
945        then
946            cat $seq.casenotrun
947            casenotrun="$casenotrun $seq"
948        fi
949    fi
950
951    # come here for each test, except when $showme is true
952    #
953    _report_test_result $seq "$status" "$starttime" "$lasttime" "$thistime" "$results"
954    case "$status" in
955        "pass")
956            try=$(expr $try + 1)
957            ;;
958        "fail")
959            try=$(expr $try + 1)
960            if $makecheck; then
961                _full_env_details
962            fi
963            if $printdiff; then
964                $diff -w "$reference" "$PWD"/$seq.out.bad
965            fi
966            bad="$bad $seq"
967            n_bad=$(expr $n_bad + 1)
968            quick=false
969            ;;
970        "not run")
971            notrun="$notrun $seq"
972            ;;
973    esac
974
975    seq="after_$seq"
976done
977
978interrupt=false
979status=$(expr $n_bad)
980exit
981