2021-05-15 22:13:31 -04:00
# Running SerenityOS Tests
2021-02-26 17:12:04 -05:00
2021-05-15 22:13:31 -04:00
There are two classes of tests built during a SerenityOS build: host tests and target tests. Host tests run on the build
machine, and use [Lagom ](../Meta/Lagom/ReadMe.md ) to build SerenityOS userspace libraries for the host platform. Target
tests run on the SerenityOS machine, either emulated or bare metal.
2021-02-26 17:12:04 -05:00
2021-05-15 22:13:31 -04:00
## Running Host Tests
2021-02-26 17:12:04 -05:00
There are two ways to build host tests: from a full build, or from a Lagom-only build. The only difference is the CMake
2021-04-18 04:30:03 -04:00
command used to initialize the build directory.
2021-02-26 17:12:04 -05:00
For a full build, pass `-DBUILD_LAGOM=ON` to the CMake command.
```sh
2022-10-03 20:32:37 -04:00
cmake -GNinja -S Meta/CMake/Superbuild -B Build/superbuild-x86_64 -DBUILD_LAGOM=ON
2021-02-26 17:12:04 -05:00
```
2021-05-15 22:13:31 -04:00
For a Lagom-only build, pass the Lagom directory to CMake. The `BUILD_LAGOM` CMake option is still required.
2021-02-26 17:12:04 -05:00
```sh
2021-09-07 04:35:29 -04:00
cmake -GNinja -S Meta/Lagom -B Build/lagom -DBUILD_LAGOM=ON
2021-02-26 17:12:04 -05:00
```
2021-05-15 22:13:31 -04:00
In both cases, the tests can be run via ninja after doing a build. Note that `test-js` requires the `SERENITY_SOURCE_DIR` environment variable to be set
to the root of the serenity source tree when running on a non-SerenityOS host.
2021-02-26 17:12:04 -05:00
```sh
2021-05-15 22:13:31 -04:00
# /path/to/serenity repository
2021-09-07 04:35:29 -04:00
export SERENITY_SOURCE_DIR=${PWD}
cd Build/lagom
2022-05-08 12:26:10 -04:00
ninja
2021-09-07 04:35:29 -04:00
ninja test
2021-02-26 17:12:04 -05:00
```
2021-05-17 12:48:55 -04:00
To see the stdout/stderr output of failing tests, the recommended way is to set the environment variable [`CTEST_OUTPUT_ON_FAILURE` ](https://cmake.org/cmake/help/latest/manual/ctest.1.html#options ) to 1.
2021-02-26 17:12:04 -05:00
2021-05-15 22:13:31 -04:00
```sh
CTEST_OUTPUT_ON_FAILURE=1 ninja test
# or, using ctest directly...
ctest --output-on-failure
```
### Running with Sanitizers
CI runs host tests with Address Sanitizer and Undefined Sanitizer instrumentation enabled. These tools catch many
classes of common C++ errors, including memory leaks, out of bounds access to stack and heap allocations, and
signed integer overflow. For more info on the sanitizers, check out the Address Sanitizer [wiki page ](https://github.com/google/sanitizers/wiki ),
or the Undefined Sanitizer [documentation ](https://clang.llvm.org/docs/UndefinedBehaviorSanitizer.html ) from clang.
2022-01-06 09:07:15 -05:00
Note that a sanitizer build will take significantly longer than a non-sanitizer build, and will mess with caches in tools such as `ccache` .
2021-09-07 04:35:29 -04:00
The sanitizers can be enabled with the `-DENABLE_FOO_SANITIZER` set of flags. For the Serenity target, only the Undefined Sanitizers is supported.
2021-05-15 22:13:31 -04:00
```sh
2021-09-07 04:35:29 -04:00
cmake -GNinja -S Meta/Lagom -B Build/lagom -DBUILD_LAGOM=ON -DENABLE_ADDRESS_SANITIZER=ON -DENABLE_UNDEFINED_SANITIZER=ON
cd Build/lagom
2021-05-15 22:13:31 -04:00
ninja
2021-09-07 04:35:29 -04:00
CTEST_OUTPUT_ON_FAILURE=1 SERENITY_SOURCE_DIR=${PWD}/../.. ninja test
2021-05-15 22:13:31 -04:00
```
To ensure that Undefined Sanitizer errors fail the test, the `halt_on_error` flag should be set to 1 in the environment variable `UBSAN_OPTIONS` .
```sh
UBSAN_OPTIONS=halt_on_error=1 CTEST_OUTPUT_ON_FAILURE=1 SERENITY_SOURCE_DIR=${PWD}/.. ninja test
```
## Running Target Tests
Tests built for the SerenityOS target get installed either into `/usr/Tests` or `/bin` . `/usr/Tests` is preferred, but
2021-02-26 17:12:04 -05:00
some system tests are installed into `/bin` for historical reasons.
The easiest way to run all of the known tests in the system is to use the `run-tests-and-shutdown.sh` script that gets
2022-03-20 14:55:50 -04:00
installed into `/home/anon/Tests` . When running in CI, the environment variable `$DO_SHUTDOWN_AFTER_TESTS` is set, which
2021-02-26 17:12:04 -05:00
will run `shutdown -n` after running all the tests.
2021-05-15 22:13:31 -04:00
For completeness, a basic on-target test run will need the SerenityOS image built and run via QEMU.
2021-02-26 17:12:04 -05:00
```sh
2022-10-03 20:32:37 -04:00
cmake -GNinja -S Meta/CMake/Superbuild -B Build/superbuild-x86_64
cmake --build Build/superbuild-x86_64
cd Build/x86_64
2023-03-11 20:12:37 -05:00
ninja install & & ninja qemu-image & & ninja run
2021-02-26 17:12:04 -05:00
```
In the initial terminal, one can easily run the test runner script:
```
2022-03-20 14:55:50 -04:00
courage ~ $ ./Tests/run-tests-and-shutdown.sh
2021-02-26 17:12:04 -05:00
=== Running Tests on SerenityOS ===
...
```
CI runs the tests in self-test mode, using the 'ci' run options and the TestRunner entry in /etc/SystemServer.ini to run
tests automatically on startup.
The system server entry looks as below:
```ini
[TestRunner@ttyS0]
2022-03-20 14:55:50 -04:00
Executable=/home/anon/Tests/run-tests-and-shutdown.sh
2021-02-26 17:12:04 -05:00
StdIO=/dev/ttyS0
2021-08-07 08:05:33 -04:00
Environment=DO_SHUTDOWN_AFTER_TESTS=1 TERM=xterm PATH=/usr/local/bin:/usr/bin:/bin
2021-02-26 17:12:04 -05:00
User=anon
WorkingDirectory=/home/anon
2021-10-23 13:15:01 -04:00
SystemModes=self-test
2021-02-26 17:12:04 -05:00
```
`/dev/ttyS0` is used as stdio because that serial port is connected when qemu is run with `-display none` and
2023-10-24 07:11:04 -04:00
`-serial stdio` , and output to it will show up in the stdout of the qemu window. Separately, the CI run script redirects
2021-02-26 17:12:04 -05:00
the serial debug output to `./debug.log` so that both stdout of the tests and the dbgln from the kernel/tests can be
captured.
2021-05-15 22:13:31 -04:00
To run with CI's TestRunner system server entry, SerenityOS needs booted in self-test mode. Running the following shell
2021-10-23 11:31:00 -04:00
lines will boot SerenityOS in self-test mode, run tests, and exit. Note that CI also sets `panic=shutdown` to terminate qemu;
the default value `halt` keeps qemu around, which allows you to inspect the state.
2021-02-26 17:12:04 -05:00
```sh
export SERENITY_RUN=ci
2022-08-12 01:37:23 -04:00
export SERENITY_KERNEL_CMDLINE="graphics_subsystem_mode=off system_mode=self-test"
2021-02-26 17:12:04 -05:00
ninja run
```