浏览代码

* ERM parsing mostly done, some strange cases of body syntax, undocumented expressions and encoding issues still remain.

mateuszb 14 年之前
父节点
当前提交
3c393843e6
共有 2 个文件被更改,包括 197 次插入65 次删除
  1. 3 3
      client/CAnimation.cpp
  2. 194 62
      lib/ERMParser.cpp

+ 3 - 3
client/CAnimation.cpp

@@ -1117,7 +1117,7 @@ size_t CAnimation::size(size_t group) const
 
 
 std::set<CAnimation*> CAnimation::loadedAnims;
 std::set<CAnimation*> CAnimation::loadedAnims;
 
 
-void CAnimation::getAnimInfo()
+void CAnimation::getAnimInfo()
 {
 {
 	tlog1<<"Animation stats: Loaded "<<loadedAnims.size()<<" total\n";
 	tlog1<<"Animation stats: Loaded "<<loadedAnims.size()<<" total\n";
 	for (std::set<CAnimation*>::iterator it = loadedAnims.begin(); it != loadedAnims.end(); it++)
 	for (std::set<CAnimation*>::iterator it = loadedAnims.begin(); it != loadedAnims.end(); it++)
@@ -1127,8 +1127,8 @@ void CAnimation::getAnimInfo()
 		if (!anim->images.empty())
 		if (!anim->images.empty())
 			tlog1<<", "<<anim->images.begin()->second.size()<<" image loaded in group "<< anim->images.begin()->first;
 			tlog1<<", "<<anim->images.begin()->second.size()<<" image loaded in group "<< anim->images.begin()->first;
 		tlog1<<"\n";
 		tlog1<<"\n";
-	}
-}
+	}
+}
 
 
 CAnimImage::CAnimImage(std::string name, size_t Frame, size_t Group, int x, int y, unsigned char Flags):
 CAnimImage::CAnimImage(std::string name, size_t Frame, size_t Group, int x, int y, unsigned char Flags):
 	frame(Frame),
 	frame(Frame),

+ 194 - 62
lib/ERMParser.cpp

@@ -55,6 +55,7 @@ void ERMParser::parseFile()
 	parsedLine = 1;
 	parsedLine = 1;
 	std::string wholeLine; //used for buffering multiline lines
 	std::string wholeLine; //used for buffering multiline lines
 	bool inString = false;
 	bool inString = false;
