From 2807e0f4f1db04fcf3851918d654d265338c7fe5 Mon Sep 17 00:00:00 2001 From: Riley Alston Date: Wed, 20 Nov 2024 10:16:53 -0800 Subject: [PATCH] Ray tMax NaN Fix --- .../rtxdi/RtxdiApplicationBridge.slangh | 1 + src/dxvk/shaders/rtx/concept/ray/ray.slangh | 10 +++-- src/dxvk/shaders/rtx/utility/math.slangh | 40 +++++++++++++------ submodules/rtxdi | 2 +- 4 files changed, 36 insertions(+), 17 deletions(-) diff --git a/src/dxvk/shaders/rtx/algorithm/rtxdi/RtxdiApplicationBridge.slangh b/src/dxvk/shaders/rtx/algorithm/rtxdi/RtxdiApplicationBridge.slangh index 94f2c1c1..dbaf7b9e 100644 --- a/src/dxvk/shaders/rtx/algorithm/rtxdi/RtxdiApplicationBridge.slangh +++ b/src/dxvk/shaders/rtx/algorithm/rtxdi/RtxdiApplicationBridge.slangh @@ -582,6 +582,7 @@ DecodedPolymorphicLight RAB_GetDecodedPolymorphicLightWithTypeHint(uint lightID, return decodedPolymorphicLight; } +// Note: Assumes lightID is valid (e.g. not the invalid light ID reservoirs use to indicate they are invalid). RAB_LightSample RAB_GetLightSample(uint lightID, float2 lightUV, MinimalSurfaceInteraction minimalSurfaceInteraction, bool usePreviousLights = false) { const MemoryPolymorphicLight memoryPolymorphicLight = RAB_GetMemoryPolymorphicLight(lightID, usePreviousLights); diff --git a/src/dxvk/shaders/rtx/concept/ray/ray.slangh b/src/dxvk/shaders/rtx/concept/ray/ray.slangh index 8647a557..897f7c3e 100644 --- a/src/dxvk/shaders/rtx/concept/ray/ray.slangh +++ b/src/dxvk/shaders/rtx/concept/ray/ray.slangh @@ -159,7 +159,11 @@ Ray rayCreatePosition( vec3 targetPosition, bool penetrateSurface) { const vec3 rayVector = targetPosition - minimalSurfaceInteraction.position; - const f16vec3 direction = f16vec3(normalize(rayVector)); + // Note: safeNormalizeGetLength not used here as while it could be used to calculate tMax, the length that should be returned + // is a bit ambigious. If it's supposed to be the length of the input vector then this would fork fine as the length would be + // 0 for zero vectors, but if it's set to 1 to match the fallback vector then the ray will have an incorrect tMax. As such we + // just calculate the tMax manually for clarity and hope the compiler optimizes the redundant length calculations together. + const f16vec3 direction = f16vec3(safeNormalize(rayVector, vec3(0.0f, 0.0f, 1.0f))); const float tMax = length(rayVector); return rayCreateInternal(minimalRayInteraction, minimalSurfaceInteraction, viewRay, direction, tMax, penetrateSurface); @@ -170,7 +174,7 @@ Ray rayCreatePosition( vec3 targetPosition) { const vec3 rayVector = targetPosition - volumeInteraction.position; - const f16vec3 direction = f16vec3(normalize(rayVector)); + const f16vec3 direction = f16vec3(safeNormalize(rayVector, vec3(0.0f, 0.0f, 1.0f))); const float tMax = length(rayVector); return rayCreateInternal(minimalRayInteraction, volumeInteraction, viewRay, direction, tMax); @@ -181,7 +185,7 @@ Ray rayCreatePositionSubsurface( vec3 targetPosition, vec3 shadingNormal) { const vec3 rayVector = targetPosition - minimalSurfaceInteraction.position; - const f16vec3 direction = f16vec3(normalize(rayVector)); + const f16vec3 direction = f16vec3(safeNormalize(rayVector, vec3(0.0f, 0.0f, 1.0f))); const float tMax = length(rayVector); // If the new tracing ray is on the other side of SSS surface, treat the surface as penetrateSurface diff --git a/src/dxvk/shaders/rtx/utility/math.slangh b/src/dxvk/shaders/rtx/utility/math.slangh index 1b693c24..cfe4eac3 100644 --- a/src/dxvk/shaders/rtx/utility/math.slangh +++ b/src/dxvk/shaders/rtx/utility/math.slangh @@ -127,19 +127,33 @@ GENERIC_SAFER_POSITIVE_DIVIDE(float, float) GENERIC_SAFER_POSITIVE_DIVIDE(f16vec3, float16_t) GENERIC_SAFER_POSITIVE_DIVIDE(vec3, float) -// Normalizes a vector "safely" by falling back to another vector in the case of -// an inability to normalize (to avoid NaNs from normalization). -#define GENERIC_SAFE_NORMALIZE(type, lengthType) \ -type safeNormalize(type vector, type fallbackVector) \ -{ \ - const lengthType vectorLength = length(vector); \ - \ - if (vectorLength == lengthType(0.0f)) \ - { \ - return fallbackVector; \ - } \ - \ - return vector / vectorLength; \ +// Normalizes a vector "safely" by falling back to another (ideally normalized) vector in the case of +// an inability to normalize (to avoid NaNs from normalization). Also calculates +// and outputs the length of the input vector. If the vector is invalid (the zero vector) +// then the length will be 0. +#define GENERIC_SAFE_NORMALIZE_GET_LENGTH(type, lengthType) \ +type safeNormalizeGetLength(type vector, type fallbackVector, inout lengthType vectorLength) \ +{ \ + vectorLength = length(vector); \ + \ + if (vectorLength == lengthType(0.0f)) \ + { \ + return fallbackVector; \ + } \ + \ + return vector / vectorLength; \ +} + +GENERIC_SAFE_NORMALIZE_GET_LENGTH(f16vec3, float16_t) +GENERIC_SAFE_NORMALIZE_GET_LENGTH(vec3, float) + +// Same as the get length variant of the safe normalize function, just without a length output. +#define GENERIC_SAFE_NORMALIZE(type, lengthType) \ +type safeNormalize(type vector, type fallbackVector) \ +{ \ + lengthType dummyLength; \ + \ + return safeNormalizeGetLength(vector, fallbackVector, dummyLength); \ } GENERIC_SAFE_NORMALIZE(f16vec3, float16_t) diff --git a/submodules/rtxdi b/submodules/rtxdi index 37ea43d7..ed737de7 160000 --- a/submodules/rtxdi +++ b/submodules/rtxdi @@ -1 +1 @@ -Subproject commit 37ea43d76c1215b1155f49b58c6e8861bde53e7a +Subproject commit ed737de74170ea1b4814e6eb7d81042565f47270