314 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
		
		
			
		
	
	
			314 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
|  | # Yivanlee 添加Pass与GBuffer笔记
 | |||
|  | ### 给BaseScalability.ini 添加渲染质量命令行
 | |||
|  | ```ini | |||
|  | [EffectsQuality@0] | |||
|  | [EffectsQuality@1] | |||
|  | r.ToonDataMaterials=0 | |||
|  | 
 | |||
|  | [EffectsQuality@2] | |||
|  | [EffectsQuality@3] | |||
|  | [EffectsQuality@Cine] | |||
|  | r.ToonDataMaterials=1 | |||
|  | ``` | |||
|  | ### 增加bUsesToonData选项
 | |||
|  | 1. MaterialRelevance.h的FMaterialRelevance | |||
|  | 2. HLSLMaterialTranslator.h与HLSLMaterialTranslator.cpp的FHLSLMaterialTranslator类 | |||
|  | 3. MaterialInterface.cpp的UMaterialInterface::GetRelevance_Internal | |||
|  | 4. PrimitiveSceneInfo.cpp的FBatchingSPDI.DrawMesh() | |||
|  | 5. SceneCore.h的FStaticMeshBatchRelevance类 | |||
|  | 
 | |||
|  | ### 定义Stat宏
 | |||
|  | RenderCore.cpp与RenderCore.h里定义ToonDataPass渲染Stat。 | |||
|  | ```c# | |||
|  | //h | |||
|  | DECLARE_CYCLE_STAT_EXTERN(TEXT("ToonData pass drawing"), STAT_ToonDataPassDrawTime, STATGROUP_SceneRendering, RENDERCORE_API); | |||
|  | 
 | |||
|  | //cpp | |||
|  | DEFINE_STAT(STAT_ToonDataPassDrawTime); | |||
|  | ``` | |||
|  | 
 | |||
|  | BasePassRendering.cpp里定义渲染状态宏。 | |||
|  | ```c# | |||
|  | DECLARE_CYCLE_STAT(TEXT("ToonDataPass"), STAT_CLM_ToonDataPass, STATGROUP_CommandListMarkers); | |||
|  | DECLARE_CYCLE_STAT(TEXT("AfterToonDataPass"), STAT_CLM_AfterToonDataPass, STATGROUP_CommandListMarkers); | |||
|  | ``` | |||
|  | 
 | |||
|  | ### 添加渲染用的RT
 | |||
|  | SceneRenderTargets.h与SceneRenderTargets.cpp | |||
|  | ```c++ | |||
|  | //h | |||
|  | TRefCountPtr<IPooledRenderTarget> ToonBufferA; | |||
|  | 
 | |||
|  | //cpp | |||
|  | FSceneRenderTargets::FSceneRenderTargets(const FViewInfo& View, const FSceneRenderTargets& SnapshotSource) | |||
|  | 	: LightAccumulation(GRenderTargetPool.MakeSnapshot(SnapshotSource.LightAccumulation)) | |||
|  |     ··· | |||
|  | 	, ToonBufferA(GRenderTargetPool.MakeSnapshot(SnapshotSource.ToonBufferA)) | |||
|  | ``` | |||
|  | 修改SetupSceneTextureUniformParameters(),在GBuffer代码段中增加`SceneTextureParameters.ToonBufferATexture = bCanReadGBufferUniforms && EnumHasAnyFlags(SetupMode, ESceneTextureSetupMode::GBufferF) && SceneContext.ToonBufferA ? GetRDG(SceneContext.ToonBufferA) : BlackDefault2D;` | |||
|  | 
 | |||
|  | 在SceneTextureParameters.h与SceneTextureParameters.cpp中将新增加的RT添加到FSceneTextureParameters中;并且在GetSceneTextureParameters中注册RT,并在另一个同名函数中添加`Parameters.ToonBufferATexture = (*SceneTextureUniformBuffer)->ToonBufferATexture;`。 | |||
|  | 
 | |||
|  | 在FSceneTextureUniformParameters中添加`SHADER_PARAMETER_RDG_TEXTURE(Texture2D, ToonBufferATexture)` | |||
|  | 
 | |||
