/* This file is part of Mitsuba, a physically based rendering system. Copyright (c) 2007-2010 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 MTS_NAMESPACE_BEGIN SceneHandler::SceneHandler() : m_isIncludedFile(false) { m_pluginManager = PluginManager::getInstance(); #if !defined(WIN32) setlocale(LC_NUMERIC, "C"); #endif } SceneHandler::SceneHandler(const std::map ¶ms, bool isIncludedFile) : m_params(params), m_isIncludedFile(isIncludedFile) { m_pluginManager = PluginManager::getInstance(); } SceneHandler::~SceneHandler() { } // ----------------------------------------------------------------------- // Implementation of the SAX DocumentHandler interface // ----------------------------------------------------------------------- void SceneHandler::startDocument() { m_objects.clear(); } void SceneHandler::endDocument() { SAssert(m_scene != NULL); } void SceneHandler::characters(const XMLCh* const name, const unsigned int length) { } static Float parseFloat(const std::string &name, const std::string &str, Float defVal = -1) { char *end_ptr = NULL; if (str == "") { if (defVal == -1) SLog(EError, "Missing floating point value (in <%s>)", name.c_str()); return defVal; } Float result = (Float) std::strtod(str.c_str(), &end_ptr); if (*end_ptr != '\0') SLog(EError, "Invalid floating point value specified (in <%s>)", name.c_str()); return result; } void SceneHandler::startElement(const XMLCh* const xmlName, AttributeList &xmlAttributes) { std::string name = transcode(xmlName); ParseContext context((name == "scene") ? NULL : &m_context.top()); /* Convert attributes to ISO-8859-1 */ for (unsigned int i=0; i 0 && attrValue.find('$') != attrValue.npos) { for (std::map::const_iterator it = m_params.begin(); it != m_params.end(); ++it) { std::string::size_type pos = 0; std::string searchString = "$" + (*it).first; while ((pos = attrValue.find(searchString, pos)) != std::string::npos) { attrValue.replace(pos, searchString.size(), (*it).second); ++pos; } } if (attrValue.find('$') != attrValue.npos) SLog(EError, "The scene referenced an undefined parameter: \"%s\"", attrValue.c_str()); } context.attributes[transcode(xmlAttributes.getName(i))] = attrValue; } if (name == "transform") m_transform = Transform(); m_context.push(context); } void SceneHandler::endElement(const XMLCh* const xmlName) { std::string name = transcode(xmlName); ParseContext &context = m_context.top(); std::string type = toLowerCase(context.attributes["type"]); context.properties.setPluginName(type); if (context.attributes.find("id") != context.attributes.end()) context.properties.setID(context.attributes["id"]); ref object = NULL; /* Construct configurable objects */ if (name == "scene") { object = m_scene = new Scene(context.properties); } else if (name == "shape") { object = static_cast (m_pluginManager->createObject( Shape::m_theClass, context.properties)); } else if (name == "sampler") { object = static_cast (m_pluginManager->createObject( Sampler::m_theClass, context.properties)); } else if (name == "film") { object = static_cast (m_pluginManager->createObject( Film::m_theClass, context.properties)); } else if (name == "integrator") { object = static_cast (m_pluginManager->createObject( Integrator::m_theClass, context.properties)); } else if (name == "texture") { object = static_cast (m_pluginManager->createObject( Texture::m_theClass, context.properties)); } else if (name == "camera") { object = static_cast (m_pluginManager->createObject( Camera::m_theClass, context.properties)); } else if (name == "subsurface") { object = static_cast (m_pluginManager->createObject( Subsurface::m_theClass, context.properties)); } else if (name == "luminaire") { object = static_cast (m_pluginManager->createObject( Luminaire::m_theClass, context.properties)); } else if (name == "medium") { object = static_cast (m_pluginManager->createObject( Medium::m_theClass, context.properties)); } else if (name == "volume") { object = static_cast (m_pluginManager->createObject( VolumeDataSource::m_theClass, context.properties)); } else if (name == "phase") { object = static_cast (m_pluginManager->createObject( PhaseFunction::m_theClass, context.properties)); } else if (name == "bsdf") { object = static_cast (m_pluginManager->createObject( BSDF::m_theClass, context.properties)); } else if (name == "rfilter") { object = static_cast (m_pluginManager->createObject( ReconstructionFilter::m_theClass, context.properties)); } else if (name == "ref") { std::string id = context.attributes["id"]; if (m_objects.find(id) == m_objects.end()) SLog(EError, "Referenced object '%s' not found!", id.c_str()); object = m_objects[id]; /* Construct properties */ } else if (name == "integer") { char *end_ptr = NULL; #ifdef WIN32 int64_t i = _strtoi64(context.attributes["value"].c_str(), &end_ptr, 10); #else int64_t i = strtoll(context.attributes["value"].c_str(), &end_ptr, 10); #endif if (*end_ptr != '\0') SLog(EError, "Invalid integer value specified (in <%s>)", context.attributes["name"].c_str()); context.parent->properties.setLong(context.attributes["name"], i); } else if (name == "float") { Float f = parseFloat(name, context.attributes["value"]); context.parent->properties.setFloat(context.attributes["name"], f); } else if (name == "boolean") { bool value = false; if (context.attributes["value"] == "true") value = true; context.parent->properties.setBoolean(context.attributes["name"], value); } else if (name == "string") { context.parent->properties.setString(context.attributes["name"], context.attributes["value"]); } else if (name == "translate") { Float x = parseFloat(name, context.attributes["x"], 0); Float y = parseFloat(name, context.attributes["y"], 0); Float z = parseFloat(name, context.attributes["z"], 0); Transform translate = Transform::translate(Vector(x, y, z)); m_transform = translate * m_transform; } else if (name == "rotate") { Float x = parseFloat(name, context.attributes["x"], 0); Float y = parseFloat(name, context.attributes["y"], 0); Float z = parseFloat(name, context.attributes["z"], 0); Float angle = parseFloat(name, context.attributes["angle"]); Transform rotate = Transform::rotate(Vector(x, y, z), angle); m_transform = rotate * m_transform; } else if (name == "lookAt") { Float ox = parseFloat(name, context.attributes["ox"]); Float oy = parseFloat(name, context.attributes["oy"]); Float oz = parseFloat(name, context.attributes["oz"]); Float tx = parseFloat(name, context.attributes["tx"]); Float ty = parseFloat(name, context.attributes["ty"]); Float tz = parseFloat(name, context.attributes["tz"]); Float ux = parseFloat(name, context.attributes["ux"], 0); Float uy = parseFloat(name, context.attributes["uy"], 0); Float uz = parseFloat(name, context.attributes["uz"], 0); Point o(ox, oy, oz), t(tx, ty, tz); Vector u(ux, uy, uz); if (u.lengthSquared() == 0) { /* If 'up' was not specified, use an arbitrary axis */ Vector v; coordinateSystem(normalize(t-o), u, v); } Transform lookAt = Transform::lookAt(o, t, u); m_transform = lookAt * m_transform; } else if (name == "scale") { Float x = parseFloat(name, context.attributes["x"], 1); Float y = parseFloat(name, context.attributes["y"], 1); Float z = parseFloat(name, context.attributes["z"], 1); Transform scale = Transform::scale(Vector(x, y, z)); m_transform = scale * m_transform; } else if (name == "matrix") { std::vector tokens = tokenize( context.attributes["value"], ", "); if (tokens.size() != 16) SLog(EError, "Invalid matrix specified"); int index = 0; Float tmp[4][4]; for (int i=0; i<4; ++i) for (int j=0; j<4; ++j) tmp[i][j] = parseFloat(name, tokens[index++]); m_transform = Transform(new Matrix4x4(tmp)) * m_transform; } else if (name == "point" || name == "vector") { Float x = parseFloat(name, context.attributes["x"]); Float y = parseFloat(name, context.attributes["y"]); Float z = parseFloat(name, context.attributes["z"]); context.parent->properties.setPoint(context.attributes["name"], Point(x, y, z)); } else if (name == "rgb") { std::string valueStr = context.attributes["value"]; std::vector tokens = tokenize(valueStr, ", "); Float value[3]; if (tokens.size() == 1 && tokens[0].length() == 7 && tokens[0][0] == '#') { char *end_ptr = NULL; /* Parse HTML-style hexadecimal colors */ int encoded = strtol(tokens[0].c_str()+1, &end_ptr, 16); if (*end_ptr != '\0') SLog(EError, "Invalid rgb value specified (in <%s>)", context.attributes["name"].c_str()); value[0] = ((encoded & 0xFF0000) >> 16) / 255.0f; value[1] = ((encoded & 0x00FF00) >> 8) / 255.0f; value[2] = (encoded & 0x0000FF) / 255.0f; } else if (tokens.size() == 1) { value[0] = value[1] = value[2] = parseFloat(name, tokens[0]); } else if (tokens.size() == 3) { for (int i=0; i<3; i++) value[i] = parseFloat(name, tokens[i]); } else { value[0] = value[1] = value[2] = 0; // avoid warning SLog(EError, "Invalid RGB value specified"); } Spectrum specValue; specValue.fromLinearRGB(value[0], value[1], value[2]); context.parent->properties.setSpectrum(context.attributes["name"], specValue); } else if (name == "srgb") { std::string valueStr = context.attributes["value"]; std::vector tokens = tokenize(valueStr, ", "); Float value[3]; if (tokens.size() == 1 && tokens[0].length() == 7 && tokens[0][0] == '#') { char *end_ptr = NULL; /* Parse HTML-style hexadecimal colors */ int encoded = strtol(tokens[0].c_str()+1, &end_ptr, 16); if (*end_ptr != '\0') SLog(EError, "Invalid sRGB value specified (in <%s>)", context.attributes["name"].c_str()); value[0] = ((encoded & 0xFF0000) >> 16) / 255.0f; value[1] = ((encoded & 0x00FF00) >> 8) / 255.0f; value[2] = (encoded & 0x0000FF) / 255.0f; } else if (tokens.size() == 1) { value[0] = value[1] = value[2] = parseFloat(name, tokens[0]); } else if (tokens.size() == 3) { for (int i=0; i<3; i++) value[i] = parseFloat(name, tokens[i]); } else { value[0] = value[1] = value[2] = 0; // avoid warning SLog(EError, "Invalid sRGB value specified"); } Spectrum specValue; specValue.fromSRGB(value[0], value[1], value[2]); context.parent->properties.setSpectrum(context.attributes["name"], specValue); } else if (name == "blackbody") { Float temperature = parseFloat(name, context.attributes["temperature"]); BlackBodySpectrum *spec = new BlackBodySpectrum(temperature); context.parent->properties.setSpectrum(context.attributes["name"], Spectrum(spec)); delete spec; } else if (name == "spectrum") { std::vector tokens = tokenize( context.attributes["value"], ", "); Float value[SPECTRUM_SAMPLES]; if (tokens.size() == 1) { value[0] = parseFloat(name, tokens[0]); context.parent->properties.setSpectrum(context.attributes["name"], Spectrum(value[0])); } else { if (tokens[0].find(':') != std::string::npos) { InterpolatedSpectrum spec(tokens.size()); /* Wavelength -> Value mapping */ for (size_t i=0; i tokens2 = tokenize(tokens[i], ":"); if (tokens2.size() != 2) SLog(EError, "Invalid spectrum->value mapping specified"); Float wavelength = parseFloat(name, tokens2[0]); Float value = parseFloat(name, tokens2[1]); spec.appendSample(wavelength, value); } context.parent->properties.setSpectrum(context.attributes["name"], Spectrum(&spec)); } else { if (tokens.size() != SPECTRUM_SAMPLES) SLog(EError, "Invalid spectrum value specified (incorrect length)"); for (int i=0; iproperties.setSpectrum(context.attributes["name"], Spectrum(value)); } } } else if (name == "transform") { context.parent->properties.setTransform(context.attributes["name"], m_transform); /* Do nothing */ } else if (name == "include") { SAXParser* parser = new SAXParser(); FileResolver *resolver = FileResolver::getInstance(); std::string schemaPath = resolver->resolveAbsolute("schema/scene.xsd"); /* Check against the 'scene.xsd' XML Schema */ parser->setDoSchema(true); parser->setValidationSchemaFullChecking(true); parser->setValidationScheme(SAXParser::Val_Always); parser->setExternalNoNamespaceSchemaLocation(schemaPath.c_str()); /* Set the handler and start parsing */ SceneHandler *handler = new SceneHandler(m_params, true); parser->setDoNamespaces(true); parser->setDocumentHandler(handler); parser->setErrorHandler(handler); std::string filename = resolver->resolve(context.attributes["filename"]); SLog(EInfo, "Parsing included file \"%s\" ..", filename.c_str()); parser->parse(filename.c_str()); object = handler->getScene(); delete parser; delete handler; } else { SLog(EError, "Unhandled tag \"%s\" encountered!", name.c_str()); } if (object != NULL) { std::string id = context.attributes["id"]; std::string nodeName = context.attributes["name"]; if (id != "" && name != "ref") { if (m_objects.find(id) != m_objects.end()) SLog(EError, "Duplicate ID '%s' used in scene description!", id.c_str()); m_objects[id] = object; } /* If the object has a parent, add it to the parent's children list */ if (context.parent != NULL) { object->incRef(); context.parent->children.push_back( std::pair(nodeName, object)); } /* If the object has children, append them */ for (std::vector > ::iterator it = context.children.begin(); it != context.children.end(); ++it) { object->addChild((*it).first, (*it).second); (*it).second->setParent(object); (*it).second->decRef(); } /* Don't configure a scene object if it is from an included file */ if (name != "include" && (!m_isIncludedFile || !object->getClass()->derivesFrom(Scene::m_theClass))) object->configure(); } /* Warn about unqueried properties */ std::vector unq = context.properties.getUnqueried(); for (unsigned int i=0; i