diff options
Diffstat (limited to 'test/libmpv_test.c')
-rw-r--r-- | test/libmpv_test.c | 271 |
1 files changed, 271 insertions, 0 deletions
diff --git a/test/libmpv_test.c b/test/libmpv_test.c new file mode 100644 index 0000000..fafef6a --- /dev/null +++ b/test/libmpv_test.c @@ -0,0 +1,271 @@ +/* + * This file is part of mpv. + * + * mpv 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. + * + * mpv 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 mpv. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <inttypes.h> +#include <libmpv/client.h> +#include <stdarg.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +// Stolen from osdep/compiler.h +#ifdef __GNUC__ +#define PRINTF_ATTRIBUTE(a1, a2) __attribute__ ((format(printf, a1, a2))) +#define MP_NORETURN __attribute__((noreturn)) +#else +#define PRINTF_ATTRIBUTE(a1, a2) +#define MP_NORETURN +#endif + +// Broken crap with __USE_MINGW_ANSI_STDIO +#if defined(__MINGW32__) && defined(__GNUC__) && !defined(__clang__) +#undef PRINTF_ATTRIBUTE +#define PRINTF_ATTRIBUTE(a1, a2) __attribute__ ((format (gnu_printf, a1, a2))) +#endif + +// Dummy values for test_options_and_properties +static const char *str = "string"; +static int flag = 1; +static int64_t int_ = 20; +static double double_ = 1.5; + +// Global handle. +static mpv_handle *ctx; + + +MP_NORETURN PRINTF_ATTRIBUTE(1, 2) +static void fail(const char *fmt, ...) +{ + if (fmt) { + va_list va; + va_start(va, fmt); + vfprintf(stderr, fmt, va); + va_end(va); + } + mpv_destroy(ctx); + exit(1); +} + +static void check_api_error(int status) +{ + if (status < 0) + fail("mpv API error: %s\n", mpv_error_string(status)); +} + +static void check_double(const char *property, double expect) +{ + double result_double; + check_api_error(mpv_get_property(ctx, property, MPV_FORMAT_DOUBLE, &result_double)); + if (expect != result_double) + fail("Double: expected '%f' but got '%f'!\n", expect, result_double); +} + +static void check_flag(const char *property, int expect) +{ + int result_flag; + check_api_error(mpv_get_property(ctx, property, MPV_FORMAT_FLAG, &result_flag)); + if (expect != result_flag) + fail("Flag: expected '%d' but got '%d'!\n", expect, result_flag); +} + +static void check_int(const char *property, int64_t expect) +{ + int64_t result_int; + check_api_error(mpv_get_property(ctx, property, MPV_FORMAT_INT64, &result_int)); + if (expect != result_int) + fail("Int: expected '%" PRId64 "' but got '%" PRId64 "'!\n", expect, result_int); +} + +static void check_string(const char *property, const char *expect) +{ + char *result_string; + check_api_error(mpv_get_property(ctx, property, MPV_FORMAT_STRING, &result_string)); + if (strcmp(expect, result_string) != 0) + fail("String: expected '%s' but got '%s'!\n", expect, result_string); + mpv_free(result_string); +} + +static void check_results(const char *properties[], enum mpv_format formats[]) +{ + for (int i = 0; properties[i]; i++) { + switch (formats[i]) { + case MPV_FORMAT_STRING: + check_string(properties[i], str); + break; + case MPV_FORMAT_FLAG: + check_flag(properties[i], flag); + break; + case MPV_FORMAT_INT64: + check_int(properties[i], int_); + break; + case MPV_FORMAT_DOUBLE: + check_double(properties[i], double_); + break; + } + } +} + +static void set_options_and_properties(const char *options[], const char *properties[], + enum mpv_format formats[]) +{ + for (int i = 0; options[i]; i++) { + switch (formats[i]) { + case MPV_FORMAT_STRING: + check_api_error(mpv_set_option(ctx, options[i], formats[i], &str)); + check_api_error(mpv_set_property(ctx, properties[i], formats[i], &str)); + break; + case MPV_FORMAT_FLAG: + check_api_error(mpv_set_option(ctx, options[i], formats[i], &flag)); + check_api_error(mpv_set_property(ctx, properties[i], formats[i], &flag)); + break; + case MPV_FORMAT_INT64: + check_api_error(mpv_set_option(ctx, options[i], formats[i], &int_)); + check_api_error(mpv_set_property(ctx, properties[i], formats[i], &int_)); + break; + case MPV_FORMAT_DOUBLE: + check_api_error(mpv_set_option(ctx, options[i], formats[i], &double_)); + check_api_error(mpv_set_property(ctx, properties[i], formats[i], &double_)); + break; + } + } +} + +static void test_file_loading(char *file) +{ + const char *cmd[] = {"loadfile", file, NULL}; + check_api_error(mpv_command(ctx, cmd)); + int loaded = 0; + int finished = 0; + while (!finished) { + mpv_event *event = mpv_wait_event(ctx, 0); + switch (event->event_id) { + case MPV_EVENT_FILE_LOADED: + // make sure it loads before exiting + loaded = 1; + break; + case MPV_EVENT_END_FILE: + if (loaded) + finished = 1; + break; + } + } + if (!finished) + fail("Unable to load test file!\n"); +} + +static void test_lavfi_complex(char *file) +{ + const char *cmd[] = {"loadfile", file, NULL}; + check_api_error(mpv_command(ctx, cmd)); + int finished = 0; + int loaded = 0; + while (!finished) { + mpv_event *event = mpv_wait_event(ctx, 0); + switch (event->event_id) { + case MPV_EVENT_FILE_LOADED: + // Add file as external and toggle lavfi-complex on. + if (!loaded) { + check_api_error(mpv_set_property_string(ctx, "external-files", file)); + const char *add_cmd[] = {"video-add", file, "auto", NULL}; + check_api_error(mpv_command(ctx, add_cmd)); + check_api_error(mpv_set_property_string(ctx, "lavfi-complex", "[vid1] [vid2] vstack [vo]")); + } + loaded = 1; + break; + case MPV_EVENT_END_FILE: + if (loaded) + finished = 1; + break; + } + } + if (!finished) + fail("Lavfi complex failed!\n"); +} + +// Ensure that setting options/properties work correctly and +// have the expected values. +static void test_options_and_properties(void) +{ + // Order matters. string -> flag -> int -> double (repeat) + // One for set_option the other for set_property + const char *options[] = { + "screen-name", + "save-position-on-quit", + "cursor-autohide", + "speed", + NULL + }; + + const char *properties[] = { + "fs-screen-name", + "shuffle", + "sub-pos", + "window-scale", + NULL + }; + + // Must match above ordering. + enum mpv_format formats[] = { + MPV_FORMAT_STRING, + MPV_FORMAT_FLAG, + MPV_FORMAT_INT64, + MPV_FORMAT_DOUBLE, + }; + + set_options_and_properties(options, properties, formats); + + check_api_error(mpv_initialize(ctx)); + + check_results(options, formats); + check_results(properties, formats); + + // Ensure the format is still MPV_FORMAT_FLAG for these property types. + mpv_node result_node; + check_api_error(mpv_get_property(ctx, "idle-active", MPV_FORMAT_NODE, &result_node)); + if (result_node.format != MPV_FORMAT_FLAG) + fail("Node: expected mpv format '%d' but got '%d'!\n", MPV_FORMAT_FLAG, result_node.format); + + // Always should be true. + if (result_node.u.flag != 1) + fail("Node: expected 1 but got %d'!\n", result_node.u.flag); +} + +int main(int argc, char *argv[]) +{ + if (argc != 2) + return 1; + + ctx = mpv_create(); + if (!ctx) + return 1; + + check_api_error(mpv_set_option_string(ctx, "vo", "null")); + check_api_error(mpv_set_option_string(ctx, "terminal", "yes")); + check_api_error(mpv_set_option_string(ctx, "msg-level", "all=debug")); + + const char *fmt = "================ TEST: %s ================\n"; + + printf(fmt, "test_options_and_properties"); + test_options_and_properties(); + printf(fmt, "test_file_loading"); + test_file_loading(argv[1]); + printf(fmt, "test_lavfi_complex"); + test_lavfi_complex(argv[1]); + + mpv_destroy(ctx); + return 0; +} |