All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jboss.classfilewriter.code.StackFrame Maven / Gradle / Ivy

/*
 * JBoss, Home of Professional Open Source.
 *
 * Copyright 2012 Red Hat, Inc.
 *
 * 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.jboss.classfilewriter.code;

import org.jboss.classfilewriter.ClassMethod;

/**
 * Represents a stack frame in the virtual machine. Holds the state of the local variable array and the stack
 *
 * @author Stuart Douglas
 *
 */
public class StackFrame {

    /**
     * The current state of the stack
     */
    private final StackState stackState;

    /**
     * The local variable state
     */
    private final LocalVariableState localVariableState;

    private final StackFrameType type;

    /**
     * Creates the initial stack frame
     */
    public StackFrame(ClassMethod method) {
        this.stackState = new StackState(method.getClassFile().getConstPool());
        this.localVariableState = new LocalVariableState(method);
        this.type = StackFrameType.FULL_FRAME;
    }

    public StackFrame(StackState stackState, LocalVariableState localVariableState, StackFrameType type) {
        this.stackState = stackState;
        this.localVariableState = localVariableState;
        this.type = type;
    }

    public StackState getStackState() {
        return stackState;
    }

    public LocalVariableState getLocalVariableState() {
        return localVariableState;
    }

    /**
     * push an operand of the given type onto the stack
     * 

* If the entry is wide then a corresponding TOP type will be created */ public StackFrame push(String type) { StackState ns = stackState.push(type); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * push an operand of the given type onto the stack. *

* If the entry is wide then a corresponding TOP type will be created */ public StackFrame push(StackEntry entry) { StackState ns = stackState.push(entry); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * pushes a null type onto the stack * * @return */ public StackFrame aconstNull() { StackState ns = stackState.aconstNull(); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } public StackFrame pop(int no) { StackState ns = stackState.pop(no); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * pops an operand from the stack */ public StackFrame pop() { StackState ns = stackState.pop(1); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * pops 2 operands from the stack */ public StackFrame pop2() { StackState ns = stackState.pop(2); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * pops 3 operands from the stack */ public StackFrame pop3() { StackState ns = stackState.pop(3); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * pops 4 operands from the stack */ public StackFrame pop4() { StackState ns = stackState.pop(4); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * replace the operand at the top of the stack with the given operand */ public StackFrame replace(String type) { // TODO: inefficinet StackState ns = stackState.pop(1).push(type); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } public StackFrame dup() { StackState ns = stackState.dup(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } public StackFrame dupX1() { StackState ns = stackState.dupX1(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } public StackFrame dupX2() { StackState ns = stackState.dupX2(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } public StackFrame dup2() { StackState ns = stackState.dup2(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } public StackFrame dup2X1() { StackState ns = stackState.dup2X1(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } public StackFrame dup2X2() { StackState ns = stackState.dup2X2(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } /** * Store the variable on top of the stack into a local variable, poping the variable from the stack. Wide types are handled * automatically */ public StackFrame store(int no) { StackEntry top = stackState.top(); StackState ns; LocalVariableState ls; if(top.getType() == StackEntryType.TOP) { //wide type StackEntry type = stackState.top_1(); ns = stackState.pop(2); ls = localVariableState.storeWide(no, type); } else { StackEntry type = stackState.top(); ns = stackState.pop(1); ls = localVariableState.store(no, type); } return new StackFrame(ns, ls, StackFrameType.FULL_FRAME); } /** * remote the top two operands and replace them with an different operand * */ public StackFrame pop2push1(String type) { StackState ns = stackState.pop(2).push(type); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } /** * remote the top two operands and replace them with an different operand * */ public StackFrame pop4push1(String type) { StackState ns = stackState.pop(4).push(type); return new StackFrame(ns, localVariableState, typeNoLocalChange(ns)); } @Override public String toString() { return "StackFrame [localVariableState=" + localVariableState + ", stackState=" + stackState + "]"; } /** * marks the value in potition initializedValueStackPosition as initialized. This also pops this value and everything above * it */ public StackFrame constructorCall(int initializedValueStackPosition) { StackEntry entry = stackState.getContents().get(stackState.getContents().size() - 1 - initializedValueStackPosition); StackState ns = stackState.constructorCall(initializedValueStackPosition, entry); LocalVariableState locals = localVariableState.constructorCall(entry); return new StackFrame(ns, locals, StackFrameType.FULL_FRAME); } public StackFrame swap() { StackState ns = stackState.swap(); return new StackFrame(ns, localVariableState, StackFrameType.FULL_FRAME); } /** * determins the type of stack frame this is when no local variables have changed */ private StackFrameType typeNoLocalChange(StackState ns) { int size = ns.getContents().size(); if (size == 0) { return StackFrameType.SAME_FRAME; } else if (size == 1) { return StackFrameType.SAME_LOCALS_1_STACK; } return StackFrameType.FULL_FRAME; } public StackFrame mergeStack(int pos, StackEntry frame) { StackState stack = stackState.updateMerged(pos, frame); return new StackFrame(stack, localVariableState, StackFrameType.FULL_FRAME); } public StackFrame mergeLocals(int pos, StackEntry frame) { LocalVariableState locals = localVariableState.updateMerged(pos, frame); return new StackFrame(stackState, locals, StackFrameType.FULL_FRAME); } public StackFrameType getType() { return type; } public StackFrame createFull() { return new StackFrame(stackState, localVariableState, StackFrameType.FULL_FRAME); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy