ULRE/src/ShaderGen/MaterialCreateInfo.cpp

256 lines
6.4 KiB
C++
Raw Normal View History

2023-03-17 10:08:41 +08:00
#include<hgl/shadergen/MaterialCreateInfo.h>
2023-03-17 10:14:07 +08:00
#include<hgl/shadergen/ShaderDescriptorInfo.h>
#include<hgl/graph/mtl/UBOCommon.h>
2023-06-02 20:45:19 +08:00
#include<hgl/graph/VKDeviceAttribute.h>
#include"common/MFCommon.h"
2023-10-07 17:02:00 +08:00
#include"ShaderLibrary.h"
2023-03-03 23:02:40 +08:00
using namespace hgl;
using namespace hgl::graph;
STD_MTL_NAMESPACE_BEGIN
2023-06-02 20:45:19 +08:00
MaterialCreateInfo::MaterialCreateInfo(const MaterialCreateConfig *mc)
2023-03-03 23:02:40 +08:00
{
config=mc;
2023-03-03 23:02:40 +08:00
2023-03-22 15:58:59 +08:00
if(hasVertex ())shader_map.Add(vert=new ShaderCreateInfoVertex (&mdi));else vert=nullptr;
if(hasGeometry ())shader_map.Add(geom=new ShaderCreateInfoGeometry(&mdi));else geom=nullptr;
if(hasFragment ())shader_map.Add(frag=new ShaderCreateInfoFragment(&mdi));else frag=nullptr;
2023-05-12 20:30:29 +08:00
ubo_range=0;
ssbo_range=0;
{
mi_data_bytes=0;
mi_shader_stage=0;
mi_max_count=0;
mi_ubo=nullptr;
}
2023-03-06 14:06:20 +08:00
}
2023-03-03 23:02:40 +08:00
2023-03-17 10:08:41 +08:00
bool MaterialCreateInfo::AddStruct(const AnsiString &struct_name,const AnsiString &codes)
2023-03-03 23:02:40 +08:00
{
2023-03-09 23:21:52 +08:00
if(struct_name.IsEmpty()||codes.IsEmpty())
2023-03-06 14:06:20 +08:00
return(false);
2023-03-03 23:02:40 +08:00
2023-03-22 15:58:59 +08:00
return mdi.AddStruct(struct_name,codes);
2023-03-06 14:06:20 +08:00
}
2023-03-03 23:02:40 +08:00
bool MaterialCreateInfo::AddUBO(const VkShaderStageFlagBits flag_bit,const DescriptorSetType set_type,const AnsiString &struct_name,const AnsiString &name)
2023-03-03 23:02:40 +08:00
{
if(!shader_map.ContainsKey(flag_bit))
2023-03-06 14:06:20 +08:00
return(false);
2023-03-03 23:02:40 +08:00
if(!mdi.hasStruct(struct_name))
2023-03-06 14:06:20 +08:00
return(false);
2023-03-03 23:02:40 +08:00
2023-03-17 10:08:41 +08:00
ShaderCreateInfo *sc=shader_map[flag_bit];
2023-03-03 23:02:40 +08:00
2023-03-06 14:06:20 +08:00
if(!sc)
return(false);
2023-03-22 15:58:59 +08:00
UBODescriptor *ubo=mdi.GetUBO(name);
2023-03-06 14:06:20 +08:00
if(ubo)
2023-03-03 23:02:40 +08:00
{
if(ubo->type!=struct_name)
2023-03-03 23:02:40 +08:00
return(false);
2023-03-08 14:02:51 +08:00
ubo->stage_flag|=flag_bit;
return sc->AddUBO(set_type,ubo);
2023-03-03 23:02:40 +08:00
}
2023-03-06 14:06:20 +08:00
else
2023-03-03 23:02:40 +08:00
{
2023-03-06 14:06:20 +08:00
ubo=new UBODescriptor();
ubo->type=struct_name;
2023-03-22 15:58:59 +08:00
hgl::strcpy(ubo->name,DESCRIPTOR_NAME_MAX_LENGTH,name);
2023-03-03 23:02:40 +08:00
return sc->AddUBO(set_type,mdi.AddUBO(flag_bit,set_type,ubo));
2023-03-03 23:02:40 +08:00
}
2023-03-06 14:06:20 +08:00
}
2023-03-03 23:02:40 +08:00
bool MaterialCreateInfo::AddUBO(const uint32_t flag_bits,const DescriptorSetType &set_type,const AnsiString &struct_name,const AnsiString &name)
{
if(flag_bits==0)return(false); //没有任何SHADER用?
if(!mdi.hasStruct(struct_name))
return(false);
uint result=0;
VkShaderStageFlagBits bit;
for(int i=0;i<shader_map.GetCount();i++)
{
shader_map.GetKey(i,bit);
if(flag_bits&bit)
if(AddUBO(bit,set_type,struct_name,name))
++result;
}
return(result==shader_map.GetCount());
}
2023-03-17 10:08:41 +08:00
bool MaterialCreateInfo::AddSampler(const VkShaderStageFlagBits flag_bit,const DescriptorSetType set_type,const SamplerType &st,const AnsiString &name)
{
if(!shader_map.ContainsKey(flag_bit))
2023-03-06 14:06:20 +08:00
return(false);
2023-03-06 14:06:20 +08:00
RANGE_CHECK_RETURN_FALSE(st);
2023-03-17 10:08:41 +08:00
ShaderCreateInfo *sc=shader_map[flag_bit];
2023-03-06 14:06:20 +08:00
if(!sc)
return(false);
2023-03-22 15:58:59 +08:00
SamplerDescriptor *sampler=mdi.GetSampler(name);
2023-03-06 14:06:20 +08:00
AnsiString st_name=GetSamplerTypeName(st);
2023-03-06 14:06:20 +08:00
if(sampler)
{
2023-03-06 14:06:20 +08:00
if(sampler->type!=st_name)
return(false);
2023-03-08 14:02:51 +08:00
sampler->stage_flag|=flag_bit;
return sc->AddSampler(set_type,sampler);
}
2023-03-06 14:06:20 +08:00
else
{
sampler=new SamplerDescriptor();
2023-03-06 14:06:20 +08:00
sampler->type=st_name;
2023-03-22 15:58:59 +08:00
hgl::strcpy(sampler->name,DESCRIPTOR_NAME_MAX_LENGTH,name);
2023-03-03 23:02:40 +08:00
return sc->AddSampler(set_type,mdi.AddSampler(flag_bit,set_type,sampler));
2023-03-06 14:06:20 +08:00
}
2023-03-03 23:02:40 +08:00
}
2023-03-08 14:02:51 +08:00
2023-09-28 17:42:14 +08:00
namespace
{
UBODescriptor *CreateUBODescriptor(const ShaderBufferSource &sbs,const uint32_t flag_bits)
{
UBODescriptor *ubo=new UBODescriptor;
ubo->type=sbs.struct_name;
hgl::strcpy(ubo->name,DESCRIPTOR_NAME_MAX_LENGTH,sbs.name);
ubo->stage_flag=flag_bits;
return ubo;
}
}//namespace
/**
*
2023-06-02 14:23:33 +08:00
* @param glsl_codes GLSL代码
* @param data_bytes
* @param shader_stage_flag_bits 使shader
* @return
*/
bool MaterialCreateInfo::SetMaterialInstance(const AnsiString &glsl_codes,const uint32_t data_bytes,const uint32_t shader_stage_flag_bits)
2023-03-08 14:02:51 +08:00
{
2023-06-02 14:22:07 +08:00
if(mi_data_bytes>0)return(false); //已经有数据了
if(shader_stage_flag_bits==0)return(false);
2023-06-02 14:23:33 +08:00
if(data_bytes>0&&glsl_codes.Length()<4)return(false);
2023-06-02 14:23:33 +08:00
mi_data_bytes=data_bytes;
2023-06-02 14:23:33 +08:00
if(data_bytes>0)
mi_codes=glsl_codes;
2023-09-05 23:36:59 +08:00
mi_max_count=hgl_min<uint32_t>(ubo_range/data_bytes,HGL_U16_MAX);
2023-06-02 20:45:19 +08:00
mdi.AddStruct(MaterialInstanceStruct,mi_codes);
mdi.AddStruct(SBS_MaterialInstance);
2023-09-28 17:42:14 +08:00
mi_ubo=CreateUBODescriptor(SBS_MaterialInstance,shader_stage_flag_bits);
mdi.AddUBO(shader_stage_flag_bits,DST_MaterialInstance,mi_ubo);
const AnsiString MI_MAX_COUNT_STRING=AnsiString::numberOf(mi_max_count);
2023-03-08 14:02:51 +08:00
auto *it=shader_map.GetDataList();
for(int i=0;i<shader_map.GetCount();i++)
2023-05-12 20:30:29 +08:00
{
if((*it)->key&shader_stage_flag_bits)
2023-06-02 20:45:19 +08:00
{
(*it)->value->AddDefine("MI_MAX_COUNT",MI_MAX_COUNT_STRING);
(*it)->value->SetMaterialInstance(mi_ubo,mi_codes);
2023-06-02 20:45:19 +08:00
}
++it;
2023-05-12 20:30:29 +08:00
}
mi_shader_stage=shader_stage_flag_bits;
2023-05-31 20:14:45 +08:00
return(true);
}
bool MaterialCreateInfo::SetLocalToWorld(const uint32_t shader_stage_flag_bits)
{
if(shader_stage_flag_bits==0)return(false);
l2w_max_count=hgl_min<uint32_t>(ubo_range/sizeof(Matrix4f),HGL_U16_MAX);
mdi.AddStruct(SBS_LocalToWorld);
l2w_ubo=CreateUBODescriptor(SBS_LocalToWorld,shader_stage_flag_bits);
mdi.AddUBO(shader_stage_flag_bits,DST_LocalToWorld,l2w_ubo);
const AnsiString L2W_MAX_COUNT_STRING=AnsiString::numberOf(l2w_max_count);
auto *it=shader_map.GetDataList();
for(int i=0;i<shader_map.GetCount();i++)
{
if((*it)->key&shader_stage_flag_bits)
{
(*it)->value->AddDefine("L2W_MAX_COUNT",L2W_MAX_COUNT_STRING);
}
++it;
}
l2w_shader_stage=shader_stage_flag_bits;
return(true);
}
2025-05-17 20:13:23 +08:00
bool MaterialCreateInfo::CreateShader(const VkDevAttr *dev_attr)
{
if(shader_map.IsEmpty())
return(false);
ubo_range=dev_attr->physical_device->GetUBORange(); //Mali-T系/G71为16knVidia和Mali-G系列除G71外为64kIntel/PowerVR为128MAMD无限制。
ssbo_range=dev_attr->physical_device->GetSSBORange();
2023-03-22 15:58:59 +08:00
mdi.Resort();
2023-03-14 22:22:35 +08:00
2023-03-17 10:08:41 +08:00
ShaderCreateInfo *sc,*last=nullptr;
2023-03-08 14:02:51 +08:00
2023-03-15 14:55:10 +08:00
for(int i=0;i<shader_map.GetCount();i++)
{
if(!shader_map.GetValue(i,sc))
return(false);
2023-03-08 21:41:57 +08:00
if(static_cast<uint32_t>(sc->GetShaderStage())<mi_shader_stage)
sc->AddMaterialInstanceOutput();
2023-03-15 14:55:10 +08:00
sc->CreateShader(last);
2023-03-08 21:41:57 +08:00
2023-03-15 14:55:10 +08:00
last=sc;
}
2023-03-08 14:02:51 +08:00
return(true);
}
STD_MTL_NAMESPACE_END