ULRE/example/common/VulkanAppFramework.h

414 lines
11 KiB
C
Raw Normal View History

2019-05-08 00:25:40 +08:00
#pragma once
#include<hgl/platform/Window.h>
#include<hgl/graph/VKInstance.h>
#include<hgl/graph/VKPhysicalDevice.h>
#include<hgl/graph/VKDevice.h>
#include<hgl/graph/VKSemaphore.h>
#include<hgl/graph/VKBuffer.h>
#include<hgl/graph/VKShaderModule.h>
#include<hgl/graph/VKImageView.h>
#include<hgl/graph/VKRenderable.h>
#include<hgl/graph/VKDescriptorSets.h>
#include<hgl/graph/VKRenderPass.h>
#include<hgl/graph/VKPipeline.h>
#include<hgl/graph/VKCommandBuffer.h>
#include<hgl/graph/VKFormat.h>
#include<hgl/graph/VKFramebuffer.h>
#include<hgl/graph/VKMaterial.h>
#include<hgl/graph/VKMaterialInstance.h>
#include<hgl/graph/VKRenderTarget.h>
2020-10-21 12:39:22 +08:00
#include<hgl/graph/VKRenderResource.h>
#include<hgl/graph/RenderList.h>
using namespace hgl;
using namespace hgl::graph;
class VulkanApplicationFramework
{
2019-06-11 16:18:49 +08:00
private:
2020-10-21 12:47:06 +08:00
Window * win =nullptr;
VulkanInstance * inst =nullptr;
void OnKeyPressed (KeyboardButton kb){key_status[kb]=true;}
void OnKeyReleased (KeyboardButton kb){key_status[kb]=false;}
void OnKeyRepeat (KeyboardButton kb){KeyRepeat(kb);}
2019-06-14 19:56:21 +08:00
protected:
uint mouse_key=0;
Vector2f mouse_pos;
void OnMousePressed (int,int,uint mk){mouse_key=mk;MousePressed(mk);}
void OnMouseReleased(int,int,uint mk){mouse_key=0;MouseReleased(mk);}
void OnMouseMove (int x,int y){mouse_pos.Set(x,y);MouseMove();}
void OnMouseWheel (int v,int h,uint mk){MouseWheel(v,h,mk);}
protected:
2020-10-21 12:52:32 +08:00
GPUDevice * device =nullptr;
2020-10-21 12:47:06 +08:00
SwapchainRenderTarget * sc_render_target =nullptr;
2020-01-23 21:00:35 +08:00
2019-07-17 04:49:16 +08:00
protected:
2020-10-21 12:52:32 +08:00
int32_t swap_chain_count =0;
RenderCmdBuffer ** cmd_buf =nullptr;
2020-10-21 12:52:32 +08:00
Color4f clear_color;
protected:
2020-10-21 12:52:32 +08:00
RenderResource * db =nullptr;
2020-10-21 12:52:32 +08:00
bool key_status[kbRangeSize];
public:
virtual ~VulkanApplicationFramework()
{
SAFE_CLEAR(db);
SAFE_CLEAR_OBJECT_ARRAY(cmd_buf,swap_chain_count);
SAFE_CLEAR(device);
2020-01-23 21:00:35 +08:00
SAFE_CLEAR(win);
SAFE_CLEAR(inst);
}
virtual bool Init(int w,int h)
{
hgl_zero(key_status);
clear_color.Zero();
#ifdef _DEBUG
2020-10-21 12:47:06 +08:00
if(!CheckStrideBytesByFormat())
return(false);
#endif//
2019-04-30 19:01:44 +08:00
InitNativeWindowSystem();
2020-10-28 17:28:07 +08:00
InitVulkanInstanceProperties();
2019-11-14 17:57:46 +08:00
win=CreateRenderWindow(OS_TEXT("VulkanTest"));
if(!win)
return(false);
if(!win->Create(w,h))
return(false);
2019-11-14 17:57:46 +08:00
{
2020-10-21 12:52:32 +08:00
CreateInstanceLayerInfo cili;
2020-09-28 11:16:45 +08:00
hgl_zero(cili);
2019-11-14 17:57:46 +08:00
cili.lunarg.standard_validation = true;
cili.khronos.validation = true;
// cili.RenderDoc.Capture = true;
2019-11-14 17:57:46 +08:00
2020-10-21 12:47:06 +08:00
inst=CreateInstance("VulkanTest",nullptr,&cili);
2019-11-14 17:57:46 +08:00
if(!inst)
return(false);
}
device=CreateRenderDevice(inst,win);
if(!device)
return(false);
2020-10-21 12:47:06 +08:00
db=new RenderResource(device);
InitCommandBuffer();
SetEventCall(win->OnResize, this,VulkanApplicationFramework,OnResize );
SetEventCall(win->OnKeyPressed, this,VulkanApplicationFramework,OnKeyPressed );
SetEventCall(win->OnKeyReleased, this,VulkanApplicationFramework,OnKeyReleased );
SetEventCall(win->OnKeyRepeat, this,VulkanApplicationFramework,OnKeyRepeat );
SetEventCall(win->OnMousePressed, this,VulkanApplicationFramework,OnMousePressed );
SetEventCall(win->OnMouseReleased, this,VulkanApplicationFramework,OnMouseReleased );
SetEventCall(win->OnMouseMove, this,VulkanApplicationFramework,OnMouseMove );
SetEventCall(win->OnMouseWheel, this,VulkanApplicationFramework,OnMouseWheel );
2019-06-14 19:56:21 +08:00
return(true);
}
virtual void Resize(int,int)=0;
virtual void KeyRepeat(KeyboardButton){}
virtual void MousePressed(uint){}
virtual void MouseReleased(uint){}
2019-06-14 19:56:21 +08:00
virtual void MouseMove(){}
virtual void MouseWheel(int,int,uint){}
2020-01-23 21:00:35 +08:00
void SetClearColor(COLOR cc)
{
clear_color.Use(cc,1.0);
}
2019-07-17 04:49:16 +08:00
void OnResize(int w,int h)
{
if(w>0&&h>0)
device->Resize(w,h);
2019-07-17 04:49:16 +08:00
InitCommandBuffer();
Resize(w,h);
}
void InitCommandBuffer()
{
if(cmd_buf)
2019-06-28 20:46:32 +08:00
SAFE_CLEAR_OBJECT_ARRAY(cmd_buf,swap_chain_count);
2020-01-23 21:00:35 +08:00
2019-07-17 04:49:16 +08:00
sc_render_target=device->GetSwapchainRT();
swap_chain_count=sc_render_target->GetImageCount();
{
const VkExtent2D extent=sc_render_target->GetExtent();
cmd_buf=hgl_zero_new<RenderCmdBuffer *>(swap_chain_count);
for(int32_t i=0;i<swap_chain_count;i++)
cmd_buf[i]=device->CreateRenderCommandBuffer();
}
}
bool BuildCommandBuffer(RenderCmdBuffer *cb,RenderPass *rp,Framebuffer *fb,RenderableInstance *ri)
{
if(!ri)return(false);
2020-10-21 12:47:06 +08:00
const IndexBuffer *ib=ri->GetIndexBuffer();
cb->Begin();
cb->BindFramebuffer(rp,fb);
cb->SetClearColor(0,clear_color.r,clear_color.g,clear_color.b);
cb->BeginRenderPass();
cb->BindPipeline(ri->GetPipeline());
cb->BindDescriptorSets(ri->GetDescriptorSets());
cb->BindVAB(ri);
if (ib)
cb->DrawIndexed(ib->GetCount());
else
cb->Draw(ri->GetDrawCount());
cb->EndRenderPass();
cb->End();
return(true);
}
void BuildCommandBuffer(RenderCmdBuffer *cb,RenderTarget *rt,RenderableInstance *ri)
{
if(!cb||!rt||!ri)
return;
BuildCommandBuffer(cb,rt->GetRenderPass(),rt->GetFramebuffer(),ri);
}
2020-10-24 19:11:54 +08:00
bool BuildCommandBuffer(uint32_t index,RenderableInstance *ri)
{
if(!ri)return(false);
return BuildCommandBuffer(cmd_buf[index],sc_render_target->GetRenderPass(),sc_render_target->GetFramebuffer(index),ri);
2019-06-18 00:48:05 +08:00
}
2020-01-23 21:00:35 +08:00
2020-10-24 19:11:54 +08:00
bool BuildCommandBuffer(RenderableInstance *ri)
2019-06-18 00:48:05 +08:00
{
2020-10-24 19:11:54 +08:00
if(!ri)return(false);
for(int32_t i=0;i<swap_chain_count;i++)
BuildCommandBuffer(i,ri);
2020-10-24 19:11:54 +08:00
return(true);
}
2020-10-24 19:11:54 +08:00
bool BuildCurrentCommandBuffer(RenderableInstance *ri)
{
2020-10-24 19:11:54 +08:00
if(!ri)return(false);
return BuildCommandBuffer(sc_render_target->GetCurrentFrameIndices(),ri);
2019-06-18 00:48:05 +08:00
}
2019-06-18 00:48:05 +08:00
void BuildCommandBuffer(uint32_t index,RenderList *rl)
{
2020-01-23 21:00:35 +08:00
if(!rl)return;
2019-06-18 00:48:05 +08:00
RenderCmdBuffer *cb=cmd_buf[index];
2019-06-18 00:48:05 +08:00
cb->Begin();
cb->BindFramebuffer(sc_render_target->GetRenderPass(),sc_render_target->GetFramebuffer(index));
cb->SetClearColor(0,clear_color.r,clear_color.g,clear_color.b);
cb->BeginRenderPass();
rl->Render(cb);
cb->EndRenderPass();
2019-06-18 00:48:05 +08:00
cb->End();
}
void BuildCommandBuffer(RenderList *rl)
{
for(int32_t i=0;i<swap_chain_count;i++)
2019-06-18 00:48:05 +08:00
BuildCommandBuffer(i,rl);
}
2020-01-23 21:00:35 +08:00
2019-06-18 00:48:05 +08:00
void BuildCurrentCommandBuffer(RenderList *rl)
2020-01-23 21:00:35 +08:00
{
BuildCommandBuffer(sc_render_target->GetCurrentFrameIndices(),rl);
}
template<typename ...ARGS>
Pipeline *CreatePipeline(ARGS...args){return sc_render_target->CreatePipeline(args...);}
public:
2019-06-18 00:48:05 +08:00
int AcquireNextImage()
{
return sc_render_target->AcquireNextImage();
2019-06-18 00:48:05 +08:00
}
virtual void SubmitDraw(int index)
2019-06-18 00:48:05 +08:00
{
VkCommandBuffer cb=*cmd_buf[index];
2020-01-23 21:00:35 +08:00
sc_render_target->Submit(cb);
sc_render_target->PresentBackbuffer();
sc_render_target->WaitQueue();
sc_render_target->WaitFence();
}
2019-06-18 00:48:05 +08:00
virtual void Draw()
{
int index=AcquireNextImage();
2019-07-01 17:04:02 +08:00
if(index<0||index>=swap_chain_count)return;
2019-06-18 00:48:05 +08:00
SubmitDraw(index);
}
2019-04-30 19:01:44 +08:00
bool Run()
{
if(!win->Update())return(false);
if(win->IsVisible())
Draw();
return(true);
}
};//class VulkanApplicationFramework
class CameraAppFramework:public VulkanApplicationFramework
{
private:
2020-11-17 17:20:59 +08:00
GPUBuffer * ubo_world_matrix =nullptr;
protected:
2020-11-19 22:12:31 +08:00
WalkerCamera * camera;
2020-11-17 17:20:59 +08:00
float move_speed=1;
2020-11-17 17:20:59 +08:00
Vector2f mouse_last_pos;
2019-06-14 19:56:21 +08:00
public:
2020-11-17 17:20:59 +08:00
virtual ~CameraAppFramework()
{
2020-11-19 22:12:31 +08:00
SAFE_CLEAR(camera);
2020-11-17 17:20:59 +08:00
}
virtual bool Init(int w,int h)
{
if(!VulkanApplicationFramework::Init(w,h))
return(false);
2020-01-23 21:00:35 +08:00
InitCamera(w,h);
return(true);
}
virtual void InitCamera(int w,int h)
{
2020-11-19 22:12:31 +08:00
camera=new WalkerCamera;
camera->type=CameraType::Perspective;
camera->width=w;
camera->height=h;
camera->vp_width=w;
camera->vp_height=h;
2020-11-19 22:12:31 +08:00
camera->pos=Vector4f(10,10,10,1);
camera->target=Vector4f(0,0,0,1);
2020-11-17 17:20:59 +08:00
2020-11-19 22:12:31 +08:00
camera->Refresh(); //更新矩阵计算
2020-11-19 22:12:31 +08:00
ubo_world_matrix=db->CreateUBO(sizeof(WorldMatrix),&camera->matrix);
}
2020-01-23 21:00:35 +08:00
void Resize(int w,int h)override
{
2020-11-19 22:12:31 +08:00
camera->width=w;
camera->height=h;
}
2020-10-21 12:47:06 +08:00
GPUBuffer *GetCameraMatrixBuffer()
{
return ubo_world_matrix;
}
2019-06-18 00:48:05 +08:00
virtual void BuildCommandBuffer(uint32_t index)=0;
2019-06-14 20:01:46 +08:00
virtual void Draw()override
{
2020-11-19 22:12:31 +08:00
camera->Refresh(); //更新相机矩阵
ubo_world_matrix->Write(&camera->matrix); //写入缓冲区
const uint32_t index=AcquireNextImage();
2019-06-18 00:48:05 +08:00
BuildCommandBuffer(index);
SubmitDraw(index);
2020-11-19 22:12:31 +08:00
if(key_status[kbW])camera->Up (move_speed);else
if(key_status[kbS])camera->Down (move_speed);else
if(key_status[kbA])camera->Left (move_speed);else
if(key_status[kbD])camera->Right (move_speed);else
if(key_status[kbR])camera->Forward (move_speed);else
if(key_status[kbF])camera->Backward (move_speed);else
if(key_status[kbLeft ])camera->HoriRotate( move_speed);else
if(key_status[kbRight ])camera->HoriRotate(-move_speed);else
if(key_status[kbUp ])camera->VertRotate( move_speed);else
if(key_status[kbDown ])camera->VertRotate(-move_speed);else
return;
}
virtual void KeyRepeat(KeyboardButton kb)override
{
if(kb==kbMinus)move_speed*=0.9f;else
if(kb==kbEquals)move_speed*=1.1f;else
return;
}
2019-06-14 19:56:21 +08:00
virtual void MousePressed(uint) override
2019-06-14 19:56:21 +08:00
{
mouse_last_pos=mouse_pos;
}
virtual void MouseMove() override
{
if(!(mouse_key&(mbLeft|mbRight)))return;
2019-06-14 19:56:21 +08:00
Vector2f gap=mouse_pos-mouse_last_pos;
bool update=false;
if(gap.x!=0){update=true;if(mouse_key&mbLeft)camera->HoriRotate( gap.x/10.0f);else camera->WrapHoriRotate(gap.x);}
if(gap.y!=0){update=true;if(mouse_key&mbLeft)camera->VertRotate(-gap.y/10.0f);else camera->WrapVertRotate(gap.y);}
2019-06-14 19:56:21 +08:00
mouse_last_pos=mouse_pos;
}
virtual void MouseWheel(int v,int h,uint)
{
2020-11-19 22:12:31 +08:00
camera->Distance(1+(v/1000.0f));
}
};//class CameraAppFramework