mirror of
https://github.com/vulkano-rs/vulkano.git
synced 2024-11-25 16:25:31 +00:00
4c515a81cb
* Make each example its own workspace member * Fix runtime-shader example * Fix shader-include example * Fix teapot example * Fix `run_all.sh` * Fix output files getting saved in cwd * Fix spelling for examples readme filename * Remove wrong leftover dependencies for gl-interop * Fix pipeline-cache example * Clearer .gitignore * Help cargo be less useless
294 lines
9.9 KiB
Rust
294 lines
9.9 KiB
Rust
// Copyright (c) 2021 The vulkano developers
|
|
// Licensed under the Apache License, Version 2.0
|
|
// <LICENSE-APACHE or
|
|
// https://www.apache.org/licenses/LICENSE-2.0> or the MIT
|
|
// license <LICENSE-MIT or https://opensource.org/licenses/MIT>,
|
|
// at your option. All files in the project carrying such
|
|
// notice may not be copied, modified, or distributed except
|
|
// according to those terms.
|
|
|
|
use std::sync::Arc;
|
|
use vulkano::{
|
|
buffer::{Buffer, BufferContents, BufferCreateInfo, BufferUsage, Subbuffer},
|
|
command_buffer::{
|
|
allocator::StandardCommandBufferAllocator, AutoCommandBufferBuilder,
|
|
CommandBufferInheritanceInfo, CommandBufferUsage, SecondaryAutoCommandBuffer,
|
|
},
|
|
descriptor_set::{
|
|
allocator::StandardDescriptorSetAllocator, PersistentDescriptorSet, WriteDescriptorSet,
|
|
},
|
|
device::Queue,
|
|
image::{
|
|
sampler::{Filter, Sampler, SamplerAddressMode, SamplerCreateInfo, SamplerMipmapMode},
|
|
view::ImageView,
|
|
},
|
|
memory::allocator::{AllocationCreateInfo, MemoryTypeFilter, StandardMemoryAllocator},
|
|
pipeline::{
|
|
graphics::{
|
|
color_blend::{ColorBlendAttachmentState, ColorBlendState},
|
|
input_assembly::InputAssemblyState,
|
|
multisample::MultisampleState,
|
|
rasterization::RasterizationState,
|
|
vertex_input::{Vertex, VertexDefinition},
|
|
viewport::{Viewport, ViewportState},
|
|
GraphicsPipelineCreateInfo,
|
|
},
|
|
layout::PipelineDescriptorSetLayoutCreateInfo,
|
|
DynamicState, GraphicsPipeline, Pipeline, PipelineBindPoint, PipelineLayout,
|
|
PipelineShaderStageCreateInfo,
|
|
},
|
|
render_pass::Subpass,
|
|
};
|
|
|
|
/// Vertex for textured quads.
|
|
#[derive(BufferContents, Vertex)]
|
|
#[repr(C)]
|
|
pub struct TexturedVertex {
|
|
#[format(R32G32_SFLOAT)]
|
|
pub position: [f32; 2],
|
|
#[format(R32G32_SFLOAT)]
|
|
pub tex_coords: [f32; 2],
|
|
}
|
|
|
|
pub fn textured_quad(width: f32, height: f32) -> (Vec<TexturedVertex>, Vec<u32>) {
|
|
(
|
|
vec![
|
|
TexturedVertex {
|
|
position: [-(width / 2.0), -(height / 2.0)],
|
|
tex_coords: [0.0, 1.0],
|
|
},
|
|
TexturedVertex {
|
|
position: [-(width / 2.0), height / 2.0],
|
|
tex_coords: [0.0, 0.0],
|
|
},
|
|
TexturedVertex {
|
|
position: [width / 2.0, height / 2.0],
|
|
tex_coords: [1.0, 0.0],
|
|
},
|
|
TexturedVertex {
|
|
position: [width / 2.0, -(height / 2.0)],
|
|
tex_coords: [1.0, 1.0],
|
|
},
|
|
],
|
|
vec![0, 2, 1, 0, 3, 2],
|
|
)
|
|
}
|
|
|
|
/// A subpass pipeline that fills a quad over frame.
|
|
pub struct PixelsDrawPipeline {
|
|
gfx_queue: Arc<Queue>,
|
|
subpass: Subpass,
|
|
pipeline: Arc<GraphicsPipeline>,
|
|
command_buffer_allocator: Arc<StandardCommandBufferAllocator>,
|
|
descriptor_set_allocator: Arc<StandardDescriptorSetAllocator>,
|
|
vertices: Subbuffer<[TexturedVertex]>,
|
|
indices: Subbuffer<[u32]>,
|
|
}
|
|
|
|
impl PixelsDrawPipeline {
|
|
pub fn new(
|
|
gfx_queue: Arc<Queue>,
|
|
subpass: Subpass,
|
|
memory_allocator: Arc<StandardMemoryAllocator>,
|
|
command_buffer_allocator: Arc<StandardCommandBufferAllocator>,
|
|
descriptor_set_allocator: Arc<StandardDescriptorSetAllocator>,
|
|
) -> PixelsDrawPipeline {
|
|
let (vertices, indices) = textured_quad(2.0, 2.0);
|
|
let vertex_buffer = Buffer::from_iter(
|
|
memory_allocator.clone(),
|
|
BufferCreateInfo {
|
|
usage: BufferUsage::VERTEX_BUFFER,
|
|
..Default::default()
|
|
},
|
|
AllocationCreateInfo {
|
|
memory_type_filter: MemoryTypeFilter::PREFER_DEVICE
|
|
| MemoryTypeFilter::HOST_SEQUENTIAL_WRITE,
|
|
..Default::default()
|
|
},
|
|
vertices,
|
|
)
|
|
.unwrap();
|
|
let index_buffer = Buffer::from_iter(
|
|
memory_allocator,
|
|
BufferCreateInfo {
|
|
usage: BufferUsage::INDEX_BUFFER,
|
|
..Default::default()
|
|
},
|
|
AllocationCreateInfo {
|
|
memory_type_filter: MemoryTypeFilter::PREFER_DEVICE
|
|
| MemoryTypeFilter::HOST_SEQUENTIAL_WRITE,
|
|
..Default::default()
|
|
},
|
|
indices,
|
|
)
|
|
.unwrap();
|
|
|
|
let pipeline = {
|
|
let device = gfx_queue.device();
|
|
let vs = vs::load(device.clone())
|
|
.expect("failed to create shader module")
|
|
.entry_point("main")
|
|
.expect("shader entry point not found");
|
|
let fs = fs::load(device.clone())
|
|
.expect("failed to create shader module")
|
|
.entry_point("main")
|
|
.expect("shader entry point not found");
|
|
let vertex_input_state = TexturedVertex::per_vertex()
|
|
.definition(&vs.info().input_interface)
|
|
.unwrap();
|
|
let stages = [
|
|
PipelineShaderStageCreateInfo::new(vs),
|
|
PipelineShaderStageCreateInfo::new(fs),
|
|
];
|
|
let layout = PipelineLayout::new(
|
|
device.clone(),
|
|
PipelineDescriptorSetLayoutCreateInfo::from_stages(&stages)
|
|
.into_pipeline_layout_create_info(device.clone())
|
|
.unwrap(),
|
|
)
|
|
.unwrap();
|
|
|
|
GraphicsPipeline::new(
|
|
device.clone(),
|
|
None,
|
|
GraphicsPipelineCreateInfo {
|
|
stages: stages.into_iter().collect(),
|
|
vertex_input_state: Some(vertex_input_state),
|
|
input_assembly_state: Some(InputAssemblyState::default()),
|
|
viewport_state: Some(ViewportState::default()),
|
|
rasterization_state: Some(RasterizationState::default()),
|
|
multisample_state: Some(MultisampleState::default()),
|
|
color_blend_state: Some(ColorBlendState::with_attachment_states(
|
|
subpass.num_color_attachments(),
|
|
ColorBlendAttachmentState::default(),
|
|
)),
|
|
dynamic_state: [DynamicState::Viewport].into_iter().collect(),
|
|
subpass: Some(subpass.clone().into()),
|
|
..GraphicsPipelineCreateInfo::layout(layout)
|
|
},
|
|
)
|
|
.unwrap()
|
|
};
|
|
|
|
PixelsDrawPipeline {
|
|
gfx_queue,
|
|
subpass,
|
|
pipeline,
|
|
command_buffer_allocator,
|
|
descriptor_set_allocator,
|
|
vertices: vertex_buffer,
|
|
indices: index_buffer,
|
|
}
|
|
}
|
|
|
|
fn create_descriptor_set(&self, image: Arc<ImageView>) -> Arc<PersistentDescriptorSet> {
|
|
let layout = self.pipeline.layout().set_layouts().get(0).unwrap();
|
|
let sampler = Sampler::new(
|
|
self.gfx_queue.device().clone(),
|
|
SamplerCreateInfo {
|
|
mag_filter: Filter::Linear,
|
|
min_filter: Filter::Linear,
|
|
address_mode: [SamplerAddressMode::Repeat; 3],
|
|
mipmap_mode: SamplerMipmapMode::Linear,
|
|
..Default::default()
|
|
},
|
|
)
|
|
.unwrap();
|
|
|
|
PersistentDescriptorSet::new(
|
|
&self.descriptor_set_allocator,
|
|
layout.clone(),
|
|
[
|
|
WriteDescriptorSet::sampler(0, sampler),
|
|
WriteDescriptorSet::image_view(1, image),
|
|
],
|
|
[],
|
|
)
|
|
.unwrap()
|
|
}
|
|
|
|
/// Draws input `image` over a quad of size -1.0 to 1.0.
|
|
pub fn draw(
|
|
&self,
|
|
viewport_dimensions: [u32; 2],
|
|
image: Arc<ImageView>,
|
|
) -> Arc<SecondaryAutoCommandBuffer> {
|
|
let mut builder = AutoCommandBufferBuilder::secondary(
|
|
self.command_buffer_allocator.as_ref(),
|
|
self.gfx_queue.queue_family_index(),
|
|
CommandBufferUsage::MultipleSubmit,
|
|
CommandBufferInheritanceInfo {
|
|
render_pass: Some(self.subpass.clone().into()),
|
|
..Default::default()
|
|
},
|
|
)
|
|
.unwrap();
|
|
let desc_set = self.create_descriptor_set(image);
|
|
builder
|
|
.set_viewport(
|
|
0,
|
|
[Viewport {
|
|
offset: [0.0, 0.0],
|
|
extent: [viewport_dimensions[0] as f32, viewport_dimensions[1] as f32],
|
|
depth_range: 0.0..=1.0,
|
|
}]
|
|
.into_iter()
|
|
.collect(),
|
|
)
|
|
.unwrap()
|
|
.bind_pipeline_graphics(self.pipeline.clone())
|
|
.unwrap()
|
|
.bind_descriptor_sets(
|
|
PipelineBindPoint::Graphics,
|
|
self.pipeline.layout().clone(),
|
|
0,
|
|
desc_set,
|
|
)
|
|
.unwrap()
|
|
.bind_vertex_buffers(0, self.vertices.clone())
|
|
.unwrap()
|
|
.bind_index_buffer(self.indices.clone())
|
|
.unwrap()
|
|
.draw_indexed(self.indices.len() as u32, 1, 0, 0, 0)
|
|
.unwrap();
|
|
builder.build().unwrap()
|
|
}
|
|
}
|
|
|
|
mod vs {
|
|
vulkano_shaders::shader! {
|
|
ty: "vertex",
|
|
src: r"
|
|
#version 450
|
|
layout(location=0) in vec2 position;
|
|
layout(location=1) in vec2 tex_coords;
|
|
|
|
layout(location = 0) out vec2 f_tex_coords;
|
|
|
|
void main() {
|
|
gl_Position = vec4(position, 0.0, 1.0);
|
|
f_tex_coords = tex_coords;
|
|
}
|
|
",
|
|
}
|
|
}
|
|
|
|
mod fs {
|
|
vulkano_shaders::shader! {
|
|
ty: "fragment",
|
|
src: r"
|
|
#version 450
|
|
layout(location = 0) in vec2 v_tex_coords;
|
|
|
|
layout(location = 0) out vec4 f_color;
|
|
|
|
layout(set = 0, binding = 0) uniform sampler s;
|
|
layout(set = 0, binding = 1) uniform texture2D tex;
|
|
|
|
void main() {
|
|
f_color = texture(sampler2D(tex, s), v_tex_coords);
|
|
}
|
|
",
|
|
}
|
|
}
|