mirror of https://github.com/mpv-player/mpv
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();
|
|
}
|