Skip to content

Commit

Permalink
Generate traits and impls for all validstructs on command parameters
Browse files Browse the repository at this point in the history
  • Loading branch information
MarijnS95 committed May 6, 2023
1 parent 77bed8c commit 6e347d7
Show file tree
Hide file tree
Showing 4 changed files with 73 additions and 17 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ jobs:
- name: Run generator
run: cargo run -p generator
- name: Diff autogen result
run: git diff --quiet || (echo "::error::Generated files are different, please regenerate with cargo run -p generator!"; git diff; false)
run: test -z "$(git status --porcelain)" || (echo "::error::Generated files are different, please regenerate with cargo run -p generator!"; git diff; false)

test:
name: Test Suite
Expand Down
12 changes: 7 additions & 5 deletions ash/src/extensions/ext/pipeline_properties.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,16 +21,18 @@ impl PipelineProperties {
}

/// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkGetPipelinePropertiesEXT.html>
///
/// Currently only accepts [`vk::PipelinePropertiesIdentifierEXT`].
#[inline]
pub unsafe fn get_pipeline_properties(
&self,
pipeline_info: &vk::PipelineInfoEXT,
pipeline_properties: *mut vk::BaseOutStructure,
pipeline_properties: &mut impl vk::GetPipelinePropertiesEXTParamPipelineProperties,
) -> VkResult<()> {
(self.fp.get_pipeline_properties_ext)(self.handle, pipeline_info, pipeline_properties)
.result()
(self.fp.get_pipeline_properties_ext)(
self.handle,
pipeline_info,
<*mut _>::cast(pipeline_properties),
)
.result()
}

pub const NAME: &'static CStr = vk::ExtPipelinePropertiesFn::NAME;
Expand Down
7 changes: 7 additions & 0 deletions ash/src/vk/extensions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17672,6 +17672,13 @@ impl ExtPipelinePropertiesFn {
pub const SPEC_VERSION: u32 = 1u32;
}
#[allow(non_camel_case_types)]
#[doc = "Implemented for all types that can be passed as argument to `pipeline_properties` in [`PFN_vkGetPipelinePropertiesEXT`]"]
pub unsafe trait GetPipelinePropertiesEXTParamPipelineProperties {}
unsafe impl GetPipelinePropertiesEXTParamPipelineProperties
for PipelinePropertiesIdentifierEXT<'_>
{
}
#[allow(non_camel_case_types)]
pub type PFN_vkGetPipelinePropertiesEXT = unsafe extern "system" fn(
device: Device,
p_pipeline_info: *const PipelineInfoEXT,
Expand Down
69 changes: 58 additions & 11 deletions generator/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -906,50 +906,59 @@ fn generate_function_pointers<'a>(
struct Command<'a> {
type_needs_defining: bool,
type_name: Ident,
pfn_type_name: Ident,
function_name_c: &'a str,
function_name_rust: Ident,
parameters: TokenStream,
parameters_unused: TokenStream,
returns: TokenStream,
parameter_validstructs: Vec<(Ident, Vec<String>)>,
}

