Mercurial Hosting > luan
changeset 666:2f449ccf54d2
use lists to assemble source
author | Franklin Schmidt <fschmidt@gmail.com> |
---|---|
date | Thu, 07 Apr 2016 23:36:56 -0600 (2016-04-08) |
parents | 41f8fdbc3a0a |
children | 08966099aa6d |
files | core/src/luan/impl/LuanParser.java |
diffstat | 1 files changed, 334 insertions(+), 165 deletions(-) [+] |
line wrap: on
line diff
diff -r 41f8fdbc3a0a -r 2f449ccf54d2 core/src/luan/impl/LuanParser.java --- a/core/src/luan/impl/LuanParser.java Thu Apr 07 17:06:22 2016 -0600 +++ b/core/src/luan/impl/LuanParser.java Thu Apr 07 23:36:56 2016 -0600 @@ -176,13 +176,13 @@ private FnDef newFnDef(int start,StmtString stmt) { if( !stmt.hasReturn ) - stmt = new StmtString( stmt.code + "return LuanFunction.NOTHING;\n" ); + stmt.list.add( "return LuanFunction.NOTHING;\n" ); return toFnDef( stmt, frame.stackSize, symbolsSize(), frame.isVarArg, frame.upValueGetters.toArray(NO_UP_VALUE_GETTERS) ); } private ExpString newFnExpStr(int start,StmtString stmt) { if( !stmt.hasReturn ) - stmt = new StmtString( stmt.code + "return LuanFunction.NOTHING;\n" ); + stmt.list.add( "return LuanFunction.NOTHING;\n" ); return toFnExpStr( stmt, frame.stackSize, symbolsSize(), frame.isVarArg, frame.upValueGetters.toArray(NO_UP_VALUE_GETTERS) ); } @@ -191,8 +191,10 @@ int start = parser.begin(); ExpString expr = ExprZ(In.NOTHING); if( expr != null && parser.endOfInput() ) { - String code = "return " + expr.code + ";\n"; - StmtString stmt = new StmtString(code); + StmtString stmt = new StmtString(); + stmt.list.add( "return " ); + stmt.list.addAll( expr.list ); + stmt.list.add( ";\n" ); stmt.hasReturn = true; return parser.success(newFnDef(start,stmt)); } @@ -210,7 +212,7 @@ } private StmtString RequiredBlock() throws ParseException { - StringBuilder stmts = new StringBuilder(); + StmtString stmts = new StmtString(); int stackStart = symbolsSize(); boolean isReturn = Stmt(stmts); while( !isReturn && StmtSep(stmts) ) { @@ -221,23 +223,22 @@ Spaces(In.NOTHING); int stackEnd = symbolsSize(); popSymbols( stackEnd - stackStart ); - if( stmts.length() == 0 ) - return EMPTY_STMT; + if( stmts.list.isEmpty() ) + return stmts; String code = stmts.toString(); - if( stackStart < stackEnd ) - code = "" - +"try {\n" - + code + if( stackStart < stackEnd ) { + stmts.list.add( 0, "try {\n" ); + stmts.list.add( "" +"} finally {\n" +" luan.stackClear("+stackStart+","+stackEnd+");\n" +"}\n" - ; - StmtString stmt = new StmtString(code); - stmt.hasReturn = isReturn; - return stmt; + ); + } + stmts.hasReturn = isReturn; + return stmts; } - private boolean StmtSep(StringBuilder stmts) throws ParseException { + private boolean StmtSep(StmtString stmts) throws ParseException { parser.begin(); if( parser.match( ';' ) ) return parser.success(); @@ -247,7 +248,7 @@ // parser.rollback(); StmtString stmt = TemplateStmt(); if( stmt != null ) { - stmts.append(stmt.code); + stmts.list.addAll(stmt.list); return parser.success(); } } @@ -258,12 +259,12 @@ return parser.match( "\r\n" ) || parser.match( '\r' ) || parser.match( '\n' ); } - private boolean Stmt(StringBuilder stmts) throws ParseException { + private boolean Stmt(StmtString stmts) throws ParseException { if( LocalStmt(stmts) ) return false; StmtString stmt; if( (stmt=ReturnStmt()) != null ) { - stmts.append(stmt.code); + stmts.list.addAll(stmt.list); return true; } if( (stmt=FunctionStmt()) != null @@ -277,31 +278,44 @@ || (stmt=SetStmt()) != null || (stmt=ExpressionsStmt()) != null ) { - stmts.append(stmt.code); + stmts.list.addAll(stmt.list); } return false; } private ExpString indexExpStr(ExpString exp1,ExpString exp2) { - String code = "luan.index(" + exp1.expr().code + "," + exp2.expr().code + ")"; - return new ExpString(code,true,false); + ExpString exp = new ExpString(true,false); + exp.list.add( "luan.index(" ); + exp.list.addAll( exp1.expr().list ); + exp.list.add( "," ); + exp.list.addAll( exp2.expr().list ); + exp.list.add( ")" ); + return exp; } private ExpString callExpStr(ExpString fn,ExpString args) { - String code = "Luan.checkFunction(" + fn.expr().code + ").call(luan,Luan.array(" + args.code + "))"; - return new ExpString(code,false,true); + ExpString exp = new ExpString(false,true); + exp.list.add( "Luan.checkFunction(" ); + exp.list.addAll( fn.expr().list ); + exp.list.add( ").call(luan,Luan.array(" ); + exp.list.addAll( args.list ); + exp.list.add( "))" ); + return exp; } private StmtString TemplateStmt() throws ParseException { ExpString exprs = TemplateExpressions(In.NOTHING); if( exprs == null ) return null; - String code = "PackageLuan.require(luan,\"luan:Io\")"; - ExpString requireCall = new ExpString(code,true,false); + ExpString requireCall = new ExpString(true,false); + requireCall.list.add( "PackageLuan.require(luan,\"luan:Io\")" ); ExpString stdoutExp = indexExpStr( requireCall.expr(), constExpStr("stdout") ); ExpString writeExp = indexExpStr( stdoutExp, constExpStr("write") ); ExpString writeCall = callExpStr( writeExp, exprs ); - return new StmtString( writeCall.code + ";\n" ); + StmtString stmt = new StmtString(); + stmt.list.addAll( writeCall.list ); + stmt.list.add( ";\n" ); + return stmt; } private ExpString TemplateExpressions(In in) throws ParseException { @@ -340,8 +354,14 @@ if( !Keyword("return",In.NOTHING) ) return parser.failure(null); ExpString exprs = ExpStringList(In.NOTHING); - String code = "return " + (exprs!=null ? exprs.code : "LuanFunction.NOTHING") + ";\n"; - return parser.success( new StmtString(code) ); + StmtString stmt = new StmtString(); + stmt.list.add( "return " ); + if( exprs != null ) + stmt.list.addAll( exprs.list ); + else + stmt.list.add( "LuanFunction.NOTHING" ); + stmt.list.add( ";\n" ); + return parser.success( stmt ); } private StmtString FunctionStmt() throws ParseException { @@ -371,8 +391,12 @@ addSymbol( name ); ExpString fnDef = RequiredFunction(In.NOTHING); Settable s = new SetLocalVar(symbolsSize()-1); - String code = new SettableString(s).code + ".set(luan," + fnDef.code + ");\n"; - return parser.success( new StmtString(code) ); + StmtString stmt = new StmtString(); + stmt.list.add( settableToString(s) ); + stmt.list.add( ".set(luan," ); + stmt.list.addAll( fnDef.list ); + stmt.list.add( ");\n" ); + return parser.success( stmt ); } private StmtString BreakStmt() throws ParseException { @@ -381,7 +405,9 @@ return parser.failure(null); if( frame.loops <= 0 ) throw parser.exception("'break' outside of loop"); - return parser.success( new StmtString("break;\n") ); + StmtString stmt = new StmtString(); + stmt.list.add( "break;\n" ); + return parser.success( stmt ); } int forCounter = 0; @@ -408,22 +434,30 @@ StmtString loop = RequiredLoopBlock(); RequiredKeyword("end",In.NOTHING); String fnVar = "fn"+ ++forCounter; - String code = "" + StmtString stmt = new StmtString(); + stmt.list.add( "" +"try {\n" - +"LuanFunction "+fnVar+" = Luan.checkFunction(" + expr.code + ");\n" + +"LuanFunction "+fnVar+" = Luan.checkFunction(" + ); + stmt.list.addAll( expr.list ); + stmt.list.add( "" + + ");\n" +"while(true) {\n" +"for( int i="+stackStart+"; i<"+symbolsSize()+"; i++ ) {\n" +"luan.stackSet(i,null);\n" +"}\n" +"LuanImpl.set(luan," + varsStr + ", "+fnVar+".call(luan) );\n" - +"if( " + firstVar.code + "==null ) break;\n" - + loop.code + +"if( " + ); + stmt.list.addAll( firstVar.list ); + stmt.list.add( "==null ) break;\n" ); + stmt.list.addAll( loop.list ); + stmt.list.add( "" +"}" +"} finally {\n" +"luan.stackClear("+stackStart+","+symbolsSize()+");\n" +"}\n" - ; - StmtString stmt = new StmtString(code); + ); popSymbols( symbolsSize() - stackStart ); return parser.success(stmt); } @@ -437,7 +471,7 @@ return parser.success(stmt); } - private boolean LocalStmt(StringBuilder stmts) throws ParseException { + private boolean LocalStmt(StmtString stmts) throws ParseException { parser.begin(); if( !Keyword("local",In.NOTHING) ) return parser.failure(); @@ -458,8 +492,9 @@ vars[i] = new SetLocalVar(stackStart+i); } String varsStr = varsToString(vars); - String code = "LuanImpl.set(luan," + varsStr + "," + values.code + ");\n"; - stmts.append(code); + stmts.list.add( "LuanImpl.set(luan," + varsStr + "," ); + stmts.list.addAll( values.list ); + stmts.list.add( ");\n" ); } addSymbols(names); return parser.success(); @@ -504,12 +539,13 @@ RequiredKeyword("do",In.NOTHING); StmtString loop = RequiredLoopBlock(); RequiredKeyword("end",In.NOTHING); - String code = "" - +"while( Luan.checkBoolean(" + cnd.code + ") ) {\n" - + loop.code - +"}\n" - ; - return parser.success( new StmtString(code) ); + StmtString stmt = new StmtString(); + stmt.list.add( "while( Luan.checkBoolean(" ); + stmt.list.addAll( cnd.list ); + stmt.list.add( ") ) {\n" ); + stmt.list.addAll( loop.list ); + stmt.list.add( "}\n" ); + return parser.success( stmt ); } private StmtString RepeatStmt() throws ParseException { @@ -519,12 +555,13 @@ StmtString loop =RequiredLoopBlock(); RequiredKeyword("until",In.NOTHING); ExpString cnd = RequiredExpr(In.NOTHING).expr(); - String code = "" - +"do {\n" - + loop.code - +"} while( !Luan.checkBoolean(" + cnd.code + ") );\n" - ; - return parser.success( new StmtString(code) ); + StmtString stmt = new StmtString(); + stmt.list.add( "do {\n" ); + stmt.list.addAll( loop.list ); + stmt.list.add( "} while( !Luan.checkBoolean(" ); + stmt.list.addAll( cnd.list ); + stmt.list.add( ") );\n" ); + return parser.success( stmt ); } private StmtString RequiredLoopBlock() throws ParseException { @@ -538,26 +575,33 @@ parser.begin(); if( !Keyword("if",In.NOTHING) ) return parser.failure(null); - StringBuilder sb = new StringBuilder(); + StmtString stmt = new StmtString(); ExpString cnd; StmtString block; cnd = RequiredExpr(In.NOTHING).expr(); RequiredKeyword("then",In.NOTHING); block = RequiredBlock(); - sb.append( "if( Luan.checkBoolean(" ).append( cnd.code ).append( ") ) {\n" ).append( block.code ); + stmt.list.add( "if( Luan.checkBoolean(" ); + stmt.list.addAll( cnd.list ); + stmt.list.add( ") ) {\n" ); + stmt.list.addAll( block.list ); while( Keyword("elseif",In.NOTHING) ) { cnd = RequiredExpr(In.NOTHING).expr(); RequiredKeyword("then",In.NOTHING); block = RequiredBlock(); - sb.append( "} else if( Luan.checkBoolean(" ).append( cnd.code ).append( ") ) {\n" ).append( block.code ); + stmt.list.add( "} else if( Luan.checkBoolean(" ); + stmt.list.addAll( cnd.list ); + stmt.list.add( ") ) {\n" ); + stmt.list.addAll( block.list ); } if( Keyword("else",In.NOTHING) ) { block = RequiredBlock(); - sb.append( "} else {\n" ).append( block.code ); + stmt.list.add( "} else {\n" ); + stmt.list.addAll( block.list ); } RequiredKeyword("end",In.NOTHING); - sb.append( "}\n" ); - return parser.success( new StmtString(sb.toString()) ); + stmt.list.add( "}\n" ); + return parser.success( stmt ); } private StmtString SetStmt() throws ParseException { @@ -584,21 +628,26 @@ int n = vars.size(); if( n == 1 ) return parser.success( vars.get(0).set(values) ); - StringBuilder sb = new StringBuilder(); - sb.append( "t = " + values.code + ";\n" ); - ExpString t = new ExpString("t",true,false); - sb.append( vars.get(0).set(new ExpString("t",true,false)).code ); + StmtString stmt = new StmtString(); + stmt.list.add( "t = " ); + stmt.list.addAll( values.list ); + stmt.list.add( ";\n" ); + ExpString t = new ExpString(true,false); + t.list.add( "t" ); + stmt.list.addAll( vars.get(0).set(t).list ); for( int i=1; i<n; i++ ) { - sb.append( vars.get(i).set(new ExpString("LuanImpl.pick(t,"+i+")",true,false)).code ); + t.list.clear(); + t.list.add( "LuanImpl.pick(t,"+i+")" ); + stmt.list.addAll( vars.get(i).set(t).list ); } - return parser.success( new StmtString(sb.toString()) ); + return parser.success( stmt ); } private static String varsToString(Settable[] vars) { StringBuilder sb = new StringBuilder(); sb.append( "new Settable[]{" ); for( Settable v : vars ) { - sb.append( new SettableString(v).code ).append( ',' ); + sb.append( settableToString(v) ).append( ',' ); } sb.append( "}" ); return sb.toString(); @@ -608,11 +657,16 @@ parser.begin(); ExpString exp = ExprZ(In.NOTHING); if( exp != null && exp.isStmt ) { - String code = exp.code; - if( exp.isExpr ) - code = "LuanImpl.nop(" + code + ")"; - code += ";\n"; - return parser.success( new StmtString(code) ); + StmtString stmt = new StmtString(); + if( exp.isExpr ) { + stmt.list.add( "LuanImpl.nop(" ); + stmt.list.addAll( exp.list ); + stmt.list.add( ")" ); + } else { + stmt.list.addAll( exp.list ); + } + stmt.list.add( ";\n" ); + return parser.success( stmt ); } return parser.failure(null); } @@ -642,8 +696,13 @@ while( Keyword("or",in) ) { exp = exp.expr(); ExpString exp2 = required(RelExpr(in)).expr(); - String code = "(LuanImpl.cnd(t = " + exp.code + ") ? t : (" + exp2.code + "))"; - exp = new ExpString(code,true,true); + ExpString newExp = new ExpString(true,true); + newExp.list.add( "(LuanImpl.cnd(t = " ); + newExp.list.addAll( exp.list ); + newExp.list.add( ") ? t : (" ); + newExp.list.addAll( exp2.list ); + newExp.list.add( "))" ); + exp = newExp; } return parser.success(exp); } @@ -656,8 +715,13 @@ while( Keyword("and",in) ) { exp = exp.expr(); ExpString exp2 = required(RelExpr(in)).expr(); - String code = "(LuanImpl.cnd(t = " + exp.code + ") ? (" + exp2.code + ") : t)"; - exp = new ExpString(code,true,true); + ExpString newExp = new ExpString(true,true); + newExp.list.add( "(LuanImpl.cnd(t = " ); + newExp.list.addAll( exp.list ); + newExp.list.add( ") ? (" ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ") : t)" ); + exp = newExp; } return parser.success(exp); } @@ -672,38 +736,68 @@ Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "LuanImpl.eq(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.eq(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( parser.match("~=") ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "!LuanImpl.eq(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "!LuanImpl.eq(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( parser.match("<=") ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "LuanImpl.le(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.le(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( parser.match(">=") ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "LuanImpl.le(luan," + exp2.code + "," + exp.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.le(luan," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( parser.match("<") ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "LuanImpl.lt(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); - } else if( parser.match(">") ) { + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.lt(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; + } else if( parser.match(">") ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "LuanImpl.lt(luan," + exp2.code + "," + exp.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.lt(luan," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp.list ); + newExp.list.add( ")" ); + exp = newExp; } else break; } @@ -719,8 +813,13 @@ Spaces(in); exp = exp.expr(); ExpString exp2 = required(ConcatExpr(in)).expr(); - String code = "LuanImpl.concat(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.concat(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } return parser.success(exp); } @@ -735,14 +834,24 @@ Spaces(in); exp = exp.expr(); ExpString exp2 = required(TermExpr(in)).expr(); - String code = "LuanImpl.add(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.add(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( Minus() ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(TermExpr(in)).expr(); - String code = "LuanImpl.sub(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.sub(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else break; } @@ -764,20 +873,35 @@ Spaces(in); exp = exp.expr(); ExpString exp2 = required(UnaryExpr(in)).expr(); - String code = "LuanImpl.mul(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.mul(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( parser.match('/') ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(UnaryExpr(in)).expr(); - String code = "LuanImpl.div(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.div(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else if( Mod() ) { Spaces(in); exp = exp.expr(); ExpString exp2 = required(UnaryExpr(in)).expr(); - String code = "LuanImpl.mod(luan," + exp.code + "," + exp2.code + ")"; - exp = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.mod(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.list ); + newExp.list.add( ")" ); + exp = newExp; } else break; } @@ -794,23 +918,29 @@ if( parser.match('#') ) { Spaces(in); ExpString exp = required(UnaryExpr(in)).expr(); - String code = "LuanImpl.len(luan," + exp.code + ")"; - exp = new ExpString(code,true,false); - return parser.success(exp); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.len(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( ")" ); + return parser.success(newExp); } if( Minus() ) { Spaces(in); ExpString exp = required(UnaryExpr(in)).expr(); - String code = "LuanImpl.unm(luan," + exp.code + ")"; - exp = new ExpString(code,true,false); - return parser.success(exp); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.unm(luan," ); + newExp.list.addAll( exp.list ); + newExp.list.add( ")" ); + return parser.success(newExp); } if( Keyword("not",in) ) { Spaces(in); ExpString exp = required(UnaryExpr(in)).expr(); - String code = "!Luan.checkBoolean(" + exp.code + ")"; - exp = new ExpString(code,true,false); - return parser.success(exp); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "!Luan.checkBoolean(" ); + newExp.list.addAll( exp.list ); + newExp.list.add( ")" ); + return parser.success(newExp); } ExpString exp = PowExpr(in); if( exp==null ) @@ -826,8 +956,13 @@ if( parser.match('^') ) { Spaces(in); ExpString exp2 = required(PowExpr(in)); - String code = "LuanImpl.pow(luan," + exp1.code + "," + exp2.code + ")"; - exp1 = new ExpString(code,true,false); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "LuanImpl.pow(luan," ); + newExp.list.addAll( exp1.expr().list ); + newExp.list.add( "," ); + newExp.list.addAll( exp2.expr().list ); + newExp.list.add( ")" ); + exp1 = newExp; } return parser.success(exp1); } @@ -906,8 +1041,11 @@ if( !parser.match('}') ) throw parser.exception("Expected table element or '}'"); Spaces(in); - String code = "LuanImpl.table(" + expString(builder).code + ")"; - return parser.success( new ExpString(code,true,false) ); + exp = new ExpString(true,false); + exp.list.add( "LuanImpl.table(" ); + exp.list.addAll( expString(builder).list ); + exp.list.add( ")" ); + return parser.success( exp ); } private boolean FieldSep() throws ParseException { @@ -922,8 +1060,13 @@ if( exp!=null && parser.match('=') ) { Spaces(In.NOTHING); ExpString val = RequiredExpr(In.NOTHING).expr(); - String code = "new TableField(" + exp.code + "," + val.code + ")"; - builder.add( new ExpString(code,true,false) ); + ExpString newExp = new ExpString(true,false); + newExp.list.add( "new TableField(" ); + newExp.list.addAll( exp.list ); + newExp.list.add( "," ); + newExp.list.addAll( val.list ); + newExp.list.add( ")" ); + builder.add( newExp ); return parser.success(); } parser.rollback(); @@ -1045,8 +1188,11 @@ } public StmtString set(ExpString val) throws ParseException { - String code = new SettableString(settable()).code + ".set(luan," + val.expr().code + ");\n"; - return new StmtString(code); + StmtString stmt = new StmtString(); + stmt.list.add( settableToString(settable()) + ".set(luan," ); + stmt.list.addAll( val.expr().list ); + stmt.list.add( ");\n" ); + return stmt; } }; } @@ -1084,8 +1230,15 @@ } */ public StmtString set(ExpString val) { - String code = "LuanImpl.put(luan," + table.expr().code + "," + key.expr().code + "," + val.expr().code + ");\n"; - return new StmtString(code); + StmtString stmt = new StmtString(); + stmt.list.add( "LuanImpl.put(luan," ); + stmt.list.addAll( table.expr().list ); + stmt.list.add( "," ); + stmt.list.addAll( key.expr().list ); + stmt.list.add( "," ); + stmt.list.addAll( val.expr().list ); + stmt.list.add( ");\n" ); + return stmt; } }; } @@ -1253,17 +1406,25 @@ private ExpString Literal(In in) throws ParseException { parser.begin(); - if( NilLiteral(in) ) - return parser.success(new ExpString("null",true,false)); + if( NilLiteral(in) ) { + ExpString exp = new ExpString(true,false); + exp.list.add( "null" ); + return parser.success(exp); + } Boolean b = BooleanLiteral(in); - if( b != null ) - return parser.success(new ExpString(b.toString(),true,false)); + if( b != null ) { + ExpString exp = new ExpString(true,false); + exp.list.add( b.toString() ); + return parser.success(exp); + } Number n = NumberLiteral(in); if( n != null ) { String s = n.toString(); if( n instanceof Long ) s += "L"; - return parser.success(new ExpString(s,true,false)); + ExpString exp = new ExpString(true,false); + exp.list.add( s ); + return parser.success(exp); } String s = StringLiteral(in); if( s != null ) @@ -1280,7 +1441,9 @@ .replace("\t","\\t") .replace("\b","\\b") ; - return new ExpString( "\""+s+"\"", true,false); + ExpString exp = new ExpString(true,false); + exp.list.add( "\""+s+"\"" ); + return exp; } private boolean NilLiteral(In in) throws ParseException { @@ -1533,67 +1696,78 @@ private static int classCounter = 0; private class ExpString { - final String code; + final List list = new ArrayList(); final boolean isExpr; final boolean isStmt; ExpString(Expressions exp) { if( exp==null ) throw new NullPointerException(); int i = LuanImpl.addObj(exp); - code = "((Expressions)LuanImpl.getObj(" + i + ")).eval(luan)"; + list.add( "((Expressions)LuanImpl.getObj(" + i + ")).eval(luan)" ); isExpr = exp instanceof Expr; isStmt = exp instanceof StmtExp; } - ExpString(String code,boolean isExpr,boolean isStmt) { - this.code = code; + ExpString(boolean isExpr,boolean isStmt) { this.isExpr = isExpr; this.isStmt = isStmt; } ExpString expr() { - return isExpr ? this : new ExpString( "Luan.first(" + code + ")", true, isStmt ); + if( isExpr ) + return this; + ExpString exp = new ExpString(true,isStmt); + exp.list.addAll( list ); + return exp; } } private ExpString expString(List<ExpString> list) { + ExpString exp = new ExpString(false,false); switch(list.size()) { case 0: - return new ExpString("LuanFunction.NOTHING",false,false); + exp.list.add("LuanFunction.NOTHING"); + return exp; case 1: return list.get(0); default: int lastI = list.size() - 1; - StringBuilder sb = new StringBuilder(); - sb.append( "new Object[]{" ); + exp.list.add( "new Object[]{" ); for( int i=0; i<lastI; i++ ) { - sb.append( list.get(i).expr().code ).append( ',' ); + exp.list.addAll( list.get(i).expr().list ); + exp.list.add( "," ); } ExpString last = list.get(lastI); if( last.isExpr ) { - sb.append( last.code ).append( '}' ); - return new ExpString(sb.toString(),false,false); + exp.list.addAll( last.list ); + exp.list.add( "}" ); } else { - sb.append( '}' ); - String s = "LuanImpl.concatArgs(" + sb + "," + last.code + ")"; - return new ExpString(s,false,false); + exp.list.add( "}" ); + exp.list.add( 0, "LuanImpl.concatArgs(" ); + exp.list.add( "," ); + exp.list.addAll( last.list ); + exp.list.add( ")" ); } + return exp; } } private static class StmtString { - final String code; + final List list = new ArrayList(); boolean hasReturn = false; - - StmtString(String code) { - this.code = code; - if( code.contains("LuanParser") ) throw new RuntimeException("\n"+code); - } } - private static StmtString EMPTY_STMT = new StmtString(""); + private static FnDef toFnDef(StmtString stmt,int stackSize,int numArgs,boolean isVarArg,UpValue.Getter[] upValueGetters) { + StringBuilder sb = new StringBuilder(); + for( Object o : stmt.list ) { + if( o instanceof List ) throw new RuntimeException(); + if( o instanceof ExpString ) throw new RuntimeException(); + if( o instanceof StmtString ) throw new RuntimeException(); + sb.append( o.toString() ); + } + String code = sb.toString(); +//System.out.println("code:\n"+code); - private static FnDef toFnDef(StmtString stmt,int stackSize,int numArgs,boolean isVarArg,UpValue.Getter[] upValueGetters) { int i = LuanImpl.addObj(upValueGetters); String className = "EXP" + ++classCounter; String classCode = "" @@ -1610,7 +1784,7 @@ +"\n" +" @Override public Object run(LuanStateImpl luan) throws LuanException {\n" +" Object t;\n" - +" " + stmt.code + +" " + code +" }\n" +"}\n" ; @@ -1629,32 +1803,27 @@ private ExpString toFnExpStr(StmtString stmt,int stackSize,int numArgs,boolean isVarArg,UpValue.Getter[] upValueGetters) { int i = LuanImpl.addObj(upValueGetters); - String classCode = "" + ExpString exp = new ExpString(true,false); + exp.list.add( "" +"\n" +"new FnDef("+stackSize+","+numArgs+","+isVarArg+",(UpValue.Getter[])LuanImpl.getObj("+i+")) {\n" +" @Override public Object run(LuanStateImpl luan) throws LuanException {\n" +" Object t;\n" - +" " + stmt.code + +" " + ); + exp.list.addAll( stmt.list ); + exp.list.add( "" +" }\n" +"}.eval(luan)\n" - ; - return new ExpString(classCode,true,false); + ); + return exp; } - private static class SettableString { - final String code; - - SettableString(Settable settable) { - if( settable==null ) throw new NullPointerException(); - int i = LuanImpl.addObj(settable); - code = "((Settable)LuanImpl.getObj(" + i + "))"; - } - - SettableString(String code) { - this.code = code; - } - + private static String settableToString(Settable settable) { + if( settable==null ) throw new NullPointerException(); + int i = LuanImpl.addObj(settable); + return"((Settable)LuanImpl.getObj(" + i + "))"; } }