2022-12-22 13:39:31 +11:00
|
|
|
use crate::{error, util};
|
2022-12-10 17:25:36 +11:00
|
|
|
use ash::vk;
|
2023-01-13 17:48:04 +11:00
|
|
|
|
2023-02-08 15:08:44 +11:00
|
|
|
use crate::framebuffer::OutputImage;
|
2023-01-29 18:30:58 +11:00
|
|
|
use crate::render_pass::VulkanRenderPass;
|
2022-12-10 17:25:36 +11:00
|
|
|
use librashader_reflect::back::ShaderCompilerOutput;
|
|
|
|
use librashader_reflect::reflect::semantics::{TextureBinding, UboReflection};
|
|
|
|
use librashader_reflect::reflect::ShaderReflection;
|
2023-02-08 15:08:44 +11:00
|
|
|
use librashader_runtime::render_target::RenderTarget;
|
2022-12-22 13:39:31 +11:00
|
|
|
use std::ffi::CStr;
|
2022-12-10 17:25:36 +11:00
|
|
|
|
2023-01-16 03:08:13 +11:00
|
|
|
const ENTRY_POINT: &CStr = unsafe { CStr::from_bytes_with_nul_unchecked(b"main\0") };
|
2023-01-13 16:07:03 +11:00
|
|
|
|
2022-12-10 17:25:36 +11:00
|
|
|
pub struct PipelineDescriptors {
|
|
|
|
pub replicas: u32,
|
|
|
|
pub layout_bindings: Vec<vk::DescriptorSetLayoutBinding>,
|
2022-12-22 13:39:31 +11:00
|
|
|
pub pool_sizes: Vec<vk::DescriptorPoolSize>,
|
2022-12-10 17:25:36 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
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<&UboReflection>) {
|
|
|
|
if let Some(ubo_meta) = ubo_meta && !ubo_meta.stage_mask.is_empty() {
|
2023-01-12 10:04:35 +11:00
|
|
|
let ubo_mask = util::binding_stage_to_vulkan_stage(ubo_meta.stage_mask);
|
2022-12-10 17:25:36 +11:00
|
|
|
|
|
|
|
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<Item = &'a TextureBinding>) {
|
2023-01-12 10:04:35 +11:00
|
|
|
let texture_mask = vk::ShaderStageFlags::FRAGMENT;
|
2022-12-10 17:25:36 +11:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
2022-12-22 13:39:31 +11:00
|
|
|
pub fn create_descriptor_set_layout(
|
|
|
|
&self,
|
|
|
|
device: &ash::Device,
|
|
|
|
) -> error::Result<vk::DescriptorSetLayout> {
|
2022-12-10 17:25:36 +11:00
|
|
|
unsafe {
|
|
|
|
let layout = device.create_descriptor_set_layout(
|
|
|
|
&vk::DescriptorSetLayoutCreateInfo::builder()
|
|
|
|
.bindings(self.bindings())
|
|
|
|
.build(),
|
2022-12-22 13:39:31 +11:00
|
|
|
None,
|
|
|
|
)?;
|
2022-12-10 17:25:36 +11:00
|
|
|
Ok(layout)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct PipelineLayoutObjects {
|
|
|
|
pub layout: vk::PipelineLayout,
|
|
|
|
pub pool: vk::DescriptorPool,
|
|
|
|
pub descriptor_sets: Vec<vk::DescriptorSet>,
|
|
|
|
pub descriptor_set_layout: [vk::DescriptorSetLayout; 1],
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PipelineLayoutObjects {
|
2022-12-22 13:39:31 +11:00
|
|
|
pub fn new(
|
|
|
|
reflection: &ShaderReflection,
|
|
|
|
replicas: u32,
|
|
|
|
device: &ash::Device,
|
|
|
|
) -> error::Result<Self> {
|
2022-12-10 17:25:36 +11:00
|
|
|
let mut descriptors = PipelineDescriptors::new(replicas);
|
|
|
|
descriptors.add_ubo_binding(reflection.ubo.as_ref());
|
|
|
|
descriptors.add_texture_bindings(reflection.meta.texture_meta.values());
|
|
|
|
|
2023-01-13 17:48:04 +11:00
|
|
|
let descriptor_set_layout = [descriptors.create_descriptor_set_layout(device)?];
|
2022-12-10 17:25:36 +11:00
|
|
|
|
2023-01-12 10:04:35 +11:00
|
|
|
let pipeline_create_info =
|
2022-12-22 13:39:31 +11:00
|
|
|
vk::PipelineLayoutCreateInfo::builder().set_layouts(&descriptor_set_layout);
|
2022-12-10 17:25:36 +11:00
|
|
|
|
|
|
|
let pipeline_create_info = if let Some(push_constant) = &reflection.push_constant {
|
2023-01-12 10:04:35 +11:00
|
|
|
let stage_mask = util::binding_stage_to_vulkan_stage(push_constant.stage_mask);
|
2022-12-22 13:39:31 +11:00
|
|
|
let push_constant_range = [vk::PushConstantRange::builder()
|
|
|
|
.stage_flags(stage_mask)
|
|
|
|
.size(push_constant.size)
|
|
|
|
.build()];
|
|
|
|
pipeline_create_info
|
|
|
|
.push_constant_ranges(&push_constant_range)
|
|
|
|
.build()
|
2022-12-10 17:25:36 +11:00
|
|
|
} else {
|
|
|
|
pipeline_create_info.build()
|
|
|
|
};
|
|
|
|
|
2022-12-22 13:39:31 +11:00
|
|
|
let layout = unsafe { device.create_pipeline_layout(&pipeline_create_info, None)? };
|
2022-12-10 17:25:36 +11:00
|
|
|
|
2023-01-13 16:07:03 +11:00
|
|
|
let pool_info = vk::DescriptorPoolCreateInfo::builder()
|
|
|
|
.max_sets(replicas)
|
|
|
|
.pool_sizes(&descriptors.pool_sizes)
|
|
|
|
.build();
|
|
|
|
|
2023-01-13 17:48:04 +11:00
|
|
|
let pool = unsafe { device.create_descriptor_pool(&pool_info, None)? };
|
2022-12-10 17:25:36 +11:00
|
|
|
|
|
|
|
let mut descriptor_sets = Vec::new();
|
|
|
|
let alloc_info = vk::DescriptorSetAllocateInfo::builder()
|
|
|
|
.descriptor_pool(pool)
|
|
|
|
.set_layouts(&descriptor_set_layout)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
for _ in 0..replicas {
|
2023-01-13 16:07:03 +11:00
|
|
|
let set = unsafe { device.allocate_descriptor_sets(&alloc_info)? };
|
|
|
|
descriptor_sets.push(set)
|
2022-12-10 17:25:36 +11:00
|
|
|
}
|
|
|
|
|
2022-12-22 13:39:31 +11:00
|
|
|
let descriptor_sets: Vec<vk::DescriptorSet> =
|
|
|
|
descriptor_sets.into_iter().flatten().collect();
|
2022-12-10 17:25:36 +11:00
|
|
|
|
2023-01-16 03:08:13 +11:00
|
|
|
Ok(PipelineLayoutObjects {
|
2022-12-10 17:25:36 +11:00
|
|
|
layout,
|
|
|
|
descriptor_set_layout,
|
|
|
|
descriptor_sets,
|
|
|
|
pool,
|
2023-01-16 03:08:13 +11:00
|
|
|
})
|
2022-12-10 17:25:36 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct VulkanShaderModule {
|
|
|
|
shader: vk::ShaderModule,
|
|
|
|
device: ash::Device,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl VulkanShaderModule {
|
2022-12-22 13:39:31 +11:00
|
|
|
pub fn new(
|
|
|
|
device: &ash::Device,
|
|
|
|
info: &vk::ShaderModuleCreateInfo,
|
|
|
|
) -> error::Result<VulkanShaderModule> {
|
2022-12-10 17:25:36 +11:00
|
|
|
Ok(VulkanShaderModule {
|
|
|
|
shader: unsafe { device.create_shader_module(info, None)? },
|
2022-12-22 13:39:31 +11:00
|
|
|
device: device.clone(),
|
2022-12-10 17:25:36 +11:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for VulkanShaderModule {
|
|
|
|
fn drop(&mut self) {
|
2022-12-22 13:39:31 +11:00
|
|
|
unsafe { self.device.destroy_shader_module(self.shader, None) }
|
2022-12-10 17:25:36 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct VulkanGraphicsPipeline {
|
2022-12-29 16:50:48 +11:00
|
|
|
pub layout: PipelineLayoutObjects,
|
2023-01-10 16:45:48 +11:00
|
|
|
pub pipeline: vk::Pipeline,
|
2023-01-29 18:30:58 +11:00
|
|
|
pub render_pass: Option<VulkanRenderPass>,
|
2022-12-10 17:25:36 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl VulkanGraphicsPipeline {
|
2022-12-22 13:39:31 +11:00
|
|
|
pub fn new(
|
|
|
|
device: &ash::Device,
|
|
|
|
cache: &vk::PipelineCache,
|
|
|
|
shader_assembly: &ShaderCompilerOutput<Vec<u32>>,
|
|
|
|
reflection: &ShaderReflection,
|
|
|
|
replicas: u32,
|
2023-01-29 18:30:58 +11:00
|
|
|
render_pass_format: vk::Format,
|
2022-12-22 13:39:31 +11:00
|
|
|
) -> error::Result<VulkanGraphicsPipeline> {
|
2023-01-16 03:08:13 +11:00
|
|
|
let pipeline_layout = PipelineLayoutObjects::new(reflection, replicas, device)?;
|
2022-12-10 17:25:36 +11:00
|
|
|
|
|
|
|
let input_assembly = vk::PipelineInputAssemblyStateCreateInfo::builder()
|
|
|
|
.topology(vk::PrimitiveTopology::TRIANGLE_STRIP)
|
|
|
|
.build();
|
|
|
|
|
2022-12-22 13:39:31 +11:00
|
|
|
let vao_state = [
|
|
|
|
vk::VertexInputAttributeDescription {
|
|
|
|
location: 0,
|
|
|
|
binding: 0,
|
|
|
|
format: vk::Format::R32G32_SFLOAT,
|
|
|
|
offset: 0,
|
|
|
|
},
|
|
|
|
vk::VertexInputAttributeDescription {
|
|
|
|
location: 1,
|
|
|
|
binding: 0,
|
|
|
|
format: vk::Format::R32G32_SFLOAT,
|
|
|
|
offset: (2 * std::mem::size_of::<f32>()) as u32,
|
|
|
|
},
|
|
|
|
];
|
2022-12-10 17:25:36 +11:00
|
|
|
|
2023-01-13 16:07:03 +11:00
|
|
|
let input_binding = [vk::VertexInputBindingDescription::builder()
|
2022-12-10 17:25:36 +11:00
|
|
|
.binding(0)
|
|
|
|
.stride(4 * std::mem::size_of::<f32>() as u32)
|
|
|
|
.input_rate(vk::VertexInputRate::VERTEX)
|
2023-01-13 16:07:03 +11:00
|
|
|
.build()];
|
2022-12-10 17:25:36 +11:00
|
|
|
|
|
|
|
let pipeline_input_state = vk::PipelineVertexInputStateCreateInfo::builder()
|
2023-01-13 16:07:03 +11:00
|
|
|
.vertex_binding_descriptions(&input_binding)
|
2022-12-10 17:25:36 +11:00
|
|
|
.vertex_attribute_descriptions(&vao_state)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
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)
|
|
|
|
.build();
|
|
|
|
|
2023-01-13 16:07:03 +11:00
|
|
|
let attachments = [vk::PipelineColorBlendAttachmentState::builder()
|
|
|
|
.blend_enable(false)
|
|
|
|
.color_write_mask(vk::ColorComponentFlags::from_raw(0xf))
|
|
|
|
.build()];
|
|
|
|
|
2022-12-10 17:25:36 +11:00
|
|
|
let blend_state = vk::PipelineColorBlendStateCreateInfo::builder()
|
2023-01-13 16:07:03 +11:00
|
|
|
.attachments(&attachments)
|
2022-12-10 17:25:36 +11:00
|
|
|
.build();
|
|
|
|
|
|
|
|
let viewport_state = vk::PipelineViewportStateCreateInfo::builder()
|
|
|
|
.viewport_count(1)
|
|
|
|
.scissor_count(1)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
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)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let multisample_state = vk::PipelineMultisampleStateCreateInfo::builder()
|
|
|
|
.rasterization_samples(vk::SampleCountFlags::TYPE_1)
|
|
|
|
.build();
|
|
|
|
|
2023-01-13 16:07:03 +11:00
|
|
|
let states = [vk::DynamicState::VIEWPORT, vk::DynamicState::SCISSOR];
|
2022-12-10 17:25:36 +11:00
|
|
|
let dynamic_state = vk::PipelineDynamicStateCreateInfo::builder()
|
2023-01-13 16:07:03 +11:00
|
|
|
.dynamic_states(&states)
|
2022-12-10 17:25:36 +11:00
|
|
|
.build();
|
|
|
|
|
|
|
|
let vertex_info = vk::ShaderModuleCreateInfo::builder()
|
|
|
|
.code(shader_assembly.vertex.as_ref())
|
|
|
|
.build();
|
|
|
|
let fragment_info = vk::ShaderModuleCreateInfo::builder()
|
|
|
|
.code(shader_assembly.fragment.as_ref())
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let vertex_module = VulkanShaderModule::new(device, &vertex_info)?;
|
|
|
|
let fragment_module = VulkanShaderModule::new(device, &fragment_info)?;
|
|
|
|
|
|
|
|
let shader_stages = [
|
|
|
|
vk::PipelineShaderStageCreateInfo::builder()
|
|
|
|
.stage(vk::ShaderStageFlags::VERTEX)
|
2023-01-16 03:08:13 +11:00
|
|
|
.name(ENTRY_POINT)
|
|
|
|
.module(vertex_module.shader)
|
2022-12-10 17:25:36 +11:00
|
|
|
.build(),
|
|
|
|
vk::PipelineShaderStageCreateInfo::builder()
|
|
|
|
.stage(vk::ShaderStageFlags::FRAGMENT)
|
2023-01-16 03:08:13 +11:00
|
|
|
.name(ENTRY_POINT)
|
|
|
|
.module(fragment_module.shader)
|
2022-12-10 17:25:36 +11:00
|
|
|
.build(),
|
|
|
|
];
|
|
|
|
|
2023-01-26 15:45:10 +11:00
|
|
|
let mut pipeline_info = vk::GraphicsPipelineCreateInfo::builder()
|
2022-12-10 17:25:36 +11:00
|
|
|
.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)
|
2023-01-26 15:45:10 +11:00
|
|
|
.layout(pipeline_layout.layout);
|
|
|
|
|
|
|
|
let mut render_pass = None;
|
|
|
|
if render_pass_format != vk::Format::UNDEFINED {
|
2023-01-29 18:30:58 +11:00
|
|
|
render_pass = Some(VulkanRenderPass::create_render_pass(
|
|
|
|
device,
|
|
|
|
render_pass_format,
|
|
|
|
)?);
|
2023-01-26 15:45:10 +11:00
|
|
|
pipeline_info = pipeline_info.render_pass(render_pass.as_ref().unwrap().handle)
|
|
|
|
}
|
|
|
|
|
|
|
|
let pipeline_info = pipeline_info.build();
|
2022-12-10 17:25:36 +11:00
|
|
|
|
|
|
|
let pipeline = unsafe {
|
|
|
|
// panic_safety: if this is successful this should return 1 pipelines.
|
2022-12-22 13:39:31 +11:00
|
|
|
device
|
2023-01-16 03:08:13 +11:00
|
|
|
.create_graphics_pipelines(*cache, &[pipeline_info], None)
|
2022-12-22 13:39:31 +11:00
|
|
|
.map_err(|e| e.1)
|
|
|
|
.unwrap()[0]
|
2022-12-10 17:25:36 +11:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(VulkanGraphicsPipeline {
|
|
|
|
layout: pipeline_layout,
|
2022-12-22 13:39:31 +11:00
|
|
|
pipeline,
|
2023-01-26 15:45:10 +11:00
|
|
|
render_pass,
|
2022-12-10 17:25:36 +11:00
|
|
|
})
|
|
|
|
}
|
2023-01-26 15:45:10 +11:00
|
|
|
|
|
|
|
#[inline(always)]
|
2023-01-29 18:30:58 +11:00
|
|
|
pub(crate) fn begin_rendering(
|
|
|
|
&self,
|
|
|
|
device: &ash::Device,
|
2023-02-08 15:08:44 +11:00
|
|
|
output: &RenderTarget<OutputImage>,
|
2023-01-29 18:30:58 +11:00
|
|
|
cmd: vk::CommandBuffer,
|
|
|
|
) -> error::Result<Option<vk::Framebuffer>> {
|
2023-01-26 15:45:10 +11:00
|
|
|
if let Some(render_pass) = &self.render_pass {
|
|
|
|
let attachments = [output.output.image_view];
|
|
|
|
let framebuffer = unsafe {
|
|
|
|
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)
|
|
|
|
.build(),
|
|
|
|
None,
|
|
|
|
)?
|
|
|
|
};
|
|
|
|
|
|
|
|
let clear_values = [vk::ClearValue {
|
|
|
|
color: vk::ClearColorValue {
|
2023-01-29 18:30:58 +11:00
|
|
|
float32: [0.0, 0.0, 0.0, 0.0],
|
|
|
|
},
|
2023-01-26 15:45:10 +11:00
|
|
|
}];
|
|
|
|
|
|
|
|
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 {
|
2023-01-29 18:30:58 +11:00
|
|
|
offset: vk::Offset2D { x: 0, y: 0 },
|
2023-01-26 15:45:10 +11:00
|
|
|
extent: output.output.size.into(),
|
2023-01-29 18:30:58 +11:00
|
|
|
})
|
|
|
|
.build();
|
2023-01-26 15:45:10 +11:00
|
|
|
unsafe {
|
|
|
|
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)
|
|
|
|
.build()];
|
|
|
|
|
|
|
|
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)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
device.cmd_begin_rendering(cmd, &rendering_info);
|
|
|
|
}
|
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn end_rendering(&self, device: &ash::Device, cmd: vk::CommandBuffer) {
|
|
|
|
unsafe {
|
|
|
|
if self.render_pass.is_none() {
|
|
|
|
device.cmd_end_rendering(cmd);
|
|
|
|
} else {
|
|
|
|
device.cmd_end_render_pass(cmd)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-12-22 13:39:31 +11:00
|
|
|
}
|