// 27 february 2018 #include #include #include #include #include #include #include #include "../ui.h" #ifdef libuiOSHeader #include libuiOSHeader #endif #ifdef __cplusplus extern "C" { #endif #define testingprivImplName(basename) testingprivImpl ## basename #define testingprivScaffoldName(basename) testingprivScaffold ## basename #define testingprivMkScaffold(basename) \ static int testingprivScaffoldName(basename)(void) { int ret = 0; testingprivImplName(basename)(&ret); return ret; } // references: // - https://gitlab.gnome.org/GNOME/glib/blob/master/glib/gconstructor.h // - https://gitlab.gnome.org/GNOME/glib/blob/master/gio/glib-compile-resources.c // - https://msdn.microsoft.com/en-us/library/bb918180.aspx #define testingprivCtorName(basename) testingprivCtor ## basename #define testingprivCtorPtrName(basename) testingprivCtorPtr ## basename #if defined(__GNUC__) #define testingprivMkCtor(basename) \ __attribute__((constructor)) static void testingprivCtorName(basename)(void) { testingprivRegisterTest(#basename, testingprivScaffoldName(basename)); } #elif defined(_MSC_VER) #define testingprivMkCtor(basename) \ static int testingprivCtorName(basename)(void) { testingprivRegisterTest(#basename, testingprivScaffoldName(basename)); return 0; } \ __pragma(section(".CRT$XCU",read)) \ __declspec(allocate(".CRT$XCU")) static int (*testingprivCtorPtrName(basename))(void) = testingprivCtorName(basename); #elif defined(__SUNPRO_C) #define testingprivMkCtor(basename) \ static void testingprivCtorName(basename)(void) { testingprivRegisterTest(#basename, testingprivScaffoldName(basename)); } \ _Pragma(init(testingprivCtorName(basename))) #else #error unknown compiler for making constructors in C; cannot continue #endif #define Test(basename) \ void testingprivImplName(basename)(int *testingprivRet); \ testingprivMkScaffold(basename) \ testingprivMkCtor(basename) \ void testingprivImplName(basename)(int *testingprivRet) #define Test(Name) \ testingprivMk(Test ## Name) // These can only be called directly from the Test functions. // TODO make it otherwise #define TestFail() (*testingprivRet = 1) #define TestFailNow() {TestFail(); return;} // see https://mesonbuild.com/Unit-tests.html#skipped-tests-and-hard-errors #define TestSkipNow() {*testingprivRet = 77; return;} #define TestLogf(...) \ (testingprivLogf(stdout, __FILE__, __LINE__, __VA_ARGS__)) #define TestErrorf(...) \ {testingprivLogf(stderr, __FILE__, __LINE__, __VA_ARGS__); TestFail();} #define TestFatalf(...) \ {testingprivLogf(stderr, __FILE__, __LINE__, __VA_ARGS__); TestFailNow();} // TODO remember if this needs to go to stdout or to stderr #define TestSkipf(...) \ {testingprivLogf(stderr, __FILE__, __LINE__, __VA_ARGS__); TestSkipNow();} extern void testingprivRegisterTest(const char *name, int (*f)(void)); #include "../sharedbits/printfwarn_header.h" sharedbitsPrintfFunc( extern void testingprivLogf(FILE *f, const char *filename, long line, const char *fmt, ...), 4, 5); #undef sharedbitsPrintfFunc // end of test framework definitions #define diff(fmt) "\ngot " fmt "\nwant " fmt #ifdef __cplusplus } #endif