| /* |
| * 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; |
| } |