initial set of timeline unit tests

This commit is contained in:
Sebastian Krzyszkowiak 2019-04-03 02:59:48 +02:00
parent c3096eee7b
commit 83bd62277f
No known key found for this signature in database
GPG key ID: E8F235CF3BDBC3FF
6 changed files with 569 additions and 0 deletions

View file

@ -7,3 +7,4 @@ list(APPEND CMAKE_MODULE_PATH "${PROJECT_SOURCE_DIR}/cmake")
include(libsuperderpy) include(libsuperderpy)
add_subdirectory(src) add_subdirectory(src)
add_subdirectory(test)

49
cmake/FindCMocka.cmake Normal file
View file

@ -0,0 +1,49 @@
# - Try to find CMocka
# Once done this will define
#
# CMOCKA_ROOT_DIR - Set this variable to the root installation of CMocka
#
# Read-Only variables:
# CMOCKA_FOUND - system has CMocka
# CMOCKA_INCLUDE_DIR - the CMocka include directory
# CMOCKA_LIBRARIES - Link these to use CMocka
# CMOCKA_DEFINITIONS - Compiler switches required for using CMocka
#
#=============================================================================
# Copyright (c) 2011-2012 Andreas Schneider <asn@cryptomilk.org>
#
# Distributed under the OSI-approved BSD License (the "License");
# see accompanying file Copyright.txt for details.
#
# This software is distributed WITHOUT ANY WARRANTY; without even the
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
# See the License for more information.
#=============================================================================
#
find_path(CMOCKA_INCLUDE_DIR
NAMES
cmocka.h
PATHS
${CMOCKA_ROOT_DIR}/include
)
find_library(CMOCKA_LIBRARY
NAMES
cmocka cmocka_shared
PATHS
${CMOCKA_ROOT_DIR}/include
)
if (CMOCKA_LIBRARY)
set(CMOCKA_LIBRARIES
${CMOCKA_LIBRARIES}
${CMOCKA_LIBRARY}
)
endif (CMOCKA_LIBRARY)
include(FindPackageHandleStandardArgs)
find_package_handle_standard_args(CMocka DEFAULT_MSG CMOCKA_LIBRARIES CMOCKA_INCLUDE_DIR)
# show the CMOCKA_INCLUDE_DIR and CMOCKA_LIBRARIES variables only in the advanced view
mark_as_advanced(CMOCKA_INCLUDE_DIR CMOCKA_LIBRARIES)

9
test/CMakeLists.txt Normal file
View file

@ -0,0 +1,9 @@
find_package(CMocka)
if (CMOCKA_FOUND)
set(CMAKE_INSTALL_RPATH "\$ORIGIN/../src")
add_executable(engine-tests tests.c timeline.c)
target_link_libraries(engine-tests cmocka libsuperderpy)
else(CMOCKA_FOUND)
message("CMocka not found; tests disabled.")
endif(CMOCKA_FOUND)

34
test/tests.c Normal file
View file

