hwcontext: Add test for device creation and derivation

This uses any devices it can find on the host system - on a system with no
hardware device support or in builds with no support included it will do
nothing and pass.
This commit is contained in:
Mark Thompson 2018-05-15 23:15:03 +01:00
parent 70d25268c2
commit ba7d1377e8
5 changed files with 239 additions and 0 deletions

View File

@ -206,6 +206,7 @@ TESTPROGS = adler32 \
fifo \
hash \
hmac \
hwdevice \
integer \
imgutils \
lfg \

View File

@ -22,6 +22,7 @@
/file
/hash
/hmac
/hwdevice
/imgutils
/lfg
/lls

226
libavutil/tests/hwdevice.c Normal file
View File

@ -0,0 +1,226 @@
/*
* This file is part of FFmpeg.
*
* FFmpeg is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* FFmpeg is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <stdio.h>
#include "libavutil/hwcontext.h"
static int test_derivation(AVBufferRef *src_ref, const char *src_name)
{
enum AVHWDeviceType derived_type;
const char *derived_name;
AVBufferRef *derived_ref = NULL, *back_ref = NULL;
AVHWDeviceContext *src_dev, *derived_dev;
int err;
src_dev = (AVHWDeviceContext*)src_ref->data;
derived_type = AV_HWDEVICE_TYPE_NONE;
while (1) {
derived_type = av_hwdevice_iterate_types(derived_type);
if (derived_type == AV_HWDEVICE_TYPE_NONE)
break;
derived_name = av_hwdevice_get_type_name(derived_type);
err = av_hwdevice_ctx_create_derived(&derived_ref, derived_type,
src_ref, 0);
if (err < 0) {
fprintf(stderr, "Unable to derive %s -> %s: %d.\n",
src_name, derived_name, err);
continue;
}
derived_dev = (AVHWDeviceContext*)derived_ref->data;
if (derived_dev->type != derived_type) {
fprintf(stderr, "Device derived as type %d has type %d.\n",
derived_type, derived_dev->type);
goto fail;
}
if (derived_type == src_dev->type) {
if (derived_dev != src_dev) {
fprintf(stderr, "Derivation of %s from itself succeeded "
"but did not return the same device.\n", src_name);
goto fail;
}
av_buffer_unref(&derived_ref);
continue;
}
err = av_hwdevice_ctx_create_derived(&back_ref, src_dev->type,
derived_ref, 0);
if (err < 0) {
fprintf(stderr, "Derivation %s to %s succeeded, but derivation "
"back again failed: %d.\n",
src_name, derived_name, err);
goto fail;
}
if (back_ref->data != src_ref->data) {
fprintf(stderr, "Derivation %s to %s succeeded, but derivation "
"back again did not return the original device.\n",
src_name, derived_name);
goto fail;
}
fprintf(stderr, "Successfully tested derivation %s -> %s.\n",
src_name, derived_name);
av_buffer_unref(&derived_ref);
av_buffer_unref(&back_ref);
}
return 0;
fail:
av_buffer_unref(&derived_ref);
av_buffer_unref(&back_ref);
return -1;
}
static int test_device(enum AVHWDeviceType type, const char *name,
const char *device, AVDictionary *opts, int flags)
{
AVBufferRef *ref;
AVHWDeviceContext *dev;
int err;
err = av_hwdevice_ctx_create(&ref, type, device, opts, flags);
if (err < 0) {
fprintf(stderr, "Failed to create %s device: %d.\n", name, err);
return 1;
}
dev = (AVHWDeviceContext*)ref->data;
if (dev->type != type) {
fprintf(stderr, "Device created as type %d has type %d.\n",
type, dev->type);
av_buffer_unref(&ref);
return -1;
}
fprintf(stderr, "Device type %s successfully created.\n", name);
err = test_derivation(ref, name);
av_buffer_unref(&ref);
return err;
}
static const struct {
enum AVHWDeviceType type;
const char *possible_devices[5];
} test_devices[] = {
{ AV_HWDEVICE_TYPE_CUDA,
{ "0", "1", "2" } },
{ AV_HWDEVICE_TYPE_DRM,
{ "/dev/dri/card0", "/dev/dri/card1",
"/dev/dri/renderD128", "/dev/dri/renderD129" } },
{ AV_HWDEVICE_TYPE_DXVA2,
{ "0", "1", "2" } },
{ AV_HWDEVICE_TYPE_D3D11VA,
{ "0", "1", "2" } },
{ AV_HWDEVICE_TYPE_OPENCL,
{ "0.0", "0.1", "1.0", "1.1" } },
{ AV_HWDEVICE_TYPE_VAAPI,
{ "/dev/dri/renderD128", "/dev/dri/renderD129", ":0" } },
};
static int test_device_type(enum AVHWDeviceType type)
{
enum AVHWDeviceType check;
const char *name;
int i, j, found, err;
name = av_hwdevice_get_type_name(type);
if (!name) {
fprintf(stderr, "No name available for device type %d.\n", type);
return -1;
}
check = av_hwdevice_find_type_by_name(name);
if (check != type) {
fprintf(stderr, "Type %d maps to name %s maps to type %d.\n",
type, name, check);
return -1;
}
found = 0;
err = test_device(type, name, NULL, NULL, 0);
if (err < 0) {
fprintf(stderr, "Test failed for %s with default options.\n", name);
return -1;
}
if (err == 0) {
fprintf(stderr, "Test passed for %s with default options.\n", name);
++found;
}
for (i = 0; i < FF_ARRAY_ELEMS(test_devices); i++) {
if (test_devices[i].type != type)
continue;
for (j = 0; test_devices[i].possible_devices[j]; j++) {
err = test_device(type, name,
test_devices[i].possible_devices[j],
NULL, 0);
if (err < 0) {
fprintf(stderr, "Test failed for %s with device %s.\n",
name, test_devices[i].possible_devices[j]);
return -1;
}
if (err == 0) {
fprintf(stderr, "Test passed for %s with device %s.\n",
name, test_devices[i].possible_devices[j]);
++found;
}
}
}
return !found;
}
int main(void)
{
enum AVHWDeviceType type = AV_HWDEVICE_TYPE_NONE;
int pass, fail, skip, err;
pass = fail = skip = 0;
while (1) {
type = av_hwdevice_iterate_types(type);
if (type == AV_HWDEVICE_TYPE_NONE)
break;
err = test_device_type(type);
if (err == 0)
++pass;
else if (err < 0)
++fail;
else
++skip;
}
fprintf(stderr, "Attempted to test %d device types: "
"%d passed, %d failed, %d skipped.\n",
pass + fail + skip, pass, fail, skip);
return fail > 0;
}

View File

@ -131,6 +131,7 @@ include $(SRC_PATH)/tests/fate/gif.mak
include $(SRC_PATH)/tests/fate/h264.mak
include $(SRC_PATH)/tests/fate/hap.mak
include $(SRC_PATH)/tests/fate/hevc.mak
include $(SRC_PATH)/tests/fate/hw.mak
include $(SRC_PATH)/tests/fate/id3v2.mak
include $(SRC_PATH)/tests/fate/image.mak
include $(SRC_PATH)/tests/fate/indeo.mak
@ -215,6 +216,10 @@ $(addprefix fate-, $(IGNORE_TESTS)): REPORT=ignore
fate:: $(FATE)
# Tests requiring hardware support are not included in a default fate run.
fate-hw: $(FATE_HW-yes)
FATE += $(FATE_HW-yes)
$(FATE) $(FATE_TESTS-no): export PROGSUF = $(PROGSSUF)
$(FATE) $(FATE_TESTS-no): $(FATE_UTILS:%=tests/%$(HOSTEXESUF))
@echo "TEST $(@:fate-%=%)"

6
tests/fate/hw.mak Normal file
View File

@ -0,0 +1,6 @@
FATE_HWCONTEXT += fate-hwdevice
fate-hwdevice: libavutil/tests/hwdevice$(EXESUF)
fate-hwdevice: CMD = run libavutil/tests/hwdevice
fate-hwdevice: CMP = null
FATE_HW-$(CONFIG_AVUTIL) += $(FATE_HWCONTEXT)