org.formulacompiler.compiler.internal.bytecode.ExpressionCompilerForStrings_Base.java Source code

Java tutorial

Introduction

Here is the source code for org.formulacompiler.compiler.internal.bytecode.ExpressionCompilerForStrings_Base.java

Source

/*
 * Copyright (c) 2006-2009 by Abacus Research AG, Switzerland.
 * All rights reserved.
 *
 * This file is part of the Abacus Formula Compiler (AFC).
 *
 * For commercial licensing, please contact sales(at)formulacompiler.com.
 *
 * AFC is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * AFC 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with AFC.  If not, see <http://www.gnu.org/licenses/>.
 */

package org.formulacompiler.compiler.internal.bytecode;

import java.util.Collection;
import java.util.List;

import org.formulacompiler.compiler.CompilerException;
import org.formulacompiler.compiler.internal.expressions.DataType;
import org.formulacompiler.compiler.internal.expressions.ExpressionNode;
import org.formulacompiler.compiler.internal.expressions.ExpressionNodeForOperator;
import org.formulacompiler.compiler.internal.model.ExpressionNodeForCount;
import org.formulacompiler.runtime.ScaledLong;
import org.objectweb.asm.Opcodes;

abstract class ExpressionCompilerForStrings_Base extends ExpressionCompilerForAll_Generated {
    private static final String SNAME = TypeCompilerForStrings.SNAME;
    private static final String S = TypeCompilerForStrings.S;
    private static final String S2I = "(" + S + ")I";
    private static final String S2Z = "(" + S + ")Z";

    public ExpressionCompilerForStrings_Base(MethodCompiler _methodCompiler) {
        super(_methodCompiler);
    }

    @Override
    protected TypeCompiler typeCompiler() {
        return section().engineCompiler().stringCompiler();
    }

    protected abstract void compile_util_toLong() throws CompilerException;

    protected abstract void compile_util_toInt() throws CompilerException;

    protected abstract void compile_util_toShort() throws CompilerException;

    protected abstract void compile_util_toByte() throws CompilerException;

    protected abstract void compile_util_toBigDecimal() throws CompilerException;

    protected abstract void compile_util_toBigInteger() throws CompilerException;

    protected abstract void compile_util_toFloat() throws CompilerException;

    protected abstract void compile_util_toDouble() throws CompilerException;

    protected abstract void compile_util_toBoolean() throws CompilerException;

    protected abstract void compile_util_toDate() throws CompilerException;

    @Override
    protected void compileConversionFrom(DataType _type) throws CompilerException {
        switch (_type) {
        case NULL:
            compile_util_fromNull();
            return;
        case NUMERIC:
            method().numericCompiler().compileConversionToString();
            return;
        default:
            throw new CompilerException.UnsupportedDataType(
                    "Cannot convert from a " + _type + " to a " + this + ".");
        }
    }

    @Override
    protected final void compileConversionToNumber() throws CompilerException {
        compile_util_toBigDecimal();
    }

    @Override
    protected final void compileConversionToLong() throws CompilerException {
        compile_util_toLong();
    }

    @Override
    protected void compileConversionToInt() throws CompilerException {
        compile_util_toInt();
    }

    @Override
    protected void compileConversionToShort() throws CompilerException {
        compile_util_toShort();
    }

    @Override
    protected void compileConversionToByte() throws CompilerException {
        compile_util_toByte();
    }

    @Override
    protected void compileConversionToBigDecimal() throws CompilerException {
        compile_util_toBigDecimal();
    }

    @Override
    protected void compileConversionToBigInteger() throws CompilerException {
        compile_util_toBigInteger();
    }

    @Override
    protected void compileConversionToFloat() throws CompilerException {
        compile_util_toFloat();
    }

    @Override
    protected void compileConversionToDouble() throws CompilerException {
        compile_util_toDouble();
    }

    @Override
    protected void compileConversionToBoolean() throws CompilerException {
        compile_util_toBoolean();
    }

    @Override
    protected void compileConversionToString() throws CompilerException {
        // Nothing to do here.
    }

    @Override
    protected void compileConversionToDate() throws CompilerException {
        compile_util_toDate();
    }

    @Override
    protected void compileConversionToObject() throws CompilerException {
        // Nothing to do here.
    }

    @Override
    protected void compileConversionFrom(final ScaledLong _scale) throws CompilerException {
        compile_util_fromScaledLong(_scale.value());
    }

    @Override
    protected void compileConversionTo(ScaledLong _scale) throws CompilerException {
        compile_util_toScaledLong(_scale.value());
    }

    protected abstract void compile_util_fromScaledLong(int _scale) throws CompilerException;

    protected abstract void compile_util_toScaledLong(int _scale) throws CompilerException;

    @Override
    protected void compileOperator(ExpressionNodeForOperator _node) throws CompilerException {
        final List<ExpressionNode> args = _node.arguments();
        switch (_node.getOperator()) {

        case CONCAT:
            switch (args.size()) {
            case 0:
                throw new CompilerException.UnsupportedExpression("CONCAT needs at least one argument.");
            case 1:
                compile(args.get(0));
                break;
            default:
                compileConcatenation(args);
            }
            return;

        }
        super.compileOperator(_node);
    }

    private final void compileConcatenation(Collection<ExpressionNode> _args) throws CompilerException {
        boolean first = true;
        for (ExpressionNode arg : _args) {
            if (first) {
                compile_utilFun_newBuilder(arg);
                first = false;
            } else {
                compile_utilOp_appendBuilder(arg);
            }
        }
        compile_utilOp_fromBuilder();
    }

    @Override
    protected int compileComparison(int _ifOpcode, int _comparisonOpcode) throws CompilerException {
        switch (_ifOpcode) {

        /*
         * This may seem counter-intuitive here, but the contract is to return 0 for equality.
         * Boolean true, however, is 1. So we invert the test for EQ and NE.
         */

        case Opcodes.IFEQ:
            mv().visitMethodInsn(Opcodes.INVOKEVIRTUAL, SNAME, "equalsIgnoreCase", S2Z);
            return Opcodes.IFNE;

        case Opcodes.IFNE:
            mv().visitMethodInsn(Opcodes.INVOKEVIRTUAL, SNAME, "equalsIgnoreCase", S2Z);
            return Opcodes.IFEQ;

        default:
            mv().visitMethodInsn(Opcodes.INVOKEVIRTUAL, SNAME, "compareToIgnoreCase", S2I);
            return _ifOpcode;

        }
    }

    @Override
    protected void compileCount(ExpressionNodeForCount _node) throws CompilerException {
        throw new AbstractMethodError();
    }

    @Override
    public String toString() {
        return "string";
    }

    protected abstract void compile_util_fromNull() throws CompilerException;

    protected abstract void compile_utilFun_newBuilder(ExpressionNode _arg) throws CompilerException;

    protected abstract void compile_utilOp_appendBuilder(ExpressionNode _arg) throws CompilerException;

    protected abstract void compile_utilOp_fromBuilder() throws CompilerException;

}