ryzom-core/code/ryzom/tools/pd_parser/parser.cpp

4913 lines
139 KiB
C++

// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
// Copyright (C) 2010 Winch Gate Property Limited
//
// This source file has been modified by the following contributors:
// Copyright (C) 2015 Jan BOON (Kaetemi) <jan.boon@kaetemi.be>
//
// 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)
{
std::string::size_type pos = 0;
while ((pos = str.find(search)) != string::npos)
str.replace(pos, search.size(), replace);
return str;
}
string xmlSpecialChars(string str)
{
str = strReplace(str, "&", "&amp;");
str = strReplace(str, "<", "&lt;");
str = strReplace(str, ">", "&gt;");
str = strReplace(str, "\"", "&quot;");
str = strReplace(str, "'", "&apos;");
return str;
}
string getFullStdPathNoExt(const string &path)
{
string dir = NLMISC::toLower(NLMISC::CFile::getPath(path));
string file = NLMISC::toLower(NLMISC::CFile::getFilenameWithoutExtension(path));
return dir.empty() ? file : NLMISC::CPath::standardizePath(dir)+file;
}
string getFullStdPath(const string &path)
{
string dir = NLMISC::toLower(NLMISC::CFile::getPath(path));
string file = NLMISC::toLower(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::toLower(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.clear();
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.clear();
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.clear();
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 += (uint)child->Logs.size();
child->generateContent();
}
}
// get file path from this file
string CDbNode::getFileNoExtPath(const std::string& file)
{
string thisPath = NLMISC::CFile::getPath(NLMISC::toLower(getDbFile()));
string filePath = NLMISC::CFile::getPath(NLMISC::toLower(file));
string fileName = NLMISC::CFile::getFilename(NLMISC::toLower(file));
if (thisPath == filePath)
return CFile::getFilenameWithoutExtension(fileName);
else
return CPath::standardizePath(filePath)+CFile::getFilenameWithoutExtension(NLMISC::toLower(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 = NLMISC::toLower(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::toLower(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 = NLMISC::toLower(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(NLMISC::toLower(Name));
string filePath = NLMISC::CFile::getPath(NLMISC::toLower(file));
string fileName = NLMISC::CFile::getFilename(NLMISC::toLower(file));
if (thisPath == filePath)
return CFile::getFilenameWithoutExtension(fileName);
else
return CFile::getFilenameWithoutExtension(NLMISC::toLower(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(std::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(std::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(std::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 = (uint)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.clear();
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("+keyVariable+", "+objectVariable+"));");
}
else
{
FetchId.add(cppName()+".insert(std::make_pair("+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(std::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(std::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(std::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";
}
}