fixed the composite material
parent
97ba51d675
commit
071b7379c1
|
@ -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
|
|
|
@ -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>
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in New Issue