|  | ### 添加SceneVisibility中的ToonDataPass定义
 | |||
|  | 在SceneVisibility.h中的MarkRelevant()添加 | |||
|  | ```c# | |||
|  | if (StaticMeshRelevance.bUseToonData) | |||
|  | { | |||
|  |     DrawCommandPacket.AddCommandsForMesh(PrimitiveIndex, PrimitiveSceneInfo, StaticMeshRelevance, StaticMesh, Scene, bCanCache, EMeshPass::ToonDataPass); | |||
|  | } | |||
|  | ``` | |||
|  | 
 | |||
|  | 在ComputeDynamicMeshRelevance()中添加 | |||
|  | ```c# | |||
|  | if (ViewRelevance.bUsesToonData) | |||
|  | { | |||
|  |     PassMask.Set(EMeshPass::ToonDataPass); | |||
|  |     View.NumVisibleDynamicMeshElements[EMeshPass::ToonDataPass] += NumElements; | |||
|  | } | |||
|  | ``` | |||
|  |    | |||
|  | #### 修改DecodeGBufferData()以及相关函数
 | |||
|  | - 修改RayTracingDeferredShadingCommon.ush的DecodeGBufferData() | |||
|  | - 修改DeferredShadingCommon.ush中的FGBufferData,添加ToonDataA变量,并修改DecodeGBufferData()、GetGBufferDataUint()、GetGBufferData()、 | |||
|  | - 修改SceneTextureParameters.ush中的ToonData变量声明:`Texture2D ToonBufferATexture;`、`#define ToonBufferATextureSampler GlobalPointClampedSampler`以及`GetGBufferDataFromSceneTextures();`;SceneTexturesCommon.ush中的`#define SceneTexturesStruct_ToonBufferATextureSampler SceneTexturesStruct.PointClampSampler` | |||
|  |    | |||
|  | ### 增加ToonDataPass MeshDrawPass已实现增加GBuffer
 | |||
|  | - 在MeshPassProcessor.h增加ToonDataPass MeshDrawPass定义。 | |||
|  | - 在DeferredShadingRenderer.h添加渲染函数声明。 | |||
|  | - 在新添加的ToonDataRendering.h与ToonDataRendering.cpp中添加MeshDrawPass声明与定义。 | |||
|  | - 在ToonDataPassShader.usf中实现 | |||
|  | 
 | |||