let commands = commands
.iter()
.map(|cmd| {
let name = &cmd.proto.name;
let type_name = format_ident!("PFN_{}", name);
let pfn_type_name = format_ident!("PFN_{}", name);

// We might need to generate a function pointer for an extension, where we are given the original
// `cmd` and a rename back to the extension alias (typically with vendor suffix) in `rename_commands`:
let function_name_c = rename_commands.get(name.as_str()).cloned().unwrap_or(name);

let function_name_rust = format_ident!(
"{}",
function_name_c.strip_prefix("vk").unwrap().to_snake_case()
);
let type_name = function_name_c.strip_prefix("vk").unwrap();
let function_name_rust = format_ident!("{}", type_name.to_snake_case());
let type_name = format_ident!("{}", type_name);

let params: Vec<_> = cmd
let params = cmd
.params
.iter()
.filter(|param| matches!(param.api.as_deref(), None | Some(DESIRED_API)))
.filter(|param| matches!(param.api.as_deref(), None | Some(DESIRED_API)));

let params_tokens: Vec<_> = params
.clone()
.map(|param| {
let name = param.param_ident();
let ty = param.type_tokens(true);
(name, ty)
})
.collect();

let params_iter = params
let params_iter = params_tokens
.iter()
.map(|(param_name, param_ty)| quote!(#param_name: #param_ty));
let parameters = quote!(#(#params_iter,)*);

let params_iter = params.iter().map(|(param_name, param_ty)| {
let params_iter = params_tokens.iter().map(|(param_name, param_ty)| {
let unused_name = format_ident!("_{}", param_name);
quote!(#unused_name: #param_ty)
});
let parameters_unused = quote!(#(#params_iter,)*);

let parameter_validstructs: Vec<_> = params
.filter(|param| !param.validstructs.is_empty())
.map(|param| (param.param_ident(), param.validstructs.clone()))
.collect();

let ret = cmd
.proto
.type_name
Expand All @@ -961,6 +970,7 @@ fn generate_function_pointers<'a>(
// This can happen because there are aliases to commands
type_needs_defining: fn_cache.insert(name),
type_name,
pfn_type_name,
function_name_c,
function_name_rust,
parameters,
Expand All @@ -971,14 +981,49 @@ fn generate_function_pointers<'a>(
let ret_ty_tokens = name_to_tokens(ret);
quote!(-> #ret_ty_tokens)
},
parameter_validstructs,
}
})
.collect::<Vec<_>>();

struct CommandToParamTraits<'a>(&'a Command<'a>);
impl<'a> quote::ToTokens for CommandToParamTraits<'a> {
fn to_tokens(&self, tokens: &mut TokenStream) {
for (param_ident, validstructs) in &self.0.parameter_validstructs {
let param_ident = param_ident.to_string();
let param_ident = param_ident
.strip_prefix("pp_")
.or_else(|| param_ident.strip_prefix("p_"))
.unwrap_or(&param_ident);

let doc_string = format!(
"Implemented for all types that can be passed as argument to `{}` in [`{}`]",
param_ident, self.0.pfn_type_name
);
let param_trait_name = format_ident!(
"{}Param{}",
self.0.type_name,
param_ident.to_upper_camel_case()
);
quote! {
#[allow(non_camel_case_types)]
#[doc = #doc_string]
pub unsafe trait #param_trait_name {}
}
.to_tokens(tokens);

for validstruct in validstructs {
let structname = name_to_tokens(validstruct);
quote!(unsafe impl #param_trait_name for #structname<'_> {}).to_tokens(tokens);
}
}
}
}

struct CommandToType<'a>(&'a Command<'a>);
impl<'a> quote::ToTokens for CommandToType<'a> {
fn to_tokens(&self, tokens: &mut TokenStream) {
let type_name = &self.0.type_name;
let type_name = &self.0.pfn_type_name;
let parameters = &self.0.parameters;
let returns = &self.0.returns;
quote!(
Expand All @@ -992,7 +1037,7 @@ fn generate_function_pointers<'a>(
struct CommandToMember<'a>(&'a Command<'a>);
impl<'a> quote::ToTokens for CommandToMember<'a> {
fn to_tokens(&self, tokens: &mut TokenStream) {
let type_name = &self.0.type_name;
let type_name = &self.0.pfn_type_name;
let type_name = if self.0.type_needs_defining {
// Type is defined in local scope
quote!(#type_name)
Expand Down Expand Up @@ -1033,6 +1078,7 @@ fn generate_function_pointers<'a>(
}
}

let param_traits = commands.iter().map(CommandToParamTraits);
let pfn_typedefs = commands
.iter()
.filter(|pfn| pfn.type_needs_defining)
Expand All @@ -1041,6 +1087,7 @@ fn generate_function_pointers<'a>(
let loaders = commands.iter().map(CommandToLoader);

quote! {
#(#param_traits)*
#(#pfn_typedefs)*

#[derive(Clone)]
Expand Down

0 comments on commit 6e347d7

Please sign in to comment.