This feature adds the following functions:

 - set_malloc()
 - set_free()

Which is a safe malloc wrapper, that frees all memory after a test has finished.

Reviewed-on: #1
This commit is contained in:
2025-06-25 23:17:02 +00:00
parent 3f78db4cb7
commit ed41c04209
10 changed files with 377 additions and 70 deletions

View File

@@ -2,15 +2,11 @@
Tiny C testing framework.
## Build
## Building a test.
Building is quite simple. To build a test `test_fac.c` just pick your favourite compiler and run.
To build a test, first build the library. Then link it to the `test.c` via:
gcc Small-Enough-Tester/set.c test_fac.c -I Small-Enough-Tester/ -D COLORIZED -lm -o test_fac
Running a test is equally simple just execute the compiled binary.
./test_fac
gcc src/*.c test.c -o test
## LICENSE

View File

@@ -1,8 +0,0 @@
#include "set.h"
#include "set_asserts.h"
TEST(Basic) { ASSERT_EQ(2, 2); }
SUIT(Basic) { ADD_TEST(Basic); }
BUNDLE() { ADD_SUIT(Basic); }

33
include/list.h Normal file
View File

@@ -0,0 +1,33 @@
#include <stddef.h>
struct SETBlockMeta
{
struct SETBlockMeta *next;
struct SETBlockMeta *prev;
struct SETBlockMeta *end;
};
/**
* Adds a new element to the list. The MetaData should sit at the start
* of a block.
*
* head (SETBlockMeta) List head.
* next (SETBlockMeta) Next meta to append.
*/
void set_ll_append(struct SETBlockMeta *head, struct SETBlockMeta *next);
/**
* Frees all blocks in the list.
*
* head (SETBlockMeta) List head.
*/
void set_ll_free_all(struct SETBlockMeta *head);
/**
* Frees the block with the given address and returns the new head.
*
* head (SETBlockMeta) List head.
* address (void*) Address to free.
*/
struct SETBlockMeta *set_ll_free_one(struct SETBlockMeta *head, void *address);

View File

@@ -2,17 +2,23 @@
#define _GNU_SOURCE
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/mman.h>
#include <sys/shm.h>
#include "utils.h"
#ifndef INCLUDE_SET_H
#define INCLUDE_SET_H
#define SET_MAX_ERROR_MSG_SIZE 256
#define SET_MAX_NAME_SIZE 64
/**
* Meta data for one test function.
*
* See also: struct SETSuit
*/
struct SETest
{
void (*function)(struct SETest *test);
@@ -21,6 +27,11 @@ struct SETest
bool passed;
};
/**
* Meta data for one test suit.
*
* See also: struct SETest
*/
struct SETSuit
{
const char *name;
@@ -32,26 +43,98 @@ struct SETSuit
bool passed;
};
char *format_string(const char *fmt, ...);
// To be implemented by user.
/**
* Internal function header for bundle. Use BUNDLE() macro to define bundle in
* test.
*/
void set_bundle_suits(struct SETSuit **suits, int *counter, bool count);
int create_shared_suit_space(size_t size);
/**
* Empty suit_setup. Resolving EMPTY in a suit constructor gets
* us here.
*
* Note: Use EMPTY instead of this function.
*/
static bool EMPTY_suit_setup() { return true; }
/**
* Empty suit_tear_down. Resolving EMPTY in a suit constructor gets
* us here.
*
* Note: Use EMPTY instead of this function.
*/
static bool EMPTY_suit_tear_down() { return true; }
/**
* Internal header for global setup. Use SETUP() macro instead.
*/
bool set_up();
/**
* Internal header for global tear down. Use TEAR_DOWN() macro instead.
*/
bool tear_down();
/**
* Global setup function.
* Gets called at the start of the whole test program.
*
* Should return True if setup was successfull, false if not.
* Tests are not executed on false.
*
* Note: This function has to be set in the tests even if it is not used.
*
* See also: NO_SETUP()
*/
#define SETUP() bool set_up()
/**
* Empty setup function returning true.
*/
#define NO_SETUP \
bool set_up() { return true; }
/**
* Setup function that allways fails. May be usefull to set if tests shouldn't
* be executed.
*/
#define DEACTIVATE \
bool set_up() \
{ \
printf("All tests are deactivated.\n"); \
return false; \
}
/**
* Global tear down function.
* Gets called at the end of the while test program.
*
* Should return true if tear down was successfull, false if not.
*
* Note: This function has to be set in the tests even if it is not used.
*/
#define TEAR_DOWN() bool tear_down()
/**
* Empty tear down function returning true.
*/
#define NO_TEAR_DOWN \
bool tear_down() { return true; }
/**
* Section in the tests to register all suits for execution.
*
* See also: ADD_SUIT()
* See also: SUIT, SUIT_ST
*/
#define BUNDLE() \
void set_bundle_suits(struct SETSuit **suits, int *counter, bool count)
/**
* Used in BUNDLE() to register suit.
*
* suit_name (Parameter) Name of the test suit to add.
*/
#define ADD_SUIT(suit_name) \
if (!count) \
{ \
@@ -59,8 +142,20 @@ bool tear_down();
} \
(*counter)++;
// We manually allocate suits via mmap be able to mark them DONTFORK.
// Since we don't need the suits in a test fork.
/**
* Creates a new test suit with local setup and tear_down function.
*
* suit_name (Parameter) Name of the suit. (Must be globally unique for suits.)
* setup_func (SUIT_SETUP) Suit setup function name. (Write name EMPTY if no
* setup is required.) tear_down_func (SUIT_TEAR_DOWN) Suit tear down function
* name. (Write name EMPTY if no tear down is required)
*
* See also: SUIT_SETUP()
* See also: SUIT_TEAR_DOWN()
*
* Technical Note: We manually allocate suits via mmap to be able to mark them
* DONTFORK. Since we don't need the suits in a test fork.
*/
#define SUIT_ST(suit_name, setup_func, tear_down_func) \
void suit_name##_suit(struct SETSuit *suit, bool count); \
struct SETSuit *suit_name##_suit_contructor() \
@@ -89,12 +184,38 @@ bool tear_down();
} \
void suit_name##_suit(struct SETSuit *suit, bool count)
/**
* Suit local setup function. If registered in suit it get's executed before the
* suit starts the first test.
*
* suit_name (Parameter) Name of the suit setup function. (Must be globally
* unique for suit setup functions!)
*/
#define SUIT_SETUP(suit_name) bool suit_name##_suit_setup()
/**
* Suit local tear down function. If registered in suit it get's executed after
* the last test was executed.
*
* suit_name (Parameter) Name of the suit setup function. (Must be globally
* unique for suit tear down functions!)
*/
#define SUIT_TEAR_DOWN(suit_name) bool suit_name##_suit_tear_down()
/**
* Suit without setup or tear down.
*
* suit_name (Parameter) Name of the suit. (Must be globally unique for suits!)
*/
#define SUIT(suit_name) SUIT_ST(suit_name, EMPTY, EMPTY)
/**
* Registers a test with a test suit.
*
* test_name (Parameter) Name of the test.
*
* See also: TEST()
*/
#define ADD_TEST(test_name) \
if (!count) \
{ \
@@ -102,7 +223,13 @@ bool tear_down();
} \
suit->len++;
// TEST MACRO
/**
* Test function. If registered in suit it gets executed be that suit.
*
* test_name (Parameter) Name of the test (Must be globally unique for tests!)
*
* See also: set_asserts.h
*/
#define TEST(test_name) \
void test_name##_test(struct SETest *test); \
void test_name##_test_constructor(struct SETest *test) \

54
include/utils.h Normal file
View File

@@ -0,0 +1,54 @@
#include <stddef.h>
/*
* Returns pointer to formatted the string.
*
* fmt (const char*) Formatter string.
* ... (args) Arguments.
*
* Note: The string is dynamically allocated and must be freed.
*/
char *format_string(const char *fmt, ...);
/**
* Internal function to create shared memory for all tests in a suit.
*
* size (size_t) Size of space in bytes.
*/
int create_shared_suit_space(size_t size);
/**
* Wrapper for malloc function that notes down the allocated blocks. All
* blocks allocated with this function get freed after test execution.
* This function should only be used if the object might still be in memory
* once a test fails. Manual allocation should be preferred.
*
* n (size_t) Amount of bytes to allocate.
*/
void *set_malloc(size_t n);
/**
* Wrapper for fee(void*).
*
* See also: set_malloc()
*/
void set_free(void *address);
/**
* Internal function to free all allocated and registered memory.
*/
void set_free_all();
/**
* Wrapper for calloc.
*
* See also: set_malloc()
*/
void *set_calloc(size_t n, size_t size);
/**
* Wrapper for realloc().
*
* See also: set_malloc()
*/
void *set_realloc(size_t n);

44
src/list.c Normal file
View File

@@ -0,0 +1,44 @@
#include "list.h"
#include <stdlib.h>
void set_ll_append(struct SETBlockMeta *head, struct SETBlockMeta *next)
{
while (head->next)
head = head->next;
head->next = next;
next->prev = head;
}
void set_ll_free_all(struct SETBlockMeta *head)
{
while (head)
{
void *block_start = head;
head = head->next;
free(block_start);
}
}
struct SETBlockMeta *set_ll_free_one(struct SETBlockMeta *head, void *address)
{
struct SETBlockMeta *meta = address - (sizeof(struct SETBlockMeta));
if (meta == head)
{
struct SETBlockMeta *ret = meta->next;
ret->prev = NULL;
ret->end = meta->end;
free(meta);
return ret;
}
meta->prev->next = meta->next;
if (meta->next)
meta->next->prev = meta->prev;
free(meta);
return head;
}

