use crate::{error, util}; use ash::vk; use crate::error::FilterChainError; use crate::framebuffer::OutputImage; use crate::render_pass::VulkanRenderPass; use ash::vk::PushConstantRange; use bytemuck::offset_of; use librashader_cache::cache_pipeline; use librashader_reflect::back::ShaderCompilerOutput; use librashader_reflect::reflect::semantics::{BufferReflection, TextureBinding}; use librashader_reflect::reflect::ShaderReflection; use librashader_runtime::quad::VertexInput; use librashader_runtime::render_target::RenderTarget; use std::ffi::CStr; use std::sync::Arc; const ENTRY_POINT: &CStr = unsafe { CStr::from_bytes_with_nul_unchecked(b"main\0") }; pub struct PipelineDescriptors { pub replicas: u32, pub layout_bindings: Vec, pub pool_sizes: Vec, } impl PipelineDescriptors { pub fn new(duplicates: u32) -> Self { Self { replicas: duplicates, layout_bindings: vec![], pool_sizes: vec![], } } pub fn add_ubo_binding(&mut self, ubo_meta: Option<&BufferReflection>) { if let Some(ubo_meta) = ubo_meta && !ubo_meta.stage_mask.is_empty() { let ubo_mask = util::binding_stage_to_vulkan_stage(ubo_meta.stage_mask); self.layout_bindings.push(vk::DescriptorSetLayoutBinding { binding: ubo_meta.binding, descriptor_type: vk::DescriptorType::UNIFORM_BUFFER, descriptor_count: 1, stage_flags: ubo_mask, p_immutable_samplers: std::ptr::null(), }); self.pool_sizes.push(vk::DescriptorPoolSize { ty: vk::DescriptorType::UNIFORM_BUFFER, descriptor_count: self.replicas, }) } } pub fn add_texture_bindings<'a>(&mut self, textures: impl Iterator) { let texture_mask = vk::ShaderStageFlags::FRAGMENT; for texture in textures { self.layout_bindings.push(vk::DescriptorSetLayoutBinding { binding: texture.binding, descriptor_type: vk::DescriptorType::COMBINED_IMAGE_SAMPLER, descriptor_count: 1, stage_flags: texture_mask, p_immutable_samplers: std::ptr::null(), }); self.pool_sizes.push(vk::DescriptorPoolSize { ty: vk::DescriptorType::COMBINED_IMAGE_SAMPLER, descriptor_count: self.replicas, }) } } pub fn bindings(&self) -> &[vk::DescriptorSetLayoutBinding] { self.layout_bindings.as_ref() } pub fn create_descriptor_set_layout( &self, device: &ash::Device, ) -> error::Result { unsafe { let layout = device.create_descriptor_set_layout( &vk::DescriptorSetLayoutCreateInfo::builder().bindings(self.bindings()), None, )?; Ok(layout) } } } pub struct PipelineLayoutObjects { pub layout: vk::PipelineLayout, pub pool: vk::DescriptorPool, pub descriptor_sets: Vec, pub descriptor_set_layout: [vk::DescriptorSetLayout; 1], } impl PipelineLayoutObjects { pub fn new( reflection: &ShaderReflection, replicas: u32, device: &ash::Device, ) -> error::Result { let mut descriptors = PipelineDescriptors::new(replicas); descriptors.add_ubo_binding(reflection.ubo.as_ref()); descriptors.add_texture_bindings(reflection.meta.texture_meta.values()); let descriptor_set_layout = [descriptors.create_descriptor_set_layout(device)?]; let pipeline_create_info = vk::PipelineLayoutCreateInfo::builder().set_layouts(&descriptor_set_layout); let push_constant_range = reflection.push_constant.as_ref().map(|push_constant| { let stage_mask = util::binding_stage_to_vulkan_stage(push_constant.stage_mask); [*vk::PushConstantRange::builder() .stage_flags(stage_mask) .size(push_constant.size)] }); let push_constant_range: &[PushConstantRange] = push_constant_range.as_ref().map_or(&[], |o| o); let pipeline_create_info = pipeline_create_info.push_constant_ranges(push_constant_range); let layout = unsafe { device.create_pipeline_layout(&pipeline_create_info, None)? }; let pool_info = vk::DescriptorPoolCreateInfo::builder() .max_sets(replicas) .pool_sizes(&descriptors.pool_sizes); let pool = unsafe { device.create_descriptor_pool(&pool_info, None)? }; let mut descriptor_sets = Vec::new(); let alloc_info = vk::DescriptorSetAllocateInfo::builder() .descriptor_pool(pool) .set_layouts(&descriptor_set_layout); for _ in 0..replicas { let set = unsafe { device.allocate_descriptor_sets(&alloc_info)? }; descriptor_sets.push(set) } let descriptor_sets: Vec = descriptor_sets.into_iter().flatten().collect(); Ok(PipelineLayoutObjects { layout, descriptor_set_layout, descriptor_sets, pool, }) } } pub struct VulkanShaderModule { shader: vk::ShaderModule, device: ash::Device, } impl VulkanShaderModule { pub fn new( device: &ash::Device, info: &vk::ShaderModuleCreateInfo, ) -> error::Result { Ok(VulkanShaderModule { shader: unsafe { device.create_shader_module(info, None)? }, device: device.clone(), }) } } impl Drop for VulkanShaderModule { fn drop(&mut self) { unsafe { self.device.destroy_shader_module(self.shader, None) } } } pub struct VulkanGraphicsPipeline { pub layout: PipelineLayoutObjects, pub pipeline: vk::Pipeline, pub render_pass: Option, device: Arc, vertex: VulkanShaderModule, fragment: VulkanShaderModule, cache: vk::PipelineCache, } impl VulkanGraphicsPipeline { fn create_pipeline( device: &ash::Device, cache: &vk::PipelineCache, pipeline_layout: &PipelineLayoutObjects, vertex_module: &VulkanShaderModule, fragment_module: &VulkanShaderModule, render_pass: Option<&VulkanRenderPass>, ) -> error::Result { let input_assembly = vk::PipelineInputAssemblyStateCreateInfo::builder() .topology(vk::PrimitiveTopology::TRIANGLE_STRIP); let vao_state = [ vk::VertexInputAttributeDescription { location: 0, binding: 0, format: vk::Format::R32G32_SFLOAT, offset: offset_of!(VertexInput, position) as u32, }, vk::VertexInputAttributeDescription { location: 1, binding: 0, format: vk::Format::R32G32_SFLOAT, offset: offset_of!(VertexInput, texcoord) as u32, }, ]; let input_binding = vk::VertexInputBindingDescription::builder() .binding(0) .stride(std::mem::size_of::() as u32) .input_rate(vk::VertexInputRate::VERTEX); let input_binding = [*input_binding]; let pipeline_input_state = vk::PipelineVertexInputStateCreateInfo::builder() .vertex_binding_descriptions(&input_binding) .vertex_attribute_descriptions(&vao_state); let raster_state = vk::PipelineRasterizationStateCreateInfo::builder() .polygon_mode(vk::PolygonMode::FILL) .cull_mode(vk::CullModeFlags::NONE) .front_face(vk::FrontFace::COUNTER_CLOCKWISE) .depth_clamp_enable(false) .rasterizer_discard_enable(false) .depth_bias_enable(false) .line_width(1.0); let attachments = vk::PipelineColorBlendAttachmentState::builder() .blend_enable(false) .color_write_mask(vk::ColorComponentFlags::from_raw(0xf)); let attachments = [*attachments]; let blend_state = vk::PipelineColorBlendStateCreateInfo::builder().attachments(&attachments); let viewport_state = vk::PipelineViewportStateCreateInfo::builder() .viewport_count(1) .scissor_count(1); let depth_stencil_state = vk::PipelineDepthStencilStateCreateInfo::builder() .depth_test_enable(false) .depth_write_enable(false) .stencil_test_enable(false) .depth_bounds_test_enable(false) .min_depth_bounds(1.0) .max_depth_bounds(1.0); let multisample_state = vk::PipelineMultisampleStateCreateInfo::builder() .rasterization_samples(vk::SampleCountFlags::TYPE_1); let states = [vk::DynamicState::VIEWPORT, vk::DynamicState::SCISSOR]; let dynamic_state = vk::PipelineDynamicStateCreateInfo::builder().dynamic_states(&states); let shader_stages = [ vk::PipelineShaderStageCreateInfo::builder() .stage(vk::ShaderStageFlags::VERTEX) .name(ENTRY_POINT) .module(vertex_module.shader), vk::PipelineShaderStageCreateInfo::builder() .stage(vk::ShaderStageFlags::FRAGMENT) .name(ENTRY_POINT) .module(fragment_module.shader), ]; let shader_stages = [*shader_stages[0], *shader_stages[1]]; let mut pipeline_info = vk::GraphicsPipelineCreateInfo::builder() .stages(&shader_stages) .vertex_input_state(&pipeline_input_state) .input_assembly_state(&input_assembly) .rasterization_state(&raster_state) .color_blend_state(&blend_state) .multisample_state(&multisample_state) .viewport_state(&viewport_state) .depth_stencil_state(&depth_stencil_state) .dynamic_state(&dynamic_state) .layout(pipeline_layout.layout); if let Some(render_pass) = render_pass { pipeline_info = pipeline_info.render_pass(render_pass.handle) } let pipeline = unsafe { // panic_safety: if this is successful this should return 1 pipelines. device .create_graphics_pipelines(*cache, &[*pipeline_info], None) .map_err(|e| e.1)?[0] }; Ok(pipeline) } pub fn new( device: &Arc, shader_assembly: &ShaderCompilerOutput>, reflection: &ShaderReflection, replicas: u32, render_pass_format: vk::Format, bypass_cache: bool, ) -> error::Result { let pipeline_layout = PipelineLayoutObjects::new(reflection, replicas, device)?; let vertex_info = vk::ShaderModuleCreateInfo::builder().code(shader_assembly.vertex.as_ref()); let fragment_info = vk::ShaderModuleCreateInfo::builder().code(shader_assembly.fragment.as_ref()); let vertex_module = VulkanShaderModule::new(device, &vertex_info)?; let fragment_module = VulkanShaderModule::new(device, &fragment_info)?; let mut render_pass = None; if render_pass_format != vk::Format::UNDEFINED { render_pass = Some(VulkanRenderPass::create_render_pass( device, render_pass_format, )?); } let (pipeline, pipeline_cache) = cache_pipeline( "vulkan", &[&shader_assembly.vertex, &shader_assembly.fragment], |pipeline_data| { let mut cache_info = vk::PipelineCacheCreateInfo::builder(); if let Some(pipeline_data) = pipeline_data.as_ref() { cache_info = cache_info.initial_data(pipeline_data); } let cache_info = cache_info; let pipeline_cache = unsafe { device.create_pipeline_cache(&cache_info, None)? }; let pipeline = Self::create_pipeline( &device, &pipeline_cache, &pipeline_layout, &vertex_module, &fragment_module, render_pass.as_ref(), )?; Ok::<_, FilterChainError>((pipeline, pipeline_cache)) }, |(_pipeline, cache)| unsafe { Ok(device.get_pipeline_cache_data(*cache)?) }, bypass_cache, )?; Ok(VulkanGraphicsPipeline { device: Arc::clone(device), layout: pipeline_layout, pipeline, render_pass, vertex: vertex_module, fragment: fragment_module, cache: pipeline_cache, }) } pub(crate) fn recompile(&mut self, format: vk::Format) -> error::Result<()> { let mut new_renderpass = if self.render_pass.is_some() { Some(VulkanRenderPass::create_render_pass(&self.device, format)?) } else { None }; let mut new_pipeline = Self::create_pipeline( &self.device, &self.cache, &self.layout, &self.vertex, &self.fragment, new_renderpass.as_ref(), )?; std::mem::swap(&mut self.render_pass, &mut new_renderpass); std::mem::swap(&mut self.pipeline, &mut new_pipeline); unsafe { if new_pipeline != vk::Pipeline::null() { self.device.destroy_pipeline(new_pipeline, None) } } Ok(()) } #[inline(always)] pub(crate) fn begin_rendering( &self, output: &RenderTarget, cmd: vk::CommandBuffer, ) -> error::Result> { if let Some(render_pass) = &self.render_pass { let attachments = [output.output.image_view]; let framebuffer = unsafe { self.device.create_framebuffer( &vk::FramebufferCreateInfo::builder() .render_pass(render_pass.handle) .attachments(&attachments) .width(output.output.size.width) .height(output.output.size.height) .layers(1), None, )? }; let clear_values = [vk::ClearValue { color: vk::ClearColorValue { float32: [0.0, 0.0, 0.0, 0.0], }, }]; let render_pass_info = vk::RenderPassBeginInfo::builder() .framebuffer(framebuffer) .render_pass(render_pass.handle) .clear_values(&clear_values) // always render into the full output, regardless of viewport settings. .render_area(vk::Rect2D { offset: vk::Offset2D { x: 0, y: 0 }, extent: output.output.size.into(), }); unsafe { self.device.cmd_begin_render_pass( cmd, &render_pass_info, vk::SubpassContents::INLINE, ); } Ok(Some(framebuffer)) } else { let attachments = vk::RenderingAttachmentInfo::builder() .load_op(vk::AttachmentLoadOp::DONT_CARE) .store_op(vk::AttachmentStoreOp::STORE) .image_layout(vk::ImageLayout::COLOR_ATTACHMENT_OPTIMAL) .image_view(output.output.image_view); let attachments = [*attachments]; let rendering_info = vk::RenderingInfo::builder() .layer_count(1) .render_area(vk::Rect2D { offset: vk::Offset2D { x: 0, y: 0 }, extent: output.output.size.into(), }) .color_attachments(&attachments); unsafe { self.device.cmd_begin_rendering(cmd, &rendering_info); } Ok(None) } } pub(crate) fn end_rendering(&self, cmd: vk::CommandBuffer) { unsafe { if self.render_pass.is_none() { self.device.cmd_end_rendering(cmd); } else { self.device.cmd_end_render_pass(cmd) } } } } impl Drop for VulkanGraphicsPipeline { fn drop(&mut self) { unsafe { if self.pipeline != vk::Pipeline::null() { self.device.destroy_pipeline(self.pipeline, None) } if self.cache != vk::PipelineCache::null() { self.device.destroy_pipeline_cache(self.cache, None) } } } }