| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 1 | #!/bin/bash |
| 2 | |
| 3 | # Self-tests for gm, based on tools/tests/run.sh |
| epoger@google.com | 454008a | 2012-11-13 20:46:50 +0000 | [diff] [blame] | 4 | # |
| 5 | # These tests are run by the Skia_PerCommit_House_Keeping bot at every commit, |
| 6 | # so make sure that they still pass when you make changes to gm! |
| 7 | # |
| epoger@google.com | 0b62b3d | 2013-03-20 17:59:28 +0000 | [diff] [blame] | 8 | # To generate new baselines when gm behavior changes, run gm/tests/rebaseline.sh |
| 9 | # |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 10 | # TODO: because this is written as a shell script (instead of, say, Python) |
| 11 | # it only runs on Linux and Mac. |
| epoger@google.com | 3aa3358 | 2013-01-02 15:53:25 +0000 | [diff] [blame] | 12 | # See https://code.google.com/p/skia/issues/detail?id=677 |
| 13 | # ('make tools/tests/run.sh work cross-platform') |
| epoger@google.com | 454008a | 2012-11-13 20:46:50 +0000 | [diff] [blame] | 14 | # Ideally, these tests should pass on all development platforms... |
| 15 | # otherwise, how can developers be expected to test them before committing a |
| 16 | # change? |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 17 | |
| 18 | # cd into .../trunk so all the paths will work |
| 19 | cd $(dirname $0)/../.. |
| 20 | |
| 21 | # TODO(epoger): make it look in Release and/or Debug |
| 22 | GM_BINARY=out/Debug/gm |
| 23 | |
| epoger@google.com | fed9db6 | 2013-05-08 14:10:28 +0000 | [diff] [blame^] | 24 | # If WRITE_IMAGE_FILES is nonzero, then the self-test will pass --writePath |
| 25 | # and --mismatchPath arguments to GM. Currently, for various reasons, we |
| 26 | # cannot run these arguments on the production buildbots, so this should |
| 27 | # only be set to nonzero for local testing. |
| 28 | WRITE_IMAGE_FILES=0 |
| 29 | |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 30 | OUTPUT_ACTUAL_SUBDIR=output-actual |
| 31 | OUTPUT_EXPECTED_SUBDIR=output-expected |
| scroggo@google.com | 09fd4d2 | 2013-03-20 14:20:18 +0000 | [diff] [blame] | 32 | CONFIGS="--config 8888 565" |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 33 | |
| epoger@google.com | 0cc99cf | 2013-04-26 17:45:06 +0000 | [diff] [blame] | 34 | ENCOUNTERED_ANY_ERRORS=0 |
| 35 | |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 36 | # Compare contents of all files within directories $1 and $2, |
| 37 | # EXCEPT for any dotfiles. |
| 38 | # If there are any differences, a description is written to stdout and |
| 39 | # we exit with a nonzero return value. |
| 40 | # Otherwise, we write nothing to stdout and return. |
| 41 | function compare_directories { |
| 42 | if [ $# != 2 ]; then |
| 43 | echo "compare_directories requires exactly 2 parameters, got $#" |
| 44 | exit 1 |
| 45 | fi |
| 46 | diff -r --exclude=.* $1 $2 |
| 47 | if [ $? != 0 ]; then |
| 48 | echo "failed in: compare_directories $1 $2" |
| epoger@google.com | 0cc99cf | 2013-04-26 17:45:06 +0000 | [diff] [blame] | 49 | ENCOUNTERED_ANY_ERRORS=1 |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 50 | fi |
| 51 | } |
| 52 | |
| 53 | # Run gm... |
| 54 | # - with the arguments in $1 |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 55 | # - writing stdout into $2/$OUTPUT_ACTUAL_SUBDIR/stdout |
| 56 | # - writing json summary into $2/$OUTPUT_ACTUAL_SUBDIR/json-summary.txt |
| 57 | # - writing return value into $2/$OUTPUT_ACTUAL_SUBDIR/return_value |
| 58 | # Then compare all of those against $2/$OUTPUT_EXPECTED_SUBDIR . |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 59 | function gm_test { |
| 60 | if [ $# != 2 ]; then |
| 61 | echo "gm_test requires exactly 2 parameters, got $#" |
| 62 | exit 1 |
| 63 | fi |
| 64 | GM_ARGS="$1" |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 65 | ACTUAL_OUTPUT_DIR="$2/$OUTPUT_ACTUAL_SUBDIR" |
| 66 | EXPECTED_OUTPUT_DIR="$2/$OUTPUT_EXPECTED_SUBDIR" |
| 67 | JSON_SUMMARY_FILE="$ACTUAL_OUTPUT_DIR/json-summary.txt" |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 68 | |
| 69 | rm -rf $ACTUAL_OUTPUT_DIR |
| 70 | mkdir -p $ACTUAL_OUTPUT_DIR |
| epoger@google.com | fed9db6 | 2013-05-08 14:10:28 +0000 | [diff] [blame^] | 71 | |
| scroggo@google.com | 09fd4d2 | 2013-03-20 14:20:18 +0000 | [diff] [blame] | 72 | COMMAND="$GM_BINARY $GM_ARGS --writeJsonSummaryPath $JSON_SUMMARY_FILE" |
| epoger@google.com | fed9db6 | 2013-05-08 14:10:28 +0000 | [diff] [blame^] | 73 | if [ $WRITE_IMAGE_FILES != 0 ]; then |
| 74 | COMMAND="$COMMAND --writePath $ACTUAL_OUTPUT_DIR/writePath --mismatchPath $ACTUAL_OUTPUT_DIR/mismatchPath" |
| 75 | fi |
| 76 | |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 77 | echo "$COMMAND" >$ACTUAL_OUTPUT_DIR/command_line |
| epoger@google.com | 5efdd0c | 2013-03-13 14:18:40 +0000 | [diff] [blame] | 78 | $COMMAND >$ACTUAL_OUTPUT_DIR/stdout 2>$ACTUAL_OUTPUT_DIR/stderr |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 79 | echo $? >$ACTUAL_OUTPUT_DIR/return_value |
| 80 | |
| epoger@google.com | 80d4478 | 2013-01-18 20:03:58 +0000 | [diff] [blame] | 81 | # Only compare selected lines in the stdout, to ignore any spurious lines |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 82 | # as noted in http://code.google.com/p/skia/issues/detail?id=1068 . |
| 83 | # |
| 84 | # TODO(epoger): This is still hacky... we need to rewrite this script in |
| 85 | # Python soon, and make stuff like this more maintainable. |
| epoger@google.com | 5efdd0c | 2013-03-13 14:18:40 +0000 | [diff] [blame] | 86 | grep ^GM: $ACTUAL_OUTPUT_DIR/stdout >$ACTUAL_OUTPUT_DIR/stdout-tmp |
| epoger@google.com | 98204f9 | 2013-01-16 04:19:01 +0000 | [diff] [blame] | 87 | mv $ACTUAL_OUTPUT_DIR/stdout-tmp $ACTUAL_OUTPUT_DIR/stdout |
| epoger@google.com | 5efdd0c | 2013-03-13 14:18:40 +0000 | [diff] [blame] | 88 | grep ^GM: $ACTUAL_OUTPUT_DIR/stderr >$ACTUAL_OUTPUT_DIR/stderr-tmp |
| 89 | mv $ACTUAL_OUTPUT_DIR/stderr-tmp $ACTUAL_OUTPUT_DIR/stderr |
| epoger@google.com | 98204f9 | 2013-01-16 04:19:01 +0000 | [diff] [blame] | 90 | |
| epoger@google.com | fed9db6 | 2013-05-08 14:10:28 +0000 | [diff] [blame^] | 91 | if [ $WRITE_IMAGE_FILES != 0 ]; then |
| 92 | for IMAGEFILE in $(ls $ACTUAL_OUTPUT_DIR/*/*/*.png); do |
| 93 | SUM=$(sum $IMAGEFILE) |
| 94 | echo "Replaced image bytes with a checksum, because of https://code.google.com/p/chromium/issues/detail?id=169600 ('gcl/upload.py fail to upload binary files to rietveld')" >$IMAGEFILE |
| 95 | echo $SUM >> $IMAGEFILE |
| 96 | done |
| 97 | for MISMATCHDIR in $(ls -d $ACTUAL_OUTPUT_DIR/mismatchPath/*); do |
| 98 | echo "Created additional file to make sure directory isn't empty, because self-test cannot handle empty directories." >$MISMATCHDIR/bogusfile |
| 99 | done |
| 100 | fi |
| 101 | |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 102 | compare_directories $EXPECTED_OUTPUT_DIR $ACTUAL_OUTPUT_DIR |
| 103 | } |
| 104 | |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 105 | # Create input dir (at path $1) with expectations (both image and json) |
| 106 | # that gm will match or mismatch as appropriate. |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 107 | # |
| 108 | # We used to check these files into SVN, but then we needed to rebasline them |
| 109 | # when our drawing changed at all... so, as proposed in |
| 110 | # http://code.google.com/p/skia/issues/detail?id=1068 , we generate them |
| 111 | # new each time. |
| 112 | function create_inputs_dir { |
| 113 | if [ $# != 1 ]; then |
| 114 | echo "create_inputs_dir requires exactly 1 parameter, got $#" |
| 115 | exit 1 |
| 116 | fi |
| 117 | INPUTS_DIR="$1" |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 118 | IMAGES_DIR=$INPUTS_DIR/images |
| 119 | JSON_DIR=$INPUTS_DIR/json |
| 120 | mkdir -p $IMAGES_DIR $JSON_DIR |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 121 | |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 122 | mkdir -p $IMAGES_DIR/identical-bytes |
| 123 | # Run GM to write out the images actually generated. |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 124 | $GM_BINARY --hierarchy --match selftest1 $CONFIGS \ |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 125 | -w $IMAGES_DIR/identical-bytes |
| 126 | # Run GM again to read in those images and write them out as a JSON summary. |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 127 | $GM_BINARY --hierarchy --match selftest1 $CONFIGS \ |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 128 | -r $IMAGES_DIR/identical-bytes \ |
| scroggo@google.com | 09fd4d2 | 2013-03-20 14:20:18 +0000 | [diff] [blame] | 129 | --writeJsonSummaryPath $JSON_DIR/identical-bytes.json |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 130 | |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 131 | mkdir -p $IMAGES_DIR/identical-pixels |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 132 | $GM_BINARY --hierarchy --match selftest1 $CONFIGS \ |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 133 | -w $IMAGES_DIR/identical-pixels |
| 134 | echo "more bytes that do not change the image pixels" \ |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 135 | >> $IMAGES_DIR/identical-pixels/8888/selftest1.png |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 136 | echo "more bytes that do not change the image pixels" \ |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 137 | >> $IMAGES_DIR/identical-pixels/565/selftest1.png |
| 138 | $GM_BINARY --hierarchy --match selftest1 $CONFIGS \ |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 139 | -r $IMAGES_DIR/identical-pixels \ |
| scroggo@google.com | 09fd4d2 | 2013-03-20 14:20:18 +0000 | [diff] [blame] | 140 | --writeJsonSummaryPath $JSON_DIR/identical-pixels.json |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 141 | |
| 142 | mkdir -p $IMAGES_DIR/different-pixels |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 143 | $GM_BINARY --hierarchy --match selftest2 $CONFIGS \ |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 144 | -w $IMAGES_DIR/different-pixels |
| commit-bot@chromium.org | c61c3c3 | 2013-03-01 15:32:34 +0000 | [diff] [blame] | 145 | mv $IMAGES_DIR/different-pixels/8888/selftest2.png \ |
| 146 | $IMAGES_DIR/different-pixels/8888/selftest1.png |
| 147 | mv $IMAGES_DIR/different-pixels/565/selftest2.png \ |
| 148 | $IMAGES_DIR/different-pixels/565/selftest1.png |
| 149 | $GM_BINARY --hierarchy --match selftest1 $CONFIGS \ |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 150 | -r $IMAGES_DIR/different-pixels \ |
| scroggo@google.com | 09fd4d2 | 2013-03-20 14:20:18 +0000 | [diff] [blame] | 151 | --writeJsonSummaryPath $JSON_DIR/different-pixels.json |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 152 | |
| epoger@google.com | e460a47 | 2013-02-06 18:41:04 +0000 | [diff] [blame] | 153 | mkdir -p $IMAGES_DIR/empty-dir |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 154 | } |
| 155 | |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 156 | GM_TESTDIR=gm/tests |
| 157 | GM_INPUTS=$GM_TESTDIR/inputs |
| 158 | GM_OUTPUTS=$GM_TESTDIR/outputs |
| epoger@google.com | 3726960 | 2013-01-19 04:21:27 +0000 | [diff] [blame] | 159 | GM_TEMPFILES=$GM_TESTDIR/tempfiles |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 160 | |
| epoger@google.com | 407f8da | 2013-01-18 19:19:47 +0000 | [diff] [blame] | 161 | create_inputs_dir $GM_INPUTS |
| 162 | |
| epoger@google.com | 570aafe | 2012-11-28 20:08:32 +0000 | [diff] [blame] | 163 | # Compare generated image against an input image file with identical bytes. |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 164 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/images/identical-bytes" "$GM_OUTPUTS/compared-against-identical-bytes-images" |
| 165 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/json/identical-bytes.json" "$GM_OUTPUTS/compared-against-identical-bytes-json" |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 166 | |
| epoger@google.com | 570aafe | 2012-11-28 20:08:32 +0000 | [diff] [blame] | 167 | # Compare generated image against an input image file with identical pixels but different PNG encoding. |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 168 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/images/identical-pixels" "$GM_OUTPUTS/compared-against-identical-pixels-images" |
| 169 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/json/identical-pixels.json" "$GM_OUTPUTS/compared-against-identical-pixels-json" |
| epoger@google.com | 570aafe | 2012-11-28 20:08:32 +0000 | [diff] [blame] | 170 | |
| 171 | # Compare generated image against an input image file with different pixels. |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 172 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/images/different-pixels" "$GM_OUTPUTS/compared-against-different-pixels-images" |
| 173 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/json/different-pixels.json" "$GM_OUTPUTS/compared-against-different-pixels-json" |
| epoger@google.com | a413a53 | 2012-11-12 18:04:51 +0000 | [diff] [blame] | 174 | |
| epoger@google.com | ee8a8e3 | 2012-12-18 19:13:49 +0000 | [diff] [blame] | 175 | # Compare generated image against an empty "expected image" dir. |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 176 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/images/empty-dir" "$GM_OUTPUTS/compared-against-empty-dir" |
| 177 | |
| epoger@google.com | 318a059 | 2013-04-12 19:05:57 +0000 | [diff] [blame] | 178 | # Compare generated image against a nonexistent "expected image" dir. |
| 179 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS -r ../path/to/nowhere" "$GM_OUTPUTS/compared-against-nonexistent-dir" |
| 180 | |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 181 | # Compare generated image against an empty "expected image" dir, but NOT in verbose mode. |
| 182 | gm_test "--hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/images/empty-dir" "$GM_OUTPUTS/nonverbose" |
| epoger@google.com | ee8a8e3 | 2012-12-18 19:13:49 +0000 | [diff] [blame] | 183 | |
| epoger@google.com | 9c56a8d | 2012-12-20 18:34:29 +0000 | [diff] [blame] | 184 | # If run without "-r", the JSON's "actual-results" section should contain |
| 185 | # actual checksums marked as "failure-ignored", but the "expected-results" |
| 186 | # section should be empty. |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 187 | gm_test "--verbose --hierarchy --match selftest1 $CONFIGS" "$GM_OUTPUTS/no-readpath" |
| epoger@google.com | 9c56a8d | 2012-12-20 18:34:29 +0000 | [diff] [blame] | 188 | |
| epoger@google.com | caac3db | 2013-04-04 19:23:11 +0000 | [diff] [blame] | 189 | # Test what happens if a subset of the renderModes fail (e.g. pipe) |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 190 | gm_test "--simulatePipePlaybackFailure --verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/json/identical-pixels.json" "$GM_OUTPUTS/pipe-playback-failure" |
| epoger@google.com | caac3db | 2013-04-04 19:23:11 +0000 | [diff] [blame] | 191 | |
| epoger@google.com | c8263e7 | 2013-04-10 12:17:34 +0000 | [diff] [blame] | 192 | # Confirm that IntentionallySkipped tests are recorded as such. |
| epoger@google.com | 51dbabe | 2013-04-10 15:24:53 +0000 | [diff] [blame] | 193 | gm_test "--verbose --hierarchy --match selftest1 selftest2 $CONFIGS" "$GM_OUTPUTS/intentionally-skipped-tests" |
| epoger@google.com | c8263e7 | 2013-04-10 12:17:34 +0000 | [diff] [blame] | 194 | |
| epoger@google.com | 5079d2c | 2013-04-12 14:11:21 +0000 | [diff] [blame] | 195 | # Ignore some error types (including ExpectationsMismatch) |
| 196 | gm_test "--ignoreErrorTypes ExpectationsMismatch NoGpuContext --verbose --hierarchy --match selftest1 $CONFIGS -r $GM_INPUTS/json/different-pixels.json" "$GM_OUTPUTS/ignore-expectations-mismatch" |
| 197 | |
| epoger@google.com | 0cc99cf | 2013-04-26 17:45:06 +0000 | [diff] [blame] | 198 | if [ $ENCOUNTERED_ANY_ERRORS == 0 ]; then |
| 199 | echo "All tests passed." |
| 200 | exit 0 |
| 201 | else |
| 202 | exit 1 |
| 203 | fi |