Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

run tests inside SVSM #120

Merged
merged 10 commits into from
Oct 23, 2023

Conversation

Freax13
Copy link
Contributor

@Freax13 Freax13 commented Oct 8, 2023

This PR makes it possible to run tests inside the SVSM. Simply setting the TEST environment variable before running make will produce an SVSM build that runs tests instead of running the guest.

This is implemented using the nightly custom_test_frameworks feature to specify our own tests runner. This nightly feature is only required when building tests for running inside the SVSM, non-test builds and normal test execution can continue to use a stable toolchain. The #[test] macro expects a crate named test to exist, so a small, no_std compatible subset of the test crate has been copied.

Some tests currently fail when run inside the SVSM (AFAICT mostly for benign reasons related to setup of the allocator and file system). Those tests are currently disabled using the ignore attribute. Fixing those tests is probably best left to separate PRs.

Makefile Outdated
@@ -9,7 +9,17 @@ TARGET_PATH="debug"
endif

STAGE2_ELF = "target/x86_64-unknown-none/${TARGET_PATH}/stage2"
ifdef TEST
KERNEL_ELF = "target/x86_64-unknown-none/${TARGET_PATH}/deps/svsm-0651abaef6a489bb"
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure the metadata hash can be hardcoded here. I think it can change based on compiler version and other parameters.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you're right in that there is no guarantee that the hash will be stable. Cargo doesn't seem to have way to fix the path: rust-lang/cargo#1924). Anecdotally, over the course of development for this pr, I've been using multiple nightly versions and the hash remained the same.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I spotted in when I was trying it out - I was playing around with my own build script that deploys to our test machine and I ended up making the hash value change. Unfortunately I don't know what I did to cause it to change and it reverted back to the original when I tried to reproduce it.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed in ace2c55

Makefile Outdated Show resolved Hide resolved
@roy-hopkins
Copy link
Collaborator

We don't have a README specific to testing at the moment but there is reference to the unit tests in INSTALL.md. Could you maybe add some instructions on how to build and deploy the test binary in INSTALL.md?

@joergroedel
Copy link
Member

Thanks for doing this, it looks very promising! Somehow I have issues getting the test binary to build, it can't find the core crate. Any hint on how I can fix this on my side?

@Freax13
Copy link
Contributor Author

Freax13 commented Oct 12, 2023

Thanks for doing this, it looks very promising! Somehow I have issues getting the test binary to build, it can't find the core crate. Any hint on how I can fix this on my side?

Sounds like the x86_64-unknown-none target on the nightly toolchain isn't installed. Try

rustup target add --toolchain nightly x86_64-unknown-none

@joergroedel
Copy link
Member

rustup target add --toolchain nightly x86_64-unknown-none

That works, thanks. I can run the tests now.

Copy link
Member

@joergroedel joergroedel left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In general this looks great. A few things I'd like to see changed:

  1. We can safely assume that the test SVSM binary will run in QEMU, so I'd like to use the QEMU exit device to get exit qemu with a given exit-code. This way running the in-kernel tests can be more easily automated.
  2. Please add a Makefile target to build and run the in-kernel tests.

Is there an easy way to limit certain tests to a single test-runner? So that we run some tests only in user-space and others only with the in-kernel runner?

src/testing.rs Outdated

info!("All tests passed!");

request_termination_msr();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wonder if we can use the qemu exit device here?

Copy link
Contributor Author

@Freax13 Freax13 Oct 15, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Turns out we can: 0ef34e2

@Freax13 Freax13 force-pushed the feature/custom-test-framework branch 2 times, most recently from 3ea5805 to f3d1e5e Compare October 15, 2023 09:54
@Freax13
Copy link
Contributor Author

Freax13 commented Oct 15, 2023

In general this looks great. A few things I'd like to see changed:

  1. We can safely assume that the test SVSM binary will run in QEMU, so I'd like to use the QEMU exit device to get exit qemu with a given exit-code. This way running the in-kernel tests can be more easily automated.

