ULRE/src/SceneGraph/MaterialRenderList.cpp

385 lines
9.2 KiB
C++
Raw Normal View History

#include<hgl/graph/MaterialRenderList.h>
#include<hgl/graph/VKRenderable.h>
#include<hgl/graph/VKDevice.h>
#include<hgl/graph/VKCommandBuffer.h>
#include<hgl/graph/VKVertexInput.h>
2023-09-06 16:55:04 +08:00
#include<hgl/graph/VKRenderAssign.h>
#include<hgl/util/sort/Sort.h>
#include"RenderAssignBuffer.h"
2024-05-28 23:49:28 +08:00
#include<hgl/graph/VertexDataManager.h>
/**
*
*
*
* for(material)
* for(pipeline)
* for(material_instance)
2024-04-24 01:38:55 +08:00
* for(vab)
*
*
* Indirect Command Buffer
IndirectCommandBuffer使Indirect渲染的
VBOINDIRECT缓冲区便
*/
template<>
int Comparator<hgl::graph::RenderNode>::compare(const hgl::graph::RenderNode &obj_one,const hgl::graph::RenderNode &obj_two) const
{
2024-05-28 23:49:28 +08:00
hgl::int64 off;
hgl::graph::Renderable *ri_one=obj_one.ri;
hgl::graph::Renderable *ri_two=obj_two.ri;
//比较管线
{
off=ri_one->GetPipeline()
-ri_two->GetPipeline();
if(off)
return off;
}
2024-05-28 23:49:28 +08:00
auto *prim_one=ri_one->GetPrimitive();
auto *prim_two=ri_two->GetPrimitive();
//比如VDM
{
off=prim_one->GetVDM()
-prim_two->GetVDM();
if(off)
return off;
}
//比较模型
{
2024-05-28 23:49:28 +08:00
off=prim_one
-prim_two;
if(off)
2024-05-28 23:49:28 +08:00
{
off=prim_one->GetVertexOffset()-prim_two->GetVertexOffset(); //保存vertex offset小的在前面
return off;
2024-05-28 23:49:28 +08:00
}
}
return 0;
}
VK_NAMESPACE_BEGIN
MaterialRenderList::MaterialRenderList(GPUDevice *d,bool l2w,Material *m)
{
device=d;
cmd_buf=nullptr;
material=m;
assign_buffer=new RenderAssignBuffer(device,material);
vab_list=new VABList(material->GetVertexInput()->GetCount());
icb_draw=nullptr;
icb_draw_indexed=nullptr;
}
MaterialRenderList::~MaterialRenderList()
{
SAFE_CLEAR(icb_draw_indexed)
SAFE_CLEAR(icb_draw)
SAFE_CLEAR(vab_list);
SAFE_CLEAR(assign_buffer);
}
void MaterialRenderList::Add(Renderable *ri,const Matrix4f &mat)
{
RenderNode rn;
rn.local_to_world=mat;
rn.ri=ri;
rn_list.Add(rn);
}
void MaterialRenderList::End()
{
//排序
2023-07-28 20:41:02 +08:00
Sort(rn_list.GetArray());
const uint node_count=rn_list.GetCount();
if(node_count<=0)return;
Stat();
if(assign_buffer)
assign_buffer->WriteNode(rn_list);
}
void MaterialRenderList::RenderItem::Set(Renderable *ri)
{
2024-05-27 01:42:10 +08:00
pipeline=ri->GetPipeline();
mi =ri->GetMaterialInstance();
2024-05-29 00:55:12 +08:00
pdb =ri->GetDataBuffer();
2024-05-27 01:42:10 +08:00
prd =ri->GetRenderData();
}
void MaterialRenderList::ReallocICB()
{
const uint32_t icb_new_count=power_to_2(rn_list.GetCount());
if(icb_draw)
{
if(icb_new_count<=icb_draw->GetMaxCount())
return;
delete icb_draw;
icb_draw=nullptr;
delete icb_draw_indexed;
icb_draw_indexed=nullptr;
}
icb_draw=device->CreateIndirectDrawBuffer(icb_new_count);
icb_draw_indexed=device->CreateIndirectDrawIndexedBuffer(icb_new_count);
}
void MaterialRenderList::WriteICB(VkDrawIndirectCommand *dicp,RenderItem *ri)
{
dicp->vertexCount =ri->prd->vertex_count;
dicp->instanceCount =ri->instance_count;
dicp->firstVertex =ri->prd->vertex_offset;
dicp->firstInstance =ri->first_instance;
}
void MaterialRenderList::WriteICB(VkDrawIndexedIndirectCommand *diicp,RenderItem *ri)
{
diicp->indexCount =ri->prd->index_count;
diicp->instanceCount=ri->instance_count;
diicp->firstIndex =ri->prd->first_index;
diicp->vertexOffset =ri->prd->vertex_offset;
diicp->firstInstance=ri->first_instance;
}
void MaterialRenderList::Stat()
{
const uint count=rn_list.GetCount();
RenderNode *rn=rn_list.GetData();
ReallocICB();
VkDrawIndirectCommand *dicp=icb_draw->MapCmd();
VkDrawIndexedIndirectCommand *diicp=icb_draw_indexed->MapCmd();
ri_array.Clear();
ri_array.Alloc(count);
RenderItem *ri=ri_array.GetData();
ri_count=1;
ri->first_instance=0;
ri->instance_count=1;
ri->Set(rn->ri);
last_pipeline =ri->pipeline;
2024-05-28 23:49:28 +08:00
last_data_buffer=ri->pdb;
last_vdm =ri->pdb->vdm;
2024-05-27 01:42:10 +08:00
last_render_data=ri->prd;
++rn;
for(uint i=1;i<count;i++)
{
if(last_pipeline==rn->ri->GetPipeline())
if(last_data_buffer->Comp(rn->ri->GetDataBuffer()))
if(last_render_data->_Comp(rn->ri->GetRenderData())==0)
{
++ri->instance_count;
++rn;
continue;
}
if(ri->pdb->vdm)
{
if(ri->pdb->ibo)
WriteICB(diicp,ri);
2024-05-29 00:55:12 +08:00
else
WriteICB(dicp,ri);
++dicp;
++diicp;
2024-05-29 00:55:12 +08:00
}
++ri_count;
++ri;
ri->first_instance=i;
ri->instance_count=1;
ri->Set(rn->ri);
last_pipeline =ri->pipeline;
2024-05-29 00:55:12 +08:00
last_data_buffer=ri->pdb;
last_vdm =ri->pdb->vdm;
2024-05-27 01:42:10 +08:00
last_render_data=ri->prd;
++rn;
}
2024-05-30 01:14:27 +08:00
if(ri->pdb->vdm)
{
if(ri->pdb->ibo)
WriteICB(diicp,ri);
else
WriteICB(dicp,ri);
}
icb_draw->Unmap();
icb_draw_indexed->Unmap();
}
bool MaterialRenderList::BindVAB(const PrimitiveDataBuffer *pdb,const uint ri_index)
{
//binding号都是在VertexInput::CreateVIL时连续紧密排列生成的所以bind时first_binding写0就行了。
//const VIL *vil=last_vil;
2024-05-27 01:42:10 +08:00
//if(vil->GetCount(VertexInputGroup::Basic)!=prb->vab_count)
// return(false); //这里基本不太可能因为CreateRenderable时就会检查值是否一样
vab_list->Restart();
2024-04-24 01:44:01 +08:00
//Basic组它所有的VAB信息均来自于Primitive由vid参数传递进来
{
vab_list->Add(pdb->vab_list,
nullptr,//prb->vab_offset,
pdb->vab_count);
}
if(assign_buffer) //L2W/MI分发组
vab_list->Add(assign_buffer->GetVAB(),0);//ASSIGN_VAB_STRIDE_BYTES*ri_index);
//if(!vab_list.IsFull()) //Joint组暂未支持
//{
// const uint joint_id_binding_count=vil->GetCount(VertexInputGroup::JointID);
// if(joint_id_binding_count>0) //有矩阵信息
// {
// count+=joint_id_binding_count;
// if(count<vab_count) //JointWeight组
// {
// const uint joing_weight_binding_count=vil->GetCount(VertexInputGroup::JointWeight);
// if(joing_weight_binding_count!=1)
// {
// ++count;
// }
// else //JointWieght不为1是个bug除非未来支持8权重
// {
// return(false);
// }
// }
// else //有JointID没有JointWeight? 这是个BUG
// {
// return(false);
// }
// }
//}
//if(count!=vab_count)
//{
// //还有没支持的绑定组????
// return(false);
//}
cmd_buf->BindVAB(vab_list);
return(true);
}
void MaterialRenderList::ProcIndirectRender()
{
if(last_data_buffer->ibo)
icb_draw_indexed->DrawIndexed(*cmd_buf,first_indirect_draw_index,indirect_draw_count);
else
icb_draw->Draw(*cmd_buf,first_indirect_draw_index,indirect_draw_count);
first_indirect_draw_index=-1;
indirect_draw_count=0;
}
void MaterialRenderList::Render(RenderItem *ri)
{
if(last_pipeline!=ri->pipeline)
{
cmd_buf->BindPipeline(ri->pipeline);
last_pipeline=ri->pipeline;
last_data_buffer=nullptr;
//这里未来尝试换pipeline同时不换mi/primitive是否需要重新绑定mi/primitive
}
if(!ri->pdb->Comp(last_data_buffer)) //换buf了
{
if(indirect_draw_count) //如果有间接绘制的数据,赶紧给画了
ProcIndirectRender();
last_data_buffer=ri->pdb;
2024-05-27 01:42:10 +08:00
last_render_data=nullptr;
BindVAB(ri->pdb,ri->first_instance);
2024-05-28 02:21:33 +08:00
if(ri->pdb->ibo)
cmd_buf->BindIBO(ri->pdb->ibo);
}
//if(device-> support indirect)
{
if(indirect_draw_count==0)
first_indirect_draw_index=ri->first_instance;
++indirect_draw_count;
}
//else
//{
// cmd_buf->Draw(ri->pdb,ri->prd,ri->instance_count,ri->first_instance);
//}
}
void MaterialRenderList::Render(RenderCmdBuffer *rcb)
{
if(!rcb)return;
const uint count=rn_list.GetCount();
if(count<=0)return;
if(ri_count<=0)return;
cmd_buf=rcb;
last_pipeline =nullptr;
2024-05-28 23:49:28 +08:00
last_data_buffer=nullptr;
last_vdm =nullptr;
last_render_data=nullptr;
if(assign_buffer)
assign_buffer->Bind(material);
cmd_buf->BindDescriptorSets(material);
2024-05-29 00:55:12 +08:00
RenderItem *ri=ri_array.GetData();
for(uint i=0;i<ri_count;i++)
{
Render(ri);
++ri;
}
if(indirect_draw_count) //如果有间接绘制的数据,赶紧给画了
ProcIndirectRender();
}
VK_NAMESPACE_END