348 lines
11 KiB
HLSL
348 lines
11 KiB
HLSL
#ifndef UNITY_SPACE_TRANSFORMS_INCLUDED
|
|
#define UNITY_SPACE_TRANSFORMS_INCLUDED
|
|
|
|
#if SHADER_API_MOBILE || SHADER_API_GLES3 || SHADER_API_SWITCH || defined(UNITY_UNIFIED_SHADER_PRECISION_MODEL)
|
|
#pragma warning (disable : 3205) // conversion of larger type to smaller
|
|
#endif
|
|
|
|
// Caution: For HDRP, adding a function in this file requires adding the appropriate #define in PickingSpaceTransforms.hlsl
|
|
|
|
// Return the PreTranslated ObjectToWorld Matrix (i.e matrix with _WorldSpaceCameraPos apply to it if we use camera relative rendering)
|
|
float4x4 GetObjectToWorldMatrix()
|
|
{
|
|
return UNITY_MATRIX_M;
|
|
}
|
|
|
|
float4x4 GetWorldToObjectMatrix()
|
|
{
|
|
return UNITY_MATRIX_I_M;
|
|
}
|
|
|
|
float4x4 GetPrevObjectToWorldMatrix()
|
|
{
|
|
return UNITY_PREV_MATRIX_M;
|
|
}
|
|
|
|
float4x4 GetPrevWorldToObjectMatrix()
|
|
{
|
|
return UNITY_PREV_MATRIX_I_M;
|
|
}
|
|
|
|
float4x4 GetWorldToViewMatrix()
|
|
{
|
|
return UNITY_MATRIX_V;
|
|
}
|
|
|
|
float4x4 GetViewToWorldMatrix()
|
|
{
|
|
return UNITY_MATRIX_I_V;
|
|
}
|
|
|
|
// Transform to homogenous clip space
|
|
float4x4 GetWorldToHClipMatrix()
|
|
{
|
|
return UNITY_MATRIX_VP;
|
|
}
|
|
|
|
// Transform to homogenous clip space
|
|
float4x4 GetViewToHClipMatrix()
|
|
{
|
|
return UNITY_MATRIX_P;
|
|
}
|
|
|
|
// This function always return the absolute position in WS
|
|
float3 GetAbsolutePositionWS(float3 positionRWS)
|
|
{
|
|
#if (SHADEROPTIONS_CAMERA_RELATIVE_RENDERING != 0)
|
|
positionRWS += _WorldSpaceCameraPos.xyz;
|
|
#endif
|
|
return positionRWS;
|
|
}
|
|
|
|
// This function return the camera relative position in WS
|
|
float3 GetCameraRelativePositionWS(float3 positionWS)
|
|
{
|
|
#if (SHADEROPTIONS_CAMERA_RELATIVE_RENDERING != 0)
|
|
positionWS -= _WorldSpaceCameraPos.xyz;
|
|
#endif
|
|
return positionWS;
|
|
}
|
|
|
|
real GetOddNegativeScale()
|
|
{
|
|
// FIXME: We should be able to just return unity_WorldTransformParams.w, but it is not
|
|
// properly set at the moment, when doing ray-tracing; once this has been fixed in cpp,
|
|
// we can revert back to the former implementation.
|
|
return unity_WorldTransformParams.w >= 0.0 ? 1.0 : -1.0;
|
|
}
|
|
|
|
float3 TransformObjectToWorld(float3 positionOS)
|
|
{
|
|
#if defined(SHADER_STAGE_RAY_TRACING)
|
|
return mul(ObjectToWorld3x4(), float4(positionOS, 1.0)).xyz;
|
|
#else
|
|
return mul(GetObjectToWorldMatrix(), float4(positionOS, 1.0)).xyz;
|
|
#endif
|
|
}
|
|
|
|
float3 TransformWorldToObject(float3 positionWS)
|
|
{
|
|
#if defined(SHADER_STAGE_RAY_TRACING)
|
|
return mul(WorldToObject3x4(), float4(positionWS, 1.0)).xyz;
|
|
#else
|
|
return mul(GetWorldToObjectMatrix(), float4(positionWS, 1.0)).xyz;
|
|
#endif
|
|
}
|
|
|
|
float3 TransformWorldToView(float3 positionWS)
|
|
{
|
|
return mul(GetWorldToViewMatrix(), float4(positionWS, 1.0)).xyz;
|
|
}
|
|
|
|
float3 TransformViewToWorld(float3 positionVS)
|
|
{
|
|
return mul(GetViewToWorldMatrix(), float4(positionVS, 1.0)).xyz;
|
|
}
|
|
|
|
// Transforms position from object space to homogenous space
|
|
float4 TransformObjectToHClip(float3 positionOS)
|
|
{
|
|
// More efficient than computing M*VP matrix product
|
|
return mul(GetWorldToHClipMatrix(), mul(GetObjectToWorldMatrix(), float4(positionOS, 1.0)));
|
|
}
|
|
|
|
// Transforms position from world space to homogenous space
|
|
float4 TransformWorldToHClip(float3 positionWS)
|
|
{
|
|
return mul(GetWorldToHClipMatrix(), float4(positionWS, 1.0));
|
|
}
|
|
|
|
// Transforms position from view space to homogenous space
|
|
float4 TransformWViewToHClip(float3 positionVS)
|
|
{
|
|
return mul(GetViewToHClipMatrix(), float4(positionVS, 1.0));
|
|
}
|
|
|
|
// Normalize to support uniform scaling
|
|
float3 TransformObjectToWorldDir(float3 dirOS, bool doNormalize = true)
|
|
{
|
|
#ifndef SHADER_STAGE_RAY_TRACING
|
|
float3 dirWS = mul((float3x3)GetObjectToWorldMatrix(), dirOS);
|
|
#else
|
|
float3 dirWS = mul((float3x3)ObjectToWorld3x4(), dirOS);
|
|
#endif
|
|
if (doNormalize)
|
|
return SafeNormalize(dirWS);
|
|
|
|
return dirWS;
|
|
}
|
|
|
|
// Normalize to support uniform scaling
|
|
float3 TransformWorldToObjectDir(float3 dirWS, bool doNormalize = true)
|
|
{
|
|
#ifndef SHADER_STAGE_RAY_TRACING
|
|
float3 dirOS = mul((float3x3)GetWorldToObjectMatrix(), dirWS);
|
|
#else
|
|
float3 dirOS = mul((float3x3)WorldToObject3x4(), dirWS);
|
|
#endif
|
|
if (doNormalize)
|
|
return normalize(dirOS);
|
|
|
|
return dirOS;
|
|
}
|
|
|
|
// Transforms vector from world space to view space
|
|
real3 TransformWorldToViewDir(real3 dirWS, bool doNormalize = false)
|
|
{
|
|
float3 dirVS = mul((real3x3)GetWorldToViewMatrix(), dirWS).xyz;
|
|
if (doNormalize)
|
|
return normalize(dirVS);
|
|
|
|
return dirVS;
|
|
}
|
|
|
|
// Transforms vector from view space to world space
|
|
real3 TransformViewToWorldDir(real3 dirVS, bool doNormalize = false)
|
|
{
|
|
float3 dirWS = mul((real3x3)GetViewToWorldMatrix(), dirVS).xyz;
|
|
if (doNormalize)
|
|
return normalize(dirWS);
|
|
|
|
return dirWS;
|
|
}
|
|
|
|
// Transforms normal from world space to view space
|
|
real3 TransformWorldToViewNormal(real3 normalWS, bool doNormalize = false)
|
|
{
|
|
// assuming view matrix is uniformly scaled, we can use direction transform
|
|
return TransformWorldToViewDir(normalWS, doNormalize);
|
|
}
|
|
|
|
// Transforms normal from view space to world space
|
|
real3 TransformViewToWorldNormal(real3 normalVS, bool doNormalize = false)
|
|
{
|
|
// assuming view matrix is uniformly scaled, we can use direction transform
|
|
return TransformViewToWorldDir(normalVS, doNormalize);
|
|
}
|
|
|
|
// Transforms vector from world space to homogenous space
|
|
real3 TransformWorldToHClipDir(real3 directionWS, bool doNormalize = false)
|
|
{
|
|
float3 dirHCS = mul((real3x3)GetWorldToHClipMatrix(), directionWS).xyz;
|
|
if (doNormalize)
|
|
return normalize(dirHCS);
|
|
|
|
return dirHCS;
|
|
}
|
|
|
|
// Transforms normal from object to world space
|
|
float3 TransformObjectToWorldNormal(float3 normalOS, bool doNormalize = true)
|
|
{
|
|
#ifdef UNITY_ASSUME_UNIFORM_SCALING
|
|
return TransformObjectToWorldDir(normalOS, doNormalize);
|
|
#else
|
|
// Normal need to be multiply by inverse transpose
|
|
float3 normalWS = mul(normalOS, (float3x3)GetWorldToObjectMatrix());
|
|
if (doNormalize)
|
|
return SafeNormalize(normalWS);
|
|
|
|
return normalWS;
|
|
#endif
|
|
}
|
|
|
|
// Transforms normal from world to object space
|
|
float3 TransformWorldToObjectNormal(float3 normalWS, bool doNormalize = true)
|
|
{
|
|
#ifdef UNITY_ASSUME_UNIFORM_SCALING
|
|
return TransformWorldToObjectDir(normalWS, doNormalize);
|
|
#else
|
|
// Normal need to be multiply by inverse transpose
|
|
float3 normalOS = mul(normalWS, (float3x3)GetObjectToWorldMatrix());
|
|
if (doNormalize)
|
|
return SafeNormalize(normalOS);
|
|
|
|
return normalOS;
|
|
#endif
|
|
}
|
|
|
|
real3x3 CreateTangentToWorld(real3 normal, real3 tangent, real flipSign)
|
|
{
|
|
// For odd-negative scale transforms we need to flip the sign
|
|
real sgn = flipSign * GetOddNegativeScale();
|
|
real3 bitangent = cross(normal, tangent) * sgn;
|
|
|
|
return real3x3(tangent, bitangent, normal);
|
|
}
|
|
|
|
// this function is intended to work on Normals (handles non-uniform scale)
|
|
// tangentToWorld is the matrix representing the transformation of a normal from tangent to world space
|
|
real3 TransformTangentToWorld(float3 normalTS, real3x3 tangentToWorld, bool doNormalize = false)
|
|
{
|
|
// Note matrix is in row major convention with left multiplication as it is build on the fly
|
|
real3 result = mul(normalTS, tangentToWorld);
|
|
if (doNormalize)
|
|
return SafeNormalize(result);
|
|
return result;
|
|
}
|
|
|
|
// this function is intended to work on Normals (handles non-uniform scale)
|
|
// This function does the exact inverse of TransformTangentToWorld() and is
|
|
// also decribed within comments in mikktspace.h and it follows implicitly
|
|
// from the scalar triple product (google it).
|
|
// tangentToWorld is the matrix representing the transformation of a normal from tangent to world space
|
|
real3 TransformWorldToTangent(real3 normalWS, real3x3 tangentToWorld, bool doNormalize = true)
|
|
{
|
|
// Note matrix is in row major convention with left multiplication as it is build on the fly
|
|
float3 row0 = tangentToWorld[0];
|
|
float3 row1 = tangentToWorld[1];
|
|
float3 row2 = tangentToWorld[2];
|
|
|
|
// these are the columns of the inverse matrix but scaled by the determinant
|
|
float3 col0 = cross(row1, row2);
|
|
float3 col1 = cross(row2, row0);
|
|
float3 col2 = cross(row0, row1);
|
|
|
|
float determinant = dot(row0, col0);
|
|
|
|
// inverse transposed but scaled by determinant
|
|
// Will remove transpose part by using matrix as the first arg in the mul() below
|
|
// this makes it the exact inverse of what TransformTangentToWorld() does.
|
|
real3x3 matTBN_I_T = real3x3(col0, col1, col2);
|
|
real3 result = mul(matTBN_I_T, normalWS);
|
|
if (doNormalize)
|
|
{
|
|
float sgn = determinant < 0.0 ? (-1.0) : 1.0;
|
|
return SafeNormalize(sgn * result);
|
|
}
|
|
else
|
|
return result / determinant;
|
|
}
|
|
|
|
// this function is intended to work on Vectors/Directions
|
|
// tangentToWorld is the matrix representing the transformation of a normal from tangent to world space
|
|
real3 TransformWorldToTangentDir(real3 dirWS, real3x3 tangentToWorld, bool doNormalize = false)
|
|
{
|
|
// Note matrix is in row major convention with left multiplication as it is build on the fly
|
|
real3 result = mul(tangentToWorld, dirWS);
|
|
if (doNormalize)
|
|
return SafeNormalize(result);
|
|
return result;
|
|
}
|
|
|
|
// this function is intended to work on Vectors/Directions
|
|
// This function does the exact inverse of TransformWorldToTangentDir()
|
|
// tangentToWorld is the matrix representing the transformation of a normal from tangent to world space
|
|
real3 TransformTangentToWorldDir(real3 dirWS, real3x3 tangentToWorld, bool doNormalize = false)
|
|
{
|
|
// Note matrix is in row major convention with left multiplication as it is build on the fly
|
|
float3 row0 = tangentToWorld[0];
|
|
float3 row1 = tangentToWorld[1];
|
|
float3 row2 = tangentToWorld[2];
|
|
|
|
// these are the columns of the inverse matrix but scaled by the determinant
|
|
float3 col0 = cross(row1, row2);
|
|
float3 col1 = cross(row2, row0);
|
|
float3 col2 = cross(row0, row1);
|
|
|
|
float determinant = dot(row0, col0);
|
|
|
|
// inverse transposed but scaled by determinant
|
|
// Will remove transpose part by using matrix as the second arg in the mul() below
|
|
// this makes it the exact inverse of what TransformWorldToTangentDir() does.
|
|
real3x3 matTBN_I_T = real3x3(col0, col1, col2);
|
|
real3 result = mul(dirWS, matTBN_I_T);
|
|
if (doNormalize)
|
|
{
|
|
float sgn = determinant < 0.0 ? (-1.0) : 1.0;
|
|
return SafeNormalize(sgn * result);
|
|
}
|
|
else
|
|
return result / determinant;
|
|
}
|
|
|
|
// tangentToWorld is the matrix representing the transformation of a normal from tangent to world space
|
|
real3 TransformTangentToObject(real3 dirTS, real3x3 tangentToWorld)
|
|
{
|
|
// Note matrix is in row major convention with left multiplication as it is build on the fly
|
|
real3 normalWS = TransformTangentToWorld(dirTS, tangentToWorld);
|
|
return TransformWorldToObjectNormal(normalWS);
|
|
}
|
|
|
|
// tangentToWorld is the matrix representing the transformation of a normal from tangent to world space
|
|
real3 TransformObjectToTangent(real3 dirOS, real3x3 tangentToWorld)
|
|
{
|
|
// Note matrix is in row major convention with left multiplication as it is build on the fly
|
|
|
|
// don't normalize, as normalWS will be normalized after TransformWorldToTangent
|
|
float3 normalWS = TransformObjectToWorldNormal(dirOS, false);
|
|
|
|
// transform from world to tangent
|
|
return TransformWorldToTangent(normalWS, tangentToWorld);
|
|
}
|
|
|
|
#if SHADER_API_MOBILE || SHADER_API_GLES3 || SHADER_API_SWITCH
|
|
#pragma warning (enable : 3205) // conversion of larger type to smaller
|
|
#endif
|
|
|
|
#endif
|