// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/> // 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 <http://www.gnu.org/licenses/>. #include "parser.h" #include "cpp_output.h" #include "templatizer.h" #include "parser_rules.h" #include <nel/misc/path.h> #include <nel/misc/sha1.h> using namespace std; using namespace NLMISC; //using namespace RY_PDS; bool GenerateCpp = true; bool GenerateXmlDescription = true; bool VerboseMode = false; bool GenerateDebugMessages = false; bool GenerateHAuto = false; bool GenerateOnlyLogs = false; CTokenizer Tokenizer; CTemplatizer Templatizer; CHashKey HashKey; string CurrentEnum; CEnumNode *CurrentEnumNode = NULL; string DbDescriptionVersion("0.0"); string formatDescription(const string &str) { string result; uint pos = 0; while (pos < str.size() && (str[pos] == ' ' || str[pos] == '\t' || str[pos] == '\r' || str[pos] == '\n')) ++pos; bool first = true; while (pos < str.size()) { if (!first) result += "\n"; first = false; result += "* "; while (pos < str.size() && str[pos] != '\n') result += str[pos++]; while (pos < str.size() && (str[pos] == ' ' || str[pos] == '\t' || str[pos] == '\r' || str[pos] == '\n')) ++pos; } return result; } string strReplace(string str, const string &search, const string &replace) { uint pos; while ((pos = str.find(search)) != string::npos) str.replace(pos, search.size(), replace); return str; } string xmlSpecialChars(string str) { str = strReplace(str, "&", "&"); str = strReplace(str, "<", "<"); str = strReplace(str, ">", ">"); str = strReplace(str, "\"", """); str = strReplace(str, "'", "'"); return str; } string getFullStdPathNoExt(const string &path) { string dir = strlwr(NLMISC::CFile::getPath(path)); string file = strlwr(NLMISC::CFile::getFilenameWithoutExtension(path)); return dir.empty() ? file : NLMISC::CPath::standardizePath(dir)+file; } string getFullStdPath(const string &path) { string dir = strlwr(NLMISC::CFile::getPath(path)); string file = strlwr(NLMISC::CFile::getFilename(path)); return dir.empty() ? file : NLMISC::CPath::standardizePath(dir)+file; } string appendArg(const std::string& firstArgs, const std::string& nextArg) { if (firstArgs.empty()) return nextArg; return firstArgs + ", " + nextArg; } /* * Start parsing */ CParseNode *parse(const string &file) { HashKey = getSHA1(file); Tokenizer.readFile(file); if (Tokenizer.tokenize()) return parseMain(Tokenizer); return NULL; } /* * Execute nodes */ // DB Node bool CDbNode::prolog() { addTypeNode("CSheetId", "NLMISC::CSheetId", "NLMISC::CSheetId::Unknown"); addTypeNode("CEntityId", "NLMISC::CEntityId", "NLMISC::CEntityId::Unknown"); addTypeNode("double", "double", "0.0"); addTypeNode("float", "float", "0.0f"); addTypeNode("sint64", "sint64", "(sint64)0"); addTypeNode("uint64", "uint64", "(uint64)0"); addTypeNode("sint32", "sint32", "0"); addTypeNode("uint32", "uint32", "0"); addTypeNode("sint16", "sint16", "0"); addTypeNode("uint16", "uint16", "0"); addTypeNode("sint8", "sint8", "0"); addTypeNode("uint8", "uint8", "0"); addTypeNode("ucchar", "ucchar", "0"); addTypeNode("char", "char", "0"); addTypeNode("bool", "bool", "false"); return true; } bool inlineAccessors = false; string getFunctionPrefix = "get"; string setFunctionPrefix = "set"; string newFunction = "addTo"; string deleteFunction = "deleteFrom"; string indexVariable = "__i"; string valueVariable = "__v"; string keyVariable = "__k"; string objectVariable = "__o"; bool inlineUserInitDefaultCode = false; string userInitFunction = "init"; string userReleaseFunction = "release"; bool inlineStaticPublic = false; string staticCreateFunction = "create"; string staticRemoveFunction = "remove"; string staticSetUserFactoryFunction = "setFactory"; string staticLoadFunction = "load"; string staticUnloadFunction = "unload"; string staticSetLoadCbFunction = "setLoadCallback"; string staticGetFunction = "get"; string staticCastFunction = "cast"; string staticConstCastFunction = "cast"; string staticBeginFunction = "begin"; string staticEndFunction = "end"; bool inlineInternal = false; string initFunction = "pds__init"; string destroyFunction = "pds__destroy"; string registerFunction = "pds__register"; string registerAttributesFunction = "pds__registerAttributes"; string unregisterFunction = "pds__unregister"; string unregisterAttributesFunction = "pds__unregisterAttributes"; string fetchFunction = "pds__fetch"; string setParentFunction = "pds__setParent"; string setUnnotifiedParentFunction = "pds__setParentUnnotified"; string getTableFunction = "pds__getTable"; string unlinkFunction = "pds__unlink"; string notifyInitFunction = "pds__notifyInit"; string notifyReleaseFunction = "pds__notifyRelease"; string clearFunction = "clear"; string storeFunction = "store"; string applyFunction = "apply"; string declareTokensFunction = "pds__declareTokens"; bool inlineStaticInternal = false; string staticInitFactoryFunction = "pds_static__setFactory"; string staticFactoryFunction = "pds_static__factory"; string staticFetchFunction = "pds_static__fetch"; string staticInitFunction = "pds_static__init"; string staticNotifyLoadFailure = "pds_static__notifyFailure"; string staticLoadCbAttribute = "__pds__LoadCallback"; bool inlineLog = false; string logStartFunction = "pds__startLog"; string logStopFunction = "pds__stopLog"; string PDSNamespace = "RY_PDS"; string CPDSLibName = PDSNamespace+"::CPDSLib"; string objectIndexName = PDSNamespace+"::CObjectIndex"; string nullIndexName = PDSNamespace+"::CObjectIndex::null()"; string TTableIndexName = PDSNamespace+"::TTableIndex"; string TRowIndexName = PDSNamespace+"::TRowIndex"; string TColumnIndexName = PDSNamespace+"::TColumnIndex"; string INVALID_TABLE_INDEXName = PDSNamespace+"::INVALID_TABLE_INDEX"; string INVALID_ROW_INDEXName = PDSNamespace+"::INVALID_ROW_INDEX"; string indexAllocatorName = PDSNamespace+"::CIndexAllocator"; string pdBaseDataName = PDSNamespace+"::IPDBaseData"; //string pdBaseInheritDataName = PDSNamespace+"::IPDBaseInheritData"; string CPDataName = PDSNamespace+"::CPData"; string TPDFactoryName = PDSNamespace+"::TPDFactory"; string TPDFetchName = PDSNamespace+"::TPDFetch"; string TPDFetchFailureName = PDSNamespace+"::TPDFetchFailure"; bool inlineLogFunctions = false; string pdslibFunc(const std::string& func) { //return CPDSLibName+"::"+func; return "PDSLib."+func; } bool CDbNode::epilog() { uint i; Env = Templatizer.RootEnv; setEnv("db", Name); string fullfile = getFullStdPathNoExt(MainFile.empty() ? Name : MainFile); string filename = NLMISC::strlwr(NLMISC::CFile::getFilenameWithoutExtension(fullfile)); setEnv("filename", filename); setEnv("fullfilename", fullfile); setEnv("headerfilename", strReplace(strupr(filename+".h"), ".", "_")); DbHpp.setFileHeader(filename+".h", "Initialisation of the "+Name+" database, declarations\n"+Description); DbCpp.setFileHeader(filename+".cpp", "Initialisation of the "+Name+" database, implementation\n"+Description); DbHppInline.setFileHeader(filename+"_inline.h", "Initialisation of the "+Name+" database, inline implementation\n"+Description); DbSummary << "\n\n"; DbSummary << "Summary of " << Name << " database classes of database\n"; DbSummary << "-----------------------------------------------------------------------------\n\n"; DbSummary << "This file is automatically generated.\n"; DbSummary << "This is a reminder of all classes generated and methods implemented.\n\n\n"; DbSummary << "Database " << Name << " is managed through 4 functions located in " << fullfile << ".h:\n\n"; DbSummary << Name << "::init():\n"; DbSummary << "Initialises database context and connection towards database server (refered as PDS).\n"; DbSummary << "All user factories must have been set before call to this function.\n"; DbSummary << "Call this function in service init method.\n\n"; DbSummary << Name << "::ready():\n"; DbSummary << "Tells whether the whole database engine is ready to work.\n"; DbSummary << "You must not update any value nor call update() unless ready() is true.\n\n"; DbSummary << Name << "::update():\n"; DbSummary << "Updates the database engine and sends updates to the PDS.\n"; DbSummary << "Call this function each tick, provided ready() returned true.\n\n"; DbSummary << Name << "::release():\n"; DbSummary << "Releases the database engine. Drops all data, closes the connection to the PDS.\n"; DbSummary << "Call this function in service release method.\n\n"; DbSummary << "\n\n"; DbSummary << "Summary of generated classes for " << Name << "\n\n"; DbHpp << "\n#ifndef " << strReplace(strupr(filename+".h"), ".", "_") << "\n"; DbHpp << "#define " << strReplace(strupr(filename+".h"), ".", "_") << "\n"; DbHpp << "\n"; DbHpp << "#include <nel/misc/types_nl.h>\n"; DbHpp << "#include <pd_lib/pd_lib.h>\n"; DbHpp << "\n"; DbHpp << "namespace " << Name << "\n{\n\n"; DbHpp.unindent(); if (!Pch.empty()) { DbCpp << "\n#include \"" << Pch << "\"\n\n"; } DbCpp << "#include \"" << filename << ".h\"\n\n"; DbCpp << "namespace " << Name << "\n{\n\n"; DbCpp.unindent(); DbCpp << "RY_PDS::CPDSLib PDSLib;\n"; DbHppInline << "namespace " << Name << "\n{\n\n"; DbHppInline.unindent(); uint maxClassId = 0; for (i=0; i<ClassNodes.size(); ++i) { CClassNode *classnd = ClassNodes[i]; ++maxClassId; } uint maxTypeId = 0; for (i=0; i<TypeNodes.size(); ++i) { CTypeNode *typend = TypeNodes[i]; ++maxTypeId; } xmlDescription.clear(); xmlDescription.push_back("<?xml version='1.0'?>"); xmlDescription.push_back("<dbdescription version='"+DbDescriptionVersion+"'>"); //xmlDescription.push_back("<db name='"+Name+"' types='"+toString(maxTypeId)+"' classes='"+toString(maxClassId)+"' hashkey='"+HashKey.toString()+"'>"); xmlDescription.push_back("<db name='"+Name+"' types='"+toString(maxTypeId)+"' classes='"+toString(maxClassId)+"'>"); pass1(); pass2(); pass3(); // Check dependencies order vector<CClassNode*> classesOrder; vector<CFileNode*> filesOrder; buildClassOrder(classesOrder, filesOrder); // generate all file prologs for (i=0; i<FileNodes.size(); ++i) { CFileNode *child = FileNodes[i]; child->generateProlog(); } pass4(); // initDb.init("init"); initDb.IsInline = false; initDb.Proto = "uint32 overrideDbId"; initDb.Type = "void"; initDb.Description = "Initialise the whole database engine.\nCall this function at service init."; readyDb.init("ready"); readyDb.IsInline = false; readyDb.Proto = ""; readyDb.Type = "bool"; readyDb.Description = "Tells if database engine is ready to work.\nEngine may not be ready because PDS is down, not yet ready\nor message queue to PDS is full."; updateDb.init("update"); updateDb.IsInline = false; updateDb.Proto = ""; updateDb.Type = "void"; updateDb.Description = "Update the database engine.\nCall this method once per tick, only if engine is ready (see also ready() above)."; logChatDb.init("logChat"); logChatDb.IsInline = false; logChatDb.Proto = "const ucstring& sentence, const NLMISC::CEntityId& from, const std::vector<NLMISC::CEntityId>& to"; logChatDb.Type = "void"; logChatDb.Description = "Logs chat sentence with sender and receipiants."; logTellDb.init("logTell"); logTellDb.IsInline = false; logTellDb.Proto = "const ucstring& sentence, const NLMISC::CEntityId& from, const NLMISC::CEntityId& to"; logTellDb.Type = "void"; logTellDb.Description = "Logs tell sentence with sender and single recipient (might be player or group)."; releaseDb.init("release"); releaseDb.IsInline = false; releaseDb.Proto = ""; releaseDb.Type = "void"; releaseDb.Description = "Release the whole database engine.\nCall this function at service release."; // generateClassesDeclaration(); // generateClassesContent(classesOrder); // generateLogContent(); xmlDescription.push_back("</db>"); xmlDescription.push_back("</dbdescription>");; initDb.add("std::string\txmlDescription;"); DbXml.setXmlMode(); for (i=0; i<xmlDescription.size(); ++i) { initDb.add("xmlDescription += \""+xmlDescription[i]+"\\n\";"); DbXml << xmlDescription[i] << "\n"; } DbXml.flush(fullfile+".xml"); DbHpp << "/// \\name Public API for " << Name << " database\n"; DbHpp << "// @{\n"; DbHpp.unindent(); initDb.add(pdslibFunc("init")+"(xmlDescription, overrideDbId);"); initDb.flush(DbHpp, DbCpp, DbHppInline); readyDb.add("return "+pdslibFunc("PDSReady")+"();"); readyDb.flush(DbHpp, DbCpp, DbHppInline); updateDb.add(pdslibFunc("update")+"();"); updateDb.flush(DbHpp, DbCpp, DbHppInline); logChatDb.add(pdslibFunc("logChat")+"(sentence, from, to);"); logChatDb.flush(DbHpp, DbCpp, DbHppInline); logTellDb.add("std::vector<NLMISC::CEntityId>\tids;"); logTellDb.add("ids.push_back(to);"); logTellDb.add(pdslibFunc("logChat")+"(sentence, from, ids);"); logTellDb.flush(DbHpp, DbCpp, DbHppInline); releaseDb.add(pdslibFunc("release")+"();"); releaseDb.flush(DbHpp, DbCpp, DbHppInline); DbHpp << "\n// @}\n\n"; DbHpp << "extern RY_PDS::CPDSLib PDSLib;\n"; DbHpp.indent(); DbHpp << "\n} // End of " << Name <<"\n"; generateIncludes(filesOrder); DbHpp << "\n#include \"" << filename << "_inline.h\"\n\n"; DbHpp << "\n#endif\n"; DbHpp.flush(fullfile+".h"); DbCpp.indent(); DbCpp << "\n} // End of " << Name <<"\n"; DbCpp.flush(fullfile+".cpp"); DbHppInline.indent(); DbHppInline << "\n} // End of " << Name <<"\n"; DbHppInline.flush(fullfile+"_inline.h"); DbSummary.flush(fullfile+"_summary.txt"); for (i=0; i<FileNodes.size(); ++i) { CFileNode *child = FileNodes[i]; child->generateEpilog(); } return true; } void CDbNode::pass1() { /* * PASS 1 * - all type names and class names are known * -> look for classes in set or backreferences */ uint i; uint classId = 0; uint typeId = 0; for (i=0; i<ClassNodes.size(); ++i) { CClassNode *classnd = ClassNodes[i]; classnd->checkClassReferences(); classnd->Id = classId++; classnd->getFileNode()->IncludeStandard = true; classnd->getFileNode()->IncludeDbFile = true; } for (i=0; i<TypeNodes.size(); ++i) { CTypeNode *typend = TypeNodes[i]; typend->Id = typeId++; uint tsize = 0; if (typend->StorageType == "bool") tsize = 1; if (typend->StorageType == "char") tsize = 1; if (typend->StorageType == "ucchar") tsize = 2; if (typend->StorageType == "uint8") tsize = 1; if (typend->StorageType == "sint8") tsize = 1; if (typend->StorageType == "uint16") tsize = 2; if (typend->StorageType == "sint16") tsize = 2; if (typend->StorageType == "uint32") tsize = 4; if (typend->StorageType == "sint32") tsize = 4; if (typend->StorageType == "uint64") tsize = 8; if (typend->StorageType == "sint64") tsize = 8; if (typend->StorageType == "float") tsize = 4; if (typend->StorageType == "double") tsize = 8; if (typend->StorageType == "CEntityId") tsize = 8; if (typend->StorageType == "CSheetId") tsize = 4; typend->Size = tsize; string xmlnode = "<typedef name='"+typend->Name+"' id='"+toString(typend->Id)+"' size='"+toString(tsize)+"' storage='"+typend->StorageType+"'"; if (typend->isEnum()) { CEnumNode *enumnd = static_cast<CEnumNode*>(typend); xmlnode += " type='enum'>"; xmlDescription.push_back(xmlnode); uint j; for (j=0; j<enumnd->Values.size(); ++j) xmlDescription.push_back("<enumvalue name='"+enumnd->Values[j].first+"' value='"+toString(enumnd->Values[j].second)+"'/>"); xmlDescription.push_back("</typedef>"); } else if (typend->isDimension()) { CDimensionNode *dimnd = static_cast<CDimensionNode*>(typend); xmlnode += " type='dimension' dimension='"+toString(dimnd->Dimension)+"'/>"; xmlDescription.push_back(xmlnode); } else { xmlnode += " type='type'/>"; xmlDescription.push_back(xmlnode); } } } void CDbNode::pass2() { /* * PASS 2 * - class hierarchy, backreferences and in set information are known * -> fill up attributes */ uint i; for (i=0; i<ClassNodes.size(); ++i) { CClassNode *child = ClassNodes[i]; if (!child->Inherited.empty() && child->MappedFlag && !child->HasParent) child->error("class cannot inherit another class and be mapped. Try to map base class instead."); if (child->MappedFlag && child->getClassKey() == NULL) child->error("class is mapped and has no key defined"); child->ForceReference = (child->HasInheritance || child->MappedFlag || child->DerivatedFlag || (child->HasParent && !child->ParentIsHidden)); if (child->ForceReference && child->ParentIsHidden) child->error("Parent attribute cannot be hidden because class has inheritance, is mapped or is derivated."); child->getFileNode()->IncludePDSLib = true; if (!child->Inherited.empty()) { CClassNode *icln = child; CClassNode *lastMapped = (child->MappedFlag ? child : NULL); while (!icln->Inherited.empty()) { icln = getClassNode(icln->Inherited); if (icln->MappedFlag) { if (lastMapped != NULL) lastMapped->error("class cannot be remapped since parent "+icln->Name+" is already mapped"); lastMapped = icln; } } /* if (icln->MappedFlag) child->MapClass = icln; */ child->MapClass = lastMapped; } else if (child->MappedFlag) { child->MapClass = child; } child->fillAttributes(); } } void CDbNode::pass3() { /* * PASS 3 * - attributes are known * -> fill up backrefs and array/set forwardrefs */ uint i; for (i=0; i<ClassNodes.size(); ++i) { CClassNode *child = ClassNodes[i]; child->fillRefs(); } for (i=0; i<ClassNodes.size(); ++i) { CClassNode *child = ClassNodes[i]; child->computeFriends(); } } void CDbNode::pass4() { /* * PASS 4 * - everything is ok in database descriptor * -> output c++ code */ uint i; for (i=0; i<FileNodes.size(); ++i) { CFileNode* file = FileNodes[i]; file->Hpp << "\n\n//\n// Typedefs & Enums\n//\n\n"; } for (i=0; i<TypeNodes.size(); ++i) { CTypeNode *type = TypeNodes[i]; if (type->ExternFlag || type->InternFlag) continue; type->generateContent(); } } void CDbNode::generateClassesDeclaration() { uint i; DbHpp << "\n//\n"; DbHpp << "// Global Forward Declarations\n"; DbHpp << "//\n\n"; for (i=0; i<ClassNodes.size(); ++i) { DbHpp << "class " << ClassNodes[i]->Name << ";\n"; } DbHpp << "\n"; DbHpp << "//\n\n"; } void CDbNode::generateIncludes(vector<CFileNode*>& filesOrder) { uint i; DbHpp << "\n//\n"; DbHpp << "// Includes\n"; DbHpp << "//\n\n"; for (i=0; i<filesOrder.size(); ++i) { if (!filesOrder[i]->SeparatedFlag) { filesOrder[i]->setEnv("as", getFileNoExtPath(filesOrder[i]->IncludeAs)); DbHpp << "#include \"" << getFileNoExtPath(filesOrder[i]->IncludeAs) << ".h\"\n"; } } DbHpp << "\n"; for (i=0; i<filesOrder.size(); ++i) { if (filesOrder[i]->IncludeDbFile && !filesOrder[i]->SeparatedFlag) { filesOrder[i]->define("incinline"); DbHpp << "#include \"" << getFileNoExtPath(filesOrder[i]->IncludeAs) << "_inline.h\"\n"; } } DbHpp << "\n"; DbHpp << "//\n\n"; } void CDbNode::generateClassesContent(vector<CClassNode*>& classesOrder) { uint i; // // output classes content // for (i=0; i<classesOrder.size(); ++i) { CClassNode *cln = classesOrder[i]; initDb.add(pdslibFunc("registerClassMapping")+"("+toString(cln->Id)+", \""+cln->Name+"\");"); cln->generateContent(); } } void CDbNode::buildClassOrder(vector<CClassNode*>& classesOrder, vector<CFileNode*>& filesOrder) { set<CClassNode*> checkedClasses; uint i; for (i=0; i<ClassNodes.size(); ++i) { CClassNode *child = ClassNodes[i]; if (checkedClasses.find(child) != checkedClasses.end()) continue; set<CClassNode*> beingChecked; child->checkDependencies(beingChecked, checkedClasses, classesOrder); } set<CFileNode*> checkedFiles; filesOrder.clear(); for (i=0; i<FileNodes.size(); ++i) { CFileNode *child = FileNodes[i]; if (checkedFiles.find(child) != checkedFiles.end()) continue; set<CFileNode*> beingChecked; child->checkDependencies(beingChecked, checkedFiles, filesOrder); } for (i=0; i<filesOrder.size(); ++i) filesOrder[i]->Env = Env->nextArrayNode("files"); for (i=0; i<classesOrder.size(); ++i) classesOrder[i]->Env = classesOrder[i]->getFileNode()->Env->nextArrayNode("classes"); } void CDbNode::generateLogContent() { uint logid = 0; uint i; for (i=0; i<LogNodes.size(); ++i) { CLogMsgNode *child = LogNodes[i]; child->Id = logid; logid += child->Logs.size(); child->generateContent(); } } // get file path from this file string CDbNode::getFileNoExtPath(const std::string& file) { string thisPath = NLMISC::CFile::getPath(strlwr(getDbFile())); string filePath = NLMISC::CFile::getPath(strlwr(file)); string fileName = NLMISC::CFile::getFilename(strlwr(file)); if (thisPath == filePath) return CFile::getFilenameWithoutExtension(fileName); else return CPath::standardizePath(filePath)+CFile::getFilenameWithoutExtension(strlwr(file)); } // File Node bool CFileNode::prolog() { CDbNode* db = getDbNode(); db->FileNodes.push_back(this); if (GenerateOnlyLogs) Generate = false; return true; } bool CFileNode::epilog() { return true; } bool CFileNode::generateProlog() { CDbNode* db = getDbNode(); if (!db->Description.empty()) Hpp << db->Description << "\n"; string filename = strlwr(CFile::getFilenameWithoutExtension(Name)); setEnv("fullfilename", getFullStdPathNoExt(Name)); setEnv("filename", filename); setEnv("headerfilename", strReplace(strupr(filename+".h"), ".", "_")); setEnv("description", Description); Hpp.setFileHeader(filename+".h", Description); Cpp.setFileHeader(filename+".cpp", Description); HppInline.setFileHeader(filename+"_inline.h", Description); Hpp << "\n#ifndef " << strReplace(strupr(filename+".h"), ".", "_") << "\n"; Hpp << "#define " << strReplace(strupr(filename+".h"), ".", "_") << "\n\n"; Hpp << "#include <nel/misc/types_nl.h>\n"; Hpp << "#include <nel/misc/debug.h>\n"; Hpp << "#include <nel/misc/common.h>\n"; if (GenerateHAuto) { Hpp << "#include <nel/misc/hierarchical_timer.h>\n"; } if (IncludeStandard) { define("incstd"); Hpp << "#include <nel/misc/entity_id.h>\n"; Hpp << "#include <nel/misc/sheet_id.h>\n"; } Hpp << "#include <vector>\n"; Hpp << "#include <map>\n"; if (IncludePDSLib) { define("incpdslib"); Hpp << "#include <pd_lib/pd_lib.h>\n"; Hpp << "#include <game_share/persistent_data.h>\n"; } Hpp << "\n"; if (SeparatedFlag) { string fullfile = getFullStdPathNoExt(db->MainFile.empty() ? db->Name : db->MainFile); string filename = NLMISC::strlwr(NLMISC::CFile::getFilenameWithoutExtension(fullfile)); Hpp << "#include \"" << filename << ".h\"\n"; Hpp << "\n"; } Hpp << "// User #includes\n"; uint i; for (i=0; i<IncludeNodes.size(); ++i) { Env->nextArrayNode("incuser")->set("as", IncludeNodes[i]->Name); Hpp << "#include \"" << IncludeNodes[i]->Name << "\"\n"; } Hpp << "\nnamespace " << db->Name << "\n{\n\n"; Hpp.unindent(); if (!db->Description.empty()) HppInline << db->Description; HppInline << "namespace " << db->Name << "\n{\n\n"; HppInline.unindent(); Hpp << "//\n// Forward declarations\n//\n\n"; if (!db->Pch.empty()) { Cpp << "\n"; Cpp << "#include \""+db->Pch+"\""; } Cpp << "\n"; if (SeparatedFlag || !IncludeDbFile) { Cpp << "#include \"" << filename << ".h\"\n"; } else { Cpp << "#include \"" << getFileNoExtPath(getDbNode()->getDbFile()) << ".h\"\n"; } Cpp << "\n"; Cpp << "namespace " << db->Name << "\n{\n\n"; Cpp.unindent(); return true; } bool CFileNode::generateEpilog() { CDbNode* db = getDbNode(); string fullfile = getFullStdPathNoExt(Name); string filename = strlwr(CFile::getFilenameWithoutExtension(Name)); Hpp.indent(); Hpp << "\n} // End of " << db->Name <<"\n"; if (!IncludeDbFile || SeparatedFlag) { // add inline #include Hpp << "\n\n//\n// Inline implementations\n//\n\n"; Hpp << "#include \"" << filename << "_inline.h\"\n"; } Hpp << "\n#endif\n"; HppInline.indent(); HppInline << "\n} // End of " << db->Name <<"\n"; Cpp.indent(); Cpp << "\n} // End of " << db->Name <<"\n"; if (Generate) writeFile(); return true; } string CFileNode::getFileNoExtPath(const string& file) { string thisPath = NLMISC::CFile::getPath(strlwr(Name)); string filePath = NLMISC::CFile::getPath(strlwr(file)); string fileName = NLMISC::CFile::getFilename(strlwr(file)); if (thisPath == filePath) return CFile::getFilenameWithoutExtension(fileName); else return CFile::getFilenameWithoutExtension(strlwr(file)); } void CFileNode::writeFile() { string fullfile = getFullStdPathNoExt(Name); Hpp.flush(fullfile+".h"); Cpp.flush(fullfile+".cpp"); HppInline.flush(fullfile+"_inline.h"); } void CFileNode::checkDependencies(set<CFileNode*> &beingChecked, set<CFileNode*> &checkedFiles, vector<CFileNode*> &filesOrder) { if (beingChecked.find(this) != beingChecked.end()) error("circular dependency in file '"+Name+"'"); if (checkedFiles.find(this) != checkedFiles.end()) return; beingChecked.insert(this); checkedFiles.insert(this); set<CFileNode*>::iterator it; for (it=Dependencies.begin(); it!=Dependencies.end(); ++it) { CFileNode *fileNode = *it; if (fileNode == this) continue; fileNode->checkDependencies(beingChecked, checkedFiles, filesOrder); } filesOrder.push_back(this); } // Type Node bool CTypeNode::prolog() { CDbNode* db = getDbNode(); db->TypeNodes.push_back(this); return true; } bool CTypeNode::generateContent() { hOutput() << "/** " << Name << "\n"; if (!Description.empty()) { hOutput() << Description << "\n"; } uint line, col; string file; getFileLine(line, col, file); hOutput() << "defined at " << file << ":" << line << "\n"; hOutput() << "*/\n"; hOutput() << "typedef " << getCppType() << " " << Name << ";\n\n"; if (ToCppType != NULL) { CCppCodeNode *tocpp = static_cast<CCppCodeNode*>(ToCppType); CFunctionGenerator toCppFunc; toCppFunc.init(storageToCpp()); toCppFunc.setType(getName()); toCppFunc.IsInline = true; toCppFunc.Proto = StorageType+" _v"; toCppFunc.add(getName()+"\t__res;"); toCppFunc.add(strReplace(strReplace(tocpp->RawCode, "$("+CppType+")", "__res"), "$("+StorageType+")", "_v")); toCppFunc.add("return __res;"); toCppFunc.flush(hOutput(), cppOutput(), inlineOutput()); } if (ToStorageType != NULL) { CCppCodeNode *tostorage = static_cast<CCppCodeNode*>(ToStorageType); CFunctionGenerator toStorageFunc; toStorageFunc.init(cppToStorage()); toStorageFunc.setType(StorageType); toStorageFunc.IsInline = true; toStorageFunc.Proto = getName()+" _v"; toStorageFunc.add(StorageType+"\t__res;"); toStorageFunc.add(strReplace(strReplace(tostorage->RawCode, "$("+StorageType+")", "__res"), "$("+CppType+")", "_v")); toStorageFunc.add("return __res;"); toStorageFunc.flush(hOutput(), cppOutput(), inlineOutput()); } hOutput() << "\n"; return true; } // Include Node bool CIncludeNode::prolog() { CFileNode* file = getFileNode(); file->IncludeNodes.push_back(this); return true; } // Include Node bool CUsePchNode::prolog() { CDbNode* db = getDbNode(); db->Pch = Name; return true; } // CppCode Node bool CCppCodeNode::prolog() { return true; } // Dimension Nodes bool CDimensionNode::prolog() { CDbNode* db = getDbNode(); db->TypeNodes.push_back(this); if (Dimension < 256) StorageType = "uint8"; else if (Dimension < 65536) StorageType = "uint16"; else StorageType = "uint32"; CppType = "uint32"; return true; } bool CDimensionNode::epilog() { return true; } bool CDimensionNode::generateContent() { hOutput() << "/** " << Name << "\n"; if (!Description.empty()) { hOutput() << Description << "\n"; } uint line, col; string file; getFileLine(line, col, file); hOutput() << "defined at " << file << ":" << line << "\n"; hOutput() << "*/\n"; hOutput() << "typedef " << CppType << " " << Name << ";\n"; hOutput() << "const " << getName() << "\t" << getSizeName() << " = " << Dimension << ";\n\n"; return true; } // Enum Nodes bool CEnumNode::prolog() { CDbNode* db = getDbNode(); db->TypeNodes.push_back(this); if (Name.empty() || Name[0] != 'T') error("enum name '"+Name+"' is invalid, must begin with a 'T'"); CurrentValue = 0; CurrentEnum = Name; CurrentEnumNode = this; MinValue = 0; MaxValue = 0; return true; } bool CEnumNode::epilog() { uint i; for (i=0; i<Nodes.size(); ++i) { CEnumNode *nd = dynamic_cast<CEnumNode*>(Nodes[i]); if (!nd) continue; Values.insert(Values.end(), nd->Values.begin(), nd->Values.end()); } for (i=0; i<Values.size(); ++i) { if (MinValue > Values[i].second) MinValue = Values[i].second; if (MaxValue < Values[i].second) MaxValue = Values[i].second; } CurrentEnumNode = NULL; return true; } bool CEnumSimpleValueNode::prolog() { CEnumNode *parent = dynamic_cast<CEnumNode*>(Parent); if (parent != NULL) { CurrentValue = parent->CurrentValue; } else { CurrentValue = 0; } uint i; for (i=0; i<Names.size(); ++i) { CurrentEnumNode->Values.push_back(make_pair<string, uint32>(Names[i], CurrentValue)); } if (parent != NULL) ++(parent->CurrentValue); return true; } bool CEnumSimpleValueNode::epilog() { return true; } bool CEnumRangeNode::prolog() { CEnumNode *parent = dynamic_cast<CEnumNode*>(Parent); if (parent != NULL) { CurrentValue = parent->CurrentValue; } else { CurrentValue = 0; } CurrentEnumNode->Values.push_back(make_pair<string, uint32>(Name, CurrentValue)); return true; } bool CEnumRangeNode::epilog() { CEnumNode *parent = dynamic_cast<CEnumNode*>(Parent); if (parent != NULL) { parent->CurrentValue = CurrentValue; } uint i; for (i=0; i<Nodes.size(); ++i) { CEnumNode *nd = dynamic_cast<CEnumNode*>(Nodes[i]); if (!nd) continue; Values.insert(Values.end(), nd->Values.begin(), nd->Values.end()); } if (!EndRange.empty()) { CurrentEnumNode->Values.push_back(make_pair<string, uint32>(EndRange, CurrentValue)); } return true; } bool CEnumNode::generateContent() { hOutput() << "/** " << Name << "\n"; if (!Description.empty()) { hOutput() << Description << "\n"; } uint line, col; string file; getFileLine(line, col, file); hOutput() << "defined at " << file << ":" << line << "\n"; hOutput() << "*/\n"; string enumTruncName = Name.substr(1); CClassGenerator gen; gen.init("C"+enumTruncName); gen.createPublic("enum", "Enum values", ""); gen.createPublic("conv", "Conversion methods", "Use these methods to convert from enum value to string (and vice versa)"); gen.createPrivate("init", "Enum initialisation", ""); uint j; gen.addOther("enum "+Name+"\n", "enum"); gen.addOther("{\n", "enum"); for (j=0; j<Values.size(); ++j) gen.addOther(Values[j].first+" = "+toString(Values[j].second)+",\n", "enum"); gen.addOther(getUnscopedUseSize()+" = "+toString(MaxValue-MinValue+1)+",\n", "enum"); gen.addOther(getUnknownValue()+" = "+toString(MaxValue-MinValue+1)+",\n", "enum"); if (!EndRange.empty()) gen.addOther(EndRange+" = "+toString(MaxValue-MinValue+1)+",\n", "enum"); gen.addOther("};\n", "enum"); gen.startMethod("const std::string&", "toString", Name+" v", "conv", false, true, true); gen.add("if (v < 0 || v >= "+getUnscopedUseSize()+")"); gen.add("{"); gen.add("nlwarning(\""+Name+"::toString(): value '%u' is not matched, \\\"Unknown\\\" string returned\", v);"); gen.add("return _UnknownString;"); gen.add("}"); //gen.add(checkCode("v")); gen.add("if (!_Initialised)"); gen.add("{"); gen.add("init();"); gen.add("}"); gen.add("return _StrTable[v];"); gen.startMethod(getName(), "fromString", "const std::string& v", "conv", false, true, true); gen.add("if (!_Initialised)"); gen.add("{"); gen.add("init();"); gen.add("}"); gen.add("if(v==_UnknownString)"); gen.add("{"); gen.add("return Unknown;"); gen.add("}"); gen.add("const std::map<std::string, "+Name+">::const_iterator\tit = _ValueMap.find(NLMISC::toLower(v));"); gen.add("if (it == _ValueMap.end())"); gen.add("{"); gen.add("nlwarning(\""+Name+"::toString(): string '%s' is not matched, 'Unknown' enum value returned\", v.c_str());"); gen.add("return "+getUnknownValue()+";"); gen.add("}"); gen.add("return (*it).second;"); gen.startMethod("void", "init", "", "init", false, false, true); gen.add("_StrTable.clear();"); gen.add("_ValueMap.clear();"); gen.add("_StrTable.resize("+toString(getSize()+1)+");"); gen.add("uint\ti;"); gen.add("for (i=0; i<"+toString(Values.size())+"; ++i)"); gen.add("{"); gen.add("_StrTable["+Name+"Convert[i].Value] = "+Name+"Convert[i].Name;"); gen.add("_ValueMap[NLMISC::toLower(std::string("+Name+"Convert[i].Name))] = "+Name+"Convert[i].Value;"); gen.add("}"); gen.add("_Initialised = true;"); gen.addAttribute("bool", "_Initialised", "init", true, "false"); gen.addAttribute("std::string", "_UnknownString", "init", true, "\""+getUnknownValue()+"\""); gen.addAttribute("std::vector<std::string>", "_StrTable", "init", true); gen.addAttribute("std::map<std::string, "+Name+">", "_ValueMap", "init", true, "", false, "std::map<std::string, "+getName()+">"); cppOutput() << "static const struct { char* Name; " << getName() << " Value; } " << Name << "Convert[] =\n"; cppOutput() << "{\n"; for (j=0; j<Values.size(); ++j) cppOutput() << "{ \"" << Values[j].first << "\", C"+enumTruncName+"::"+Values[j].first+" },\n"; cppOutput() << "};\n"; gen.flush(hOutput(), cppOutput(), inlineOutput()); return true; } // Class Node bool CClassNode::prolog() { CDbNode* db = getDbNode(); db->ClassNodes.push_back(this); return true; } bool CClassNode::epilog() { return true; } string CClassNode::getUserCode(const string& name) { uint i; for (i=0; i<Nodes.size(); ++i) { CCppCodeNode *code = dynamic_cast<CCppCodeNode*>(Nodes[i]); if (!code) continue; if (code->Name == name) return code->RawCode; } return ""; } void CClassNode::checkClassReferences() { if (!Implements.empty()) { Gen.Inherit += (HasRowAccess ? string(", ") : string(""))+"public "+Implements; getDbNode()->Implemented.insert(Implements); } if (!Inherited.empty()) { HasInheritance = true; CClassNode *nd = getClassNode(Inherited); nd->HasInheritance = true; nd->ChildrenClasses.push_back(Name); Dependencies.insert(nd); getFileNode()->Dependencies.insert(nd->getFileNode()); } CClassNode* inherit = this; while (inherit != NULL) { if (MappedFlag) inherit->PDSMapped = true; inherit->Legacy.insert(this); inherit = getClassNode(inherit->Inherited, false); } uint i; uint id = 0; for (i=0; i<Nodes.size(); ++i) { CDeclarationNode *decl = dynamic_cast<CDeclarationNode*>(Nodes[i]); if (!decl) continue; decl->Id = id++; if (decl->ParentFlag) { if (HasParent) decl->error("class '"+Name+"' already has a parent"); //if (MappedFlag) // decl->error("class '"+Name+"' can't have a parent and be mapped at the same time"); CClassNode* inherit = this; while (inherit != NULL) { inherit->PDSMapped = false; inherit = getClassNode(inherit->Inherited, false); } ParentClass = decl->ParentClass; HasParent = true; ParentIsHidden = decl->HiddenFlag; decl->getClassNode(decl->ParentClass)->IsBackReferenced = true; } else if (decl->SetFlag) { decl->getClassNode(decl->Type)->IsInSet = true; } } } void CClassNode::fillAttributes() { if (HasParent && !IsBackReferenced && !HasInheritance && !IsInSet && !DerivatedFlag && !MappedFlag) error("class '"+Name+"' has a parent whereas it is not backreferenced, has no inherited link and is not mapped"); uint i; for (i=0; i<Nodes.size(); ++i) { CDeclarationNode *decl = dynamic_cast<CDeclarationNode*>(Nodes[i]); if (!decl) continue; uint j; for (j=0; j<Nodes.size(); ++j) if (j != i && dynamic_cast<CDeclarationNode*>(Nodes[j]) != NULL && Nodes[j]->Name == decl->Name) decl->error("attribute '"+decl->Name+"' already defined"); if (decl->ParentFlag) { decl->DeclarationType = BackRef; } else if (decl->ArrayFlag) { CClassNode *classNd = NULL; CTypeNode *typeNd = NULL; if (classNd = decl->getClassNode(decl->Type, false)) { if (classNd->IsBackReferenced || classNd->ForceReference) { if (decl->ForwardRefAttribute.empty()) decl->error("no forward reference to parent in array declaration, class '"+decl->Type+"' is backref'd or has inheritance"); classNd->IsInArrayRef = true; decl->IsRef = true; decl->DeclarationType = ArrayRef; if (classNd->ParentIsHidden) Dependencies.insert(classNd); } else { if (!decl->ForwardRefAttribute.empty()) decl->error("forward reference declared whereas subclass is not backreferenced and has no inheritance link"); Dependencies.insert(classNd); classNd->IsInArray = true; decl->IsRef = false; decl->DeclarationType = ArrayClass; } } else if (typeNd = decl->getTypeNode(decl->Type, false)) { decl->DeclarationType = ArrayType; } else { decl->error("type or class '"+decl->Type+"' not found"); } } else if (decl->SetFlag) { decl->IsRef = true; decl->DeclarationType = Set; } else { CClassNode *classNd = NULL; CTypeNode *typeNd = NULL; if (classNd = decl->getClassNode(decl->Type, false)) { if (classNd->IsBackReferenced || classNd->ForceReference) { if (decl->ForwardRefAttribute.empty()) decl->error("no forward reference to parent in array declaration, class '"+decl->Type+"' is backref'd or has inheritance"); decl->IsRef = true; decl->DeclarationType = ForwardRef; } else { if (!decl->ForwardRefAttribute.empty()) decl->error("forward reference declared whereas subclass is not backreferenced and has no inheritance link"); Dependencies.insert(classNd); decl->IsRef = false; decl->DeclarationType = SimpleClass; } } else if (typeNd = decl->getTypeNode(decl->Type, false)) { decl->IsType = true; decl->DeclarationType = SimpleType; } else { decl->error("type or class '"+decl->Type+"' not found"); } } } CDeclarationNode *declNd = getClassKey(); if (declNd != NULL) { if (!declNd->IsType) error("attribute '"+declNd->Name+"' can't be a key, only simple type allowed"); declNd->IsKey = true; } } void CClassNode::fillRefs() { uint i; for (i=0; i<Nodes.size(); ++i) { CDeclarationNode *decl = dynamic_cast<CDeclarationNode*>(Nodes[i]); if (!decl) continue; switch (decl->DeclarationType) { case BackRef: { // check parent is a valid class CClassNode *cln = decl->getClassNode(decl->ParentClass); CDeclarationNode *dln = dynamic_cast<CDeclarationNode*>(cln->getNode(decl->ParentField)); if (!dln) decl->error("attribute '"+decl->ParentField+"' not found in class '"+decl->ParentClass+"'"); if (!dln->ArrayFlag && !dln->SetFlag && !ForceReference) decl->error("back reference 'parent "+decl->ParentClass+":"+decl->ParentField+" "+decl->Name+"' is not forwarded in class '"+decl->ParentClass+"' with an array or a set"); if (dln->Type != Name || dln->ForwardRefAttribute != decl->Name) decl->error("back reference 'parent "+decl->ParentClass+":"+decl->ParentField+" "+decl->Name+"' is not correctly forwarded in class '"+decl->ParentClass+"'"); Friends.insert(cln->Name); } break; case Set: { CClassNode *cln = decl->getClassNode(decl->Type); CDeclarationNode *dln = dynamic_cast<CDeclarationNode*>(cln->getNode(decl->ForwardRefAttribute)); if (!dln) decl->error("attribute '"+decl->ForwardRefAttribute+"' not found in class '"+decl->Type+"'"); if (!dln->ParentFlag) decl->error("set '"+decl->Type+":"+decl->ForwardRefAttribute+"<> "+decl->Name+"' is not backref'd in class '"+decl->Type+"'"); if (dln->ParentClass != Name || dln->ParentField != decl->Name) decl->error("set '"+decl->Type+":"+decl->ForwardRefAttribute+"<> "+decl->Name+"' is not correctly backref'd in class '"+decl->Type+"'"); if (cln->getClassKey() == NULL) decl->error("class '"+decl->Type+"' has no key defined, whereas it is used in a set"); cln->Friends.insert(Name); ForwardFriends.insert(cln->Name); } break; case ArrayRef: { if (decl->ForwardRefAttribute.empty()) decl->error("No forward reference defined"); CClassNode *cln = decl->getClassNode(decl->Type); CDeclarationNode *dln = dynamic_cast<CDeclarationNode*>(cln->getNode(decl->ForwardRefAttribute)); CTypeNode *tln = decl->getTypeNode(decl->ArrayIndex); getFileNode()->Dependencies.insert(tln->getFileNode()); if (!dln) decl->error("attribute '"+decl->ForwardRefAttribute+"' not found in class '"+decl->Type+"'"); if (!dln->ParentFlag) decl->error("array '"+decl->Type+":"+decl->ForwardRefAttribute+"["+decl->ArrayIndex+"] "+decl->Name+"' is not backref'd in class '"+decl->Type+"'"); if (dln->ParentClass != Name || dln->ParentField != decl->Name) decl->error("array '"+decl->Type+":"+decl->ForwardRefAttribute+"["+decl->ArrayIndex+"] "+decl->Name+"' is not correctly backref'd in class '"+decl->Type+"'"); if (cln->getClassKey() == NULL) decl->error("class '"+decl->Type+"' has no key defined, whereas it is used in an array of ref"); CDeclarationNode *kdn = dynamic_cast<CDeclarationNode*>(cln->getClassKey()); if (!kdn) decl->error("attribute '"+cln->ClassKey+"' not found in class '"+cln->Name+"'"); if (kdn->Type != decl->ArrayIndex) decl->error("type in array definition mismatch class '"+cln->Name+"' key definition"); cln->Friends.insert(Name); ForwardFriends.insert(cln->Name); } break; case ForwardRef: { if (decl->ForwardRefAttribute.empty()) decl->error("No forward reference defined"); CClassNode *cln = decl->getClassNode(decl->Type); CDeclarationNode *dln = dynamic_cast<CDeclarationNode*>(cln->getNode(decl->ForwardRefAttribute)); if (!dln) decl->error("attribute '"+decl->ForwardRefAttribute+"' not found in class '"+decl->Type+"'"); if (!dln->ParentFlag) decl->error("set '"+decl->Type+":"+decl->ForwardRefAttribute+"<> "+decl->Name+"' is not backref'd in class '"+decl->Type+"'"); if (dln->ParentClass != Name || dln->ParentField != decl->Name) decl->error("set '"+decl->Type+":"+decl->ForwardRefAttribute+"<> "+decl->Name+"' is not correctly backref'd in class '"+decl->Type+"'"); cln->Friends.insert(Name); ForwardFriends.insert(cln->Name); } break; case ArrayType: { CTypeNode *tln = decl->getTypeNode(decl->ArrayIndex); getFileNode()->Dependencies.insert(tln->getFileNode()); } case SimpleType: break; case ArrayClass: { CTypeNode *tln = decl->getTypeNode(decl->ArrayIndex); getFileNode()->Dependencies.insert(tln->getFileNode()); } case SimpleClass: { CClassNode *cln = decl->getClassNode(decl->Type); cln->Friends.insert(Name); } break; default: decl->error("Can't decide declaration type"); break; } } } void CClassNode::computeFriends() { bool added; do { added = false; set<string>::iterator itf, itsf; for (itf=Friends.begin(); !added && itf!=Friends.end(); ++itf) { CClassNode* pfriend = getClassNode(*itf); for (itsf=pfriend->Friends.begin(); !added && itsf!=pfriend->Friends.end(); ++itsf) { const string& sfriend = *itsf; if (Friends.find(*itsf) == Friends.end()) { Friends.insert(*itsf); added = true; } } } } while (added); } void CClassNode::checkDependencies(set<CClassNode*> &beingChecked, set<CClassNode*> &checkedClasses, vector<CClassNode*> &classesOrder) { if (beingChecked.find(this) != beingChecked.end()) error("circular dependency in class '"+Name+"'"); if (checkedClasses.find(this) != checkedClasses.end()) return; beingChecked.insert(this); checkedClasses.insert(this); set<CClassNode*>::iterator it; for (it=Dependencies.begin(); it!=Dependencies.end(); ++it) { CClassNode *classNode = *it; classNode->checkDependencies(beingChecked, checkedClasses, classesOrder); } classesOrder.push_back(this); } // void CClassNode::buildInit() { CDbNode *db = getDbNode(); if (!Init.empty()) return; if (!Inherited.empty()) { CClassNode *mother = getClassNode(Inherited); if (mother) { mother->buildInit(); Init = mother->Init; uint i; for (i=0; i<Init.size(); ++i) { if (!InitProto.empty()) InitProto += ", "; if (!InitCallArgs.empty()) InitCallArgs += ", "; CTypeNode* typeNode = getTypeNode(Init[i]->Type); InitProto += "const "+typeNode->getName()+" &"+Init[i]->Name; InitCallArgs += Init[i]->Name; } } } if (!ClassKey.empty()) { CDeclarationNode *decl = dynamic_cast<CDeclarationNode*>(getNode(ClassKey)); if (decl) { Init.push_back(decl); if (!InitProto.empty()) InitProto += ", "; CTypeNode* typeNode = getTypeNode(decl->Type); InitProto += "const "+typeNode->getName()+" &"+decl->Name; } } } void CClassNode::computeAttributesColumns() { if (Columns >= 0) return; CDbNode *db = getDbNode(); Columns = 0; if (!Inherited.empty()) { CClassNode *mother = getClassNode(Inherited); mother->computeAttributesColumns(); Columns = mother->Columns; Attributes = mother->Attributes; } uint attribId = Attributes.size(); uint i; for (i=0; i<Nodes.size(); ++i) { CDeclarationNode *decl = dynamic_cast<CDeclarationNode*>(Nodes[i]); if (decl == NULL) continue; Attributes.push_back(decl); decl->Column = Columns; decl->Id = attribId++; CColumn col; // All for backref, set, forwardref, type switch (decl->DeclarationType) { case ForwardRef: decl->Columns = 1; col.Name = decl->Name; col.Type = ForwardRef; col.TypeStr = "forwardref"; col.TypeId = decl->getClassNode(decl->Type)->Id; col.ByteSize = 8; decl->ColumnList.push_back(col); break; case BackRef: decl->Columns = 1; col.Name = decl->Name; col.Type = BackRef; col.TypeStr = "backref"; col.TypeId = decl->getClassNode(decl->ParentClass)->Id; col.ByteSize = 8; decl->ColumnList.push_back(col); break; case SimpleType: decl->Columns = 1; col.Name = decl->Name; col.Type = SimpleType; col.TypeStr = "type"; col.TypeId = decl->getTypeNode(decl->Type)->Id; col.ByteSize = decl->getTypeNode(decl->Type)->Size; decl->ColumnList.push_back(col); break; case Set: decl->Columns = 1; col.Name = decl->Name; col.Type = Set; col.TypeStr = "set"; col.TypeId = decl->getClassNode(decl->Type)->Id; col.ByteSize = 4; decl->ColumnList.push_back(col); break; case SimpleClass: { CClassNode *sub = decl->getClassNode(decl->Type); sub->computeAttributesColumns(); decl->Columns = sub->Columns; uint i, j; for (i=0; i<sub->Attributes.size(); ++i) { CDeclarationNode *attrib = sub->Attributes[i]; for (j=0; j<attrib->ColumnList.size(); ++j) { col.Name = decl->Name+"."+attrib->ColumnList[j].Name; col.Type = attrib->ColumnList[j].Type; col.TypeStr = attrib->ColumnList[j].TypeStr; col.TypeId = attrib->ColumnList[j].TypeId; col.ByteSize = attrib->ColumnList[j].ByteSize; decl->ColumnList.push_back(col); } } } break; case ArrayRef: { CIndexNode *indexNd = decl->getIndexNode(decl->ArrayIndex); uint numInEnum = indexNd->getSize(); decl->Columns = numInEnum; uint i; for (i=0; i<numInEnum; ++i) { col.Name = decl->Name+"["+indexNd->getIndexName(i)+"]"; col.Type = ForwardRef; col.TypeStr = "forwardref"; col.TypeId = decl->getClassNode(decl->Type)->Id; col.ByteSize = 8; decl->ColumnList.push_back(col); } } break; case ArrayType: { CIndexNode *indexNd = decl->getIndexNode(decl->ArrayIndex); uint numInEnum = indexNd->getSize(); decl->Columns = numInEnum; uint i; for (i=0; i<numInEnum; ++i) { col.Name = decl->Name+"["+indexNd->getIndexName(i)+"]"; col.Type = SimpleType; col.TypeStr = "type"; col.TypeId = decl->getTypeNode(decl->Type)->Id; col.ByteSize = decl->getTypeNode(decl->Type)->Size; decl->ColumnList.push_back(col); } } break; case ArrayClass: { CIndexNode *indexNd = decl->getIndexNode(decl->ArrayIndex); CClassNode *sub = decl->getClassNode(decl->Type); sub->computeAttributesColumns(); uint numInEnum = indexNd->getSize(); decl->Columns = numInEnum*sub->Columns; uint i, j, k; for (k=0; k<numInEnum; ++k) { for (i=0; i<sub->Attributes.size(); ++i) { CDeclarationNode *attrib = sub->Attributes[i]; for (j=0; j<attrib->ColumnList.size(); ++j) { col.Name = decl->Name+"["+indexNd->getIndexName(k)+"]."+attrib->ColumnList[j].Name; col.Type = attrib->ColumnList[j].Type; col.TypeStr = attrib->ColumnList[j].TypeStr; col.TypeId = attrib->ColumnList[j].TypeId; col.ByteSize = attrib->ColumnList[j].ByteSize; decl->ColumnList.push_back(col); } } } } break; } Columns += decl->Columns; } } bool CClassNode::generateContent() { nlassert(Env != NULL); uint line, col; string file; getFileLine(line, col, file); setEnv("name", Name); if (!Description.empty()) setEnv("description", Description); setEnv("deffile", file); setEnv("defline", line); computeAttributesColumns(); uint j; // // generate description // hOutput() << "/** " << Name << "\n"; if (!Description.empty()) hOutput() << Description << "\n"; hOutput() << "defined at " << file << ":" << line << "\n"; hOutput() << "*/\n"; CCppOutput& DbSummary = getDbNode()->DbSummary; DbSummary << "Class " << getDbNode()->Name << "::" << Name << ":\n"; DbSummary << "----------------------------------------------------------\n"; DbSummary << "located in file \"" << getFullStdPathNoExt(getFileNode()->Name) << ".h\"\n"; DbSummary << "defined in file \"" << getFullStdPath(file) << "\"\n"; DbSummary << "The class contains:\n\n"; Gen.init(Name); Gen.createPublic("methods", "Accessors and Mutators methods", "Use these methods to change a value, add or delete elements."); Gen.createPublic("map", "Public Management methods", "Use these methods to create, load, unload and get\nan object from database."); Gen.createPublic("user", "User defined attributes and methods", "This code was verbatim copied from source file"); Gen.createPublic("construct", "Public constructor", "This constructor is public to allow direct instanciation of the class"); Gen.createPublic("persist", "Persistent methods declaration", ""); Gen.createProtected("userinit", "User defined init and release methods", "Overload those methods to implement init and release behaviours"); Gen.createProtected("attributes", "Attributes", "Don't modify those value manually, use accessors and mutators above"); Gen.createProtected("internal", "Internal Management methods"); Gen.createProtected("inherit map"); Gen.createProtected("factories", "Default Factory and Fetch methods"); Gen.createProtected("friends"); // EGS Compat // -- begin Gen.startRaw("persist", false); ApplyId = Gen.startMethod("void", applyFunction, "CPersistentDataRecord &__pdr", "persist", false, inlineInternal, false, false, "", HasInheritance); StoreId = Gen.startMethod("void", storeFunction, "CPersistentDataRecord &__pdr", "persist", true, inlineInternal, false, false, "", HasInheritance); ClearId = Gen.startMethod("void", clearFunction, "", "map", false, inlineStaticPublic, false, false, "", HasInheritance); Gen.setDescription("Clear whole object content but key (delete subobjects if there are, key is left unmodified), default clear value is 0."); StoreId.add("uint16\t__Tok_MapKey = __pdr.addString(\"__Key__\");"); StoreId.add("uint16\t__Tok_MapVal = __pdr.addString(\"__Val__\");"); StoreId.add("uint16\t__Tok_ClassName = __pdr.addString(\"__Class__\");"); ApplyId.add("uint16\t__Tok_MapKey = __pdr.addString(\"__Key__\");"); ApplyId.add("uint16\t__Tok_MapVal = __pdr.addString(\"__Val__\");"); ApplyId.add("uint16\t__Tok_ClassName = __pdr.addString(\"__Class__\");"); if (!Inherited.empty()) { StoreId.add("uint16\t__Tok_Parent = __pdr.addString(\"__Parent__\");"); ApplyId.add("uint16\t__Tok_Parent = __pdr.addString(\"__Parent__\");"); } for (j=0; j<Attributes.size(); ++j) { CDeclarationNode* decl = Attributes[j]; if (decl->Parent != this) continue; if (decl->DeclarationType == BackRef) { ApplyId.add(decl->cppName()+" = NULL;"); } else { StoreId.add("uint16\t"+decl->tokenName()+" = __pdr.addString(\""+decl->Name+"\");"); ApplyId.add("uint16\t"+decl->tokenName()+" = __pdr.addString(\""+decl->Name+"\");"); } } ApplyId.add("while (!__pdr.isEndOfStruct())"); ApplyId.add("{"); ApplyId.add( "if (false) {}"); if (!Inherited.empty()) { StoreId.add("__pdr.pushStructBegin(__Tok_Parent);"); StoreId.add(Inherited+"::store(__pdr);"); StoreId.add("__pdr.pushStructEnd(__Tok_Parent);"); ApplyId.add("else if (__pdr.peekNextToken() == __Tok_Parent)"); ApplyId.add("{"); ApplyId.add( "__pdr.popStructBegin(__Tok_Parent);"); ApplyId.add( Inherited+"::apply(__pdr);"); ApplyId.add( "__pdr.popStructEnd(__Tok_Parent);"); ApplyId.add("}"); } // -- end for (j=0; j<Nodes.size(); ++j) { CCppCodeNode* cpp = dynamic_cast<CCppCodeNode*>(Nodes[j]); if (cpp == NULL || !cpp->Name.empty()) continue; Gen.addOther(cpp->RawCode, "user"); } HasRowAccess = false; HasTableAccess = false; if (!Inherited.empty()) { setEnv("inherit", Inherited); setEnv("inheritclass", Inherited); Gen.Inherit += "public "+Inherited; HasRowAccess = true; HasTableAccess = true; } else if (HasInheritance || IsBackReferenced || IsInSet || ForceReference) { setEnv("inherit", pdBaseDataName); Gen.Inherit += "public "+pdBaseDataName; HasTableAccess = true; HasRowAccess = true; } if (Legacy.size() > 1 && HasTableAccess) { CClassGenerator::SMethodId castId = Gen.startMethod(Name+"*", staticCastFunction, pdBaseDataName+"* obj", "map", false, inlineStaticPublic, true, false, "", false, false); Gen.setDescription("Cast base object to "+Name); castId.add("switch (obj->getTable())"); castId.add("{"); std::set<CClassNode*>::iterator itl; for (itl=Legacy.begin(); itl!=Legacy.end(); ++itl) { CClassNode* child = (*itl); castId.add("case "+toString(child->Id)+":"); } castId.add("return static_cast<"+Name+"*>(obj);"); castId.add("}"); castId.add("return NULL;"); CClassGenerator::SMethodId constCastId = Gen.startMethod("const "+Name+"*", staticConstCastFunction, "const "+pdBaseDataName+"* obj", "map", false, inlineStaticPublic, true, false, "", false, false); Gen.setDescription("Cast base object to const "+Name); constCastId.add("switch (obj->getTable())"); constCastId.add("{"); for (itl=Legacy.begin(); itl!=Legacy.end(); ++itl) { CClassNode* child = (*itl); constCastId.add("case "+toString(child->Id)+":"); } constCastId.add("return static_cast<const "+Name+"*>(obj);"); constCastId.add("}"); constCastId.add("return NULL;"); } else if (Legacy.size() == 1 && HasTableAccess) { CClassGenerator::SMethodId castId = Gen.startMethod(Name+"*", staticCastFunction, pdBaseDataName+"* obj", "map", false, inlineStaticPublic, true, false, "", false, false); std::set<CClassNode*>::iterator itl = Legacy.begin(); Gen.setDescription("Cast base object to "+Name); castId.add("return (obj->getTable() == "+toString((*itl)->Id)+") ? static_cast<"+Name+"*>(obj) : NULL;"); CClassGenerator::SMethodId constCastId = Gen.startMethod("const "+Name+"*", staticConstCastFunction, "const "+pdBaseDataName+"* obj", "map", false, inlineStaticPublic, true, false, "", false, false); Gen.setDescription("Cast base object to const "+Name); constCastId.add("return (obj->getTable() == "+toString((*itl)->Id)+") ? static_cast<const "+Name+"*>(obj) : NULL;"); } else { } if (HasRowAccess) define("hasrowaccess"); if (HasTableAccess) define("hastableaccess"); if (!Implements.empty()) { setEnv("implements", Implements); Gen.Inherit += (HasRowAccess ? string(", ") : string(""))+"public "+Implements; } // // generate init method // buildInit(); setEnv("initproto", InitProto); InitId = Gen.startMethod("void", initFunction, InitProto, "internal", false, inlineInternal, false, false, "", HasInheritance); if (!Inherited.empty()) { InitId.add(Inherited + "::" + initFunction + "(" + InitCallArgs + ");"); setEnv("initcallargs", InitCallArgs); } DestroyId = Gen.startMethod("void", destroyFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); FetchId = Gen.startMethod("void", fetchFunction, CPDataName+" &data", "internal", false, inlineInternal, false, false, "", HasInheritance); RegisterId = Gen.startMethod("void", registerFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); RegisterAttributesId = Gen.startMethod("void", registerAttributesFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); UnregisterId = Gen.startMethod("void", unregisterFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); UnregisterAttributesId = Gen.startMethod("void", unregisterAttributesFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); SetParentId; SetUnnotifiedParentId; if (HasParent) { SetParentId = Gen.startMethod("void", setParentFunction, ParentClass+"* __parent", "internal", false, inlineInternal); SetUnnotifiedParentId = Gen.startMethod("void", setUnnotifiedParentFunction, ParentClass+"* __parent", "internal", false, inlineInternal); } if (HasRowAccess && GenerateDebugMessages) { DestroyId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": destroy %u:%u\", "+getId()+", __BaseRow);"); FetchId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": fetch %u:%u\", "+getId()+", __BaseRow);"); } if (!Inherited.empty()) { FetchId.add(Inherited+"::"+fetchFunction+"(data);"); } if (DerivatedFlag) { UserInitId = Gen.startMethod("void", userInitFunction, "", "userinit", false, inlineUserInitDefaultCode, false, false, "", true); UserReleaseId = Gen.startMethod("void", userReleaseFunction, "", "userinit", false, inlineUserInitDefaultCode, false, false, "", true); } NotifyInitId = Gen.startMethod("void", notifyInitFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); string initUCode = getUserCode("onInit"); if (!Inherited.empty()) Gen.add(Inherited+"::"+notifyInitFunction+"();"); if (DerivatedFlag) Gen.add(userInitFunction+"();"); if (!initUCode.empty()) { Gen.add("{"); uint line, col; string file; getFileLine(line, col, file); Gen.add("// "+Name+" init user code, defined at "+file+":"+toString(line)); Gen.add(initUCode); Gen.add("}"); } NotifyReleaseId = Gen.startMethod("void", notifyReleaseFunction, "", "internal", false, inlineInternal, false, false, "", HasInheritance); string releaseUCode = getUserCode("onRelease"); if (!releaseUCode.empty()) { Gen.add("{"); uint line, col; string file; getFileLine(line, col, file); Gen.add("// "+Name+" release user code, defined at "+file+":"+toString(line)); Gen.add(releaseUCode); Gen.add("}"); } if (DerivatedFlag) Gen.add(userReleaseFunction+"();"); if (!Inherited.empty()) Gen.add(Inherited+"::"+notifyReleaseFunction+"();"); else if (HasRowAccess) Gen.add(pdslibFunc("release")+"("+getId()+", __BaseRow);"); if (!Inherited.empty()) { DestroyId.add(Inherited+"::"+destroyFunction+"();"); ClearId.add(Inherited+"::"+clearFunction+"();"); } // // Generate XML description // string xmlnode; xmlnode += "<classdef"; xmlnode += " name='"+Name+"'"; xmlnode += " id='"+toString(Id)+"'"; if (!Inherited.empty()) { CClassNode *inh = getClassNode(Inherited); xmlnode += " inherit='"+toString(inh->Id)+"'"; } CDeclarationNode *dln = getClassKey(); if (dln != NULL) { xmlnode += " key='"+toString(dln->Id)+"'"; } if (MapClass && !MapClass->HasParent) { xmlnode += " mapped='"+toString(MapClass->Id)+"'"; } if (HasRowAccess) { /* if (!Reserve.empty()) { xmlnode += " allocate='"+(Reserve)+"'"; } else { xmlnode += " allocate='10000'"; } */ } xmlnode += " columns='"+toString(Columns)+"'"; xmlnode += ">"; getDbNode()->xmlDescription.push_back(xmlnode); indexUsedInInit = false; indexUsedInDestroy = false; indexUsedInFetch = false; tableAndRowIndicesUsedInFetch = false; indexUsedInRegister = false; indexUsedInUnregister = false; // generate code for init of new index if (HasRowAccess) { RegisterId.add("__BaseRow = _IndexAllocator.allocate();"); if (GenerateDebugMessages) { if (MapClass != NULL) { CDeclarationNode* key = MapClass->getKey(); RegisterId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": register %u:%u, key="+key->displayPrintfPrefix()+"\", "+getId()+", __BaseRow, "+key->displayCppCode()+");"); } else { RegisterId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": register %u:%u\", "+getId()+", __BaseRow);"); } } string oeid; if (useEntityId()) { oeid = ", "+getClassKey()->cppName(); } RegisterId.add(pdslibFunc("allocateRow")+"("+getId()+", __BaseRow, "+(PDSMapped ? MapClass->getKey()->toUint64() : "0")+oeid+");"); RegisterId.add(registerAttributesFunction + "();"); RegisterAttributesId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": registerAttributes %u:%u\", "+getId()+", __BaseRow);"); // send key to the pds (if key exists) if (!ClassKey.empty()) { CDeclarationNode *keyNode = getKey(); CTypeNode *keyTypeNode = getTypeNode(keyNode->Type); ///// TYPE CAST RegisterAttributesId.add(pdslibFunc("set")+"("+getId()+", __BaseRow, ("+TColumnIndexName+")("+toString(keyNode->Column)+"), "+keyTypeNode->castToPDS(getKey()->cppName())+");"); } if (!Inherited.empty()) { RegisterAttributesId.add(Inherited + "::" + registerAttributesFunction + "();"); } UnregisterAttributesId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": unregisterAttributes %u:%u\", "+getId()+", __BaseRow);"); if (!Inherited.empty()) { UnregisterAttributesId.add(Inherited + "::" + unregisterAttributesFunction + "();"); } if (HasParent) { UnregisterAttributesId.add(setParentFunction+"(NULL);"); } if (GenerateDebugMessages) { if (MapClass != NULL) { CDeclarationNode* key = MapClass->getKey(); UnregisterId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": unregister %u:%u, key="+key->displayPrintfPrefix()+"\", "+getId()+", __BaseRow, "+key->displayCppCode()+");"); } else UnregisterId.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": unregister %u:%u\", "+getId()+", __BaseRow);"); } UnregisterId.add(unregisterAttributesFunction + "();"); oeid = ""; if (useEntityId()) { oeid = ", "+getClassKey()->cppName(); } UnregisterId.add(pdslibFunc("deallocateRow")+"("+getId()+", __BaseRow"+oeid+");"); UnregisterId.add("_IndexAllocator.deallocate(__BaseRow);"); //UnregisterId.add(destroyFunction+"();"); } // // add attributes and methods // - attributes are in private part // - read accessor are public // - write accessor are public or delegated in public accessor objects // for (j=0; j<Nodes.size(); ++j) { CDeclarationNode *dln = dynamic_cast<CDeclarationNode*>(Nodes[j]); if (!dln) continue; dln->Env = Env->nextArrayNode("dcl"); dln->generateContent(); } uint columnId = 0; for (j=0; j<Attributes.size(); ++j) { CDeclarationNode *dln = Attributes[j]; getDbNode()->xmlDescription.push_back("<attribute "+dln->XmlNode+"/>"); uint k; for (k=0; k<dln->ColumnList.size(); ++k) { CColumn &column = dln->ColumnList[k]; //getDbNode()->xmlDescription.push_back("<column id='"+toString(k)+"' name='"+column.Name+"' type='"+column.TypeStr+"' typeid='"+toString(column.TypeId)+"' size='"+toString(column.ByteSize)+"' columnid='"+toString(columnId)+"'/>"); ++columnId; } //getDbNode()->xmlDescription.push_back("</attribute>"); } getDbNode()->xmlDescription.push_back("</classdef>"); if (HasTableAccess) { Gen.startConstructor("", "construct"); Gen.add("__BaseTable = "+toString(Id)+";"); Gen.startDestructor("construct", true, DerivatedFlag || HasInheritance); } // when inited/fetched a mapped class, map id to object if (MappedFlag) { InitId.add("_Map["+getKey()->getFunc()+"()] = this;"); FetchId.add("_Map["+getKey()->getFunc()+"()] = this;"); DestroyId.add("_Map.erase("+getKey()->getFunc()+"());"); } // // generate IPDBaseData API // if (MappedFlag || DerivatedFlag || HasInheritance || ForceReference) { if (DerivatedFlag) { Gen.startMethod("void", staticSetUserFactoryFunction, TPDFactoryName+" userFactory", "map", false, inlineStaticPublic, true); Gen.setDescription("Set user factory for this class (as class is indicated as derived, a home made constructor must be provided)"); Gen.add(staticInitFactoryFunction+"(userFactory);"); } Gen.startMethod("void", staticInitFactoryFunction, TPDFactoryName+" userFactory", "factories", false, inlineStaticInternal, true); Gen.add("if (!_FactoryInitialised)"); Gen.add("{"); Gen.add(pdslibFunc("registerClass")+"(" + toString(Id) + ", userFactory, "+staticFetchFunction+", "+((MappedFlag && !HasParent) ? staticNotifyLoadFailure : string("NULL"))+");"); Gen.add("_FactoryInitialised = true;"); Gen.add("}"); Gen.addAttribute("bool", "_FactoryInitialised", "factories", true); if (MappedFlag || HasInheritance || ForceReference) { // // create: create an object, then init attributes and register // Gen.startMethod(Name+"*", staticCreateFunction, InitProto, "map", false, inlineStaticPublic, true); Gen.setDescription("Create an object of the "+Name+" class, and declare it to the PDS."); if (GenerateDebugMessages) { Gen.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": create\");"); } if (DerivatedFlag) { Gen.add(Name + "\t*"+objectVariable+" = static_cast<" + Name + "*>("+pdslibFunc("create")+"("+toString(Id)+"));"); } else { Gen.add(Name + "\t*"+objectVariable+" = static_cast<" + Name + "*>("+staticFactoryFunction+"());"); } string str = objectVariable+"->" + initFunction + "("; uint i; for (i=0; i<Init.size(); ++i) str += (i != 0 ? ", " : "") + Init[i]->Name; str += ");"; Gen.add(str); Gen.add(objectVariable+"->" + registerFunction + "();"); /* if (MappedFlag) { Gen.add("_Map["+MapClass->ClassKey+"] = "+objectVariable+";"); } */ Gen.add(objectVariable+"->"+notifyInitFunction+"();"); Gen.add("return "+objectVariable+";"); } if (MappedFlag) { CDeclarationNode *dln = (MapClass != NULL ? MapClass->getKey() : NULL); CTypeNode *keyType = getTypeNode(dln->Type); // only authorize remove/load/unload for mapped objects that are roots if (!HasParent) { Gen.startMethod("void", staticRemoveFunction, "const "+keyType->getName()+"& "+dln->Name, "map", false, inlineStaticPublic, true); Gen.setDescription("Destroy an object from the PDS. Caution! Object will no longer exist in database.\nAlso children (that is objects that belong to this object) are also destroyed."); if (GenerateDebugMessages) { Gen.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": remove "+dln->displayPrintfPrefix()+"\", "+dln->displayCppCode(dln->Name)+");"); } Gen.add("std::map<" + keyType->getName() + "," + Name + "*>::iterator\tit = _Map.find("+dln->Name+");"); Gen.add("if (it != _Map.end())"); Gen.add("{"); Gen.add(Name + "*\t__o = (*it).second;"); Gen.add("__o->"+notifyReleaseFunction+"();"); Gen.add("__o->"+unregisterFunction+"();"); Gen.add("__o->"+destroyFunction+"();"); Gen.add("delete __o;"); //Gen.add("_Map.erase(it);"); Gen.add("}"); Gen.startMethod("void", staticLoadFunction, "const "+keyType->getName()+"& "+dln->Name, "map", false, inlineStaticPublic, true); Gen.setDescription("Retrieve an object from the database.\nData are sent asynchronously, so the load callback is called when data are ready.\nUse get() to access to the loaded object."); if (GenerateDebugMessages) { Gen.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": load "+dln->displayPrintfPrefix()+"\", "+dln->displayCppCode(dln->Name)+");"); } Gen.add(pdslibFunc("load")+"("+toString(Id)+", "+dln->toUint64(dln->Name)+");"); Gen.startMethod("void", staticSetLoadCbFunction, "void (*callback)(const "+keyType->getName()+"& key, "+Name+"* object)", "map", false, inlineStaticPublic, true); Gen.setDescription("Setup load callback so client is warned that load succeded or failed."); Gen.add(staticLoadCbAttribute+" = callback;"); Gen.startMethod("void", staticNotifyLoadFailure, "uint64 key", "factories", false, inlineStaticInternal, true); Gen.add("if ("+staticLoadCbAttribute+" != NULL)"); Gen.add("{"); Gen.add(staticLoadCbAttribute+"("+keyType->castFromUser("key")+", NULL);"); Gen.add("}"); Gen.addAttribute("void", staticLoadCbAttribute, "factories", true, "NULL", true, "const "+keyType->getName()+"& key, "+Name+"* object"); // Gen.startMethod("void", staticUnloadFunction, "const " + keyType->getName() + " &" + dln->Name, "map", false, inlineStaticPublic, true); Gen.setDescription("Unload an object from the client memory. Object still exists in database and can be retrieved again using load."); if (GenerateDebugMessages) { Gen.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": unload "+dln->displayPrintfPrefix()+"\", "+dln->displayCppCode(dln->Name)+");"); } Gen.add("std::map<" + keyType->getName() + "," + Name + "*>::iterator\tit = _Map.find("+dln->Name+");"); Gen.add("if (it != _Map.end())"); Gen.add("{"); Gen.add(Name + "*\t__o = (*it).second;"); Gen.add("__o->"+notifyReleaseFunction+"();"); Gen.add("__o->"+destroyFunction+"();"); Gen.add("delete __o;"); //Gen.add("_Map.erase(it);"); Gen.add("}"); } std::string mapType = "std::map<"+keyType->getName()+", "+Name+"*>"; Gen.startMethod(Name+"*", staticGetFunction, "const " + keyType->getName() + " &" + dln->Name, "map", false, inlineStaticPublic, true); Gen.setDescription("Get an object in client. Object must have been previously loaded from database with a load."); if (GenerateDebugMessages) { Gen.add("if (RY_PDS::PDVerbose)\tnldebug(\""+Name+": get "+dln->displayPrintfPrefix()+"\", "+dln->displayCppCode(dln->Name)+");"); } Gen.add(mapType+"::iterator\t__it = _Map.find("+dln->Name+");"); if (GenerateDebugMessages) { Gen.add("if (__it == _Map.end())"); Gen.add("nlwarning(\""+Name+": unable to get %\"NL_I64\"u, not found in map.\", "+dln->toUint64(dln->Name)+");"); } Gen.add("return (__it != _Map.end()) ? (*__it).second : NULL;"); Gen.startMethod(mapType+"::iterator", staticBeginFunction, "", "map", false, inlineStaticPublic, true); Gen.setDescription("Return the begin iterator of the global map of "+Name); Gen.add("return _Map.begin();"); Gen.startMethod(mapType+"::iterator", staticEndFunction, "", "map", false, inlineStaticPublic, true); Gen.setDescription("Return the end iterator of the global map of "+Name); Gen.add("return _Map.end();"); } } // // generate internal management functions // if (HasRowAccess || MappedFlag) { if (MappedFlag) { CDeclarationNode *dln = (MappedFlag ? getKey() : NULL); CTypeNode *keyType = getTypeNode(dln->Type); Gen.addAttribute("std::map<" + keyType->getName() + "," + Name + "*>", "_Map", "inherit map", true); } Gen.addAttribute(indexAllocatorName, "_IndexAllocator", "factories", true); Gen.startMethod("void", staticInitFunction, "", "internal", false, inlineInternal, true, false, "", false); Gen.add(pdslibFunc("setIndexAllocator")+"("+toString(Id)+", _IndexAllocator);"); if (MappedFlag || DerivatedFlag || HasInheritance || ForceReference) { if (DerivatedFlag) { // check factory has been set Gen.add("nlassertex(_FactoryInitialised, (\"User Factory for class "+Name+" not set!\"));"); Gen.add("// factory must have been set by user before database init called!"); Gen.add("// You must provide a factory for the class "+Name+" as it is marked as derived"); Gen.add("// Call "+getDbNode()->Name+"::"+Name+"::"+staticSetUserFactoryFunction+"() with a factory before any call to "+getDbNode()->Name+"::init()!"); } else { Gen.add(staticInitFactoryFunction+"("+staticFactoryFunction+");"); } } getDbNode()->initDb.add(Name+"::"+staticInitFunction+"();"); if (ForceReference) { if (!DerivatedFlag) // forbid factory function for derivated classes { Gen.startMethod(pdBaseDataName+"*", staticFactoryFunction, "", "factories", false, inlineStaticInternal, true); Gen.add("return new " + Name + "();"); } if (Inherited.empty()) { Gen.startMethod("void", staticFetchFunction, pdBaseDataName+" *object, "+CPDataName+" &data", "factories", false, inlineStaticInternal, true); Gen.add(Name + "\t*"+objectVariable+" = static_cast<" + Name + "*>(object);"); Gen.add(objectVariable+"->"+fetchFunction+"(data);"); if (MappedFlag) { //Gen.add("_Map["+objectVariable+"->"+getKey()->getFunc()+"()] = "+objectVariable+";"); if (!HasParent) { Gen.add("if ("+staticLoadCbAttribute+" != NULL)"); Gen.add("{"); Gen.add(staticLoadCbAttribute+"("+objectVariable+"->"+getKey()->getFunc()+"(), "+objectVariable+");"); Gen.add("}"); } } Gen.add(objectVariable+"->"+notifyInitFunction+"();"); } } } // EGS Compat // -- begin ApplyId.add( "else"); ApplyId.add( "{"); ApplyId.add( "nlwarning(\"Skipping unrecognised token: %s\", __pdr.peekNextTokenName().c_str());"); ApplyId.add( "__pdr.skipData();"); ApplyId.add( "}"); ApplyId.add("}"); if (MappedFlag && !HasParent) { ApplyId.add(notifyInitFunction+"();"); } //EGSImplId.add("\n#include \"game_share/persistent_data_template.h\""); //EGSImplId.add("#undef PERSISTENT_CLASS"); //EGSImplId.add("#undef PERSISTENT_DATA"); // -- end set<string>::iterator itf; for (itf=Friends.begin(); itf!=Friends.end(); ++itf) if (*itf != Name) Gen.addOther("friend class "+(*itf)+";\n", "friends"); for (itf=ForwardFriends.begin(); itf!=ForwardFriends.end(); ++itf) if (*itf != Name) Gen.addOther("friend class "+(*itf)+";\n", "friends"); Gen.addOther("friend class "+CPDSLibName+";\n", "friends"); CDbNode* dbNode = getDbNode(); Gen.addOther("friend void "+dbNode->Name+"::init(uint32);\n", "friends"); Gen.flush(hOutput(), cppOutput(), inlineOutput()); DbSummary << "\n\n"; return true; } void CClassNode::generateContentInCall(CCallContext *context) { uint j; for (j=0; j<Nodes.size(); ++j) { CDeclarationNode *dln = dynamic_cast<CDeclarationNode*>(Nodes[j]); if (!dln) continue; dln->generateContent(context); } } // Declaration Node bool CDeclarationNode::prolog() { return true; } bool CDeclarationNode::epilog() { return true; } void CDeclarationNode::generateContent(CCallContext *context) { ClassNode = static_cast<CClassNode*>(Parent); nlassert(Env != NULL); setEnv("name", Name); XmlNode = "name='"+Name+"' id='"+toString(Id)+"' columnid='"+toString(Column)+"' columns='"+toString(Columns)+"'"; if (context == NULL) { CCppOutput& DbSummary = getDbNode()->DbSummary; DbSummary << "Attribute " << Name << ":\n"; } switch (DeclarationType) { case SimpleType: generateTypeContent(context); break; case SimpleClass: generateClassContent(context); break; case BackRef: generateBackRefContent(); break; case ForwardRef: generateForwardRefContent(); break; case ArrayType: generateArrayTypeContent(context); break; case ArrayClass: generateArrayClassContent(context); break; case ArrayRef: generateArrayRefContent(context); break; case Set: generateSetContent(context); break; default: error("Can't decide declaration type"); break; } if (context == NULL) { ClassNode->Gen.separator("methods"); } } std::string CDeclarationNode::getAccessorName(CCallContext *context, const std::string& accessortype, const std::string& sep) { return context->getRootCaller()->Name + sep +accessortype + context->getCallString(); } void CDeclarationNode::generateTypeContent(CCallContext *context) { CClassGenerator& Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; string UCodeContext; if (context != NULL) UCodeContext = context->getUserCodeContext(); string onGetUser = getUserCode("onGet", UCodeContext); string onSetUser = getUserCode("onSet", UCodeContext); string onChangeUser = getUserCode("onChange", UCodeContext); CTypeNode *tnd = getTypeNode(Type); XmlNode += " type='type' typeid='"+toString(tnd->Id)+"'"; setEnv("decltype", "type"); setEnv("type", tnd->getName()); define(IsKey, "iskey"); setEnv("defaultvalue", tnd->getDefaultValue()); setEnv("checkcode", tnd->checkCode(Name)); CCallContext ctx(this); if (context != NULL) ctx = context->getSubContext(this); CClassGenerator &gen = ctx.getRootCaller()->Gen; // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; ApplyId.add("else if (__pdr.peekNextToken() == "+tokenName()+")"); ApplyId.add("{"); tnd->generateApplyCode(ApplyId, tokenName(), cppName()); ApplyId.add("}"); tnd->generateStoreCode(StoreId, tokenName(), cppName()); // -- end CTemplatizerEnv* env = ctx.getRootDeclaration()->Env->nextArrayNode("accessors"); env->set("name", Name); env->set("type", tnd->getName()); env->set("defaultvalue", tnd->getDefaultValue()); env->set("checkcode", tnd->checkCode(Name)); env->define(ctx.getRootCaller()->HasRowAccess, "rowaccess"); env->define(IsKey, "iskey"); env->set("rootcallerid", ctx.getRootCaller()->getId()); env->set("callstr", ctx.getCallString()); env->set("callargs", ctx.getCallArgList()); env->set("callpath", ctx.getCallPath()); env->set("column", ctx.getColumn()); env->set("valuevar", valueVariable); env->set("castcpp", tnd->castToCpp(valueVariable)); env->set("castpds", tnd->castToPDS(valueVariable)); vector<string> checks = ctx.getCheckCode(); for (uint i=0; i<checks.size(); ++i) env->nextArrayNode("checks")->set("check", checks[i]); env->nextArrayNode("checks")->set("check", tnd->checkCode(valueVariable)); if (!onGetUser.empty()) env->set("onget", onGetUser); if (!onSetUser.empty()) env->set("onset", onSetUser); if (!onChangeUser.empty()) env->set("onchange", onChangeUser); // // generate read accessor // if (ctx.getRootCaller()->HasRowAccess) { string arglist = ctx.getCallArgList(); gen.startMethod(tnd->getName(), getFunctionPrefix+ctx.getCallString(), arglist, "methods", true, inlineAccessors); DbSummary << "\t" << getFunctionPrefix+ctx.getCallString() << "\n"; uint i; vector<string> checks = ctx.getCheckCode(); for (i=0; i<checks.size(); ++i) { gen.add(checks[i]); } if (!onGetUser.empty()) { gen.add("{"); gen.add(onGetUser); gen.add("}"); } gen.add("return "+ctx.getCallPath()+";"); } // // generate write accessor // if (ctx.getRootCaller()->HasRowAccess && !IsKey) { string arglist = ctx.getCallArgList(); if (!arglist.empty()) arglist += ", "; arglist += tnd->getName()+" "+valueVariable; gen.startMethod("void", setFunctionPrefix+ctx.getCallString(), appendArg(arglist, "bool forceWrite=false"), "methods", false, inlineAccessors); DbSummary << "\t" << setFunctionPrefix << ctx.getCallString() << "\n"; if (GenerateHAuto) { gen.add("H_AUTO("+getAccessorName(&ctx, setFunctionPrefix, "_")+")"); } if (VerboseMode) { string verbStr; string callStr; verbStr = "nlinfo(\"" + ctx.getRootCaller()->Name + "(%d:%d)::" +setFunctionPrefix + ctx.getCallString() + "("; callStr = ctx.getDebugCallStringFmt(); if (!callStr.empty()) callStr += ", "; callStr += valueVariable+"="+tnd->getPrintfFmt(); verbStr += callStr; verbStr += ")\", __BaseTable, __BaseRow, "; callStr = ctx.getDebugCallStringVal(); if (!callStr.empty()) callStr += ", "; callStr += tnd->getPrintfVal(valueVariable); verbStr += callStr; verbStr += ");"; gen.add(verbStr); } uint i; vector<string> checks = ctx.getCheckCode(); for (i=0; i<checks.size(); ++i) { gen.add(checks[i]); } gen.add(tnd->checkCode(valueVariable)); ///// TYPE CAST if (!onChangeUser.empty()) { gen.add("if ("+ctx.getCallPath()+" != "+tnd->castToCpp(valueVariable)+")"); gen.add("{"); gen.add(onChangeUser); gen.add("}"); } if (!onSetUser.empty()) { gen.add("{"); gen.add(onSetUser); gen.add("}"); } if (WriteTriggerFlag) { gen.add("if (forceWrite && ("+ctx.getCallPath()+" != "+tnd->castToCpp(valueVariable)+"))"); } else { gen.add("if (("+ctx.getCallPath()+" != "+tnd->castToCpp(valueVariable)+") || forceWrite)"); } gen.add("{"); bool useEntityId = ctx.hasRootEntityIdKey(); if (useEntityId) { gen.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(valueVariable)+", "+ctx.getRootCaller()->getKey()->cppName()+");"); } else { gen.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(valueVariable)+");"); } gen.add("}"); gen.add(ctx.getCallPath()+" = "+tnd->castToCpp(valueVariable)+";"); } // // generate attribute // Gen.addAttribute(tnd->getName(), cppName(), "attributes"); string defaultValue; if (!DefaultValue.empty()) defaultValue = DefaultValue; else defaultValue = tnd->getDefaultValue(); // // generate init // if (!IsKey) { InitId.add(cppName()+" = "+defaultValue+";"); } else { InitId.add(tnd->checkCode(Name)); InitId.add(cppName()+" = "+Name+";"); } // // generate create code // // // generate fetch code // if (tnd->isEnum()) FetchId.add("data.serialEnum("+cppName()+");"); else if (tnd->CppType != tnd->StorageType) { FetchId.add("{"); FetchId.add(tnd->StorageType+"\t_v;"); FetchId.add("data.serial(_v);"); FetchId.add(cppName()+" = "+tnd->castToCpp("_v")+";"); FetchId.add("}"); } else { FetchId.add("data.serial("+cppName()+");"); } // // generate clear code // if (ctx.getRootCaller()->HasRowAccess && !IsKey) { ctx.getRootCaller()->ClearId.add(ctx.getCallPath()+" = "+defaultValue+";"); ctx.getRootCaller()->ClearId.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(defaultValue)+");"); } } void CDeclarationNode::generateClassContent(CCallContext *context) { setEnv("decltype", "class"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; CClassNode *cnd = getClassNode(Type); XmlNode += " type='class' classid='"+toString(cnd->Id)+"'"; setEnv("type", Type); // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; StoreId.add("__pdr.pushStructBegin("+tokenName()+");"); StoreId.add(cppName()+".store(__pdr);"); StoreId.add("__pdr.pushStructEnd("+tokenName()+");"); ApplyId.add("else if (__pdr.peekNextToken() == "+tokenName()+")"); ApplyId.add("{"); ApplyId.add("__pdr.popStructBegin("+tokenName()+");"); ApplyId.add(cppName()+".apply(__pdr);"); ApplyId.add("__pdr.popStructEnd("+tokenName()+");"); ApplyId.add("}"); // -- end // // export class accessors into root caller // CCallContext ctx; if (context != NULL) ctx = *context; ctx.Context.push_back(this); cnd->generateContentInCall(&ctx); // // generate attribute // Gen.addAttribute(Type, cppName(), "attributes"); // // generate init // InitId.add(cppName()+"."+initFunction+"();"); // // generate create code // // // generate fetch code // FetchId.add(cppName()+"."+fetchFunction+"(data);"); // // generate clear code // //ClearId.add(cppName()+"."+clearFunction+"();"); } void CDeclarationNode::generateBackRefContent() { setEnv("decltype", "backref"); define(HiddenFlag, "hidden"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; CClassNode *cnd = getClassNode(ParentClass); CDeclarationNode *dnd = cnd->getDeclarationNode(ParentField); CDeclarationNode *knd = (ClassNode->ClassKey.empty() ? NULL : ClassNode->getKey()); XmlNode += " type='backref' classid='"+toString(cnd->Id)+"' backreferentid='"+toString(dnd->Id)+"'"; if (knd != NULL) XmlNode += " key='"+toString(knd->Id)+"'"; setEnv("type", ParentClass); // // generate read accessor // Gen.startMethod(ParentClass+"*", getFunc(), "", "methods", false, inlineAccessors); Gen.add("return "+cppName()+";"); Gen.startMethod("const "+ParentClass+"*", getFunc(), "", "methods", true, inlineAccessors); Gen.add("return "+cppName()+";"); DbSummary << "\t" << getFunc() << "\n"; // // generate write accessor // // // generate attribute // Gen.addAttribute(ParentClass+"*", cppName(), "attributes"); // // generate init // InitId.add(cppName()+" = NULL;"); // // generate create code // // // generate fetch code // FetchId.add(cppName()+" = NULL;"); // // generate set parent code // bool useId = ClassNode->useEntityId(); bool parentUseId = cnd->useEntityId(); if (parentUseId) SetParentId.add("NLMISC::CEntityId\tprevId;"); if (!HiddenFlag) { //SetParentId.add(pdslibFunc("set")+"("+ClassNode->getId()+", getRow(), ("+TColumnIndexName+")("+toString(Column)+"), (__parent != NULL ? "+objectIndexName+"("+(cnd->HasInheritance ? toString("__parent->getTable()") : toString(cnd->Id))+", __parent->getRow()) : "+nullIndexName+"));"); SetParentId.add("if ("+cppName()+" != NULL)"); SetParentId.add("{"); if (parentUseId) SetParentId.add("prevId = "+cppName()+"->"+getFunctionPrefix+cnd->getKey()->Name+"();"); if (ClassNode->getClassKey() == NULL) { SetParentId.add(cppName()+"->"+dnd->unlinkFunc()+"();"); } else { SetParentId.add(cppName()+"->"+dnd->unlinkFunc()+"("+ClassNode->getKey()->cppName()+");"); } SetParentId.add("}"); } else { if (parentUseId) { SetParentId.add("if ("+cppName()+" != NULL)"); SetParentId.add("{"); SetParentId.add("prevId = "+cppName()+"->"+cnd->getKey()->cppName()+";"); SetParentId.add("}"); } } string oeid; string peid; if (useId) oeid = ", "+ClassNode->getKey()->cppName(); if (parentUseId) peid = ", ("+cppName()+" != NULL ? "+cppName()+"->"+getFunctionPrefix+cnd->getKey()->Name+"() : NLMISC::CEntityId::Unknown), prevId"; SetParentId.add(cppName()+" = __parent;"); SetParentId.add(pdslibFunc("setParent")+"("+ClassNode->getId()+", getRow(), ("+TColumnIndexName+")("+toString(Column)+"), (__parent != NULL ? "+objectIndexName+"("+(cnd->HasInheritance ? toString("__parent->getTable()") : toString(cnd->Id))+", __parent->getRow()) : "+nullIndexName+")"+oeid+peid+");"); SetUnnotifiedParentId.add(cppName()+" = __parent;"); } void CDeclarationNode::generateClassPtrApplyCode(const std::string& value) { CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; ApplyId.add(value+" = NULL;"); ApplyId.add("if (__pdr.peekNextToken() == __Tok_ClassName)"); ApplyId.add("{"); ApplyId.add( "std::string\t__className;"); ApplyId.add( "__pdr.pop(__Tok_ClassName, __className);"); ApplyId.add( value+" = "+Type+"::cast("+pdslibFunc("create")+"(__className));"); ApplyId.add( "if ("+value+" != NULL)"); ApplyId.add( "{"); ApplyId.add( "__pdr.popStructBegin("+tokenName()+");"); ApplyId.add( value+"->apply(__pdr);"); ApplyId.add( value+"->"+setUnnotifiedParentFunction+"(this);"); ApplyId.add( "__pdr.popStructEnd("+tokenName()+");"); ApplyId.add( "}"); ApplyId.add( "else"); ApplyId.add( "{"); ApplyId.add( "__pdr.skipStruct();"); ApplyId.add( "}"); ApplyId.add("}"); } void CDeclarationNode::generateClassPtrStoreCode(const std::string& value) { CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; StoreId.add("if ("+value+" != NULL)"); StoreId.add("{"); StoreId.add( "std::string\t__className = "+pdslibFunc("getClassName")+"("+value+");"); StoreId.add( "__pdr.push(__Tok_ClassName, __className);"); StoreId.add( "__pdr.pushStructBegin("+tokenName()+");"); StoreId.add( value+"->store(__pdr);"); StoreId.add( "__pdr.pushStructEnd("+tokenName()+");"); StoreId.add("}"); } void CDeclarationNode::generateForwardRefContent() { setEnv("decltype", "forwardref"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &NotifyInitId = ClassNode->NotifyInitId; CClassGenerator::SMethodId &NotifyReleaseId = ClassNode->NotifyReleaseId; CClassNode *cnd = getClassNode(Type); CDeclarationNode *dnd = cnd->getDeclarationNode(ForwardRefAttribute); CDeclarationNode *knd = (cnd->ClassKey.empty() ? NULL : cnd->getKey()); XmlNode += " type='forwardref' classid='"+toString(cnd->Id)+"' forwardreferedid='"+toString(dnd->Id)+"'"; if (knd != NULL) XmlNode += " key='"+toString(knd->Id)+"'"; setEnv("type", Type); // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; StoreId.add("// store "+Name); StoreId.add("__pdr.pushStructBegin("+tokenName()+");"); generateClassPtrStoreCode(cppName()); StoreId.add("__pdr.pushStructEnd("+tokenName()+");"); ApplyId.add("// apply "+Name); ApplyId.add("else if (__pdr.peekNextToken() == "+tokenName()+")"); ApplyId.add("{"); ApplyId.add( "__pdr.popStructBegin("+tokenName()+");"); generateClassPtrApplyCode(cppName()); ApplyId.add( "__pdr.popStructEnd("+tokenName()+");"); ApplyId.add("}"); // -- end // // generate read accessor // Gen.startMethod(Type+"*", getFunc(), "", "methods", false, inlineAccessors); Gen.add("return "+cppName()+";"); Gen.startMethod("const "+Type+"*", getFunc(), "", "methods", true, inlineAccessors); Gen.add("return "+cppName()+";"); DbSummary << "\t" << getFunc() << "\n"; // // generate write accessor // Gen.startMethod("void", setFunc(), Type+"* "+valueVariable, "methods", false, inlineAccessors); Gen.add("if ("+cppName()+" != NULL)"); Gen.add("{"); Gen.add(cppName()+"->"+setParentFunction+"(NULL);"); Gen.add("}"); Gen.add(valueVariable+"->"+setParentFunction+"(this);"); Gen.add(cppName()+" = "+valueVariable+";"); DbSummary << "\t" << setFunc() << "\n"; // // generate attribute // Gen.addAttribute(Type+"*", cppName(), "attributes"); // // generate init // InitId.add(cppName()+" = NULL;"); // // generate destroy code // DestroyId.add("if ("+cppName()+" != NULL)"); DestroyId.add("{"); DestroyId.add(Type+"*\t__o = "+cppName()+";"); DestroyId.add("__o->"+destroyFunction+"();"); DestroyId.add("delete __o;"); DestroyId.add("}"); // // generate create code // // // generate fetch code // FetchId.add("// read table and row, create an object, affect to the ref, and fetch it"); if (!ClassNode->tableAndRowIndicesUsedInFetch) { FetchId.add(TTableIndexName+"\ttableIndex;\n"+TRowIndexName+"\trowIndex;"); ClassNode->tableAndRowIndicesUsedInFetch = true; } FetchId.add(cppName()+" = NULL;"); FetchId.add("data.serial(tableIndex, rowIndex);"); FetchId.add("if (rowIndex != "+INVALID_ROW_INDEXName+" && tableIndex != "+INVALID_TABLE_INDEXName+")"); FetchId.add("{"); FetchId.add(cppName()+" = static_cast<"+Type+"*>("+pdslibFunc("create")+"(tableIndex));"); FetchId.add(pdslibFunc("setRowIndex")+"(rowIndex, "+cppName()+");"); FetchId.add(cppName()+"->"+fetchFunction+"(data);"); FetchId.add(cppName()+"->"+setUnnotifiedParentFunction+"(this);"); FetchId.add("}"); // // generate register/unregister code // UnregisterAttributesId.add("if ("+cppName()+" != NULL)"); UnregisterAttributesId.add("{"); UnregisterAttributesId.add(Type+"*\t"+objectVariable+" = "+cppName()+";"); UnregisterAttributesId.add(objectVariable+"->"+unregisterFunction+"();"); UnregisterAttributesId.add(objectVariable+"->"+destroyFunction+"();"); UnregisterAttributesId.add("delete "+objectVariable+";"); UnregisterAttributesId.add("}"); // // generate init/release notification // NotifyInitId.add("if ("+cppName()+" != NULL)"); NotifyInitId.add("{"); NotifyInitId.add(cppName()+"->"+notifyInitFunction+"();"); NotifyInitId.add("}"); NotifyReleaseId.add("if ("+cppName()+" != NULL)"); NotifyReleaseId.add("{"); NotifyReleaseId.add(cppName()+"->"+notifyReleaseFunction+"();"); NotifyReleaseId.add("}"); // // generate unlink code // string unlinkProto; if (cnd->getClassKey() != NULL) { CDeclarationNode* kd = cnd->getClassKey(); CTypeNode* keyType = getTypeNode(kd->Type); unlinkProto = keyType->getName()+" dummy"; } Gen.startMethod("void", unlinkFunc(), unlinkProto, "internal", false, inlineInternal); Gen.add("{"); Gen.add(cppName()+" = NULL;"); Gen.add("}"); // // generate clear code // //ClearId.add(cppName()+"->"+setParentFunction+"(NULL);"); ClearId.add(Type+"*\t"+objectVariable+" = "+cppName()+";"); ClearId.add(objectVariable+"->"+unregisterFunction+"();"); ClearId.add(objectVariable+"->"+destroyFunction+"();"); ClearId.add("delete "+objectVariable+";"); } void CDeclarationNode::generateArrayApplyCode() { CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; ApplyId.add("// apply "+Name); ApplyId.add("else if (__pdr.peekNextToken() == "+tokenName()+")"); ApplyId.add("{"); ApplyId.add( "__pdr.popStructBegin("+tokenName()+");"); ApplyId.add( "uint\tindex = 0;"); ApplyId.add( "while (!__pdr.isEndOfStruct())"); ApplyId.add( "{"); CIndexNode *ind = getIndexNode(ArrayIndex); if (ind->isEnum()) { ApplyId.add("std::string\tindexname;"); ApplyId.add("__pdr.pop(__Tok_MapKey, indexname);"); ApplyId.add("index = "+ind->getFromStringCode("indexname")+";"); } } void CDeclarationNode::generateArrayStoreCode() { CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; StoreId.add("// store "+Name); StoreId.add("__pdr.pushStructBegin("+tokenName()+");"); CIndexNode *ind = getIndexNode(ArrayIndex); StoreId.add("for (uint index=0; index<"+ind->getSizeName()+"; ++index)"); StoreId.add("{"); if (ind->isEnum()) { StoreId.add("std::string\tindexname = "+ind->getToStringCode(ind->castFromUser("index"))+";"); StoreId.add("__pdr.push(__Tok_MapKey, indexname);"); } } void CDeclarationNode::generateArrayEndCode() { CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; ApplyId.add( "++index;"); ApplyId.add( "}"); ApplyId.add( "__pdr.popStructEnd("+tokenName()+");"); ApplyId.add("}"); StoreId.add("}"); StoreId.add("__pdr.pushStructEnd("+tokenName()+");"); } void CDeclarationNode::generateArrayTypeContent(CCallContext *context) { setEnv("decltype", "arraytype"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; string UCodeContext; if (context != NULL) UCodeContext = context->getUserCodeContext(); string onGetUser = getUserCode("onGet", UCodeContext); string onSetUser = getUserCode("onSet", UCodeContext); string onChangeUser = getUserCode("onChange", UCodeContext); CTypeNode *tnd = getTypeNode(Type); CIndexNode *ind = getIndexNode(ArrayIndex); XmlNode += " type='arraytype' typeid='"+toString(tnd->Id)+"' indexid='"+toString(ind->Id)+"'"; CCallContext ctx(this); if (context != NULL) ctx = context->getSubContext(this); CClassGenerator &gen = ctx.getRootCaller()->Gen; setEnv("type", tnd->getName()); setEnv("indexsize", ind->getSizeName()); // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; generateArrayApplyCode(); generateArrayStoreCode(); ApplyId.add(tnd->getName()+"\tvalue;"); tnd->generateApplyCode(ApplyId, "__Tok_MapVal", "value"); ApplyId.add("if (index != "+toString(ind->getSize())+")"); ApplyId.add("{"); ApplyId.add( cppName()+"[index] = value;"); ApplyId.add("}"); tnd->generateStoreCode(ApplyId, "__Tok_MapVal", cppName()+"[index]"); generateArrayEndCode(); // -- end // // generate read accessor // if (ctx.getRootCaller()->HasRowAccess) { string arglist = ctx.getCallArgList(); ctx.getRootCaller()->Gen.startMethod(tnd->getName(), getFunctionPrefix+ctx.getCallString(), arglist, "methods", true, inlineAccessors); DbSummary << "\t" << getFunctionPrefix+ctx.getCallString() << "\n"; uint i; vector<string> checks = ctx.getCheckCode(); for (i=0; i<checks.size(); ++i) gen.add(checks[i]); if (!onGetUser.empty()) { gen.add("{"); gen.add(onGetUser); gen.add("}"); } gen.add("return "+ctx.getCallPath()+";"); } // // generate write accessor // if (ctx.getRootCaller()->HasRowAccess) { string arglist = ctx.getCallArgList(); if (!arglist.empty()) arglist += ", "; arglist += tnd->getName()+" "+valueVariable; gen.startMethod("void", setFunctionPrefix+ctx.getCallString(), appendArg(arglist, "bool forceWrite=false"), "methods", false, inlineAccessors); DbSummary << "\t" << setFunctionPrefix+ctx.getCallString() << "\n"; if (GenerateHAuto) { gen.add("H_AUTO("+getAccessorName(&ctx, getFunctionPrefix, "_")+")"); } if (VerboseMode) { string verbStr; string callStr; verbStr = "nlinfo(\"" + ctx.getRootCaller()->Name + "(%d:%d)::" +setFunctionPrefix + ctx.getCallString() + "("; callStr = ctx.getDebugCallStringFmt(); if (!callStr.empty()) callStr += ", "; callStr += valueVariable+"="+tnd->getPrintfFmt(); verbStr += callStr; verbStr += ")\", __BaseTable, __BaseRow, "; callStr = ctx.getDebugCallStringVal(); if (!callStr.empty()) callStr += ", "; callStr += tnd->getPrintfVal(valueVariable); verbStr += callStr; verbStr += ");"; gen.add(verbStr); } uint i; vector<string> checks = ctx.getCheckCode(); for (i=0; i<checks.size(); ++i) gen.add(checks[i]); gen.add(tnd->checkCode(valueVariable)); if (!onChangeUser.empty()) { gen.add("if ("+ctx.getCallPath()+" != "+tnd->castToCpp(valueVariable)+")"); gen.add("{"); gen.add(onChangeUser); gen.add("}"); } if (!onSetUser.empty()) { gen.add("{"); gen.add(onGetUser); gen.add("}"); } if (WriteTriggerFlag) { gen.add("if (forceWrite && ("+ctx.getCallPath()+" != "+tnd->castToCpp(valueVariable)+"))"); } else { gen.add("if (("+ctx.getCallPath()+" != "+tnd->castToCpp(valueVariable)+") || forceWrite)"); } gen.add("{"); bool useEntityId = ctx.hasRootEntityIdKey(); if (useEntityId) { gen.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(valueVariable)+", "+ctx.getRootCaller()->getKey()->cppName()+");"); } else { gen.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(valueVariable)+");"); } //gen.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(valueVariable)+");"); gen.add("}"); gen.add(ctx.getCallPath()+" = "+tnd->castToCpp(valueVariable)+";"); } // // generate attribute // Gen.addAttribute(tnd->getName(), cppName()+"["+ind->getSizeName()+"]", "attributes"); // // generate init // if (!ClassNode->indexUsedInInit) { InitId.add("uint\ti;"); ClassNode->indexUsedInInit = true; } InitId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)\t"+cppName()+"[i] = "+tnd->getDefaultValue()+";"); // // generate create code // // // generate fetch code // if (!ClassNode->indexUsedInFetch) { FetchId.add("uint\ti;"); ClassNode->indexUsedInFetch = true; } if (tnd->isEnum()) FetchId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)\tdata.serialEnum("+cppName()+"[i]);"); else if (tnd->CppType != tnd->StorageType) { FetchId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); FetchId.add("{"); FetchId.add(tnd->StorageType+"\t_v;"); FetchId.add("data.serial(_v);"); FetchId.add(cppName()+"[i] = "+tnd->castToCpp("_v")+";"); FetchId.add("}"); } else { FetchId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)\tdata.serial("+cppName()+"[i]);"); } // // generate clear code // if (ctx.getRootCaller()->HasRowAccess) { string forIndex = "__i"+toString(ctx.getContextIndex()); ctx.getRootCaller()->ClearId.add("for (uint "+forIndex+"=0; "+forIndex+"<"+ind->getSizeName()+"; ++"+forIndex+")"); ctx.getRootCaller()->ClearId.add("{"); ctx.getRootCaller()->ClearId.add(ctx.getCallPath()+" = "+tnd->getDefaultValue()+";"); ctx.getRootCaller()->ClearId.add(pdslibFunc("set")+"("+ctx.getRootCaller()->getId()+", __BaseRow, ("+TColumnIndexName+")("+ctx.getColumn()+"), "+tnd->castToPDS(tnd->getDefaultValue())+");"); ctx.getRootCaller()->ClearId.add("}"); } } void CDeclarationNode::generateArrayClassContent(CCallContext *context) { setEnv("decltype", "arrayclass"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; CClassNode *sub = getClassNode(Type); CIndexNode *ind = getIndexNode(ArrayIndex); XmlNode += " type='arrayclass' classid='"+toString(sub->Id)+"' indexid='"+toString(ind->Id)+"'"; setEnv("type", Type); setEnv("indexsize", ind->getSizeName()); // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; generateArrayApplyCode(); generateArrayStoreCode(); // StoreId.add("__pdr.pushStructBegin("+tokenName()+");"); StoreId.add("__pdr.pushStructBegin(__Tok_MapVal);"); StoreId.add(cppName()+"[index].store(__pdr);"); StoreId.add("__pdr.pushStructEnd(__Tok_MapVal);"); // StoreId.add("__pdr.pushStructEnd("+tokenName()+");"); ApplyId.add("if (index < "+ind->getSizeName()+")"); ApplyId.add("{"); // ApplyId.add("__pdr.popStructBegin("+tokenName()+");"); ApplyId.add("__pdr.popStructBegin(__pdr.peekNextToken());"); ApplyId.add(cppName()+"[index].apply(__pdr);"); ApplyId.add("__pdr.popStructEnd(__pdr.peekNextToken());"); // ApplyId.add("__pdr.popStructEnd("+tokenName()+");"); ApplyId.add("}"); ApplyId.add("else"); ApplyId.add("{"); ApplyId.add( "__pdr.skipStruct();"); ApplyId.add("}"); generateArrayEndCode(); // -- end // // export class accessors into root caller // CCallContext ctx; if (context != NULL) ctx = *context; ctx.Context.push_back(this); if (ctx.getRootCaller()->HasRowAccess) { string forIndex = "__i"+toString(ctx.getContextIndex()); ctx.getRootCaller()->ClearId.add("for (uint "+forIndex+"=0; "+forIndex+"<"+ind->getSizeName()+"; ++"+forIndex+")"); ctx.getRootCaller()->ClearId.add("{"); } sub->generateContentInCall(&ctx); if (ctx.getRootCaller()->HasRowAccess) { ctx.getRootCaller()->ClearId.add("}"); } // // generate attribute // Gen.addAttribute(Type, cppName()+"["+ind->getSizeName()+"]", "attributes"); // // generate init // // // generate create code // // // generate fetch code // if (!ClassNode->indexUsedInFetch) { FetchId.add("uint\ti;"); ClassNode->indexUsedInFetch = true; } FetchId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)\t"+cppName()+"[i]."+fetchFunction+"(data);"); // // generate clear code // } void CDeclarationNode::generateArrayRefContent(CCallContext *context) { setEnv("decltype", "arrayref"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; CClassGenerator::SMethodId &NotifyInitId = ClassNode->NotifyInitId; CClassGenerator::SMethodId &NotifyReleaseId = ClassNode->NotifyReleaseId; CClassNode *cnd = getClassNode(Type); bool useReference = cnd->ForceReference; CDeclarationNode *dnd = cnd->getDeclarationNode(ForwardRefAttribute); CIndexNode *ind = getIndexNode(ArrayIndex); CDeclarationNode *knd = cnd->getKey(); XmlNode += " type='arrayref' classid='"+toString(cnd->Id)+"' forwardreferedid='"+toString(dnd->Id)+"' key='"+toString(knd->Id)+"' indexid='"+toString(ind->Id)+"' allownull='"+(useReference ? "true" : "false")+"'"; string arrayType = Type+(useReference ? "*" : ""); string access = (useReference ? "->" : "."); setEnv("type", Type); setEnv("indexsize", ind->getSizeName()); define(useReference, "useref"); string UCodeContext; if (context != NULL) UCodeContext = context->getUserCodeContext(); string onChangeUser = getUserCode("onChange", UCodeContext); // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; generateArrayStoreCode(); generateArrayApplyCode(); StoreId.add("__pdr.pushStructBegin("+tokenName()+");"); ApplyId.add("if (index < "+ind->getSizeName()+")"); ApplyId.add("{"); ApplyId.add( "if (__pdr.peekNextToken() == "+tokenName()+")"); ApplyId.add( "{"); ApplyId.add( "__pdr.popStructBegin("+tokenName()+");"); if (useReference) { generateClassPtrStoreCode(cppName()+"[index]"); generateClassPtrApplyCode(cppName()+"[index]"); } else { StoreId.add(cppName()+"[index].store(__pdr);"); ApplyId.add(cppName()+"[index].apply(__pdr);"); ApplyId.add(cppName()+"[index]."+setUnnotifiedParentFunction+"(this);"); } StoreId.add("__pdr.pushStructEnd("+tokenName()+");"); ApplyId.add( "__pdr.popStructEnd("+tokenName()+");"); ApplyId.add( "}"); ApplyId.add("}"); ApplyId.add("else"); ApplyId.add("{"); ApplyId.add( "__pdr.skipStruct();"); ApplyId.add("}"); generateArrayEndCode(); // -- end // // generate read accessor // Gen.startMethod((useReference ? arrayType : Type+"&"), getFunc(), ind->getName()+" "+indexVariable, "methods", false, inlineAccessors); Gen.add(ind->checkCode(indexVariable)); Gen.add("return "+cppName()+"["+indexVariable+"];"); Gen.startMethod(string("const ")+(useReference ? arrayType : Type+"&"), getFunc(), ind->getName()+" "+indexVariable, "methods", true, inlineAccessors); Gen.add(ind->checkCode(indexVariable)); Gen.add("return "+cppName()+"["+indexVariable+"];"); DbSummary << "\t" << getFunc() << "\n"; // // generate write accessor // if (useReference) { Gen.startMethod("void", setFunc(), arrayType+" "+valueVariable, "methods", false, inlineAccessors); DbSummary << "\t" << setFunc() << "\n"; Gen.add("if ("+valueVariable+" == NULL)\treturn;"); Gen.add(ind->getName()+"\t"+keyVariable+" = "+valueVariable+"->"+cnd->getKey()->getFunc()+"();"); Gen.add(ind->checkCode(keyVariable)); Gen.add(Type+"*\t__prev = "+cppName()+"["+keyVariable+"];"); Gen.add("if (__prev != NULL)"); Gen.add("{"); Gen.add("__prev->"+setParentFunction+"(NULL);"); if (cnd->MapClass == NULL) { Gen.add("__prev->"+unregisterFunction+"();"); Gen.add("__prev->"+destroyFunction+"();"); Gen.add("delete __prev;"); } Gen.add("}"); Gen.add(valueVariable+"->"+setParentFunction+"(this);"); Gen.add(cppName()+"["+keyVariable+"] = "+valueVariable+";"); } if (cnd->MapClass == NULL && useReference && !cnd->HasInheritance && !cnd->DerivatedFlag) { Gen.startMethod(Type+"*", newFunc(), ind->getName()+" "+indexVariable, "methods", false, inlineAccessors); DbSummary << "\t" << newFunc() << "\n"; Gen.add(ind->checkCode(indexVariable)); Gen.add(Type+"*\t"+objectVariable+" = new "+Type+"();"); Gen.add(objectVariable+"->"+initFunction+"("+indexVariable+");"); Gen.add(objectVariable+"->"+registerFunction+"();"); Gen.add(setFunc()+"("+objectVariable+");"); } if (cnd->MapClass == NULL && useReference) { Gen.startMethod("void", deleteFunc(), ind->getName()+" "+indexVariable, "methods", false, inlineAccessors); DbSummary << "\t" << deleteFunc() << "\n"; Gen.add(ind->checkCode(indexVariable)); Gen.add(Type+"*\t"+objectVariable+" = "+cppName()+"["+indexVariable+"];"); Gen.add(objectVariable+"->"+setParentFunction+"(NULL);"); Gen.add(objectVariable+"->"+unregisterFunction+"();"); Gen.add(objectVariable+"->"+destroyFunction+"();"); Gen.add("delete "+objectVariable+";"); } // // generate attribute // Gen.addAttribute(arrayType, cppName()+"["+ind->getSizeName()+"]", "attributes"); // // generate init // if (!ClassNode->indexUsedInInit) { InitId.add("uint\ti;"); ClassNode->indexUsedInInit = true; } if (useReference) { InitId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)\t"+cppName()+"[i] = NULL;"); } else { InitId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)\t"+cppName()+"[i]."+initFunction+"(("+ind->getName()+")i);"); } // // generate destroy code // if (!ClassNode->indexUsedInDestroy) { DestroyId.add("uint\ti;"); ClassNode->indexUsedInDestroy = true; } DestroyId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); DestroyId.add("{"); if (useReference) { DestroyId.add("if ("+cppName()+"[i] != NULL)"); DestroyId.add("{"); DestroyId.add(Type+"*\t"+objectVariable+" = "+cppName()+"[i];"); DestroyId.add(objectVariable+"->"+destroyFunction+"();"); DestroyId.add("delete "+objectVariable+";"); DestroyId.add("}"); } else { DestroyId.add(cppName()+"[i]."+destroyFunction+"();"); } DestroyId.add("}"); // // generate create code // // // generate fetch code // if (useReference) { if (!ClassNode->indexUsedInFetch) { FetchId.add("uint\ti;"); ClassNode->indexUsedInFetch = true; } if (!ClassNode->tableAndRowIndicesUsedInFetch) { FetchId.add(TTableIndexName+"\ttableIndex;\n"+TRowIndexName+"\trowIndex;"); ClassNode->tableAndRowIndicesUsedInFetch = true; } FetchId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); FetchId.add("{"); FetchId.add(cppName()+"[i] = NULL;"); FetchId.add("// read table and row, create an object, affect to the ref, and fetch it"); FetchId.add("data.serial(tableIndex, rowIndex);"); FetchId.add("if (rowIndex != "+INVALID_ROW_INDEXName+" && tableIndex != "+INVALID_TABLE_INDEXName+")"); FetchId.add("{"); FetchId.add(cppName()+"[i] = static_cast<"+Type+"*>("+pdslibFunc("create")+"(tableIndex));"); FetchId.add(pdslibFunc("setRowIndex")+"(rowIndex, "+cppName()+"[i]);"); FetchId.add(cppName()+"[i]->"+fetchFunction+"(data);"); FetchId.add(cppName()+"[i]->"+setUnnotifiedParentFunction+"(this);"); FetchId.add("}"); FetchId.add("}"); } else { if (!ClassNode->indexUsedInFetch) { FetchId.add("uint\ti;"); ClassNode->indexUsedInFetch = true; } if (!ClassNode->tableAndRowIndicesUsedInFetch) { FetchId.add(TTableIndexName+"\ttableIndex;\n"+TRowIndexName+"\trowIndex;"); ClassNode->tableAndRowIndicesUsedInFetch = true; } FetchId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); FetchId.add("{"); FetchId.add("// read table and row, create an object, affect to the ref, and fetch it"); FetchId.add("data.serial(tableIndex, rowIndex);"); FetchId.add("if (rowIndex != "+INVALID_ROW_INDEXName+" && tableIndex != "+INVALID_TABLE_INDEXName+")"); FetchId.add("{"); FetchId.add(pdslibFunc("setRowIndex")+"(rowIndex, &"+cppName()+"[i]);"); FetchId.add(cppName()+"[i]."+fetchFunction+"(data);"); FetchId.add(cppName()+"[i]."+setUnnotifiedParentFunction+"(this);"); FetchId.add("}"); FetchId.add("}"); } // // generate register/unregister code // if (!useReference) { if (!ClassNode->indexUsedInRegister) { RegisterAttributesId.add("uint\ti;"); ClassNode->indexUsedInRegister = true; } RegisterAttributesId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); RegisterAttributesId.add("{"); RegisterAttributesId.add(cppName()+"[i]."+registerFunction+"();"); RegisterAttributesId.add(cppName()+"[i]."+setParentFunction+"(this);"); RegisterAttributesId.add("}"); if (!ClassNode->indexUsedInUnregister) { UnregisterAttributesId.add("uint\ti;"); ClassNode->indexUsedInUnregister = true; } UnregisterAttributesId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); UnregisterAttributesId.add("{"); UnregisterAttributesId.add(cppName()+"[i]."+unregisterFunction+"();"); UnregisterAttributesId.add("}"); } else { if (!ClassNode->indexUsedInUnregister) { UnregisterAttributesId.add("uint\ti;"); ClassNode->indexUsedInUnregister = true; } UnregisterAttributesId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); UnregisterAttributesId.add("{"); UnregisterAttributesId.add("if ("+cppName()+"[i] != NULL)"); UnregisterAttributesId.add("{"); UnregisterAttributesId.add(Type+"*\t"+objectVariable+" = "+cppName()+"[i];"); UnregisterAttributesId.add(objectVariable+"->"+unregisterFunction+"();"); UnregisterAttributesId.add(objectVariable+"->"+destroyFunction+"();"); UnregisterAttributesId.add("delete "+objectVariable+";"); UnregisterAttributesId.add("}"); UnregisterAttributesId.add("}"); } // // generate init/release notification // if (!ClassNode->indexUsedInNotifyInit) { NotifyInitId.add("uint\ti;"); ClassNode->indexUsedInNotifyInit = true; } NotifyInitId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); NotifyInitId.add("{"); if (!ClassNode->indexUsedInNotifyRelease) { NotifyReleaseId.add("uint\ti;"); ClassNode->indexUsedInNotifyRelease = true; } NotifyReleaseId.add("for (i=0; i<"+ind->getSizeName()+"; ++i)"); NotifyReleaseId.add("{"); if (!useReference) { NotifyInitId.add(cppName()+"[i]."+notifyInitFunction+"();"); NotifyReleaseId.add(cppName()+"[i]."+notifyReleaseFunction+"();"); } else { NotifyInitId.add("if ("+cppName()+"[i] != NULL)"); NotifyInitId.add("{"); NotifyInitId.add(cppName()+"[i]->"+notifyInitFunction+"();"); NotifyInitId.add("}"); NotifyReleaseId.add("if ("+cppName()+"[i] != NULL)"); NotifyReleaseId.add("{"); NotifyReleaseId.add(cppName()+"[i]->"+notifyReleaseFunction+"();"); NotifyReleaseId.add("}"); } NotifyInitId.add("}"); NotifyReleaseId.add("}"); // // generate unlink code // if (useReference) { Gen.startMethod("void", unlinkFunc(), ind->getName()+" "+keyVariable, "internal", false, inlineInternal); Gen.add(cppName()+"["+keyVariable+"] = NULL;"); } // // generate clear code // ClearId.add("for (uint i=0; i<"+ind->getSizeName()+"; ++i)"); ClearId.add("{"); if (!useReference) { ClearId.add(cppName()+"[i]."+clearFunction+"();"); } else { //ClearId.add(cppName()+"[i]->"+setParentFunction+"(NULL);"); ClearId.add(Type+"*\t"+objectVariable+" = "+cppName()+"[i];"); ClearId.add(objectVariable+"->"+unregisterFunction+"();"); ClearId.add(objectVariable+"->"+destroyFunction+"();"); ClearId.add("delete "+objectVariable+";"); } ClearId.add("}"); } void CDeclarationNode::generateSetContent(CCallContext *context) { setEnv("decltype", "set"); CClassGenerator &Gen = ClassNode->Gen; CCppOutput& DbSummary = getDbNode()->DbSummary; CClassGenerator::SMethodId &InitId = ClassNode->InitId; CClassGenerator::SMethodId &ClearId = ClassNode->ClearId; CClassGenerator::SMethodId &DestroyId = ClassNode->DestroyId; CClassGenerator::SMethodId &FetchId = ClassNode->FetchId; CClassGenerator::SMethodId &RegisterId = ClassNode->RegisterId; CClassGenerator::SMethodId &RegisterAttributesId = ClassNode->RegisterAttributesId; CClassGenerator::SMethodId &UnregisterId = ClassNode->UnregisterId; CClassGenerator::SMethodId &UnregisterAttributesId = ClassNode->UnregisterAttributesId; CClassGenerator::SMethodId &SetParentId = ClassNode->SetParentId; CClassGenerator::SMethodId &SetUnnotifiedParentId = ClassNode->SetUnnotifiedParentId; CClassGenerator::SMethodId &NotifyInitId = ClassNode->NotifyInitId; CClassGenerator::SMethodId &NotifyReleaseId = ClassNode->NotifyReleaseId; CClassNode *sub = getClassNode(Type); CDeclarationNode *key = sub->getKey(); CTypeNode *keyType = getTypeNode(key->Type); bool useReference = sub->ForceReference; string access = (useReference ? "->" : "."); string onChangeUser = getUserCode("onChange", context != NULL ? context->getUserCodeContext() : ""); string setType = "std::map<"+keyType->getName()+", " + Type + (useReference ? "*" : "") + ">"; CDeclarationNode *dnd = sub->getDeclarationNode(ForwardRefAttribute); XmlNode += " type='set' classid='"+toString(sub->Id)+"' forwardreferedid='"+toString(dnd->Id)+"' key='"+toString(key->Id)+"' "; setEnv("type", Type); setEnv("keytype", keyType->getName()); define(useReference, "useref"); // EGS Compat // -- begin CClassGenerator::SMethodId& StoreId = ClassNode->StoreId; CClassGenerator::SMethodId& ApplyId = ClassNode->ApplyId; StoreId.add("// store "+Name); StoreId.add("__pdr.pushStructBegin("+tokenName()+");"); StoreId.add("for ("+setType+"::const_iterator it="+cppName()+".begin(); it!="+cppName()+".end(); ++it)"); StoreId.add("{"); StoreId.add( keyType->getName()+"\tkey = (*it).first;"); keyType->generateStoreCode(StoreId, "__Tok_MapKey", "key"); StoreId.add( "__pdr.pushStructBegin(__Tok_MapVal);"); if (useReference) { generateClassPtrStoreCode("(*it).second"); } else { StoreId.add("(*it).second.store(__pdr);"); } StoreId.add( "__pdr.pushStructEnd(__Tok_MapVal);"); StoreId.add("}"); StoreId.add("__pdr.pushStructEnd("+tokenName()+");"); StoreId.add("// end of store "+Name); ApplyId.add("// apply "+Name); ApplyId.add("else if (__pdr.peekNextToken() == "+tokenName()+")"); ApplyId.add("{"); ApplyId.add( "__pdr.popStructBegin("+tokenName()+");"); ApplyId.add( "while (!__pdr.isEndOfStruct())"); ApplyId.add( "{"); ApplyId.add( keyType->getName()+"\tkey;"); keyType->generateApplyCode(ApplyId, "__Tok_MapKey", "key"); ApplyId.add( "__pdr.popStructBegin(__Tok_MapVal);"); if (useReference) { ApplyId.add( Type+"*\tobj;"); generateClassPtrApplyCode("obj"); ApplyId.add( "if (obj != NULL)"); ApplyId.add( "{"); ApplyId.add( cppName()+"[key] = obj;"); ApplyId.add( "}"); } else { ApplyId.add( Type+"&\tobj = "+cppName()+"[key];"); ApplyId.add( "obj.apply(__pdr);"); ApplyId.add( "obj."+setUnnotifiedParentFunction+"(this);"); } ApplyId.add( "__pdr.popStructEnd(__Tok_MapVal);"); ApplyId.add( "}"); ApplyId.add( "__pdr.popStructEnd("+tokenName()+");"); ApplyId.add("}"); ApplyId.add("// end of apply "+Name); // -- end // // generate read accessor // Gen.startMethod(Type+"*", getFunc(), "const "+keyType->getName()+"& "+keyVariable, "methods", false, inlineAccessors); DbSummary << "\t" << getFunc() << "\n"; Gen.add(setType+"::iterator _it = "+cppName()+".find("+keyVariable+");"); Gen.add("return (_it=="+cppName()+".end() ? NULL : "+(useReference ? "(*_it).second" : "&((*_it).second)")+");"); Gen.startMethod("const "+Type+"*", getFunc(), "const "+keyType->getName()+"& "+keyVariable, "methods", true, inlineAccessors); Gen.add(setType+"::const_iterator _it = "+cppName()+".find("+keyVariable+");"); Gen.add("return (_it=="+cppName()+".end() ? NULL : "+(useReference ? "(*_it).second" : "&((*_it).second)")+");"); // generate map accessor string stypedef = "T"+Name+"Map"; //Gen.addOther("typedef "+setType+"\t"+stypedef+";"); Gen.startMethod(setType+"::iterator", getFunc()+"Begin", "", "methods", false, inlineAccessors); Gen.add("return "+cppName()+".begin();"); DbSummary << "\t" << getFunc() << "Begin" << "\n"; Gen.startMethod(setType+"::iterator", getFunc()+"End", "", "methods", false, inlineAccessors); Gen.add("return "+cppName()+".end();"); DbSummary << "\t" << getFunc() << "End" << "\n"; Gen.startMethod(setType+"::const_iterator", getFunc()+"Begin", "", "methods", true, inlineAccessors); Gen.add("return "+cppName()+".begin();"); Gen.startMethod(setType+"::const_iterator", getFunc()+"End", "", "methods", true, inlineAccessors); Gen.add("return "+cppName()+".end();"); Gen.startMethod("const "+setType+" &", getFunc(), "", "methods", true, inlineAccessors); Gen.add("return "+cppName()+";"); // // generate write accessor // if (useReference) { Gen.startMethod("void", setFunc(), Type+"* "+valueVariable, "methods", false, inlineAccessors); DbSummary << "\t" << setFunc() << "\n"; Gen.add("if ("+valueVariable+" == NULL)\treturn;"); Gen.add(keyType->getName()+"\t"+keyVariable+" = "+valueVariable+"->"+sub->getKey()->getFunc()+"();"); Gen.add(setType+"::iterator\t_it = "+cppName()+".find("+keyVariable+");"); Gen.add("if (_it != "+cppName()+".end())"); Gen.add("{"); Gen.add(Type+"*\t__prev = (*_it).second;"); Gen.add("if (__prev == "+valueVariable+")\treturn;"); Gen.add("__prev->"+setParentFunction+"(NULL);"); Gen.add("__prev->"+unregisterFunction+"();"); Gen.add("__prev->"+destroyFunction+"();"); Gen.add("delete __prev;"); Gen.add("}"); Gen.add(valueVariable+"->"+setParentFunction+"(this);"); Gen.add(cppName()+"["+keyVariable+"] = "+valueVariable+";"); if (!onChangeUser.empty()) { Gen.add("{"); Gen.add(onChangeUser); Gen.add("}"); } } if (sub->MapClass == NULL && !sub->HasInheritance && !sub->DerivatedFlag) { Gen.startMethod(Type+"*", newFunc(), "const "+keyType->getName()+" &"+keyVariable, "methods", false, inlineAccessors); DbSummary << "\t" << newFunc() << "\n"; Gen.add(setType+"::iterator\t__it = "+cppName()+".find("+keyVariable+");"); Gen.add("if (__it == "+cppName()+".end())"); Gen.add("{"); Gen.add("__it = "+cppName()+".insert("+setType+"::value_type("+keyVariable+", "+(useReference ? toString("new ") : toString(""))+Type+"())).first;"); Gen.add(Type+"*\t"+objectVariable+" = "+(useReference ? toString("") : toString("&"))+"((*__it).second);"); Gen.add(objectVariable+"->"+initFunction+"("+keyVariable+");"); Gen.add(objectVariable+"->"+registerFunction+"();"); Gen.add(objectVariable+"->"+setParentFunction+"(this);"); Gen.add("}"); if (!onChangeUser.empty()) { Gen.add("{"); Gen.add(onChangeUser); Gen.add("}"); } Gen.add("return "+(useReference ? toString("") : toString("&"))+"((*__it).second);"); } Gen.startMethod("void", deleteFunc(), "const "+keyType->getName()+" &"+keyVariable, "methods", false, inlineAccessors); DbSummary << "\t" << deleteFunc() << "\n"; Gen.add(setType+"::iterator\t__it = "+cppName()+".find("+keyVariable+");"); Gen.add("if (__it == "+cppName()+".end())\treturn;"); if (useReference) { Gen.add(Type+"*\t"+objectVariable+" = (*__it).second;"); Gen.add(objectVariable+"->"+unregisterFunction+"();"); Gen.add(objectVariable+"->"+destroyFunction+"();"); Gen.add("delete "+objectVariable+";"); } else { Gen.add(Type+"&\t"+objectVariable+" = (*__it).second;"); Gen.add(objectVariable+"."+unregisterFunction+"();"); Gen.add(cppName()+".erase(__it);"); } if (!onChangeUser.empty()) { Gen.add("{"); Gen.add(onChangeUser); Gen.add("}"); } // // generate attribute // Gen.addAttribute(setType, cppName(), "attributes"); // // generate init // // // generate destroy code // DestroyId.add("for ("+setType+"::iterator __it="+cppName()+".begin(); __it!="+cppName()+".end(); )"); DestroyId.add("{"); DestroyId.add(setType+"::iterator __itr=__it++;"); if (useReference) { DestroyId.add(Type+"*\t"+objectVariable+" = ((*__itr).second);"); DestroyId.add("if ("+objectVariable+" != NULL)"); DestroyId.add("{"); DestroyId.add(objectVariable+"->"+destroyFunction+"();"); DestroyId.add("delete "+objectVariable+";"); DestroyId.add("}"); } else { DestroyId.add("((*__itr).second)."+destroyFunction+"();"); } DestroyId.add("}"); DestroyId.add(cppName()+".clear();"); // // generate create code // // // generate fetch code // if (!ClassNode->tableAndRowIndicesUsedInFetch) { FetchId.add(TTableIndexName+"\ttableIndex;\n"+TRowIndexName+"\trowIndex;"); ClassNode->tableAndRowIndicesUsedInFetch = true; } FetchId.add("do"); FetchId.add("{"); FetchId.add("// read table and row, create an object, affect to the ref, and fetch it"); FetchId.add("data.serial(tableIndex, rowIndex);"); FetchId.add("if (rowIndex == "+INVALID_ROW_INDEXName+" || tableIndex == "+INVALID_TABLE_INDEXName+")\tbreak;"); FetchId.add(keyType->getName()+"\t"+keyVariable+";"); if (keyType->isEnum()) { FetchId.add("data.serialEnum("+keyVariable+");"); } else if (keyType->CppType != keyType->StorageType) { FetchId.add("{"); FetchId.add(keyType->StorageType+"\t_v;"); FetchId.add("data.serial(_v);"); FetchId.add(keyVariable+" = "+keyType->castToCpp("_v")+";"); FetchId.add("}"); } else { FetchId.add("data.serial("+keyVariable+");"); } if (useReference) { FetchId.add(Type+"*\t"+objectVariable+" = static_cast<"+Type+"*>("+pdslibFunc("create")+"(tableIndex));"); FetchId.add(cppName()+".insert(std::make_pair<"+keyType->getName()+","+Type+"*>("+keyVariable+", "+objectVariable+"));"); } else { FetchId.add(cppName()+".insert(std::make_pair<"+keyType->getName()+","+Type+">("+keyVariable+", "+Type+"()));"); FetchId.add(Type+"*\t"+objectVariable+" = &("+cppName()+"["+keyVariable+"]);"); } FetchId.add(pdslibFunc("setRowIndex")+"(rowIndex, "+objectVariable+");"); FetchId.add(objectVariable+"->"+fetchFunction+"(data);"); FetchId.add(objectVariable+"->"+setUnnotifiedParentFunction+"(this);"); FetchId.add("}"); FetchId.add("while (true);"); // // generate register/unregister code // UnregisterAttributesId.add("for ("+setType+"::iterator __it="+cppName()+".begin(); __it!="+cppName()+".end(); )"); UnregisterAttributesId.add("{"); UnregisterAttributesId.add(setType+"::iterator __itr=__it++;"); if (useReference) { UnregisterAttributesId.add(Type+"*\t"+objectVariable+" = (*__itr).second;"); UnregisterAttributesId.add(objectVariable+"->"+unregisterFunction+"();"); UnregisterAttributesId.add(objectVariable+"->"+destroyFunction+"();"); UnregisterAttributesId.add("delete "+objectVariable+";"); } else { UnregisterAttributesId.add(Type+"&\t"+objectVariable+" = (*__itr).second;"); UnregisterAttributesId.add(objectVariable+"."+unregisterFunction+"();"); } UnregisterAttributesId.add("}"); // // generate init/release notification // NotifyInitId.add("for ("+setType+"::iterator __it="+cppName()+".begin(); __it!="+cppName()+".end(); )"); NotifyInitId.add("{"); NotifyInitId.add(setType+"::iterator __itr=__it++;"); NotifyReleaseId.add("for ("+setType+"::iterator __it="+cppName()+".begin(); __it!="+cppName()+".end(); )"); NotifyReleaseId.add("{"); NotifyReleaseId.add(setType+"::iterator __itr=__it++;"); if (!useReference) { NotifyInitId.add("(*__itr).second."+notifyInitFunction+"();"); NotifyReleaseId.add("(*__itr).second."+notifyReleaseFunction+"();"); } else { NotifyInitId.add("(*__itr).second->"+notifyInitFunction+"();"); NotifyReleaseId.add("(*__itr).second->"+notifyReleaseFunction+"();"); } NotifyInitId.add("}"); NotifyReleaseId.add("}"); // // generate unlink code // if (useReference) { Gen.startMethod("void", unlinkFunc(), keyType->getName()+" "+keyVariable, "internal", false, inlineInternal); Gen.add(cppName()+".erase("+keyVariable+");"); } // // generate clear code // ClearId.add("for ("+setType+"::iterator __it="+cppName()+".begin(); __it!="+cppName()+".end(); )"); ClearId.add("{"); ClearId.add(setType+"::iterator __itr=__it++;"); if (!useReference) { ClearId.add(Type+"*\t"+objectVariable+" = &((*__itr).second);"); } else { ClearId.add(Type+"*\t"+objectVariable+" = (*__itr).second;"); } //ClearId.add(objectVariable+"->"+setParentFunction+"(NULL);"); ClearId.add(objectVariable+"->"+unregisterFunction+"();"); ClearId.add(objectVariable+"->"+destroyFunction+"();"); if (useReference) { ClearId.add("delete "+objectVariable+";"); } ClearId.add("}"); ClearId.add(cppName()+".clear();"); } // string CDeclarationNode::displayPrintfPrefix() { if (DeclarationType != SimpleType) error("can't display other declaration than simple type", "internal"); CTypeNode* type = getTypeNode(Type); if (type->CppType == "CEntityId" || type->CppType == "CSheetId") return "%s"; else if (type->CppType == "uint64" || type->CppType == "sint64") return "%\"NL_I64\"d"; else if (type->CppType == "float" || type->CppType == "double") return "%g"; else return "%d"; } string CDeclarationNode::displayCppCode(string replVar) { if (DeclarationType != SimpleType) error("can't display other declaration than simple type", "internal"); if (replVar.empty()) replVar = cppName(); CTypeNode* type = getTypeNode(Type); if (type->CppType == "CEntityId" || type->CppType == "CSheetId") return replVar+".toString().c_str()"; else return replVar; } string CDeclarationNode::toUint64(string replVar) { if (DeclarationType != SimpleType) error("can't display other declaration than simple type", "internal"); if (replVar.empty()) replVar = cppName(); CTypeNode* type = getTypeNode(Type); if (type->CppType == "CEntityId") return replVar+".asUint64()"; else if (type->CppType == "CSheetId") return "(uint64)("+replVar+".asInt())"; else return "(uint64)"+replVar; } bool CLogMsgNode::prolog() { getDbNode()->LogNodes.push_back(this); getFileNode()->IncludeDbFile = true; getFileNode()->IncludeStandard = true; getFileNode()->IncludePDSLib = true; return true; } void CLogMsgNode::generateContent() { // uint j; CFunctionGenerator logfunc; CClassGenerator logclass; string argcall; CFunctionGenerator& initDb = getDbNode()->initDb; logfunc.init(Name); logfunc.setType("void"); logfunc.IsInline = inlineLogFunctions; logclass.init("Log"+Name); logclass.createPublic("pub"); logclass.createPrivate("priv"); CClassGenerator::SMethodId startlogid = logclass.startMethod("void", logStartFunction, "", "priv", false, inlineLogFunctions); CClassGenerator::SMethodId stoplogid = logclass.startMethod("void", logStopFunction, "", "priv", false, inlineLogFunctions); startlogid.add(pdslibFunc("pushContext")+"();"); stoplogid.add(pdslibFunc("popContext")+"();"); map<string, CParseNode*> params; for (j=0; j<Params.size(); ++j) { string type = Params[j].first; string name = Params[j].second; CTypeNode *tnd; CClassNode *cnd; if (tnd = getTypeNode(type, false)) { pair<map<string, CParseNode*>::iterator, bool> res = params.insert(make_pair<string, CParseNode*>(name, tnd)); if (!res.second) error("log parameter '"+name+"' already defined"); if (!logfunc.Proto.empty()) logfunc.Proto += ", "; if (!argcall.empty()) argcall += ", "; logfunc.Proto += tnd->getName()+" "+name; argcall += name; } else if (cnd = getClassNode(type, false)) { pair<map<string, CParseNode*>::iterator, bool> res = params.insert(make_pair<string, CParseNode*>(name, cnd)); if (!res.second) error("log parameter '"+name+"' already defined"); if (!logfunc.Proto.empty()) logfunc.Proto += ", "; if (!argcall.empty()) argcall += ", "; logfunc.Proto += "const "+type+"& "+name; argcall += name; } else if (type == "string") { CExtLogTypeNode* extnd = new CExtLogTypeNode(); extnd->ExtLogType = "string"; pair<map<string, CParseNode*>::iterator, bool> res = params.insert(make_pair<string, CParseNode*>(name, extnd)); if (!res.second) error("log parameter '"+name+"' already defined"); if (!logfunc.Proto.empty()) logfunc.Proto += ", "; if (!argcall.empty()) argcall += ", "; logfunc.Proto += "const std::string& "+name; argcall += name; } else { error("'"+type+"' not found as a class or a type"); } } // initDb.add("// Init "+Name+" log message and parameters"); for (j=0; j<Logs.size(); ++j) { uint logId = Id+j; logfunc.add(pdslibFunc("log")+"("+toString(logId)+");"); startlogid.add(pdslibFunc("log")+"("+toString(logId)+");"); const char *cptr = Logs[j].c_str(); string log; // parse log line getDbNode()->xmlDescription.push_back("<logmsg id='"+toString(logId)+"' context='"+string(Context ? "true" : "false")+"'>"); uint paramNum = 0; initDb.add(pdslibFunc("initLog")+"("+toString(logId)+");"); while (*cptr != '\0') { if (*cptr == '$') { ++cptr; if (*cptr == '\0') error("log format corrupted in line \""+Logs[j]+"\""); if (*cptr == '$') { log += "$$"; ++cptr; continue; } string param; string var; while (*cptr!='\0' && *cptr!='$' && *cptr!='.') param += *(cptr++); if (*cptr == '\0') error("log format corrupted in line \""+Logs[j]+"\""); if (*cptr == '.') { ++cptr; while (*cptr!='\0' && *cptr!='$') var += *(cptr++); } if (*cptr != '$') error("log format corrupted in line \""+Logs[j]+"\""); ++cptr; map<string, CParseNode*>::iterator it = params.find(param); if (it == params.end()) error("'"+param+"' ot found in prototype, at line \""+Logs[j]+"\""); CTypeNode *tnd = NULL; CExtLogTypeNode* extnd = NULL; if (var.empty()) { // is simple type tnd = dynamic_cast<CTypeNode*>((*it).second); extnd = dynamic_cast<CExtLogTypeNode*>((*it).second); if (tnd != NULL) { logfunc.add(pdslibFunc("logPush")+"("+tnd->castToPDS(param)+");"); startlogid.add(pdslibFunc("logPush")+"("+tnd->castToPDS(param)+");"); } else if (extnd != NULL && extnd->ExtLogType == "string") { logfunc.add(pdslibFunc("logPush")+"("+param+");"); startlogid.add(pdslibFunc("logPush")+"("+param+");"); } else { error("misuse of parameter '"+param+"' at line \""+Logs[j]+"\", missing attribute name"); } } else { // is class CClassNode *cnd = dynamic_cast<CClassNode*>((*it).second); if (!cnd) error("misuse of parameter '"+param+"' at line \""+Logs[j]+"\", should be a class"); CDeclarationNode *dnd = cnd->getDeclaration(var); if (!dnd->IsType) error("attribute '"+var+"' is not a simple type"); tnd = getTypeNode(dnd->Type); logfunc.add(pdslibFunc("logPush")+"("+tnd->castToPDS(param+"."+dnd->getFunc()+"()")+");"); startlogid.add(pdslibFunc("logPush")+"("+tnd->castToPDS(param+"."+dnd->getFunc()+"()")+");"); } if (tnd != NULL) { getDbNode()->xmlDescription.push_back("<param id='"+toString(paramNum)+"' typeid='"+toString(tnd->Id)+"'/>"); initDb.add(pdslibFunc("initLogParam")+"("+toString(logId)+", "+toString(paramNum)+", "+toString(tnd->Size)+");"); } else { getDbNode()->xmlDescription.push_back("<param id='"+toString(paramNum)+"' typeid='"+extnd->ExtLogType+"'/>"); initDb.add(pdslibFunc("initLogParam")+"("+toString(logId)+", "+toString(paramNum)+", "+toString(sizeof(uint16))+");"); } log += "$"+toString(paramNum); ++paramNum; } else { log += *(cptr++); } } getDbNode()->xmlDescription.push_back("<msg>"+xmlSpecialChars(log)+"</msg>"); getDbNode()->xmlDescription.push_back("</logmsg>"); } logclass.get(startlogid).Proto = logfunc.Proto; CClassGenerator::SMethodId construct = logclass.startConstructor(logfunc.Proto, "pub", inlineLogFunctions, ""); construct.add(logStartFunction+"("+argcall+");"); CClassGenerator::SMethodId destruct = logclass.startDestructor("pub", inlineLogFunctions); destruct.add(logStopFunction+"();"); CFileNode* file = getFileNode(); if (Context) { logclass.flush(file->hOutput(), file->cppOutput(), file->inlineOutput()); file->hOutput() << "\n"; } else { logfunc.flush(file->hOutput(), file->cppOutput(), file->inlineOutput()); file->hOutput() << "\n"; } }