Transform Bitflags and Enums to constants

This commit is contained in:
Maik Klein 2018-07-09 08:49:28 +02:00
parent fb13cc402e
commit 72f1cdf6ad

View file

@ -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);
} }