2024-04-15 16:35:44 +00:00
|
|
|
/*
|
|
|
|
* 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>
|
|
|
|
|
2024-05-03 00:30:15 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <io.h>
|
|
|
|
#else
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2024-04-15 16:35:44 +00:00
|
|
|
// 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
|
|
|
|
|
|
|
|
// Global handle
|
|
|
|
static mpv_handle *ctx;
|
|
|
|
// Temporary output file
|
|
|
|
static const char *out_path;
|
|
|
|
|
|
|
|
static void exit_cleanup(void)
|
|
|
|
{
|
|
|
|
if (ctx)
|
|
|
|
mpv_destroy(ctx);
|
2024-05-21 20:31:17 +00:00
|
|
|
if (out_path && *out_path)
|
2024-04-15 16:35:44 +00:00
|
|
|
unlink(out_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void check_api_error(int status)
|
|
|
|
{
|
|
|
|
if (status < 0)
|
|
|
|
fail("libmpv error: %s\n", mpv_error_string(status));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wait_done(void)
|
|
|
|
{
|
|
|
|
while (1) {
|
|
|
|
mpv_event *ev = mpv_wait_event(ctx, -1.0);
|
|
|
|
if (ev->event_id == MPV_EVENT_NONE)
|
|
|
|
continue;
|
|
|
|
printf("event: %s\n", mpv_event_name(ev->event_id));
|
|
|
|
if (ev->event_id == MPV_EVENT_SHUTDOWN)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-21 20:31:17 +00:00
|
|
|
static void check_output(FILE *fp)
|
2024-04-15 16:35:44 +00:00
|
|
|
{
|
2024-05-21 20:31:17 +00:00
|
|
|
fseek(fp, 0, SEEK_END);
|
|
|
|
long size = ftell(fp);
|
2024-04-15 16:35:44 +00:00
|
|
|
if (size < 100)
|
2024-05-21 20:31:17 +00:00
|
|
|
fail("did not encode anything\n");
|
2024-04-15 16:35:44 +00:00
|
|
|
|
2024-07-07 03:24:20 +00:00
|
|
|
char magic[4];
|
2024-05-21 20:31:17 +00:00
|
|
|
fseek(fp, 0, SEEK_SET);
|
2024-07-07 03:24:20 +00:00
|
|
|
size_t ret = fread(magic, sizeof(magic), 1, fp);
|
2024-04-15 16:35:44 +00:00
|
|
|
static const char ebml_magic[] = {26, 69, 223, 163};
|
2024-07-07 03:24:20 +00:00
|
|
|
if (ret != 1 || memcmp(magic, ebml_magic, sizeof(magic)) != 0)
|
2024-05-21 20:31:17 +00:00
|
|
|
fail("output was not Matroska\n");
|
2024-04-15 16:35:44 +00:00
|
|
|
|
|
|
|
puts("output file ok");
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
atexit(exit_cleanup);
|
|
|
|
|
|
|
|
ctx = mpv_create();
|
|
|
|
if (!ctx)
|
|
|
|
return 1;
|
|
|
|
|
2024-05-21 20:31:17 +00:00
|
|
|
static char path[] = "./testout.XXXXXX";
|
2024-06-18 18:22:07 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
out_path = _mktemp(path);
|
2024-05-21 20:31:17 +00:00
|
|
|
if (!out_path || !*out_path)
|
|
|
|
fail("tmpfile failed\n");
|
2024-06-18 18:22:07 +00:00
|
|
|
#else
|
|
|
|
int fd = mkstemp(path);
|
|
|
|
if (fd == -1)
|
|
|
|
fail("tmpfile failed\n");
|
|
|
|
out_path = path;
|
|
|
|
#endif
|
2024-05-21 20:31:17 +00:00
|
|
|
|
2024-04-15 16:35:44 +00:00
|
|
|
check_api_error(mpv_set_option_string(ctx, "o", out_path));
|
|
|
|
check_api_error(mpv_set_option_string(ctx, "of", "matroska"));
|
|
|
|
check_api_error(mpv_set_option_string(ctx, "end", "1.5"));
|
2024-05-21 22:52:09 +00:00
|
|
|
check_api_error(mpv_set_option_string(ctx, "terminal", "yes"));
|
|
|
|
check_api_error(mpv_set_option_string(ctx, "msg-level", "all=v"));
|
2024-04-15 16:35:44 +00:00
|
|
|
|
|
|
|
if (mpv_initialize(ctx) != 0)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
check_api_error(mpv_set_option_string(ctx, "idle", "once"));
|
|
|
|
|
|
|
|
const char *cmd[] = {"loadfile", "av://lavfi:testsrc", NULL};
|
|
|
|
check_api_error(mpv_command(ctx, cmd));
|
|
|
|
|
|
|
|
wait_done();
|
|
|
|
mpv_destroy(ctx);
|
|
|
|
ctx = NULL;
|
|
|
|
|
2024-05-21 20:31:17 +00:00
|
|
|
FILE *output = fopen(out_path, "rb");
|
|
|
|
if (!output)
|
|
|
|
fail("output file doesn't exist\n");
|
|
|
|
check_output(output);
|
|
|
|
fclose(output);
|
2024-04-15 16:35:44 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|