diff --git a/crates/renderling-shader/src/debug.rs b/crates/renderling-shader/src/debug.rs index 83fd6518..3cd747ef 100644 --- a/crates/renderling-shader/src/debug.rs +++ b/crates/renderling-shader/src/debug.rs @@ -4,6 +4,7 @@ use crate::slab::Slabbed; /// Used to debug shaders by early exiting the shader and attempting to display /// the value as shaded colors. +// TODO: Change DebugChannel to DebugMode and remove the previous DebugMode. #[repr(u32)] #[cfg_attr(not(target_arch = "spirv"), derive(Debug))] #[derive(Clone, Copy, PartialEq, PartialOrd)] diff --git a/crates/renderling-shader/src/id.rs b/crates/renderling-shader/src/id.rs index 6cc0537d..90ab5288 100644 --- a/crates/renderling-shader/src/id.rs +++ b/crates/renderling-shader/src/id.rs @@ -71,11 +71,15 @@ impl Default for Id { impl core::fmt::Debug for Id { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { - f.write_fmt(format_args!( - "Id<{}>({})", - &core::any::type_name::(), - &self.0 - )) + if self.is_none() { + f.write_fmt(format_args!("Id<{}>(null)", &core::any::type_name::(),)) + } else { + f.write_fmt(format_args!( + "Id<{}>({})", + &core::any::type_name::(), + &self.0 + )) + } } } diff --git a/crates/renderling-shader/src/pbr.rs b/crates/renderling-shader/src/pbr.rs index c5acc1ed..a13e3a13 100644 --- a/crates/renderling-shader/src/pbr.rs +++ b/crates/renderling-shader/src/pbr.rs @@ -4,16 +4,11 @@ //! * https://learnopengl.com/PBR/Theory //! * https://github.com/KhronosGroup/glTF-Sample-Viewer/blob/5b1b7f48a8cb2b7aaef00d08fdba18ccc8dd331b/source/Renderer/shaders/pbr.frag //! * https://github.khronos.org/glTF-Sample-Viewer-Release/ -use renderling_derive::Slabbed; +use glam::{Vec2, Vec3, Vec4, Vec4Swizzles}; +use renderling_derive::Slabbed; #[cfg(target_arch = "spirv")] use spirv_std::num_traits::Float; -use spirv_std::{ - image::{Cubemap, Image2d}, - Sampler, -}; - -use glam::{Vec2, Vec3, Vec4, Vec4Swizzles}; use crate::{ self as renderling_shader, @@ -154,21 +149,42 @@ fn outgoing_radiance( metalness: f32, roughness: f32, ) -> Vec3 { + crate::println!("outgoing_radiance"); + crate::println!(" light_color: {light_color:?}"); + crate::println!(" albedo: {albedo:?}"); + crate::println!(" attenuation: {attenuation:?}"); + crate::println!(" v: {v:?}"); + crate::println!(" l: {l:?}"); + crate::println!(" n: {n:?}"); + crate::println!(" metalness: {metalness:?}"); + crate::println!(" roughness: {roughness:?}"); + let f0 = Vec3::splat(0.4).lerp(albedo, metalness); + crate::println!(" f0: {f0:?}"); let radiance = light_color.xyz() * attenuation; + crate::println!(" radiance: {radiance:?}"); let h = (v + l).alt_norm_or_zero(); + crate::println!(" h: {h:?}"); // cook-torrance brdf let ndf: f32 = normal_distribution_ggx(n, h, roughness); + crate::println!(" ndf: {ndf:?}"); let g: f32 = geometry_smith(n, v, l, roughness); + crate::println!(" g: {g:?}"); let f: Vec3 = fresnel_schlick(h.dot(v).max(0.0), f0); + crate::println!(" f: {f:?}"); let k_s = f; let k_d = (Vec3::splat(1.0) - k_s) * (1.0 - metalness); + crate::println!(" k_s: {k_s:?}"); let numerator: Vec3 = ndf * g * f; + crate::println!(" numerator: {numerator:?}"); let n_dot_l = n.dot(l).max(0.0); + crate::println!(" n_dot_l: {n_dot_l:?}"); let denominator: f32 = 4.0 * n.dot(v).max(0.0) * n_dot_l + 0.0001; + crate::println!(" denominator: {denominator:?}"); let specular: Vec3 = numerator / denominator; + crate::println!(" specular: {specular:?}"); (k_d * albedo / core::f32::consts::PI + specular) * radiance * n_dot_l } @@ -349,7 +365,9 @@ pub fn stage_shade_fragment( ) -> Vec4 { let n = in_norm.alt_norm_or_zero(); let v = (camera_pos - in_pos).alt_norm_or_zero(); - + crate::println!("lights: {lights:?}"); + crate::println!("n: {n:?}"); + crate::println!("v: {v:?}"); // reflectance let mut lo = Vec3::ZERO; for i in 0..lights.len() { @@ -405,9 +423,12 @@ pub fn stage_shade_fragment( LightStyle::Directional => { let dir_light = slab.read(light.into_directional_id()); - let l = (-dir_light.direction).alt_norm_or_zero(); + let l = -dir_light.direction.alt_norm_or_zero(); let attenuation = dir_light.intensity; - lo += outgoing_radiance( + crate::println!("dir_light: {dir_light:?}"); + crate::println!("l: {l:?}"); + crate::println!("attenuation: {attenuation:?}"); + let radiance = outgoing_radiance( dir_light.color, albedo, attenuation, @@ -417,10 +438,13 @@ pub fn stage_shade_fragment( metallic, roughness, ); + crate::println!("radiance: {radiance:?}"); + lo += radiance; } } } + crate::println!("lo: {lo:?}"); // calculate reflectance at normal incidence; if dia-electric (like plastic) use // F0 of 0.04 and if it's a metal, use the albedo color as F0 (metallic // workflow) diff --git a/crates/renderling-shader/src/stage.rs b/crates/renderling-shader/src/stage.rs index d75d0366..b20663e6 100644 --- a/crates/renderling-shader/src/stage.rs +++ b/crates/renderling-shader/src/stage.rs @@ -506,14 +506,18 @@ pub struct DrawIndirect { pub base_instance: u32, } -fn texture_color( +fn texture_color( texture_id: Id, uv: Vec2, - atlas: &Image2d, - sampler: &Sampler, + atlas: &T, + sampler: &S, atlas_size: UVec2, textures: &[GpuTexture], -) -> Vec4 { +) -> Vec4 +where + T: Sample2d, + S: IsSampler, +{ let texture = if texture_id.is_none() { GpuTexture::default() } else { @@ -636,6 +640,65 @@ pub fn main_fragment_scene( output: &mut Vec4, brigtness: &mut Vec4, ) { + main_fragment_impl( + atlas, + atlas_sampler, + irradiance, + irradiance_sampler, + prefiltered, + prefiltered_sampler, + brdf, + brdf_sampler, + constants, + lights, + materials, + textures, + in_material, + in_color, + in_uv0, + in_uv1, + in_norm, + in_tangent, + in_bitangent, + in_pos, + output, + brigtness, + ); +} + +/// Scene fragment shader, callable from the CPU or GPU. +pub fn main_fragment_impl( + atlas: &T, + atlas_sampler: &S, + + irradiance: &C, + irradiance_sampler: &S, + prefiltered: &C, + prefiltered_sampler: &S, + brdf: &T, + brdf_sampler: &S, + + constants: &GpuConstants, + lights: &[GpuLight], + materials: &[pbr::PbrMaterial], + textures: &[GpuTexture], + + in_material: u32, + in_color: Vec4, + in_uv0: Vec2, + in_uv1: Vec2, + in_norm: Vec3, + in_tangent: Vec3, + in_bitangent: Vec3, + in_pos: Vec3, + + output: &mut Vec4, + brigtness: &mut Vec4, +) where + T: Sample2d, + C: SampleCube, + S: IsSampler, +{ let material = if in_material == ID_NONE || !constants.toggles.get_use_lighting() { // without an explicit material (or if the entire render has no lighting) // the entity will not participate in any lighting calculations @@ -871,6 +934,12 @@ pub fn main_fragment_scene( } /// A camera used for transforming the stage during rendering. +/// +/// Use `Camera::new(projection, view)` to create a new camera. +/// Or use `Camera::default` followed by `Camera::with_projection_and_view` +/// to set the projection and view matrices. Using the `with_*` or `set_*` +/// methods is preferred over setting the fields directly because they will +/// also update the camera's position. #[cfg_attr(not(target_arch = "spirv"), derive(Debug))] #[repr(C)] #[derive(Default, Clone, Copy, PartialEq, Slabbed)] @@ -880,6 +949,41 @@ pub struct Camera { pub position: Vec3, } +impl Camera { + pub fn new(projection: Mat4, view: Mat4) -> Self { + Camera::default().with_projection_and_view(projection, view) + } + + pub fn set_projection_and_view(&mut self, projection: Mat4, view: Mat4) { + self.projection = projection; + self.view = view; + self.position = view.inverse().transform_point3(Vec3::ZERO); + } + + pub fn with_projection_and_view(mut self, projection: Mat4, view: Mat4) -> Self { + self.set_projection_and_view(projection, view); + self + } + + pub fn set_projection(&mut self, projection: Mat4) { + self.set_projection_and_view(projection, self.view); + } + + pub fn with_projection(mut self, projection: Mat4) -> Self { + self.set_projection(projection); + self + } + + pub fn set_view(&mut self, view: Mat4) { + self.set_projection_and_view(self.projection, view); + } + + pub fn with_view(mut self, view: Mat4) -> Self { + self.set_view(view); + self + } +} + /// Holds important info about the stage. /// /// This should be the first struct in the stage's slab. @@ -1212,14 +1316,18 @@ pub fn stage_fragment_impl( C: SampleCube, S: IsSampler, { + let legend = get_stage_legend(slab); + crate::println!("legend: {:?}", legend); let StageLegend { atlas_size, debug_mode, has_skybox: _, has_lighting, light_array, - } = get_stage_legend(slab); + } = legend; + let material = get_material(in_material, has_lighting, slab); + crate::println!("material: {:?}", material); let albedo_tex_uv = if material.albedo_tex_coord == 0 { in_uv0 @@ -1234,6 +1342,7 @@ pub fn stage_fragment_impl( atlas_size, slab, ); + crate::println!("albedo_tex_color: {:?}", albedo_tex_color); let metallic_roughness_uv = if material.metallic_roughness_tex_coord == 0 { in_uv0 @@ -1248,6 +1357,10 @@ pub fn stage_fragment_impl( atlas_size, slab, ); + crate::println!( + "metallic_roughness_tex_color: {:?}", + metallic_roughness_tex_color + ); let normal_tex_uv = if material.normal_tex_coord == 0 { in_uv0 @@ -1262,6 +1375,7 @@ pub fn stage_fragment_impl( atlas_size, slab, ); + crate::println!("normal_tex_color: {:?}", normal_tex_color); let ao_tex_uv = if material.ao_tex_coord == 0 { in_uv0 diff --git a/crates/renderling/src/atlas.rs b/crates/renderling/src/atlas.rs index 8c25fbc5..d0424a2f 100644 --- a/crates/renderling/src/atlas.rs +++ b/crates/renderling/src/atlas.rs @@ -497,7 +497,7 @@ impl Atlas { #[cfg(test)] impl Atlas { - fn atlas_img(&self, device: &wgpu::Device, queue: &wgpu::Queue) -> RgbaImage { + pub fn atlas_img(&self, device: &wgpu::Device, queue: &wgpu::Queue) -> RgbaImage { let buffer = crate::Texture::read( &self.texture.texture, device, diff --git a/crates/renderling/src/lib.rs b/crates/renderling/src/lib.rs index 857819b7..576bb541 100644 --- a/crates/renderling/src/lib.rs +++ b/crates/renderling/src/lib.rs @@ -612,7 +612,7 @@ mod test { /// A helper struct that contains all outputs of the vertex shader. #[allow(unused)] - #[derive(Debug, Default)] + #[derive(Clone, Debug, Default, PartialEq)] pub struct VertexInvocation { pub instance_index: u32, pub vertex_index: u32, @@ -634,6 +634,7 @@ mod test { } impl VertexInvocation { + #[allow(dead_code)] pub fn invoke(instance_index: u32, vertex_index: u32, slab: &[u32]) -> Self { let mut v = Self { instance_index, @@ -1163,12 +1164,7 @@ mod test { assert_eq!(Vec2::splat(0.1), tex.uv(Vec2::ZERO, UVec2::splat(100))); } - #[test] - /// Tests shading with directional light. - fn old_scene_cube_directional() { - let mut r = - Renderling::headless(100, 100).with_background_color(Vec3::splat(0.0).extend(1.0)); - + fn old_scene_cube_directional_builder(r: &Renderling) -> SceneBuilder { let mut builder = r.new_scene(); let red = Vec3::X.extend(1.0); let green = Vec3::Y.extend(1.0); @@ -1208,14 +1204,29 @@ mod test { .with_material(material) .build(); - let mut scene = builder.build().unwrap(); + builder + } + fn old_scene_cube_directional_camera() -> (Mat4, Mat4) { let (projection, _) = camera::default_perspective(100.0, 100.0); let view = Mat4::look_at_rh( Vec3::new(1.8, 1.8, 1.8), Vec3::ZERO, Vec3::new(0.0, 1.0, 0.0), ); + (projection, view) + } + + #[test] + /// Tests shading with directional light. + fn old_scene_cube_directional() { + let mut r = + Renderling::headless(100, 100).with_background_color(Vec3::splat(0.0).extend(1.0)); + + let builder = old_scene_cube_directional_builder(&r); + let mut scene = builder.build().unwrap(); + + let (projection, view) = old_scene_cube_directional_camera(); scene.set_camera(projection, view); r.setup_render_graph(RenderGraphConfig { @@ -1242,11 +1253,7 @@ mod test { Vec3::ZERO, Vec3::new(0.0, 1.0, 0.0), ); - let camera = stage.append(&Camera { - projection, - view, - ..Default::default() - }); + let camera = stage.append(&Camera::default().with_projection_and_view(projection, view)); let red = Vec3::X.extend(1.0); let green = Vec3::Y.extend(1.0); @@ -1293,21 +1300,12 @@ mod test { ..Default::default() }); - let cube = stage.draw_unit(&RenderUnit { + let _cube = stage.draw_unit(&RenderUnit { vertex_data: VertexData::new_native(vertex_data), vertex_count: vertices.len() as u32, camera, ..Default::default() }); - - let data = stage.read_all_raw().unwrap(); - let invocation = VertexInvocation::invoke(cube.inner(), 0, &data); - println!("vertex invocation: {:#?}", invocation); - - let atlas: image::DynamicImage = stage.atlas.read().unwrap().atlas_img().into(); - let atlas = CpuTexture2d { image: atlas }; - let _ = crate::shader::stage_fragment_impl(&atlas, &CpuSampler); - let img = r.render_image().unwrap(); img_diff::assert_img_eq("scene_cube_directional.png", img); } diff --git a/crates/renderling/src/linkage/stage-main_fragment_scene.spv b/crates/renderling/src/linkage/stage-main_fragment_scene.spv index 57781915..cbde1de8 100644 Binary files a/crates/renderling/src/linkage/stage-main_fragment_scene.spv and b/crates/renderling/src/linkage/stage-main_fragment_scene.spv differ diff --git a/crates/renderling/src/linkage/stage-stage_fragment.spv b/crates/renderling/src/linkage/stage-stage_fragment.spv index 2b2a66b4..5f18e07e 100644 Binary files a/crates/renderling/src/linkage/stage-stage_fragment.spv and b/crates/renderling/src/linkage/stage-stage_fragment.spv differ diff --git a/crates/renderling/src/stage/gltf_support.rs b/crates/renderling/src/stage/gltf_support.rs index 1695331a..19976fbf 100644 --- a/crates/renderling/src/stage/gltf_support.rs +++ b/crates/renderling/src/stage/gltf_support.rs @@ -1378,14 +1378,10 @@ mod test { .load_gltf_document_from_path("../../gltf/gltfTutorial_013_SimpleTexture.gltf") .unwrap(); - let position = Vec3::new(0.5, 0.5, 1.25); let projection = crate::camera::perspective(size as f32, size as f32); - let view = crate::camera::look_at(position, Vec3::new(0.5, 0.5, 0.0), Vec3::Y); - let camera = stage.append(&Camera { - projection, - view, - position, - }); + let view = + crate::camera::look_at(Vec3::new(0.5, 0.5, 1.25), Vec3::new(0.5, 0.5, 0.0), Vec3::Y); + let camera = stage.append(&Camera::new(projection, view)); let _unit_ids = stage.draw_gltf_scene(&gpu_doc, camera, cpu_doc.default_scene().unwrap()); let img = r.render_image().unwrap(); diff --git a/shaders/Cargo.toml b/shaders/Cargo.toml index 551b2b4e..498ec4d8 100644 --- a/shaders/Cargo.toml +++ b/shaders/Cargo.toml @@ -12,10 +12,17 @@ env_logger = "^0.10" log = "^0.4" spirv-builder = "^0.9" -# Compile build-dependencies in release mode with -# the same settings as regular dependencies. +# Enable incremental by default in release mode. +[profile.release] +incremental = true +# HACK(eddyb) this is the default but without explicitly specifying it, Cargo +# will treat the identical settings in `[profile.release.build-override]` below +# as different sets of `rustc` flags and will not reuse artifacts between them. +codegen-units = 256 + +# Compile build-dependencies in release mode with the same settings +# as regular dependencies (including the incremental enabled above). [profile.release.build-override] opt-level = 3 -codegen-units = 16 -[profile.dev.build-override] -opt-level = 3 \ No newline at end of file +incremental = true +codegen-units = 256 \ No newline at end of file diff --git a/shaders/shader-crate/Cargo.toml b/shaders/shader-crate/Cargo.toml index 5173cb9a..9b375b65 100644 --- a/shaders/shader-crate/Cargo.toml +++ b/shaders/shader-crate/Cargo.toml @@ -15,3 +15,18 @@ crate-type = ["dylib"] renderling-shader = { path = "../../crates/renderling-shader" } spirv-std = "0.9" glam = { version = "0.24.2", default-features = false, features = ["libm"]} + +# Enable incremental by default in release mode. +[profile.release] +incremental = true +# HACK(eddyb) this is the default but without explicitly specifying it, Cargo +# will treat the identical settings in `[profile.release.build-override]` below +# as different sets of `rustc` flags and will not reuse artifacts between them. +codegen-units = 256 + +# Compile build-dependencies in release mode with the same settings +# as regular dependencies (including the incremental enabled above). +[profile.release.build-override] +opt-level = 3 +incremental = true +codegen-units = 256 \ No newline at end of file