org.adjective.stout.operation.InvokeVirtualOperation.java Source code

Java tutorial

Introduction

Here is the source code for org.adjective.stout.operation.InvokeVirtualOperation.java

Source

/* ------------------------------------------------------------------------
 * Copyright 2009 Tim Vernum
 * ------------------------------------------------------------------------
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * ------------------------------------------------------------------------
 */

package org.adjective.stout.operation;

import org.objectweb.asm.Opcodes;

import org.adjective.stout.core.ElementModifier;
import org.adjective.stout.core.ExecutionStack;
import org.adjective.stout.core.ExtendedType;
import org.adjective.stout.core.InstructionCollector;
import org.adjective.stout.core.MethodSignature;
import org.adjective.stout.core.Operation;
import org.adjective.stout.core.UnresolvedType;
import org.adjective.stout.core.UnresolvedType.Sort;
import org.adjective.stout.instruction.MethodInstruction;

/**
 * @author <a href="http://blog.adjective.org/">Tim Vernum</a>
 */
public class InvokeVirtualOperation implements Operation {
    private final Expression _target;
    private final UnresolvedType _targetType;
    private final MethodSignature _method;
    private final Expression[] _arguments;

    public InvokeVirtualOperation(Expression target, UnresolvedType targetType, MethodSignature method,
            Expression... arguments) {
        if (method.getModifiers().contains(ElementModifier.STATIC)) {
            throw new IllegalArgumentException(
                    "The static method " + method.getName() + " cannot have a virtual invocation");
        }
        _target = target;
        _targetType = targetType;
        _method = method;
        _arguments = arguments;
        if (targetType == null && target != null) {
            throw new NullPointerException("Target type cannot be null when target expression is specified");
        }
    }

    public boolean hasReturnValue() {
        return getReturnType().getRawClass() != Void.TYPE;
    }

    public ExtendedType getReturnType() {
        return _method.getReturnType();
    }

    public void getInstructions(ExecutionStack stack, InstructionCollector collector) {
        if (_target == null) {
            ThisExpression.INSTANCE.getInstructions(stack, collector);
        } else {
            _target.getInstructions(stack, collector);
        }

        for (Expression expression : _arguments) {
            expression.getInstructions(stack, collector);
        }
        UnresolvedType targetType = _targetType == null ? stack.currentClass() : _targetType;
        int opCode = targetType.getSort() == Sort.INTERFACE ? Opcodes.INVOKEINTERFACE : Opcodes.INVOKEVIRTUAL;
        collector.add(new MethodInstruction(opCode, targetType.getInternalName(), _method));
    }

}