diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml
index ddbfd90b776db..b800f542bd393 100644
--- a/.github/workflows/build.yml
+++ b/.github/workflows/build.yml
@@ -217,7 +217,7 @@ jobs:
vulkan-headers \
wayland-protocols
./ci/build-freebsd.sh
- meson test -C build
+ meson test -v -C build
msys2:
runs-on: windows-latest
diff --git a/ci/build-freebsd.sh b/ci/build-freebsd.sh
index c393fb25fee32..5168b67c183d3 100755
--- a/ci/build-freebsd.sh
+++ b/ci/build-freebsd.sh
@@ -10,6 +10,7 @@ export LDFLAGS="$LDFLAGS -L/usr/local/lib"
meson setup build \
--werror \
-Dc_args="-Wno-error=deprecated -Wno-error=deprecated-declarations" \
+ -Db_sanitize=address \
-Diconv=disabled \
-Dlibmpv=true \
-Dlua=enabled \
diff --git a/player/video.c b/player/video.c
index 48a3165f9d150..baa6db8d84c10 100644
--- a/player/video.c
+++ b/player/video.c
@@ -129,9 +129,9 @@ void uninit_video_out(struct MPContext *mpctx)
uninit_video_chain(mpctx);
if (mpctx->video_out) {
vo_destroy(mpctx->video_out);
+ mpctx->video_out = NULL;
mp_notify(mpctx, MPV_EVENT_VIDEO_RECONFIG, NULL);
}
- mpctx->video_out = NULL;
}
static void vo_chain_uninit(struct vo_chain *vo_c)
diff --git a/test/libmpv_test.c b/test/libmpv_test.c
index fafef6ade901d..e8b27b064f249 100644
--- a/test/libmpv_test.c
+++ b/test/libmpv_test.c
@@ -1,271 +1,28 @@
-/*
- * 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 .
- */
+struct A {
+ char *a;
+ float b;
+ float c;
+};
-#include
-#include
-#include
-#include
-#include
-#include
+struct B {
+ float b;
+ float c;
+};
-// 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
+int foo(struct A *a, struct B *b);
-// 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)
+__attribute__((noinline))
+int foo(struct A *a, struct B *b)
{
- 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);
-}
+ b->b = a->b;
+ b->c = a->c;
-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);
+ return b->c;
}
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;
+ struct A a = {0};
+ struct B b;
+ return foo(&a, &b);
}