Transform Bitflags and Enums to constants
This commit is contained in:
parent
fb13cc402e
commit
72f1cdf6ad
1 changed files with 75 additions and 67 deletions
|
@ -20,6 +20,10 @@ use quote::Tokens;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use syn::Ident;
|
use syn::Ident;
|
||||||
|
|
||||||
|
pub struct Foo;
|
||||||
|
impl Foo {
|
||||||
|
const FOO: u32 = 1;
|
||||||
|
}
|
||||||
#[derive(Copy, Clone, Debug)]
|
#[derive(Copy, Clone, Debug)]
|
||||||
pub enum CType {
|
pub enum CType {
|
||||||
USize,
|
USize,
|
||||||
|
@ -734,8 +738,12 @@ pub fn generate_extension(extension: &vkxml::Extension, commands: &CommandMap) -
|
||||||
vkxml::ExtensionSpecificationElement::CommandReference(ref cmd_ref) => {
|
vkxml::ExtensionSpecificationElement::CommandReference(ref cmd_ref) => {
|
||||||
Some(cmd_ref)
|
Some(cmd_ref)
|
||||||
}
|
}
|
||||||
vkxml::ExtensionSpecificationElement::DefinitionReference(ref field) => {
|
vkxml::ExtensionSpecificationElement::Constant(ref field) => {
|
||||||
println!("EXT {:?}", field);
|
//println!("EXT {:?}", field);
|
||||||
|
None
|
||||||
|
}
|
||||||
|
vkxml::ExtensionSpecificationElement::Enum(ref field) => {
|
||||||
|
//println!("Enum {:?}", field);
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
_ => None,
|
_ => None,
|
||||||
|
@ -778,6 +786,7 @@ pub fn generate_bitmask(bitmask: &vkxml::Bitmask) -> Option<Tokens> {
|
||||||
vk_bitflags_wrapped!(#ident, 0b0, Flags);
|
vk_bitflags_wrapped!(#ident, 0b0, Flags);
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn to_variant_ident(enum_name: &str, variant_name: &str) -> Ident {
|
pub fn to_variant_ident(enum_name: &str, variant_name: &str) -> Ident {
|
||||||
let tag = ["AMD", "NN", "KHR", "NV", "EXT", "NVX", "KHX"]
|
let tag = ["AMD", "NN", "KHR", "NV", "EXT", "NVX", "KHX"]
|
||||||
.iter()
|
.iter()
|
||||||
|
@ -794,7 +803,7 @@ pub fn to_variant_ident(enum_name: &str, variant_name: &str) -> Ident {
|
||||||
.map(|t| enum_name.replace(t, ""))
|
.map(|t| enum_name.replace(t, ""))
|
||||||
.unwrap_or(enum_name.into());
|
.unwrap_or(enum_name.into());
|
||||||
let variant_without_tag = tag
|
let variant_without_tag = tag
|
||||||
.map(|t| variant_name.replace(t, ""))
|
.map(|t| variant_name.replace(&format!("_{}", t), ""))
|
||||||
.unwrap_or(variant_name.into());
|
.unwrap_or(variant_name.into());
|
||||||
let camel_case_name_enum = &name_without_tag.to_camel_case();
|
let camel_case_name_enum = &name_without_tag.to_camel_case();
|
||||||
let name = variant_without_tag.to_camel_case()[2..].replace(camel_case_name_enum, "");
|
let name = variant_without_tag.to_camel_case()[2..].replace(camel_case_name_enum, "");
|
||||||
|
@ -817,6 +826,35 @@ pub fn generate_enum(
|
||||||
) -> EnumType {
|
) -> EnumType {
|
||||||
let name = &_enum.name[2..];
|
let name = &_enum.name[2..];
|
||||||
let _name = name.replace("FlagBits", "Flags");
|
let _name = name.replace("FlagBits", "Flags");
|
||||||
|
let ident = Ident::from(_name.as_str());
|
||||||
|
let variants = _enum.elements.iter().filter_map(|elem| {
|
||||||
|
let (variant_name, value) = match *elem {
|
||||||
|
vkxml::EnumerationElement::Enum(ref constant) => {
|
||||||
|
let c = Constant::from_constant(constant);
|
||||||
|
(constant.name.as_str(), c.to_tokens())
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let _name = _name.split("Flags").nth(0).expect("split");
|
||||||
|
let struct_name = _name.to_shouty_snake_case();
|
||||||
|
println!("{}", struct_name);
|
||||||
|
let new_variant_name = variant_name.replace(&struct_name, "").replace("VK", "");
|
||||||
|
let new_variant_name = new_variant_name.trim_matches('_');
|
||||||
|
let is_digit = new_variant_name
|
||||||
|
.chars()
|
||||||
|
.nth(0)
|
||||||
|
.map(|c| c.is_digit(10))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let variant_ident = if is_digit {
|
||||||
|
Ident::from(format!("TYPE_{}", new_variant_name).as_str())
|
||||||
|
} else {
|
||||||
|
Ident::from(new_variant_name)
|
||||||
|
};
|
||||||
|
Some((variant_ident, value))
|
||||||
|
}).collect_vec();
|
||||||
if name.contains("Bit") {
|
if name.contains("Bit") {
|
||||||
let ident = Ident::from(_name.as_str());
|
let ident = Ident::from(_name.as_str());
|
||||||
let all_bits = _enum
|
let all_bits = _enum
|
||||||
|
@ -832,58 +870,36 @@ pub fn generate_enum(
|
||||||
.fold(0, |acc, next| acc | next.bits());
|
.fold(0, |acc, next| acc | next.bits());
|
||||||
let all_bits_term = Term::intern(&format!("0b{:b}", all_bits));
|
let all_bits_term = Term::intern(&format!("0b{:b}", all_bits));
|
||||||
|
|
||||||
let variants = _enum.elements.iter().filter_map(|elem| {
|
let variants = variants.iter().map(|(variant_ident, value)|{
|
||||||
let (variant_name, value) = match *elem {
|
quote!{
|
||||||
vkxml::EnumerationElement::Enum(ref constant) => {
|
pub const #variant_ident: Self = #ident { flags: #value }
|
||||||
let variant_name = &constant.name[3..];
|
|
||||||
let c = Constant::from_constant(constant);
|
|
||||||
if c.value().map(|v| v.bits() == 0).unwrap_or(false) {
|
|
||||||
return None;
|
|
||||||
}
|
}
|
||||||
(variant_name, c.to_tokens())
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
let variant_ident = Ident::from(variant_name);
|
|
||||||
Some(quote!{
|
|
||||||
pub const #variant_ident: #ident = #ident { flags: #value };
|
|
||||||
})
|
|
||||||
});
|
});
|
||||||
let q = quote!{
|
let q = quote!{
|
||||||
#(#variants)*
|
|
||||||
vk_bitflags_wrapped!(#ident, #all_bits_term, Flags);
|
vk_bitflags_wrapped!(#ident, #all_bits_term, Flags);
|
||||||
|
impl #ident {
|
||||||
|
#(#variants;)*
|
||||||
|
}
|
||||||
};
|
};
|
||||||
EnumType::Bitflags(q)
|
EnumType::Bitflags(q)
|
||||||
} else {
|
} else {
|
||||||
let q = match _name.as_str() {
|
let variants = variants.iter().map(|(variant_ident, value)|{
|
||||||
"StructureType" => generate_structure_type(&_name, _enum, create_info_constants),
|
quote!{
|
||||||
"Result" => generate_result(&_name, _enum),
|
pub const #variant_ident: Self = #ident(#value)
|
||||||
_ => {
|
|
||||||
let ident = Ident::from(_name.as_str());
|
|
||||||
let variants = _enum.elements.iter().filter_map(|elem| {
|
|
||||||
let (variant_name, value) = match *elem {
|
|
||||||
vkxml::EnumerationElement::Enum(ref constant) => {
|
|
||||||
let c = Constant::from_constant(constant);
|
|
||||||
//println!("value {:?}", c.value());
|
|
||||||
(constant.name.as_str(), c.to_tokens())
|
|
||||||
}
|
}
|
||||||
_ => {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let variant_ident = to_variant_ident(&_name, variant_name);
|
|
||||||
Some(quote!{
|
|
||||||
#variant_ident = #value
|
|
||||||
})
|
|
||||||
});
|
});
|
||||||
|
let q = match _name.as_str() {
|
||||||
|
//"StructureType" => generate_structure_type(&_name, _enum, create_info_constants),
|
||||||
|
//"Result" => generate_result(&_name, _enum),
|
||||||
|
_ => {
|
||||||
quote!{
|
quote!{
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
#[repr(C)]
|
#[repr(C)]
|
||||||
pub enum #ident {
|
pub struct #ident(pub i32);
|
||||||
#(#variants,)*
|
impl #ident {
|
||||||
|
#(
|
||||||
|
#variants;
|
||||||
|
)*
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -891,6 +907,7 @@ pub fn generate_enum(
|
||||||
EnumType::Enum(q)
|
EnumType::Enum(q)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn generate_structure_type(
|
pub fn generate_structure_type(
|
||||||
name: &str,
|
name: &str,
|
||||||
_enum: &vkxml::Enumeration,
|
_enum: &vkxml::Enumeration,
|
||||||
|
@ -1226,25 +1243,16 @@ pub fn write_source_code(spec: &vkxml::Registry) {
|
||||||
.flat_map(|constants| constants.elements.iter())
|
.flat_map(|constants| constants.elements.iter())
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let (create_info_constants, constants): (Vec<_>, Vec<_>) =
|
let (enum_code, bitflags_code) = enums.into_iter().map(|e| generate_enum(e, &[])).fold(
|
||||||
constants.iter().partition_map(|&c| {
|
(Vec::new(), Vec::new()),
|
||||||
if c.name.contains("CreateInfo") {
|
|mut acc, elem| {
|
||||||
itertools::Either::Left(c)
|
|
||||||
} else {
|
|
||||||
itertools::Either::Right(c)
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let (enum_code, bitflags_code) = enums
|
|
||||||
.into_iter()
|
|
||||||
.map(|e| generate_enum(e, &create_info_constants))
|
|
||||||
.fold((Vec::new(), Vec::new()), |mut acc, elem| {
|
|
||||||
match elem {
|
match elem {
|
||||||
EnumType::Enum(token) => acc.0.push(token),
|
EnumType::Enum(token) => acc.0.push(token),
|
||||||
EnumType::Bitflags(token) => acc.1.push(token),
|
EnumType::Bitflags(token) => acc.1.push(token),
|
||||||
};
|
};
|
||||||
acc
|
acc
|
||||||
});
|
},
|
||||||
|
);
|
||||||
let constants_code: Vec<_> = constants
|
let constants_code: Vec<_> = constants
|
||||||
.iter()
|
.iter()
|
||||||
.map(|constant| generate_constant(constant))
|
.map(|constant| generate_constant(constant))
|
||||||
|
@ -1264,7 +1272,7 @@ pub fn write_source_code(spec: &vkxml::Registry) {
|
||||||
.iter()
|
.iter()
|
||||||
.map(|ext| generate_extension(ext, &commands))
|
.map(|ext| generate_extension(ext, &commands))
|
||||||
.collect();
|
.collect();
|
||||||
let mut file = File::create("../ash/src/vk_test.rs").expect("vk");
|
let mut file = File::create("../ash/src/vk.rs").expect("vk");
|
||||||
let bitflags_macro = vk_bitflags_wrapped_macro();
|
let bitflags_macro = vk_bitflags_wrapped_macro();
|
||||||
let handle_nondispatchable_macro = handle_nondispatchable_macro();
|
let handle_nondispatchable_macro = handle_nondispatchable_macro();
|
||||||
let define_handle_macro = define_handle_macro();
|
let define_handle_macro = define_handle_macro();
|
||||||
|
@ -1272,17 +1280,17 @@ pub fn write_source_code(spec: &vkxml::Registry) {
|
||||||
let platform_specific_types = platform_specific_types();
|
let platform_specific_types = platform_specific_types();
|
||||||
let source_code = quote!{
|
let source_code = quote!{
|
||||||
pub use libc::*;
|
pub use libc::*;
|
||||||
#version_macros
|
// #version_macros
|
||||||
#platform_specific_types
|
// #platform_specific_types
|
||||||
#bitflags_macro
|
#bitflags_macro
|
||||||
#handle_nondispatchable_macro
|
// #handle_nondispatchable_macro
|
||||||
#define_handle_macro
|
// #define_handle_macro
|
||||||
#(#feature_code)*
|
// #(#feature_code)*
|
||||||
#(#definition_code)*
|
// #(#definition_code)*
|
||||||
#(#enum_code)*
|
#(#enum_code)*
|
||||||
#(#bitflags_code)*
|
#(#bitflags_code)*
|
||||||
#(#constants_code)*
|
// #(#constants_code)*
|
||||||
#(#extension_code)*
|
// #(#extension_code)*
|
||||||
};
|
};
|
||||||
write!(&mut file, "{}", source_code);
|
write!(&mut file, "{}", source_code);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue