fix sampling issues in twosided.cpp

metadata
Wenzel Jakob 2011-06-23 11:31:29 +02:00
parent 504e19e420
commit 226b639d28
2 changed files with 22 additions and 11 deletions

View File

@ -182,7 +182,13 @@ public:
if (bRec.wo.z <= 0) if (bRec.wo.z <= 0)
return Spectrum(0.0f); return Spectrum(0.0f);
return f(bRec) / pdf(bRec); Float pdfVal = pdf(bRec);
// guard against numerical issues
if (pdfVal == 0)
return Spectrum(0.0f);
else
return f(bRec) / pdfVal;
} }
inline Float pdfDiffuse(const BSDFQueryRecord &bRec) const { inline Float pdfDiffuse(const BSDFQueryRecord &bRec) const {

View File

@ -22,6 +22,11 @@
MTS_NAMESPACE_BEGIN MTS_NAMESPACE_BEGIN
/**
* Turns a one-sided BRDF onto a two-sided one that
* can be used to render meshes where the back-side
* is visible.
*/
class TwoSidedBRDF : public BSDF { class TwoSidedBRDF : public BSDF {
public: public:
TwoSidedBRDF(const Properties &props) TwoSidedBRDF(const Properties &props)
@ -87,31 +92,31 @@ public:
Spectrum sample(BSDFQueryRecord &bRec, const Point2 &sample) const { Spectrum sample(BSDFQueryRecord &bRec, const Point2 &sample) const {
bool flip = false; bool flipped = false;
if (bRec.wi.z < 0) { if (bRec.wi.z < 0) {
bRec.wi.z *= -1; bRec.wi.z *= -1;
flip = true; flipped = true;
} }
Spectrum result = m_nestedBRDF->sample(bRec, sample); Spectrum result = m_nestedBRDF->sample(bRec, sample);
if (bRec.wi.z < 0 && !result.isZero()) { if (flipped) {
bRec.wi.z *= -1; bRec.wi.z *= -1;
if (!result.isZero())
bRec.wo.z *= -1; bRec.wo.z *= -1;
flip = true;
} }
return result; return result;
} }
Spectrum sample(BSDFQueryRecord &bRec, Float &pdf, const Point2 &sample) const { Spectrum sample(BSDFQueryRecord &bRec, Float &pdf, const Point2 &sample) const {
bool flip = false; bool flipped = false;
if (bRec.wi.z < 0) { if (bRec.wi.z < 0) {
bRec.wi.z *= -1; bRec.wi.z *= -1;
flip = true; flipped = true;
} }
Spectrum result = m_nestedBRDF->sample(bRec, pdf, sample); Spectrum result = m_nestedBRDF->sample(bRec, pdf, sample);
if (bRec.wi.z < 0 && !result.isZero()) { if (flipped) {
bRec.wi.z *= -1; bRec.wi.z *= -1;
if (!result.isZero() && pdf != 0)
bRec.wo.z *= -1; bRec.wo.z *= -1;
flip = true;
} }
return result; return result;
} }