mirror of
https://github.com/Ninjdai1/pokeemerald.git
synced 2025-01-13 07:03:40 +01:00
Recover from test crashes
This commit is contained in:
parent
b31f10d124
commit
4637d7e7ee
@ -45,6 +45,13 @@ SECTIONS {
|
|||||||
test/*.o(COMMON);
|
test/*.o(COMMON);
|
||||||
*libc.a:sbrkr.o(COMMON);
|
*libc.a:sbrkr.o(COMMON);
|
||||||
end = .;
|
end = .;
|
||||||
|
|
||||||
|
/* .persistent starts at 0x3007F00 */
|
||||||
|
/* WARNING: This is the end of the IRQ stack, if there's too
|
||||||
|
* much data it WILL be overwritten. */
|
||||||
|
. = 0x7F00;
|
||||||
|
test/*.o(.persistent);
|
||||||
|
|
||||||
. = 0x8000;
|
. = 0x8000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,6 +13,7 @@ enum TestResult
|
|||||||
TEST_RESULT_INVALID,
|
TEST_RESULT_INVALID,
|
||||||
TEST_RESULT_ERROR,
|
TEST_RESULT_ERROR,
|
||||||
TEST_RESULT_TIMEOUT,
|
TEST_RESULT_TIMEOUT,
|
||||||
|
TEST_RESULT_CRASH,
|
||||||
TEST_RESULT_TODO,
|
TEST_RESULT_TODO,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -38,8 +39,6 @@ struct TestRunnerState
|
|||||||
{
|
{
|
||||||
u8 state;
|
u8 state;
|
||||||
u8 exitCode;
|
u8 exitCode;
|
||||||
s32 tests;
|
|
||||||
s32 passes;
|
|
||||||
const char *skipFilename;
|
const char *skipFilename;
|
||||||
const struct Test *test;
|
const struct Test *test;
|
||||||
u32 processCosts[MAX_PROCESSES];
|
u32 processCosts[MAX_PROCESSES];
|
||||||
|
@ -15,6 +15,16 @@ void CB2_TestRunner(void);
|
|||||||
EWRAM_DATA struct TestRunnerState gTestRunnerState;
|
EWRAM_DATA struct TestRunnerState gTestRunnerState;
|
||||||
EWRAM_DATA struct FunctionTestRunnerState *gFunctionTestRunnerState;
|
EWRAM_DATA struct FunctionTestRunnerState *gFunctionTestRunnerState;
|
||||||
|
|
||||||
|
enum {
|
||||||
|
CURRENT_TEST_STATE_ESTIMATE,
|
||||||
|
CURRENT_TEST_STATE_RUN,
|
||||||
|
};
|
||||||
|
|
||||||
|
__attribute__((section(".persistent"))) static struct {
|
||||||
|
u32 address:28;
|
||||||
|
u32 state:1;
|
||||||
|
} sCurrentTest = {0};
|
||||||
|
|
||||||
void TestRunner_Battle(const struct Test *);
|
void TestRunner_Battle(const struct Test *);
|
||||||
|
|
||||||
static bool32 MgbaOpen_(void);
|
static bool32 MgbaOpen_(void);
|
||||||
@ -51,6 +61,47 @@ enum
|
|||||||
STATE_EXIT,
|
STATE_EXIT,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static u32 MinCostProcess(void)
|
||||||
|
{
|
||||||
|
u32 i;
|
||||||
|
u32 minCost, minCostProcess;
|
||||||
|
|
||||||
|
minCost = gTestRunnerState.processCosts[0];
|
||||||
|
minCostProcess = 0;
|
||||||
|
for (i = 1; i < gTestRunnerN; i++)
|
||||||
|
{
|
||||||
|
if (gTestRunnerState.processCosts[i] < minCost)
|
||||||
|
{
|
||||||
|
minCost = gTestRunnerState.processCosts[i];
|
||||||
|
minCostProcess = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return minCostProcess;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Greedily assign tests to processes based on estimated cost.
|
||||||
|
// TODO: Make processCosts a min heap.
|
||||||
|
static u32 AssignCostToRunner(void)
|
||||||
|
{
|
||||||
|
u32 minCostProcess;
|
||||||
|
|
||||||
|
if (gTestRunnerState.test->runner == &gAssumptionsRunner)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
minCostProcess = MinCostProcess();
|
||||||
|
|
||||||
|
// XXX: If estimateCost returns only on some processes, or
|
||||||
|
// returns inconsistent results then processCosts will be
|
||||||
|
// inconsistent and some tests may not run.
|
||||||
|
if (gTestRunnerState.test->runner->estimateCost)
|
||||||
|
gTestRunnerState.processCosts[minCostProcess] += gTestRunnerState.test->runner->estimateCost(gTestRunnerState.test->data);
|
||||||
|
else
|
||||||
|
gTestRunnerState.processCosts[minCostProcess] += 1;
|
||||||
|
|
||||||
|
return minCostProcess;
|
||||||
|
}
|
||||||
|
|
||||||
void CB2_TestRunner(void)
|
void CB2_TestRunner(void)
|
||||||
{
|
{
|
||||||
switch (gTestRunnerState.state)
|
switch (gTestRunnerState.state)
|
||||||
@ -65,12 +116,43 @@ void CB2_TestRunner(void)
|
|||||||
|
|
||||||
gIntrTable[7] = Intr_Timer2;
|
gIntrTable[7] = Intr_Timer2;
|
||||||
|
|
||||||
gTestRunnerState.state = STATE_NEXT_TEST;
|
// The current test restarted the ROM (e.g. by jumping to NULL).
|
||||||
|
if (sCurrentTest.address != 0)
|
||||||
|
{
|
||||||
|
gTestRunnerState.test = __start_tests;
|
||||||
|
while ((uintptr_t)gTestRunnerState.test != sCurrentTest.address)
|
||||||
|
{
|
||||||
|
AssignCostToRunner();
|
||||||
|
gTestRunnerState.test++;
|
||||||
|
}
|
||||||
|
if (sCurrentTest.state == CURRENT_TEST_STATE_ESTIMATE)
|
||||||
|
{
|
||||||
|
u32 runner = MinCostProcess();
|
||||||
|
gTestRunnerState.processCosts[runner] += 1;
|
||||||
|
if (runner == gTestRunnerI)
|
||||||
|
{
|
||||||
|
gTestRunnerState.state = STATE_REPORT_RESULT;
|
||||||
|
gTestRunnerState.result = TEST_RESULT_CRASH;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gTestRunnerState.state = STATE_NEXT_TEST;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gTestRunnerState.state = STATE_REPORT_RESULT;
|
||||||
|
gTestRunnerState.result = TEST_RESULT_CRASH;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gTestRunnerState.state = STATE_NEXT_TEST;
|
||||||
|
gTestRunnerState.test = __start_tests - 1;
|
||||||
|
}
|
||||||
gTestRunnerState.exitCode = 0;
|
gTestRunnerState.exitCode = 0;
|
||||||
gTestRunnerState.tests = 0;
|
|
||||||
gTestRunnerState.passes = 0;
|
|
||||||
gTestRunnerState.skipFilename = NULL;
|
gTestRunnerState.skipFilename = NULL;
|
||||||
gTestRunnerState.test = __start_tests - 1;
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case STATE_NEXT_TEST:
|
case STATE_NEXT_TEST:
|
||||||
@ -108,40 +190,19 @@ void CB2_TestRunner(void)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Greedily assign tests to processes based on estimated cost.
|
sCurrentTest.address = (uintptr_t)gTestRunnerState.test;
|
||||||
// TODO: Make processCosts a min heap.
|
sCurrentTest.state = CURRENT_TEST_STATE_ESTIMATE;
|
||||||
if (gTestRunnerState.test->runner != &gAssumptionsRunner)
|
|
||||||
{
|
|
||||||
u32 i;
|
|
||||||
u32 minCost, minCostProcess;
|
|
||||||
minCost = gTestRunnerState.processCosts[0];
|
|
||||||
minCostProcess = 0;
|
|
||||||
for (i = 1; i < gTestRunnerN; i++)
|
|
||||||
{
|
|
||||||
if (gTestRunnerState.processCosts[i] < minCost)
|
|
||||||
{
|
|
||||||
minCost = gTestRunnerState.processCosts[i];
|
|
||||||
minCostProcess = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (minCostProcess == gTestRunnerI)
|
if (AssignCostToRunner() == gTestRunnerI)
|
||||||
gTestRunnerState.state = STATE_RUN_TEST;
|
gTestRunnerState.state = STATE_RUN_TEST;
|
||||||
else
|
else
|
||||||
gTestRunnerState.state = STATE_NEXT_TEST;
|
gTestRunnerState.state = STATE_NEXT_TEST;
|
||||||
|
|
||||||
// XXX: If estimateCost exits only on some processes then
|
|
||||||
// processCosts will be inconsistent.
|
|
||||||
if (gTestRunnerState.test->runner->estimateCost)
|
|
||||||
gTestRunnerState.processCosts[minCostProcess] += gTestRunnerState.test->runner->estimateCost(gTestRunnerState.test->data);
|
|
||||||
else
|
|
||||||
gTestRunnerState.processCosts[minCostProcess] += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case STATE_RUN_TEST:
|
case STATE_RUN_TEST:
|
||||||
gTestRunnerState.state = STATE_REPORT_RESULT;
|
gTestRunnerState.state = STATE_REPORT_RESULT;
|
||||||
|
sCurrentTest.state = CURRENT_TEST_STATE_RUN;
|
||||||
if (gTestRunnerState.test->runner->setUp)
|
if (gTestRunnerState.test->runner->setUp)
|
||||||
gTestRunnerState.test->runner->setUp(gTestRunnerState.test->data);
|
gTestRunnerState.test->runner->setUp(gTestRunnerState.test->data);
|
||||||
gTestRunnerState.test->runner->run(gTestRunnerState.test->data);
|
gTestRunnerState.test->runner->run(gTestRunnerState.test->data);
|
||||||
@ -186,11 +247,8 @@ void CB2_TestRunner(void)
|
|||||||
const char *color;
|
const char *color;
|
||||||
const char *result;
|
const char *result;
|
||||||
|
|
||||||
gTestRunnerState.tests++;
|
|
||||||
|
|
||||||
if (gTestRunnerState.result == gTestRunnerState.expectedResult)
|
if (gTestRunnerState.result == gTestRunnerState.expectedResult)
|
||||||
{
|
{
|
||||||
gTestRunnerState.passes++;
|
|
||||||
color = "\e[32m";
|
color = "\e[32m";
|
||||||
MgbaPrintf_(":N%s", gTestRunnerState.test->name);
|
MgbaPrintf_(":N%s", gTestRunnerState.test->name);
|
||||||
}
|
}
|
||||||
@ -243,6 +301,9 @@ void CB2_TestRunner(void)
|
|||||||
case TEST_RESULT_TIMEOUT:
|
case TEST_RESULT_TIMEOUT:
|
||||||
result = "TIMEOUT";
|
result = "TIMEOUT";
|
||||||
break;
|
break;
|
||||||
|
case TEST_RESULT_CRASH:
|
||||||
|
result = "CRASH";
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
result = "UNKNOWN";
|
result = "UNKNOWN";
|
||||||
break;
|
break;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user