view src/tests/gov/nasa/jpf/test/mc/basic/SkipInstructionTest.java @ 0:61d41facf527

initial v8 import (history reset)
author Peter Mehlitz <Peter.C.Mehlitz@nasa.gov>
date Fri, 23 Jan 2015 10:14:01 -0800
parents
children
line wrap: on
line source

/*
 * Copyright (C) 2014, United States Government, as represented by the
 * Administrator of the National Aeronautics and Space Administration.
 * All rights reserved.
 *
 * The Java Pathfinder core (jpf-core) platform is 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 gov.nasa.jpf.test.mc.basic;

import gov.nasa.jpf.ListenerAdapter;
import gov.nasa.jpf.jvm.bytecode.GETFIELD;
import gov.nasa.jpf.jvm.bytecode.IRETURN;
import gov.nasa.jpf.jvm.bytecode.JVMInvokeInstruction;
import gov.nasa.jpf.util.test.TestJPF;
import gov.nasa.jpf.vm.ClassInfo;
import gov.nasa.jpf.vm.Instruction;
import gov.nasa.jpf.vm.VM;
import gov.nasa.jpf.vm.MethodInfo;
import gov.nasa.jpf.vm.StackFrame;
import gov.nasa.jpf.vm.ThreadInfo;
import gov.nasa.jpf.vm.choice.IntChoiceFromList;

import org.junit.Test;

public class SkipInstructionTest extends TestJPF {

  //--- replacing field access

  int answer = 0;

  public static class GetFieldListener extends ListenerAdapter {
    
    @Override
    public void executeInstruction(VM vm, ThreadInfo ti, Instruction insnToExecute) {
      Instruction pc = ti.getPC();

      if (pc instanceof GETFIELD) {
        GETFIELD gf = (GETFIELD) pc;
        if (gf.getVariableId().equals(SkipInstructionTest.class.getName() + ".answer")) {
          System.out.println("now intercepting: " + pc);

          // simulate the operand stack behavior of the skipped insn
          StackFrame frame = ti.getModifiableTopFrame();

          frame.pop();
          frame.push(42);

          ti.skipInstruction(pc.getNext());
        }
      }
    }
  }
  
  @Test
  public void testGETFIELD () {

    if (verifyNoPropertyViolation("+listener=gov.nasa.jpf.test.mc.basic.SkipInstructionTest$GetFieldListener")){
      int i = answer; // to be intercepted by listener
    
      System.out.println(i);
      assert (i == 42) : "get_field not intercepted: " + i;
    }
  }
  
  //--- replacing method execution
  // this is not using skipInstruction but is setting the next one to enter, which
  // means it can skip over a whole bunch of things. The use that comes to mind is to
  // skip over method bodies, but still preserve the invoke processing so that we
  // preserve the locking/sync (which we otherwise would have to do explicitly from
  // a pre-exec listener
  
  // the intercepted method
  int foo (int a, int b) {
    int result = a + b;
    if (result > 10) {
      return 10;
    } else {
      return result;
    }
  }
  
  // the listener that does the interception
  public static class InvokeListener extends ListenerAdapter {
    MethodInfo interceptedMethod;
    
    @Override
    public void classLoaded (VM vm, ClassInfo ci) {
      if (ci.getName().equals("gov.nasa.jpf.test.mc.basic.SkipInstructionTest")) {
        interceptedMethod = ci.getMethod("foo(II)I", false);
        assert interceptedMethod != null : "foo(II)I not found";
        System.out.println("method to intercept: " + interceptedMethod);
      }
    }
        
    @Override
    public void instructionExecuted (VM vm, ThreadInfo ti, Instruction nextInsn, Instruction executedInsn) {
      MethodInfo mi = ti.getTopFrameMethodInfo();
      if (mi == interceptedMethod) {
        System.out.println("in " + mi);

        if (!ti.isFirstStepInsn()) {
          System.out.println("in top half: " + executedInsn);
          if (executedInsn instanceof JVMInvokeInstruction) { // we just entered the interceptedMethod
            IntChoiceFromList cg = new IntChoiceFromList("Test", 42, 43);
            if (vm.setNextChoiceGenerator(cg)) {
              return; // we are done here, next insn in this method will skip to return
            }
          }
        } else {
          // note - this is the first insn WITHIN the interceptedMethod
          System.out.println("in bottom half: " + executedInsn);
          IntChoiceFromList cg = vm.getCurrentChoiceGenerator("Test", IntChoiceFromList.class);
          if (cg != null) {
            int choice = cg.getNextChoice();
            Instruction lastInsn = mi.getLastInsn();
            assert lastInsn instanceof IRETURN : "last instruction not an IRETURN ";
            StackFrame frame = ti.getModifiableTopFrame(); // we are modifying it
            System.out.println("listener is skipping method body of " + mi + " returning " + choice);
            frame.push(choice);
            ti.setNextPC(lastInsn);
          } else {
            System.out.println("unexpected CG: " + cg);
          }
        }
      }      
    }    
  }
  
  @Test
  public void testSkipMethodBody() {
    if (verifyNoPropertyViolation("+listener=gov.nasa.jpf.test.mc.basic.SkipInstructionTest$InvokeListener")){
      int ret = foo( 3, 4);
      System.out.println(ret);
      //assertTrue( ret == 42);
    }    
  }
}