|  | ``` | |||
|  | // Copyright Epic Games, Inc. All Rights Reserved. | |||
|  | 
 | |||
|  | /*============================================================================= | |||
|  | 	AnisotropyPassShader.usf: Outputs Anisotropy and World Tangent to GBufferF | |||
|  | =============================================================================*/ | |||
|  | 
 | |||
|  | #include "Common.ush"
 | |||
|  | #include "/Engine/Generated/Material.ush"
 | |||
|  | #include "/Engine/Generated/VertexFactory.ush"
 | |||
|  | #include "DeferredShadingCommon.ush"
 | |||
|  | 
 | |||
|  | struct FToonDataPassVSToPS | |||
|  | { | |||
|  | 	float4 Position : SV_POSITION; | |||
|  | 	FVertexFactoryInterpolantsVSToPS Interps; | |||
|  | 
 | |||
|  | 	#if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS | |||
|  | 		float3 PixelPositionExcludingWPO : TEXCOORD7; | |||
|  | 	#endif | |||
|  | }; | |||
|  | 
 | |||
|  | #if USING_TESSELLATION
 | |||
|  | 	struct FAnisotropyPassVSToDS | |||
|  | 	{ | |||
|  | 		FVertexFactoryInterpolantsVSToDS FactoryInterpolants; | |||
|  | 		float4 Position : VS_To_DS_Position; | |||
|  | 		#if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS | |||
|  | 			float3 PixelPositionExcludingWPO : TEXCOORD7; | |||
|  | 		#endif | |||
|  | 		OPTIONAL_VertexID_VS_To_DS | |||
|  | 	}; | |||
|  | 	 | |||
|  | 	#define FVertexOutput FAnisotropyPassVSToDS | |||
|  | 	#define VertexFactoryGetInterpolants VertexFactoryGetInterpolantsVSToDS | |||
|  | #else
 | |||
|  | 	#define FVertexOutput FToonDataPassVSToPS | |||
|  | 	#define VertexFactoryGetInterpolants VertexFactoryGetInterpolantsVSToPS | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if USING_TESSELLATION
 | |||
|  | 	#define FPassSpecificVSToDS FAnisotropyPassVSToDS | |||
|  | 	#define FPassSpecificVSToPS FToonDataPassVSToPS | |||
|  | 		 | |||
|  | 	FAnisotropyPassVSToDS PassInterpolate(FAnisotropyPassVSToDS a, float aInterp, FAnisotropyPassVSToDS b, float bInterp) | |||
|  | 	{ | |||
|  | 		FAnisotropyPassVSToDS O; | |||
|  | 
 | |||
|  | 		O.FactoryInterpolants = VertexFactoryInterpolate(a.FactoryInterpolants, aInterp, b.FactoryInterpolants, bInterp); | |||
|  | 		 | |||
|  | 		#if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS | |||
|  | 			TESSELLATION_INTERPOLATE_MEMBER(PixelPositionExcludingWPO); | |||
|  | 		#endif | |||
|  | 
 | |||
|  | 		return O; | |||
|  | 	} | |||
|  | 	 | |||
|  | 	FToonDataPassVSToPS PassFinalizeTessellationOutput(FAnisotropyPassVSToDS Interpolants, float4 WorldPosition, FMaterialTessellationParameters MaterialParameters) | |||
|  | 	{ | |||
|  | 		FToonDataPassVSToPS O; | |||
|  | 		 | |||
|  | 		O.Interps = VertexFactoryAssignInterpolants(Interpolants.FactoryInterpolants); | |||
|  | 		O.Position = mul(WorldPosition, ResolvedView.TranslatedWorldToClip); | |||
|  | 
 | |||
|  | 		#if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS | |||
|  | 			O.PixelPositionExcludingWPO = Interpolants.PixelPositionExcludingWPO; | |||
|  | 		#endif | |||
|  | 
 | |||
|  | 		return O; | |||
|  | 	} | |||
|  | 	 | |||
|  | 	#include "Tessellation.ush" | |||
|  | #endif
 | |||
|  | 	 | |||
|  | /*============================================================================= | |||
|  |  * Vertex Shader | |||
|  |  *============================================================================*/ | |||
|  | 
 | |||
|  | void MainVertexShader( | |||
|  | 	FVertexFactoryInput Input, | |||
|  | 	OPTIONAL_VertexID | |||
|  | 	out FVertexOutput Output  | |||
|  | #if USE_GLOBAL_CLIP_PLANE && !USING_TESSELLATION
 | |||
|  | 	, out float OutGlobalClipPlaneDistance : SV_ClipDistance | |||
|  | #endif
 | |||
|  | #if INSTANCED_STEREO
 | |||
|  | 	, uint InstanceId : SV_InstanceID | |||
|  | 	#if !MULTI_VIEW | |||
|  | 		, out float OutClipDistance : SV_ClipDistance1 | |||
|  | 	#else | |||
|  | 		, out uint ViewportIndex : SV_ViewPortArrayIndex | |||
|  | 	#endif | |||
|  | #endif
 | |||
|  | 	) | |||
|  | { | |||
|  | #if INSTANCED_STEREO
 | |||
|  | 	const uint EyeIndex = GetEyeIndex(InstanceId); | |||
|  | 	ResolvedView = ResolveView(EyeIndex); | |||
|  | 	#if !MULTI_VIEW | |||
|  | 		OutClipDistance = 0.0; | |||
|  | 	#else | |||
|  | 		ViewportIndex = EyeIndex; | |||
|  | 	#endif | |||
|  | #else
 | |||
|  | 	uint EyeIndex = 0; | |||
|  | 	ResolvedView = ResolveView(); | |||
|  | #endif
 | |||
|  | 	 | |||
|  | 	FVertexFactoryIntermediates VFIntermediates = GetVertexFactoryIntermediates(Input); | |||
|  | 	float4 WorldPos = VertexFactoryGetWorldPosition(Input, VFIntermediates); | |||
|  | 	float4 WorldPositionExcludingWPO = WorldPos; | |||
|  | 
 | |||
|  | 	float3x3 TangentToLocal = VertexFactoryGetTangentToLocal(Input, VFIntermediates); | |||
|  | 	FMaterialVertexParameters VertexParameters = GetMaterialVertexParameters(Input, VFIntermediates, WorldPos.xyz, TangentToLocal); | |||
|  | 
 | |||
|  | 	// Isolate instructions used for world position offset | |||
|  | 	// As these cause the optimizer to generate different position calculating instructions in each pass, resulting in self-z-fighting. | |||
|  | 	// This is only necessary for shaders used in passes that have depth testing enabled. | |||
|  | 	{ | |||
|  | 		WorldPos.xyz += GetMaterialWorldPositionOffset(VertexParameters); | |||
|  | 	} | |||
|  | 
 | |||
|  | #if USING_TESSELLATION
 | |||
|  | 	// Transformation is done in Domain shader when tessellating | |||
|  | 	Output.Position = WorldPos;  | |||
|  | #else
 | |||
|  | 	{ | |||
|  | 		float4 RasterizedWorldPosition = VertexFactoryGetRasterizedWorldPosition(Input, VFIntermediates, WorldPos); | |||
|  | 	#if ODS_CAPTURE | |||
|  | 		float3 ODS = OffsetODS(RasterizedWorldPosition.xyz, ResolvedView.TranslatedWorldCameraOrigin.xyz, ResolvedView.StereoIPD); | |||
|  | 		Output.Position = INVARIANT(mul(float4(RasterizedWorldPosition.xyz + ODS, 1.0), ResolvedView.TranslatedWorldToClip)); | |||
|  | 	#else | |||
|  | 		Output.Position = INVARIANT(mul(RasterizedWorldPosition, ResolvedView.TranslatedWorldToClip)); | |||
|  | 	#endif | |||
|  | 	} | |||
|  | 
 | |||
|  | 	#if INSTANCED_STEREO && !MULTI_VIEW | |||
|  | 		BRANCH  | |||
|  | 		if (IsInstancedStereo())   | |||
|  | 		{ | |||
|  | 			// Clip at the center of the screen | |||
|  | 			OutClipDistance = dot(Output.Position, EyeClipEdge[EyeIndex]); | |||
|  | 
 | |||
|  | 			// Scale to the width of a single eye viewport | |||
|  | 			Output.Position.x *= 0.5 * ResolvedView.HMDEyePaddingOffset; | |||
|  | 
 | |||
|  | 			// Shift to the eye viewport | |||
|  | 			Output.Position.x += (EyeOffsetScale[EyeIndex] * Output.Position.w) * (1.0f - 0.5 * ResolvedView.HMDEyePaddingOffset); | |||
|  | 		} | |||
|  | 	#elif XBOXONE_BIAS_HACK | |||
|  | 		// XB1 needs a bias in the opposite direction to fix FORT-40853 | |||
|  | 		// XBOXONE_BIAS_HACK is defined only in a custom node in a particular material | |||
|  | 		// This should be removed with a future shader compiler update | |||
|  | 		Output.Position.z -= 0.0001 * Output.Position.w; | |||
|  | 	#endif | |||
|  | 
 | |||
|  | 	#if USE_GLOBAL_CLIP_PLANE | |||
|  | 		OutGlobalClipPlaneDistance = dot(ResolvedView.GlobalClippingPlane, float4(WorldPos.xyz - ResolvedView.PreViewTranslation.xyz, 1)); | |||
|  | 	#endif | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if USING_TESSELLATION
 | |||
|  | 	Output.FactoryInterpolants = VertexFactoryGetInterpolants( Input, VFIntermediates, VertexParameters ); | |||
|  | #else
 | |||
|  | 	Output.Interps = VertexFactoryGetInterpolants(Input, VFIntermediates, VertexParameters); | |||
|  | #endif // #if USING_TESSELLATION
 | |||
|  | 
 | |||
|  | #if INSTANCED_STEREO
 | |||
|  | 	#if USING_TESSELLATION	 | |||
|  | 		Output.Interps.InterpolantsVSToPS.EyeIndex = EyeIndex; | |||
|  | 	#else | |||
|  | 		Output.Interps.EyeIndex = EyeIndex; | |||
|  | 	#endif | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS
 | |||
|  | 	Output.PixelPositionExcludingWPO = WorldPositionExcludingWPO.xyz; | |||
|  | #endif	
 | |||
|  | 
 | |||
|  | 	OutputVertexID( Output ); | |||
|  | } | |||
|  | 
 | |||
|  | /*============================================================================= | |||
|  |  * Pixel Shader | |||
|  |  *============================================================================*/ | |||
|  | 
 | |||
|  | void MainPixelShader( | |||
|  | 	in INPUT_POSITION_QUALIFIERS float4 SvPosition : SV_Position, | |||
|  | 	FVertexFactoryInterpolantsVSToPS Input | |||
|  | #if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS
 | |||
|  | 	, float3 PixelPositionExcludingWPO : TEXCOORD7 | |||
|  | #endif
 | |||
|  | 	OPTIONAL_IsFrontFace | |||
|  | 	OPTIONAL_OutDepthConservative | |||
|  | 	, out float4 ToonBufferA : SV_Target0 | |||
|  | #if MATERIALBLENDING_MASKED_USING_COVERAGE
 | |||
|  | 	, out uint OutCoverage : SV_Coverage | |||
|  | #endif
 | |||
|  | 	) | |||
|  | { | |||
|  | #if INSTANCED_STEREO
 | |||
|  | 	ResolvedView = ResolveView(Input.EyeIndex); | |||
|  | #else
 | |||
|  | 	ResolvedView = ResolveView(); | |||
|  | #endif
 | |||
|  | 
 | |||
|  | 	// Manual clipping here (alpha-test, etc) | |||
|  | 	FMaterialPixelParameters MaterialParameters = GetMaterialPixelParameters(Input, SvPosition); | |||
|  | 	FPixelMaterialInputs PixelMaterialInputs; | |||
|  | 
 | |||
|  | 	#if USE_WORLD_POSITION_EXCLUDING_SHADER_OFFSETS | |||
|  | 		float4 ScreenPosition = SvPositionToResolvedScreenPosition(SvPosition); | |||
|  | 		float3 TranslatedWorldPosition = SvPositionToResolvedTranslatedWorld(SvPosition); | |||
|  | 		CalcMaterialParametersEx(MaterialParameters, PixelMaterialInputs, SvPosition, ScreenPosition, bIsFrontFace, TranslatedWorldPosition, PixelPositionExcludingWPO);	 | |||
|  | 	#else | |||
|  | 		CalcMaterialParameters(MaterialParameters, PixelMaterialInputs, SvPosition, bIsFrontFace); | |||
|  | 	#endif | |||
|  | 
 | |||
|  | #if OUTPUT_PIXEL_DEPTH_OFFSET
 | |||
|  | 	ApplyPixelDepthOffsetToMaterialParameters(MaterialParameters, PixelMaterialInputs, OutDepth); | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if MATERIALBLENDING_MASKED_USING_COVERAGE
 | |||
|  | 	OutCoverage = DiscardMaterialWithPixelCoverage(MaterialParameters, PixelMaterialInputs); | |||
|  | #endif
 | |||
|  | 
 | |||
|  | 	//float Anisotropy = GetMaterialAnisotropy(PixelMaterialInputs); | |||
|  | 	//float3 WorldTangent = MaterialParameters.WorldTangent; | |||
|  | 	 | |||
|  | 	ToonBufferA = float4(0.2, 0.1, 0.8, 1.0); | |||
|  | } | |||
|  | ``` |