fixed the composite material

metadata
Wenzel Jakob 2011-06-25 00:24:41 +02:00
parent 97ba51d675
commit 071b7379c1
7 changed files with 154 additions and 134 deletions

View File

@ -1,9 +0,0 @@
#!/bin/bash
tar --exclude-vcs -czvf mitsuba-$1.tar.gz ChangeLog README SConstruct \
config include/ schema/ src/bsdfs/ src/cameras/ src/collada/ src/films/ src/libcore/ src/libhw/ \
src/librender/ src/luminaires/ src/medium/ src/mitsuba/ src/phase/ src/qtgui/ src/rfilters/ src/samplers/ \
src/shapes/ src/subsurface/ src/textures/ src/utils/ src/volume/ src/integrators/direct src/integrators/misc \
src/integrators/path src/integrators/photonmapper src/integrators/vpl tools/boost tools/darwin \
tools/build.sh tools/scons-1.2.0 tools/windows tools/linux/*.sh tools/linux/mitsuba.desktop tools/qt4.py \
tools/build.bat doc/license.txt setpath.sh

View File

@ -41,6 +41,31 @@
</bsdf> </bsdf>
</bsdf> </bsdf>
<!-- Test the composite material adapter with
a mix of two previously tested materials -->
<bsdf type="composite">
<string name="weights" value="0.4, 0.6"/>
<bsdf type="phong">
<float name="diffuseAmount" value="0.5"/>
<float name="specularAmount" value="0.5"/>
<float name="exponent" value="20"/>
<spectrum name="diffuseReflectance" value="1"/>
<spectrum name="specularReflectance" value="1"/>
</bsdf>
<bsdf type="ward">
<float name="diffuseAmount" value="0.5"/>
<float name="specularAmount" value="0.5"/>
<float name="alphaX" value="0.1"/>
<float name="alphaY" value="0.3"/>
<spectrum name="diffuseReflectance" value="1"/>
<spectrum name="specularReflectance" value="1"/>
</bsdf>
</bsdf>
<!-- Test the microfacet model --> <!-- Test the microfacet model -->
<bsdf type="microfacet"> <bsdf type="microfacet">
<float name="diffuseAmount" value="0.5"/> <float name="diffuseAmount" value="0.5"/>
@ -55,15 +80,6 @@
<float name="alphaB" value="0.1"/> <float name="alphaB" value="0.1"/>
</bsdf> </bsdf>
<!-- Test the rough glass model with the
Beckmann microfacet distribution -->
<bsdf type="roughglass">
<string name="distribution" value="beckmann"/>
<float name="alpha" value=".3"/>
<float name="intIOR" value="1.5"/>
<float name="extIOR" value="1.0"/>
</bsdf>
<!-- Test the rough glass model with the <!-- Test the rough glass model with the
GGX microfacet distribution --> GGX microfacet distribution -->
<bsdf type="roughglass"> <bsdf type="roughglass">
@ -82,5 +98,14 @@
<float name="extIOR" value="1.0"/> <float name="extIOR" value="1.0"/>
</bsdf> </bsdf>
<!-- Test the rough glass model with the
Beckmann microfacet distribution -->
<bsdf type="roughglass">
<string name="distribution" value="beckmann"/>
<float name="alpha" value=".3"/>
<float name="intIOR" value="1.5"/>
<float name="extIOR" value="1.0"/>
</bsdf>
<camera type="perspective"/> <camera type="perspective"/>
</scene> </scene>

View File

@ -29,13 +29,12 @@ MTS_NAMESPACE_BEGIN
class Composite : public BSDF { class Composite : public BSDF {
public: public:
Composite(const Properties &props) Composite(const Properties &props)
: BSDF(props), m_bsdfWeight(NULL) { : BSDF(props), m_bsdfWeights(NULL), m_componentIndex(NULL), m_bsdfOffset(NULL) {
/* Parse the weight parameter */ /* Parse the weight parameter */
std::vector<std::string> weights = std::vector<std::string> weights =
tokenize(props.getString("weights", ""), " ,;"); tokenize(props.getString("weights", ""), " ,;");
m_bsdfCount = weights.size(); m_bsdfCount = weights.size();
m_bsdfWeight = new Float[m_bsdfCount]; m_bsdfWeights = new Float[m_bsdfCount];
m_bsdfOffset = new int[m_bsdfCount];
Float totalWeight = 0; Float totalWeight = 0;
char *end_ptr = NULL; char *end_ptr = NULL;
@ -45,7 +44,7 @@ public:
SLog(EError, "Could not parse the BRDF weights!"); SLog(EError, "Could not parse the BRDF weights!");
if (weight < 0) if (weight < 0)
SLog(EError, "Invalid BRDF weight!"); SLog(EError, "Invalid BRDF weight!");
m_bsdfWeight[i] = weight; m_bsdfWeights[i] = weight;
totalWeight += weight; totalWeight += weight;
} }
@ -54,12 +53,11 @@ public:
} }
Composite(Stream *stream, InstanceManager *manager) Composite(Stream *stream, InstanceManager *manager)
: BSDF(stream, manager), m_bsdfWeight(NULL) { : BSDF(stream, manager), m_bsdfWeights(NULL), m_componentIndex(NULL), m_bsdfOffset(NULL) {
m_bsdfCount = stream->readSize(); m_bsdfCount = stream->readSize();
m_bsdfWeight = new Float[m_bsdfCount]; m_bsdfWeights = new Float[m_bsdfCount];
m_bsdfOffset = new int[m_bsdfCount];
for (size_t i=0; i<m_bsdfCount; ++i) { for (size_t i=0; i<m_bsdfCount; ++i) {
m_bsdfWeight[i] = stream->readFloat(); m_bsdfWeights[i] = stream->readFloat();
BSDF *bsdf = static_cast<BSDF *>(manager->getInstance(stream)); BSDF *bsdf = static_cast<BSDF *>(manager->getInstance(stream));
bsdf->incRef(); bsdf->incRef();
m_bsdfs.push_back(bsdf); m_bsdfs.push_back(bsdf);
@ -72,18 +70,20 @@ public:
m_bsdfs[i]->decRef(); m_bsdfs[i]->decRef();
if (m_type) if (m_type)
delete[] m_type; delete[] m_type;
if (m_bsdfWeight) { if (m_bsdfWeights)
delete[] m_bsdfWeight; delete[] m_bsdfWeights;
if (m_componentIndex)
delete[] m_componentIndex;
if (m_bsdfOffset)
delete[] m_bsdfOffset; delete[] m_bsdfOffset;
} }
}
void serialize(Stream *stream, InstanceManager *manager) const { void serialize(Stream *stream, InstanceManager *manager) const {
BSDF::serialize(stream, manager); BSDF::serialize(stream, manager);
stream->writeSize(m_bsdfCount); stream->writeSize(m_bsdfCount);
for (size_t i=0; i<m_bsdfCount; ++i) { for (size_t i=0; i<m_bsdfCount; ++i) {
stream->writeFloat(m_bsdfWeight[i]); stream->writeFloat(m_bsdfWeights[i]);
manager->serialize(stream, m_bsdfs[i]); manager->serialize(stream, m_bsdfs[i]);
} }
} }
@ -99,24 +99,37 @@ public:
Log(EError, "BSDF count mismatch: " SIZE_T_FMT " bsdfs, but specified " SIZE_T_FMT " weights", Log(EError, "BSDF count mismatch: " SIZE_T_FMT " bsdfs, but specified " SIZE_T_FMT " weights",
m_bsdfs.size(), m_bsdfCount); m_bsdfs.size(), m_bsdfCount);
int offset = 0;
for (size_t i=0; i<m_bsdfCount; ++i) for (size_t i=0; i<m_bsdfCount; ++i)
m_componentCount = m_bsdfs[i]->getComponentCount(); m_componentCount += m_bsdfs[i]->getComponentCount();
m_pdf = DiscretePDF(m_bsdfs.size()); m_pdf = DiscretePDF(m_bsdfs.size());
if (m_type)
delete[] m_type;
if (m_componentIndex)
delete[] m_componentIndex;
if (m_bsdfOffset)
delete[] m_bsdfOffset;
m_type = new unsigned int[m_componentCount]; m_type = new unsigned int[m_componentCount];
m_componentIndex = new std::pair<int, int>[m_componentCount];
m_bsdfOffset = new int[m_bsdfCount];
int ctr = 0, offset = 0;
for (size_t i=0; i<m_bsdfCount; ++i) { for (size_t i=0; i<m_bsdfCount; ++i) {
BSDF *bsdf = m_bsdfs[i]; const BSDF *bsdf = m_bsdfs[i];
m_bsdfOffset[i] = offset; m_bsdfOffset[i] = offset;
for (int j=0; j<bsdf->getComponentCount(); ++j) { for (int j=0; j<bsdf->getComponentCount(); ++j) {
int componentType = bsdf->getType(j); int componentType = bsdf->getType(j);
m_type[offset+j] = componentType; m_type[offset+j] = componentType;
m_componentIndex[ctr++] = std::make_pair((int) i, j);
} }
m_combinedType |= bsdf->getType(); m_combinedType |= bsdf->getType();
offset += bsdf->getComponentCount(); offset += bsdf->getComponentCount();
m_usesRayDifferentials |= bsdf->usesRayDifferentials(); m_usesRayDifferentials |= bsdf->usesRayDifferentials();
m_pdf[i] = m_bsdfWeight[i]; m_pdf[i] = m_bsdfWeights[i];
} }
m_pdf.build(); m_pdf.build();
} }
@ -124,7 +137,7 @@ public:
Spectrum getDiffuseReflectance(const Intersection &its) const { Spectrum getDiffuseReflectance(const Intersection &its) const {
Spectrum result(0.0f); Spectrum result(0.0f);
for (size_t i=0; i<m_bsdfCount; ++i) for (size_t i=0; i<m_bsdfCount; ++i)
result+= m_bsdfs[i]->getDiffuseReflectance(its) * m_bsdfWeight[i]; result+= m_bsdfs[i]->getDiffuseReflectance(its) * m_bsdfWeights[i];
return result; return result;
} }
@ -133,18 +146,13 @@ public:
if (bRec.component == -1) { if (bRec.component == -1) {
for (size_t i=0; i<m_bsdfCount; ++i) for (size_t i=0; i<m_bsdfCount; ++i)
result += m_bsdfs[i]->f(bRec) * m_bsdfWeight[i]; result += m_bsdfs[i]->f(bRec) * m_bsdfWeights[i];
} else { } else {
/* Pick out an individual component */ /* Pick out an individual component */
for (size_t i=0; i<m_bsdfCount; ++i) { int idx = m_componentIndex[bRec.component].first;
int component = bRec.component - m_bsdfOffset[i];
if (component < 0 || component >= m_bsdfs[i]->getComponentCount())
continue;
BSDFQueryRecord bRec2(bRec); BSDFQueryRecord bRec2(bRec);
bRec2.component = component; bRec2.component = m_componentIndex[bRec.component].second;
return m_bsdfs[i]->f(bRec2) * m_bsdfWeight[i]; return m_bsdfs[idx]->f(bRec2) * m_bsdfWeights[idx];
}
} }
return result; return result;
@ -155,18 +163,13 @@ public:
if (bRec.component == -1) { if (bRec.component == -1) {
for (size_t i=0; i<m_bsdfCount; ++i) for (size_t i=0; i<m_bsdfCount; ++i)
result += m_bsdfs[i]->fDelta(bRec) * m_bsdfWeight[i]; result += m_bsdfs[i]->fDelta(bRec) * m_bsdfWeights[i];
} else { } else {
/* Pick out an individual component */ /* Pick out an individual component */
for (size_t i=0; i<m_bsdfCount; ++i) { int idx = m_componentIndex[bRec.component].first;
int component = bRec.component - m_bsdfOffset[i];
if (component < 0 || component >= m_bsdfs[i]->getComponentCount())
continue;
BSDFQueryRecord bRec2(bRec); BSDFQueryRecord bRec2(bRec);
bRec2.component = component; bRec2.component = m_componentIndex[bRec.component].second;
return m_bsdfs[i]->fDelta(bRec2) * m_bsdfWeight[i]; return m_bsdfs[idx]->fDelta(bRec2) * m_bsdfWeights[idx];
}
} }
return result; return result;
@ -180,15 +183,10 @@ public:
result += m_bsdfs[i]->pdf(bRec) * m_pdf[i]; result += m_bsdfs[i]->pdf(bRec) * m_pdf[i];
} else { } else {
/* Pick out an individual component */ /* Pick out an individual component */
for (size_t i=0; i<m_bsdfCount; ++i) { int idx = m_componentIndex[bRec.component].first;
int component = bRec.component - m_bsdfOffset[i];
if (component < 0 || component >= m_bsdfs[i]->getComponentCount())
continue;
BSDFQueryRecord bRec2(bRec); BSDFQueryRecord bRec2(bRec);
bRec2.component = component; bRec2.component = m_componentIndex[bRec.component].second;
return m_bsdfs[i]->pdf(bRec2); return m_bsdfs[idx]->pdf(bRec2);
}
} }
return result; return result;
@ -202,15 +200,10 @@ public:
result += m_bsdfs[i]->pdfDelta(bRec) * m_pdf[i]; result += m_bsdfs[i]->pdfDelta(bRec) * m_pdf[i];
} else { } else {
/* Pick out an individual component */ /* Pick out an individual component */
for (size_t i=0; i<m_bsdfCount; ++i) { int idx = m_componentIndex[bRec.component].first;
int component = bRec.component - m_bsdfOffset[i];
if (component < 0 || component >= m_bsdfs[i]->getComponentCount())
continue;
BSDFQueryRecord bRec2(bRec); BSDFQueryRecord bRec2(bRec);
bRec2.component = component; bRec2.component = m_componentIndex[bRec.component].second;
return m_bsdfs[i]->pdfDelta(bRec2); return m_bsdfs[idx]->pdfDelta(bRec2);
}
} }
return result; return result;
@ -220,7 +213,11 @@ public:
Point2 sample(_sample); Point2 sample(_sample);
if (bRec.component == -1) { if (bRec.component == -1) {
int entry = m_pdf.sampleReuse(sample.x); int entry = m_pdf.sampleReuse(sample.x);
m_bsdfs[entry]->sample(bRec, sample); Spectrum result = m_bsdfs[entry]->sample(bRec, sample);
if (result.isZero()) // sampling failed
return result;
bRec.sampledComponent += m_bsdfOffset[entry]; bRec.sampledComponent += m_bsdfOffset[entry];
if (bRec.sampledType & BSDF::EDelta) { if (bRec.sampledType & BSDF::EDelta) {
@ -232,28 +229,24 @@ public:
} }
} else { } else {
/* Pick out an individual component */ /* Pick out an individual component */
for (size_t i=0; i<m_bsdfCount; ++i) { int idx = m_componentIndex[bRec.component].first;
int component = bRec.component - m_bsdfOffset[i];
if (component < 0 || component >= m_bsdfs[i]->getComponentCount())
continue;
int tempComponent = bRec.component; int tempComponent = bRec.component;
bRec.component = component; bRec.component = m_componentIndex[bRec.component].second;
Spectrum result = m_bsdfs[i]->sample(bRec, pdf, sample); Spectrum result = m_bsdfs[idx]->sample(bRec, pdf, sample) * m_bsdfWeights[idx];
bRec.component = bRec.sampledComponent = tempComponent; bRec.component = bRec.sampledComponent = tempComponent;
return result;
return result * m_bsdfWeight[i];
} }
} }
Log(EError, "Internal error!");
return Spectrum(0.0f);
}
Spectrum sample(BSDFQueryRecord &bRec, const Point2 &_sample) const { Spectrum sample(BSDFQueryRecord &bRec, const Point2 &_sample) const {
Point2 sample(_sample); Point2 sample(_sample);
if (bRec.component == -1) { if (bRec.component == -1) {
int entry = m_pdf.sampleReuse(sample.x); int entry = m_pdf.sampleReuse(sample.x);
m_bsdfs[entry]->sample(bRec, sample); Spectrum result = m_bsdfs[entry]->sample(bRec, sample);
if (result.isZero()) // sampling failed
return result;
bRec.sampledComponent += m_bsdfOffset[entry]; bRec.sampledComponent += m_bsdfOffset[entry];
if (bRec.sampledType & BSDF::EDelta) if (bRec.sampledType & BSDF::EDelta)
@ -262,22 +255,14 @@ public:
return f(bRec)/pdf(bRec); return f(bRec)/pdf(bRec);
} else { } else {
/* Pick out an individual component */ /* Pick out an individual component */
for (size_t i=0; i<m_bsdfCount; ++i) { int idx = m_componentIndex[bRec.component].first;
int component = bRec.component - m_bsdfOffset[i];
if (component < 0 || component >= m_bsdfs[i]->getComponentCount())
continue;
int tempComponent = bRec.component; int tempComponent = bRec.component;
bRec.component = component; bRec.component = m_componentIndex[bRec.component].second;
Spectrum result = m_bsdfs[i]->sample(bRec, sample); Spectrum result = m_bsdfs[idx]->sample(bRec, sample) * m_bsdfWeights[idx];
bRec.component = bRec.sampledComponent = tempComponent; bRec.component = bRec.sampledComponent = tempComponent;
return result;
return result * m_bsdfWeight[i];
} }
} }
Log(EError, "Internal error!");
return Spectrum(0.0f);
}
void addChild(const std::string &name, ConfigurableObject *child) { void addChild(const std::string &name, ConfigurableObject *child) {
if (child->getClass()->derivesFrom(MTS_CLASS(BSDF))) { if (child->getClass()->derivesFrom(MTS_CLASS(BSDF))) {
@ -292,11 +277,17 @@ public:
std::string toString() const { std::string toString() const {
std::ostringstream oss; std::ostringstream oss;
oss << "Composite[" << endl oss << "Composite[" << endl
<< " weights = {";
for (size_t i=0; i<m_bsdfCount; ++i) {
oss << " " << m_bsdfWeights[i];
if (i + 1 < m_bsdfCount)
oss << ",";
}
oss << " }," << endl
<< " bsdfs = {" << endl; << " bsdfs = {" << endl;
for (size_t i=0; i<m_bsdfs.size(); ++i) for (size_t i=0; i<m_bsdfs.size(); ++i)
oss << indent(m_bsdfs[i]->toString()) << "," << endl; oss << " " << indent(m_bsdfs[i]->toString(), 2) << "," << endl;
oss << " }" << endl oss << " }" << endl
<< " pdf = " << m_pdf.toString() << endl
<< "]"; << "]";
return oss.str(); return oss.str();
} }
@ -306,7 +297,8 @@ public:
MTS_DECLARE_CLASS() MTS_DECLARE_CLASS()
private: private:
size_t m_bsdfCount; size_t m_bsdfCount;
Float *m_bsdfWeight; Float *m_bsdfWeights;
std::pair<int, int> *m_componentIndex;
int *m_bsdfOffset; int *m_bsdfOffset;
std::vector<BSDF *> m_bsdfs; std::vector<BSDF *> m_bsdfs;
DiscretePDF m_pdf; DiscretePDF m_pdf;
@ -424,7 +416,7 @@ private:
}; };
Shader *Composite::createShader(Renderer *renderer) const { Shader *Composite::createShader(Renderer *renderer) const {
return new CompositeShader(renderer, m_bsdfs, m_bsdfWeight); return new CompositeShader(renderer, m_bsdfs, m_bsdfWeights);
} }
MTS_IMPLEMENT_CLASS(CompositeShader, false, Shader) MTS_IMPLEMENT_CLASS(CompositeShader, false, Shader)

View File

@ -41,7 +41,7 @@ public:
virtual ~Mask() { virtual ~Mask() {
if (m_type) if (m_type)
delete m_type; delete[] m_type;
} }
void serialize(Stream *stream, InstanceManager *manager) const { void serialize(Stream *stream, InstanceManager *manager) const {

View File

@ -111,6 +111,37 @@ public:
return Spectrum(0.0f); return Spectrum(0.0f);
} }
inline Vector reflect(const Vector &wi, const Normal &m) const {
return 2 * dot(wi, m) * Vector(m) - wi;
}
inline bool refract(const Vector &wi, Vector &wo, const Normal &m) const {
/* Determine the appropriate indices of refraction */
Float etaI = m_extIOR, etaT = m_intIOR;
if (Frame::cosTheta(wi) < 0)
std::swap(etaI, etaT);
Float eta = etaI / etaT, c = dot(wi, m);
/* Using Snell's law, calculate the squared cosine of the
angle between the normal and the transmitted ray */
Float cosThetaTSqr = 1 + eta * eta * (c*c-1);
if (cosThetaTSqr < 0)
return false; // Total internal reflection
/* Compute the transmitted direction */
wo = m * (eta*c - signum(wi.z)
* std::sqrt(cosThetaTSqr)) - wi * eta;
return true;
}
inline Float signum(Float value) const {
return (value < 0) ? -1.0f : 1.0f;
}
/** /**
* \brief Implements the microfacet distribution function D * \brief Implements the microfacet distribution function D
* *
@ -248,29 +279,6 @@ public:
return Normal(sphericalDirection(thetaM, phiM)); return Normal(sphericalDirection(thetaM, phiM));
} }
inline Vector reflect(const Vector &wi, const Normal &m) const {
return 2 * dot(wi, m) * Vector(m) - wi;
}
inline bool refract(const Vector &wi, Vector &wo, const Normal &m) const {
Float etaI = m_extIOR, etaT = m_intIOR;
if (Frame::cosTheta(wi) < 0)
std::swap(etaI, etaT);
Float eta = etaI / etaT, c = dot(wi, m);
Float cosThetaTSqr = 1 + eta * eta * (c*c-1);
if (cosThetaTSqr < 0) // Total internal reflection
return false;
wo = m * (eta*c - signum(wi.z)
* std::sqrt(cosThetaTSqr)) - wi * eta;
return true;
}
inline Float signum(Float value) const {
return (value < 0) ? -1.0f : 1.0f;
}
inline Spectrum fReflection(const BSDFQueryRecord &bRec) const { inline Spectrum fReflection(const BSDFQueryRecord &bRec) const {
Float intIOR = m_intIOR, extIOR = m_extIOR; Float intIOR = m_intIOR, extIOR = m_extIOR;
@ -448,7 +456,7 @@ public:
/* Sample the microfacet normal */ /* Sample the microfacet normal */
Vector m = sampleD(sample, alpha); Vector m = sampleD(sample, alpha);
/* Refract */ /* Refract based on 'm' */
if (!refract(bRec.wi, bRec.wo, m)) if (!refract(bRec.wi, bRec.wo, m))
return Spectrum(0.0f); return Spectrum(0.0f);

View File

@ -66,6 +66,8 @@ void help() {
FileResolver *fileResolver = Thread::getThread()->getFileResolver(); FileResolver *fileResolver = Thread::getThread()->getFileResolver();
std::ostringstream utilities, testcases; std::ostringstream utilities, testcases;
cout << "[ Loading plugin list .. ]" << endl << endl;
testcases << "The following testcases are available:" << endl << endl; testcases << "The following testcases are available:" << endl << endl;
utilities << endl << "The following utilities are available:" << endl << endl; utilities << endl << "The following utilities are available:" << endl << endl;

View File

@ -73,7 +73,7 @@ public:
SAssert(a >= 0 && b >= 0); SAssert(a >= 0 && b >= 0);
Float min = std::min(a, b); Float min = std::min(a, b);
Float err = std::abs(a - b); Float err = std::abs(a - b);
m_largestWeight = std::max(m_largestWeight, a * Frame::cosTheta(bRec.wo)); m_largestWeight = std::max(m_largestWeight, a * std::abs(Frame::cosTheta(bRec.wo)));
if (min < Epsilon && err > Epsilon) // absolute error threshold if (min < Epsilon && err > Epsilon) // absolute error threshold
mismatch = true; mismatch = true;
@ -205,8 +205,10 @@ public:
progress->update(j+1); progress->update(j+1);
} }
} }
Log(EInfo, "Done with this BSDF. The largest encountered "
"importance weight was = %.2f", largestWeight);
largestWeight = 0;
} }
Log(EInfo, "Done with this model. The largest encountered importance weight was = %.2f", largestWeight);
} }
Log(EInfo, "%i/%i BSDF checks succeeded", testCount-failureCount, testCount); Log(EInfo, "%i/%i BSDF checks succeeded", testCount-failureCount, testCount);
delete progress; delete progress;