// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "nel/ligo/ligo_config.h" #include "nel/ligo/primitive.h" #include "nel/misc/config_file.h" #include "nel/misc/i_xml.h" #include "nel/misc/path.h" #include "nel/misc/file.h" #include using namespace std; using namespace NLMISC; namespace NLLIGO { // *************************************************************************** CLigoConfig::CLigoConfig() : _DynamicAliasBitCount(32) { } // *************************************************************************** bool CLigoConfig::readConfigFile (const char *fileName, bool parsePrimitiveComboContent) { // The CF CConfigFile cf; // Load and parse the file cf.load (fileName); // Read the parameters CConfigFile::CVar &cell_size = cf.getVar ("cell_size"); CellSize = cell_size.asFloat (); CConfigFile::CVar &snap = cf.getVar ("snap"); Snap = snap.asFloat (); CConfigFile::CVar &snapShot = cf.getVar ("zone_snapeshot_res"); ZoneSnapShotRes = (uint)snapShot.asInt (); CConfigFile::CVar &primitiveClassFilename = cf.getVar ("primitive_class_filename"); PrimitiveClassFilename= primitiveClassFilename.asString (); // Clear the previous classes _Contexts.clear(); _Contexts.push_back ("default"); _PrimitiveClasses.clear(); _PrimitiveConfigurations.clear(); // Read the primitive class name if (!PrimitiveClassFilename.empty()) { return readPrimitiveClass (PrimitiveClassFilename.c_str(), parsePrimitiveComboContent); } return true; } // *************************************************************************** bool CLigoConfig::readPrimitiveClass (const char *_fileName, bool parsePrimitiveComboContent) { // File exist ? string filename = _fileName; filename = CPath::lookup (_fileName, false, false, false); if (filename.empty()) filename = _fileName; // The context strings set contextStrings; // Read the document CIFile file; if (file.open (filename)) { try { // XML stream CIXml xml; xml.init (file); // Get the root node xmlNodePtr root = xml.getRootNode (); nlassert (root); // Check the header if (strcmp ((const char*)root->name, "NEL_LIGO_PRIMITIVE_CLASS") == 0) { // ALIAS_DYNAMIC_BITS xmlNodePtr aliasBits = CIXml::getFirstChildNode (root, "ALIAS_DYNAMIC_BITS"); if (aliasBits) { string bits; if (getPropertyString (bits, filename.c_str(), aliasBits, "BIT_COUNT")) { uint32 uBits; NLMISC::fromString(bits, uBits); _DynamicAliasBitCount = std::min((uint32)32, uBits); } else return false; } else { // by default, set the size of dynamic alias to 32 bits _DynamicAliasBitCount = 32; } // ALIAS_STATIC_FILE_ID xmlNodePtr indexFileNameNode = CIXml::getFirstChildNode (root, "ALIAS_STATIC_FILE_ID"); if (indexFileNameNode) { string indexFileName; if (getPropertyString (indexFileName, filename.c_str(), indexFileNameNode, "FILE_NAME")) { if (CPath::lookup(indexFileName, false, false, true).empty()) { // try to append the class file path indexFileName = CFile::getPath(_fileName)+indexFileName; } // load the configuration file reloadIndexFile(indexFileName); } else nlwarning("Can't find XML element , no file index available for alias" ); } else { // by default, set the size of dynamic alias to 32 bits _DynamicAliasBitCount = 32; } // Get the first primitive description xmlNodePtr primitive = CIXml::getFirstChildNode (root, "PRIMITIVE"); if (primitive) { do { // Get the primitive name std::string name; if (getPropertyString (name, filename.c_str(), primitive, "CLASS_NAME")) { // Add the primitive pair::iterator, bool> insertResult = _PrimitiveClasses.insert (std::map::value_type (name, CPrimitiveClass ())); if (insertResult.second) { if (!insertResult.first->second.read (primitive, filename.c_str(), name.c_str (), contextStrings, _ContextFilesLookup, *this, parsePrimitiveComboContent)) return false; } else { syntaxError (filename.c_str(), root, "Class (%s) already defined", name.c_str ()); } } else return false; primitive = CIXml::getNextChildNode (primitive, "PRIMITIVE"); } while (primitive); } // Add the context strings { set::iterator ite = contextStrings.begin (); while (ite != contextStrings.end ()) { if (*ite != "default") _Contexts.push_back (*ite); ite++; } } // Get the first primitive configuration _PrimitiveConfigurations.reserve (_PrimitiveConfigurations.size()+CIXml::countChildren (root, "CONFIGURATION")); xmlNodePtr configuration = CIXml::getFirstChildNode (root, "CONFIGURATION"); if (configuration) { do { // Get the configuration name std::string name; if (getPropertyString (name, filename.c_str(), configuration, "NAME")) { // Add the configuration _PrimitiveConfigurations.resize (_PrimitiveConfigurations.size()+1); if (!_PrimitiveConfigurations.back().read (configuration, filename.c_str(), name.c_str (), *this)) return false; } else return false; configuration = CIXml::getNextChildNode (configuration, "CONFIGURATION"); } while (configuration); } // Ok return true; } else { syntaxError (filename.c_str(), root, "Wrong root node, should be NEL_LIGO_PRIMITIVE_CLASS"); } } catch (Exception &e) { errorMessage ("File read error (%s):%s", filename.c_str(), e.what ()); } } else { errorMessage ("Can't open the file %s for reading.", filename.c_str()); } return false; } // *************************************************************************** bool CLigoConfig::reloadIndexFile(const std::string &indexFileName) { if (_IndexFileName.empty() && indexFileName.empty()) { nlwarning("CLigoConfig::reloadIndexFile: no file name specified and index file not previously loaded, can't load anything"); return false; } if (!_IndexFileName.empty() && !indexFileName.empty() && _IndexFileName != indexFileName) { nlwarning("CLigoConfig::reloadIndexFile: index file already loaded as '%s', can't load another file '%s'!", _IndexFileName.c_str(), indexFileName.c_str()); return false; } if (_IndexFileName.empty()) _IndexFileName = indexFileName; // load the configuration file CConfigFile cf; string pathName = CPath::lookup(_IndexFileName, false); if (pathName.empty()) { nlwarning("Can't find index file '%s' in search path, no file index available for alias", indexFileName.c_str()); return false; } cf.load(pathName); // get the variable CConfigFile::CVar *files = cf.getVarPtr("Files"); if (files != NULL) { for (uint i=0; isize()/2; ++i) { string fileName; uint32 index; fileName = files->asString(i*2); index = files->asInt(i*2+1); if (isFileStaticAliasMapped(fileName)) { // check that the mapping as not changed if (getFileStaticAliasMapping(fileName) != index) { nlwarning("CLigoConfig::reloadIndexFile: the mapping for the file '%s' as changed from %u to %u in the config file, the change is ignored", fileName.c_str(), index, getFileStaticAliasMapping(fileName)); } } else { registerFileToStaticAliasTranslation(fileName, index); } } } return true; } // *************************************************************************** NLMISC::CRGBA CLigoConfig::getPrimitiveColor (const NLLIGO::IPrimitive &primitive) { // Get the class string className; if (primitive.getPropertyByName ("class", className)) { // Get the class std::map::iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { return ite->second.Color; } } return DEFAULT_PRIMITIVE_COLOR; } // *************************************************************************** bool CLigoConfig::isPrimitiveLinked (const NLLIGO::IPrimitive &primitive) { // Get the class string className; if (primitive.getPropertyByName ("class", className)) { // Get the class std::map::iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { return ite->second.LinkBrothers; } } return false; } // *************************************************************************** const NLLIGO::IPrimitive *CLigoConfig::getLinkedPrimitive (const NLLIGO::IPrimitive &primitive) const { // Get the parent const IPrimitive *parent = primitive.getParent (); if (parent) { uint childId; if (parent->getChildId (childId, &primitive)) { // Test the next primitive // Get the primitive class string className; if (primitive.getPropertyByName ("class", className)) { // Get the class std::map::const_iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { if (ite->second.LinkBrothers) { // Add the next child const IPrimitive *brother; if (parent->getChild (brother, childId+1)) return brother; } } } } } return NULL; } // *************************************************************************** const NLLIGO::IPrimitive *CLigoConfig::getPreviousLinkedPrimitive (const NLLIGO::IPrimitive &primitive) const { // Get the parent const IPrimitive *parent = primitive.getParent (); if (parent) { uint childId; if (parent->getChildId (childId, &primitive)) { // Test the previous primitive if (childId > 0) { const IPrimitive *brother; if (parent->getChild (brother, childId-1) && brother) { // Get the primitive class string className; if (brother->getPropertyByName ("class", className)) { // Get the class std::map::const_iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { if (ite->second.LinkBrothers) { // Return the previous child return brother; } } } } } } } return NULL; } // *************************************************************************** bool CLigoConfig::isPrimitiveDeletable (const NLLIGO::IPrimitive &primitive) { // If it is a static child, it can't be deleted. if (isStaticChild (primitive)) return false; // Get the class string className; if (primitive.getPropertyByName ("class", className)) { // Get the class std::map::iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { return ite->second.Deletable; } } return false; } // *************************************************************************** bool CLigoConfig::canBeChild (const NLLIGO::IPrimitive &child, const NLLIGO::IPrimitive &parent) { // Get the child class string childClassName; if (child.getPropertyByName ("class", childClassName)) { // Get the parent class const CPrimitiveClass *parentClass = getPrimitiveClass (parent); if (parentClass) { // Search for the child class uint i; for (i=0; iDynamicChildren.size (); i++) { // The same ? if (parentClass->DynamicChildren[i].ClassName == childClassName) break; } if (iDynamicChildren.size ()) return true; for (i=0; iGeneratedChildren.size (); i++) { // The same ? if (parentClass->GeneratedChildren[i].ClassName == childClassName) break; } return (iGeneratedChildren.size ()); } else return true; } else { // Only if it is a root node or parent class doesn't exist string parentClassName; return ( (parent.getParent () == NULL) || (!parent.getPropertyByName ("class", parentClassName) ) ); } } // *************************************************************************** bool CLigoConfig::canBeRoot (const NLLIGO::IPrimitive &child) { // Get the child class string childClassName; if (child.getPropertyByName ("class", childClassName)) { // Get the parent class const CPrimitiveClass *parentClass = getPrimitiveClass ("root"); if (parentClass) { // Search for the child class uint i; for (i=0; iDynamicChildren.size (); i++) { // The same ? if (parentClass->DynamicChildren[i].ClassName == childClassName) break; } return (iDynamicChildren.size ()); } else return true; } else { // Root class doesn't exist return ( !getPrimitiveClass ("root") ); } } // *************************************************************************** bool CLigoConfig::getPropertyString (std::string &result, const char *filename, xmlNodePtr xmlNode, const char *propName) { // Call the CIXml version if (!CIXml::getPropertyString (result, xmlNode, propName)) { // Output a formated error syntaxError (filename, xmlNode, "Missing XML node property (%s)", propName); return false; } return true; } // *************************************************************************** void CLigoConfig::syntaxError (const char *filename, xmlNodePtr xmlNode, const char *format, ...) { va_list args; va_start( args, format ); char buffer[1024]; vsnprintf( buffer, 1024, format, args ); va_end( args ); errorMessage ("(%s), node (%s), line (%d) :\n%s", filename, xmlNode->name, (ptrdiff_t)xmlNode->content, buffer); } // *************************************************************************** void CLigoConfig::errorMessage (const char *format, ... ) { // Make a buffer string va_list args; va_start( args, format ); char buffer[1024]; vsnprintf( buffer, 1024, format, args ); va_end( args ); nlwarning (buffer); } // *************************************************************************** const std::vector &CLigoConfig::getContextString () const { return _Contexts; } // *************************************************************************** const CPrimitiveClass *CLigoConfig::getPrimitiveClass (const IPrimitive &primitive) const { const CPrimitiveClass *primClass = NULL; // Get property class string className; if (primitive.getPropertyByName ("class", className)) { std::map::const_iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { primClass = &(ite->second); } } // Not found ? if (!primClass) { // Root ? if (!primitive.getParent ()) { std::map::const_iterator ite = _PrimitiveClasses.find ("root"); if (ite != _PrimitiveClasses.end ()) { primClass = &(ite->second); } } } return primClass; } // *************************************************************************** const CPrimitiveClass *CLigoConfig::getPrimitiveClass (const char *className) const { std::map::const_iterator ite = _PrimitiveClasses.find (className); if (ite != _PrimitiveClasses.end ()) { return &(ite->second); } return NULL; } // *************************************************************************** void CLigoConfig::resetPrimitiveConfiguration () { _PrimitiveConfigurations.clear (); } // *************************************************************************** bool CLigoConfig::isStaticChild (const NLLIGO::IPrimitive &primitive) { // Has a parent ? const IPrimitive *parent = primitive.getParent (); if (parent) { // Get the classes const CPrimitiveClass *parentClass = getPrimitiveClass (*parent); string className; string name; if (parentClass && primitive.getPropertyByName ("class", className) && primitive.getPropertyByName ("name", name)) { // Does it belong to the static children ? uint i; for (i=0; iStaticChildren.size(); i++) { if (parentClass->StaticChildren[i].Name == name && parentClass->StaticChildren[i].ClassName == className) { // Found return true; } } } } return false; } // *************************************************************************** /// Get the dynamic bit size for alias uint32 CLigoConfig::getDynamicAliasSize() const { return _DynamicAliasBitCount; } // *************************************************************************** /// Get the dynamic bit mask for alias uint32 CLigoConfig::getDynamicAliasMask() const { // this 'strange' test because VC fail to generate a correct shift if // _DynamicAliasBitCount is 32 bits. // The generated code lead to no shift at all and return 0 instead of 0xffffffff if (_DynamicAliasBitCount >= 32) return 0xffffffff; else return (1U<<_DynamicAliasBitCount)-1; } // *************************************************************************** /// Get the static bit size for alias uint32 CLigoConfig::getStaticAliasSize() const { return 32-_DynamicAliasBitCount; } // *************************************************************************** /// Get the static bit mask for alias uint32 CLigoConfig::getStaticAliasMask() const { // the opposite of the dynamic mask return ~getDynamicAliasMask(); } // *************************************************************************** /// Build an alias given a static and dynamic part uint32 CLigoConfig::buildAlias(uint32 staticPart, uint32 dynamicPart, bool warnIfOverload) const { if (warnIfOverload) { if (staticPart != (staticPart & (getStaticAliasMask()>>getDynamicAliasSize()))) { nlwarning("CLigoConfig::buildAlias: staticPart 0x%x is outside the mask 0x%x", staticPart, getStaticAliasMask()>>getDynamicAliasSize()); } if (dynamicPart != (dynamicPart & getDynamicAliasMask())) { nlwarning("CLigoConfig::buildAlias: dynamicPart 0x%x is outside the mask 0x%x", dynamicPart, getDynamicAliasMask()); } } return dynamicPart | (staticPart << _DynamicAliasBitCount); } // *************************************************************************** void CLigoConfig::registerFileToStaticAliasTranslation(const std::string &fileName, uint32 staticPart) { // check the existing mapping std::map::iterator first(_StaticAliasFileMapping.begin()), last(_StaticAliasFileMapping.end()); for (; first != last; ++first) { if (first->second == staticPart) { nlassertex(false, ("While registering static alias %u to file '%s', the alias is already assigned to file '%s'", staticPart, fileName.c_str(), first->first.c_str())); } } if ((staticPart<::const_iterator first(_StaticAliasFileMapping.begin()), last(_StaticAliasFileMapping.end()); for (; first != last; ++first) { if (first->second == staticAlias) return first->first; } static string emptyString; return emptyString; } // *************************************************************************** uint32 CLigoConfig::getFileStaticAliasMapping(const std::string &fileName) const { std::map::const_iterator it(_StaticAliasFileMapping.find(fileName)); if (it != _StaticAliasFileMapping.end()) { return it->second; } else // no mapping defined. return 0; } // *************************************************************************** bool CLigoConfig::isFileStaticAliasMapped(const std::string &fileName) const { std::map::const_iterator it(_StaticAliasFileMapping.find(fileName)); if (it != _StaticAliasFileMapping.end()) { return true; } else // no mapping defined. return false; } std::string CLigoConfig::aliasToString(uint32 fullAlias) { uint32 staticPart; uint32 dynPart; staticPart = (fullAlias & getStaticAliasMask())>>getDynamicAliasSize(); dynPart = fullAlias & getDynamicAliasMask(); return toString("(A:%u:%u)", staticPart, dynPart); } uint32 CLigoConfig::aliasFromString(std::string fullAlias) { uint32 staticPart; uint32 dynPart; sscanf(fullAlias.c_str(), "(A:%u:%u)", &staticPart, &dynPart); return ((staticPart<::iterator first(_StaticAliasFileMapping.begin()), last(_StaticAliasFileMapping.end()); for ( ; first != last; ++first) { first->second = first->second << diff; } _DynamicAliasBitCount = newDynamicAliasBitCount; } }