package railo.transformer.bytecode.cast; import org.objectweb.asm.Type; import org.objectweb.asm.commons.GeneratorAdapter; import org.objectweb.asm.commons.Method; import railo.runtime.exp.TemplateException; import railo.transformer.bytecode.BytecodeContext; import railo.transformer.bytecode.BytecodeException; import railo.transformer.bytecode.Literal; import railo.transformer.bytecode.expression.ExprBoolean; import railo.transformer.bytecode.expression.ExprDouble; import railo.transformer.bytecode.expression.ExprFloat; import railo.transformer.bytecode.expression.ExprString; import railo.transformer.bytecode.expression.Expression; import railo.transformer.bytecode.expression.ExpressionBase; import railo.transformer.bytecode.literal.LitFloat; import railo.transformer.bytecode.op.OpDouble; import railo.transformer.bytecode.util.Methods; import railo.transformer.bytecode.util.Types; /** * cast a Expression to a Double */ public final class CastFloat extends ExpressionBase implements ExprFloat,Cast { private Expression expr; private CastFloat(Expression expr) { super(expr.getStart(),expr.getEnd()); this.expr=expr; } /** * Create a String expression from a Expression * @param expr * @return String expression * @throws TemplateException */ public static ExprFloat toExprFloat(Expression expr) { if(expr instanceof ExprFloat) return (ExprFloat) expr; if(expr instanceof Literal) { Double dbl = ((Literal)expr).getDouble(null); if(dbl!=null) return new LitFloat((float)dbl.doubleValue(),expr.getStart(),expr.getEnd()); } return new CastFloat(expr); } /** * @see railo.transformer.bytecode.expression.Expression#_writeOut(org.objectweb.asm.commons.GeneratorAdapter, int) */ public Type _writeOut(BytecodeContext bc, int mode) throws BytecodeException { GeneratorAdapter adapter = bc.getAdapter(); if(expr instanceof OpDouble) { ((OpDouble)expr).writeOutDouble(bc,MODE_VALUE); if(mode==MODE_VALUE)adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE_FROM_DOUBLE); else adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_DOUBLE); } else if(expr instanceof ExprBoolean) { expr.writeOut(bc,MODE_VALUE); if(mode==MODE_VALUE)adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE_FROM_BOOLEAN); else adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_BOOLEAN); } else if(expr instanceof ExprFloat) { expr.writeOut(bc,mode); } else if(expr instanceof ExprDouble) { expr.writeOut(bc,MODE_VALUE); if(mode==MODE_VALUE)adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE_FROM_DOUBLE); else adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_DOUBLE); } else if(expr instanceof ExprString) { expr.writeOut(bc,MODE_REF); if(mode==MODE_VALUE)adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE_FROM_STRING); else adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_STRING); } else { Type rtn = expr.writeOut(bc,mode); if(mode==MODE_VALUE) { if(!Types.isPrimitiveType(rtn)) { adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE); } else if(Types.DOUBLE_VALUE.equals(rtn)) { adapter.cast(Types.DOUBLE_VALUE, Types.FLOAT_VALUE); } else if(Types.FLOAT_VALUE.equals(rtn)) {} else if(Types.BOOLEAN_VALUE.equals(rtn)) { adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE_FROM_BOOLEAN); } else { adapter.invokeStatic(Types.CASTER,new Method("toRef",Types.toRefType(rtn),new Type[]{rtn})); adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_VALUE); } return Types.FLOAT_VALUE; } else if(Types.isPrimitiveType(rtn)) { if(Types.DOUBLE_VALUE.equals(rtn)) { adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_DOUBLE); } else if(Types.FLOAT_VALUE.equals(rtn)) { adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_FLOAT); } else if(Types.BOOLEAN_VALUE.equals(rtn)) { adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT_FROM_BOOLEAN); } else { adapter.invokeStatic(Types.CASTER,new Method("toRef",Types.toRefType(rtn),new Type[]{rtn})); adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT); } return Types.FLOAT; } //else { if(!Types.FLOAT.equals(rtn)) adapter.invokeStatic(Types.CASTER,Methods.METHOD_TO_FLOAT); return Types.FLOAT; //} } if(mode==MODE_VALUE)return Types.FLOAT_VALUE; return Types.FLOAT; } @Override public Expression getExpr() { return expr; } }