/* * Copyright (c) 2007, 2020, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* * jnistress004 is a class that sets up classes that do the work * for the test. * * The Interrupter objects send interrupts to the JNIters. * The GarbageGenerator objects generate garbage. * * sync[0] synchronizes the test cycles. * sync[1] synchronizes access to exception counters. * sync[2] synchronizes the cycle count update. It also insures that * the interrupts do not interfere with the cycle count updates. * This is because cycle count updates are used to define cycles. */ /* * @test * @key stress * * @summary converted from VM testbase nsk/stress/jni/jnistress004. * VM testbase keywords: [stress, quick, feature_283, nonconcurrent] * * @library /vmTestbase * /test/lib * @run main/othervm/native * nsk.stress.jni.jnistress004 * -numTHREADer 20 * -threadInterval 200 * -numInterrupter 2 * -interruptInterval 500 * -numGarbage 80 * -garbageInterval 5 * -numIteration 260 */ package nsk.stress.jni; import nsk.share.Consts; import nsk.share.Debug; import nsk.share.test.StressOptions; public class jnistress004 extends Thread { /* Maximum number of iterations. Ignored if <= 0L */ static long numIteration = 2L; /* Timeout */ static long timeOut; /* Number of test class objects */ static int numJNIter = 100; /* Time between JNI stressing by the threads under test */ /* (in milliseconds) */ static int jniInterval = 25; /* Number of interrupting threads */ static int numInterrupter = 10; /* Time between interrupts in milliseconds */ static int interruptInterval = 45; /* Number of garbage generating threads */ static int numGarbage = 1; /* Time between garbage allocations in milliseconds */ static int garbageInterval = 100; // The MAX quantity of critical operations static int jniStringAllocSize = 50000; private static StressOptions stressOptions; public static void main(String[] argv) { try { int i = 0; int nJNISync = 10; jnistress004 dm = null; boolean errArg = false; stressOptions = new StressOptions(argv); /* Process arguments */ while (!errArg && i < argv.length) { /* Number of iterations. Ignored if <= 0. */ if (i < argv.length && argv[i].equals("-numIteration")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { numIteration = Long.parseLong(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].equals("-numTHREADer")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { numJNIter = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } if (numJNIter <= 0) errArg = true; } } else if (i < argv.length && argv[i].equals("-threadInterval")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { jniInterval = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].equals("-numInterrupter")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { numInterrupter = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].equals("-interruptInterval")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { interruptInterval = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].equals("-numGarbage")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { numGarbage = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].equals("-garbageInterval")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { garbageInterval = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].equals("-jniStringAllocSize")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { try { jniStringAllocSize = Integer.parseInt(argv[i++]); } catch (NumberFormatException e) { errArg = true; } } } else if (i < argv.length && argv[i].startsWith("-stress")) { ++i; if (i < argv.length && Character.isDigit(argv[i].charAt(0))) { ++i; } } else System.out.println("Argument #" + i++ + " is incorrect"); } numIteration *= stressOptions.getIterationsFactor(); numJNIter *= stressOptions.getThreadsFactor(); numInterrupter *= stressOptions.getThreadsFactor(); numGarbage *= stressOptions.getThreadsFactor(); timeOut = stressOptions.getTime() * 1000; sync = new Synchronizer[10]; for (i = 0; i < nJNISync; i++) sync[i] = new Synchronizer(); dm = new jnistress004(numIteration, numJNIter, jniInterval, numInterrupter, interruptInterval, numGarbage, garbageInterval); dm.start(); try { dm.join(timeOut); } catch (InterruptedException e) { System.out.println("TESTER THREAD WAS INTERRUPTED"); System.exit(Consts.TEST_FAILED); } if (DEBUG) System.out.println("jnistress004::main(): halt!"); if (dm.isAlive()) { System.out.println("TIME LIMIT EXCEEDED"); dm.halt(); if (DEBUG) System.out.println("jnistress004::main(): join!"); try { dm.join(10000L); } catch (InterruptedException e) { System.out.println("TESTER THREAD WAS INTERRUPTED"); System.exit(Consts.TEST_FAILED); } } else { System.out.println("TESTER THREAD FINISHED"); } if (DEBUG) System.out.println("jnistress004::main(): zzzz..."); if (!JNIter004.passed()) System.exit(Consts.TEST_FAILED); } catch (Throwable e) { Debug.Fail(e); } } jnistress004( long iters, int nJNI, int jniInterval, int nInter, int iruptInterval, int nGarb, int garbInterval ) { int i = 0; nCycles = iters; /* Should have at least one of nCycles>0 */ if (nCycles <= 0) nCycles = Long.MAX_VALUE; jniter = new JNIter004[nJNI]; interval = jniInterval; irupt = new Interrupter[nInter]; garb = new GarbageGenerator[nGarb]; for (i = 0; i < nJNI; i++) jniter[i] = new JNIter004(sync); for (i = 0; i < nInter; i++) { irupt[i] = new Interrupter(jniter, sync); irupt[i].setInterval(iruptInterval); } for (i = 0; i < nGarb; i++) { garb[i] = new GarbageGenerator(); garb[i].setInterval(garbInterval); } } public void run() { try { int i = 0; long iCycle = 0L; JNIter004.clearCount(); JNIter004.clearInterruptCount(); for (i = 0; i < jniter.length; i++) jniter[i].start(); while (JNIter004.getCount() < jniter.length) { try { sleep(100); } catch (InterruptedException e) { } } JNIter004.clearCount(); // JNIter004.clearInterruptCount(); synchronized (sync[0]) { sync[0].notifyAll(); } for (i = 0; i < garb.length; i++) garb[i].start(); for (i = 0; i < irupt.length; i++) irupt[i].start(); if (DEBUG) System.out.println("Cycles=" + nCycles); for (iCycle = 0; iCycle < nCycles && !done && JNIter004.passed(); iCycle++) { System.out.print("Cycle: " + iCycle); try { sleep(interval); } catch (InterruptedException e) { } synchronized (sync[1]) { System.out.println(" Interrupt count=" + JNIter004.getInterruptCount()); } JNIter004.clearCount(); synchronized (sync[0]) { sync[0].notifyAll(); } int n = 0; for (i = 0; i < jniter.length; i++) if (jniter[i].finished()) n++; if (n == jniter.length) break; } if (JNIter004.passed()) System.out.println("JNI TEST PASSED"); else System.out.println("JNI TEST FAILED"); for (i = 0; i < irupt.length; i++) irupt[i].halt(); for (i = 0; i < garb.length; i++) garb[i].halt(); for (i = 0; i < jniter.length; i++) jniter[i].halt(); /* Flush any waiters */ if (DEBUG) System.out.println("jnistress004::run(): before sync[0]"); synchronized (sync[0]) { sync[0].notifyAll(); } if (DEBUG) System.out.println("jnistress004::run(): after sync[0]"); for (i = 0; i < irupt.length; i++) { try { irupt[i].join(); } catch (InterruptedException e) { } } if (DEBUG) System.out.println("jnistress004::run(): X"); for (i = 0; i < garb.length; i++) { try { garb[i].join(); } catch (InterruptedException e) { } } if (DEBUG) System.out.println("jnistress004::run(): Y"); synchronized (sync[0]) { sync[0].notifyAll(); } for (i = 0; i < jniter.length; i++) { try { if (jniter[i].isAlive()) { jniter[i].join(); } } catch (InterruptedException e) { } } if (DEBUG) System.out.println("jnistress004::run(): Z"); } catch (Throwable e) { Debug.Fail(e); } } public void halt() { done = true; } public boolean finished() { return done; } long nCycles = 0; JNIter004[] jniter; static Synchronizer[] sync; private int interval = 100; Interrupter[] irupt; GarbageGenerator[] garb; private boolean done = false; final private static boolean DEBUG = false; } class JNIter004 extends Thread { // The native methods for testing JNI critical calls public native char[] CheckSum(String str); public native boolean CheckCompare(String name, char[] sum, int upper); static { System.loadLibrary("jnistress004"); } Runtime myRT = Runtime.getRuntime(); static int Count = 0; public JNIter004(Synchronizer[] aSync) { sync = aSync; } public void run() { try { char[] Sum; int iter = 0; /* Synchronize start of work */ incCount(); synchronized (sync[0]) { try { sync[0].wait(); } catch (InterruptedException e) { } } while (!done && pass) { try { /* Synchronized the JNI stressing */ synchronized (sync[2]) { incCount(); } synchronized (sync[0]) { try { sync[0].wait(); } catch (InterruptedException e) { synchronized (sync[1]) { JNIter004.incInterruptCount(); } } } synchronized (sync[0]) { try { if (Count++ < jnistress004.jniStringAllocSize) { System.out.println("JAVA: comparing " + (getName()) + " with " + CheckSum(getName())); if (!CheckCompare(getName(), CheckSum(getName()), jnistress004.jniStringAllocSize)) pass = true; } } catch (OutOfMemoryError e) { System.out.println("Error in Java code" + e); } } if (DEBUG) System.out.println(getName() + "\t\t" + myRT.freeMemory()); synchronized (this) { try { wait(1L); } catch (InterruptedException e) { throw new InterruptedException(); } } } catch (Exception e) { synchronized (sync[1]) { JNIter004.incInterruptCount(); } } iter++; iter = iter % CASECOUNT; } if (DEBUG) System.out.println("JNITer::run(): done=" + done); done = true; if (DEBUG) System.out.println("JNITer::run(): pass=" + JNIter004.pass); if (DEBUG) System.out.println("JNIter004::run(): done"); } catch (Throwable e) { Debug.Fail(e); } } private synchronized static void incCount() { count++; } public static int getCount() { return count; } public synchronized static void clearCount() { count = 0; } private synchronized static void incInterruptCount() { interruptCount++; } public static int getInterruptCount() { return interruptCount; } public synchronized static void clearInterruptCount() { interruptCount = 0; } public static void halt() { done = true; } public boolean finished() { return done; } public static boolean passed() { return pass; } public static void setpass(boolean value) { pass = value; } Synchronizer[] sync; private static int count = 0; private static int interruptCount = 0; private static boolean done = false; private static boolean pass = true; final private static int CASECOUNT = 2; final private static boolean DEBUG = false; }