UNREAL ENGINE

[짜투리] 커스텀 셰이더 익스프레션 사용할 때 Pow(a,b)

jplee 2023. 12. 16. 22:17

일단 유니티 에서는 아래와 같은 매크로 형식으로 정의 되어 있다.

// Using pow often result to a warning like this
// "pow(f, e) will not work for negative f, use abs(f) or conditionally handle negative values if you expect them"
// PositivePow remove this warning when you know the value is positive or 0 and avoid inf/NAN.
// Note: https://msdn.microsoft.com/en-us/library/windows/desktop/bb509636(v=vs.85).aspx pow(0, >0) == 0
TEMPLATE_2_REAL(PositivePow, base, power, return pow(abs(base), power))

// pow를 사용하면 종종 다음과 같은 경고가 표시됩니다. 
// "pow(f, e)는 음수 f에 작동하지 않으므로, 음수 값이 예상되는 경우 abs(f)를 사용하거나 조건부로 처리하십시오." 
// PositivePow는 값이 양수 또는 0이라는 것을 알면 이 경고를 제거하고 inf/NAN을 피합니다.

암튼... 나같은 경우에는

언리얼 엔진의 내장 셰이더를 보면 common.ush 에서 찾을 수 있다.

#define POW_CLAMP 0.000001f

// Clamp the base, so it's never <= 0.0f (INF/NaN).
MaterialFloat ClampedPow(MaterialFloat X,MaterialFloat Y)
{
	return pow(max(abs(X),POW_CLAMP),Y);
}
MaterialFloat2 ClampedPow(MaterialFloat2 X,MaterialFloat2 Y)
{
	return pow(max(abs(X),MaterialFloat2(POW_CLAMP,POW_CLAMP)),Y);
}
MaterialFloat3 ClampedPow(MaterialFloat3 X,MaterialFloat3 Y)
{
	return pow(max(abs(X),MaterialFloat3(POW_CLAMP,POW_CLAMP,POW_CLAMP)),Y);
}  
MaterialFloat4 ClampedPow(MaterialFloat4 X,MaterialFloat4 Y)
{
	return pow(max(abs(X),MaterialFloat4(POW_CLAMP,POW_CLAMP,POW_CLAMP,POW_CLAMP)),Y);
} 

/* Pow function that will return 0 if Base is <=0. This ensures that no compiler expands pow into exp(Exponent * log(Base)) with Base=0 */
MaterialFloat PositiveClampedPow(MaterialFloat Base, MaterialFloat Exponent)
{
	return (Base <= 0.0f) ? 0.0f : pow(Base, Exponent);
}
MaterialFloat2 PositiveClampedPow(MaterialFloat2 Base, MaterialFloat2 Exponent)
{
	return MaterialFloat2(PositiveClampedPow(Base.x, Exponent.x), PositiveClampedPow(Base.y, Exponent.y)); 
}
MaterialFloat3 PositiveClampedPow(MaterialFloat3 Base, MaterialFloat3 Exponent)
{
	return MaterialFloat3(PositiveClampedPow(Base.xy, Exponent.xy), PositiveClampedPow(Base.z, Exponent.z)); 
}  
MaterialFloat4 PositiveClampedPow(MaterialFloat4 Base, MaterialFloat4 Exponent)
{
	return MaterialFloat4(PositiveClampedPow(Base.xy, Exponent.xy), PositiveClampedPow(Base.zw, Exponent.zw)); 
}

 

개인적으로 Utility.ush 파일을 따로 만들어서 사용하는 편인지라... 

#pragma once
#include "/Engine/Public/Platform.ush"
#include"/Engine/Private/Common.ush"

#define POSITIVEPOW ClampedPow

간단히 이렇게 디파인 해서 익스프레션으로 셰이더 작성할 때 사용하는 편을 선호 한다.

만약 Common.ush 를 include 해서 사용하지 않을 경우에는..

#pragma once
// Utility shader
#define HALF_MAX        65504.0 // (2 - 2^-10) * 2^15
#define HALF_MAX_MINUS1 65472.0 // (2 - 2^-9) * 2^15
#define EPSILON         1.0e-4
#define PI              3.14159265359
#define TWO_PI          6.28318530718
#define FOUR_PI         12.56637061436
#define INV_PI          0.31830988618
#define INV_TWO_PI      0.15915494309
#define INV_FOUR_PI     0.07957747155
#define HALF_PI         1.57079632679
#define INV_HALF_PI     0.636619772367

#define FLT_EPSILON     1.192092896e-07 // Smallest positive number, such that 1.0 + FLT_EPSILON != 1.0
#define FLT_MIN         1.175494351e-38 // Minimum representable positive floating-point number
#define FLT_MAX         3.402823466e+38 // Maximum representable floating-point number

// pow를 사용하면 종종 다음과 같은 경고가 표시됩니다.
// "pow(f, e)는 음수 f에 대해 작동하지 않으므로, 음수 값이 예상되는 경우 abs(f)를 사용하거나 조건부로 처리하십시오."
// 값이 양수라는 것을 알면 PositivePow는 이 경고를 제거하고 inf/NAN을 피합니다.

float PositivePow(float base, float power)
{
    return pow(max(abs(base), float(FLT_EPSILON)), power);
}

float2 PositivePow(float2 base, float2 power)
{
    return pow(max(abs(base), float2(FLT_EPSILON, FLT_EPSILON)), power);
}

float3 PositivePow(float3 base, float3 power)
{
    return pow(max(abs(base), float3(FLT_EPSILON, FLT_EPSILON, FLT_EPSILON)), power);
}

float4 PositivePow(float4 base, float4 power)
{
    return pow(max(abs(base), float4(FLT_EPSILON, FLT_EPSILON, FLT_EPSILON, FLT_EPSILON)), power);
}

#define POSITIVEPOW PositivePow

//--------------------------------------------------------------------------------------------------

뭐 이런 식으로 만들어서 사용했었는데... 내 생각에 굳이 만들어 쓸 일은 없는 것 같고... 원래 언리얼 엔진에서 제공하는 common.ush 있는거 바로 활용하면 되겠다.