added a normalization feature
parent
2f553660c4
commit
854cd049e7
|
@ -28,6 +28,7 @@
|
||||||
#include <mitsuba/core/fresolver.h>
|
#include <mitsuba/core/fresolver.h>
|
||||||
#include <mitsuba/core/fstream.h>
|
#include <mitsuba/core/fstream.h>
|
||||||
#include <mitsuba/core/bitmap.h>
|
#include <mitsuba/core/bitmap.h>
|
||||||
|
#include <mitsuba/render/scene.h>
|
||||||
#include "irawan.h"
|
#include "irawan.h"
|
||||||
|
|
||||||
MTS_NAMESPACE_BEGIN
|
MTS_NAMESPACE_BEGIN
|
||||||
|
@ -76,7 +77,7 @@ MTS_NAMESPACE_BEGIN
|
||||||
class IrawanClothBRDF : public BSDF {
|
class IrawanClothBRDF : public BSDF {
|
||||||
public:
|
public:
|
||||||
IrawanClothBRDF(const Properties &props)
|
IrawanClothBRDF(const Properties &props)
|
||||||
: BSDF(props) {
|
: BSDF(props), m_specularNormalization(0) {
|
||||||
|
|
||||||
FileResolver *fResolver = Thread::getThread()->getFileResolver();
|
FileResolver *fResolver = Thread::getThread()->getFileResolver();
|
||||||
fs::path path = fResolver->resolve(props.getString("filename"));
|
fs::path path = fResolver->resolve(props.getString("filename"));
|
||||||
|
@ -90,8 +91,8 @@ public:
|
||||||
WeavePatternGrammar<iterator_type> g(props);
|
WeavePatternGrammar<iterator_type> g(props);
|
||||||
SkipGrammar<iterator_type> sg;
|
SkipGrammar<iterator_type> sg;
|
||||||
|
|
||||||
bool result = phrase_parse(begin, end, g, sg, m_pattern);
|
bool success = phrase_parse(begin, end, g, sg, m_pattern);
|
||||||
if (!result)
|
if (!success)
|
||||||
Log(EError, "Unable to parse the weave pattern file \"%s\"!",
|
Log(EError, "Unable to parse the weave pattern file \"%s\"!",
|
||||||
path.file_string().c_str());
|
path.file_string().c_str());
|
||||||
|
|
||||||
|
@ -106,9 +107,10 @@ public:
|
||||||
m_repeatU = props.getFloat("repeatU");
|
m_repeatU = props.getFloat("repeatU");
|
||||||
m_repeatV = props.getFloat("repeatV");
|
m_repeatV = props.getFloat("repeatV");
|
||||||
|
|
||||||
/* Diffuse and specular multipliers */
|
if (props.hasProperty("ksMultiplier") || props.hasProperty("kdMultiplier"))
|
||||||
m_kdMultiplier = props.getFloat("kdMultiplier");
|
Log(EError, "The 'ksMultiplier' and 'kdMultiplier' parameters were "
|
||||||
m_ksMultiplier = props.getFloat("ksMultiplier");
|
"replaced by a normalization scheme. Please remove them and"
|
||||||
|
"potentially adapt the 'kd' and 'ks'-values used in your model.");
|
||||||
}
|
}
|
||||||
|
|
||||||
IrawanClothBRDF(Stream *stream, InstanceManager *manager)
|
IrawanClothBRDF(Stream *stream, InstanceManager *manager)
|
||||||
|
@ -116,17 +118,54 @@ public:
|
||||||
m_pattern = WeavePattern(stream);
|
m_pattern = WeavePattern(stream);
|
||||||
m_repeatU = stream->readFloat();
|
m_repeatU = stream->readFloat();
|
||||||
m_repeatV = stream->readFloat();
|
m_repeatV = stream->readFloat();
|
||||||
m_kdMultiplier = stream->readFloat();
|
m_specularNormalization = stream->readFloat();
|
||||||
m_ksMultiplier = stream->readFloat();
|
|
||||||
configure();
|
configure();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void serialize(Stream *stream, InstanceManager *manager) const {
|
||||||
|
BSDF::serialize(stream, manager);
|
||||||
|
|
||||||
|
m_pattern.serialize(stream);
|
||||||
|
stream->writeFloat(m_repeatU);
|
||||||
|
stream->writeFloat(m_repeatV);
|
||||||
|
stream->writeFloat(m_specularNormalization);
|
||||||
|
}
|
||||||
|
|
||||||
void configure() {
|
void configure() {
|
||||||
m_components.clear();
|
m_components.clear();
|
||||||
m_components.push_back(EGlossyReflection | EFrontSide
|
m_components.push_back(EGlossyReflection | EFrontSide
|
||||||
| EAnisotropic | ESpatiallyVarying);
|
| EAnisotropic | ESpatiallyVarying);
|
||||||
m_components.push_back(EDiffuseReflection | EFrontSide
|
m_components.push_back(EDiffuseReflection | EFrontSide
|
||||||
| ESpatiallyVarying);
|
| ESpatiallyVarying);
|
||||||
|
|
||||||
|
/* Estimate the average reflectance under diffuse
|
||||||
|
illumination and use it to normalize the specular
|
||||||
|
component */
|
||||||
|
ref<Random> random = new Random();
|
||||||
|
size_t nSamples = 10000;
|
||||||
|
|
||||||
|
if (m_specularNormalization == 0) {
|
||||||
|
Intersection its;
|
||||||
|
BSDFQueryRecord bRec(its, NULL, ERadiance);
|
||||||
|
Spectrum result(0.0f);
|
||||||
|
m_initialization = true;
|
||||||
|
for (size_t i=0; i<nSamples; ++i) {
|
||||||
|
bRec.wi = squareToHemispherePSA(Point2(random->nextFloat(), random->nextFloat()));
|
||||||
|
bRec.wo = squareToHemispherePSA(Point2(random->nextFloat(), random->nextFloat()));
|
||||||
|
its.uv = Point2(random->nextFloat(), random->nextFloat());
|
||||||
|
|
||||||
|
result += eval(bRec, ESolidAngle) / Frame::cosTheta(bRec.wo);
|
||||||
|
}
|
||||||
|
m_initialization = false;
|
||||||
|
|
||||||
|
if (result.max() == 0)
|
||||||
|
m_specularNormalization = 0;
|
||||||
|
else
|
||||||
|
m_specularNormalization = nSamples / (result.max() * M_PI);
|
||||||
|
Log(EDebug, "Specular normalization factor = %f",
|
||||||
|
m_specularNormalization);
|
||||||
|
}
|
||||||
|
|
||||||
BSDF::configure();
|
BSDF::configure();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -140,15 +179,15 @@ public:
|
||||||
int yarnID = m_pattern.pattern[lookup.x + lookup.y * m_pattern.tileWidth] - 1;
|
int yarnID = m_pattern.pattern[lookup.x + lookup.y * m_pattern.tileWidth] - 1;
|
||||||
const Yarn &yarn = m_pattern.yarns.at(yarnID);
|
const Yarn &yarn = m_pattern.yarns.at(yarnID);
|
||||||
|
|
||||||
return yarn.kd * m_kdMultiplier;
|
return yarn.kd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Spectrum eval(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
Spectrum eval(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
||||||
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 0) && m_ksMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 0);
|
||||||
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 1) && m_kdMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 1);
|
||||||
|
|
||||||
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
||||||
Frame::cosTheta(bRec.wo) <= 0 ||
|
Frame::cosTheta(bRec.wo) <= 0 ||
|
||||||
|
@ -261,24 +300,28 @@ public:
|
||||||
intensityVariation = std::min(-std::fastlog(xi), (Float) 10.0f);
|
intensityVariation = std::min(-std::fastlog(xi), (Float) 10.0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
result = yarn.ks * (intensityVariation * m_ksMultiplier * integrand);
|
if (!m_initialization)
|
||||||
|
result = yarn.ks * (intensityVariation * integrand * m_specularNormalization);
|
||||||
|
else
|
||||||
|
result = Spectrum(intensityVariation * integrand);
|
||||||
|
|
||||||
if (type == Yarn::EWarp)
|
if (type == Yarn::EWarp)
|
||||||
result *= (m_pattern.warpArea + m_pattern.weftArea) / m_pattern.warpArea;
|
result *= (m_pattern.warpArea + m_pattern.weftArea) / m_pattern.warpArea;
|
||||||
else
|
else
|
||||||
result *= (m_pattern.warpArea + m_pattern.weftArea) / m_pattern.weftArea;
|
result *= (m_pattern.warpArea + m_pattern.weftArea) / m_pattern.weftArea;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hasDiffuse)
|
if (hasDiffuse && !m_initialization)
|
||||||
result += yarn.kd * m_kdMultiplier * INV_PI;
|
result += yarn.kd * INV_PI;
|
||||||
|
|
||||||
return result * Frame::cosTheta(bRec.wo);
|
return result * Frame::cosTheta(bRec.wo);
|
||||||
}
|
}
|
||||||
|
|
||||||
Float pdf(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
Float pdf(const BSDFQueryRecord &bRec, EMeasure measure) const {
|
||||||
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 0) && m_ksMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 0);
|
||||||
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 1) && m_kdMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 1);
|
||||||
|
|
||||||
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
||||||
Frame::cosTheta(bRec.wo) <= 0 ||
|
Frame::cosTheta(bRec.wo) <= 0 ||
|
||||||
|
@ -291,9 +334,9 @@ public:
|
||||||
|
|
||||||
Spectrum sample(BSDFQueryRecord &bRec, const Point2 &sample) const {
|
Spectrum sample(BSDFQueryRecord &bRec, const Point2 &sample) const {
|
||||||
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 0) && m_ksMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 0);
|
||||||
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 1) && m_kdMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 1);
|
||||||
|
|
||||||
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
||||||
(!hasDiffuse && !hasSpecular))
|
(!hasDiffuse && !hasSpecular))
|
||||||
|
@ -308,9 +351,9 @@ public:
|
||||||
|
|
||||||
Spectrum sample(BSDFQueryRecord &bRec, Float &pdf, const Point2 &sample) const {
|
Spectrum sample(BSDFQueryRecord &bRec, Float &pdf, const Point2 &sample) const {
|
||||||
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
bool hasSpecular = (bRec.typeMask & EGlossyReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 0) && m_ksMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 0);
|
||||||
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
bool hasDiffuse = (bRec.typeMask & EDiffuseReflection) &&
|
||||||
(bRec.component == -1 || bRec.component == 1) && m_kdMultiplier > 0;
|
(bRec.component == -1 || bRec.component == 1);
|
||||||
|
|
||||||
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
if (Frame::cosTheta(bRec.wi) <= 0 ||
|
||||||
(!hasDiffuse && !hasSpecular))
|
(!hasDiffuse && !hasSpecular))
|
||||||
|
@ -324,16 +367,6 @@ public:
|
||||||
return eval(bRec, ESolidAngle) / pdf;
|
return eval(bRec, ESolidAngle) / pdf;
|
||||||
}
|
}
|
||||||
|
|
||||||
void serialize(Stream *stream, InstanceManager *manager) const {
|
|
||||||
BSDF::serialize(stream, manager);
|
|
||||||
|
|
||||||
m_pattern.serialize(stream);
|
|
||||||
stream->writeFloat(m_repeatU);
|
|
||||||
stream->writeFloat(m_repeatV);
|
|
||||||
stream->writeFloat(m_kdMultiplier);
|
|
||||||
stream->writeFloat(m_ksMultiplier);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** parameters:
|
/** parameters:
|
||||||
* u to be compared to u(v) in texturing
|
* u to be compared to u(v) in texturing
|
||||||
* v for filament, we compute u(v)
|
* v for filament, we compute u(v)
|
||||||
|
@ -583,9 +616,7 @@ public:
|
||||||
oss << "IrawanClothBRDF[" << endl
|
oss << "IrawanClothBRDF[" << endl
|
||||||
<< " weavePattern = " << indent(m_pattern.toString()) << "," << endl
|
<< " weavePattern = " << indent(m_pattern.toString()) << "," << endl
|
||||||
<< " repeatU = " << m_repeatU << "," << endl
|
<< " repeatU = " << m_repeatU << "," << endl
|
||||||
<< " repeatV = " << m_repeatV << "," << endl
|
<< " repeatV = " << m_repeatV << endl
|
||||||
<< " kdMultiplier = " << m_kdMultiplier << "," << endl
|
|
||||||
<< " ksMultiplier = " << m_ksMultiplier << endl
|
|
||||||
<< "]";
|
<< "]";
|
||||||
return oss.str();
|
return oss.str();
|
||||||
}
|
}
|
||||||
|
@ -596,8 +627,8 @@ public:
|
||||||
private:
|
private:
|
||||||
WeavePattern m_pattern;
|
WeavePattern m_pattern;
|
||||||
Float m_repeatU, m_repeatV;
|
Float m_repeatU, m_repeatV;
|
||||||
Float m_kdMultiplier;
|
Float m_specularNormalization;
|
||||||
Float m_ksMultiplier;
|
bool m_initialization;
|
||||||
};
|
};
|
||||||
|
|
||||||
// ================ Hardware shader implementation ================
|
// ================ Hardware shader implementation ================
|
||||||
|
@ -638,7 +669,7 @@ Shader *IrawanClothBRDF::createShader(Renderer *renderer) const {
|
||||||
Spectrum albedo(0.0f);
|
Spectrum albedo(0.0f);
|
||||||
for (size_t i=0; i<m_pattern.yarns.size(); ++i)
|
for (size_t i=0; i<m_pattern.yarns.size(); ++i)
|
||||||
albedo += m_pattern.yarns[i].kd;
|
albedo += m_pattern.yarns[i].kd;
|
||||||
albedo *= m_kdMultiplier / (Float) m_pattern.yarns.size();
|
albedo /= (Float) m_pattern.yarns.size();
|
||||||
return new IrawanShader(renderer, albedo);
|
return new IrawanShader(renderer, albedo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue