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.
		
			
				
	
	
		
			227 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			227 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * 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;
 | 
						|
}
 |