#![recursion_limit = "256"] use heck::{CamelCase, ShoutySnakeCase, SnakeCase}; use itertools::Itertools; use nom::{ alt, character::complete::digit1, delimited, do_parse, map, named, one_of, opt, pair, preceded, tag, terminated, }; use once_cell::sync::Lazy; use proc_macro2::{Delimiter, Group, Literal, Span, TokenStream, TokenTree}; use quote::*; use regex::Regex; use std::collections::{BTreeMap, HashMap, HashSet}; use std::fmt::Display; use std::hash::BuildHasher; use std::path::Path; use syn::Ident; pub trait ExtensionExt {} #[derive(Copy, Clone, Debug)] pub enum CType { USize, U32, U64, Float, Bool32, } impl quote::ToTokens for CType { fn to_tokens(&self, tokens: &mut TokenStream) { let term = match self { CType::USize => format_ident!("usize"), CType::U32 => format_ident!("u32"), CType::U64 => format_ident!("u64"), CType::Float => format_ident!("f32"), CType::Bool32 => format_ident!("Bool32"), }; term.to_tokens(tokens); } } named!(ctype<&str, CType>, alt!( tag!("ULL") => { |_| CType::U64 } | tag!("U") => { |_| CType::U32 } ) ); named!(cexpr<&str, (CType, String)>, alt!( map!(cfloat, |f| (CType::Float, format!("{:.2}", f))) | inverse_number ) ); named!(decimal_number<&str, (CType, String)>, do_parse!( num: digit1 >> typ: ctype >> ((typ, num.to_string())) ) ); named!(inverse_number<&str, (CType, String)>, map!( delimited!( tag!("("), pair!( preceded!(tag!("~"), decimal_number), opt!(preceded!(tag!("-"), digit1)) ), tag!(")") ), |((ctyp, num), minus_num)| { let expr = if let Some(minus) = minus_num { format!("!{}-{}", num, minus) } else{ format!("!{}", num) }; (ctyp, expr) } ) ); named!(cfloat<&str, f32>, terminated!(nom::number::complete::float, one_of!("fF")) ); fn khronos_link(name: &S) -> Literal { Literal::string(&format!( "", name = name )) } pub fn define_handle_macro() -> TokenStream { quote! { #[macro_export] macro_rules! define_handle{ ($name: ident, $ty: ident) => { define_handle!($name, $ty, doc = ""); }; ($name: ident, $ty: ident, $doc_link: meta) => { #[repr(transparent)] #[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Hash)] #[$doc_link] pub struct $name(*mut u8); impl Default for $name { fn default() -> $name { $name::null() } } impl Handle for $name { const TYPE: ObjectType = ObjectType::$ty; fn as_raw(self) -> u64 { self.0 as u64 } fn from_raw(x: u64) -> Self { $name(x as _) } } unsafe impl Send for $name {} unsafe impl Sync for $name {} impl $name{ pub const fn null() -> Self{ $name(::std::ptr::null_mut()) } } impl fmt::Pointer for $name { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Pointer::fmt(&self.0, f) } } impl fmt::Debug for $name { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Debug::fmt(&self.0, f) } } } } } } pub fn handle_nondispatchable_macro() -> TokenStream { quote! { #[macro_export] macro_rules! handle_nondispatchable { ($name: ident, $ty: ident) => { handle_nondispatchable!($name, $ty, doc = ""); }; ($name: ident, $ty: ident, $doc_link: meta) => { #[repr(transparent)] #[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Hash, Default)] #[$doc_link] pub struct $name(u64); impl Handle for $name { const TYPE: ObjectType = ObjectType::$ty; fn as_raw(self) -> u64 { self.0 as u64 } fn from_raw(x: u64) -> Self { $name(x as _) } } impl $name{ pub const fn null() -> $name{ $name(0) } } impl fmt::Pointer for $name { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "0x{:x}", self.0) } } impl fmt::Debug for $name { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "0x{:x}", self.0) } } } } } } pub fn vk_version_macros() -> TokenStream { quote! { #[doc = ""] pub const fn make_version(major: u32, minor: u32, patch: u32) -> u32 { (major << 22) | (minor << 12) | patch } #[doc = ""] pub const fn version_major(version: u32) -> u32 { version >> 22 } #[doc = ""] pub const fn version_minor(version: u32) -> u32 { (version >> 12) & 0x3ff } #[doc = ""] pub const fn version_patch(version: u32) -> u32 { version & 0xfff } } } pub fn vk_bitflags_wrapped_macro() -> TokenStream { quote! { #[macro_export] macro_rules! vk_bitflags_wrapped { ($name: ident, $all: expr, $flag_type: ty) => { impl Default for $name{ fn default() -> $name { $name(0) } } impl $name { #[inline] pub const fn empty() -> $name { $name(0) } #[inline] pub const fn all() -> $name { $name($all) } #[inline] pub const fn from_raw(x: $flag_type) -> Self { $name(x) } #[inline] pub const fn as_raw(self) -> $flag_type { self.0 } #[inline] pub fn is_empty(self) -> bool { self == $name::empty() } #[inline] pub fn is_all(self) -> bool { self & $name::all() == $name::all() } #[inline] pub fn intersects(self, other: $name) -> bool { self & other != $name::empty() } /// Returns whether `other` is a subset of `self` #[inline] pub fn contains(self, other: $name) -> bool { self & other == other } } impl ::std::ops::BitOr for $name { type Output = $name; #[inline] fn bitor(self, rhs: $name) -> $name { $name (self.0 | rhs.0 ) } } impl ::std::ops::BitOrAssign for $name { #[inline] fn bitor_assign(&mut self, rhs: $name) { *self = *self | rhs } } impl ::std::ops::BitAnd for $name { type Output = $name; #[inline] fn bitand(self, rhs: $name) -> $name { $name (self.0 & rhs.0) } } impl ::std::ops::BitAndAssign for $name { #[inline] fn bitand_assign(&mut self, rhs: $name) { *self = *self & rhs } } impl ::std::ops::BitXor for $name { type Output = $name; #[inline] fn bitxor(self, rhs: $name) -> $name { $name (self.0 ^ rhs.0 ) } } impl ::std::ops::BitXorAssign for $name { #[inline] fn bitxor_assign(&mut self, rhs: $name) { *self = *self ^ rhs } } impl ::std::ops::Sub for $name { type Output = $name; #[inline] fn sub(self, rhs: $name) -> $name { self & !rhs } } impl ::std::ops::SubAssign for $name { #[inline] fn sub_assign(&mut self, rhs: $name) { *self = *self - rhs } } impl ::std::ops::Not for $name { type Output = $name; #[inline] fn not(self) -> $name { self ^ $name::all() } } } } } } fn is_opaque_type(ty: &str) -> bool { matches!( ty, "void" | "wl_display" | "wl_surface" | "Display" | "xcb_connection_t" | "ANativeWindow" | "AHardwareBuffer" | "CAMetalLayer" | "IDirectFB" | "IDirectFBSurface" ) } #[derive(Debug, Copy, Clone)] pub enum ConstVal { U32(u32), U64(u64), Float(f32), } impl ConstVal { pub fn bits(&self) -> u64 { match self { ConstVal::U64(n) => *n, _ => panic!("Constval not supported"), } } } pub trait ConstantExt { fn constant(&self, enum_name: &str) -> Constant; fn variant_ident(&self, enum_name: &str) -> Ident; fn notation(&self) -> Option<&str>; fn is_alias(&self) -> bool { false } } impl ConstantExt for vkxml::ExtensionEnum { fn constant(&self, _enum_name: &str) -> Constant { Constant::from_extension_enum(self).unwrap() } fn variant_ident(&self, enum_name: &str) -> Ident { variant_ident(enum_name, &self.name) } fn notation(&self) -> Option<&str> { self.notation.as_deref() } } impl ConstantExt for vk_parse::Enum { fn constant(&self, enum_name: &str) -> Constant { Constant::from_vk_parse_enum_spec(&self.spec, Some(enum_name), None) .unwrap() .0 } fn variant_ident(&self, enum_name: &str) -> Ident { variant_ident(enum_name, &self.name) } fn notation(&self) -> Option<&str> { self.comment.as_deref() } fn is_alias(&self) -> bool { matches!(self.spec, vk_parse::EnumSpec::Alias { .. }) } } impl ConstantExt for vkxml::Constant { fn constant(&self, _enum_name: &str) -> Constant { Constant::from_constant(self) } fn variant_ident(&self, enum_name: &str) -> Ident { variant_ident(enum_name, &self.name) } fn notation(&self) -> Option<&str> { self.notation.as_deref() } } #[derive(Clone, Debug)] pub enum Constant { Number(i32), Hex(String), BitPos(u32), CExpr(vkxml::CExpression), Text(String), Alias(Ident), } impl quote::ToTokens for Constant { fn to_tokens(&self, tokens: &mut TokenStream) { match *self { Constant::Number(n) => { let number = interleave_number('_', 3, &n.to_string()); syn::LitInt::new(&number, Span::call_site()).to_tokens(tokens); } Constant::Hex(ref s) => { let number = interleave_number('_', 4, s); syn::LitInt::new(&format!("0x{}", number), Span::call_site()).to_tokens(tokens); } Constant::Text(ref text) => text.to_tokens(tokens), Constant::CExpr(ref expr) => { let (_, (_, rexpr)) = cexpr(expr).expect("Unable to parse cexpr"); tokens.extend(rexpr.parse::()); } Constant::BitPos(pos) => { let value = 1u64 << pos; let bit_string = format!("{:b}", value); let bit_string = interleave_number('_', 4, &bit_string); syn::LitInt::new(&format!("0b{}", bit_string), Span::call_site()).to_tokens(tokens); } Constant::Alias(ref value) => tokens.extend(quote!(Self::#value)), } } } impl quote::ToTokens for ConstVal { fn to_tokens(&self, tokens: &mut TokenStream) { match self { ConstVal::U32(n) => n.to_tokens(tokens), ConstVal::U64(n) => n.to_tokens(tokens), ConstVal::Float(f) => f.to_tokens(tokens), } } } // Interleaves a number, for example 100000 => 100_000. Mostly used to make clippy happy fn interleave_number(symbol: char, count: usize, n: &str) -> String { let number: String = n .chars() .rev() .enumerate() .fold(String::new(), |mut acc, (idx, next)| { if idx != 0 && idx % count == 0 { acc.push(symbol); } acc.push(next); acc }); number.chars().rev().collect() } impl Constant { pub fn value(&self) -> Option { match *self { Constant::Number(n) => Some(ConstVal::U64(n as u64)), Constant::Hex(ref hex) => u64::from_str_radix(&hex, 16).ok().map(ConstVal::U64), Constant::BitPos(pos) => Some(ConstVal::U64(1u64 << pos)), _ => None, } } pub fn ty(&self) -> CType { match self { Constant::Number(_) | Constant::Hex(_) => CType::USize, Constant::CExpr(expr) => { let (_, (ty, _)) = cexpr(expr).expect("Unable to parse cexpr"); ty } _ => unimplemented!(), } } pub fn from_extension_enum(constant: &vkxml::ExtensionEnum) -> Option { let number = constant.number.map(Constant::Number); let hex = constant.hex.as_ref().map(|hex| Constant::Hex(hex.clone())); let bitpos = constant.bitpos.map(Constant::BitPos); let expr = constant .c_expression .as_ref() .map(|e| Constant::CExpr(e.clone())); number.or(hex).or(bitpos).or(expr) } pub fn from_constant(constant: &vkxml::Constant) -> Self { let number = constant.number.map(Constant::Number); let hex = constant.hex.as_ref().map(|hex| Constant::Hex(hex.clone())); let bitpos = constant.bitpos.map(Constant::BitPos); let expr = constant .c_expression .as_ref() .map(|e| Constant::CExpr(e.clone())); number.or(hex).or(bitpos).or(expr).expect("") } /// Returns (Constant, optional base type, is_alias) pub fn from_vk_parse_enum_spec( spec: &vk_parse::EnumSpec, enum_name: Option<&str>, extension_number: Option, ) -> Option<(Self, Option, bool)> { use vk_parse::EnumSpec; match spec { EnumSpec::Bitpos { bitpos, extends } => { Some((Self::BitPos(*bitpos as u32), extends.clone(), false)) } EnumSpec::Offset { offset, extends, extnumber, dir: positive, } => { let ext_base = 1_000_000_000; let ext_block_size = 1000; let extnumber = extnumber .or(extension_number) .expect("Need an extension number"); let value = ext_base + (extnumber - 1) * ext_block_size + offset; let value = if *positive { value } else { -value }; Some((Self::Number(value as i32), Some(extends.clone()), false)) } EnumSpec::Value { value, extends } => { let value = value .strip_prefix("0x") .map(|hex| Self::Hex(hex.to_owned())) .or_else(|| value.parse::().ok().map(Self::Number))?; Some((value, extends.clone(), false)) } EnumSpec::Alias { alias, extends } => { let base_type = extends.as_deref().or(enum_name)?; let key = variant_ident(base_type, &alias); if key == "DISPATCH_BASE" { None } else { Some((Self::Alias(key), Some(base_type.to_owned()), true)) } } _ => None, } } } pub trait FeatureExt { fn version_string(&self) -> String; fn is_version(&self, major: u32, minor: u32) -> bool; } impl FeatureExt for vkxml::Feature { fn is_version(&self, major: u32, minor: u32) -> bool { let self_major = self.version as u32; let self_minor = (self.version * 10.0) as u32 - self_major * 10; major == self_major && self_minor == minor } fn version_string(&self) -> String { let mut version = format!("{}", self.version); if version.len() == 1 { version = format!("{}_0", version) } version.replace(".", "_") } } #[derive(Debug, Copy, Clone)] pub enum FunctionType { Static, Entry, Instance, Device, } pub trait CommandExt { /// Returns the ident in snake_case and without the 'vk' prefix. fn function_type(&self) -> FunctionType; /// /// Returns true if the command is a device level command. This is indicated by /// the type of the first parameter. fn command_ident(&self) -> Ident; } impl CommandExt for vkxml::Command { fn command_ident(&self) -> Ident { format_ident!("{}", &self.name[2..].to_snake_case()) } fn function_type(&self) -> FunctionType { let is_first_param_device = self .param .get(0) .map(|field| { matches!( field.basetype.as_str(), "VkDevice" | "VkCommandBuffer" | "VkQueue" ) }) .unwrap_or(false); match self.name.as_str() { "vkGetInstanceProcAddr" => FunctionType::Static, "vkCreateInstance" | "vkEnumerateInstanceLayerProperties" | "vkEnumerateInstanceExtensionProperties" | "vkEnumerateInstanceVersion" => FunctionType::Entry, // This is actually not a device level function "vkGetDeviceProcAddr" => FunctionType::Instance, _ => { if is_first_param_device { FunctionType::Device } else { FunctionType::Instance } } } } } pub trait FieldExt { /// Returns the name of the parameter that doesn't clash with Rusts reserved /// keywords fn param_ident(&self) -> Ident; /// The inner type of this field, with one level of pointers removed fn inner_type_tokens(&self) -> TokenStream; /// Returns reference-types wrapped in their safe variant. (Dynamic) arrays become /// slices, pointers become Rust references. fn safe_type_tokens(&self, lifetime: TokenStream) -> TokenStream; /// Returns the basetype ident and removes the 'Vk' prefix. When `is_ffi_param` is `true` /// array types (e.g. `[f32; 3]`) will be converted to pointer types (e.g. `&[f32; 3]`), /// which is needed for `C` function parameters. Set to `false` for struct definitions. fn type_tokens(&self, is_ffi_param: bool) -> TokenStream; fn is_clone(&self) -> bool; /// Whether this is C's `void` type (not to be mistaken with a void _pointer_!) fn is_void(&self) -> bool; } pub trait ToTokens { fn to_tokens(&self, is_const: bool) -> TokenStream; /// Returns the topmost pointer as safe reference fn to_safe_tokens(&self, is_const: bool, lifetime: TokenStream) -> TokenStream; } impl ToTokens for vkxml::ReferenceType { fn to_tokens(&self, is_const: bool) -> TokenStream { let r = if is_const { quote!(*const) } else { quote!(*mut) }; match self { vkxml::ReferenceType::Pointer => quote!(#r), vkxml::ReferenceType::PointerToPointer => quote!(#r *mut), vkxml::ReferenceType::PointerToConstPointer => quote!(#r *const), } } fn to_safe_tokens(&self, is_const: bool, lifetime: TokenStream) -> TokenStream { let r = if is_const { quote!(&#lifetime) } else { quote!(&#lifetime mut) }; match self { vkxml::ReferenceType::Pointer => quote!(#r), vkxml::ReferenceType::PointerToPointer => quote!(#r *mut), vkxml::ReferenceType::PointerToConstPointer => quote!(#r *const), } } } fn name_to_tokens(type_name: &str) -> Ident { let new_name = match type_name { "uint8_t" => "u8", "uint16_t" => "u16", "uint32_t" => "u32", "uint64_t" => "u64", "int8_t" => "i8", "int16_t" => "i16", "int32_t" => "i32", "int64_t" => "i64", "size_t" => "usize", "int" => "c_int", "void" => "c_void", "char" => "c_char", "float" => "f32", "double" => "f64", "long" => "c_ulong", _ => type_name.strip_prefix("Vk").unwrap_or(type_name), }; let new_name = new_name.replace("FlagBits", "Flags"); format_ident!("{}", new_name.as_str()) } fn map_identifier_to_rust(ident: Ident) -> TokenTree { match ident.to_string().as_str() { "VK_MAKE_VERSION" => { TokenTree::Group(Group::new(Delimiter::None, quote!(crate::vk::make_version))) } s => format_ident!("{}", constant_name(s)).into(), } } /// Parse and yield a C expression that is valid to write in Rust /// Identifiers are replaced with their Rust vk equivalent. /// /// Examples: /// - `VK_MAKE_VERSION(1, 2, VK_HEADER_VERSION)` -> `crate::vk::make_version(1, 2, HEADER_VERSION)` /// - `2*VK_UUID_SIZE` -> `2 * UUID_SIZE` fn convert_c_expression(c_expr: &str) -> TokenStream { fn rewrite_token_stream(stream: TokenStream) -> TokenStream { stream .into_iter() .map(|tt| match tt { TokenTree::Group(group) => TokenTree::Group(Group::new( group.delimiter(), rewrite_token_stream(group.stream()), )), TokenTree::Ident(term) => map_identifier_to_rust(term), _ => tt, }) .collect::() } let c_expr = c_expr.parse().unwrap(); rewrite_token_stream(c_expr) } impl FieldExt for vkxml::Field { fn is_clone(&self) -> bool { true } fn param_ident(&self) -> Ident { let name = self.name.as_deref().unwrap_or("field"); let name_corrected = match name { "type" => "ty", _ => name, }; format_ident!("{}", name_corrected.to_snake_case().as_str()) } fn inner_type_tokens(&self) -> TokenStream { assert!(!self.is_void()); let ty = name_to_tokens(&self.basetype); match self.reference { Some(vkxml::ReferenceType::PointerToPointer) => quote!(*mut #ty), Some(vkxml::ReferenceType::PointerToConstPointer) => quote!(*const #ty), _ => quote!(#ty), } } fn safe_type_tokens(&self, lifetime: TokenStream) -> TokenStream { assert!(!self.is_void()); match self.array { // The outer type fn type_tokens() returns is [], which fits our "safe" prescription Some(vkxml::ArrayType::Static) => self.type_tokens(false), Some(vkxml::ArrayType::Dynamic) => { let ty = self.inner_type_tokens(); quote!([#ty]) } None => { let ty = name_to_tokens(&self.basetype); let pointer = self .reference .as_ref() .map(|r| r.to_safe_tokens(self.is_const, lifetime)); quote!(#pointer #ty) } } } fn type_tokens(&self, is_ffi_param: bool) -> TokenStream { assert!(!self.is_void()); let ty = name_to_tokens(&self.basetype); match self.array { Some(vkxml::ArrayType::Static) => { assert!(self.reference.is_none()); let size = self .size .as_ref() .or_else(|| self.size_enumref.as_ref()) .expect("Should have size"); // Make sure we also rename the constant, that is // used inside the static array let size: TokenStream = constant_name(size).parse().unwrap(); // arrays in c are always passed as a pointer if is_ffi_param { quote!(*const [#ty; #size]) } else { quote!([#ty; #size]) } } _ => { let pointer = self.reference.as_ref().map(|r| r.to_tokens(self.is_const)); quote!(#pointer #ty) } } } fn is_void(&self) -> bool { self.basetype == "void" && self.reference.is_none() } } pub type CommandMap<'a> = HashMap; fn generate_function_pointers<'a>( ident: Ident, commands: &[&'a vkxml::Command], aliases: &HashMap, fn_cache: &mut HashSet<&'a str, impl BuildHasher>, ) -> TokenStream { // Commands can have duplicates inside them because they are declared per features. But we only // really want to generate one function pointer. let commands = { let mut cache = HashSet::new(); let mut cmd_vec: Vec<&vkxml::Command> = Vec::new(); for cmd in commands { let name = cmd.name.as_str(); if !cache.contains(name) { cmd_vec.push(cmd); cache.insert(name); } } cmd_vec }; // PFN function pointers are global and can not have duplicates. This can happen because there // are aliases to commands let commands_pfn: Vec<_> = commands .iter() .filter(|cmd| { let ident = cmd.name.as_str(); if !fn_cache.contains(ident) { fn_cache.insert(ident); true } else { false } }) .collect(); let function_name_raw = |name: &str| -> String { if let Some(alias_name) = aliases.get(name) { alias_name.to_string() } else { name.to_string() } }; let function_name = |name: &str| -> Ident { let fn_name = function_name_raw(&name); format_ident!("{}", fn_name[2..].to_snake_case().as_str()) }; let names: Vec<_> = commands .iter() .map(|cmd| function_name(&cmd.name)) .collect(); let names_ref = &names; let raw_names: Vec<_> = commands .iter() .map(|cmd| { let byt = std::ffi::CString::new(function_name_raw(cmd.name.as_str())).unwrap(); Literal::byte_string(byt.to_bytes_with_nul()) }) .collect(); let raw_names_ref = &raw_names; let khronos_links: Vec<_> = commands .iter() .map(|cmd| khronos_link(&function_name_raw(cmd.name.as_str()))) .collect(); let params: Vec> = commands .iter() .map(|cmd| { let params: Vec<_> = cmd .param .iter() .map(|field| { let name = field.param_ident(); let ty = field.type_tokens(true); (name, ty) }) .collect(); params }) .collect(); let params_names: Vec> = params .iter() .map(|inner_params| { inner_params .iter() .map(|(param_name, _)| param_name.clone()) .collect() }) .collect(); let expanded_params: Vec<_> = params .iter() .map(|inner_params| { let inner_params_iter = inner_params.iter().map(|&(ref param_name, ref param_ty)| { quote! {#param_name: #param_ty} }); quote! { #(#inner_params_iter,)* } }) .collect(); let expanded_params_unused: Vec<_> = params .iter() .map(|inner_params| { let inner_params_iter = inner_params.iter().map(|&(ref param_name, ref param_ty)| { let unused_name = format_ident!("{}", format!("_{}", param_name).as_str()); quote! {#unused_name: #param_ty} }); quote! { #(#inner_params_iter,)* } }) .collect(); let expanded_params_ref = &expanded_params; let pfn_names: Vec<_> = commands_pfn .iter() .map(|cmd| format_ident!("{}", format!("PFN_{}", cmd.name.as_str()))) .collect(); let pfn_names_ref = &pfn_names; let signature_params: Vec> = commands .iter() .map(|cmd| { let params: Vec<_> = cmd .param .iter() .map(|field| { let name = field.param_ident(); let ty = field.type_tokens(true); quote! { #name: #ty } }) .collect(); params }) .collect(); let signature_params_ref = &signature_params; let return_types: Vec<_> = commands .iter() .map(|cmd| { if cmd.return_type.is_void() { quote!() } else { let ret_ty_tokens = cmd.return_type.type_tokens(true); quote!(-> #ret_ty_tokens) } }) .collect(); let return_types_ref = &return_types; quote! { #( #[allow(non_camel_case_types)] pub type #pfn_names_ref = extern "system" fn(#(#signature_params_ref),*) #return_types_ref; )* pub struct #ident { #( pub #names_ref: extern "system" fn(#expanded_params_ref) #return_types_ref, )* } unsafe impl Send for #ident {} unsafe impl Sync for #ident {} impl ::std::clone::Clone for #ident { fn clone(&self) -> Self { #ident{ #(#names: self.#names,)* } } } impl #ident { pub fn load(mut _f: F) -> Self where F: FnMut(&::std::ffi::CStr) -> *const c_void { #ident { #( #names_ref: unsafe { extern "system" fn #names_ref (#expanded_params_unused) #return_types_ref { panic!(concat!("Unable to load ", stringify!(#names_ref))) } let cname = ::std::ffi::CStr::from_bytes_with_nul_unchecked(#raw_names_ref); let val = _f(cname); if val.is_null(){ #names_ref } else{ ::std::mem::transmute(val) } }, )* } } #( #[doc = #khronos_links] pub unsafe fn #names_ref(&self, #expanded_params_ref) #return_types_ref { (self.#names)(#(#params_names,)*) } )* } } } pub struct ExtensionConstant<'a> { pub name: &'a str, pub constant: Constant, } impl<'a> ConstantExt for ExtensionConstant<'a> { fn constant(&self, _enum_name: &str) -> Constant { self.constant.clone() } fn variant_ident(&self, enum_name: &str) -> Ident { variant_ident(enum_name, self.name) } fn notation(&self) -> Option<&str> { None } } pub fn generate_extension_constants<'a>( extension_name: &str, extension_number: i64, extension_items: &'a [vk_parse::ExtensionChild], const_cache: &mut HashSet<&'a str, impl BuildHasher>, const_values: &mut BTreeMap, ) -> TokenStream { let items = extension_items .iter() .filter_map(|item| match item { vk_parse::ExtensionChild::Require { items, .. } => Some(items.iter()), _ => None, }) .flat_map(|iter| iter); let enum_tokens = items.filter_map(|item| match item { vk_parse::InterfaceItem::Enum(enum_) => { if const_cache.contains(enum_.name.as_str()) { return None; } let (constant, extends, is_alias) = Constant::from_vk_parse_enum_spec(&enum_.spec, None, Some(extension_number))?; let extends = extends?; let ext_constant = ExtensionConstant { name: &enum_.name, constant, }; let ident = name_to_tokens(&extends); const_values .get_mut(&ident) .unwrap() .values .push(ConstantMatchInfo { ident: ext_constant.variant_ident(&extends), is_alias, }); let impl_block = bitflags_impl_block(ident, &extends, &[&ext_constant]); let doc_string = format!("Generated from '{}'", extension_name); let q = quote! { #[doc = #doc_string] #impl_block }; const_cache.insert(enum_.name.as_str()); Some(q) } _ => None, }); quote! { #(#enum_tokens)* } } pub fn generate_extension_commands<'a>( extension_name: &str, items: &[vk_parse::ExtensionChild], cmd_map: &CommandMap<'a>, cmd_aliases: &HashMap, fn_cache: &mut HashSet<&'a str, impl BuildHasher>, ) -> TokenStream { let mut commands = Vec::new(); let mut aliases = HashMap::new(); items .iter() .filter_map(|ext_item| match ext_item { vk_parse::ExtensionChild::Require { items, .. } => { Some(items.iter().filter_map(|item| match item { vk_parse::InterfaceItem::Command { ref name, .. } => Some(name), _ => None, })) } _ => None, }) .flatten() .for_each(|name| { if let Some(cmd) = cmd_map.get(name).copied() { commands.push(cmd); } else if let Some(cmd) = cmd_aliases .get(name) .and_then(|alias_name| cmd_map.get(alias_name).copied()) { aliases.insert(cmd.name.clone(), name.to_string()); commands.push(cmd); } }); let name = format!("{}Fn", extension_name.to_camel_case()); let ident = format_ident!("{}", &name[2..]); let fp = generate_function_pointers(ident.clone(), &commands, &aliases, fn_cache); let byte_name = format!("{}\0", extension_name); let spec_version = items .iter() .find_map(|ext_item| match ext_item { vk_parse::ExtensionChild::Require { items, .. } => { items.iter().find_map(|item| match item { vk_parse::InterfaceItem::Enum(ref e) if e.name.contains("SPEC_VERSION") => { Some(e) } _ => None, }) } _ => None, }) .and_then(|e| { if let vk_parse::EnumSpec::Value { value, .. } = &e.spec { let v: u32 = str::parse(value).unwrap(); Some(quote!(pub const SPEC_VERSION: u32 = #v;)) } else { None } }); let byte_name_ident = syn::LitByteStr::new(byte_name.as_bytes(), Span::call_site()); let extension_cstr = quote! { impl #ident { pub fn name() -> &'static ::std::ffi::CStr { ::std::ffi::CStr::from_bytes_with_nul(#byte_name_ident).expect("Wrong extension string") } #spec_version } }; quote! { #extension_cstr #fp } } pub fn generate_extension<'a>( extension: &'a vk_parse::Extension, cmd_map: &CommandMap<'a>, const_cache: &mut HashSet<&'a str, impl BuildHasher>, const_values: &mut BTreeMap, cmd_aliases: &HashMap, fn_cache: &mut HashSet<&'a str, impl BuildHasher>, ) -> Option { // Okay this is a little bit odd. We need to generate all extensions, even disabled ones, // because otherwise some StructureTypes won't get generated. But we don't generate extensions // that are reserved if extension.name.contains("RESERVED") { return None; } let extension_tokens = generate_extension_constants( &extension.name, extension.number.unwrap_or(0), &extension.children, const_cache, const_values, ); let fp = generate_extension_commands( &extension.name, &extension.children, cmd_map, cmd_aliases, fn_cache, ); let q = quote! { #fp #extension_tokens }; Some(q) } pub fn generate_define(define: &vkxml::Define) -> TokenStream { let name = constant_name(&define.name); let ident = format_ident!("{}", name); let deprecated = define .comment .as_ref() .map_or(false, |c| c.contains("DEPRECATED")); if name == "NULL_HANDLE" || deprecated { quote!() } else if let Some(value) = &define.value { str::parse::(value).map_or(quote!(), |v| quote!(pub const #ident: u32 = #v;)) } else if let Some(c_expr) = &define.c_expression { if define.defref.contains(&"VK_MAKE_VERSION".to_string()) { let c_expr = convert_c_expression(c_expr); quote!(pub const #ident: u32 = #c_expr;) } else { quote!() } } else { quote!() } } pub fn generate_typedef(typedef: &vkxml::Typedef) -> TokenStream { if typedef.basetype.is_empty() { // Ignore forward declarations quote! {} } else { let typedef_name = name_to_tokens(&typedef.name); let typedef_ty = name_to_tokens(&typedef.basetype); let khronos_link = khronos_link(&typedef.name); quote! { #[doc = #khronos_link] pub type #typedef_name = #typedef_ty; } } } pub fn generate_bitmask( bitmask: &vkxml::Bitmask, bitflags_cache: &mut HashSet, const_values: &mut BTreeMap, ) -> Option { // Workaround for empty bitmask if bitmask.name.is_empty() { return None; } // If this enum has constants, then it will generated later in generate_enums. if bitmask.enumref.is_some() { return None; } let name = &bitmask.name[2..]; let ident = format_ident!("{}", name); if bitflags_cache.contains(&ident) { return None; }; bitflags_cache.insert(ident.clone()); const_values.insert(ident.clone(), Default::default()); let khronos_link = khronos_link(&bitmask.name); let type_ = name_to_tokens(&bitmask.basetype); Some(quote! { #[repr(transparent)] #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)] #[doc = #khronos_link] pub struct #ident(pub(crate) #type_); vk_bitflags_wrapped!(#ident, 0b0, #type_); }) } pub enum EnumType { Bitflags(TokenStream), Enum(TokenStream), } fn is_enum_variant_with_typo(variant_name: &str) -> bool { // All these names are aliases and make little sense in our // enum structure, they are better omitted entirely. matches!( variant_name, "VK_STENCIL_FRONT_AND_BACK" | "VK_COLORSPACE_SRGB_NONLINEAR" | "VK_QUERY_SCOPE_COMMAND_BUFFER" | "VK_QUERY_SCOPE_RENDER_PASS" | "VK_QUERY_SCOPE_COMMAND" ) } static TRAILING_NUMBER: Lazy = Lazy::new(|| Regex::new("(\\d+)$").unwrap()); pub fn variant_ident(enum_name: &str, variant_name: &str) -> Ident { let variant_name = variant_name.to_uppercase(); let _name = enum_name.replace("FlagBits", ""); // TODO: Should be read from vk.xml id:2 // TODO: Also needs to be more robust, vendor names can be substrings from itself, id:4 // like NVX and NV let vendors = [ "_NVX", "_KHR", "_EXT", "_NV", "_AMD", "_ANDROID", "_GOOGLE", "_INTEL", ]; let struct_name = _name.to_shouty_snake_case(); let vendor = vendors .iter() .find(|&vendor| struct_name.ends_with(vendor)) .cloned() .unwrap_or(""); let struct_name = struct_name.strip_suffix(vendor).unwrap(); let struct_name = TRAILING_NUMBER.replace(struct_name, "_$1"); let variant_name = variant_name .strip_suffix(vendor) .unwrap_or_else(|| variant_name.as_str()); let new_variant_name = variant_name .strip_prefix(struct_name.as_ref()) .unwrap_or_else(|| { if enum_name == "VkResult" || is_enum_variant_with_typo(variant_name) { variant_name.strip_prefix("VK").unwrap() } else { panic!( "Failed to strip {} prefix from enum variant {}", struct_name, variant_name ) } }); // Both of the above strip_prefix leave a leading `_`: let new_variant_name = new_variant_name.strip_prefix("_").unwrap(); // Replace _BIT anywhere in the string, also works when there's a trailing // vendor extension in the variant name that's not in the enum/type name: let new_variant_name = new_variant_name.replace("_BIT", ""); let is_digit = new_variant_name .chars() .next() .map(|c| c.is_digit(10)) .unwrap_or(false); if is_digit { format_ident!("{}", format!("TYPE_{}", new_variant_name).as_str()) } else { format_ident!("{}", new_variant_name) } } pub fn bitflags_impl_block( ident: Ident, enum_name: &str, constants: &[&impl ConstantExt], ) -> TokenStream { let variants = constants .iter() .map(|constant| { let variant_ident = constant.variant_ident(enum_name); let constant = constant.constant(enum_name); let tokens = if let Constant::Alias(_) = &constant { quote!(#constant) } else { quote!(Self(#constant)) }; (variant_ident, tokens) }) .collect_vec(); let notations = constants.iter().map(|constant| { constant.notation().map(|n| { if n.to_lowercase().contains("backwards") { quote!(#[deprecated = #n]) } else { quote!(#[doc = #n]) } }) }); let variants = variants .iter() .zip(notations.clone()) .map(|((variant_ident, value), ref notation)| { quote! { #notation pub const #variant_ident: Self = #value; } }); quote! { impl #ident { #(#variants)* } } } pub fn generate_enum<'a>( enum_: &'a vk_parse::Enums, const_cache: &mut HashSet<&'a str, impl BuildHasher>, const_values: &mut BTreeMap, bitflags_cache: &mut HashSet, ) -> EnumType { let name = enum_.name.as_ref().unwrap(); let clean_name = name.strip_prefix("Vk").unwrap(); let _name = clean_name.replace("FlagBits", "Flags"); let ident = format_ident!("{}", _name.as_str()); let constants = enum_ .children .iter() .filter_map(|elem| match *elem { vk_parse::EnumsChild::Enum(ref constant) => Some(constant), _ => None, }) .filter(|constant| match &constant.spec { vk_parse::EnumSpec::Alias { alias, .. } => { // Remove any alias whose name is identical after name de-mangling. For example // the XML contains compatibility aliases for variants without _BIT postfix // which are removed by the generator anyway, after which they become identical. let alias_name = constant.variant_ident(name); let aliases_to = variant_ident(name, alias); alias_name != aliases_to } _ => true, }) .collect_vec(); let mut values = Vec::with_capacity(constants.len()); for constant in &constants { const_cache.insert(constant.name.as_str()); values.push(ConstantMatchInfo { ident: constant.variant_ident(name), is_alias: constant.is_alias(), }); } const_values.insert( ident.clone(), ConstantTypeInfo { values, bitwidth: enum_.bitwidth, }, ); let khronos_link = khronos_link(name); if clean_name.contains("Bit") { let ident = format_ident!("{}", _name.as_str()); let all_bits = constants .iter() .filter_map(|constant| constant.constant(name).value()) .fold(0, |acc, next| acc | next.bits()); let type_ = if enum_.bitwidth == Some(64u32) { quote!(Flags64) } else { quote!(Flags) }; let bit_string = format!("{:b}", all_bits); let bit_string = interleave_number('_', 4, &bit_string); let all_bits_term = syn::LitInt::new(&format!("0b{}", bit_string), Span::call_site()); if bitflags_cache.contains(&ident) { EnumType::Bitflags(quote! {}) } else { let impl_bitflags = bitflags_impl_block(ident.clone(), name, &constants); bitflags_cache.insert(ident.clone()); let q = quote! { #[repr(transparent)] #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)] #[doc = #khronos_link] pub struct #ident(pub(crate) #type_); vk_bitflags_wrapped!(#ident, #all_bits_term, #type_); #impl_bitflags }; EnumType::Bitflags(q) } } else { let (struct_attribute, special_quote) = match _name.as_str() { //"StructureType" => generate_structure_type(&_name, _enum, create_info_constants), "Result" => (quote!(#[must_use]), generate_result(ident.clone(), enum_)), _ => (quote!(), quote!()), }; let impl_block = bitflags_impl_block(ident.clone(), name, &constants); let enum_quote = quote! { #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Default)] #[repr(transparent)] #[doc = #khronos_link] #struct_attribute pub struct #ident(pub(crate) i32); impl #ident { pub const fn from_raw(x: i32) -> Self { #ident(x) } pub const fn as_raw(self) -> i32 { self.0 } } #impl_block }; let q = quote! { #enum_quote #special_quote }; EnumType::Enum(q) } } pub fn generate_result(ident: Ident, enum_: &vk_parse::Enums) -> TokenStream { let notation = enum_.children.iter().filter_map(|elem| { let (variant_name, notation) = match *elem { vk_parse::EnumsChild::Enum(ref constant) => ( constant.name.as_str(), constant.comment.as_deref().unwrap_or(""), ), _ => { return None; } }; let variant_ident = variant_ident(&enum_.name.as_ref().unwrap(), variant_name); Some(quote! { #ident::#variant_ident => Some(#notation) }) }); quote! { impl ::std::error::Error for #ident {} impl fmt::Display for #ident { fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { let name = match *self { #(#notation),*, _ => None, }; if let Some(x) = name { fmt.write_str(x) } else { // If we don't have a nice message to show, call the generated `Debug` impl // which includes *all* enum variants, including those from extensions. ::fmt(&self, fmt) } } } } } fn is_static_array(field: &vkxml::Field) -> bool { field .array .as_ref() .map(|ty| matches!(ty, vkxml::ArrayType::Static)) .unwrap_or(false) } pub fn derive_default(_struct: &vkxml::Struct) -> Option { let name = name_to_tokens(&_struct.name); let members = _struct.elements.iter().filter_map(|elem| match *elem { vkxml::StructElement::Member(ref field) => Some(field), _ => None, }); let is_structure_type = |field: &vkxml::Field| field.basetype == "VkStructureType"; // This are also pointers, and therefor also don't implement Default. The spec // also doesn't mark them as pointers let handles = ["LPCWSTR", "HANDLE", "HINSTANCE", "HWND", "HMONITOR"]; let contains_ptr = members.clone().any(|field| field.reference.is_some()); let contains_structure_type = members.clone().any(is_structure_type); let contains_static_array = members.clone().any(is_static_array); if !(contains_ptr || contains_structure_type || contains_static_array) { return None; }; let default_fields = members.clone().map(|field| { let param_ident = field.param_ident(); if is_structure_type(field) { let ty = field .type_enums .as_ref() .and_then(|ty| ty.split(',').next()); if let Some(variant) = ty { let variant_ident = variant_ident("VkStructureType", variant); quote! { #param_ident: StructureType::#variant_ident } } else { quote! { #param_ident: unsafe { ::std::mem::zeroed() } } } } else if let Some(ref reference) = field.reference { match reference { vkxml::ReferenceType::Pointer => { if field.is_const { quote! { #param_ident: ::std::ptr::null() } } else { quote! { #param_ident: ::std::ptr::null_mut() } } } vkxml::ReferenceType::PointerToPointer => { quote! { #param_ident: ::std::ptr::null_mut() } } vkxml::ReferenceType::PointerToConstPointer => { if field.is_const { quote! { #param_ident: ::std::ptr::null() } } else { quote! { #param_ident: ::std::ptr::null_mut() } } } } } else if is_static_array(field) || handles.contains(&field.basetype.as_str()) { quote! { #param_ident: unsafe { ::std::mem::zeroed() } } } else { let ty = field.type_tokens(false); quote! { #param_ident: #ty::default() } } }); let q = quote! { impl ::std::default::Default for #name { fn default() -> #name { #name { #( #default_fields ),* } } } }; Some(q) } pub fn derive_debug( _struct: &vkxml::Struct, union_types: &HashSet<&str, impl BuildHasher>, ) -> Option { let name = name_to_tokens(&_struct.name); let members = _struct.elements.iter().filter_map(|elem| match *elem { vkxml::StructElement::Member(ref field) => Some(field), _ => None, }); let contains_pfn = members.clone().any(|field| { field .name .as_ref() .map(|n| n.contains("pfn")) .unwrap_or(false) }); let contains_static_array = members .clone() .any(|x| is_static_array(x) && x.basetype == "char"); let contains_union = members .clone() .any(|field| union_types.contains(field.basetype.as_str())); if !(contains_union || contains_static_array || contains_pfn) { return None; } let debug_fields = members.clone().map(|field| { let param_ident = field.param_ident(); let param_str = param_ident.to_string(); let debug_value = if is_static_array(field) && field.basetype == "char" { quote! { &unsafe { ::std::ffi::CStr::from_ptr(self.#param_ident.as_ptr() as *const c_char) } } } else if param_str.contains("pfn") { quote! { &(self.#param_ident.map(|x| x as *const ())) } } else if union_types.contains(field.basetype.as_str()) { quote!(&"union") } else { quote! { &self.#param_ident } }; quote! { .field(#param_str, #debug_value) } }); let name_str = name.to_string(); let q = quote! { impl fmt::Debug for #name { fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { fmt.debug_struct(#name_str) #(#debug_fields)* .finish() } } }; Some(q) } pub fn derive_setters( _struct: &vkxml::Struct, root_struct_names: &HashSet, ) -> Option { if &_struct.name == "VkBaseInStructure" || &_struct.name == "VkBaseOutStructure" || &_struct.name == "VkTransformMatrixKHR" || &_struct.name == "VkAccelerationStructureInstanceKHR" { return None; } let name = name_to_tokens(&_struct.name); let name_builder = name_to_tokens(&(_struct.name.clone() + "Builder")); let members = _struct.elements.iter().filter_map(|elem| match *elem { vkxml::StructElement::Member(ref field) => Some(field), _ => None, }); let has_next = members.clone().any(|field| field.param_ident() == "p_next"); let nofilter_count_members = [ "VkPipelineViewportStateCreateInfo.pViewports", "VkPipelineViewportStateCreateInfo.pScissors", "VkDescriptorSetLayoutBinding.pImmutableSamplers", ]; let filter_members: Vec = members .clone() .filter_map(|field| { let field_name = field.name.as_ref().unwrap(); // Associated _count members if field.array.is_some() { if let Some(ref array_size) = field.size { if !array_size.starts_with("latexmath") && !nofilter_count_members .iter() .any(|&n| n == (_struct.name.clone() + "." + field_name)) { return Some((*array_size).clone()); } } } // VkShaderModuleCreateInfo requiers a custom setter if field_name == "codeSize" { return Some(field_name.clone()); } None }) .collect(); let setters = members.clone().filter_map(|field| { let param_ident = field.param_ident(); let param_ty_tokens = field.safe_type_tokens(quote!('a)); let param_ident_string = param_ident.to_string(); if param_ident_string == "s_type" || param_ident_string == "p_next" { return None; } let param_ident_short = param_ident_string .strip_prefix("p_") .or_else(|| param_ident_string.strip_prefix("pp_")) .unwrap_or_else(|| param_ident_string.as_str()); let param_ident_short = format_ident!("{}", ¶m_ident_short); if let Some(name) = field.name.as_ref() { // Fiter if filter_members.iter().any(|n| *n == *name) { return None; } // Unique cases if name == "pCode" { return Some(quote!{ pub fn code(mut self, code: &'a [u32]) -> Self { self.inner.code_size = code.len() * 4; self.inner.p_code = code.as_ptr() as *const u32; self } }); } if name == "pSampleMask" { return Some(quote!{ pub fn sample_mask(mut self, sample_mask: &'a [SampleMask]) -> Self { self.inner.p_sample_mask = sample_mask.as_ptr() as *const SampleMask; self } }); } if name == "ppGeometries" { return Some(quote!{ pub fn geometries_ptrs(mut self, geometries: &'a [&'a AccelerationStructureGeometryKHR]) -> Self { self.inner.geometry_count = geometries.len() as _; self.inner.pp_geometries = geometries.as_ptr() as *const *const _; self } }); } } // TODO: Improve in future when https://github.com/rust-lang/rust/issues/53667 is merged id:6 if field.reference.is_some() { if field.basetype == "char" && matches!(field.reference, Some(vkxml::ReferenceType::Pointer)) { assert!(field.null_terminate); assert_eq!(field.size, None); return Some(quote!{ pub fn #param_ident_short(mut self, #param_ident_short: &'a ::std::ffi::CStr) -> Self { self.inner.#param_ident = #param_ident_short.as_ptr(); self } }); } if matches!(field.array, Some(vkxml::ArrayType::Dynamic)) { if let Some(ref array_size) = field.size { if !array_size.starts_with("latexmath") { let mut slice_param_ty_tokens = field.safe_type_tokens(quote!('a)); let mut ptr = if field.is_const { quote!(.as_ptr()) } else { quote!(.as_mut_ptr()) }; // Interpret void array as byte array if field.basetype == "void" { let mutable = if field.is_const { quote!(const) } else { quote!(mut) }; slice_param_ty_tokens = quote!([u8]); ptr = quote!(#ptr as *#mutable c_void); }; let mutable = if field.is_const { quote!() } else { quote!(mut) }; // Apply some heuristics to determine whether the size is an expression. // If so, this is a pointer to a piece of memory with statically known size. let set_size_stmt = if array_size.contains("ename:") || array_size.contains('*') { // c_size should contain the same minus `ename:`-prefixed identifiers let array_size = field.c_size.as_ref().unwrap_or(array_size); let c_size = convert_c_expression(array_size); let inner_type = field.inner_type_tokens(); slice_param_ty_tokens = quote!([#inner_type; #c_size]); quote!() } else { let array_size_ident = format_ident!("{}", array_size.to_snake_case().as_str()); quote!(self.inner.#array_size_ident = #param_ident_short.len() as _;) }; return Some(quote! { pub fn #param_ident_short(mut self, #param_ident_short: &'a #mutable #slice_param_ty_tokens) -> Self { #set_size_stmt self.inner.#param_ident = #param_ident_short#ptr; self } }); } } } } if field.basetype == "VkBool32" { return Some(quote!{ pub fn #param_ident_short(mut self, #param_ident_short: bool) -> Self { self.inner.#param_ident = #param_ident_short.into(); self } }); } let param_ty_tokens = if is_opaque_type(&field.basetype) { // Use raw pointers for void/opaque types field.type_tokens(false) } else { param_ty_tokens }; Some(quote!{ pub fn #param_ident_short(mut self, #param_ident_short: #param_ty_tokens) -> Self { self.inner.#param_ident = #param_ident_short; self } }) }); let extends_name = name_to_tokens(&format!("Extends{}", name)); let root_structs: Vec = _struct .extends .as_ref() .map(|extends| { extends .split(',') .filter(|extend| root_struct_names.contains(*extend)) .map(|extends| name_to_tokens(&format!("Extends{}", name_to_tokens(&extends)))) .collect() }) .unwrap_or_else(Vec::new); // We only implement a next methods for root structs with a `pnext` field. let next_function = if has_next && root_structs.is_empty() { quote! { /// Prepends the given extension struct between the root and the first pointer. This /// method only exists on structs that can be passed to a function directly. Only /// valid extension structs can be pushed into the chain. /// If the chain looks like `A -> B -> C`, and you call `builder.push_next(&mut D)`, then the /// chain will look like `A -> D -> B -> C`. pub fn push_next(mut self, next: &'a mut T) -> Self { unsafe{ let next_ptr = next as *mut T as *mut BaseOutStructure; // `next` here can contain a pointer chain. This means that we must correctly // attach he head to the root and the tail to the rest of the chain // For example: // // next = A -> B // Before: `Root -> C -> D -> E` // After: `Root -> A -> B -> C -> D -> E` // ^^^^^^ // next chain let last_next = ptr_chain_iter(next).last().unwrap(); (*last_next).p_next = self.inner.p_next as _; self.inner.p_next = next_ptr as _; } self } } } else { quote! {} }; // Root structs come with their own trait that structs that extends this struct will // implement let next_trait = if has_next && _struct.extends.is_none() { quote! { pub unsafe trait #extends_name { } } } else { quote! {} }; // If the struct extends something we need to implement the trait. let impl_extend_trait = root_structs.iter().map(|extends| { quote! { unsafe impl #extends for #name_builder<'_> { } unsafe impl #extends for #name { } } }); let q = quote! { impl #name { pub fn builder<'a>() -> #name_builder<'a> { #name_builder { inner: #name::default(), marker: ::std::marker::PhantomData, } } } #[repr(transparent)] pub struct #name_builder<'a> { inner: #name, marker: ::std::marker::PhantomData<&'a ()>, } #(#impl_extend_trait)* #next_trait impl<'a> ::std::ops::Deref for #name_builder<'a> { type Target = #name; fn deref(&self) -> &Self::Target { &self.inner } } impl<'a> ::std::ops::DerefMut for #name_builder<'a> { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.inner } } impl<'a> #name_builder<'a> { #(#setters)* #next_function /// Calling build will **discard** all the lifetime information. Only call this if /// necessary! Builders implement `Deref` targeting their corresponding Vulkan struct, /// so references to builders can be passed directly to Vulkan functions. pub fn build(self) -> #name { self.inner } } }; Some(q) } /// At the moment `Ash` doesn't properly derive all the necessary drives /// like Eq, Hash etc. /// To Address some cases, you can add the name of the struct that you /// require and add the missing derives yourself. pub fn manual_derives(_struct: &vkxml::Struct) -> TokenStream { match _struct.name.as_str() { "VkClearRect" | "VkExtent2D" | "VkExtent3D" | "VkOffset2D" | "VkOffset3D" | "VkRect2D" | "VkSurfaceFormatKHR" => quote! {PartialEq, Eq, Hash,}, _ => quote! {}, } } pub fn generate_struct( _struct: &vkxml::Struct, root_struct_names: &HashSet, union_types: &HashSet<&str, impl BuildHasher>, ) -> TokenStream { let name = name_to_tokens(&_struct.name); if &_struct.name == "VkTransformMatrixKHR" { return quote! { #[repr(C)] #[derive(Copy, Clone)] pub struct TransformMatrixKHR { pub matrix: [f32; 12], } }; } if &_struct.name == "VkAccelerationStructureInstanceKHR" { return quote! { #[repr(C)] #[derive(Copy, Clone)] pub union AccelerationStructureReferenceKHR { pub device_handle: DeviceAddress, pub host_handle: AccelerationStructureKHR, } #[repr(C)] #[derive(Copy, Clone)] pub struct AccelerationStructureInstanceKHR { pub transform: TransformMatrixKHR, pub instance_custom_index_and_mask: u32, pub instance_shader_binding_table_record_offset_and_flags: u32, pub acceleration_structure_reference: AccelerationStructureReferenceKHR, } }; } let members = _struct.elements.iter().filter_map(|elem| match *elem { vkxml::StructElement::Member(ref field) => Some(field), _ => None, }); let params = members.clone().map(|field| { let param_ident = field.param_ident(); let param_ty_tokens = field.type_tokens(false); quote! {pub #param_ident: #param_ty_tokens} }); let debug_tokens = derive_debug(_struct, union_types); let default_tokens = derive_default(_struct); let setter_tokens = derive_setters(_struct, root_struct_names); let manual_derive_tokens = manual_derives(_struct); let dbg_str = if debug_tokens.is_none() { quote!(Debug,) } else { quote!() }; let default_str = if default_tokens.is_none() { quote!(Default,) } else { quote!() }; let khronos_link = khronos_link(&_struct.name); quote! { #[repr(C)] #[derive(Copy, Clone, #default_str #dbg_str #manual_derive_tokens)] #[doc = #khronos_link] pub struct #name { #(#params,)* } #debug_tokens #default_tokens #setter_tokens } } pub fn generate_handle(handle: &vkxml::Handle) -> Option { if handle.name.is_empty() { return None; } let khronos_link = khronos_link(&handle.name); let tokens = match handle.ty { vkxml::HandleType::Dispatch => { let name = &handle.name[2..]; let ty = format_ident!("{}", name.to_shouty_snake_case()); let name = format_ident!("{}", name); quote! { define_handle!(#name, #ty, doc = #khronos_link); } } vkxml::HandleType::NoDispatch => { let name = &handle.name[2..]; let ty = format_ident!("{}", name.to_shouty_snake_case()); let name = format_ident!("{}", name); quote! { handle_nondispatchable!(#name, #ty, doc = #khronos_link); } } }; Some(tokens) } fn generate_funcptr(fnptr: &vkxml::FunctionPointer) -> TokenStream { let name = format_ident!("{}", fnptr.name.as_str()); let ret_ty_tokens = if fnptr.return_type.is_void() { quote!() } else { let ret_ty_tokens = fnptr.return_type.type_tokens(true); quote!(-> #ret_ty_tokens) }; let params = fnptr.param.iter().map(|field| { let ident = field.param_ident(); let type_tokens = field.type_tokens(true); quote! { #ident: #type_tokens } }); let khronos_link = khronos_link(&fnptr.name); quote! { #[allow(non_camel_case_types)] #[doc = #khronos_link] pub type #name = Option; } } fn generate_union(union: &vkxml::Union) -> TokenStream { let name = name_to_tokens(&union.name); let fields = union.elements.iter().map(|field| { let name = field.param_ident(); let ty = field.type_tokens(false); quote! { pub #name: #ty } }); let khronos_link = khronos_link(&union.name); quote! { #[repr(C)] #[derive(Copy, Clone)] #[doc = #khronos_link] pub union #name { #(#fields),* } impl ::std::default::Default for #name { fn default() -> #name { unsafe { ::std::mem::zeroed() } } } } } pub fn root_struct_names(definitions: &[&vkxml::DefinitionsElement]) -> HashSet { definitions .iter() .filter_map(|definition| match *definition { vkxml::DefinitionsElement::Struct(ref _struct) => { let is_root_struct = _struct.extends.is_none(); if is_root_struct { Some(_struct.name.clone()) } else { None } } _ => None, }) .collect() } pub fn generate_definition( definition: &vkxml::DefinitionsElement, union_types: &HashSet<&str, impl BuildHasher>, root_structs: &HashSet, bitflags_cache: &mut HashSet, const_values: &mut BTreeMap, ) -> Option { match *definition { vkxml::DefinitionsElement::Define(ref define) => Some(generate_define(define)), vkxml::DefinitionsElement::Typedef(ref typedef) => Some(generate_typedef(typedef)), vkxml::DefinitionsElement::Struct(ref _struct) => { Some(generate_struct(_struct, root_structs, union_types)) } vkxml::DefinitionsElement::Bitmask(ref mask) => { generate_bitmask(mask, bitflags_cache, const_values) } vkxml::DefinitionsElement::Handle(ref handle) => generate_handle(handle), vkxml::DefinitionsElement::FuncPtr(ref fp) => Some(generate_funcptr(fp)), vkxml::DefinitionsElement::Union(ref union) => Some(generate_union(union)), _ => None, } } pub fn generate_feature<'a>( feature: &vkxml::Feature, commands: &CommandMap<'a>, fn_cache: &mut HashSet<&'a str, impl BuildHasher>, ) -> TokenStream { let (static_commands, entry_commands, device_commands, instance_commands) = feature .elements .iter() .flat_map(|feature| { if let vkxml::FeatureElement::Require(ref spec) = feature { spec.elements .iter() .filter_map(|feature_spec| { if let vkxml::FeatureReference::CommandReference(ref cmd_ref) = feature_spec { Some(cmd_ref) } else { None } }) .collect() } else { vec![] } }) .filter_map(|cmd_ref| commands.get(&cmd_ref.name)) .fold( (Vec::new(), Vec::new(), Vec::new(), Vec::new()), |mut acc, &cmd_ref| { match cmd_ref.function_type() { FunctionType::Static => { acc.0.push(cmd_ref); } FunctionType::Entry => { acc.1.push(cmd_ref); } FunctionType::Device => { acc.2.push(cmd_ref); } FunctionType::Instance => { acc.3.push(cmd_ref); } } acc }, ); let version = feature.version_string(); let static_fn = if feature.is_version(1, 0) { generate_function_pointers( format_ident!("{}", "StaticFn"), &static_commands, &HashMap::new(), fn_cache, ) } else { quote! {} }; let entry = generate_function_pointers( format_ident!("{}", format!("EntryFnV{}", version).as_str()), &entry_commands, &HashMap::new(), fn_cache, ); let instance = generate_function_pointers( format_ident!("{}", format!("InstanceFnV{}", version).as_str()), &instance_commands, &HashMap::new(), fn_cache, ); let device = generate_function_pointers( format_ident!("{}", format!("DeviceFnV{}", version).as_str()), &device_commands, &HashMap::new(), fn_cache, ); quote! { #static_fn #entry #instance #device } } pub fn constant_name(name: &str) -> &str { name.strip_prefix("VK_").unwrap_or(name) } pub fn generate_constant<'a>( constant: &'a vkxml::Constant, cache: &mut HashSet<&'a str, impl BuildHasher>, ) -> TokenStream { cache.insert(constant.name.as_str()); let c = Constant::from_constant(constant); let name = constant_name(&constant.name); let ident = format_ident!("{}", name); let ty = if name == "TRUE" || name == "FALSE" { CType::Bool32 } else { c.ty() }; quote! { pub const #ident: #ty = #c; } } pub fn generate_feature_extension<'a>( registry: &'a vk_parse::Registry, const_cache: &mut HashSet<&'a str, impl BuildHasher>, const_values: &mut BTreeMap, ) -> TokenStream { let constants = registry.0.iter().filter_map(|item| match item { vk_parse::RegistryChild::Feature(feature) => Some(generate_extension_constants( &feature.name, 0, &feature.children, const_cache, const_values, )), _ => None, }); quote! { #(#constants)* } } pub struct ConstantMatchInfo { pub ident: Ident, pub is_alias: bool, } #[derive(Default)] pub struct ConstantTypeInfo { values: Vec, bitwidth: Option, } pub fn generate_const_debugs(const_values: &BTreeMap) -> TokenStream { let impls = const_values.iter().map(|(ty, values)| { let ConstantTypeInfo { values, bitwidth } = values; if ty.to_string().contains("Flags") { let cases = values.iter().filter_map(|value| { if value.is_alias { None } else { let ident = &value.ident; let name = ident.to_string(); Some(quote! { (#ty::#ident.0, #name) }) } }); let type_ = if bitwidth == &Some(64u32) { quote!(Flags64) } else { quote!(Flags) }; quote! { impl fmt::Debug for #ty { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { const KNOWN: &[(#type_, &str)] = &[#(#cases),*]; debug_flags(f, KNOWN, self.0) } } } } else { let cases = values.iter().filter_map(|value| { if value.is_alias { None } else { let ident = &value.ident; let name = ident.to_string(); Some(quote! { Self::#ident => Some(#name), }) } }); quote! { impl fmt::Debug for #ty { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { let name = match *self { #(#cases)* _ => None, }; if let Some(x) = name { f.write_str(x) } else { self.0.fmt(f) } } } } } }); quote! { pub(crate) fn debug_flags + Copy>( f: &mut fmt::Formatter, known: &[(Value, &'static str)], value: Value, ) -> fmt::Result { let mut first = true; let mut accum = value.into(); for &(bit, name) in known { let bit = bit.into(); if bit != 0 && accum & bit == bit { if !first { f.write_str(" | ")?; } f.write_str(name)?; first = false; accum &= !bit; } } if accum != 0 { if !first { f.write_str(" | ")?; } write!(f, "{:b}", accum)?; } Ok(()) } #(#impls)* } } pub fn generate_aliases_of_types( types: &vk_parse::Types, ty_cache: &mut HashSet, ) -> TokenStream { let aliases = types .children .iter() .filter_map(|child| match child { vk_parse::TypesChild::Type(ty) => Some((ty.name.as_ref()?, ty.alias.as_ref()?)), _ => None, }) .filter_map(|(name, alias)| { let name_ident = name_to_tokens(name); if ty_cache.contains(&name_ident) { return None; }; ty_cache.insert(name_ident.clone()); let alias_ident = name_to_tokens(alias); let tokens = quote! { pub type #name_ident = #alias_ident; }; Some(tokens) }); quote! { #(#aliases)* } } pub fn write_source_code>(vk_xml: &Path, src_dir: P) { use std::fs::File; use std::io::Write; let (spec2, _errors) = vk_parse::parse_file(vk_xml).expect("Invalid xml file"); let extensions: &Vec = spec2 .0 .iter() .filter_map(|item| match item { vk_parse::RegistryChild::Extensions(ref ext) => Some(&ext.children), _ => None, }) .next() .expect("extension"); let mut ty_cache = HashSet::new(); let aliases: Vec<_> = spec2 .0 .iter() .filter_map(|item| match item { vk_parse::RegistryChild::Types(ref ty) => { Some(generate_aliases_of_types(ty, &mut ty_cache)) } _ => None, }) .collect(); let spec = vk_parse::parse_file_as_vkxml(vk_xml).expect("Invalid xml file."); let cmd_aliases: HashMap = spec2 .0 .iter() .filter_map(|item| match item { vk_parse::RegistryChild::Commands(cmds) => { let cmd_tuple_iter = cmds.children.iter().filter_map(|cmd| match cmd { vk_parse::Command::Alias { name, alias } => { Some((name.to_string(), alias.to_string())) } _ => None, }); Some(cmd_tuple_iter) } _ => None, }) .flatten() .collect(); let commands: HashMap = spec .elements .iter() .filter_map(|elem| match elem { vkxml::RegistryElement::Commands(ref cmds) => Some(cmds), _ => None, }) .flat_map(|cmds| cmds.elements.iter().map(|cmd| (cmd.name.clone(), cmd))) .collect(); let features: Vec<&vkxml::Feature> = spec .elements .iter() .filter_map(|elem| match elem { vkxml::RegistryElement::Features(ref features) => Some(features), _ => None, }) .flat_map(|features| features.elements.iter()) .collect(); let definitions: Vec<&vkxml::DefinitionsElement> = spec .elements .iter() .filter_map(|elem| match elem { vkxml::RegistryElement::Definitions(ref definitions) => Some(definitions), _ => None, }) .flat_map(|definitions| definitions.elements.iter()) .collect(); let constants: Vec<&vkxml::Constant> = spec .elements .iter() .filter_map(|elem| match elem { vkxml::RegistryElement::Constants(ref constants) => Some(constants), _ => None, }) .flat_map(|constants| constants.elements.iter()) .collect(); let mut fn_cache = HashSet::new(); let mut bitflags_cache = HashSet::new(); let mut const_cache = HashSet::new(); let mut const_values: BTreeMap = BTreeMap::new(); let (enum_code, bitflags_code) = spec2 .0 .iter() .filter_map(|item| match item { vk_parse::RegistryChild::Enums(ref enums) if enums.kind.is_some() => Some(enums), _ => None, }) .map(|e| generate_enum(e, &mut const_cache, &mut const_values, &mut bitflags_cache)) .fold((Vec::new(), Vec::new()), |mut acc, elem| { match elem { EnumType::Enum(token) => acc.0.push(token), EnumType::Bitflags(token) => acc.1.push(token), }; acc }); let mut constants_code: Vec<_> = constants .iter() .map(|constant| generate_constant(constant, &mut const_cache)) .collect(); constants_code.push(quote! { pub const SHADER_UNUSED_NV : u32 = SHADER_UNUSED_KHR;}); let extension_code = extensions .iter() .filter_map(|ext| { generate_extension( ext, &commands, &mut const_cache, &mut const_values, &cmd_aliases, &mut fn_cache, ) }) .collect_vec(); let union_types = definitions .iter() .filter_map(|def| match def { vkxml::DefinitionsElement::Union(ref union) => Some(union.name.as_str()), _ => None, }) .collect::>(); let root_names = root_struct_names(&definitions); let definition_code: Vec<_> = definitions .into_iter() .filter_map(|def| { generate_definition( def, &union_types, &root_names, &mut bitflags_cache, &mut const_values, ) }) .collect(); let feature_code: Vec<_> = features .iter() .map(|feature| generate_feature(feature, &commands, &mut fn_cache)) .collect(); let feature_extensions_code = generate_feature_extension(&spec2, &mut const_cache, &mut const_values); let const_debugs = generate_const_debugs(&const_values); let bitflags_macro = vk_bitflags_wrapped_macro(); let handle_nondispatchable_macro = handle_nondispatchable_macro(); let define_handle_macro = define_handle_macro(); let version_macros = vk_version_macros(); let ptr_chain_code = quote! { /// Iterates through the pointer chain. Includes the item that is passed into the function. /// Stops at the last `BaseOutStructure` that has a null `p_next` field. pub(crate) unsafe fn ptr_chain_iter( ptr: &mut T, ) -> impl Iterator { let ptr: *mut BaseOutStructure = ptr as *mut T as _; (0..).scan(ptr, |p_ptr, _| { if p_ptr.is_null() { return None; } let n_ptr = (**p_ptr).p_next as *mut BaseOutStructure; let old = *p_ptr; *p_ptr = n_ptr; Some(old) }) } }; let macros_code = quote! { #version_macros #bitflags_macro #handle_nondispatchable_macro #define_handle_macro }; let src_dir = src_dir.as_ref(); let vk_dir = src_dir.join("vk"); std::fs::create_dir_all(&vk_dir).expect("failed to create vk dir"); let mut vk_rs_file = File::create(src_dir.join("vk.rs")).expect("vk.rs"); let mut vk_macros_file = File::create(vk_dir.join("macros.rs")).expect("vk/macros.rs"); let mut vk_features_file = File::create(vk_dir.join("features.rs")).expect("vk/features.rs"); let mut vk_definitions_file = File::create(vk_dir.join("definitions.rs")).expect("vk/definitions.rs"); let mut vk_enums_file = File::create(vk_dir.join("enums.rs")).expect("vk/enums.rs"); let mut vk_bitflags_file = File::create(vk_dir.join("bitflags.rs")).expect("vk/bitflags.rs"); let mut vk_constants_file = File::create(vk_dir.join("constants.rs")).expect("vk/constants.rs"); let mut vk_extensions_file = File::create(vk_dir.join("extensions.rs")).expect("vk/extensions.rs"); let mut vk_feature_extensions_file = File::create(vk_dir.join("feature_extensions.rs")).expect("vk/feature_extensions.rs"); let mut vk_const_debugs_file = File::create(vk_dir.join("const_debugs.rs")).expect("vk/const_debugs.rs"); let mut vk_aliases_file = File::create(vk_dir.join("aliases.rs")).expect("vk/aliases.rs"); let feature_code = quote! { use std::os::raw::*; use crate::vk::bitflags::*; use crate::vk::definitions::*; use crate::vk::enums::*; #(#feature_code)* }; let definition_code = quote! { use std::fmt; use std::os::raw::*; use crate::vk::{Handle, ptr_chain_iter}; use crate::vk::platform_types::*; use crate::vk::aliases::*; use crate::vk::bitflags::*; use crate::vk::constants::*; use crate::vk::enums::*; #(#definition_code)* }; let enum_code = quote! { use std::fmt; #(#enum_code)* }; let bitflags_code = quote! { use crate::vk::definitions::*; #(#bitflags_code)* }; let constants_code = quote! { use crate::vk::definitions::*; #(#constants_code)* }; let extension_code = quote! { use std::os::raw::*; use crate::vk::platform_types::*; use crate::vk::aliases::*; use crate::vk::bitflags::*; use crate::vk::definitions::*; use crate::vk::enums::*; #(#extension_code)* }; let feature_extensions_code = quote! { use crate::vk::bitflags::*; use crate::vk::enums::*; #feature_extensions_code }; let const_debugs = quote! { use std::fmt; use crate::vk::bitflags::*; use crate::vk::definitions::*; use crate::vk::enums::*; #const_debugs }; let aliases = quote! { use crate::vk::bitflags::*; use crate::vk::definitions::*; use crate::vk::enums::*; #(#aliases)* }; // These are defined outside of `quote!` because rustfmt doesn't seem // to format them correctly when they contain extra spaces. let vk_rs_clippy_lints = r#" #![allow(clippy::too_many_arguments, clippy::cognitive_complexity, clippy::wrong_self_convention)] "#; let vk_rs_code = quote! { #[macro_use] mod macros; pub use macros::*; mod aliases; pub use aliases::*; mod bitflags; pub use bitflags::*; mod const_debugs; pub(crate) use const_debugs::*; mod constants; pub use constants::*; mod definitions; pub use definitions::*; mod enums; pub use enums::*; mod extensions; pub use extensions::*; mod feature_extensions; pub use feature_extensions::*; mod features; pub use features::*; mod platform_types; pub use platform_types::*; #ptr_chain_code pub trait Handle { const TYPE: ObjectType; fn as_raw(self) -> u64; fn from_raw(_: u64) -> Self; } }; write!(&mut vk_macros_file, "{}", macros_code).expect("Unable to write vk/macros.rs"); write!(&mut vk_features_file, "{}", feature_code).expect("Unable to write vk/features.rs"); write!(&mut vk_definitions_file, "{}", definition_code) .expect("Unable to write vk/definitions.rs"); write!(&mut vk_enums_file, "{}", enum_code).expect("Unable to write vk/enums.rs"); write!(&mut vk_bitflags_file, "{}", bitflags_code).expect("Unable to write vk/bitflags.rs"); write!(&mut vk_constants_file, "{}", constants_code).expect("Unable to write vk/constants.rs"); write!(&mut vk_extensions_file, "{}", extension_code) .expect("Unable to write vk/extensions.rs"); write!( &mut vk_feature_extensions_file, "{}", feature_extensions_code ) .expect("Unable to write vk/feature_extensions.rs"); write!(&mut vk_const_debugs_file, "{}", const_debugs) .expect("Unable to write vk/const_debugs.rs"); write!(&mut vk_aliases_file, "{}", aliases).expect("Unable to write vk/aliases.rs"); write!(&mut vk_rs_file, "{} {}", vk_rs_clippy_lints, vk_rs_code) .expect("Unable to write vk.rs"); }