Export triangle to external scene
Some checks failed
Build legacy Nix package on Ubuntu / build (push) Failing after 1s

This commit is contained in:
Florian RICHER 2024-11-27 22:16:26 +01:00
parent 4b08b7359d
commit 7b5cae8322
Signed by: florian.richer
GPG key ID: C73D37CBED7BFC77
18 changed files with 121 additions and 65 deletions

View file

@ -1,13 +1,15 @@
use crate::display::window::Window; use crate::display::window::Window;
use crate::renderer::vulkan::VkRenderContext; use crate::renderer::{vulkan::VkRenderContext, Renderable};
use winit::application::ApplicationHandler; use winit::application::ApplicationHandler;
use winit::event::WindowEvent; use winit::event::WindowEvent;
use winit::event_loop::ActiveEventLoop; use winit::event_loop::ActiveEventLoop;
use winit::window::WindowId; use winit::window::WindowId;
use crate::scene::TriangleScene;
pub struct App { pub struct App {
window: Window, window: Window,
render_context: Option<VkRenderContext>, render_context: Option<VkRenderContext>,
scene: Option<Box<dyn Renderable>>,
} }
impl App { impl App {
@ -15,6 +17,18 @@ impl App {
Self { Self {
window, window,
render_context: None, render_context: None,
scene: None,
}
}
pub fn set_scene(&mut self, mut scene: Box<dyn Renderable>) {
let result = self.render_context.as_mut()
.ok_or_else(|| anyhow::anyhow!("No render context"))
.and_then(|render_context| render_context.init_scene(&mut scene));
match result {
Ok(_) => self.scene = Some(scene),
Err(err) => log::warn!("{err}"),
} }
} }
} }
@ -27,6 +41,9 @@ impl ApplicationHandler for App {
.unwrap(); .unwrap();
self.render_context = VkRenderContext::init(&self.window).ok(); self.render_context = VkRenderContext::init(&self.window).ok();
let scene = TriangleScene::new();
self.set_scene(Box::new(scene));
} }
fn window_event(&mut self, event_loop: &ActiveEventLoop, _id: WindowId, event: WindowEvent) { fn window_event(&mut self, event_loop: &ActiveEventLoop, _id: WindowId, event: WindowEvent) {
@ -59,7 +76,7 @@ impl ApplicationHandler for App {
if !event_loop.exiting() { if !event_loop.exiting() {
match self.render_context.as_mut() { match self.render_context.as_mut() {
Some(render_context) => { Some(render_context) => {
if let Err(error) = render_context.render() { if let Err(error) = render_context.render(self.scene.as_ref()) {
log::error!("Failed to render with render context : {}", error); log::error!("Failed to render with render context : {}", error);
event_loop.exit(); event_loop.exit();
} }

View file

@ -2,6 +2,7 @@ use winit::event_loop::{ControlFlow, EventLoop};
mod display; mod display;
mod renderer; mod renderer;
mod scene;
fn main() { fn main() {
env_logger::init(); env_logger::init();

View file

@ -1,7 +1,9 @@
use std::sync::Arc;
use ash::vk; use ash::vk;
pub mod vulkan; pub mod vulkan;
pub trait Renderable { pub trait Renderable {
fn render(device: &vulkan::VkDevice, command_buffer: vk::CommandBuffer); fn init(&mut self, device: &Arc<vulkan::VkDevice>, render_pass: &Arc<vulkan::VkRenderPass>) -> anyhow::Result<()>;
fn render(&self, device: &vulkan::VkDevice, swapchain: &vulkan::VkSwapchain, command_buffer: &vk::CommandBuffer) -> anyhow::Result<()>;
} }

View file

@ -6,7 +6,7 @@ use std::sync::Arc;
pub struct VkCommandPool { pub struct VkCommandPool {
device: Arc<VkDevice>, device: Arc<VkDevice>,
pub(super) handle: vk::CommandPool, pub handle: vk::CommandPool,
} }
impl VkCommandPool { impl VkCommandPool {

View file

@ -5,9 +5,9 @@ use std::sync::Arc;
pub struct VkDevice { pub struct VkDevice {
instance: Arc<VkInstance>, instance: Arc<VkInstance>,
pub(super) handle: ash::Device, pub handle: ash::Device,
pub(super) swapchain_loader: ash::khr::swapchain::Device, pub swapchain_loader: ash::khr::swapchain::Device,
pub(super) queue_family_index: u32, pub queue_family_index: u32,
// Arc not used because vk::Queue is destroyed with Device automatically // Arc not used because vk::Queue is destroyed with Device automatically
// so any references of vk::Queue must be destroyed with VkDevice // so any references of vk::Queue must be destroyed with VkDevice
@ -15,7 +15,7 @@ pub struct VkDevice {
} }
impl VkDevice { impl VkDevice {
pub(super) fn new_graphics_device( pub fn new_graphics_device(
instance: &Arc<VkInstance>, instance: &Arc<VkInstance>,
physical_device: &VkPhysicalDevice, physical_device: &VkPhysicalDevice,
queue_family_index: u32, queue_family_index: u32,
@ -64,11 +64,11 @@ impl VkDevice {
}) })
} }
pub(super) fn get_device_queue(&self, queue_index: u32) -> Option<&vk::Queue> { pub fn get_device_queue(&self, queue_index: u32) -> Option<&vk::Queue> {
self.queues.get(queue_index as usize) self.queues.get(queue_index as usize)
} }
pub(super) fn create_command_pool( pub fn create_command_pool(
&self, &self,
info: &vk::CommandPoolCreateInfo, info: &vk::CommandPoolCreateInfo,
) -> VkResult<vk::CommandPool> { ) -> VkResult<vk::CommandPool> {
@ -77,18 +77,18 @@ impl VkDevice {
unsafe { self.handle.create_command_pool(&info, None) } unsafe { self.handle.create_command_pool(&info, None) }
} }
pub(super) fn allocate_command_buffers( pub fn allocate_command_buffers(
&self, &self,
info: &vk::CommandBufferAllocateInfo, info: &vk::CommandBufferAllocateInfo,
) -> VkResult<Vec<vk::CommandBuffer>> { ) -> VkResult<Vec<vk::CommandBuffer>> {
unsafe { self.handle.allocate_command_buffers(&info) } unsafe { self.handle.allocate_command_buffers(&info) }
} }
pub(super) fn create_fence(&self, info: &vk::FenceCreateInfo) -> VkResult<vk::Fence> { pub fn create_fence(&self, info: &vk::FenceCreateInfo) -> VkResult<vk::Fence> {
unsafe { self.handle.create_fence(&info, None) } unsafe { self.handle.create_fence(&info, None) }
} }
pub(super) fn create_semaphore( pub fn create_semaphore(
&self, &self,
info: &vk::SemaphoreCreateInfo, info: &vk::SemaphoreCreateInfo,
) -> VkResult<vk::Semaphore> { ) -> VkResult<vk::Semaphore> {

View file

@ -5,7 +5,7 @@ use std::sync::Arc;
pub struct VkFence { pub struct VkFence {
device: Arc<VkDevice>, device: Arc<VkDevice>,
pub(super) handle: vk::Fence, pub handle: vk::Fence,
} }
impl VkFence { impl VkFence {

View file

@ -7,7 +7,7 @@ pub struct VkFramebuffer {
image_view: Arc<vk::ImageView>, image_view: Arc<vk::ImageView>,
render_pass: Arc<VkRenderPass>, render_pass: Arc<VkRenderPass>,
pub(super) handle: vk::Framebuffer, pub handle: vk::Framebuffer,
} }
impl VkFramebuffer { impl VkFramebuffer {

View file

@ -7,8 +7,8 @@ pub struct VkGraphicsPipeline {
device: Arc<VkDevice>, device: Arc<VkDevice>,
render_pass: Arc<VkRenderPass>, render_pass: Arc<VkRenderPass>,
pub(super) pipeline_layout: vk::PipelineLayout, pub pipeline_layout: vk::PipelineLayout,
pub(super) pipeline: vk::Pipeline, pub pipeline: vk::Pipeline,
vertex_shader: VkShaderModule, vertex_shader: VkShaderModule,
fragment_shader: VkShaderModule, fragment_shader: VkShaderModule,
} }

View file

@ -7,9 +7,9 @@ use ash::{vk, Entry, Instance};
use std::ffi::{c_char, CStr, CString}; use std::ffi::{c_char, CStr, CString};
pub struct VkInstance { pub struct VkInstance {
pub(super) entry: Entry, pub entry: Entry,
pub(super) handle: Instance, pub handle: Instance,
pub(super) surface_loader: surface::Instance, pub surface_loader: surface::Instance,
} }
impl VkInstance { impl VkInstance {

View file

@ -3,7 +3,7 @@ use ash::vk;
pub struct VkPhysicalDevice { pub struct VkPhysicalDevice {
// Vulkan properties // Vulkan properties
pub(super) handle: vk::PhysicalDevice, pub handle: vk::PhysicalDevice,
pub properties: vk::PhysicalDeviceProperties, pub properties: vk::PhysicalDeviceProperties,
pub features: vk::PhysicalDeviceFeatures, pub features: vk::PhysicalDeviceFeatures,
pub queue_family_properties: Vec<vk::QueueFamilyProperties>, pub queue_family_properties: Vec<vk::QueueFamilyProperties>,

View file

@ -4,6 +4,7 @@ use super::{
}; };
use ash::vk; use ash::vk;
use std::sync::Arc; use std::sync::Arc;
use crate::renderer::Renderable;
pub struct VkRenderContext { pub struct VkRenderContext {
instance: Arc<VkInstance>, instance: Arc<VkInstance>,
@ -13,7 +14,7 @@ pub struct VkRenderContext {
swapchain: Arc<VkSwapchain>, swapchain: Arc<VkSwapchain>,
render_pass: Arc<VkRenderPass>, render_pass: Arc<VkRenderPass>,
framebuffers: Vec<Arc<VkFramebuffer>>, framebuffers: Vec<Arc<VkFramebuffer>>,
pipeline: Arc<VkGraphicsPipeline>,
command_pool: VkCommandPool, command_pool: VkCommandPool,
command_buffers: Vec<vk::CommandBuffer>, command_buffers: Vec<vk::CommandBuffer>,
image_available_semaphore: VkSemaphore, image_available_semaphore: VkSemaphore,
@ -62,8 +63,6 @@ impl VkRenderContext {
.create_framebuffers(&render_pass) .create_framebuffers(&render_pass)
.ok_or_else(|| anyhow::anyhow!("Failed to get framebuffers"))?; .ok_or_else(|| anyhow::anyhow!("Failed to get framebuffers"))?;
let pipeline = Arc::new(VkGraphicsPipeline::new(&device, &render_pass)?);
let command_pool = VkCommandPool::new(&device)?; let command_pool = VkCommandPool::new(&device)?;
// Destroyed with command pool // Destroyed with command pool
@ -82,7 +81,6 @@ impl VkRenderContext {
swapchain, swapchain,
render_pass, render_pass,
framebuffers, framebuffers,
pipeline,
command_pool, command_pool,
command_buffers, command_buffers,
@ -93,7 +91,7 @@ impl VkRenderContext {
}) })
} }
pub fn render(&mut self) -> anyhow::Result<()> { pub fn render(&mut self, scene: Option<&Box<dyn Renderable>>) -> anyhow::Result<()> {
unsafe { self.device.handle.wait_for_fences(&[self.in_flight_fence.handle], true, u64::MAX)? }; unsafe { self.device.handle.wait_for_fences(&[self.in_flight_fence.handle], true, u64::MAX)? };
unsafe { self.device.handle.reset_fences(&[self.in_flight_fence.handle])? }; unsafe { self.device.handle.reset_fences(&[self.in_flight_fence.handle])? };
@ -133,26 +131,9 @@ impl VkRenderContext {
); );
}; };
unsafe { if let Some(scene) = scene {
self.device.handle.cmd_bind_pipeline( scene.render(&self.device, &self.swapchain, &command_buffer)?;
command_buffer, }
vk::PipelineBindPoint::GRAPHICS,
self.pipeline.pipeline,
)
};
let viewport = vk::Viewport::default()
.width(self.swapchain.surface_resolution.width as f32)
.height(self.swapchain.surface_resolution.height as f32)
.max_depth(1.0);
unsafe { self.device.handle.cmd_set_viewport(command_buffer, 0, &[viewport]) }
let scissor = self.swapchain.surface_resolution.into();
unsafe { self.device.handle.cmd_set_scissor(command_buffer, 0, &[scissor]) }
unsafe { self.device.handle.cmd_draw(command_buffer, 3, 1, 0, 0) };
unsafe { self.device.handle.cmd_end_render_pass(command_buffer) }; unsafe { self.device.handle.cmd_end_render_pass(command_buffer) };
@ -204,6 +185,10 @@ impl VkRenderContext {
unsafe { self.device.handle.device_wait_idle().unwrap() } unsafe { self.device.handle.device_wait_idle().unwrap() }
} }
pub fn init_scene(&self, scene: &mut Box<dyn Renderable>) -> anyhow::Result<()> {
scene.init(&self.device, &self.render_pass)
}
fn update_swapchain(&mut self) -> anyhow::Result<()> { fn update_swapchain(&mut self) -> anyhow::Result<()> {
match Arc::get_mut(&mut self.swapchain) { match Arc::get_mut(&mut self.swapchain) {
Some(swapchain) => { Some(swapchain) => {

View file

@ -6,7 +6,7 @@ use std::sync::Arc;
pub struct VkRenderPass { pub struct VkRenderPass {
device: Arc<VkDevice>, device: Arc<VkDevice>,
pub(super) handle: vk::RenderPass, pub handle: vk::RenderPass,
} }
impl VkRenderPass { impl VkRenderPass {

View file

@ -5,7 +5,7 @@ use std::sync::Arc;
pub struct VkSemaphore { pub struct VkSemaphore {
device: Arc<VkDevice>, device: Arc<VkDevice>,
pub(super) handle: vk::Semaphore, pub handle: vk::Semaphore,
} }
impl VkSemaphore { impl VkSemaphore {

View file

@ -6,7 +6,7 @@ use std::sync::Arc;
pub struct VkShaderModule { pub struct VkShaderModule {
device: Arc<VkDevice>, device: Arc<VkDevice>,
pub(super) handle: vk::ShaderModule, pub handle: vk::ShaderModule,
} }
impl VkShaderModule { impl VkShaderModule {

View file

@ -13,7 +13,7 @@ pub struct SwapchainSupportDetails(
pub struct VkSurface { pub struct VkSurface {
instance: Arc<VkInstance>, instance: Arc<VkInstance>,
pub(super) handle: vk::SurfaceKHR, pub handle: vk::SurfaceKHR,
} }
impl VkSurface { impl VkSurface {

View file

@ -8,22 +8,22 @@ pub struct VkSwapchain {
surface: Arc<VkSurface>, surface: Arc<VkSurface>,
device: Arc<VkDevice>, device: Arc<VkDevice>,
pub(super) handle: Option<vk::SwapchainKHR>, pub handle: Option<vk::SwapchainKHR>,
swapchain_support_details: SwapchainSupportDetails, swapchain_support_details: SwapchainSupportDetails,
pub(super) desired_image_count: u32, pub desired_image_count: u32,
pub(super) surface_format: vk::SurfaceFormatKHR, pub surface_format: vk::SurfaceFormatKHR,
pub(super) surface_resolution: vk::Extent2D, pub surface_resolution: vk::Extent2D,
pub(super) new_requested_surface_resolution: Option<vk::Extent2D>, pub new_requested_surface_resolution: Option<vk::Extent2D>,
pub(super) present_mode: vk::PresentModeKHR, pub present_mode: vk::PresentModeKHR,
pub(super) pre_transform: vk::SurfaceTransformFlagsKHR, pub pre_transform: vk::SurfaceTransformFlagsKHR,
pub(super) present_images: Option<Vec<vk::Image>>, pub present_images: Option<Vec<vk::Image>>,
pub(super) present_image_views: Option<Vec<Arc<vk::ImageView>>>, pub present_image_views: Option<Vec<Arc<vk::ImageView>>>,
} }
impl VkSwapchain { impl VkSwapchain {
pub(super) fn new( pub fn new(
window: &Window, window: &Window,
surface: &Arc<VkSurface>, surface: &Arc<VkSurface>,
device: &Arc<VkDevice>, device: &Arc<VkDevice>,
@ -78,7 +78,7 @@ impl VkSwapchain {
Ok(swapchain) Ok(swapchain)
} }
pub(super) fn create_swapchain(&mut self) -> VkResult<()> { pub fn create_swapchain(&mut self) -> VkResult<()> {
if let Some(new_requested_surface_resolution) = self.new_requested_surface_resolution { if let Some(new_requested_surface_resolution) = self.new_requested_surface_resolution {
self.surface_resolution = new_requested_surface_resolution; self.surface_resolution = new_requested_surface_resolution;
self.new_requested_surface_resolution = None; self.new_requested_surface_resolution = None;
@ -125,7 +125,7 @@ impl VkSwapchain {
Ok(()) Ok(())
} }
pub(super) fn create_framebuffers( pub fn create_framebuffers(
&self, &self,
render_pass: &Arc<VkRenderPass>, render_pass: &Arc<VkRenderPass>,
) -> Option<Vec<Arc<VkFramebuffer>>> { ) -> Option<Vec<Arc<VkFramebuffer>>> {
@ -148,7 +148,7 @@ impl VkSwapchain {
) )
} }
pub(super) fn update_resolution(&mut self, width: u32, height: u32) -> VkResult<()> { pub fn update_resolution(&mut self, width: u32, height: u32) -> VkResult<()> {
log::debug!("New resolution requested ({width}x{height})"); log::debug!("New resolution requested ({width}x{height})");
let chosen_extent = Self::choose_swapchain_extent( let chosen_extent = Self::choose_swapchain_extent(
@ -171,7 +171,7 @@ impl VkSwapchain {
Ok(()) Ok(())
} }
pub(super) fn acquire_next_image(&self, semaphore: &VkSemaphore) -> VkResult<(u32, bool)> { pub fn acquire_next_image(&self, semaphore: &VkSemaphore) -> VkResult<(u32, bool)> {
unsafe { unsafe {
self.device.swapchain_loader.acquire_next_image( self.device.swapchain_loader.acquire_next_image(
self.handle.unwrap(), self.handle.unwrap(),
@ -182,7 +182,7 @@ impl VkSwapchain {
} }
} }
pub(super) fn is_dirty(&self) -> bool { pub fn is_dirty(&self) -> bool {
self.new_requested_surface_resolution.is_some() self.new_requested_surface_resolution.is_some()
} }

2
src/scene/mod.rs Normal file
View file

@ -0,0 +1,2 @@
mod triangle;
pub use triangle::Triangle as TriangleScene;

49
src/scene/triangle.rs Normal file
View file

@ -0,0 +1,49 @@
use std::sync::Arc;
use ash::vk;
use crate::renderer::vulkan::{VkDevice, VkGraphicsPipeline, VkRenderPass, VkSwapchain};
use crate::renderer::Renderable;
use ash::vk::CommandBuffer;
pub struct Triangle {
pipeline: Option<VkGraphicsPipeline>,
}
impl Triangle {
pub fn new() -> Self {
Self { pipeline: None }
}
}
impl Renderable for Triangle {
fn init(&mut self, device: &Arc<VkDevice>, render_pass: &Arc<VkRenderPass>) -> anyhow::Result<()> {
let pipeline = VkGraphicsPipeline::new(&device, &render_pass)?;
self.pipeline = Some(pipeline);
Ok(())
}
fn render(&self, device: &VkDevice, swapchain: &VkSwapchain, command_buffer: &CommandBuffer) -> anyhow::Result<()> {
unsafe {
device.handle.cmd_bind_pipeline(
*command_buffer,
vk::PipelineBindPoint::GRAPHICS,
self.pipeline.as_ref().unwrap().pipeline,
)
};
let viewport = vk::Viewport::default()
.width(swapchain.surface_resolution.width as f32)
.height(swapchain.surface_resolution.height as f32)
.max_depth(1.0);
unsafe { device.handle.cmd_set_viewport(*command_buffer, 0, &[viewport]) }
let scissor = swapchain.surface_resolution.into();
unsafe { device.handle.cmd_set_scissor(*command_buffer, 0, &[scissor]) }
unsafe { device.handle.cmd_draw(*command_buffer, 3, 1, 0, 0) };
Ok(())
}
}