Good idea! Implemented in 0ef34e2.

  1. Please add a Makefile target to build and run the in-kernel tests.

Implemented in 773b3b2. I added a test-in-svsm target. This targets determines the QEMU and OVMF installation paths from environment variables (and errors out if they are missing) to run the tests.

Is there an easy way to limit certain tests to a single test-runner? So that we run some tests only in user-space and others only with the in-kernel runner?

We can use cfg_attr(target_os = "none", ignore) to prevent tests from being run inside the SVSM and vice versa.

@joergroedel
Copy link
Member

We can use cfg_attr(target_os = "none", ignore) to prevent tests from being run inside the SVSM and vice versa.

A more descriptive way of specifying this would be great. Can this work with defining specific features and making tests depend on them?

@Freax13
Copy link
Contributor Author

Freax13 commented Oct 18, 2023

We can use cfg_attr(target_os = "none", ignore) to prevent tests from being run inside the SVSM and vice versa.

A more descriptive way of specifying this would be great. Can this work with defining specific features and making tests depend on them?

Yes, we could either use a feature or just a raw cfg(some_feature_name).

AdamCDunlap added a commit to AdamCDunlap/svsm that referenced this pull request Oct 20, 2023
Adds tests that make sure a #VC exception is raised while executing
various instructions and that the #VC handler properly handles them via
the ghcb.

Adds tests for cpuid, msrs, port IO, MMIO, wbinvd, rdtsc, dr7, and vmmcall.

Relies on being able to run tests inside the guest VM, e.g. with PR coconut-svsm#120.

The #VC handler is not yet implemented, so these tests are all marked
with #[should_panic]. See PR coconut-svsm#55 for progress on its implementation.

TODO:
Remove added dependencies
Move the tests to vc.rs or a new file
Clean up comments, etc.
Move assembly wrapper helper functions to more sensible places
Implement page state change (PSC) test
AdamCDunlap added a commit to AdamCDunlap/svsm that referenced this pull request Oct 20, 2023
Adds tests that make sure a #VC exception is raised while executing
various instructions and that the #VC handler properly handles them via
the ghcb.

Adds tests for cpuid, msrs, port IO, MMIO, wbinvd, rdtsc, dr7, and vmmcall.

Relies on being able to run tests inside the guest VM, e.g. with PR coconut-svsm#120.

The #VC handler is not yet implemented, so these tests are all marked
with #[should_panic]. See PR coconut-svsm#55 for progress on its implementation.

TODO:
Remove added dependencies
Move the tests to vc.rs or a new file
Clean up comments, etc.
Move assembly wrapper helper functions to more sensible places
Implement page state change (PSC) test

Signed-off-by: Adam Dunlap <[email protected]>
AdamCDunlap added a commit to AdamCDunlap/svsm that referenced this pull request Oct 20, 2023
Adds tests that make sure a #VC exception is raised while executing
various instructions and that the #VC handler properly handles them via
the ghcb.

Adds tests for cpuid, msrs, port IO, MMIO, wbinvd, rdtsc, dr7, and vmmcall.

Relies on being able to run tests inside the guest VM, e.g. with PR coconut-svsm#120.

The #VC handler is not yet implemented, so these tests are all marked
with #[should_panic]. See PR coconut-svsm#55 for progress on its implementation.

TODO:
Remove added dependencies
Move the tests to vc.rs or a new file
Clean up comments, etc.
Move assembly wrapper helper functions to more sensible places
Implement page state change (PSC) test

Signed-off-by: Adam Dunlap <[email protected]>
@Freax13 Freax13 force-pushed the feature/custom-test-framework branch from f3d1e5e to b043f5b Compare October 21, 2023 10:08
@Freax13
Copy link
Contributor Author

Freax13 commented Oct 21, 2023

We can use cfg_attr(target_os = "none", ignore) to prevent tests from being run inside the SVSM and vice versa.

A more descriptive way of specifying this would be great. Can this work with defining specific features and making tests depend on them?

Yes, we could either use a feature or just a raw cfg(some_feature_name).

