/*
This file is part of Mitsuba, a physically based rendering system.
Copyright (c) 2007-2011 by Wenzel Jakob and others.
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
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
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 .
*/
#include
#include
#include
#include
MTS_NAMESPACE_BEGIN
void TestCase::init() { }
void TestCase::shutdown() { }
void TestCase::assertTrueImpl(bool value, const char *expr, const char *file, int line) {
if (!value)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion '%s == true' failed!", expr);
}
void TestCase::assertFalseImpl(bool value, const char *expr, const char *file, int line) {
if (value)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion '%s == false' failed!", expr);
}
void TestCase::assertEqualsImpl(int expected, int actual, const char *file, int line) {
if (expected != actual)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected integer value %i, got %i.", expected, actual);
}
void TestCase::assertEqualsImpl(Float expected, Float actual, Float epsilon, const char *file, int line) {
if (std::abs(expected-actual) > epsilon)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected floating point value %f, got %f.", expected, actual);
}
void TestCase::assertEqualsImpl(const Spectrum &expected, const Spectrum &actual, Float epsilon, const char *file, int line) {
bool match = true;
for (int i=0; i epsilon)
match = false;
if (!match)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected vector %s, got %s.", expected.toString().c_str(), actual.toString().c_str());
}
void TestCase::assertEqualsImpl(const Vector2 &expected, const Vector2 &actual, Float epsilon, const char *file, int line) {
bool match = true;
for (int i=0; i<2; ++i)
if (std::abs(expected[i]-actual[i]) > epsilon)
match = false;
if (!match)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected vector %s, got %s.", expected.toString().c_str(), actual.toString().c_str());
}
void TestCase::assertEqualsImpl(const Point2 &expected, const Point2 &actual, Float epsilon, const char *file, int line) {
bool match = true;
for (int i=0; i<2; ++i)
if (std::abs(expected[i]-actual[i]) > epsilon)
match = false;
if (!match)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected point %s, got %s.", expected.toString().c_str(), actual.toString().c_str());
}
void TestCase::assertEqualsImpl(const Vector &expected, const Vector &actual, Float epsilon, const char *file, int line) {
bool match = true;
for (int i=0; i<3; ++i)
if (std::abs(expected[i]-actual[i]) > epsilon)
match = false;
if (!match)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected vector %s, got %s.", expected.toString().c_str(), actual.toString().c_str());
}
void TestCase::assertEqualsImpl(const Point &expected, const Point &actual, Float epsilon, const char *file, int line) {
bool match = true;
for (int i=0; i<3; ++i)
if (std::abs(expected[i]-actual[i]) > epsilon)
match = false;
if (!match)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected point %s, got %s.", expected.toString().c_str(), actual.toString().c_str());
}
void TestCase::assertEqualsImpl(const Vector4 &expected, const Vector4 &actual, Float epsilon, const char *file, int line) {
bool match = true;
for (int i=0; i<4; ++i)
if (std::abs(expected[i]-actual[i]) > epsilon)
match = false;
if (!match)
Thread::getThread()->getLogger()->log(EError, NULL, file, line, "Assertion failure: "
"expected vector %s, got %s.", expected.toString().c_str(), actual.toString().c_str());
}
void TestCase::failAndContinueImpl(const std::string &msg, const char *file, int line) {
Thread::getThread()->getLogger()->log(EWarn, NULL, file, line, "Failure: %s", msg.c_str());
m_executed++;
}
void TestCase::succeed() {
m_executed++;
m_succeeded++;
}
struct Sample {
Float value;
Float variance;
int nSamples;
};
static std::vector parseRefFile(std::ifstream &is) {
std::string line;
std::vector result;
while (!is.eof() && !is.fail()) {
std::getline(is, line);
std::vector tokens = tokenize(line, " \t;,[]");
for (size_t i=0; i parseMFile(std::ifstream &is, int testType) {
std::string line;
std::vector result;
while (!is.eof() && !is.fail()) {
std::getline(is, line);
std::vector tokens = tokenize(line, " \t;,[]");
SAssert(testType == Scene::ERelativeError || (tokens.size() % 3) == 0);
for (size_t i=0; igetSourceFile();
result.output = scene->getDestinationFile();
result.output.replace_extension(".m");
result.success = false;
fs::path refFilename = scene->getDestinationFile();
refFilename.replace_extension(".ref");
fs::ifstream is(result.output);
fs::ifstream is_ref(refFilename);
if (is.fail()) {
result.message = formatString("Could not open '%s'!",
result.output.file_string().c_str());
m_mutex->lock();
m_numFailed++; m_results.push_back(result);
m_mutex->unlock();
return;
}
if (is_ref.fail()) {
result.message = formatString("Could not open '%s'!",
refFilename.file_string().c_str());
m_mutex->lock();
m_numFailed++; m_results.push_back(result);
m_mutex->unlock();
return;
}
std::vector actual = parseMFile(is, scene->getTestType());
std::vector ref = parseRefFile(is_ref);
is.close();
is_ref.close();
if (actual.size() != ref.size()) {
result.message = formatString("Output format does not match the reference (%i vs %i pixels)!",
(int) actual.size(), (int) ref.size());
m_mutex->lock();
m_numFailed++; m_results.push_back(result);
m_mutex->unlock();
return;
}
if (scene->getTestType() == Scene::ETTest) {
for (size_t i=0; igetTestThreshold()) {
Log(EWarn, "t-test REJECTS!");
result.message = formatString("t-test REJECTS: result=%f (ref=%f), diff=%e, var=%f T-stat=%f, df=%i, p-value=%f", actual[i].value, ref[i], actual[i].value - ref[i], var, T, df, pval);
m_mutex->lock();
m_numFailed++; m_results.push_back(result);
m_mutex->unlock();
return;
} else {
Log(EDebug, "t-test accepts.");
}
}
} else if (scene->getTestType() == Scene::ERelativeError) {
for (size_t i=0; i scene->getTestThreshold()) {
Log(EWarn, "Relativ error threshold EXCEEDED!");
result.message = formatString("Relative error threshold EXCEEDED: result=%f (ref=%f), diff=%e, relerr=%f", actual[i].value, ref[i], actual[i].value - ref[i], relerr);
m_mutex->lock();
m_numFailed++; m_results.push_back(result);
m_mutex->unlock();
return;
} else {
Log(EDebug, "Relative error accepted.");
}
}
} else if (scene->getTestType() == Scene::ENone) {
Log(EError, "No test type specified, don't know what to do!");
} else {
Log(EError, "Unknown test type!");
}
result.success = true;
m_mutex->lock();
m_numSucceeded++;
m_results.push_back(result);
m_mutex->unlock();
}
void TestSupervisor::printSummary() const {
m_mutex->lock();
Log(EInfo, "Ran %i/%i testcases, %i succeeded, %i failed.", (int) (m_numFailed+m_numSucceeded), (int) m_total,
(int) m_numSucceeded, (int) m_numFailed);
for (size_t i=0; iunlock();
}
MTS_IMPLEMENT_CLASS(TestSupervisor, false, Object)
MTS_IMPLEMENT_CLASS(TestCase, false, Utility)
MTS_NAMESPACE_END