2019-05-28 20:54:13 -05:00
|
|
|
// 28 may 2019
|
|
|
|
#include "test.h"
|
|
|
|
|
|
|
|
struct errorCase {
|
|
|
|
const char *name;
|
2019-06-02 00:36:53 -05:00
|
|
|
const char *file;
|
|
|
|
long line;
|
2019-05-28 20:54:13 -05:00
|
|
|
bool caught;
|
|
|
|
char *msgGot;
|
|
|
|
const char *msgWant;
|
|
|
|
struct errorCase *next;
|
|
|
|
};
|
|
|
|
|
2019-06-09 12:57:02 -05:00
|
|
|
static void catalogProgrammerError(const char *msg, void *data)
|
2019-05-28 20:54:13 -05:00
|
|
|
{
|
2019-06-09 12:57:02 -05:00
|
|
|
static struct errorCase *c;
|
2019-05-31 01:58:57 -05:00
|
|
|
size_t n;
|
|
|
|
|
2019-06-09 12:57:02 -05:00
|
|
|
c->caught = true;
|
|
|
|
if (strcmp(msg, c->msgWant) != 0) {
|
2019-05-31 01:58:57 -05:00
|
|
|
n = strlen(msg);
|
2019-06-09 13:03:27 -05:00
|
|
|
c->msgGot = testingUtilStrdup(msg);
|
2019-05-28 20:54:13 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct errorCase *newCase(void)
|
|
|
|
{
|
|
|
|
struct errorCase *p;
|
|
|
|
|
|
|
|
p = (struct errorCase *) malloc(sizeof (struct errorCase));
|
|
|
|
if (p == NULL) {
|
2019-05-31 09:38:47 -05:00
|
|
|
caseError = caseErrorMemoryExhausted;
|
2019-05-28 20:54:13 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
memset(p, 0, sizeof (struct errorCase));
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void freeCases(struct errorCase *first)
|
|
|
|
{
|
|
|
|
struct errorCase *p, *next;
|
|
|
|
|
|
|
|
p = first;
|
|
|
|
while (p != NULL) {
|
|
|
|
if (p->msgGot != NULL)
|
2019-06-09 13:03:27 -05:00
|
|
|
testingUtilFreeStrdup(p->msgGot);
|
2019-05-28 20:54:13 -05:00
|
|
|
next = p->next;
|
|
|
|
free(p);
|
|
|
|
p = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void reportCases(testingT *t, struct errorCase *p)
|
|
|
|
{
|
|
|
|
while (p != NULL) {
|
2019-06-02 00:36:53 -05:00
|
|
|
testingTLogfFull(t, p->file, p->line, "*** %s", p->name);
|
2019-05-28 20:54:13 -05:00
|
|
|
if (!p->caught) {
|
2019-06-02 00:36:53 -05:00
|
|
|
testingTErrorfFull(t, p->file, p->line, "%s did not throw a programmer error; should have", p->name);
|
2019-05-28 20:54:13 -05:00
|
|
|
p = p->next;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (p->msgGot != NULL)
|
2019-06-09 12:44:46 -05:00
|
|
|
// TODO use diff
|
2019-06-02 00:59:08 -05:00
|
|
|
testingTErrorfFull(t, p->file, p->line, "%s message doesn't contain expected substring:" diff("%s"),
|
2019-06-02 00:51:40 -05:00
|
|
|
p->name, p->msgGot, p->msgWant);
|
2019-05-28 20:54:13 -05:00
|
|
|
p = p->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define allcallsCase(f, ...) { \
|
|
|
|
current = newCase(); \
|
2019-05-31 09:38:47 -05:00
|
|
|
if (caseError != NULL) \
|
2019-05-28 20:54:13 -05:00
|
|
|
return first; \
|
|
|
|
current->name = #f "()"; \
|
2019-06-02 00:36:53 -05:00
|
|
|
current->file = __FILE__; \
|
|
|
|
current->line = __LINE__; \
|
2019-05-28 20:54:13 -05:00
|
|
|
current->msgWant = "attempt to call " #f "() " allcallsMsgSuffix; \
|
2019-05-28 21:53:40 -05:00
|
|
|
f(__VA_ARGS__); \
|
2019-05-28 20:54:13 -05:00
|
|
|
if (first == NULL) \
|
|
|
|
first = current; \
|
|
|
|
if (last != NULL) \
|
|
|
|
last->next = current; \
|
|
|
|
last = current; \
|
2019-05-31 09:38:47 -05:00
|
|
|
if (caseError != NULL) \
|
2019-05-28 20:54:13 -05:00
|
|
|
return first; \
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct errorCase *runCasesBeforeInit(void)
|
|
|
|
{
|
|
|
|
struct errorCase *first = NULL;
|
|
|
|
struct errorCase *last = NULL;
|
|
|
|
|
|
|
|
#define allcallsMsgSuffix "before uiInit()"
|
|
|
|
allcallsCase(uiQueueMain, NULL, NULL);
|
|
|
|
#include "allcalls.h"
|
|
|
|
#undef allcallsMsgSuffix
|
|
|
|
return first;
|
|
|
|
}
|
|
|
|
|
|
|
|
testingTestInSet(beforeTests, FunctionsFailBeforeInit)
|
|
|
|
{
|
|
|
|
struct errorCase *cases;
|
|
|
|
|
2019-05-31 09:38:47 -05:00
|
|
|
caseError = NULL;
|
2019-05-28 20:54:13 -05:00
|
|
|
uiprivTestHookReportProgrammerError(catalogProgrammerError);
|
|
|
|
cases = runCasesBeforeInit();
|
|
|
|
uiprivTestHookReportProgrammerError(NULL);
|
2019-05-31 09:38:47 -05:00
|
|
|
if (caseError != NULL) {
|
2019-05-28 20:54:13 -05:00
|
|
|
freeCases(cases);
|
2019-05-31 09:38:47 -05:00
|
|
|
testingTErrorf(t, "%s running tests", caseError);
|
|
|
|
if (caseError != caseErrorMemoryExhausted && caseError != caseErrorEncodingError)
|
|
|
|
free(caseError);
|
|
|
|
caseError = NULL;
|
2019-05-31 10:17:11 -05:00
|
|
|
testingTFailNow(t);
|
2019-05-28 20:54:13 -05:00
|
|
|
}
|
|
|
|
reportCases(t, cases);
|
|
|
|
freeCases(cases);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct errorCase *runCasesWrongThread(void)
|
|
|
|
{
|
|
|
|
struct errorCase *first = NULL;
|
|
|
|
struct errorCase *last = NULL;
|
|
|
|
|
|
|
|
#define allcallsMsgSuffix "on a thread other than the GUI thread"
|
|
|
|
#include "allcalls.h"
|
|
|
|
#undef allcallsMsgSuffix
|
|
|
|
return first;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wrongThreadThreadProc(void *data)
|
|
|
|
{
|
|
|
|
struct errorCase **pCases = (struct errorCase **) data;
|
|
|
|
|
|
|
|
uiprivTestHookReportProgrammerError(catalogProgrammerError);
|
|
|
|
*pCases = runCasesWrongThread();
|
|
|
|
uiprivTestHookReportProgrammerError(NULL);
|
2019-05-31 09:38:47 -05:00
|
|
|
// do this now in case memory was exhausted and something gets allocated before we return to the main thread
|
|
|
|
if (caseError != NULL) {
|
2019-05-28 20:54:13 -05:00
|
|
|
freeCases(*pCases);
|
|
|
|
*pCases = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
testingTest(FunctionsFailOnWrongThread)
|
|
|
|
{
|
|
|
|
struct errorCase *cases;
|
|
|
|
threadThread *thread;
|
|
|
|
threadSysError err;
|
|
|
|
|
2019-05-31 09:38:47 -05:00
|
|
|
caseError = NULL;
|
2019-05-28 20:54:13 -05:00
|
|
|
err = threadNewThread(wrongThreadThreadProc, &cases, &thread);
|
|
|
|
if (err != 0)
|
|
|
|
testingTFatalf(t, "error creating thread: " threadSysErrorFmt, threadSysErrorFmtArg(err));
|
|
|
|
err = threadThreadWaitAndFree(thread);
|
|
|
|
if (err != 0)
|
|
|
|
testingTFatalf(t, "error waiting for thread to finish: " threadSysErrorFmt, threadSysErrorFmtArg(err));
|
2019-05-31 09:38:47 -05:00
|
|
|
if (caseError != NULL) {
|
|
|
|
freeCases(cases);
|
|
|
|
testingTErrorf(t, "%s running tests", caseError);
|
|
|
|
if (caseError != caseErrorMemoryExhausted && caseError != caseErrorEncodingError)
|
|
|
|
free(caseError);
|
|
|
|
caseError = NULL;
|
2019-05-31 10:17:11 -05:00
|
|
|
testingTFailNow(t);
|
2019-05-31 09:38:47 -05:00
|
|
|
}
|
2019-05-28 20:54:13 -05:00
|
|
|
reportCases(t, cases);
|
|
|
|
freeCases(cases);
|
|
|
|
}
|