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

Generate a C API for IGVM #3

Merged
merged 12 commits into from
Feb 1, 2024
Merged

Generate a C API for IGVM #3

merged 12 commits into from
Feb 1, 2024

Conversation

roy-hopkins
Copy link
Contributor

This is a proposal on how to implement #1 and provide a C API for igvm.

The approach taken is to based on using cbindgen to analyze the rust code and produce header files usable from C. To support this, a set of C friendly functions have been added to igvm/src/c_api.rs which allows, at present, binary files to be parsed and processed.

The additional functions are hidden behind a rust feature named igvm-c. When performing a normal build (without the C API) the only change that has any effect is from the first commit 4da1043 which by necessity relaxes the unsafe_code restriction. Unfortunately this cannot be controlled by the feature.

The sample in igvm_c/sample and the unit test code in igvm/tests are probably the best places to start to see how to use the proposed API from C.

@roy-hopkins
Copy link
Contributor Author

roy-hopkins commented Oct 3, 2023 via email

@chris-oo
Copy link
Member

chris-oo commented Oct 4, 2023

Thanks for the RFC! I'll try to find some time to look at this soon.

Copy link
Member

@chris-oo chris-oo left a comment

Choose a reason for hiding this comment

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

Broad strokes look fine, but I think we can add some rust methods to remove the need to allocate Vecs that we immediately throw away for a lot of these methods.

@roy-hopkins
Copy link
Contributor Author

I've addressed the current review comments with the latest push. The most significant change was in the management of the buffers that are returned containing the headers and file data. Previously, the rust code was populating a Vec then copying the result into a C provided, pre-allocated buffer (or returning the size and BUFFER_TOO_SMALL if the buffer was not big enough.

I've now changed this so the Vec created by the rust functions are stored alongside a handle that is created when the file is initially parsed. This means that the rust code owns the buffer and knows the actual size, allowing a lot of unsafe code to be removed. Obviously to support this I have had to add functions to manage the lifetime of these handles and buffers but I think the result is safe and cleaner than the previous version.

@chris-oo
Copy link
Member

I left another round of feedback but I agree that it's much better. I think refactoring to take advantage of Result<(), IgvmApiError> and OnceLock will help clean it up a bit more and remove all the unsafe code blocks. Thanks for doing this!

@roy-hopkins
Copy link
Contributor Author

I've worked through the PR comments and suggestions and uploaded a new version. In addition I've added one more commit to package and install the library.

I've been using the version built by this branch for a few weeks now working on integrating an IGVM loader into QEMU so I am happy that the C library is working well. Therefore I consider this ready for removing the RFC/draft status.

@roy-hopkins roy-hopkins changed the title RFC: Generate a C API for IGVM Generate a C API for IGVM Dec 19, 2023
Copy link

@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.

Just a minor coding style issue, otherwise this looks good to me.

@chris-oo
Copy link
Member

This is on my radar, I'll take a look at it tomorrow. Seems like I need to fix my github email settings because I'm not getting the emails I expect.

Copy link
Member

@chris-oo chris-oo left a comment

Choose a reason for hiding this comment

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

Thanks for cleaning up the changes!

A few comments left, mainly mechanical, but I'm happy to sign off and spin a new release once that's done. Thanks again for contributing these changes!

@@ -228,8 +229,6 @@ pub enum IgvmArchitecture {
#[derive(AsBytes, FromBytes, FromZeroes, Debug, Clone, Copy, PartialEq, Eq)]
#[repr(u32)]
pub enum IgvmVariableHeaderType {
/// Invalid.
Copy link
Member

Choose a reason for hiding this comment

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

What's the reason for removing this?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This was removed as the generated C header file ends up with multiple definitions of INVALID that conflict. The alternatives were to either:

  • Rename INVALID to something like IGVM_VHT_INVALID here.
  • Modify the cbindgen configuration to add a prefix to every enum entry. But this would result in C definitions such as IGVM_VARIABLE_HEADER_TYPE_IGVM_VHT_SUPPORTED_PLATFORM.
  • Perform some post-processing of the generated header file.

I'm happy to change this to whichever option works best. Although, why do we need to allow a header to contain INVALID when we have error handling in the library?

Copy link
Member

Choose a reason for hiding this comment

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

My rationale is that even if it looks fairly obvious that 0 is not a free value, it's nice for us to document in the format exactly that "hey this value is invalid and don't try to use it".

I'm fine with renaming it to IGVM_VHT_INVALID if that satisfies the C bindgen req. A bit unfortunate, though.

Copy link
Member

Choose a reason for hiding this comment

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

I'm happy to sign off once this last little bit is in, thanks again for your changes!

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've managed to use some C preprocessor wizardry in cbindgen_igvm_defs.toml to remove the duplicate INVALID definition on the header file so I've removed the commit that deleted this INVALID enum entry, restoring it to how it was before.

Copy link
Member

Choose a reason for hiding this comment

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

Great thanks!

Copy link

@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.

Looks good to me now.

roy-hopkins and others added 5 commits February 1, 2024 10:38
Currently unsafe_code is forbidden for igvm code. In order to add
a C compatible API we will need to accept and work with raw C
pointers to data. This requires a relaxing of the unsafe rule from
'forbid' to 'deny' allowing the C API feature, when enabled,  to
override the setting.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
This adds the docsrs configuration which is already in igvm_defs to the
igvm crate and allows any future optional features to be correctly
labelled in the generated documentation.

Signed-off-by: Roy Hopkins <roy.hopkins@suse.com>
In preparation for adding the code to provide a C API this patch
adds a cargo feature to allow the C API to be conditionally built.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
The C API will need to convert variable headers from the rust
representation to C compatible structures. As part of this the
C API will need to determine the original IgvmVariableHeaderType
from the rust representation.

This patch adds the required functions behind the igvm-c feature
as they are not used outside of that feature.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
The C API will use write_binary_header() for converting from
the rust representation of headers back to raw C structures. The
write_binary_header() function was not previously implemented
for IgvmPlatformHeader so this patch adds the function behind
the igvm-c feature.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
roy-hopkins and others added 7 commits February 1, 2024 10:38
This adds the c_api.rs file which implements functions that allow
parsing and querying of igvm files in binary format exported in a
suitable way for access from the C language.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
This ensures both the existing dynamic library and a static library
that can be used to link into C applications is generated by the build.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
In preparation for using cbindgen to generate C header files
from the rust structure definitions, add an annotation that
prevents incompatible and unsupported types from being
included in the output.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
Add a new folder named igvm_c that implements the following:

* Generates C header files for igvm and igvm_defs that can be used to
   access some functionality from C using the existing static library.
* Sample application that demonstrates using the C API to read and
   dump the contents of a binary igvm file.
* Unit test executable for testing basic C API functionality.
* Makefile that builds all C API components into the target_c
   output directory.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
Add unit tests that cover initialisation and destruction of the IGVM
file using the C API and the querying of header and file data within the
parsed file.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
Add a new git workflow job that builds the C API and runs the C unit
tests via the Makefile.

Signed-off-by: Roy Hopkins <rhopkins@suse.de>
Adds an install task to the C Makefile that copies the header files into
/usr/include/igvm/ and the library into /usr/lib64/ and creates a
pkg-config entry for the library.

Signed-off-by: Roy Hopkins <roy.hopkins@suse.com>
@roy-hopkins
Copy link
Contributor Author

Rebased on microsoft:main. No other changes.

@chris-oo chris-oo enabled auto-merge (squash) February 1, 2024 17:59
@chris-oo chris-oo merged commit 4c6ee46 into microsoft:main Feb 1, 2024
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