From ee1b7fd3bc66501b252ce9e3e5ba89ac9aa54632 Mon Sep 17 00:00:00 2001 From: omniscient <17525998+omnisci3nce@users.noreply.github.com> Date: Sat, 10 Aug 2024 23:18:22 +1000 Subject: more work on bindings and ral --- bindgen/rust/celeritas-sys/Cargo.lock | 29 +++++++ bindgen/rust/celeritas-sys/build.rs | 27 ++++-- bindgen/rust/celeritas-sys/src/lib.rs | 26 ++++++ bindgen/rust/examples/scene.rs | 2 +- bindgen/rust/src/lib.rs | 50 ++++++++++- bindgen/rust/src/ral.rs | 157 +++++++++++++++++++++++++++++++--- bindgen/rust/src/resources.rs | 4 +- bindgen/rust/src/shader.rs | 1 + 8 files changed, 273 insertions(+), 23 deletions(-) (limited to 'bindgen/rust') diff --git a/bindgen/rust/celeritas-sys/Cargo.lock b/bindgen/rust/celeritas-sys/Cargo.lock index 89c7407..43ae7fb 100644 --- a/bindgen/rust/celeritas-sys/Cargo.lock +++ b/bindgen/rust/celeritas-sys/Cargo.lock @@ -39,6 +39,15 @@ dependencies = [ "memchr", ] +[[package]] +name = "approx" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cab112f0a86d568ea0e627cc1d6be74a1e9cd55214684db5561995f6dad897c6" +dependencies = [ + "num-traits", +] + [[package]] name = "autocfg" version = "1.3.0" @@ -100,6 +109,7 @@ dependencies = [ "egui", "egui_glfw", "gl", + "glam", "serde", "serde_json", ] @@ -309,6 +319,16 @@ dependencies = [ "xml-rs", ] +[[package]] +name = "glam" +version = "0.28.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "779ae4bf7e8421cf91c0b3b64e7e8b40b862fba4d393f59150042de7c4965a94" +dependencies = [ + "approx", + "serde", +] + [[package]] name = "glfw" version = "0.55.0" @@ -492,6 +512,15 @@ dependencies = [ "minimal-lexical", ] +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", +] + [[package]] name = "objc" version = "0.2.7" diff --git a/bindgen/rust/celeritas-sys/build.rs b/bindgen/rust/celeritas-sys/build.rs index 9e97034..b356bb0 100644 --- a/bindgen/rust/celeritas-sys/build.rs +++ b/bindgen/rust/celeritas-sys/build.rs @@ -13,16 +13,26 @@ const SERIALIZABLE_TYPES: &[&'static str] = &[ "DirectionalLight", "PointLight", ]; +const EQ_TYPES: &[&'static str] = &[ + "BufferHandle", + "TextureHandle", + "MeshHandle", + "MaterialHandle", + "ModelHandle", +]; #[derive(Debug)] -struct DeriveSerialize; -impl ParseCallbacks for DeriveSerialize { +struct AdditionalDerives; +impl ParseCallbacks for AdditionalDerives { fn add_derives(&self, info: &bindgen::callbacks::DeriveInfo<'_>) -> Vec { + let mut derives = vec![]; if SERIALIZABLE_TYPES.contains(&info.name) { - vec!["Serialize".to_string(), "Deserialize".to_string()] - } else { - vec![] + derives.extend_from_slice(&["Serialize".to_string(), "Deserialize".to_string()]); + } + if EQ_TYPES.contains(&info.name) { + derives.extend_from_slice(&["PartialEq".to_string()]); } + derives } } @@ -34,7 +44,8 @@ fn main() { // println!("cargo:rustc-link-search=../../build/windows/x64/debug"); let static_lib_path = - "/Users/josh/code/CodenameVentus/deps/celeritas-core/build/macosx/arm64/debug".to_string(); + "/Users/josh/code/CodenameVentus/deps/celeritas-core/build/macosx/arm64/release" + .to_string(); // let static_lib_path = std::env::var("CELERITAS_CORE_LIB") // .unwrap_or("../../../build/macosx/arm64/debug".to_string()); @@ -75,8 +86,10 @@ fn main() { .generate_inline_functions(true) // Tell cargo to invalidate the built crate whenever any of the // included header files changed. + .rustified_enum("GPU_TextureType") + .rustified_enum("GPU_TextureFormat") .parse_callbacks(Box::new(bindgen::CargoCallbacks::new())) - .parse_callbacks(Box::new(DeriveSerialize)) + .parse_callbacks(Box::new(AdditionalDerives)) // Finish the builder and generate the bindings. .generate() // Unwrap the Result and panic on failure. diff --git a/bindgen/rust/celeritas-sys/src/lib.rs b/bindgen/rust/celeritas-sys/src/lib.rs index 780ceff..90a5914 100644 --- a/bindgen/rust/celeritas-sys/src/lib.rs +++ b/bindgen/rust/celeritas-sys/src/lib.rs @@ -87,6 +87,19 @@ impl Transform { } } +impl Vec3 { + pub const ZERO: Self = Vec3 { + x: 0., + y: 0., + z: 0., + }; + pub const ONE: Self = Vec3 { + x: 1., + y: 1., + z: 1., + }; +} + impl Default for ShaderBinding { fn default() -> Self { Self { @@ -139,3 +152,16 @@ impl Default for Camera { camera } } + +// -- fat pointer string type + +impl Str8 { + pub fn from_str(s: &str) -> Self { + let s = s.to_owned(); + let s = s.leak(); // TODO: a better way than just leaking string memory :P + Self { + buf: s.as_mut_ptr(), + len: 0, + } + } +} diff --git a/bindgen/rust/examples/scene.rs b/bindgen/rust/examples/scene.rs index ba6b9ef..568ae89 100644 --- a/bindgen/rust/examples/scene.rs +++ b/bindgen/rust/examples/scene.rs @@ -56,7 +56,7 @@ fn main() { }, }, point_lights: [None, None, None, None], - camera_orientation: (camera_pos, camera_front), + camera_orientation: (camera_pos.into(), camera_front.into()), models: vec![], }; diff --git a/bindgen/rust/src/lib.rs b/bindgen/rust/src/lib.rs index f2e177f..dd83293 100644 --- a/bindgen/rust/src/lib.rs +++ b/bindgen/rust/src/lib.rs @@ -4,6 +4,7 @@ #![cfg_attr(docsrs, feature(doc_cfg))] pub use celeritas_sys as ffi; +use glam::Vec3; /// Commonly used types pub mod prelude; @@ -17,14 +18,20 @@ use std::{ fs::{self, File}, io::Write, path::Path, + ptr::addr_of_mut, }; use celeritas_sys::{ - Camera, Camera_Create, Core_Bringup, Core_Shutdown, DirectionalLight, PointLight, RenderEnt, - Transform, Vec3, + Core_Bringup, Core_Shutdown, DirectionalLight, Material, Material_Insert, Model, PointLight, + TextureHandle, Transform, }; use serde::{Deserialize, Serialize}; +pub trait IntoFFI { + type FFIType; + unsafe fn into_ffi(self) -> Self::FFIType; +} + /// Wrapper around a string that is the path to a gltf model **relative** to the configured /// `ASSETS` folder #[derive(Debug, Clone, Serialize, Deserialize)] @@ -103,3 +110,42 @@ bitflags::bitflags! { } } + +type M = Material; +/* +pub name: [::std::os::raw::c_char; 64usize], +pub kind: MaterialKind, +pub base_colour: Vec3, +pub metallic: f32_, +pub roughness: f32_, +pub ambient_occlusion: f32_, +pub albedo_map: TextureHandle, +pub normal_map: TextureHandle, +pub metallic_roughness_map: TextureHandle, +pub ambient_occlusion_map: TextureHandle, */ + +#[derive(Debug, Clone, Default, PartialEq)] +pub struct PBRMaterial { + pub name: String, + pub base_colour: glam::Vec3, + pub metallic: f32, + pub roughness: f32, + pub ambient_occlusion: f32, + pub albedo_map: Option, + pub normal_map: Option, + pub metallic_roughness_map: Option, + pub ambient_occlusion_map: Option, +} +impl PBRMaterial { + /// Creates the material in the C core returning a handle to it + pub fn create(mat: Self) -> ffi::MaterialHandle { + let mut ffi_mat = ffi::Material::from(mat); + unsafe { Material_Insert(addr_of_mut!(ffi_mat)) } + } +} + +impl From for ffi::Material { + fn from(value: PBRMaterial) -> Self { + todo!("impl conv for materials") + } +} diff --git a/bindgen/rust/src/ral.rs b/bindgen/rust/src/ral.rs index 078121b..3ea69e2 100644 --- a/bindgen/rust/src/ral.rs +++ b/bindgen/rust/src/ral.rs @@ -1,23 +1,25 @@ //! Wrapper around the RAL code in celeritas-core -use std::ffi::c_void; +use std::ffi::{c_void, CString}; use celeritas_sys::{ BufferHandle, GPU_CmdEncoder, GPU_CmdEncoder_BeginRender, GPU_CmdEncoder_EndRender, GPU_EncodeBindShaderData, GPU_GetDefaultEncoder, GPU_GetDefaultRenderpass, GPU_GraphicsPipeline_Create, GPU_Pipeline, GraphicsPipelineDesc, ShaderBindingKind_BINDING_BYTES, ShaderBinding__bindgen_ty_1, - ShaderBinding__bindgen_ty_1__bindgen_ty_1, ShaderVisibility_VISIBILITY_COMPUTE, - ShaderVisibility_VISIBILITY_FRAGMENT, ShaderVisibility_VISIBILITY_VERTEX, TextureHandle, - MAX_SHADER_DATA_LAYOUTS, + ShaderBinding__bindgen_ty_1__bindgen_ty_1, ShaderDesc, ShaderVisibility_VISIBILITY_COMPUTE, + ShaderVisibility_VISIBILITY_FRAGMENT, ShaderVisibility_VISIBILITY_VERTEX, Str8, TextureHandle, + VertexDescription, MAX_SHADER_DATA_LAYOUTS, }; use thiserror::Error; +use crate::IntoFFI; + /// Holds a pointer to the raw `GPU_CmdEncoder` pub struct FrameRenderEncoder(*mut GPU_CmdEncoder); /// Holds a pointer to the raw `GPU_Renderpass` -pub struct RenderPass(*mut celeritas_sys::GPU_Renderpass); +pub struct RenderPass(pub *mut celeritas_sys::GPU_Renderpass); /// Holds a pointer to the raw `GPU_Pipeline` pub struct Pipeline(*mut celeritas_sys::GPU_Pipeline); @@ -58,9 +60,67 @@ impl FrameRenderEncoder { } } +// Vertex Descriptions +pub enum VertexAttrKind { + Floatx1, + Floatx2, + Floatx3, + Floatx4, + U32x1, + U32x2, + U32x3, + U32x4, + I32x1, + I32x2, + I32x3, + I32x4, +} + +pub struct VertexAttribute { + name: String, + kind: VertexAttrKind, +} + +#[derive(Default)] +pub struct VertexDesc { + debug_label: String, + attributes: Vec, +} +impl VertexDesc { + pub fn new(name: String) -> Self { + Self { + debug_label: name, + attributes: vec![], + } + } + pub fn add_attr(mut self, attr_name: &str, kind: VertexAttrKind) -> Self { + self.attributes.push(VertexAttribute { + name: attr_name.to_owned(), + kind, + }); + self + } +} +impl IntoFFI for VertexDesc { + type FFIType = VertexDescription; + + unsafe fn into_ffi(self) -> Self::FFIType { + VertexDescription { + debug_label: todo!(), + attr_names: todo!(), + attributes: todo!(), + attributes_count: todo!(), + use_full_vertex_size: todo!(), + } + } +} + pub struct PipelineBuilder { + name: String, renderpass: Option, + vertex_description: VertexDesc, data_layouts: Vec, + shader_paths: Option<(String, String)>, } #[derive(Debug, Error)] @@ -70,6 +130,17 @@ pub enum RALError { } impl PipelineBuilder { + /// Create a new [PipelineBuilder] + pub fn new(name: String) -> Self { + let vertex_description = VertexDesc::new(format!("{} Vertex Description", name.clone())); + Self { + name, + renderpass: None, + vertex_description, + data_layouts: Vec::new(), + shader_paths: None, + } + } pub fn build(self) -> Result { let layouts = [celeritas_sys::ShaderDataLayout::default(); 8]; if self.data_layouts.len() > MAX_SHADER_DATA_LAYOUTS as usize { @@ -78,12 +149,32 @@ impl PipelineBuilder { for (i, layout) in self.data_layouts.iter().enumerate().take(8) { // layouts[i] = celeritas_sys::ShaderDataLayout::from(layout); } + let (vert_path, frag_path) = self.shader_paths.expect("Shader paths must be provided"); + let vert_code = std::fs::read_to_string(vert_path.clone()).expect("msg"); + let frag_code = std::fs::read_to_string(frag_path.clone()).expect("msg"); - let mut desc = GraphicsPipelineDesc { - debug_name: todo!(), - vertex_desc: todo!(), - vs: todo!(), - fs: todo!(), + // TODO: convert VertexDesc -> ffi::VertexDescription + // load shader + let vs = ShaderDesc { + debug_name: "".as_ptr() as *const i8, + filepath: Str8::from_str(&vert_path), + code: Str8::from_str(&vert_code), + is_spirv: false, + is_combined_vert_frag: false, + }; + let fs = ShaderDesc { + debug_name: "".as_ptr() as *const i8, + filepath: Str8::from_str(&frag_path), + code: Str8::from_str(&frag_code), + is_spirv: false, + is_combined_vert_frag: false, + }; + + let desc = GraphicsPipelineDesc { + debug_name: "".as_ptr() as *const _, + vertex_desc: unsafe { self.vertex_description.into_ffi() }, + vs, + fs, data_layouts: layouts, data_layouts_count: layouts.len() as u32, wireframe: false, @@ -100,11 +191,19 @@ impl PipelineBuilder { Ok(Pipeline(p)) } - pub fn add_shader_layout(&mut self) -> &mut Self { + pub fn add_vertex_desc(mut self, vertex_desc: VertexDesc) -> Self { + self.vertex_description = vertex_desc; + self + } + pub fn add_shader_layout(mut self) -> Self { let layout = S::layout(); self.data_layouts.push(layout); self } + pub fn add_shader_src(mut self, vertex_path: &str, fragment_path: &str) -> Self { + self.shader_paths = Some((vertex_path.to_owned(), fragment_path.to_owned())); + self + } } impl Pipeline { @@ -158,6 +257,11 @@ impl Default for ShaderVisibility { pub struct ShaderDataLayout { pub bindings: heapless::Vec, } +impl ShaderDataLayout { + pub fn from_slice(bindings: &[ShaderBinding]) -> Self { + todo!() + } +} impl ShaderDataLayout { pub fn into_ffi_type(self) -> celeritas_sys::ShaderDataLayout { let mut bindings = [celeritas_sys::ShaderBinding::default(); 8]; @@ -210,3 +314,34 @@ impl From for PrimitiveTopology { } } } +#[cfg(test)] +mod test { + use super::*; + + struct TestData { + a: [f32; 2], + b: [f32; 4], + } + impl ShaderData for TestData { + fn layout() -> ShaderDataLayout { + todo!() + } + + fn bind(&self) { + todo!() + } + } + + #[test] + fn typecheck_pipeline_create() { + let vertex_desc = VertexDesc::new("Empty".into()) + .add_attr("position", VertexAttrKind::Floatx2) + .add_attr("color", VertexAttrKind::Floatx4); + + let mut builder = PipelineBuilder::new("Test Pipeline".into()) + .add_shader_layout::() + .add_vertex_desc(vertex_desc); + + let pipeline = builder.build().expect("Should be valid"); + } +} diff --git a/bindgen/rust/src/resources.rs b/bindgen/rust/src/resources.rs index 89cf6d7..e7f4505 100644 --- a/bindgen/rust/src/resources.rs +++ b/bindgen/rust/src/resources.rs @@ -1,6 +1,6 @@ -use std::{ffi::CString, path::Path}; +use std::ffi::CString; -use celeritas_sys::{ModelHandle, ModelLoad, ModelLoad_gltf}; +use celeritas_sys::{ModelHandle, ModelLoad_gltf}; /// Load a gltf from disk pub fn model_load_gltf(path: &str) -> Option { diff --git a/bindgen/rust/src/shader.rs b/bindgen/rust/src/shader.rs index e80e00b..ef221a0 100644 --- a/bindgen/rust/src/shader.rs +++ b/bindgen/rust/src/shader.rs @@ -1,3 +1,4 @@ +#![allow(unused_variables)] use std::{ffi::c_void, path::Path}; use crate::ral::{Pipeline, ShaderBinding}; -- cgit v1.2.3-70-g09d2