@ -0,0 +1,34 @@
/*
* Copyright (c) Sebastian Krzyszkowiak <dos@dosowisko.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "tests.h"
int engine_setup(void** state) {
char* args[2] = {"", "--debug"};
*state = libsuperderpy_init(2, args, "test", (struct Params){});
libsuperderpy_start(*state);
return 0;
}
int engine_teardown(void** state) {
libsuperderpy_destroy(*state);
return 0;
}
int main(int argc, char** argv) {
return test_timeline();
}

37
test/tests.h Normal file
View file

@ -0,0 +1,37 @@
/*! \file tests.h
* \brief Headers for test files.
*/
/*
* Copyright (c) Sebastian Krzyszkowiak <dos@dosowisko.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef LIBSUPERDERPY_TESTS_H
#define LIBSUPERDERPY_TESTS_H
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <stdint.h>
#include <cmocka.h>
#include "internal.h"
int engine_setup(void** state);
int engine_teardown(void** state);
int test_timeline(void);
#endif

439
test/timeline.c Normal file
View file

@ -0,0 +1,439 @@
/*
* Copyright (c) Sebastian Krzyszkowiak <dos@dosowisko.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "tests.h"
// -----------------------------------------
#define EPSILON 0.001
#define HandleTest(index) \
{ \
float* del = TM_Arg(index); \
if (del) { \
if (*del) { \
PrintConsole(game, "Eating %f delta (from %f).", *del, action->delta); \
action->delta -= *del; \
} \
if (action->delta < 0 || !*del) { \
action->delta = 0; \
} \
PrintConsole(game, "Delta left: %f", action->delta); \
} \
function_called(); \
} \
(void)0
static TM_ACTION(DoNothing) {
TM_RunningOnly;
PrintConsole(game, "Doing nothing.");
HandleTest(0);
return true;
}
static void ActionInitialized(void) {
function_called();
}
static void ActionDestroyed(void) {
function_called();
}
static void ActionStarted(void) {
function_called();
}
static void ActionStopped(void) {
function_called();
}
static TM_ACTION(SetState) {
bool* quit = TM_Arg(0);
int* val = TM_Arg(1);
*val = action->state;
SUPPRESS_WARNING("-Wcovered-switch-default")
switch (action->state) {
case TM_ACTIONSTATE_INIT:
PrintConsole(game, "Init");
ActionInitialized();
break;
case TM_ACTIONSTATE_START:
PrintConsole(game, "Started");
ActionStarted();
break;
case TM_ACTIONSTATE_STOP:
PrintConsole(game, "Stopped");
ActionStopped();
break;
case TM_ACTIONSTATE_RUNNING:
PrintConsole(game, "Running");
HandleTest(2);
return *quit;
case TM_ACTIONSTATE_DESTROY:
PrintConsole(game, "Destroyed");
ActionDestroyed();
break;
default:
PrintConsole(game, "Unknown!");
assert_true(false);
break;
}
SUPPRESS_END
return true;
}
static TM_ACTION(AssertDelta) {
TM_RunningOnly;
float* delta = TM_Arg(0);
assert_float_equal(action->delta, *delta, EPSILON);
PrintConsole(game, "Delta given: %f", action->delta);
HandleTest(1);
return true;
}
static TM_ACTION(Loop) {
TM_RunningOnly;
float* duration = TM_Arg(0);
float* delta = TM_Arg(1);
float d = action->delta;
if (delta) {
d = *delta;
}
*duration -= d;
PrintConsole(game, "Looping (for %f) with duration left: %f", action->delta, *duration);
HandleTest(2);
return *duration <= 0.0;
}
// -----------------------------------------
static int timeline_setup(void** state) {
return engine_setup(state);
}
static int timeline_teardown(void** state) {
return engine_teardown(state);
}
// -----------------------------------------
static void timeline_action(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
TM_AddAction(timeline, DoNothing, NULL);
expect_function_call(DoNothing);
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_action_lifecycle(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
int val = -1;
bool quit = true;
expect_function_call(ActionInitialized);
TM_AddAction(timeline, SetState, TM_Args(&quit, &val));
assert_int_equal(val, TM_ACTIONSTATE_INIT);
expect_function_call(ActionStarted);
expect_function_call(SetState);
expect_function_call(ActionStopped);
expect_function_call(ActionDestroyed);
TM_Process(timeline, 1);
assert_int_equal(val, TM_ACTIONSTATE_DESTROY);
TM_Destroy(timeline);
}
static void timeline_action_lifecycle_repeated(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
int val = -1;
bool quit = false;
expect_function_call(ActionInitialized);
TM_AddAction(timeline, SetState, TM_Args(&quit, &val));
assert_int_equal(val, TM_ACTIONSTATE_INIT);
expect_function_call(ActionStarted);
expect_function_call(SetState);
TM_Process(timeline, 1);
assert_int_equal(val, TM_ACTIONSTATE_RUNNING);
quit = true;
expect_function_call(SetState);
expect_function_call(ActionStopped);
expect_function_call(ActionDestroyed);
TM_Process(timeline, 1);
assert_int_equal(val, TM_ACTIONSTATE_DESTROY);
TM_Destroy(timeline);
}
static void timeline_action_queue(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 1;
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
expect_function_call(DoNothing);
TM_Process(timeline, delta);
expect_function_call(DoNothing);
TM_Process(timeline, delta);
TM_Destroy(timeline);
}
static void timeline_multiple_actions_in_one_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
TM_AddAction(timeline, DoNothing, NULL);
TM_AddAction(timeline, DoNothing, NULL);
expect_function_calls(DoNothing, 2);
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_multiple_delays_in_one_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
TM_AddDelay(timeline, 1);
TM_AddDelay(timeline, 1);
TM_AddAction(timeline, DoNothing, NULL);
expect_function_call(DoNothing);
TM_Process(timeline, 3);
TM_Destroy(timeline);
}
static void timeline_delta_less_than_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 0.2;
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
expect_function_calls(DoNothing, 2);
TM_Process(timeline, 2);
TM_Destroy(timeline);
}
static void timeline_delta_equals_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 1;
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
expect_function_call(DoNothing);
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_stops_when_delta_eaten(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 0.2;
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
expect_function_calls(DoNothing, 5);
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_delay_less_than_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 1, val = 0.4;
TM_AddDelay(timeline, 0.6);
TM_AddAction(timeline, AssertDelta, TM_Args(&val, &delta));
expect_function_call(AssertDelta);
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_delay_equals_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 1;
TM_AddDelay(timeline, 1);
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_delay_more_than_process(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 1;
TM_AddDelay(timeline, 2);
TM_AddAction(timeline, DoNothing, TM_Args(&delta));
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_delay_eats_delta(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float delta = 2;
TM_AddDelay(timeline, 1);
TM_AddAction(timeline, AssertDelta, TM_Args(&delta));
expect_function_call(AssertDelta);
TM_Process(timeline, 3);
TM_Destroy(timeline);
}
static void timeline_delay_with_multiple_processes(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float val = 0.4;
TM_AddDelay(timeline, 1);
TM_AddAction(timeline, AssertDelta, TM_Args(&val));
TM_Process(timeline, 0.4);
expect_function_call(AssertDelta);
TM_Process(timeline, 1);
TM_Destroy(timeline);
}
static void timeline_repeat_when_delta_left(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
int repeats = 8;
float delta = 1, time = delta * repeats;
TM_AddAction(timeline, Loop, TM_Args(&time, &delta, &delta));
expect_function_calls(Loop, repeats);
TM_Process(timeline, time);
TM_Destroy(timeline);
}
static void timeline_no_repeating_when_eating_zero_delta(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
float time = 42, delta = 0;
TM_AddAction(timeline, Loop, TM_Args(&time, &delta));
TM_AddAction(timeline, DoNothing, NULL);
expect_function_call(Loop);
TM_Process(timeline, 69);
TM_Destroy(timeline);
}
static void timeline_destroyed(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
int val = -1;
bool quit = false;
expect_function_call(ActionInitialized);
TM_AddAction(timeline, SetState, TM_Args(&quit, &val));
assert_int_equal(val, TM_ACTIONSTATE_INIT);
expect_function_call(ActionDestroyed);
TM_Destroy(timeline);
assert_int_equal(val, TM_ACTIONSTATE_DESTROY);
}
static void timeline_destroyed_while_running(void** state) {
struct Timeline* timeline = TM_Init(*state, NULL, __func__);
int val = -1;
bool quit = false;
expect_function_call(ActionInitialized);
TM_AddAction(timeline, SetState, TM_Args(&quit, &val));
assert_int_equal(val, TM_ACTIONSTATE_INIT);
expect_function_call(ActionStarted);
expect_function_call(SetState);
TM_Process(timeline, 1);
assert_int_equal(val, TM_ACTIONSTATE_RUNNING);
expect_function_call(ActionStopped);
expect_function_call(ActionDestroyed);
TM_Destroy(timeline);
assert_int_equal(val, TM_ACTIONSTATE_DESTROY);
}
int test_timeline(void) {
al_set_app_name("libsuperderpy");
const struct CMUnitTest timeline_tests[] = {
cmocka_unit_test(timeline_action),
cmocka_unit_test(timeline_action_lifecycle),
cmocka_unit_test(timeline_action_lifecycle_repeated),
cmocka_unit_test(timeline_action_queue),
cmocka_unit_test(timeline_multiple_actions_in_one_process),
cmocka_unit_test(timeline_multiple_delays_in_one_process),
cmocka_unit_test(timeline_delta_less_than_process),
cmocka_unit_test(timeline_delta_equals_process),
cmocka_unit_test(timeline_delay_less_than_process),
cmocka_unit_test(timeline_delay_equals_process),
cmocka_unit_test(timeline_stops_when_delta_eaten),
cmocka_unit_test(timeline_delay_more_than_process),
cmocka_unit_test(timeline_delay_eats_delta),
cmocka_unit_test(timeline_repeat_when_delta_left),
cmocka_unit_test(timeline_no_repeating_when_eating_zero_delta),
cmocka_unit_test(timeline_delay_with_multiple_processes),
cmocka_unit_test(timeline_destroyed),
cmocka_unit_test(timeline_destroyed_while_running),
// TODO: delayed actions, background queue
};
return cmocka_run_group_tests(timeline_tests, timeline_setup, timeline_teardown);
}