two-sided coating
parent
c84be3969a
commit
a4acf8b379
|
@ -2,6 +2,31 @@
|
||||||
to be tested for consistency. This is done
|
to be tested for consistency. This is done
|
||||||
using the testcase 'test_chisquare' -->
|
using the testcase 'test_chisquare' -->
|
||||||
<scene version="0.3.0">
|
<scene version="0.3.0">
|
||||||
|
<!-- Test the coating model with a transmissive
|
||||||
|
+ reflective material -->
|
||||||
|
<bsdf type="coating">
|
||||||
|
<rgb name="sigmaA" value="0.1 0.2 0.3"/>
|
||||||
|
<float name="thickness" value="2"/>
|
||||||
|
|
||||||
|
<bsdf type="mixture">
|
||||||
|
<string name="weights" value=".5 .3"/>
|
||||||
|
|
||||||
|
<bsdf type="diffuse">
|
||||||
|
<rgb name="reflectance" value=".5 0 0"/>
|
||||||
|
</bsdf>
|
||||||
|
<bsdf type="difftrans">
|
||||||
|
<rgb name="transmittance" value="0 .5 0"/>
|
||||||
|
</bsdf>
|
||||||
|
</bsdf>
|
||||||
|
</bsdf>
|
||||||
|
|
||||||
|
<!-- Test the coating model with the Hanrahan-Krueger model -->
|
||||||
|
<bsdf type="coating">
|
||||||
|
<rgb name="sigmaA" value="0.1 0.2 0.3"/>
|
||||||
|
<float name="thickness" value="2"/>
|
||||||
|
<bsdf type="hk"/>
|
||||||
|
</bsdf>
|
||||||
|
|
||||||
<!-- Test the Hanrahan-Krueger model with an
|
<!-- Test the Hanrahan-Krueger model with an
|
||||||
isotropic phase function -->
|
isotropic phase function -->
|
||||||
<bsdf type="hk"/>
|
<bsdf type="hk"/>
|
||||||
|
@ -14,6 +39,7 @@
|
||||||
</phase>
|
</phase>
|
||||||
</bsdf>
|
</bsdf>
|
||||||
|
|
||||||
|
|
||||||
<!-- Test the smooth diffuse model -->
|
<!-- Test the smooth diffuse model -->
|
||||||
<bsdf type="diffuse"/>
|
<bsdf type="diffuse"/>
|
||||||
|
|
||||||
|
|
|
@ -94,6 +94,12 @@ MTS_NAMESPACE_BEGIN
|
||||||
*/
|
*/
|
||||||
class SmoothCoating : public BSDF {
|
class SmoothCoating : public BSDF {
|
||||||
public:
|
public:
|
||||||
|
/// \sa refractTo()
|
||||||
|
enum EDestination {
|
||||||
|
EInterior = 0,
|
||||||
|
EExterior = 1
|
||||||
|
};
|
||||||
|
|
||||||
SmoothCoating(const Properties &props)
|
SmoothCoating(const Properties &props)
|
||||||
: BSDF(props) {
|
: BSDF(props) {
|
||||||
/* Specifies the internal index of refraction at the interface */
|
/* Specifies the internal index of refraction at the interface */
|
||||||
|
@ -123,9 +129,6 @@ public:
|
||||||
void configure() {
|
void configure() {
|
||||||
if (!m_nested)
|
if (!m_nested)
|
||||||
Log(EError, "A child BSDF instance is required");
|
Log(EError, "A child BSDF instance is required");
|
||||||
if (m_nested->getType() & BSDF::ETransmission)
|
|
||||||
Log(EError, "Tried to put a smooth coating layer on top of a BSDF "
|
|
||||||
"with a transmission component -- this is currently not allowed!");
|
|
||||||
|
|
||||||
unsigned int extraFlags = 0;
|
unsigned int extraFlags = 0;
|
||||||
if (!m_sigmaA->isConstant())
|
if (!m_sigmaA->isConstant())
|
||||||
|
@ -135,7 +138,7 @@ public:
|
||||||
for (int i=0; i<m_nested->getComponentCount(); ++i)
|
for (int i=0; i<m_nested->getComponentCount(); ++i)
|
||||||
m_components.push_back(m_nested->getType(i) | extraFlags);
|
m_components.push_back(m_nested->getType(i) | extraFlags);
|
||||||
|
|
||||||
m_components.push_back(EDeltaReflection | EFrontSide);
|
m_components.push_back(EDeltaReflection | EFrontSide | EBackSide);
|
||||||
|
|
||||||
m_usesRayDifferentials = m_nested->usesRayDifferentials()
|
m_usesRayDifferentials = m_nested->usesRayDifferentials()
|
||||||
|| m_sigmaA->usesRayDifferentials();
|
|| m_sigmaA->usesRayDifferentials();
|
||||||
|
@ -175,26 +178,21 @@ public:
|
||||||
return Vector(-wi.x, -wi.y, wi.z);
|
return Vector(-wi.x, -wi.y, wi.z);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/// Refraction in local coordinates
|
||||||
* \brief Refraction in local coordinates
|
Vector refractTo(EDestination dest,
|
||||||
*
|
const Vector &wi, Float &F) const {
|
||||||
* To be used when some of the data is already available
|
Float etaI, etaT;
|
||||||
*/
|
if (dest == EInterior) {
|
||||||
inline Vector refract(const Vector &wi, Float eta, Float cosThetaT) const {
|
etaI = m_extIOR;
|
||||||
return Vector(-eta*wi.x, -eta*wi.y, cosThetaT);
|
etaT = m_intIOR;
|
||||||
|
} else {
|
||||||
|
etaI = m_intIOR;
|
||||||
|
etaT = m_extIOR;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Refraction in local coordinates (full version)
|
Float cosThetaI = Frame::cosTheta(wi);
|
||||||
inline Vector refract(const Vector &wi, Float &F) const {
|
|
||||||
Float cosThetaI = Frame::cosTheta(wi),
|
|
||||||
etaI = m_extIOR, etaT = m_intIOR;
|
|
||||||
|
|
||||||
bool entering = cosThetaI > 0.0f;
|
bool entering = cosThetaI > 0.0f;
|
||||||
|
|
||||||
/* Determine the respective indices of refraction */
|
|
||||||
if (!entering)
|
|
||||||
std::swap(etaI, etaT);
|
|
||||||
|
|
||||||
/* Using Snell's law, calculate the squared sine of the
|
/* Using Snell's law, calculate the squared sine of the
|
||||||
angle between the normal and the transmitted ray */
|
angle between the normal and the transmitted ray */
|
||||||
Float eta = etaI / etaT,
|
Float eta = etaI / etaT,
|
||||||
|
@ -210,18 +208,15 @@ public:
|
||||||
|
|
||||||
/* Compute the Fresnel transmittance */
|
/* Compute the Fresnel transmittance */
|
||||||
F = fresnelDielectric(std::abs(Frame::cosTheta(wi)),
|
F = fresnelDielectric(std::abs(Frame::cosTheta(wi)),
|
||||||
cosThetaT, m_extIOR, m_intIOR);
|
cosThetaT, etaI, etaT);
|
||||||
|
|
||||||
return Vector(-eta*wi.x, -eta*wi.y,
|
/* Retain the directionality of the vector */
|
||||||
entering ? -cosThetaT : cosThetaT);
|
return Vector(eta*wi.x, eta*wi.y,
|
||||||
|
entering ? cosThetaT : -cosThetaT);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Spectrum eval(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
Spectrum eval(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
||||||
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
|
||||||
Frame::cosTheta(bRec.wo) <= 0)
|
|
||||||
return Spectrum(0.0f);
|
|
||||||
|
|
||||||
bool sampleSpecular = (bRec.typeMask & EDeltaReflection)
|
bool sampleSpecular = (bRec.typeMask & EDeltaReflection)
|
||||||
&& (bRec.component == -1 || bRec.component == (int) m_components.size()-1);
|
&& (bRec.component == -1 || bRec.component == (int) m_components.size()-1);
|
||||||
bool sampleNested = (bRec.typeMask & m_nested->getType() & BSDF::EAll)
|
bool sampleNested = (bRec.typeMask & m_nested->getType() & BSDF::EAll)
|
||||||
|
@ -230,29 +225,29 @@ public:
|
||||||
if (measure == EDiscrete && sampleSpecular &&
|
if (measure == EDiscrete && sampleSpecular &&
|
||||||
std::abs(1-dot(reflect(bRec.wi), bRec.wo)) < Epsilon) {
|
std::abs(1-dot(reflect(bRec.wi), bRec.wo)) < Epsilon) {
|
||||||
return Spectrum(fresnel(
|
return Spectrum(fresnel(
|
||||||
Frame::cosTheta(bRec.wi), m_extIOR, m_intIOR));
|
std::abs(Frame::cosTheta(bRec.wi)), m_extIOR, m_intIOR));
|
||||||
} else if (sampleNested) {
|
} else if (sampleNested) {
|
||||||
Float R12, R21;
|
Float R12, R21;
|
||||||
BSDFQueryRecord bRec2(bRec);
|
BSDFQueryRecord bRecInt(bRec);
|
||||||
bRec2.wi = -refract(bRec.wi, R12);
|
bRecInt.wi = refractTo(EInterior, bRec.wi, R12);
|
||||||
bRec2.wo = -refract(bRec.wo, R21);
|
bRecInt.wo = refractTo(EInterior, bRec.wo, R21);
|
||||||
|
|
||||||
if (R12 == 1 || R21 == 1) /* Total internal reflection */
|
if (R12 == 1 || R21 == 1) /* Total internal reflection */
|
||||||
return Spectrum(0.0f);
|
return Spectrum(0.0f);
|
||||||
|
|
||||||
Float eta = m_extIOR / m_intIOR;
|
Float eta = m_extIOR / m_intIOR;
|
||||||
Spectrum result = m_nested->eval(bRec2, measure)
|
Spectrum result = m_nested->eval(bRecInt, measure)
|
||||||
* ((1-R12) * (1-R21) * eta * eta);
|
* ((1-R12) * (1-R21) * eta * eta);
|
||||||
|
|
||||||
Spectrum sigmaA = m_sigmaA->getValue(bRec.its) * m_thickness;
|
Spectrum sigmaA = m_sigmaA->getValue(bRec.its) * m_thickness;
|
||||||
if (!sigmaA.isZero())
|
if (!sigmaA.isZero())
|
||||||
result *= (-sigmaA *
|
result *= (-sigmaA *
|
||||||
(1/std::abs(Frame::cosTheta(bRec2.wi)) +
|
(1/std::abs(Frame::cosTheta(bRecInt.wi)) +
|
||||||
1/std::abs(Frame::cosTheta(bRec2.wo)))).exp();
|
1/std::abs(Frame::cosTheta(bRecInt.wo)))).exp();
|
||||||
|
|
||||||
if (measure == ESolidAngle)
|
if (measure == ESolidAngle)
|
||||||
result *= Frame::cosTheta(bRec.wo)
|
result *= std::abs(Frame::cosTheta(bRec.wo)
|
||||||
/ Frame::cosTheta(bRec2.wo);
|
/ Frame::cosTheta(bRecInt.wo));
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -261,17 +256,13 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
Float pdf(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
Float pdf(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
||||||
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
|
||||||
Frame::cosTheta(bRec.wo) <= 0)
|
|
||||||
return 0.0f;
|
|
||||||
|
|
||||||
bool sampleSpecular = (bRec.typeMask & EDeltaReflection)
|
bool sampleSpecular = (bRec.typeMask & EDeltaReflection)
|
||||||
&& (bRec.component == -1 || bRec.component == (int) m_components.size()-1);
|
&& (bRec.component == -1 || bRec.component == (int) m_components.size()-1);
|
||||||
bool sampleNested = (bRec.typeMask & m_nested->getType() & BSDF::EAll)
|
bool sampleNested = (bRec.typeMask & m_nested->getType() & BSDF::EAll)
|
||||||
&& (bRec.component == -1 || bRec.component < (int) m_components.size()-1);
|
&& (bRec.component == -1 || bRec.component < (int) m_components.size()-1);
|
||||||
|
|
||||||
Float R12;
|
Float R12;
|
||||||
Vector wiPrime = -refract(bRec.wi, R12);
|
Vector wiPrime = refractTo(EInterior, bRec.wi, R12);
|
||||||
|
|
||||||
/* Reallocate samples */
|
/* Reallocate samples */
|
||||||
Float probSpecular = (R12*m_specularSamplingWeight) /
|
Float probSpecular = (R12*m_specularSamplingWeight) /
|
||||||
|
@ -283,18 +274,18 @@ public:
|
||||||
return sampleNested ? probSpecular : 1.0f;
|
return sampleNested ? probSpecular : 1.0f;
|
||||||
} else if (sampleNested) {
|
} else if (sampleNested) {
|
||||||
Float R21;
|
Float R21;
|
||||||
BSDFQueryRecord bRec2(bRec);
|
BSDFQueryRecord bRecInt(bRec);
|
||||||
bRec2.wi = wiPrime;
|
bRecInt.wi = wiPrime;
|
||||||
bRec2.wo = -refract(bRec.wo, R21);
|
bRecInt.wo = refractTo(EInterior, bRec.wo, R21);
|
||||||
|
|
||||||
if (R12 == 1 || R21 == 1) /* Total internal reflection */
|
if (R12 == 1 || R21 == 1) /* Total internal reflection */
|
||||||
return 0.0f;
|
return 0.0f;
|
||||||
|
|
||||||
Float pdf = m_nested->pdf(bRec2, measure);
|
Float pdf = m_nested->pdf(bRecInt, measure);
|
||||||
|
|
||||||
if (measure == ESolidAngle)
|
if (measure == ESolidAngle)
|
||||||
pdf *= Frame::cosTheta(bRec.wo)
|
pdf *= std::abs(Frame::cosTheta(bRec.wo)
|
||||||
/ Frame::cosTheta(bRec2.wo);
|
/ Frame::cosTheta(bRecInt.wo));
|
||||||
|
|
||||||
Float eta = m_extIOR / m_intIOR;
|
Float eta = m_extIOR / m_intIOR;
|
||||||
pdf *= eta * eta;
|
pdf *= eta * eta;
|
||||||
|
@ -311,21 +302,11 @@ public:
|
||||||
bool sampleNested = (bRec.typeMask & m_nested->getType() & BSDF::EAll)
|
bool sampleNested = (bRec.typeMask & m_nested->getType() & BSDF::EAll)
|
||||||
&& (bRec.component == -1 || bRec.component < (int) m_components.size()-1);
|
&& (bRec.component == -1 || bRec.component < (int) m_components.size()-1);
|
||||||
|
|
||||||
if ((!sampleSpecular && !sampleNested) || Frame::cosTheta(bRec.wi) <= 0)
|
if ((!sampleSpecular && !sampleNested))
|
||||||
return Spectrum(0.0f);
|
return Spectrum(0.0f);
|
||||||
|
|
||||||
/* Refract the incident direction and compute the Fresnel reflectance */
|
Float R12;
|
||||||
Float eta = m_extIOR / m_intIOR,
|
Vector wiPrime = refractTo(EInterior, bRec.wi, R12);
|
||||||
sinThetaTSqr = eta*eta * Frame::sinTheta2(bRec.wi),
|
|
||||||
R12, cosThetaT = 0;
|
|
||||||
|
|
||||||
if (sinThetaTSqr >= 1.0f) {
|
|
||||||
R12 = 1.0f; /* Total internal reflection */
|
|
||||||
} else {
|
|
||||||
cosThetaT = -std::sqrt(1.0f - sinThetaTSqr);
|
|
||||||
R12 = fresnelDielectric(Frame::cosTheta(bRec.wi),
|
|
||||||
-cosThetaT, m_extIOR, m_intIOR);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Reallocate samples */
|
/* Reallocate samples */
|
||||||
Float probSpecular = (R12*m_specularSamplingWeight) /
|
Float probSpecular = (R12*m_specularSamplingWeight) /
|
||||||
|
@ -353,27 +334,30 @@ public:
|
||||||
return Spectrum(0.0f);
|
return Spectrum(0.0f);
|
||||||
|
|
||||||
Vector wiBackup = bRec.wi;
|
Vector wiBackup = bRec.wi;
|
||||||
bRec.wi = -refract(bRec.wi, eta, cosThetaT);
|
bRec.wi = wiPrime;
|
||||||
|
|
||||||
Spectrum result = m_nested->sample(bRec, pdf, sample);
|
Spectrum result = m_nested->sample(bRec, pdf, sample);
|
||||||
|
bRec.wi = wiBackup;
|
||||||
|
Vector woPrime = bRec.wo;
|
||||||
|
|
||||||
if (result.isZero())
|
if (result.isZero())
|
||||||
return Spectrum(0.0f);
|
return Spectrum(0.0f);
|
||||||
|
|
||||||
Spectrum sigmaA = m_sigmaA->getValue(bRec.its) * m_thickness;
|
Spectrum sigmaA = m_sigmaA->getValue(bRec.its) * m_thickness;
|
||||||
if (!sigmaA.isZero())
|
if (!sigmaA.isZero())
|
||||||
result *= (-sigmaA *
|
result *= (-sigmaA *
|
||||||
(1/std::abs(Frame::cosTheta(bRec.wi)) +
|
(1/std::abs(Frame::cosTheta(wiPrime)) +
|
||||||
1/std::abs(Frame::cosTheta(bRec.wo)))).exp();
|
1/std::abs(Frame::cosTheta(woPrime)))).exp();
|
||||||
|
|
||||||
Float R21, cosThetaWoPrime = Frame::cosTheta(bRec.wo);
|
Float R21;
|
||||||
bRec.wo = refract(-bRec.wo, R21);
|
bRec.wo = refractTo(EExterior, woPrime, R21);
|
||||||
bRec.wi = wiBackup;
|
|
||||||
|
|
||||||
if (R21 == 1.0f) /* Total internal reflection */
|
if (R21 == 1.0f) /* Total internal reflection */
|
||||||
return Spectrum(0.0f);
|
return Spectrum(0.0f);
|
||||||
|
|
||||||
|
Float eta = m_extIOR / m_intIOR;
|
||||||
bool sampledSA = (BSDF::getMeasure(bRec.sampledType) == ESolidAngle);
|
bool sampledSA = (BSDF::getMeasure(bRec.sampledType) == ESolidAngle);
|
||||||
Float cosRatio = Frame::cosTheta(bRec.wo) / cosThetaWoPrime,
|
Float cosRatio = std::abs(Frame::cosTheta(bRec.wo) / Frame::cosTheta(woPrime)),
|
||||||
commonTerms = (sampledSA ? cosRatio : 1.0f)* eta * eta;
|
commonTerms = (sampledSA ? cosRatio : 1.0f) * eta * eta;
|
||||||
|
|
||||||
pdf *= (sampleSpecular ? (1 - probSpecular) : 1.0f) * commonTerms;
|
pdf *= (sampleSpecular ? (1 - probSpecular) : 1.0f) * commonTerms;
|
||||||
result *= (1 - R12) * (1 - R21) * commonTerms;
|
result *= (1 - R12) * (1 - R21) * commonTerms;
|
||||||
|
|
|
@ -667,7 +667,7 @@ Vector squareToCone(Float cosCutoff, const Point2 &sample) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Point2 squareToStdNormal(const Point2 &sample) {
|
Point2 squareToStdNormal(const Point2 &sample) {
|
||||||
Float tmp1 = std::sqrt(-2 * std::log(sample.x)),
|
Float tmp1 = std::sqrt(-2 * std::log(1-sample.x)),
|
||||||
tmp2 = 2 * M_PI * sample.y;
|
tmp2 = 2 * M_PI * sample.y;
|
||||||
return Point2(
|
return Point2(
|
||||||
tmp1 * std::cos(tmp2),
|
tmp1 * std::cos(tmp2),
|
||||||
|
|
Loading…
Reference in New Issue