I added a cfg option test_in_svsm. See d368d23 on how this is used.

@Freax13 Freax13 force-pushed the feature/custom-test-framework branch from b043f5b to bed5654 Compare October 21, 2023 10:11
@Freax13
Copy link
Contributor Author

Freax13 commented Oct 21, 2023

I just rebased onto the latest master branch.

@joergroedel
Copy link
Member

Made a few updates and stored them in this branch. Can you please look at my changes and include them into this PR?

@Freax13
Copy link
Contributor Author

Freax13 commented Oct 23, 2023

Made a few updates and stored them in this branch. Can you please look at my changes and include them into this PR?

Looks good to me, thanks!

Freax13 and others added 9 commits October 23, 2023 11:08
Once we'll start running tests in the SVSM, we'll want to use the
correct implementations and not the fake ones for tests outside the
SVSM.

Signed-off-by: Tom Dohrmann <[email protected]>
For one reason or another some of the tests currently fail when run
inside the SVSM.

Signed-off-by: Tom Dohrmann <[email protected]>
Signed-off-by: Tom Dohrmann <[email protected]>
... and move it to `all`. This way we can use the stage1/stage1.o target
with another kernel (e.g. test kernel).

Signed-off-by: Tom Dohrmann <[email protected]>
To build and run a kernel that runs the tests use
```
QEMU=/path/to/qemu OVMF=/path/to/firmware/ make test-in-svsm
```

Signed-off-by: Tom Dohrmann <[email protected]>
This is cleaner than `request_terminator_msr` as that MSR is more
commonly used to signal errors rather than clean exits.

Signed-off-by: Tom Dohrmann <[email protected]>
Add a helper to print the C-bit position on the current platform.

Signed-off-by: Joerg Roedel <[email protected]>
Instead of having the full QEMU command line in the Makefile, extract
the test-run into a separate script.

Signed-off-by: Joerg Roedel <[email protected]>
@Freax13 Freax13 force-pushed the feature/custom-test-framework branch from aba18a1 to 845ec96 Compare October 23, 2023 11:08
@Freax13
Copy link
Contributor Author

Freax13 commented Oct 23, 2023

I also rebased onto the latest master branch so resolve some conflicts.

Copy link
Member

@joergroedel joergroedel left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, looks good now.

@joergroedel joergroedel merged commit 6138015 into coconut-svsm:main Oct 23, 2023
2 checks passed
AdamCDunlap added a commit to AdamCDunlap/svsm that referenced this pull request Oct 27, 2023
Adds tests that make sure a #VC exception is raised while executing
various instructions and that the #VC handler properly handles them via
the ghcb.

Adds tests for cpuid, msrs, port IO, MMIO, wbinvd, rdtsc, dr7, and vmmcall.

Relies on being able to run tests inside the guest VM, e.g. with PR coconut-svsm#120.

The #VC handler is not yet implemented, so these tests are all marked
with #[should_panic]. See PR coconut-svsm#55 for progress on its implementation.

TODO:
Remove added dependencies
Move the tests to vc.rs or a new file
Clean up comments, etc.
Move assembly wrapper helper functions to more sensible places
Implement page state change (PSC) test

Signed-off-by: Adam Dunlap <[email protected]>
AdamCDunlap added a commit to AdamCDunlap/svsm that referenced this pull request Oct 27, 2023
Adds tests that make sure a #VC exception is raised while executing
various instructions and that the #VC handler properly handles them via
the ghcb.

Adds tests for cpuid, msrs, port IO, MMIO, wbinvd, rdtsc, dr7, and vmmcall.

Relies on being able to run tests inside the guest VM, e.g. with PR coconut-svsm#120.

The #VC handler is not yet implemented, so these tests are all marked
with #[should_panic]. See PR coconut-svsm#55 for progress on its implementation.

TODO:
Remove added dependencies
Move the tests to vc.rs or a new file
Clean up comments, etc.
Move assembly wrapper helper functions to more sensible places
Implement page state change (PSC) test

Signed-off-by: Adam Dunlap <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants