Mercurial > hg > Members > kono > jpf-core
view src/tests/gov/nasa/jpf/test/mc/threads/MissedPathTest.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.threads; import org.junit.Test; import gov.nasa.jpf.util.test.TestJPF; import gov.nasa.jpf.vm.Verify; /** * test for missed paths in concurrent threads with very little interaction */ public class MissedPathTest extends TestJPF { static class X { boolean pass; } static class InstanceFieldPropagation extends Thread { X myX; // initially not set @Override public void run() { if (myX != null){ Verify.println("T: accessed global myX"); if (!myX.pass){ // (2) won't fail unless main is between (0) and (1) throw new AssertionError("gotcha"); } } } } @Test public void testInstanceFieldPropagation () { if (verifyAssertionErrorDetails("gotcha", "+vm.shared.break_on_exposure=true")) { InstanceFieldPropagation mp = new InstanceFieldPropagation(); mp.start(); X x = new X(); Verify.println("M: new " + x); mp.myX = x; // (0) exposure - x cannot become shared until this GOT executed //Thread.yield(); // this would expose the error Verify.println("M: x.pass=true"); x.pass = true; // (1) need to break BEFORE assignment or no error } } //---------------------------------------------------------------------------------- static class Y { X x; } static Y globalY; // initially not set static class StaticFieldPropagation extends Thread { @Override public void run(){ if (globalY != null){ if (!globalY.x.pass){ // (2) won't fail unless main is between (0) and (1) throw new AssertionError("gotcha"); } } } } @Test public void testStaticFieldPropagation () { if (verifyAssertionErrorDetails("gotcha", "+vm.shared.break_on_exposure=true")) { StaticFieldPropagation mp = new StaticFieldPropagation(); mp.start(); X x = new X(); Y y = new Y(); y.x = x; globalY = y; // (0) x not shared until this GOT executed //Thread.yield(); // this would expose the error x.pass = true; // (1) need to break BEFORE assignment or no error } } //------------------------------------------------------------------------------- static class PutContender extends Thread { X myX; @Override public void run () { myX = new X(); // competing put with exposure if (myX != null) { // doesn't matter, we just want to GET myX Verify.println("T: accessed global myX"); } } } // this does not really belong here since it doesn't test not missing paths, but // if the exposure CGs we use to avoid missing paths are not causing infinite loops. // NOTE: turning off state matching is crucial here @Test public void testCompetingExposures(){ if (verifyNoPropertyViolation("+vm.storage.class=nil")){ PutContender mp = new PutContender(); mp.start(); X x = new X(); Verify.println("M: new " + x); mp.myX = x; // this is one of the competing PUTs Verify.println("M: x.pass=true"); x.pass = true; // irrelevant in this case } } }