UE4 RHI与Render模块简解
UE4中的RHI指的是Render hardware interface,作用像Ogre里的RenderSystem,针对Dx11,Dx12,Opengl等等平台抽象出相同的接口,我们能方便能使用相同接口对应不同渲染平台.
和以前一样,先简单介绍一些类与文件的作用,我们有个抽象的了解.
RHI.h :主要定义一些硬件平台的公共变量.
一是 硬件支持项,如是否支持PF_FloatRGBA格式渲染目标,手机平台是否支持FrameBuffer拾取,支持体纹理,支持硬件合并渲染等等.
二是 硬件变量,如最大Cube纹理数,阴影贴图长宽最大值等等.
三是 常见渲染定义,如FSamplerStateInitializerRHI纹理采样,FRasterizerStateInitializerRHI栅栏化(填充格式,正方向定义,MSAA),FDepthStencilStateInitializerRHI逐片断处理中的模板与深度,FBlendStateInitializerRHI逐片断处理中的混合.FRHIDrawIndirectParameters/FRHIDrawIndexedIndirectParameters DrawCall中相关参数.
DynamicRHI.h :包含FDynamicRHI接口定义,渲染所需求所有接口,创建buffer,创建纹理,设置着色器参数,UAV等,简单来说,对应opengl,dx提供的渲染API,其DynamicRHI.cpp文件会根据平台(Windows,apple,android等等)来选择加载合适的渲染平台(如Opengl,Dx,Vulkan等),在RHI模块的private文件夹下,可能看到各个系统会如何选择相应的渲染平台.
FRenderResource:定义接口如InitDynamicRHI /ReleaseDynamicRHI /InitRHI /ReleaseRHI /InitResource /ReleaseResource /UpdateRHI等渲染资源选择实现函数.
RHICommandList相关文件是我们讲RHI主要需要讲的,在这我们先来分析出现的各个类.
FRHICommandBase: 主要定义一个函数指针,一个执行方法调用函数指针指向的函数。
函数指针:二个参数(FRHICommandListBase,FRHICommandBase)CallExecuteAndDestruct:传入自己FRHICommandBase到时函数指针指向的方向.
FRHICommand: FRHICommandBase的一个模板子类,模板需要定义Execute方法,其方法只需要FRHICommandListBase,其会退化上面CallExecuteAndDestruct的FRHICommandBase参数,默认为自己.
FRHICommand的模板具体化,对应SetRasterizerState/SetDepthStencilState/SetShaderParameter等等,几乎所有渲染API都有对应的FRHICommand的模板具体化实现.
FRHICommandListBase: 相应FRHICommandBase的链表实现,以及定义一些上下文如IRHICommandContext ,IRHIComputeContext ,并且有相关和RHI线程交互的API,RHI本身相应的FRHICommandBase与List都是存放在渲染线程中,RHI线程可以用于在渲染线程中同步执行异步的复杂操作,如压入很多FRHICommandBase到渲染线程中执行,有些操作可以放入RHI线程中与渲染线程一起执行,在某段FRHICommandBase前,调用WaitForTasks等同步渲染线程与RHI线程,大家可以这么理解,RHI线程对于渲染线程就相当于渲染线程与游戏线程的关系,大家可以看我上篇UE4里的 渲染线程 ,看到如何在渲染线程里压入RHI线程,如何用WaitForTasks与渲染线程同步等.
FRHICommandList: 简单来说,所有用于渲染API几乎都有二种方法,一种是插入FRHICommandListBase链表,一种是直接调用相应渲染平台对应FDynamicRHI的实现,在这说下,我看了下OpenGLDrv相应的FDynamicRHI实现,相应API如SetShaderParameter, SetDepthStencilState等等,并没有直接调用相应的OpenGL的API,而是把相关改动放入一个FOpenGLRHIState的结构中保存起来,等到DrawCall(如RHIDrawPrimitiveIndirect等)相关命令调用后,才把各个改动对应opengl的API调用起来,如上的glProgramUniform等.
FRHIAsyncComputeCommandList: 多GPU的FRHICommandList实现。
FRHICommandListImmediate: 直接调用相应渲染平台对应FDynamicRHI的实现,对比FRHICommandList,主要是创建资源这一块的FDynamicRHI封装,可以看到它的一些函数都是以Create开头的。
FRHICommandListExecutor: 简单来说,管理FRHICommandListBase的几个子类单例实现,方便查找到如上的FRHICommandListImmediate 与FRHIAsyncComputeCommandListImmediate 单例实现,一般我们看到渲染代码里常见的如FRHICommandList/RHICmdList就是指的是FRHICommandListExecutor::GetImmediateCommandList().
在这,关于RHI的就先简单了解下,RHI主要调用都在渲染线程中,不过也可以使用FRHICommandListBase链表与RHI线程来实现一些同步异步操作。其中渲染模块中FRHICommandList/RHICmdList一般是FRHICommandListExecutor::GetImmediateCommandList(),这个是直接调用相关FDynamicRHI实现,一般并不与RHI线程交互。
介绍RHI模块后,我们来看下渲染模块的相关实现,在说下渲染模块的实现前,简单说下,UE4中大量用到C++ 的模版,除开自动生成各个分支代码,还有二点,一是代替部分接口类,减少如虚函数表的性能,二是减少一些分支判断,还是提高性能。但是会造成阅读代码比C#等语言验证,主要在于有些模板你都不知道是那些类可以用等,还好,UE4里一般这种模板使用类都有相同的前缀或是后缀,我们可以记一些相同的前缀或后缀转化成自己认为的接口实现。
我们先看一段代码,是OpenGLDrv实现的FDynamicRHI子类FOpenGLDynamicRHI的RHIDrawPrimitiveIndirect,简接绘制多组图元集。
void FOpenGLDynamicRHI::RHIDrawPrimitiveIndirect(uint32 PrimitiveType,FVertexBufferRHIParamRef ArgumentBufferRHI,uint32 ArgumentOffset)
{
if (FOpenGL::SupportsDrawIndirect())
{
VERIFY_GL_SCOPE(); check(ArgumentBufferRHI);
GPUProfilingData.RegisterGPUWork(); FOpenGLContextState& ContextState = GetContextStateForCurrentContext();
BindPendingFramebuffer(ContextState);
SetPendingBlendStateForActiveRenderTargets(ContextState);
UpdateViewportInOpenGLContext(ContextState);
UpdateScissorRectInOpenGLContext(ContextState);
UpdateRasterizerStateInOpenGLContext(ContextState);
UpdateDepthStencilStateInOpenGLContext(ContextState);
BindPendingShaderState(ContextState);
SetupTexturesForDraw(ContextState);
CommitNonComputeShaderConstants();
CachedBindElementArrayBuffer(ContextState,); // Zero-stride buffer emulation won't work here, need to use VAB with proper zero strides
SetupVertexArrays(ContextState, , PendingState.Streams, NUM_OPENGL_VERTEX_STREAMS, ); GLenum DrawMode = GL_TRIANGLES;
GLsizei NumElements = ;
GLint PatchSize = ;
FindPrimitiveType(PrimitiveType, ContextState.bUsingTessellation, , DrawMode, NumElements, PatchSize); if (FOpenGL::SupportsTessellation() && DrawMode == GL_PATCHES )
{
FOpenGL::PatchParameteri(GL_PATCH_VERTICES, PatchSize);
} FOpenGLVertexBuffer* ArgumentBuffer = ResourceCast(ArgumentBufferRHI); glBindBuffer( GL_DRAW_INDIRECT_BUFFER, ArgumentBuffer->Resource);
{
CONDITIONAL_SCOPE_CYCLE_COUNTER(STAT_OpenGLShaderFirstDrawTime, PendingState.BoundShaderState->RequiresDriverInstantiation());
FOpenGL::DrawArraysIndirect( DrawMode, INDEX_TO_VOID(ArgumentOffset));
}
glBindBuffer( GL_DRAW_INDIRECT_BUFFER, ); FShaderCache::LogDraw();
}
else
{
UE_LOG(LogRHI, Fatal,TEXT("OpenGL RHI does not yet support indirect draw calls."));
} }
FOpenGLDynamicRHI::RHIDrawPrimitiveIndirect
前面说过,FOpenGLDynamicRHI是在DrawCall时,才把各个改动对应opengl的API调用起来,所以在这,我们可以看到一个渲染的完整过程,当然大家使用过Opengl或是DX直接写过程序也是一样,首先设定渲染目标,混合,设定viewport,设定栅栏化,设定逐片断处理(深度,模板),绑定Shader程序,设定shader纹理,设置shader参数,绑定VAO,设定VAO,DrawCall,嗯,就是这么个过程,无论UE4如何包装,每次DrawCall就是如上顺序处理。
先说一下在渲染模块里比较常见的类:
后缀Parameters: 二个主要方法,一是Bind,简单来说,对应一个或多个参数Parameter与Shader代码里参数绑定,对应opengl里的API就是如glGetUniformLocation。二是Set,简单来说,上面绑定后,我们就可以传入参数的值到GPU里,对应opengl里的API就是如glUniform等等。
模板类里的模板如果是后缀ParametersType,一般主要是指各个后缀为Parameters的类。
如下一些类写了些自己了解,后查找资料时发现UE4官方文档里 着色器开发 有说,比我说的清楚。
FVertexFactory: 用来表示顶点数据格式,顶点分布结构,顶点元素Buffer,DeclarationElementList数组,相关opengl的API如glVertexAttribPointer.从opengl3+来说,一般虽然可能有多个buffer,但是应该是在一个glgenbuffer中对应不同的区段而已。
方法Set: 只是告诉对应opengl里各个buffer的起点与终点,相应的如OffsetInstanceStreams/SetPositionStream都是类似。
FVertexFactoryType:表示网格类型,如 Local/Particle(三种sprite/beamtrail,mesh)/Landscape/GPUSkin等,
FMeshBatch: 一般来说,是一组相同顶点格式,相同材质的模型,一般可以使用GPU的实例渲染,减少DrawCall.
FShaderType: Global/Material/MeshMaterial (vertex/hull/demain/geomerty/pixel 一种)
FGlobalShader: 全局shader,简单来说,不和mesh与Material关联,一般用于后处理,固定画个方块啥的,如处理特效这种。
方法SetParameters: 设定Shader里的FViewUniformShaderParameters /FFrameUniformShaderParameters /FBuiltinSamplersParameters 参数。
FMaterialShader: 特定于过程的着色器,它们需要访问材质的某些属性,因此必须针对每个材质进行编译,但不需要访问任何网格属性。如FLightFunctionVS,FLightFunctionPS等。
对比FGlobalShader,增加一个重载的SetParameters,包含材质对Shader的设置。
FMeshMaterialShader: 着色器是特定于过程的着色器,它们依赖于材质的属性和网格类型,因此必须针对每个材质/FVertexFactory组合进行编译。例如,TBasePassVS / TBasePassPS 需要对前向渲染过程中的所有材质输入进行评估。
对比FMaterialShader,增加一个方法SetMesh,添加FMeshBatchElement,FVertexFactory对shader的设置,对应VertexFactory的Parameters针对Mesh填充不同的顶点信息。 如GPUSkin,填充骨骼信息到相应的shader参数中, 如MeshParticle,填充动画加速度 ,时间等。以及填充模型本身的FPrimitiveUniformShaderParameters等共有信息,如FPrimitiveUniformShaderParameters:localToworld ,worldTolocal ,objectBounds, LOD,FadeTimeScaleBias等。
如下这些类表示渲染主要思路,预先一些相同的渲染方式,可以先缓存起来。
FMeshDrawingPolicy: 整合渲染模型过程,从绑定Shader到调用DrawCall,各个子类对应不同的独立着色器程序。
1 初始化,根据需要生成或绑定各个对应的Shader.
2 SetSharedState,设定和Mesh无关的Shader变量。
3 SetMeshRenderState,设定和Mesh相关的Shader变量。
4 DrawMesh 调用DrawCall.
模板类里的模板如果是DrawingPolicyType,一般主要是指FMeshDrawingPolicy的各个子类。
FUniformLightMapPolicy: 封装和光照有关渲染的Shader参数设置。
方法SetMesh:绑定相应光照计算上Shader的参数,如使用GI预计算产生的间接光照图信息,直接光照图信息,天空图AO等。
TUniformLightMapPolicy: FUniformLightMapPolicy的模版子类,模版为ELightMapPolicyType,表示各种和光照有关,模版预生成多份代码对应不同光照计算表示是否缓存,Shader预编译指令。
enum ELightMapPolicyType
{
LMP_NO_LIGHTMAP,
LMP_CACHED_VOLUME_INDIRECT_LIGHTING,
LMP_CACHED_POINT_INDIRECT_LIGHTING,
LMP_SIMPLE_DYNAMIC_LIGHTING,
LMP_LQ_LIGHTMAP,
LMP_HQ_LIGHTMAP,
LMP_DISTANCE_FIELD_SHADOWS_AND_HQ_LIGHTMAP,
// Forward shading specific
LMP_DISTANCE_FIELD_SHADOWS_AND_LQ_LIGHTMAP,
LMP_SIMPLE_DIRECTIONAL_LIGHT_AND_SH_INDIRECT,
LMP_SIMPLE_DIRECTIONAL_LIGHT_AND_SH_DIRECTIONAL_INDIRECT,
LMP_SIMPLE_DIRECTIONAL_LIGHT_AND_SH_DIRECTIONAL_CSM_INDIRECT,
LMP_MOVABLE_DIRECTIONAL_LIGHT,
LMP_MOVABLE_DIRECTIONAL_LIGHT_CSM,
LMP_MOVABLE_DIRECTIONAL_LIGHT_WITH_LIGHTMAP,
LMP_MOVABLE_DIRECTIONAL_LIGHT_CSM_WITH_LIGHTMAP,
// LightMapDensity
LMP_DUMMY
};
ELightMapPolicyType
TLightMapPolicy: Shader对应预编译指令,是否缓存,模板为ELightmapQuality,有二个值,分别是LQ_LIGHTMAP,HQ_LIGHTMAP。
模板类里的模板如果是LightMapPolicyType,一般主要是指TUniformLightMapPolicy/TLightMapPolicy的各个子类。
如上一些基本比较重要的类就到此,在这我们重点说下FMeshDrawingPolicy这个类,从上面各个类的说明来看,可以看到把所有渲染基本类组合在一起,他的子类简单说几个,BasePassRendering ,CapsuleShadowing ,DepthRendering ,ForwardBasePassRendering ,VelocityRendering等等,还有别的带Rendering的渲染,如DistortionRendering ,DeferredShading ,DecalRendering,ShadowRendering等等虽然和FMeshDrawingPolicy不同,但是过程其实真差不了多少。 在每个Rendering中,都有对应的VS,PS,HS等,这些根据需要分别从上面所说的FGlobalShader /FMaterialShader /FMeshMaterialShader继承,简单来说,后处理特效针对渲染目标的一般从FGlobalShader继承,只针对Material不和具体Mesh有关的用FMaterialShader,最后针对模型渲染的从FMeshMaterialShader继承。
按BasePassRendering说下,只简单渲染emissive color与light map,对应的FMeshDrawingPolicy子类为TBasePassDrawingPolicy ,如上所说,针对Mesh产生的都继承与FMeshMaterialShader生成的VS,PS等,因为光照有影响,我们看到相应的Shader都对应模版LightMapPolicyType,用于生成正确的Shader对应预编译指令,如有无光照,光照质量,静态或动态,阴影类型等。下面还定义一些与BasePassRendering相关的parameters,如天空盒相关参数,如上TBasePassDrawingPolicy在构造函数中得到或是生成上面的VS,PS,然后在SetSharedState时针对VS,PS设定参数,然后调用SetMeshRenderState针对每个FMeshBatch设定和Mesh有关的参数,然后提交DrawCall.
每个DrawingPolicy中,对应VS,PS等对应文件可以通过宏IMPLEMENT_SHADER_TYPE查看。
本文本来还准备更详细讲述一个基本的Rendering的过程,但是新项目时间紧,只是暂停查看,后面会仔细介绍一个完整流程,从阴影渲染,前向或是后向渲染选择一部分来详细介绍,包含大部分参数的含义与作用,不过大家熟悉如上的渲染线程再加个RHI与渲染模块如上这些基本类,应该就能把UE4的源码都联系起来了。
UE4 RHI与Render模块简解的更多相关文章
- python ConfigParser、shutil、subprocess、ElementTree模块简解
ConfigParser 模块 一.ConfigParser简介ConfigParser 是用来读取配置文件的包.配置文件的格式如下:中括号“[ ]”内包含的为section.section 下面为类 ...
- Python中操作mysql的pymysql模块详解
Python中操作mysql的pymysql模块详解 前言 pymsql是Python中操作MySQL的模块,其使用方法和MySQLdb几乎相同.但目前pymysql支持python3.x而后者不支持 ...
- python之OS模块详解
python之OS模块详解 ^_^,步入第二个模块世界----->OS 常见函数列表 os.sep:取代操作系统特定的路径分隔符 os.name:指示你正在使用的工作平台.比如对于Windows ...
- python之sys模块详解
python之sys模块详解 sys模块功能多,我们这里介绍一些比较实用的功能,相信你会喜欢的,和我一起走进python的模块吧! sys模块的常见函数列表 sys.argv: 实现从程序外部向程序传 ...
- python中threading模块详解(一)
python中threading模块详解(一) 来源 http://blog.chinaunix.net/uid-27571599-id-3484048.html threading提供了一个比thr ...
- python time 模块详解
Python中time模块详解 发表于2011年5月5日 12:58 a.m. 位于分类我爱Python 在平常的代码中,我们常常需要与时间打交道.在Python中,与时间处理有关的模块就包括: ...
- python time模块详解
python time模块详解 转自:http://blog.csdn.net/kiki113/article/details/4033017 python 的内嵌time模板翻译及说明 一.简介 ...
- 小白的Python之路 day5 time,datatime模块详解
一.模块的分类 可以分成三大类: 1.标准库 2.开源模块 3.自定义模块 二.标准库模块详解 1.time与datetime 在Python中,通常有这几种方式来表示时间:1)时间戳 2)格式化的时 ...
- 小白的Python之路 day5 random模块和string模块详解
random模块详解 一.概述 首先我们看到这个单词是随机的意思,他在python中的主要用于一些随机数,或者需要写一些随机数的代码,下面我们就来整理他的一些用法 二.常用方法 1. random.r ...
随机推荐
- C# 调用外部dll(转)
C# 调用外部dll 一. DLL与应用程序 动态链接库(也称为DLL,即为"Dynamic Link Library"的缩写)是Microsoft Windows最 ...
- T4模板使用
本例使用的数据库是Northwind 1.新建tt文本模板customer.tt 2. 修改customer.tt内容为 <#@ template debug="false" ...
- sql2000/sql2005/sql2008数据库变为0字节修复/MDF文件0字节恢复
[数据恢复故障描述] 这个客户是生产型数据库,数据比较重要,产生量也比较大,客户要求必须尽快修复,保证生产尽快恢复运行.sql数据库文件,由于碎片链接过长,mdf文件突然变为0字节,开始客户尝试自行 ...
- StackExchange.Redis 官方文档(二) Configuration
配置 有多种方式可以配置redis,StackExchange.Redis提供了一个丰富的配置模型,在执行Connect (or ConnectAsync) 时被调用: var conn = Conn ...
- 【Xilinx-Petalinux学习】-07-OpenCV的软硬件处理速度对比
有空了设计一个hls的图像处理IP. 通过hls设计ip模块实现opencv的图像处理. 对比软件和硬件的图像处理速度.
- poj 3641 ——2016——3——15
传送门:http://poj.org/problem?id=3461 题目大意:给你两个字符串p和s,求出p在s中出现的次数. 题解:这一眼看过去就知道是KMP,作为模板来写是最好不过了.... 这道 ...
- iOS 之 二维码生成与扫描(LBXScan)
参考:https://github.com/MxABC/LBXScan 步骤如下: 1. 下载 通过参考网址进行下载. 2. 导入 导入整个LBXScan文件夹 3. 配置 在pch中加入 #impo ...
- 搭建自己的BT下载平台服务器
[原理基础] BT(Bit Torren比特流)是由国外的一名叫Bram Cohen的程序员开发的下载软件,可以说它是目前网络是非常流行的一个多点下载的P2P软件,它最显著的特点就是:下载的人越多,速 ...
- Memo 的当前行、当前列与当前字符
procedure TForm1.Memo1Click(Sender: TObject); begin Text := Format('当前列:%d, 当前行:%d', [Memo1.CaretP ...
- js原生设计模式——2面向对象编程之js原生的链式调用
技巧点:对象方法中返回当前对象就可以链式调用了,即方法中写return this; <!DOCTYPE html><html lang="en"><h ...