JsrInstruction.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  *  Unless required by applicable law or agreed to in writing, software
  12.  *  distributed under the License is distributed on an "AS IS" BASIS,
  13.  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  *  See the License for the specific language governing permissions and
  15.  *  limitations under the License.
  16.  */
  17. package org.apache.bcel.generic;

  18. /**
  19.  * Super class for JSR - Jump to subroutine
  20.  */
  21. public abstract class JsrInstruction extends BranchInstruction implements UnconditionalBranch, TypedInstruction, StackProducer {

  22.     /**
  23.      * Empty constructor needed for Instruction.readInstruction. Not to be used otherwise.
  24.      */
  25.     JsrInstruction() {
  26.     }

  27.     JsrInstruction(final short opcode, final InstructionHandle target) {
  28.         super(opcode, target);
  29.     }

  30.     /**
  31.      * @return return address type
  32.      */
  33.     @Override
  34.     public Type getType(final ConstantPoolGen cp) {
  35.         return new ReturnaddressType(physicalSuccessor());
  36.     }

  37.     /**
  38.      * Returns an InstructionHandle to the physical successor of this JsrInstruction. <B>For this method to work, this
  39.      * JsrInstruction object must not be shared between multiple InstructionHandle objects!</B> Formally, there must not be
  40.      * InstructionHandle objects i, j where i != j and i.getInstruction() == this == j.getInstruction().
  41.      *
  42.      * @return an InstructionHandle to the "next" instruction that will be executed when RETurned from a subroutine.
  43.      */
  44.     public InstructionHandle physicalSuccessor() {
  45.         InstructionHandle ih = super.getTarget();
  46.         // Rewind!
  47.         while (ih.getPrev() != null) {
  48.             ih = ih.getPrev();
  49.         }
  50.         // Find the handle for "this" JsrInstruction object.
  51.         while (ih.getInstruction() != this) {
  52.             ih = ih.getNext();
  53.         }
  54.         final InstructionHandle toThis = ih;
  55.         while (ih != null) {
  56.             ih = ih.getNext();
  57.             if (ih != null && ih.getInstruction() == this) {
  58.                 throw new IllegalStateException("physicalSuccessor() called on a shared JsrInstruction.");
  59.             }
  60.         }
  61.         // Return the physical successor
  62.         return toThis.getNext();
  63.     }
  64. }