mirror of
https://github.com/mpv-player/mpv
synced 2025-01-11 09:29:29 +00:00
0279a44d93
Add env and detach arguments. This means the command.c code must use the "new" mp_subprocess2(). So also take this as an opportunity to clean up. win32 support gets broken by it, because it never made the switch to the newer function. The new detach parameter makes the "run" command fully redundant, but I guess we'll keep it for simplicity. But change its implementation to use mp_subprocess2() (couldn't do this earlier, because win32). Privately, I'm going to use the "env" argument to add a key binding that starts a shell with a FILE environment variable set to the currently playing file, so this is very useful to me. Note: breaks windows, so for example youtube-dl on windows will not work anymore. mp_subprocess2() has to be implemented. The old functions are gone, and subprocess-win.c is not built anymore. It will probably work on Cygwin.
160 lines
4.0 KiB
C
160 lines
4.0 KiB
C
#include "options/path.h"
|
|
#include "osdep/subprocess.h"
|
|
#include "player/core.h"
|
|
#include "tests.h"
|
|
|
|
static const struct unittest *unittests[] = {
|
|
&test_chmap,
|
|
&test_gl_video,
|
|
&test_img_format,
|
|
&test_json,
|
|
&test_linked_list,
|
|
&test_paths,
|
|
&test_repack_sws,
|
|
#if HAVE_ZIMG
|
|
&test_repack, // zimg only due to cross-checking with zimg.c
|
|
&test_repack_zimg,
|
|
#endif
|
|
NULL
|
|
};
|
|
|
|
bool run_tests(struct MPContext *mpctx)
|
|
{
|
|
char *sel = mpctx->opts->test_mode;
|
|
assert(sel && sel[0]);
|
|
|
|
if (strcmp(sel, "help") == 0) {
|
|
MP_INFO(mpctx, "Available tests:\n");
|
|
for (int n = 0; unittests[n]; n++)
|
|
MP_INFO(mpctx, " %s\n", unittests[n]->name);
|
|
MP_INFO(mpctx, " all-simple\n");
|
|
return true;
|
|
}
|
|
|
|
struct test_ctx ctx = {
|
|
.global = mpctx->global,
|
|
.log = mpctx->log,
|
|
.ref_path = "test/ref",
|
|
.out_path = "test/out",
|
|
};
|
|
|
|
if (!mp_path_isdir(ctx.ref_path)) {
|
|
MP_FATAL(mpctx, "Must be run from git repo root dir.\n");
|
|
abort();
|
|
}
|
|
mp_mkdirp(ctx.out_path);
|
|
assert(mp_path_isdir(ctx.out_path));
|
|
|
|
int num_run = 0;
|
|
|
|
for (int n = 0; unittests[n]; n++) {
|
|
const struct unittest *t = unittests[n];
|
|
|
|
// Exactly 1 entrypoint please.
|
|
assert(MP_IS_POWER_OF_2(
|
|
(t->run ? (1 << 1) : 0)));
|
|
|
|
bool run = false;
|
|
run |= strcmp(sel, "all-simple") == 0 && !t->is_complex;
|
|
run |= strcmp(sel, t->name) == 0;
|
|
|
|
if (run) {
|
|
if (t->run)
|
|
t->run(&ctx);
|
|
num_run++;
|
|
}
|
|
}
|
|
|
|
MP_INFO(mpctx, "%d unittests successfully run.\n", num_run);
|
|
|
|
return num_run > 0; // still error if none
|
|
}
|
|
|
|
#ifdef NDEBUG
|
|
static_assert(false, "don't define NDEBUG for tests");
|
|
#endif
|
|
|
|
void assert_int_equal_impl(const char *file, int line, int64_t a, int64_t b)
|
|
{
|
|
if (a != b) {
|
|
printf("%s:%d: %"PRId64" != %"PRId64"\n", file, line, a, b);
|
|
abort();
|
|
}
|
|
}
|
|
|
|
void assert_string_equal_impl(const char *file, int line,
|
|
const char *a, const char *b)
|
|
{
|
|
if (strcmp(a, b) != 0) {
|
|
printf("%s:%d: '%s' != '%s'\n", file, line, a, b);
|
|
abort();
|
|
}
|
|
}
|
|
|
|
void assert_float_equal_impl(const char *file, int line,
|
|
double a, double b, double tolerance)
|
|
{
|
|
if (fabs(a - b) > tolerance) {
|
|
printf("%s:%d: %f != %f\n", file, line, a, b);
|
|
abort();
|
|
}
|
|
}
|
|
|
|
FILE *test_open_out(struct test_ctx *ctx, const char *name)
|
|
{
|
|
char *path = mp_tprintf(4096, "%s/%s", ctx->out_path, name);
|
|
FILE *f = fopen(path, "wb");
|
|
if (!f) {
|
|
MP_FATAL(ctx, "Could not open '%s' for writing.\n", path);
|
|
abort();
|
|
}
|
|
return f;
|
|
}
|
|
|
|
void assert_text_files_equal_impl(const char *file, int line,
|
|
struct test_ctx *ctx, const char *ref,
|
|
const char *new, const char *err)
|
|
{
|
|
char *path_ref = mp_tprintf(4096, "%s/%s", ctx->ref_path, ref);
|
|
char *path_new = mp_tprintf(4096, "%s/%s", ctx->out_path, new);
|
|
|
|
struct mp_subprocess_opts opts = {
|
|
.exe = "diff",
|
|
.args = (char*[]){"diff", "-u", "--", path_ref, path_new, 0},
|
|
.fds = { {0, .src_fd = 0}, {1, .src_fd = 1}, {2, .src_fd = 2} },
|
|
.num_fds = 3,
|
|
};
|
|
|
|
struct mp_subprocess_result res;
|
|
mp_subprocess2(&opts, &res);
|
|
|
|
if (res.error || res.exit_status) {
|
|
if (res.error)
|
|
MP_WARN(ctx, "Note: %s\n", mp_subprocess_err_str(res.error));
|
|
MP_FATAL(ctx, "Giving up.\n");
|
|
abort();
|
|
}
|
|
}
|
|
|
|
static void hexdump(const uint8_t *d, size_t size)
|
|
{
|
|
printf("|");
|
|
while (size--) {
|
|
printf(" %02x", d[0]);
|
|
d++;
|
|
}
|
|
printf(" |\n");
|
|
}
|
|
|
|
void assert_memcmp_impl(const char *file, int line,
|
|
const void *a, const void *b, size_t size)
|
|
{
|
|
if (memcmp(a, b, size) == 0)
|
|
return;
|
|
|
|
printf("%s:%d: mismatching data:\n", file, line);
|
|
hexdump(a, size);
|
|
hexdump(b, size);
|
|
abort();
|
|
}
|