+	
 	while(file.good())
 	while(file.good())
 	{
 	{
 		//reading line
 		//reading line
@@ -103,13 +104,25 @@ void callme(char const& i)
 
 
 namespace ERM
 namespace ERM
 {
 {
-	//i-expression (identifier expression) - an integral constant, variable symbol or array symbol
-	struct iexpT
+	typedef std::string TStringConstant;
+	typedef std::string TMacroUsage;
+	typedef std::string TMacroDef;
+	typedef std::string TCmdName;
+
+	
+	struct TVarExpNotMacro
 	{
 	{
-		typedef boost::optional<boost::variant<int, std::string> > valT;
-		boost::optional<std::string> varsym;
+		typedef boost::optional<int> valT;
+		std::string varsym;
 		valT val;
 		valT val;
 	};
 	};
+	typedef boost::variant<TVarExpNotMacro, TMacroUsage> TVarExp;
+
+	//write-only variable expression
+	typedef TVarExp TVarpExp;
+
+	//i-expression (identifier expression) - an integral constant, variable symbol or array symbol
+	typedef boost::variant<TVarExp, int> iexpT;
 
 
 	struct TArithmeticOp
 	struct TArithmeticOp
 	{
 	{
@@ -117,6 +130,47 @@ namespace ERM
 		char opcode;
 		char opcode;
 	};
 	};
 
 
+	struct TVRLogic
+	{
+		char opcode;
+		iexpT var;
+	};
+
+	struct TVRArithmetic
+	{
+		char opcode;
+		iexpT rhs;
+	};
+
+	struct TSemiCompare
+	{
+		std::string compSign;
+		iexpT rhs;
+	};
+
+	struct TCurriedString
+	{
+		iexpT iexp;
+		TStringConstant string;
+	};
+
+	struct TVarConcatString 
+	{
+		TVarExp var;
+		TStringConstant string;
+	};
+
+	typedef boost::variant<TVarConcatString, TStringConstant, TCurriedString, TSemiCompare, TMacroUsage, TMacroDef, iexpT, TVarpExp, qi::unused_type> TBodyOptionItem;
+
+	typedef std::vector<TBodyOptionItem> TNormalBodyOptionList;
+
+	struct TNormalBodyOption
+	{
+		char optionCode;
+		TNormalBodyOptionList params;
+	};
+	typedef boost::variant<TVRLogic, TVRArithmetic, TNormalBodyOption> TBodyOption;
+
 	typedef boost::variant<iexpT, TArithmeticOp > TIdentifierInternal;
 	typedef boost::variant<iexpT, TArithmeticOp > TIdentifierInternal;
 	typedef std::vector< TIdentifierInternal > identifierT;
 	typedef std::vector< TIdentifierInternal > identifierT;
 
 
@@ -147,7 +201,7 @@ namespace ERM
 
 
 	struct triggerT
 	struct triggerT
 	{
 	{
-		std::string name;
+		TCmdName name;
 		boost::optional<identifierT> identifier;
 		boost::optional<identifierT> identifier;
 		boost::optional<conditionT> condition;
 		boost::optional<conditionT> condition;
 	};
 	};
@@ -158,12 +212,12 @@ namespace ERM
 	//moreover, I encountered a quite serious bug in boost: http://boost.2283326.n4.nabble.com/container-hpp-111-error-C2039-value-type-is-not-a-member-of-td3352328.html
 	//moreover, I encountered a quite serious bug in boost: http://boost.2283326.n4.nabble.com/container-hpp-111-error-C2039-value-type-is-not-a-member-of-td3352328.html
 	//not sure how serious it is...
 	//not sure how serious it is...
 
 
-	typedef boost::variant<char, std::string> bodyItem;
- 	typedef std::vector<bodyItem> bodyTbody;
+	//typedef boost::variant<char, TStringConstant, TMacroUsage, TMacroDef> bodyItem;
+ 	typedef std::vector<TBodyOption> bodyTbody;
 
 
 	struct instructionT
 	struct instructionT
 	{
 	{
-		std::string name;
+		TCmdName name;
 		boost::optional<identifierT> identifier;
 		boost::optional<identifierT> identifier;
 		boost::optional<conditionT> condition;
 		boost::optional<conditionT> condition;
 		bodyTbody body;
 		bodyTbody body;
@@ -171,7 +225,7 @@ namespace ERM
 
 
 	struct receiverT
 	struct receiverT
 	{
 	{
-		std::string name;
+		TCmdName name;
 		boost::optional<identifierT> identifier;
 		boost::optional<identifierT> identifier;
 		boost::optional<conditionT> condition;
 		boost::optional<conditionT> condition;
 		bodyTbody body;
 		bodyTbody body;
@@ -202,29 +256,43 @@ namespace ERM
 
 
 	//console printer
 	//console printer
 
 
-	struct UNT : boost::static_visitor<>
+	struct VarPrinter : boost::static_visitor<>
 	{
 	{
-		void operator()(int const& val) const
+		void operator()(TVarExpNotMacro const& val) const
 		{
 		{
-			tlog2 << val << " ";
+			tlog2 << val.varsym;
+			if(val.val.is_initialized())
+			{
+				tlog2 << val.val.get();
+			}
 		}
 		}
-		void operator()(std::string const& str) const
+		void operator()(TMacroUsage const& val) const
 		{
 		{
-			tlog2 << str << " ";
+			tlog2 << "$" << val << "&";
 		}
 		}
 	};
 	};
 
 
+	void varPrinter(const TVarExp & var)
+	{
+		boost::apply_visitor(VarPrinter(), var);
+	}
 
 
-	void iexpPrinter(const iexpT exp)
+	struct _IEP : boost::static_visitor<>
 	{
 	{
-		if(exp.varsym.is_initialized())
+		void operator()(int const & constant) const
 		{
 		{
-			tlog2 << exp.varsym.get() << " ";
+			tlog2 << constant;
 		}
 		}
-		if(exp.val.is_initialized())
+		void operator()(TVarExp const & var) const
 		{
 		{
-			boost::apply_visitor(UNT(), exp.val.get());
+			varPrinter(var);
 		}
 		}
+	};
+
+
+	void iexpPrinter(const iexpT & exp)
+	{
+		boost::apply_visitor(_IEP(), exp);
 	}
 	}
 
 
 	struct IdentifierVisitor : boost::static_visitor<>
 	struct IdentifierVisitor : boost::static_visitor<>
@@ -348,9 +416,9 @@ namespace ERM
 }
 }
 
 
 BOOST_FUSION_ADAPT_STRUCT(
 BOOST_FUSION_ADAPT_STRUCT(
-	ERM::iexpT,
-	(boost::optional<std::string>, varsym)
-	(ERM::iexpT::valT, val)
+	ERM::TVarExpNotMacro,
+	(std::string, varsym)
+	(ERM::TVarExpNotMacro::valT, val)
 	)
 	)
 
 
 BOOST_FUSION_ADAPT_STRUCT(
 BOOST_FUSION_ADAPT_STRUCT(
@@ -360,9 +428,27 @@ BOOST_FUSION_ADAPT_STRUCT(
 	(ERM::iexpT, rhs)
 	(ERM::iexpT, rhs)
 	)
 	)
 
 
+BOOST_FUSION_ADAPT_STRUCT(
+	ERM::TVRLogic,
+	(char, opcode)
+	(ERM::iexpT, var)
+	)
+
+BOOST_FUSION_ADAPT_STRUCT(
+	ERM::TVRArithmetic,
+	(char, opcode)
+	(ERM::iexpT, rhs)
+	)
+
+BOOST_FUSION_ADAPT_STRUCT(
+	ERM::TNormalBodyOption,
+	(char, optionCode)
+	(ERM::TNormalBodyOptionList, params)
+	)
+
 BOOST_FUSION_ADAPT_STRUCT(
 BOOST_FUSION_ADAPT_STRUCT(
 	ERM::triggerT,
 	ERM::triggerT,
-	(std::string, name)
+	(ERM::TCmdName, name)
 	(boost::optional<ERM::identifierT>, identifier)
 	(boost::optional<ERM::identifierT>, identifier)
 	(boost::optional<ERM::conditionT>, condition)
 	(boost::optional<ERM::conditionT>, condition)
 	)
 	)
@@ -374,6 +460,24 @@ BOOST_FUSION_ADAPT_STRUCT(
 	(ERM::iexpT, rhs)
 	(ERM::iexpT, rhs)
 	)
 	)
 
 
+BOOST_FUSION_ADAPT_STRUCT(
+	ERM::TSemiCompare,
+	(std::string, compSign)
+	(ERM::iexpT, rhs)
+	)
+
+BOOST_FUSION_ADAPT_STRUCT(
+	ERM::TCurriedString,
+	(ERM::iexpT, iexp)
+	(ERM::TStringConstant, string)
+	)
+
+BOOST_FUSION_ADAPT_STRUCT(
+	ERM::TVarConcatString,
+	(ERM::TVarExp, var)
+	(ERM::TStringConstant, string)
+	)
+
 BOOST_FUSION_ADAPT_STRUCT(
 BOOST_FUSION_ADAPT_STRUCT(
 	ERM::conditionT,
 	ERM::conditionT,
 	(char, ctype)
 	(char, ctype)
@@ -381,10 +485,9 @@ BOOST_FUSION_ADAPT_STRUCT(
 	(ERM::conditionNodeT, rhs)
 	(ERM::conditionNodeT, rhs)
 	)
 	)
 
 
-
 BOOST_FUSION_ADAPT_STRUCT(
 BOOST_FUSION_ADAPT_STRUCT(
 	ERM::instructionT,
 	ERM::instructionT,
-	(std::string, name)
+	(ERM::TCmdName, name)
 	(boost::optional<ERM::identifierT>, identifier)
 	(boost::optional<ERM::identifierT>, identifier)
 	(boost::optional<ERM::conditionT>, condition)
 	(boost::optional<ERM::conditionT>, condition)
 	(ERM::bodyTbody, body)
 	(ERM::bodyTbody, body)
@@ -392,7 +495,7 @@ BOOST_FUSION_ADAPT_STRUCT(
 
 
 BOOST_FUSION_ADAPT_STRUCT(
 BOOST_FUSION_ADAPT_STRUCT(
 	ERM::receiverT,
 	ERM::receiverT,
-	(std::string, name)
+	(ERM::TCmdName, name)
 	(boost::optional<ERM::identifierT>, identifier)
 	(boost::optional<ERM::identifierT>, identifier)
 	(boost::optional<ERM::conditionT>, condition)
 	(boost::optional<ERM::conditionT>, condition)
 	(ERM::bodyTbody, body)
 	(ERM::bodyTbody, body)
@@ -413,15 +516,20 @@ BOOST_FUSION_ADAPT_STRUCT(
 namespace ERM
 namespace ERM
 {
 {
 	template<typename Iterator>
 	template<typename Iterator>
-	struct ERM_grammar : qi::grammar<Iterator, lineT()>
+	struct ERM_grammar : qi::grammar<Iterator, lineT(), ascii::space_type>
 	{
 	{
 		ERM_grammar() : ERM_grammar::base_type(rline, "ERM script line")
 		ERM_grammar() : ERM_grammar::base_type(rline, "ERM script line")
 		{
 		{
-			macro %= qi::lit('$') >> *(qi::char_ - '$') >> qi::lit('$');
-			iexp %= -(*qi::char_("a-z") - 'u') >> -(qi::int_ | macro); 
+			//do not build too complicated expressions, e.g. (a >> b) | c, qi has problems with them
+			macroUsage %= qi::lexeme[qi::lit('$') >> *(qi::char_ - '$') >> qi::lit('$')];
+			macroDef %= qi::lexeme[qi::lit('@') >> *(qi::char_ - '@') >> qi::lit('@')];
+			varExpNotMacro %= (+(qi::char_("?a-z") - 'u')) >> -qi::int_;
+			varExp %= varExpNotMacro | macroUsage;
+			iexp %= varExp | qi::int_;
+			varp %= qi::char_("?") > varExp;
  			comment %= *(qi::char_);
  			comment %= *(qi::char_);
- 			commentLine %= ~qi::char_('!') >> comment;
- 			cmdName %= qi::repeat(2)[qi::char_];
+			commentLine %= (~qi::char_('!') >> comment | (qi::char_('!') >> (~qi::char_("?!#")) >> comment ));
+ 			cmdName %= qi::lexeme[qi::repeat(2)[qi::char_]];
 			arithmeticOp %= iexp >> qi::char_ >> iexp;
 			arithmeticOp %= iexp >> qi::char_ >> iexp;
 			//identifier is usually a vector of i-expressions but VR receiver performs arithmetic operations on it
 			//identifier is usually a vector of i-expressions but VR receiver performs arithmetic operations on it
 			identifier %= (iexp | arithmeticOp) % qi::lit('/');
 			identifier %= (iexp | arithmeticOp) % qi::lit('/');
@@ -430,14 +538,25 @@ namespace ERM
 
 
 			trigger %= cmdName >> -identifier >> -condition > qi::lit(";"); /////
 			trigger %= cmdName >> -identifier >> -condition > qi::lit(";"); /////
 			string %= qi::lexeme['^' >> *(qi::char_ - '^') >> '^'];
 			string %= qi::lexeme['^' >> *(qi::char_ - '^') >> '^'];
-			body %= qi::lit(":") > *( qi::char_("a-zA-Z0-9/ @*?%+-:|&=><-") | string | macro) > qi::lit(";");
+			
+			VRLogic %= qi::char_("&|X") >> iexp;
+			VRarithmetic %= qi::char_("+*:/%-") >> iexp;
+			semiCompare %= *qi::char_("<=>") >> iexp;
+			curStr %= iexp >> string;
+			varConcatString %= varExp >> qi::lit("+") >> string;
+			bodyOptionItem %= varConcatString | curStr | string | semiCompare | macroUsage | macroDef | iexp | varp | qi::eps;
+			exactBodyOptionList %= (bodyOptionItem % qi::lit("/"));
+			normalBodyOption = qi::char_("A-Z") > exactBodyOptionList;
+			bodyOption %= VRLogic | VRarithmetic | normalBodyOption;
+			body %= qi::lit(":") >> +(bodyOption) > qi::lit(";");
+
 			instruction %= cmdName >> -identifier >> -condition >> body;
 			instruction %= cmdName >> -identifier >> -condition >> body;
 			receiver %= cmdName >> -identifier >> -condition >> body; //receiver without body exists... change needed
 			receiver %= cmdName >> -identifier >> -condition >> body; //receiver without body exists... change needed
 			postOBtrigger %= qi::lit("$OB") >> -identifier >> -condition > qi::lit(";");
 			postOBtrigger %= qi::lit("$OB") >> -identifier >> -condition > qi::lit(";");
 			command %= (qi::lit("!") >>
 			command %= (qi::lit("!") >>
 					(
 					(
 						(qi::lit("?") >> trigger) |
 						(qi::lit("?") >> trigger) |
-						((qi::lit("!") | qi::lit("d!") | qi::lit(" !")) >> receiver) |
+						(qi::lit("!") >> receiver) |
 						(qi::lit("#") >> instruction) | 
 						(qi::lit("#") >> instruction) | 
 						postOBtrigger
 						postOBtrigger
 					) >> comment
 					) >> comment
@@ -479,24 +598,37 @@ namespace ERM
 
 
 		}
 		}
 
 
-		qi::rule<Iterator, std::string()> string;
-
-		qi::rule<Iterator, std::string()> macro;
-		qi::rule<Iterator, iexpT()> iexp;
-		qi::rule<Iterator, TArithmeticOp()> arithmeticOp;
-		qi::rule<Iterator, std::string()> comment;
-		qi::rule<Iterator, std::string()> commentLine;
-		qi::rule<Iterator, std::string()> cmdName;
-		qi::rule<Iterator, identifierT()> identifier;
-		qi::rule<Iterator, TComparison()> comparison;
-		qi::rule<Iterator, conditionT()> condition;
-		qi::rule<Iterator, triggerT()> trigger;
-		qi::rule<Iterator, bodyTbody()> body;
-		qi::rule<Iterator, instructionT()> instruction;
-		qi::rule<Iterator, receiverT()> receiver;
-		qi::rule<Iterator, postOBtriggerT()> postOBtrigger;
-		qi::rule<Iterator, commandT()> command;
-		qi::rule<Iterator, lineT()> rline;
+		qi::rule<Iterator, TStringConstant(), ascii::space_type> string;
+
+		qi::rule<Iterator, TMacroUsage(), ascii::space_type> macroUsage;
+		qi::rule<Iterator, TMacroDef(), ascii::space_type> macroDef;
+		qi::rule<Iterator, TVarExpNotMacro(), ascii::space_type> varExpNotMacro;
+		qi::rule<Iterator, TVarExp(), ascii::space_type> varExp;
+		qi::rule<Iterator, iexpT(), ascii::space_type> iexp;
+		qi::rule<Iterator, TVarpExp(), ascii::space_type> varp;
+		qi::rule<Iterator, TArithmeticOp(), ascii::space_type> arithmeticOp;
+		qi::rule<Iterator, std::string(), ascii::space_type> comment;
+		qi::rule<Iterator, std::string(), ascii::space_type> commentLine;
+		qi::rule<Iterator, TCmdName(), ascii::space_type> cmdName;
+		qi::rule<Iterator, identifierT(), ascii::space_type> identifier;
+		qi::rule<Iterator, TComparison(), ascii::space_type> comparison;
+		qi::rule<Iterator, conditionT(), ascii::space_type> condition;
+		qi::rule<Iterator, TVRLogic(), ascii::space_type> VRLogic;
+		qi::rule<Iterator, TVRArithmetic(), ascii::space_type> VRarithmetic;
+		qi::rule<Iterator, TSemiCompare(), ascii::space_type> semiCompare;
+		qi::rule<Iterator, TCurriedString(), ascii::space_type> curStr;
+		qi::rule<Iterator, TVarConcatString(), ascii::space_type> varConcatString;
+		qi::rule<Iterator, TBodyOptionItem(), ascii::space_type> bodyOptionItem;
+		qi::rule<Iterator, TNormalBodyOptionList(), ascii::space_type> exactBodyOptionList;
+		qi::rule<Iterator, TNormalBodyOption(), ascii::space_type> normalBodyOption;
+		qi::rule<Iterator, TBodyOption(), ascii::space_type> bodyOption;
+		qi::rule<Iterator, triggerT(), ascii::space_type> trigger;
+		qi::rule<Iterator, bodyTbody(), ascii::space_type> body;
+		qi::rule<Iterator, instructionT(), ascii::space_type> instruction;
+		qi::rule<Iterator, receiverT(), ascii::space_type> receiver;
+		qi::rule<Iterator, postOBtriggerT(), ascii::space_type> postOBtrigger;
+		qi::rule<Iterator, commandT(), ascii::space_type> command;
+		qi::rule<Iterator, lineT(), ascii::space_type> rline;
 	};
 	};
 };
 };
 
 
@@ -508,17 +640,17 @@ void ERMParser::parseLine( const std::string & line )
 	ERM::ERM_grammar<std::string::const_iterator> ERMgrammar;
 	ERM::ERM_grammar<std::string::const_iterator> ERMgrammar;
 	ERM::lineT AST;
 	ERM::lineT AST;
 
 
-	bool r = qi::parse(beg, end, ERMgrammar, AST);
-	if(!r || beg != end)
-	{
-		tlog1 << "Parse error for line (" << parsedLine << ") : " << line << std::endl;
-		tlog1 << "\tCannot parse: " << std::string(beg, end) << std::endl;
-	}
-	else
-	{
-		//parsing succeeded
-		//ERM::printLineAST(AST);
-	}
+// 	bool r = qi::phrase_parse(beg, end, ERMgrammar, ascii::space, AST);
+// 	if(!r || beg != end)
+// 	{
+// 		tlog1 << "Parse error for line (" << parsedLine << ") : " << line << std::endl;
+// 		tlog1 << "\tCannot parse: " << std::string(beg, end) << std::endl;
+// 	}
+// 	else
+// 	{
+// 		//parsing succeeded
+// 		//ERM::printLineAST(AST);
+// 	}
 }
 }
 
 
 ERMParser::ELineType ERMParser::classifyLine( const std::string & line, bool inString ) const
 ERMParser::ELineType ERMParser::classifyLine( const std::string & line, bool inString ) const