View File

@@ -2,7 +2,6 @@
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/types.h>
@@ -56,41 +55,6 @@
""
//clang-format on
char *format_string(const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
size_t size = vsnprintf(NULL, 0, fmt, args);
char *out = (char *)malloc(size + 1);
if (!out)
{
fprintf(stderr, "%s", "Failed to allocate for format string.\n");
}
va_end(args);
va_start(args, fmt);
vsnprintf(out, size + 1, fmt, args);
va_end(args);
return out;
}
int create_shared_suit_space(size_t size)
{
int suit_space_id = shmget(IPC_PRIVATE, size, IPC_CREAT | 0600);
if (suit_space_id == -1)
{
fprintf(stdout, "Couldn't create suite space of size: %lu\n", size);
perror("shmget");
exit(1);
}
return suit_space_id;
}
static void log_test_summary(struct SETest *test)
{
@@ -122,6 +86,7 @@ static void dispatch_single_test(struct SETest *test)
{
test->function(test);
log_test_summary(test);
set_free_all();
exit(0);
}
}

78
src/utils.c Normal file
View File

@@ -0,0 +1,78 @@
#include "utils.h"
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/shm.h>
#include "list.h"
#include <unistd.h>
struct SETBlockMeta *block_meta_head = NULL;
char *format_string(const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
size_t size = vsnprintf(NULL, 0, fmt, args);
char *out = (char *)malloc(size + 1);
if (!out)
{
fprintf(stderr, "%s", "Failed to allocate for format string.\n");
}
va_end(args);
va_start(args, fmt);
vsnprintf(out, size + 1, fmt, args);
va_end(args);
return out;
}
int create_shared_suit_space(size_t size)
{
int suit_space_id = shmget(IPC_PRIVATE, size, IPC_CREAT | 0600);
if (suit_space_id == -1)
{
fprintf(stdout, "Couldn't create suite space of size: %lu\n", size);
perror("shmget");
exit(1);
}
return suit_space_id;
}
void *set_malloc(size_t n)
{
void *blocks = malloc(n + sizeof(struct SETBlockMeta));
struct SETBlockMeta *meta = blocks;
meta->next = NULL;
if (block_meta_head == NULL)
{
meta->end = meta;
block_meta_head = meta;
}
else
{
meta->end = NULL;
set_ll_append(block_meta_head->end, meta);
block_meta_head->end = meta;
}
return blocks + sizeof(struct SETBlockMeta);
}
void set_free(void *address)
{
block_meta_head = set_ll_free_one(block_meta_head, address);
}
void set_free_all()
{
set_ll_free_all(block_meta_head);
block_meta_head = NULL;
}

View File

@@ -1,5 +1,6 @@
#include "set.h"
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include "set_asserts.h"
@@ -18,12 +19,7 @@ SETUP()
printf("Setting up.\n");
return true;
}
TEAR_DOWN()
{
printf("Ending execution.");
return true;
}
NO_TEAR_DOWN
TEST(Faculty_Basic)
{
@@ -38,19 +34,19 @@ TEST(Faculty_Negative)
ASSERT_EQ(fac(0), 1);
}
SUIT_SETUP(Basic)
SUIT_SETUP(Basic_Setup)
{
fprintf(stdout, "Some suit setup.\n");
return true;
}
SUIT_TEAR_DOWN(Basic)
SUIT_TEAR_DOWN(Basic_Tear_Down)
{
fprintf(stdout, "Some suit teardown\n");
return true;
}
SUIT_ST(Basic, Basic, Basic)
SUIT_ST(Basic, Basic_Setup, Basic_Tear_Down)
{
ADD_TEST(Faculty_Basic);
ADD_TEST(Faculty_Negative);
@@ -62,7 +58,29 @@ TEST(Other_Basic)
ASSERT_FALSE(false);
}
SUIT_ST(Other, Basic, EMPTY) { ADD_TEST(Other_Basic); }
TEST(Other_With_Malloc)
{
int *some_array = set_malloc(20 * sizeof(int));
for (int i = 0; i < 20; i++)
{
some_array[i] = i;
}
for (int i = 0; i < 20; i++)
{
fprintf(stdout, "%d ", some_array[i]);
}
fprintf(stdout, ".\n");
ASSERT_TRUE(false);
}
SUIT_ST(Other, Basic_Setup, EMPTY)
{
ADD_TEST(Other_Basic);
ADD_TEST(Other_With_Malloc);
}
BUNDLE()
{