Successfully replace the old vk.rs file

This commit is contained in:
Maik Klein 2018-07-07 12:54:31 +02:00
parent ad24467c95
commit e6e8bbd91b
7 changed files with 13738 additions and 5079 deletions

View file

@ -1,9 +1,9 @@
#![allow(dead_code)] #![allow(dead_code)]
use prelude::*; use prelude::*;
use std::mem; use std::mem;
use version::{FunctionPointers, V1_0};
use vk; use vk;
use RawPtr; use RawPtr;
use version::{FunctionPointers, V1_0};
#[allow(non_camel_case_types)] #[allow(non_camel_case_types)]
pub trait DeviceV1_0 { pub trait DeviceV1_0 {
@ -463,7 +463,8 @@ pub trait DeviceV1_0 {
pool: vk::DescriptorPool, pool: vk::DescriptorPool,
flags: vk::DescriptorPoolResetFlags, flags: vk::DescriptorPoolResetFlags,
) -> VkResult<()> { ) -> VkResult<()> {
let err_code = self.fp_v1_0() let err_code = self
.fp_v1_0()
.reset_descriptor_pool(self.handle(), pool, flags); .reset_descriptor_pool(self.handle(), pool, flags);
match err_code { match err_code {
vk::Result::Success => Ok(()), vk::Result::Success => Ok(()),
@ -476,7 +477,8 @@ pub trait DeviceV1_0 {
command_pool: vk::CommandPool, command_pool: vk::CommandPool,
flags: vk::CommandPoolResetFlags, flags: vk::CommandPoolResetFlags,
) -> VkResult<()> { ) -> VkResult<()> {
let err_code = self.fp_v1_0() let err_code = self
.fp_v1_0()
.reset_command_pool(self.handle(), command_pool, flags); .reset_command_pool(self.handle(), command_pool, flags);
match err_code { match err_code {
vk::Result::Success => Ok(()), vk::Result::Success => Ok(()),
@ -636,7 +638,7 @@ pub trait DeviceV1_0 {
descriptor_sets.as_ptr(), descriptor_sets.as_ptr(),
dynamic_offsets.len() as vk::uint32_t, dynamic_offsets.len() as vk::uint32_t,
dynamic_offsets.as_ptr(), dynamic_offsets.as_ptr(),
) );
} }
unsafe fn cmd_push_constants( unsafe fn cmd_push_constants(
@ -699,15 +701,9 @@ pub trait DeviceV1_0 {
); );
} }
unsafe fn cmd_set_line_width( unsafe fn cmd_set_line_width(&self, command_buffer: vk::CommandBuffer, line_width: f32) {
&self, self.fp_v1_0()
command_buffer: vk::CommandBuffer, .cmd_set_line_width(command_buffer, line_width);
line_width: f32,
) {
self.fp_v1_0().cmd_set_line_width(
command_buffer,
line_width,
);
} }
unsafe fn cmd_bind_vertex_buffers( unsafe fn cmd_bind_vertex_buffers(
@ -802,12 +798,8 @@ pub trait DeviceV1_0 {
clamp: f32, clamp: f32,
slope_factor: f32, slope_factor: f32,
) { ) {
self.fp_v1_0().cmd_set_depth_bias( self.fp_v1_0()
command_buffer, .cmd_set_depth_bias(command_buffer, constant_factor, clamp, slope_factor);
constant_factor,
clamp,
slope_factor,
);
} }
unsafe fn cmd_set_blend_constants( unsafe fn cmd_set_blend_constants(
@ -816,7 +808,7 @@ pub trait DeviceV1_0 {
blend_constants: [f32; 4], blend_constants: [f32; 4],
) { ) {
self.fp_v1_0() self.fp_v1_0()
.cmd_set_blend_constants(command_buffer, &blend_constants); .cmd_set_blend_constants(command_buffer, blend_constants);
} }
unsafe fn cmd_set_depth_bounds( unsafe fn cmd_set_depth_bounds(
@ -1178,7 +1170,8 @@ pub trait DeviceV1_0 {
command_buffer: vk::CommandBuffer, command_buffer: vk::CommandBuffer,
create_info: &vk::CommandBufferBeginInfo, create_info: &vk::CommandBufferBeginInfo,
) -> VkResult<()> { ) -> VkResult<()> {
let err_code = self.fp_v1_0() let err_code = self
.fp_v1_0()
.begin_command_buffer(command_buffer, create_info); .begin_command_buffer(command_buffer, create_info);
match err_code { match err_code {
vk::Result::Success => Ok(()), vk::Result::Success => Ok(()),
@ -1373,8 +1366,12 @@ pub trait DeviceV1_0 {
) -> vk::SubresourceLayout { ) -> vk::SubresourceLayout {
unsafe { unsafe {
let mut layout = mem::uninitialized(); let mut layout = mem::uninitialized();
self.fp_v1_0() self.fp_v1_0().get_image_subresource_layout(
.get_image_subresource_layout(self.handle(), image, &subresource, &mut layout); self.handle(),
image,
&subresource,
&mut layout,
);
layout layout
} }
} }

View file

@ -1,14 +1,14 @@
use prelude::*;
use std::mem;
use std::ptr;
use vk;
use instance::Instance; use instance::Instance;
use prelude::*;
use shared_library::dynamic_library::DynamicLibrary; use shared_library::dynamic_library::DynamicLibrary;
use std::error::Error; use std::error::Error;
use std::fmt; use std::fmt;
use std::mem;
use std::path::Path; use std::path::Path;
use RawPtr; use std::ptr;
use version::{EntryLoader, FunctionPointers, InstanceLoader, V1_0}; use version::{EntryLoader, FunctionPointers, InstanceLoader, V1_0};
use vk;
use RawPtr;
#[cfg(windows)] #[cfg(windows)]
const LIB_PATH: &'static str = "vulkan-1.dll"; const LIB_PATH: &'static str = "vulkan-1.dll";
@ -22,8 +22,9 @@ const LIB_PATH: &'static str = "libvulkan.so";
#[cfg(any(target_os = "macos", target_os = "ios"))] #[cfg(any(target_os = "macos", target_os = "ios"))]
const LIB_PATH: &'static str = "libMoltenVK.dylib"; const LIB_PATH: &'static str = "libMoltenVK.dylib";
lazy_static!{ lazy_static! {
static ref VK_LIB: Result<DynamicLibrary, String> = DynamicLibrary::open(Some(&Path::new(LIB_PATH))); static ref VK_LIB: Result<DynamicLibrary, String> =
DynamicLibrary::open(Some(&Path::new(LIB_PATH)));
} }
#[derive(Clone)] #[derive(Clone)]
@ -97,7 +98,8 @@ pub trait EntryV1_0 {
.enumerate_instance_layer_properties(&mut num, ptr::null_mut()); .enumerate_instance_layer_properties(&mut num, ptr::null_mut());
let mut v = Vec::with_capacity(num as usize); let mut v = Vec::with_capacity(num as usize);
let err_code = self.fp_v1_0() let err_code = self
.fp_v1_0()
.enumerate_instance_layer_properties(&mut num, v.as_mut_ptr()); .enumerate_instance_layer_properties(&mut num, v.as_mut_ptr());
v.set_len(num as usize); v.set_len(num as usize);
match err_code { match err_code {

View file

@ -1,14 +1,14 @@
#![allow(dead_code)] #![allow(dead_code)]
use prelude::*;
use std::ptr;
use std::mem;
use vk;
use device::Device; use device::Device;
use prelude::*;
use std::error::Error; use std::error::Error;
use std::fmt; use std::fmt;
use RawPtr; use std::mem;
use version::{FunctionPointers, V1_0}; use std::ptr;
use version::DeviceLoader; use version::DeviceLoader;
use version::{FunctionPointers, V1_0};
use vk;
use RawPtr;
#[derive(Debug)] #[derive(Debug)]
pub enum DeviceError { pub enum DeviceError {

View file

@ -2,19 +2,18 @@
extern crate lazy_static; extern crate lazy_static;
extern crate libc; extern crate libc;
extern crate shared_library; extern crate shared_library;
pub use instance::{DeviceError, Instance};
pub use device::Device; pub use device::Device;
pub use entry::{Entry, InstanceError, LoadingError}; pub use entry::{Entry, InstanceError, LoadingError};
pub use instance::{DeviceError, Instance};
mod instance;
mod device; mod device;
mod entry; mod entry;
pub mod prelude;
pub mod vk;
pub mod extensions; pub mod extensions;
pub mod version; mod instance;
pub mod prelude;
pub mod util; pub mod util;
mod vk_test; pub mod version;
pub mod vk;
pub trait RawPtr<T> { pub trait RawPtr<T> {
fn as_raw_ptr(&self) -> *const T; fn as_raw_ptr(&self) -> *const T;
} }

View file

@ -1,8 +1,8 @@
use vk;
pub use instance::InstanceV1_0;
pub use device::DeviceV1_0; pub use device::DeviceV1_0;
pub use entry::EntryV1_0; pub use entry::EntryV1_0;
pub use instance::InstanceV1_0;
use std::mem; use std::mem;
use vk;
pub trait FunctionPointers { pub trait FunctionPointers {
type InstanceFp: InstanceLoader + Clone; type InstanceFp: InstanceLoader + Clone;
type DeviceFp: DeviceLoader + Clone; type DeviceFp: DeviceLoader + Clone;

File diff suppressed because it is too large Load diff

View file

@ -282,6 +282,41 @@ pub fn vk_bitflags_wrapped_macro() -> Tokens {
} }
} }
} }
pub fn platform_specific_types() -> Tokens {
quote! {
pub type RROutput = c_ulong;
pub type VisualID = c_uint;
pub type Display = *const c_void;
pub type Window = c_ulong;
#[allow(non_camel_case_types)]
pub type xcb_connection_t = *const c_void;
#[allow(non_camel_case_types)]
pub type xcb_window_t = u32;
#[allow(non_camel_case_types)]
pub type xcb_visualid_t = *const c_void;
pub type MirConnection = *const c_void;
pub type MirSurface = *const c_void;
pub type HINSTANCE = *const c_void;
pub type HWND = *const c_void;
#[allow(non_camel_case_types)]
pub type wl_display = *const c_void;
#[allow(non_camel_case_types)]
pub type wl_surface = *const c_void;
pub type HANDLE = *mut c_void;
pub type DWORD = c_ulong;
pub type WCHAR = wchar_t;
pub type LPCWSTR = *const WCHAR;
// FIXME: Platform specific types that should come from a library
// typedefs are only here so that the code compiles for now
#[allow(non_camel_case_types)]
pub type SECURITY_ATTRIBUTES = ();
// Opage types
pub type ANativeWindow = c_void;
pub type AHardwareBuffer = c_void;
}
}
#[derive(Debug, Copy, Clone)] #[derive(Debug, Copy, Clone)]
pub enum ConstVal { pub enum ConstVal {
U32(u32), U32(u32),
@ -392,16 +427,23 @@ pub trait FeatureExt {
} }
impl FeatureExt for vkxml::Feature { impl FeatureExt for vkxml::Feature {
fn version_string(&self) -> String { fn version_string(&self) -> String {
let version = format!("{:.10}", self.version); let mut version = format!("{}", self.version);
let first_0 = version.find("0").expect("should have at least one 0"); if version.len() == 1 {
// + 1 is correct here because we always have 10 zeroes. version = format!("{}_0", version)
let (version, _) = version.split_at(first_0 + 1); }
version.replace(".", "_") version.replace(".", "_")
} }
} }
#[derive(Debug, Copy, Clone)]
pub enum FunctionType {
Static,
Entry,
Instance,
Device,
}
pub trait CommandExt { pub trait CommandExt {
/// Returns the ident in snake_case and without the 'vk' prefix. /// Returns the ident in snake_case and without the 'vk' prefix.
fn is_device_command(&self) -> bool; fn function_type(&self) -> FunctionType;
/// ///
/// Returns true if the command is a device level command. This is indicated by /// Returns true if the command is a device level command. This is indicated by
/// the type of the first parameter. /// the type of the first parameter.
@ -413,15 +455,31 @@ impl CommandExt for vkxml::Command {
Ident::from(self.name[2..].to_snake_case().as_str()) Ident::from(self.name[2..].to_snake_case().as_str())
} }
fn is_device_command(&self) -> bool { fn function_type(&self) -> FunctionType {
self.param let is_first_param_device = self
.param
.iter() .iter()
.nth(0) .nth(0)
.map(|field| match field.basetype.as_str() { .map(|field| match field.basetype.as_str() {
"VkDevice" | "VkCommandBuffer" | "VkQueue" => true, "VkDevice" | "VkCommandBuffer" | "VkQueue" => true,
_ => false, _ => false,
}) })
.unwrap_or(false) .unwrap_or(false);
match self.name.as_str() {
"vkGetInstanceProcAddr" => FunctionType::Static,
"vkCreateInstance"
| "vkEnumerateInstanceLayerProperties"
| "vkEnumerateInstanceExtensionProperties" => FunctionType::Entry,
// This is actually not a device level function
"vkGetDeviceProcAddr" => FunctionType::Instance,
_ => {
if is_first_param_device {
FunctionType::Device
} else {
FunctionType::Instance
}
}
}
} }
} }
@ -441,8 +499,8 @@ pub trait ToTokens {
impl ToTokens for vkxml::ReferenceType { impl ToTokens for vkxml::ReferenceType {
fn to_tokens(&self) -> Tokens { fn to_tokens(&self) -> Tokens {
let ptr_name = match self { let ptr_name = match self {
vkxml::ReferenceType::Pointer => "*mut", vkxml::ReferenceType::Pointer => "*const",
vkxml::ReferenceType::PointerToPointer => "*mut", vkxml::ReferenceType::PointerToPointer => "*mut *mut",
vkxml::ReferenceType::PointerToConstPointer => "*const", vkxml::ReferenceType::PointerToConstPointer => "*const",
}; };
let ident = Term::intern(ptr_name); let ident = Term::intern(ptr_name);
@ -453,9 +511,6 @@ impl ToTokens for vkxml::ReferenceType {
} }
fn name_to_tokens(type_name: &str) -> Tokens { fn name_to_tokens(type_name: &str) -> Tokens {
let new_name = match type_name { let new_name = match type_name {
"HANDLE" => "*const c_void",
"LPCWSTR" => "*const wchar_t",
"DWORD" => "c_uint",
"int" => "c_int", "int" => "c_int",
"void" => "c_void", "void" => "c_void",
"char" => "c_char", "char" => "c_char",
@ -478,7 +533,6 @@ fn name_to_tokens(type_name: &str) -> Tokens {
fn to_type_tokens(type_name: &str, reference: Option<&vkxml::ReferenceType>) -> Tokens { fn to_type_tokens(type_name: &str, reference: Option<&vkxml::ReferenceType>) -> Tokens {
let new_name = name_to_tokens(type_name); let new_name = name_to_tokens(type_name);
let ptr_name = reference.map(|r| r.to_tokens()).unwrap_or(quote!{}); let ptr_name = reference.map(|r| r.to_tokens()).unwrap_or(quote!{});
// let ty: syn::Type = syn::parse_str(&format!("{} {}", ptr_name, new_name)).expect("parse field");
quote!{#ptr_name #new_name} quote!{#ptr_name #new_name}
} }
@ -497,7 +551,8 @@ impl FieldExt for vkxml::Field {
fn type_tokens(&self) -> Tokens { fn type_tokens(&self) -> Tokens {
let ty = name_to_tokens(&self.basetype); let ty = name_to_tokens(&self.basetype);
let pointer = self.reference let pointer = self
.reference
.as_ref() .as_ref()
.map(|r| r.to_tokens()) .map(|r| r.to_tokens())
.unwrap_or(quote!{}); .unwrap_or(quote!{});
@ -506,7 +561,8 @@ impl FieldExt for vkxml::Field {
}; };
let array = self.array.as_ref().and_then(|arraytype| match arraytype { let array = self.array.as_ref().and_then(|arraytype| match arraytype {
vkxml::ArrayType::Static => { vkxml::ArrayType::Static => {
let size = self.size let size = self
.size
.as_ref() .as_ref()
.or(self.size_enumref.as_ref()) .or(self.size_enumref.as_ref())
.expect("Should have size"); .expect("Should have size");
@ -539,7 +595,8 @@ fn generate_function_pointers(ident: Ident, commands: &[&vkxml::Command]) -> quo
.map(|cmd| { .map(|cmd| {
let fn_name_raw = cmd.name.as_str(); let fn_name_raw = cmd.name.as_str();
let fn_name_snake = cmd.command_ident(); let fn_name_snake = cmd.command_ident();
let params: Vec<_> = cmd.param let params: Vec<_> = cmd
.param
.iter() .iter()
.map(|field| { .map(|field| {
let name = field.param_ident(); let name = field.param_ident();
@ -598,36 +655,34 @@ fn generate_function_pointers(ident: Ident, commands: &[&vkxml::Command]) -> quo
} }
} }
impl #ident { impl #ident {
pub fn load<F>(mut f: F) -> ::std::result::Result<Self, Vec<&'static str>> pub fn load<F>(mut _f: F) -> ::std::result::Result<Self, Vec<&'static str>>
where F: FnMut(&::std::ffi::CStr) -> *const c_void where F: FnMut(&::std::ffi::CStr) -> *const c_void
{ {
use std::ffi::CString; let mut _err_str = Vec::new();
use std::mem;
let mut err_str = Vec::new();
let s = #ident { let s = #ident {
#( #(
#names_ref: unsafe { #names_ref: unsafe {
let raw_name = stringify!(#raw_names_ref); let raw_name = stringify!(#raw_names_ref);
let cname = CString::new(raw_name).unwrap(); let cname = ::std::ffi::CString::new(raw_name).unwrap();
let val = f(&cname); let val = _f(&cname);
if val.is_null(){ if val.is_null(){
err_str.push(raw_name); _err_str.push(raw_name);
} }
mem::transmute(val) ::std::mem::transmute(val)
}, },
)* )*
}; };
if err_str.is_empty() { if _err_str.is_empty() {
Ok(s) Ok(s)
} }
else{ else{
Err(err_str) Err(_err_str)
} }
} }
#( #(
pub fn #names_ref(&self, #expanded_params_ref) -> #return_types_ref { pub unsafe fn #names_ref(&self, #expanded_params_ref) -> #return_types_ref {
(self.#names_left)(#(#param_names_ref,)*) (self.#names_left)(#(#param_names_ref,)*)
} }
)* )*
@ -702,9 +757,11 @@ pub fn to_variant_ident(enum_name: &str, variant_name: &str) -> Ident {
}) })
.nth(0); .nth(0);
let name_without_tag = tag.map(|t| enum_name.replace(t, "")) let name_without_tag = tag
.map(|t| enum_name.replace(t, ""))
.unwrap_or(enum_name.into()); .unwrap_or(enum_name.into());
let variant_without_tag = tag.map(|t| variant_name.replace(t, "")) let variant_without_tag = tag
.map(|t| variant_name.replace(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, "");
@ -753,7 +810,6 @@ pub fn generate_enum(_enum: &vkxml::Enumeration) -> EnumType {
return None; return None;
} }
}; };
let variant_ident = Ident::from(variant_name); let variant_ident = Ident::from(variant_name);
Some(quote!{ Some(quote!{
pub const #variant_ident: #ident = #ident { flags: #value }; pub const #variant_ident: #ident = #ident { flags: #value };
@ -857,21 +913,38 @@ pub fn generate_result(name: &str, _enum: &vkxml::Enumeration) -> Tokens {
pub trait StructExt {} pub trait StructExt {}
pub fn generate_struct(_struct: &vkxml::Struct) -> Tokens { pub fn generate_struct(_struct: &vkxml::Struct) -> Tokens {
let name = to_type_tokens(&_struct.name, None); let name = to_type_tokens(&_struct.name, None);
let params = _struct let members = _struct.elements.iter().filter_map(|elem| match *elem {
.elements vkxml::StructElement::Member(ref field) => Some(field),
.iter() _ => None,
.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();
.map(|field| { let param_ty_tokens = field.type_tokens();
let param_ident = field.param_ident(); quote!{pub #param_ident: #param_ty_tokens}
let param_ty_tokens = field.type_tokens(); });
quote!{pub #param_ident: #param_ty_tokens}
}); let contains_pfn = members.clone().any(|field| {
field
.name
.as_ref()
.map(|n| n.contains("pfn"))
.unwrap_or(false)
});
let derive = if contains_pfn {
quote!{
#[derive(Copy, Clone)]
}
} else {
// FIXME: Properly derive Debug
quote!{
#[derive(Copy, Clone)]
}
};
quote!{ quote!{
#[derive(Copy, Clone)]
#[repr(C)] #[repr(C)]
#derive
pub struct #name { pub struct #name {
#(#params,)* #(#params,)*
} }
@ -901,10 +974,10 @@ pub fn generate_handle(handle: &vkxml::Handle) -> Option<Tokens> {
Some(tokens) Some(tokens)
} }
fn generate_funcptr(fnptr: &vkxml::FunctionPointer) -> Tokens { fn generate_funcptr(fnptr: &vkxml::FunctionPointer) -> Tokens {
println!("{:#?}", fnptr);
let name = Ident::from(fnptr.name.as_str()); let name = Ident::from(fnptr.name.as_str());
let ret_ty_tokens = fnptr.return_type.type_tokens(); let ret_ty_tokens = fnptr.return_type.type_tokens();
quote!{ quote!{
#[allow(non_camel_case_types)]
pub type #name = unsafe extern "system" fn() -> #ret_ty_tokens; pub type #name = unsafe extern "system" fn() -> #ret_ty_tokens;
} }
} }
@ -937,8 +1010,8 @@ pub fn generate_definition(definition: &vkxml::DefinitionsElement) -> Option<Tok
_ => None, _ => None,
} }
} }
pub fn generate_core_spec(feature: &vkxml::Feature, commands: &CommandMap) -> quote::Tokens { pub fn generate_feature(feature: &vkxml::Feature, commands: &CommandMap) -> quote::Tokens {
let (device_commands, instance_commands) = feature let (static_commands, entry_commands, device_commands, instance_commands) = feature
.elements .elements
.iter() .iter()
.flat_map(|feature| { .flat_map(|feature| {
@ -960,16 +1033,36 @@ pub fn generate_core_spec(feature: &vkxml::Feature, commands: &CommandMap) -> qu
} }
}) })
.filter_map(|cmd_ref| commands.get(&cmd_ref.name)) .filter_map(|cmd_ref| commands.get(&cmd_ref.name))
.fold((Vec::new(), Vec::new()), |mut acc, &cmd_ref| { .fold(
if cmd_ref.is_device_command() { (Vec::new(), Vec::new(), Vec::new(), Vec::new()),
acc.0.push(cmd_ref); |mut acc, &cmd_ref| {
} else { match cmd_ref.function_type() {
acc.1.push(cmd_ref); FunctionType::Static => {
}; acc.0.push(cmd_ref);
acc }
}); FunctionType::Entry => {
let name = Ident::from("Test"); 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 version = feature.version_string();
let static_fn = if feature.version == 1.0 {
generate_function_pointers(Ident::from("StaticFn"), &static_commands)
} else {
quote!{}
};
let entry = generate_function_pointers(
Ident::from(format!("EntryFnV{}", version).as_str()),
&entry_commands,
);
let instance = generate_function_pointers( let instance = generate_function_pointers(
Ident::from(format!("InstanceFnV{}", version).as_str()), Ident::from(format!("InstanceFnV{}", version).as_str()),
&instance_commands, &instance_commands,
@ -979,6 +1072,8 @@ pub fn generate_core_spec(feature: &vkxml::Feature, commands: &CommandMap) -> qu
&device_commands, &device_commands,
); );
quote! { quote! {
#static_fn
#entry
#instance #instance
#device #device
} }
@ -997,7 +1092,8 @@ pub fn write_source_code(spec: &vkxml::Registry) {
use std::fs::File; use std::fs::File;
use std::io::Write; use std::io::Write;
println!("{:#?}", spec); println!("{:#?}", spec);
let commands: HashMap<vkxml::Identifier, &vkxml::Command> = spec.elements let commands: HashMap<vkxml::Identifier, &vkxml::Command> = spec
.elements
.iter() .iter()
.filter_map(|elem| match elem { .filter_map(|elem| match elem {
&vkxml::RegistryElement::Commands(ref cmds) => Some(cmds), &vkxml::RegistryElement::Commands(ref cmds) => Some(cmds),
@ -1006,7 +1102,8 @@ pub fn write_source_code(spec: &vkxml::Registry) {
.flat_map(|cmds| cmds.elements.iter().map(|cmd| (cmd.name.clone(), cmd))) .flat_map(|cmds| cmds.elements.iter().map(|cmd| (cmd.name.clone(), cmd)))
.collect(); .collect();
let features: Vec<&vkxml::Feature> = spec.elements let features: Vec<&vkxml::Feature> = spec
.elements
.iter() .iter()
.filter_map(|elem| match elem { .filter_map(|elem| match elem {
&vkxml::RegistryElement::Features(ref features) => Some(features), &vkxml::RegistryElement::Features(ref features) => Some(features),
@ -1015,7 +1112,8 @@ pub fn write_source_code(spec: &vkxml::Registry) {
.flat_map(|features| features.elements.iter().map(|feature| feature)) .flat_map(|features| features.elements.iter().map(|feature| feature))
.collect(); .collect();
let extensions: Vec<&vkxml::Extension> = spec.elements let extensions: Vec<&vkxml::Extension> = spec
.elements
.iter() .iter()
.filter_map(|elem| match elem { .filter_map(|elem| match elem {
&vkxml::RegistryElement::Extensions(ref extensions) => Some(extensions), &vkxml::RegistryElement::Extensions(ref extensions) => Some(extensions),
@ -1024,7 +1122,8 @@ pub fn write_source_code(spec: &vkxml::Registry) {
.flat_map(|extensions| extensions.elements.iter().map(|extension| extension)) .flat_map(|extensions| extensions.elements.iter().map(|extension| extension))
.collect(); .collect();
let definitions: Vec<&vkxml::DefinitionsElement> = spec.elements let definitions: Vec<&vkxml::DefinitionsElement> = spec
.elements
.iter() .iter()
.filter_map(|elem| match elem { .filter_map(|elem| match elem {
&vkxml::RegistryElement::Definitions(ref definitions) => Some(definitions), &vkxml::RegistryElement::Definitions(ref definitions) => Some(definitions),
@ -1033,7 +1132,8 @@ pub fn write_source_code(spec: &vkxml::Registry) {
.flat_map(|definitions| definitions.elements.iter().map(|definition| definition)) .flat_map(|definitions| definitions.elements.iter().map(|definition| definition))
.collect(); .collect();
let enums: Vec<&vkxml::Enumeration> = spec.elements let enums: Vec<&vkxml::Enumeration> = spec
.elements
.iter() .iter()
.filter_map(|elem| match elem { .filter_map(|elem| match elem {
&vkxml::RegistryElement::Enums(ref enums) => Some(enums), &vkxml::RegistryElement::Enums(ref enums) => Some(enums),
@ -1047,7 +1147,8 @@ pub fn write_source_code(spec: &vkxml::Registry) {
}) })
.collect(); .collect();
let constants: Vec<&vkxml::Constant> = spec.elements let constants: Vec<&vkxml::Constant> = spec
.elements
.iter() .iter()
.filter_map(|elem| match elem { .filter_map(|elem| match elem {
&vkxml::RegistryElement::Constants(ref constants) => Some(constants), &vkxml::RegistryElement::Constants(ref constants) => Some(constants),
@ -1074,7 +1175,7 @@ pub fn write_source_code(spec: &vkxml::Registry) {
let feature_code: Vec<_> = features let feature_code: Vec<_> = features
.iter() .iter()
.map(|feature| generate_core_spec(feature, &commands)) .map(|feature| generate_feature(feature, &commands))
.collect(); .collect();
let extension_code: Vec<_> = extensions let extension_code: Vec<_> = extensions
@ -1089,18 +1190,19 @@ pub fn write_source_code(spec: &vkxml::Registry) {
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();
let platform_specific_types = platform_specific_types();
let source_code = quote!{ let source_code = quote!{
use libc::*; pub use libc::*;
// #bitflags_macro #platform_specific_types
// #handle_nondispatchable_macro #bitflags_macro
// #define_handle_macro #handle_nondispatchable_macro
// #(#feature_code)* #define_handle_macro
// #(#extension_code)* #(#feature_code)*
// #(#definition_code)* #(#definition_code)*
// #(#enum_code)* #(#enum_code)*
// #(#bitflags_code)* #(#bitflags_code)*
#(#constants_code)* #(#constants_code)*
#(#extension_code)*
}; };
println!("{:?}", cexpr("(~0UL)"));
write!(&mut file, "{}", source_code); write!(&mut file, "{}", source_code);
} }