This is a unit testing application for Game Boy roms. It includes a CPU emulator, and loads test configurations from TOML files.
Within the test config you can create a heading for each test you want to run, and assign default and expected values for registers.
The first heading (for example, add-one
) determines the initial state, while the ".result
" heading (for example, add-one.result
) describes the expected result.
If the expected result does not match the final state, the test will fail.
```toml [add-one] b = 1 [add-one.result] a = 2
[add-two] b = 2 [add-two.result] a = 3 ```
You can assign any cpu register to an integer.
In addition, 16-bit registers may be assigned a quoted label if a symfile is loaded.
To determine which function should run in each test, assign a label to pc
.
Possible registers are:
- a
- b
- c
- d
- e
- h
- l
- bc
- de
- hl
- pc
- sp
In addition the flags can be assigned to either true
or false
.
Possible flags are:
- f.z
- f.n
- f.h
- f.c
Note that the flags must be quoted in the config file because of the dot:
toml
"f.z" = false
Finally, memory can be assigned a value in the config file by surrounding a label name in square brackets. You can either assign an 8-bit integer, a string, or an array of either. Like the flags, memory addresses must be quoted because of the square brackets: ```toml
"[wString]" = ["Hello, world!", 0] ```
Sometimes you have configurations which should apply to all tests, like a global variable or the stack pointer. Any configurations at the top of the file (before a heading) are global and apply to all tests.
```toml sp = "wStack.end"
[my-test] pc = "MyTest" a = 42 [my-test.result] b = 42 ```
If the test result is absent, the test will always pass unless it crashes.
Creating an exhaustive set of tests by hand might be tedious, so remember that you an always generate tests in any language of your choice.
rs
fn main() {
for i in 0..8 {
println!("
[my-test{i}]
pc = \"GetBitA\"
a = {i}
[my-test{i}.result]
a = {}", 1 << i
);
}
}
Then pipe this into evunit.
You can use -
to read from stdin.
./config_generator | evunit -c - bin/rom.gb
And you can always use cat
to add a handwritten file into the mix.
./config_generator | cat config.toml - | evunit -c - bin/rom.gb
A test is complete when either a crash address is reached, the test times out, or pc
is equal to the caller
specified in the config file (default is 0xFFFF
).
evunit pushes the caller
value to the stack before running your test, meaning that in most scenarios a ret
will end the test.
When the caller
value is successfully reached, evunit checks to see if the result matches what was expected.
In addition to registers, there are a few other options you can configure. All of these can be configured globally as well as per-test.
Sets the caller address.
This address is pushed to the stack when a test begins, allowing ret
to end the test.
toml
caller = "Main"
By default, caller
is set to 0xFFFF
.
Marks an address as a "crash", causing the test to fail if pc
reaches it.
This is useful for crash handler functions such as rst $38
toml
crash = 0x38
An array of values can also be used.
toml
crash = [0x38, "crash"]
Enables or disables printing register info after executing ld b, b
and ld d, d
.
Enabled by default.
This configuration can only be used globally.
toml
enable-breakpoints = true
enable-breakpoints = false
Marks an address as an "exit", causing the test to end if pc
reaches it.
The results will then be verified.
toml
exit = "SomeFunction.exit"
An array of values can also be used.
Sets the maximum number of cycles before a test fails. This is useful if you have code that tends to get stuck in infinite loops, or code which can take an extremely long time to complete. The default value is 65536.
toml
timeout = 65536
When a test fails, it outputs some cpu registers depending on the failure reason to help you diagnose the issue.
However, sometimes you need to check the state of memory as well; this can be accomplished with the --dump-dir
(-d
) flag.
Pass a directory to this flag and when any test fails a text dump of memory will be placed in the provided directory.
bash
evunit -c fail.toml -d dump/ rom.gb
The dump is simply a giant list of bytes, with headers for each memory type:
[WRAM 0]
0xc000: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
0xc010: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
0xc020: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
0xc030: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
0xc040: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
0xc050: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
0xc060: 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0 0x0
...