2010-09-03 05:41:20 +08:00
|
|
|
/*
|
|
|
|
This file is part of Mitsuba, a physically based rendering system.
|
|
|
|
|
2011-04-14 21:15:59 +08:00
|
|
|
Copyright (c) 2007-2011 by Wenzel Jakob and others.
|
2010-09-03 05:41:20 +08:00
|
|
|
|
|
|
|
Mitsuba is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License Version 3
|
|
|
|
as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
Mitsuba is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2011-04-14 21:15:59 +08:00
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2010-09-03 05:41:20 +08:00
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2010-08-10 01:38:37 +08:00
|
|
|
#include <mitsuba/render/texture.h>
|
2010-09-10 09:14:48 +08:00
|
|
|
#include <mitsuba/render/shape.h>
|
|
|
|
#include <mitsuba/core/properties.h>
|
2010-11-05 08:03:13 +08:00
|
|
|
#include <mitsuba/hw/gpuprogram.h>
|
2010-08-10 01:38:37 +08:00
|
|
|
|
|
|
|
MTS_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checkerboard texture
|
|
|
|
*/
|
2010-11-05 08:03:13 +08:00
|
|
|
class Checkerboard : public Texture2D {
|
2010-08-10 01:38:37 +08:00
|
|
|
public:
|
2010-11-05 08:03:13 +08:00
|
|
|
Checkerboard(const Properties &props) : Texture2D(props) {
|
2010-11-13 08:53:52 +08:00
|
|
|
m_brightColor = props.getSpectrum("brightColor", Spectrum(.4f));
|
|
|
|
m_darkColor = props.getSpectrum("darkColor", Spectrum(.2f));
|
2010-08-10 01:38:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Checkerboard(Stream *stream, InstanceManager *manager)
|
2010-11-05 08:03:13 +08:00
|
|
|
: Texture2D(stream, manager) {
|
2010-11-13 08:53:52 +08:00
|
|
|
m_brightColor = Spectrum(stream);
|
|
|
|
m_darkColor = Spectrum(stream);
|
2010-08-10 01:38:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void serialize(Stream *stream, InstanceManager *manager) const {
|
2010-11-05 08:03:13 +08:00
|
|
|
Texture2D::serialize(stream, manager);
|
2010-11-13 08:53:52 +08:00
|
|
|
m_brightColor.serialize(stream);
|
|
|
|
m_darkColor.serialize(stream);
|
2010-08-10 01:38:37 +08:00
|
|
|
}
|
|
|
|
|
2010-11-05 08:03:13 +08:00
|
|
|
inline Spectrum getValue(const Point2 &uv) const {
|
2010-11-19 21:07:14 +08:00
|
|
|
int x = 2*modulo((int) (uv.x * 2), 2) - 1,
|
|
|
|
y = 2*modulo((int) (uv.y * 2), 2) - 1;
|
2010-11-05 08:03:13 +08:00
|
|
|
|
2010-08-10 01:38:37 +08:00
|
|
|
if (x*y == 1)
|
2010-11-13 08:53:52 +08:00
|
|
|
return m_brightColor;
|
2010-08-10 01:38:37 +08:00
|
|
|
else
|
2010-11-13 08:53:52 +08:00
|
|
|
return m_darkColor;
|
2010-08-10 01:38:37 +08:00
|
|
|
}
|
2010-11-05 08:03:13 +08:00
|
|
|
|
|
|
|
Spectrum getValue(const Point2 &uv, Float dudx, Float dudy, Float dvdx, Float dvdy) const {
|
|
|
|
return Checkerboard::getValue(uv);
|
|
|
|
}
|
|
|
|
|
2010-08-10 01:38:37 +08:00
|
|
|
bool usesRayDifferentials() const {
|
|
|
|
return false;
|
|
|
|
}
|
2010-11-05 08:03:13 +08:00
|
|
|
|
2010-08-10 01:38:37 +08:00
|
|
|
Spectrum getAverage() const {
|
2010-11-13 08:53:52 +08:00
|
|
|
return m_darkColor * .5f;
|
2010-08-10 01:38:37 +08:00
|
|
|
}
|
2010-08-15 21:24:30 +08:00
|
|
|
|
|
|
|
Spectrum getMaximum() const {
|
2010-11-13 08:53:52 +08:00
|
|
|
return m_brightColor;
|
2010-08-15 21:24:30 +08:00
|
|
|
}
|
2010-08-10 01:38:37 +08:00
|
|
|
|
2011-07-13 23:40:26 +08:00
|
|
|
bool isConstant() const {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-08-10 01:38:37 +08:00
|
|
|
std::string toString() const {
|
2011-08-13 13:07:31 +08:00
|
|
|
std::ostringstream oss;
|
|
|
|
oss << "Checkerboard[" << endl
|
|
|
|
<< " darkColor = " << m_darkColor.toString() << "," << endl
|
|
|
|
<< " brightColor = " << m_brightColor.toString() << endl
|
|
|
|
<< "]";
|
|
|
|
return oss.str();
|
2010-08-10 01:38:37 +08:00
|
|
|
}
|
2010-11-05 08:03:13 +08:00
|
|
|
|
|
|
|
Shader *createShader(Renderer *renderer) const;
|
2010-08-10 01:38:37 +08:00
|
|
|
|
|
|
|
MTS_DECLARE_CLASS()
|
|
|
|
protected:
|
2010-11-13 08:53:52 +08:00
|
|
|
Spectrum m_darkColor;
|
|
|
|
Spectrum m_brightColor;
|
2010-08-10 01:38:37 +08:00
|
|
|
};
|
|
|
|
|
2010-11-05 08:03:13 +08:00
|
|
|
// ================ Hardware shader implementation ================
|
|
|
|
|
|
|
|
class CheckerboardShader : public Shader {
|
|
|
|
public:
|
2010-11-13 08:53:52 +08:00
|
|
|
CheckerboardShader(Renderer *renderer, const Spectrum &brightColor,
|
|
|
|
const Spectrum &darkColor, const Point2 &uvOffset,
|
2010-11-05 08:03:13 +08:00
|
|
|
const Vector2 &uvScale) : Shader(renderer, ETextureShader),
|
2010-11-13 08:53:52 +08:00
|
|
|
m_brightColor(brightColor), m_darkColor(darkColor),
|
2010-11-05 08:03:13 +08:00
|
|
|
m_uvOffset(uvOffset), m_uvScale(uvScale) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void generateCode(std::ostringstream &oss,
|
|
|
|
const std::string &evalName,
|
|
|
|
const std::vector<std::string> &depNames) const {
|
2010-11-13 08:53:52 +08:00
|
|
|
oss << "uniform vec3 " << evalName << "_brightColor;" << endl
|
|
|
|
<< "uniform vec3 " << evalName << "_darkColor;" << endl
|
2010-11-05 08:03:13 +08:00
|
|
|
<< "uniform vec2 " << evalName << "_uvOffset;" << endl
|
|
|
|
<< "uniform vec2 " << evalName << "_uvScale;" << endl
|
|
|
|
<< endl
|
|
|
|
<< "vec3 " << evalName << "(vec2 uv) {" << endl
|
|
|
|
<< " uv = vec2(" << endl
|
|
|
|
<< " uv.x * " << evalName << "_uvScale.x + " << evalName << "_uvOffset.x," << endl
|
|
|
|
<< " uv.y * " << evalName << "_uvScale.y + " << evalName << "_uvOffset.y);" << endl
|
2010-11-13 08:53:52 +08:00
|
|
|
<< " float x = 2*(mod(int(uv.x*2), 2)) - 1, y = 2*(mod(int(uv.y*2), 2)) - 1;" << endl
|
2010-11-05 08:03:13 +08:00
|
|
|
<< " if (x*y == 1)" << endl
|
2010-11-13 08:53:52 +08:00
|
|
|
<< " return " << evalName << "_brightColor;" << endl
|
2010-11-05 08:03:13 +08:00
|
|
|
<< " else" << endl
|
2010-11-13 08:53:52 +08:00
|
|
|
<< " return " << evalName << "_darkColor;" << endl
|
2010-11-05 08:03:13 +08:00
|
|
|
<< "}" << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
void resolve(const GPUProgram *program, const std::string &evalName, std::vector<int> ¶meterIDs) const {
|
2010-11-13 08:53:52 +08:00
|
|
|
parameterIDs.push_back(program->getParameterID(evalName + "_brightColor", false));
|
|
|
|
parameterIDs.push_back(program->getParameterID(evalName + "_darkColor", false));
|
2010-11-05 08:03:13 +08:00
|
|
|
parameterIDs.push_back(program->getParameterID(evalName + "_uvOffset", false));
|
|
|
|
parameterIDs.push_back(program->getParameterID(evalName + "_uvScale", false));
|
|
|
|
}
|
|
|
|
|
|
|
|
void bind(GPUProgram *program, const std::vector<int> ¶meterIDs,
|
|
|
|
int &textureUnitOffset) const {
|
2010-11-13 08:53:52 +08:00
|
|
|
program->setParameter(parameterIDs[0], m_brightColor);
|
|
|
|
program->setParameter(parameterIDs[1], m_darkColor);
|
2010-11-05 08:03:13 +08:00
|
|
|
program->setParameter(parameterIDs[2], m_uvOffset);
|
|
|
|
program->setParameter(parameterIDs[3], m_uvScale);
|
|
|
|
}
|
2011-07-13 23:40:26 +08:00
|
|
|
|
2010-11-05 08:03:13 +08:00
|
|
|
MTS_DECLARE_CLASS()
|
|
|
|
private:
|
2010-11-13 08:53:52 +08:00
|
|
|
Spectrum m_brightColor;
|
|
|
|
Spectrum m_darkColor;
|
2010-11-05 08:03:13 +08:00
|
|
|
Point2 m_uvOffset;
|
|
|
|
Vector2 m_uvScale;
|
|
|
|
};
|
|
|
|
|
|
|
|
Shader *Checkerboard::createShader(Renderer *renderer) const {
|
2010-11-13 08:53:52 +08:00
|
|
|
return new CheckerboardShader(renderer, m_brightColor, m_darkColor,
|
2010-11-05 08:03:13 +08:00
|
|
|
m_uvOffset, m_uvScale);
|
|
|
|
}
|
|
|
|
|
|
|
|
MTS_IMPLEMENT_CLASS(CheckerboardShader, false, Shader)
|
|
|
|
MTS_IMPLEMENT_CLASS_S(Checkerboard, false, Texture2D)
|
2010-08-10 01:38:37 +08:00
|
|
|
MTS_EXPORT_PLUGIN(Checkerboard, "Checkerboard texture");
|
|
|
|
MTS_NAMESPACE_END
|