380 lines
13 KiB
HLSL
380 lines
13 KiB
HLSL
#ifndef UNITY_VOLUME_RENDERING_INCLUDED
|
|
#define UNITY_VOLUME_RENDERING_INCLUDED
|
|
|
|
// Reminder:
|
|
// OpticalDepth(x, y) = Integral{x, y}{Extinction(t) dt}
|
|
// Transmittance(x, y) = Exp(-OpticalDepth(x, y))
|
|
// Transmittance(x, z) = Transmittance(x, y) * Transmittance(y, z)
|
|
// Integral{a, b}{Transmittance(0, t) dt} = Transmittance(0, a) * Integral{a, b}{Transmittance(0, t - a) dt}
|
|
|
|
real TransmittanceFromOpticalDepth(real opticalDepth)
|
|
{
|
|
return exp(-opticalDepth);
|
|
}
|
|
|
|
real3 TransmittanceFromOpticalDepth(real3 opticalDepth)
|
|
{
|
|
return exp(-opticalDepth);
|
|
}
|
|
|
|
real OpacityFromOpticalDepth(real opticalDepth)
|
|
{
|
|
return 1 - TransmittanceFromOpticalDepth(opticalDepth);
|
|
}
|
|
|
|
real3 OpacityFromOpticalDepth(real3 opticalDepth)
|
|
{
|
|
return 1 - TransmittanceFromOpticalDepth(opticalDepth);
|
|
}
|
|
|
|
real OpticalDepthFromOpacity(real opacity)
|
|
{
|
|
return -log(1 - opacity);
|
|
}
|
|
|
|
real3 OpticalDepthFromOpacity(real3 opacity)
|
|
{
|
|
return -log(1 - opacity);
|
|
}
|
|
|
|
//
|
|
// ---------------------------------- Deep Pixel Compositing ---------------------------------------
|
|
//
|
|
|
|
// TODO: it would be good to improve the perf and numerical stability
|
|
// of approximations below by finding a polynomial approximation.
|
|
|
|
// input = {radiance, opacity}
|
|
// Note that opacity must be less than 1 (not fully opaque).
|
|
real4 LinearizeRGBA(real4 value)
|
|
{
|
|
// See "Deep Compositing Using Lie Algebras".
|
|
// log(A) = {OpticalDepthFromOpacity(A.a) / A.a * A.rgb, -OpticalDepthFromOpacity(A.a)}.
|
|
// We drop redundant negations.
|
|
real a = value.a;
|
|
real d = -log(1 - a);
|
|
real r = (a >= REAL_EPS) ? (d * rcp(a)) : 1; // Prevent numerical explosion
|
|
return real4(r * value.rgb, d);
|
|
}
|
|
|
|
// input = {radiance, optical_depth}
|
|
// Note that opacity must be less than 1 (not fully opaque).
|
|
real4 LinearizeRGBD(real4 value)
|
|
{
|
|
// See "Deep Compositing Using Lie Algebras".
|
|
// log(A) = {A.a / OpacityFromOpticalDepth(A.a) * A.rgb, -A.a}.
|
|
// We drop redundant negations.
|
|
real d = value.a;
|
|
real a = 1 - exp(-d);
|
|
real r = (a >= REAL_EPS) ? (d * rcp(a)) : 1; // Prevent numerical explosion
|
|
return real4(r * value.rgb, d);
|
|
}
|
|
|
|
// output = {radiance, opacity}
|
|
// Note that opacity must be less than 1 (not fully opaque).
|
|
real4 DelinearizeRGBA(real4 value)
|
|
{
|
|
// See "Deep Compositing Using Lie Algebras".
|
|
// exp(B) = {OpacityFromOpticalDepth(-B.a) / -B.a * B.rgb, OpacityFromOpticalDepth(-B.a)}.
|
|
// We drop redundant negations.
|
|
real d = value.a;
|
|
real a = 1 - exp(-d);
|
|
real i = (a >= REAL_EPS) ? (a * rcp(d)) : 1; // Prevent numerical explosion
|
|
return real4(i * value.rgb, a);
|
|
}
|
|
|
|
// input = {radiance, optical_depth}
|
|
// Note that opacity must be less than 1 (not fully opaque).
|
|
real4 DelinearizeRGBD(real4 value)
|
|
{
|
|
// See "Deep Compositing Using Lie Algebras".
|
|
// exp(B) = {OpacityFromOpticalDepth(-B.a) / -B.a * B.rgb, -B.a}.
|
|
// We drop redundant negations.
|
|
real d = value.a;
|
|
real a = 1 - exp(-d);
|
|
real i = (a >= REAL_EPS) ? (a * rcp(d)) : 1; // Prevent numerical explosion
|
|
return real4(i * value.rgb, d);
|
|
}
|
|
|
|
//
|
|
// ----------------------------- Homogeneous Participating Media -----------------------------------
|
|
//
|
|
|
|
real OpticalDepthHomogeneousMedium(real extinction, real intervalLength)
|
|
{
|
|
return extinction * intervalLength;
|
|
}
|
|
|
|
real TransmittanceHomogeneousMedium(real extinction, real intervalLength)
|
|
{
|
|
return TransmittanceFromOpticalDepth(OpticalDepthHomogeneousMedium(extinction, intervalLength));
|
|
}
|
|
|
|
// Integral{a, b}{TransmittanceHomogeneousMedium(k, t - a) dt}.
|
|
real TransmittanceIntegralHomogeneousMedium(real extinction, real intervalLength)
|
|
{
|
|
// Note: when multiplied by the extinction coefficient, it becomes
|
|
// Albedo * (1 - TransmittanceFromOpticalDepth(d)) = Albedo * Opacity(d).
|
|
return rcp(extinction) - rcp(extinction) * exp(-extinction * intervalLength);
|
|
}
|
|
|
|
//
|
|
// ----------------------------------- Height Fog --------------------------------------------------
|
|
//
|
|
|
|
// Can be used to scale base extinction and scattering coefficients.
|
|
real ComputeHeightFogMultiplier(real height, real baseHeight, real2 heightExponents)
|
|
{
|
|
real h = max(height - baseHeight, 0);
|
|
real rcpH = heightExponents.x;
|
|
|
|
return exp(-h * rcpH);
|
|
}
|
|
|
|
// Optical depth between two endpoints.
|
|
real OpticalDepthHeightFog(real baseExtinction, real baseHeight, real2 heightExponents,
|
|
real cosZenith, real startHeight, real intervalLength)
|
|
{
|
|
// Height fog is composed of two slices of optical depth:
|
|
// - homogeneous fog below 'baseHeight': d = k * t
|
|
// - exponential fog above 'baseHeight': d = Integrate[k * e^(-(h + z * x) / H) dx, {x, 0, t}]
|
|
|
|
real H = heightExponents.y;
|
|
real rcpH = heightExponents.x;
|
|
real Z = cosZenith;
|
|
real absZ = max(abs(cosZenith), 0.001f);
|
|
real rcpAbsZ = rcp(absZ);
|
|
|
|
real endHeight = startHeight + intervalLength * Z;
|
|
real minHeight = min(startHeight, endHeight);
|
|
real h = max(minHeight - baseHeight, 0);
|
|
|
|
real homFogDist = clamp((baseHeight - minHeight) * rcpAbsZ, 0, intervalLength);
|
|
real expFogDist = intervalLength - homFogDist;
|
|
real expFogMult = exp(-h * rcpH) * (1 - exp(-expFogDist * absZ * rcpH)) * (rcpAbsZ * H);
|
|
|
|
return baseExtinction * (homFogDist + expFogMult);
|
|
}
|
|
|
|
// This version of the function assumes the interval of infinite length.
|
|
real OpticalDepthHeightFog(real baseExtinction, real baseHeight, real2 heightExponents,
|
|
real cosZenith, real startHeight)
|
|
{
|
|
real H = heightExponents.y;
|
|
real rcpH = heightExponents.x;
|
|
real Z = cosZenith;
|
|
real absZ = max(abs(cosZenith), REAL_EPS);
|
|
real rcpAbsZ = rcp(absZ);
|
|
|
|
real minHeight = (Z >= 0) ? startHeight : -rcp(REAL_EPS);
|
|
real h = max(minHeight - baseHeight, 0);
|
|
|
|
real homFogDist = max((baseHeight - minHeight) * rcpAbsZ, 0);
|
|
real expFogMult = exp(-h * rcpH) * (rcpAbsZ * H);
|
|
|
|
return baseExtinction * (homFogDist + expFogMult);
|
|
}
|
|
|
|
real TransmittanceHeightFog(real baseExtinction, real baseHeight, real2 heightExponents,
|
|
real cosZenith, real startHeight, real intervalLength)
|
|
{
|
|
real od = OpticalDepthHeightFog(baseExtinction, baseHeight, heightExponents,
|
|
cosZenith, startHeight, intervalLength);
|
|
return TransmittanceFromOpticalDepth(od);
|
|
}
|
|
|
|
real TransmittanceHeightFog(real baseExtinction, real baseHeight, real2 heightExponents,
|
|
real cosZenith, real startHeight)
|
|
{
|
|
real od = OpticalDepthHeightFog(baseExtinction, baseHeight, heightExponents,
|
|
cosZenith, startHeight);
|
|
return TransmittanceFromOpticalDepth(od);
|
|
}
|
|
|
|
//
|
|
// ----------------------------------- Phase Functions ---------------------------------------------
|
|
//
|
|
|
|
real IsotropicPhaseFunction()
|
|
{
|
|
return INV_FOUR_PI;
|
|
}
|
|
|
|
real RayleighPhaseFunction(real cosTheta)
|
|
{
|
|
real k = 3 / (16 * PI);
|
|
return k * (1 + cosTheta * cosTheta);
|
|
}
|
|
|
|
real HenyeyGreensteinPhasePartConstant(real anisotropy)
|
|
{
|
|
real g = anisotropy;
|
|
|
|
return INV_FOUR_PI * (1 - g * g);
|
|
}
|
|
|
|
real HenyeyGreensteinPhasePartVarying(real anisotropy, real cosTheta)
|
|
{
|
|
real g = anisotropy;
|
|
real x = 1 + g * g - 2 * g * cosTheta;
|
|
real f = rsqrt(max(x, REAL_EPS)); // x^(-1/2)
|
|
|
|
return f * f * f; // x^(-3/2)
|
|
}
|
|
|
|
real HenyeyGreensteinPhaseFunction(real anisotropy, real cosTheta)
|
|
{
|
|
return HenyeyGreensteinPhasePartConstant(anisotropy) *
|
|
HenyeyGreensteinPhasePartVarying(anisotropy, cosTheta);
|
|
}
|
|
|
|
real CornetteShanksPhasePartConstant(real anisotropy)
|
|
{
|
|
real g = anisotropy;
|
|
|
|
return (3 / (8 * PI)) * (1 - g * g) / (2 + g * g);
|
|
}
|
|
|
|
// Similar to the RayleighPhaseFunction.
|
|
real CornetteShanksPhasePartSymmetrical(real cosTheta)
|
|
{
|
|
real h = 1 + cosTheta * cosTheta;
|
|
return h;
|
|
}
|
|
|
|
real CornetteShanksPhasePartAsymmetrical(real anisotropy, real cosTheta)
|
|
{
|
|
real g = anisotropy;
|
|
real x = 1 + g * g - 2 * g * cosTheta;
|
|
real f = rsqrt(max(x, REAL_EPS)); // x^(-1/2)
|
|
return f * f * f; // x^(-3/2)
|
|
}
|
|
|
|
real CornetteShanksPhasePartVarying(real anisotropy, real cosTheta)
|
|
{
|
|
return CornetteShanksPhasePartSymmetrical(cosTheta) *
|
|
CornetteShanksPhasePartAsymmetrical(anisotropy, cosTheta); // h * x^(-3/2)
|
|
}
|
|
|
|
// A better approximation of the Mie phase function.
|
|
// Ref: Henyey-Greenstein and Mie phase functions in Monte Carlo radiative transfer computations
|
|
real CornetteShanksPhaseFunction(real anisotropy, real cosTheta)
|
|
{
|
|
return CornetteShanksPhasePartConstant(anisotropy) *
|
|
CornetteShanksPhasePartVarying(anisotropy, cosTheta);
|
|
}
|
|
|
|
//
|
|
// --------------------------------- Importance Sampling -------------------------------------------
|
|
//
|
|
|
|
// Samples the interval of homogeneous participating medium using the closed-form tracking approach
|
|
// (proportionally to the transmittance).
|
|
// Returns the offset from the start of the interval and the weight = (transmittance / pdf).
|
|
// Ref: Monte Carlo Methods for Volumetric Light Transport Simulation, p. 5.
|
|
void ImportanceSampleHomogeneousMedium(real rndVal, real extinction, real intervalLength,
|
|
out real offset, out real weight)
|
|
{
|
|
// pdf = extinction * exp(extinction * (intervalLength - t)) / (exp(intervalLength * extinction) - 1)
|
|
// pdf = extinction * exp(-extinction * t) / (1 - exp(-extinction * intervalLength))
|
|
// weight = TransmittanceFromOpticalDepth(t) / pdf
|
|
// weight = exp(-extinction * t) / pdf
|
|
// weight = (1 - exp(-extinction * intervalLength)) / extinction
|
|
// weight = OpacityFromOpticalDepth(extinction * intervalLength) / extinction
|
|
|
|
real x = 1 - exp(-extinction * intervalLength);
|
|
real c = rcp(extinction);
|
|
|
|
// TODO: return 'rcpPdf' to support imperfect importance sampling...
|
|
weight = x * c;
|
|
offset = -log(1 - rndVal * x) * c;
|
|
}
|
|
|
|
void ImportanceSampleExponentialMedium(real rndVal, real extinction, real falloff,
|
|
out real offset, out real rcpPdf)
|
|
{
|
|
|
|
// Extinction[t] = Extinction[0] * exp(-falloff * t).
|
|
real c = extinction;
|
|
real a = falloff;
|
|
|
|
// TODO: optimize...
|
|
offset = -log(1 - a / c * log(rndVal)) / a;
|
|
rcpPdf = rcp(c * exp(-a * offset) * exp(-c / a * (1 - exp(-a * offset))));
|
|
}
|
|
|
|
// Implements equiangular light sampling.
|
|
// Returns the distance from the origin of the ray, the squared distance from the light,
|
|
// and the reciprocal of the PDF.
|
|
// Ref: Importance Sampling of Area Lights in Participating Medium.
|
|
void ImportanceSamplePunctualLight(real rndVal, real3 lightPosition, real lightSqRadius,
|
|
real3 rayOrigin, real3 rayDirection,
|
|
real tMin, real tMax,
|
|
out real t, out real sqDist, out real rcpPdf)
|
|
{
|
|
real3 originToLight = lightPosition - rayOrigin;
|
|
real originToLightProjDist = dot(originToLight, rayDirection);
|
|
real originToLightSqDist = dot(originToLight, originToLight);
|
|
real rayToLightSqDist = originToLightSqDist - originToLightProjDist * originToLightProjDist;
|
|
|
|
// Virtually offset the light to modify the PDF distribution.
|
|
real sqD = max(rayToLightSqDist + lightSqRadius, REAL_EPS);
|
|
real rcpD = rsqrt(sqD);
|
|
real d = sqD * rcpD;
|
|
real a = tMin - originToLightProjDist;
|
|
real b = tMax - originToLightProjDist;
|
|
real x = a * rcpD;
|
|
real y = b * rcpD;
|
|
|
|
#if 0
|
|
real theta0 = FastATan(x);
|
|
real theta1 = FastATan(y);
|
|
real gamma = theta1 - theta0;
|
|
real tanTheta = tan(theta0 + rndVal * gamma);
|
|
#else
|
|
// Same but faster:
|
|
// atan(y) - atan(x) = atan((y - x) / (1 + x * y))
|
|
// tan(atan(x) + z) = (x * cos(z) + sin(z)) / (cos(z) - x * sin(z))
|
|
// Both the tangent and the angle cannot be negative.
|
|
real tanGamma = abs((y - x) * rcp(max(0, 1 + x * y)));
|
|
real gamma = FastATanPos(tanGamma);
|
|
real z = rndVal * gamma;
|
|
real numer = x * cos(z) + sin(z);
|
|
real denom = cos(z) - x * sin(z);
|
|
real tanTheta = numer * rcp(denom);
|
|
#endif
|
|
|
|
real tRelative = d * tanTheta;
|
|
|
|
sqDist = sqD + tRelative * tRelative;
|
|
rcpPdf = gamma * rcpD * sqDist;
|
|
t = originToLightProjDist + tRelative;
|
|
|
|
// Remove the virtual light offset to obtain the real geometric distance.
|
|
sqDist = max(sqDist - lightSqRadius, REAL_EPS);
|
|
}
|
|
|
|
// Returns the cosine.
|
|
// Weight = Phase / Pdf = 1.
|
|
real ImportanceSampleRayleighPhase(real rndVal)
|
|
{
|
|
// real a = sqrt(16 * (rndVal - 1) * rndVal + 5);
|
|
// real b = -4 * rndVal + a + 2;
|
|
// real c = PositivePow(b, 0.33333333);
|
|
// return rcp(c) - c;
|
|
|
|
// Approximate...
|
|
return lerp(cos(PI * rndVal + PI), 2 * rndVal - 1, 0.5);
|
|
}
|
|
|
|
//
|
|
// ------------------------------------ Miscellaneous ----------------------------------------------
|
|
//
|
|
|
|
// Absorption coefficient from Disney: http://blog.selfshadow.com/publications/s2015-shading-course/burley/s2015_pbs_disney_bsdf_notes.pdf
|
|
real3 TransmittanceColorAtDistanceToAbsorption(real3 transmittanceColor, real atDistance)
|
|
{
|
|
return -log(transmittanceColor + REAL_EPS) / max(atDistance, REAL_EPS);
|
|
}
|
|
|
|
#endif // UNITY_VOLUME_RENDERING_INCLUDED
|