| #!/bin/sh |
| ## Copyright (c) 2021, Alliance for Open Media. All rights reserved |
| ## |
| ## This source code is subject to the terms of the BSD 3-Clause Clear License and the |
| ## Alliance for Open Media Patent License 1.0. If the BSD 3-Clause Clear License was |
| ## not distributed with this source code in the LICENSE file, you can obtain it |
| ## at aomedia.org/license/software-license/bsd-3-c-c/. If the Alliance for Open Media Patent |
| ## License 1.0 was not distributed with this source code in the PATENTS file, you |
| ## can obtain it at aomedia.org/license/patent-license/. |
| ## |
| ## This file contains shell code shared by test scripts for libaom tools. |
| |
| # Use $AOM_TEST_TOOLS_COMMON_SH as a pseudo include guard. |
| if [ -z "${AOM_TEST_TOOLS_COMMON_SH}" ]; then |
| AOM_TEST_TOOLS_COMMON_SH=included |
| |
| set -e |
| devnull='> /dev/null 2>&1' |
| AOM_TEST_PREFIX="" |
| |
| elog() { |
| echo "$@" 1>&2 |
| } |
| |
| vlog() { |
| if [ "${AOM_TEST_VERBOSE_OUTPUT}" = "yes" ]; then |
| echo "$@" |
| fi |
| } |
| |
| # Sets $AOM_TOOL_TEST to the name specified by positional parameter one. |
| test_begin() { |
| AOM_TOOL_TEST="${1}" |
| } |
| |
| # Clears the AOM_TOOL_TEST variable after confirming that $AOM_TOOL_TEST matches |
| # positional parameter one. |
| test_end() { |
| if [ "$1" != "${AOM_TOOL_TEST}" ]; then |
| echo "FAIL completed test mismatch!." |
| echo " completed test: ${1}" |
| echo " active test: ${AOM_TOOL_TEST}." |
| return 1 |
| fi |
| AOM_TOOL_TEST='<unset>' |
| } |
| |
| # Echoes the target configuration being tested. |
| test_configuration_target() { |
| aom_config_c="${LIBAOM_CONFIG_PATH}/config/aom_config.c" |
| # Clean up the cfg pointer line from aom_config.c for easier re-use by |
| # someone examining a failure in the example tests. |
| # 1. Run grep on aom_config.c for cfg and limit the results to 1. |
| # 2. Split the line using ' = ' as separator. |
| # 3. Abuse sed to consume the leading " and trailing "; from the assignment |
| # to the cfg pointer. |
| cmake_config=$(awk -F ' = ' '/cfg/ { print $NF; exit }' "${aom_config_c}" \ |
| | sed -e s/\"// -e s/\"\;//) |
| echo cmake generated via command: cmake path/to/aom ${cmake_config} |
| } |
| |
| # Trap function used for failure reports and tool output directory removal. |
| # When the contents of $AOM_TOOL_TEST do not match the string '<unset>', reports |
| # failure of test stored in $AOM_TOOL_TEST. |
| cleanup() { |
| if [ -n "${AOM_TOOL_TEST}" ] && [ "${AOM_TOOL_TEST}" != '<unset>' ]; then |
| echo "FAIL: $AOM_TOOL_TEST" |
| fi |
| if [ "${AOM_TEST_PRESERVE_OUTPUT}" = "yes" ]; then |
| return |
| fi |
| if [ -n "${AOM_TEST_OUTPUT_DIR}" ] && [ -d "${AOM_TEST_OUTPUT_DIR}" ]; then |
| rm -rf "${AOM_TEST_OUTPUT_DIR}" |
| fi |
| } |
| |
| # Echoes the version string assigned to the VERSION_STRING_NOSP variable defined |
| # in $LIBAOM_CONFIG_PATH/config/aom_version.h to stdout. |
| cmake_version() { |
| aom_version_h="${LIBAOM_CONFIG_PATH}/config/aom_version.h" |
| |
| # Find VERSION_STRING_NOSP line, split it with '"' and print the next to last |
| # field to output the version string to stdout. |
| aom_version=$(awk -F \" '/VERSION_STRING_NOSP/ {print $(NF-1)}' \ |
| "${aom_version_h}") |
| echo "v${aom_version}" |
| } |
| |
| # Echoes current git version as reported by running 'git describe', or the |
| # version used by the cmake build when git is unavailable. |
| # Note: version.cmake strips optional prefix before the version. So, |
| # source_version() function needs to do the same, so that the strings can be |
| # compared in check_version_strings() later. |
| source_version() { |
| if git --version > /dev/null 2>&1; then |
| (cd "$(dirname "${0}")" |
| git describe | sed 's/.*-v/v/') |
| else |
| cmake_version |
| fi |
| } |
| |
| # Echoes warnings to stdout when source version and CMake build generated |
| # version are out of sync. |
| check_version_strings() { |
| cmake_version=$(cmake_version) |
| source_version=$(source_version) |
| |
| if [ "${cmake_version}" != "${source_version}" ]; then |
| echo "Warning: version has changed since last cmake run." |
| vlog " cmake version: ${cmake_version} version now: ${source_version}" |
| fi |
| } |
| |
| # $1 is the name of an environment variable containing a directory name to |
| # test. |
| test_env_var_dir() { |
| local dir=$(eval echo "\${$1}") |
| if [ ! -d "${dir}" ]; then |
| elog "'${dir}': No such directory" |
| elog "The $1 environment variable must be set to a valid directory." |
| return 1 |
| fi |
| } |
| |
| # This script requires that the LIBAOM_BIN_PATH, LIBAOM_CONFIG_PATH, and |
| # LIBAOM_TEST_DATA_PATH variables are in the environment: Confirm that |
| # the variables are set and that they all evaluate to directory paths. |
| verify_aom_test_environment() { |
| test_env_var_dir "LIBAOM_BIN_PATH" \ |
| && test_env_var_dir "LIBAOM_CONFIG_PATH" \ |
| && test_env_var_dir "LIBAOM_TEST_DATA_PATH" |
| } |
| |
| # Greps aom_config.h in LIBAOM_CONFIG_PATH for positional parameter one, which |
| # should be a LIBAOM preprocessor flag. Echoes yes to stdout when the feature |
| # is available. |
| aom_config_option_enabled() { |
| aom_config_option="${1}" |
| aom_config_file="${LIBAOM_CONFIG_PATH}/config/aom_config.h" |
| config_line=$(grep "${aom_config_option}" "${aom_config_file}") |
| if echo "${config_line}" | egrep -q '1$'; then |
| echo yes |
| fi |
| } |
| |
| # Echoes yes when output of test_configuration_target() contains win32 or win64. |
| is_windows_target() { |
| if test_configuration_target \ |
| | grep -q -e win32 -e win64 > /dev/null 2>&1; then |
| echo yes |
| fi |
| } |
| |
| # Echoes path to $1 when it's executable and exists in one of the directories |
| # included in $tool_paths, or an empty string. Caller is responsible for testing |
| # the string once the function returns. |
| aom_tool_path() { |
| local tool_name="$1" |
| local root_path="${LIBAOM_BIN_PATH}" |
| local suffix="${AOM_TEST_EXE_SUFFIX}" |
| local tool_paths="\ |
| ${root_path}/${tool_name}${suffix} \ |
| ${root_path}/../${tool_name}${suffix} \ |
| ${root_path}/tools/${tool_name}${suffix} \ |
| ${root_path}/../tools/${tool_name}${suffix}" |
| |
| local toolpath="" |
| |
| for tool_path in ${tool_paths}; do |
| if [ -x "${tool_path}" ] && [ -f "${tool_path}" ]; then |
| echo "${tool_path}" |
| return 0 |
| fi |
| done |
| |
| return 1 |
| } |
| |
| # Echoes yes to stdout when the file named by positional parameter one exists |
| # in LIBAOM_BIN_PATH, and is executable. |
| aom_tool_available() { |
| local tool_name="$1" |
| local tool="${LIBAOM_BIN_PATH}/${tool_name}${AOM_TEST_EXE_SUFFIX}" |
| [ -x "${tool}" ] && echo yes |
| } |
| |
| # Echoes yes to stdout when aom_config_option_enabled() reports yes for |
| # CONFIG_AV1_DECODER. |
| av1_decode_available() { |
| [ "$(aom_config_option_enabled CONFIG_AV1_DECODER)" = "yes" ] && echo yes |
| } |
| |
| # Echoes yes to stdout when aom_config_option_enabled() reports yes for |
| # CONFIG_AV1_ENCODER. |
| av1_encode_available() { |
| [ "$(aom_config_option_enabled CONFIG_AV1_ENCODER)" = "yes" ] && echo yes |
| } |
| |
| # Echoes "fast" encode params for use with aomenc. |
| aomenc_encode_test_fast_params() { |
| echo "--cpu-used=2 |
| --limit=${AV1_ENCODE_TEST_FRAME_LIMIT} |
| --lag-in-frames=0 |
| --test-decode=fatal" |
| } |
| |
| # Echoes yes to stdout when aom_config_option_enabled() reports yes for |
| # CONFIG_WEBM_IO. |
| webm_io_available() { |
| [ "$(aom_config_option_enabled CONFIG_WEBM_IO)" = "yes" ] && echo yes |
| } |
| |
| # Filters strings from $1 using the filter specified by $2. Filter behavior |
| # depends on the presence of $3. When $3 is present, strings that match the |
| # filter are excluded. When $3 is omitted, strings matching the filter are |
| # included. |
| # The filtered result is echoed to stdout. |
| filter_strings() { |
| strings=${1} |
| filter=${2} |
| exclude=${3} |
| |
| if [ -n "${exclude}" ]; then |
| # When positional parameter three exists the caller wants to remove strings. |
| # Tell grep to invert matches using the -v argument. |
| exclude='-v' |
| else |
| unset exclude |
| fi |
| |
| if [ -n "${filter}" ]; then |
| for s in ${strings}; do |
| if echo "${s}" | egrep -q ${exclude} "${filter}" > /dev/null 2>&1; then |
| filtered_strings="${filtered_strings} ${s}" |
| fi |
| done |
| else |
| filtered_strings="${strings}" |
| fi |
| echo "${filtered_strings}" |
| } |
| |
| # Runs user test functions passed via positional parameters one and two. |
| # Functions in positional parameter one are treated as environment verification |
| # functions and are run unconditionally. Functions in positional parameter two |
| # are run according to the rules specified in aom_test_usage(). |
| run_tests() { |
| local env_tests="verify_aom_test_environment $1" |
| local tests_to_filter="$2" |
| local test_name="${AOM_TEST_NAME}" |
| |
| if [ -z "${test_name}" ]; then |
| test_name="$(basename "${0%.*}")" |
| fi |
| |
| if [ "${AOM_TEST_RUN_DISABLED_TESTS}" != "yes" ]; then |
| # Filter out DISABLED tests. |
| tests_to_filter=$(filter_strings "${tests_to_filter}" ^DISABLED exclude) |
| fi |
| |
| if [ -n "${AOM_TEST_FILTER}" ]; then |
| # Remove tests not matching the user's filter. |
| tests_to_filter=$(filter_strings "${tests_to_filter}" ${AOM_TEST_FILTER}) |
| fi |
| |
| # User requested test listing: Dump test names and return. |
| if [ "${AOM_TEST_LIST_TESTS}" = "yes" ]; then |
| for test_name in $tests_to_filter; do |
| echo ${test_name} |
| done |
| return |
| fi |
| |
| # Don't bother with the environment tests if everything else was disabled. |
| [ -z "${tests_to_filter}" ] && return |
| |
| # Combine environment and actual tests. |
| local tests_to_run="${env_tests} ${tests_to_filter}" |
| |
| check_version_strings |
| |
| # Run tests. |
| for test in ${tests_to_run}; do |
| test_begin "${test}" |
| vlog " RUN ${test}" |
| "${test}" |
| vlog " PASS ${test}" |
| test_end "${test}" |
| done |
| |
| local tested_config="$(test_configuration_target) @ $(source_version)" |
| echo "${test_name}: Done, all tests pass for ${tested_config}." |
| } |
| |
| aom_test_usage() { |
| cat << EOF |
| Usage: ${0##*/} [arguments] |
| --bin-path <path to libaom binaries directory> |
| --config-path <path to libaom config directory> |
| --filter <filter>: User test filter. Only tests matching filter are run. |
| --run-disabled-tests: Run disabled tests. |
| --help: Display this message and exit. |
| --test-data-path <path to libaom test data directory> |
| --show-program-output: Shows output from all programs being tested. |
| --prefix: Allows for a user specified prefix to be inserted before all test |
| programs. Grants the ability, for example, to run test programs |
| within valgrind. |
| --list-tests: List all test names and exit without actually running tests. |
| --verbose: Verbose output. |
| |
| When the --bin-path option is not specified the script attempts to use |
| \$LIBAOM_BIN_PATH and then the current directory. |
| |
| When the --config-path option is not specified the script attempts to use |
| \$LIBAOM_CONFIG_PATH and then the current directory. |
| |
| When the -test-data-path option is not specified the script attempts to use |
| \$LIBAOM_TEST_DATA_PATH and then the current directory. |
| EOF |
| } |
| |
| # Returns non-zero (failure) when required environment variables are empty |
| # strings. |
| aom_test_check_environment() { |
| if [ -z "${LIBAOM_BIN_PATH}" ] || \ |
| [ -z "${LIBAOM_CONFIG_PATH}" ] || \ |
| [ -z "${LIBAOM_TEST_DATA_PATH}" ]; then |
| return 1 |
| fi |
| } |
| |
| # Echo aomenc command line parameters allowing use of a raw yuv file as |
| # input to aomenc. |
| yuv_raw_input() { |
| echo ""${YUV_RAW_INPUT}" |
| --width="${YUV_RAW_INPUT_WIDTH}" |
| --height="${YUV_RAW_INPUT_HEIGHT}"" |
| } |
| |
| # Do a small encode for testing decoders. |
| encode_yuv_raw_input_av1() { |
| if [ "$(av1_encode_available)" = "yes" ]; then |
| local output="$1" |
| local encoder="$(aom_tool_path aomenc)" |
| shift |
| eval "${encoder}" $(yuv_raw_input) \ |
| $(aomenc_encode_test_fast_params) \ |
| --output="${output}" \ |
| $@ \ |
| ${devnull} |
| |
| if [ ! -e "${output}" ]; then |
| elog "Output file does not exist." |
| return 1 |
| fi |
| fi |
| } |
| |
| # Parse the command line. |
| while [ -n "$1" ]; do |
| case "$1" in |
| --bin-path) |
| LIBAOM_BIN_PATH="$2" |
| shift |
| ;; |
| --config-path) |
| LIBAOM_CONFIG_PATH="$2" |
| shift |
| ;; |
| --filter) |
| AOM_TEST_FILTER="$2" |
| shift |
| ;; |
| --run-disabled-tests) |
| AOM_TEST_RUN_DISABLED_TESTS=yes |
| ;; |
| --help) |
| aom_test_usage |
| exit |
| ;; |
| --test-data-path) |
| LIBAOM_TEST_DATA_PATH="$2" |
| shift |
| ;; |
| --prefix) |
| AOM_TEST_PREFIX="$2" |
| shift |
| ;; |
| --verbose) |
| AOM_TEST_VERBOSE_OUTPUT=yes |
| ;; |
| --show-program-output) |
| devnull= |
| ;; |
| --list-tests) |
| AOM_TEST_LIST_TESTS=yes |
| ;; |
| *) |
| aom_test_usage |
| exit 1 |
| ;; |
| esac |
| shift |
| done |
| |
| # Handle running the tests from a build directory without arguments when running |
| # the tests on *nix/macosx. |
| LIBAOM_BIN_PATH="${LIBAOM_BIN_PATH:-.}" |
| LIBAOM_CONFIG_PATH="${LIBAOM_CONFIG_PATH:-.}" |
| LIBAOM_TEST_DATA_PATH="${LIBAOM_TEST_DATA_PATH:-.}" |
| |
| # Create a temporary directory for output files, and a trap to clean it up. |
| if [ -n "${TMPDIR}" ]; then |
| AOM_TEST_TEMP_ROOT="${TMPDIR}" |
| elif [ -n "${TEMPDIR}" ]; then |
| AOM_TEST_TEMP_ROOT="${TEMPDIR}" |
| else |
| AOM_TEST_TEMP_ROOT=/tmp |
| fi |
| |
| AOM_TEST_OUTPUT_DIR="${AOM_TEST_OUTPUT_DIR:-${AOM_TEST_TEMP_ROOT}/aom_test_$$}" |
| |
| if ! mkdir -p "${AOM_TEST_OUTPUT_DIR}" || \ |
| [ ! -d "${AOM_TEST_OUTPUT_DIR}" ]; then |
| echo "${0##*/}: Cannot create output directory, giving up." |
| echo "${0##*/}: AOM_TEST_OUTPUT_DIR=${AOM_TEST_OUTPUT_DIR}" |
| exit 1 |
| fi |
| |
| AOM_TEST_PRESERVE_OUTPUT=${AOM_TEST_PRESERVE_OUTPUT:-no} |
| |
| if [ "$(is_windows_target)" = "yes" ]; then |
| AOM_TEST_EXE_SUFFIX=".exe" |
| fi |
| |
| # Variables shared by tests. |
| AV1_ENCODE_CPU_USED=${AV1_ENCODE_CPU_USED:-1} |
| AV1_ENCODE_TEST_FRAME_LIMIT=${AV1_ENCODE_TEST_FRAME_LIMIT:-5} |
| AV1_IVF_FILE="${AV1_IVF_FILE:-${AOM_TEST_OUTPUT_DIR}/av1.ivf}" |
| AV1_OBU_ANNEXB_FILE="${AV1_OBU_ANNEXB_FILE:-${AOM_TEST_OUTPUT_DIR}/av1.annexb.obu}" |
| AV1_OBU_SEC5_FILE="${AV1_OBU_SEC5_FILE:-${AOM_TEST_OUTPUT_DIR}/av1.section5.obu}" |
| AV1_WEBM_FILE="${AV1_WEBM_FILE:-${AOM_TEST_OUTPUT_DIR}/av1.webm}" |
| |
| YUV_RAW_INPUT="${LIBAOM_TEST_DATA_PATH}/hantro_collage_w352h288.yuv" |
| YUV_RAW_INPUT_WIDTH=352 |
| YUV_RAW_INPUT_HEIGHT=288 |
| |
| Y4M_NOSQ_PAR_INPUT="${LIBAOM_TEST_DATA_PATH}/park_joy_90p_8_420_a10-1.y4m" |
| Y4M_720P_INPUT="${LIBAOM_TEST_DATA_PATH}/niklas_1280_720_30.y4m" |
| |
| # Setup a trap function to clean up after tests complete. |
| trap cleanup EXIT |
| |
| vlog "$(basename "${0%.*}") test configuration: |
| LIBAOM_BIN_PATH=${LIBAOM_BIN_PATH} |
| LIBAOM_CONFIG_PATH=${LIBAOM_CONFIG_PATH} |
| LIBAOM_TEST_DATA_PATH=${LIBAOM_TEST_DATA_PATH} |
| AOM_TEST_EXE_SUFFIX=${AOM_TEST_EXE_SUFFIX} |
| AOM_TEST_FILTER=${AOM_TEST_FILTER} |
| AOM_TEST_LIST_TESTS=${AOM_TEST_LIST_TESTS} |
| AOM_TEST_OUTPUT_DIR=${AOM_TEST_OUTPUT_DIR} |
| AOM_TEST_PREFIX=${AOM_TEST_PREFIX} |
| AOM_TEST_PRESERVE_OUTPUT=${AOM_TEST_PRESERVE_OUTPUT} |
| AOM_TEST_RUN_DISABLED_TESTS=${AOM_TEST_RUN_DISABLED_TESTS} |
| AOM_TEST_SHOW_PROGRAM_OUTPUT=${AOM_TEST_SHOW_PROGRAM_OUTPUT} |
| AOM_TEST_TEMP_ROOT=${AOM_TEST_TEMP_ROOT} |
| AOM_TEST_VERBOSE_OUTPUT=${AOM_TEST_VERBOSE_OUTPUT} |
| AV1_ENCODE_CPU_USED=${AV1_ENCODE_CPU_USED} |
| AV1_ENCODE_TEST_FRAME_LIMIT=${AV1_ENCODE_TEST_FRAME_LIMIT} |
| AV1_IVF_FILE=${AV1_IVF_FILE} |
| AV1_OBU_ANNEXB_FILE=${AV1_OBU_ANNEXB_FILE} |
| AV1_OBU_SEC5_FILE=${AV1_OBU_SEC5_FILE} |
| AV1_WEBM_FILE=${AV1_WEBM_FILE} |
| YUV_RAW_INPUT=${YUV_RAW_INPUT} |
| YUV_RAW_INPUT_WIDTH=${YUV_RAW_INPUT_WIDTH} |
| YUV_RAW_INPUT_HEIGHT=${YUV_RAW_INPUT_HEIGHT} |
| Y4M_NOSQ_PAR_INPUT=${Y4M_NOSQ_PAR_INPUT}" |
| |
| fi # End $AOM_TEST_TOOLS_COMMON_SH pseudo include guard. |