/* * Copyright (c) 2011, 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. */ /* * @test * @key stress randomness * * @summary converted from VM Testbase gc/gctests/StringInternSyncWithGC. * VM Testbase keywords: [gc, stress, stressopt, feature_perm_removal_jdk7, nonconcurrent] * VM Testbase readme: * The test verifies that String.intern is correctly synchronized with GC. * Test interns and drop the same strings in different threads and provokes GC. * Additionally test creates weak/soft references to interned strings. * Test fails if any string object is inaccessible. * * @library /vmTestbase * /test/lib * @run main/othervm * -Xlog:gc:gc.log * gc.gctests.StringInternSyncWithGC.StringInternSyncWithGC * -ms low * -memUsage 3 * -appTimeout 30 * -capacityVerPart 2 */ package gc.gctests.StringInternSyncWithGC; import java.util.ArrayList; import java.util.List; import nsk.share.gc.*; import nsk.share.gc.gp.MemoryStrategy; import nsk.share.gc.gp.MemoryStrategyAware; import nsk.share.gc.gp.string.RandomStringProducer; import nsk.share.test.ExecutionController; public class StringInternSyncWithGC extends ThreadedGCTest implements MemoryStrategyAware { // Maximum size of one string // Depends from all size and memory strategy private int maxStringSize; private MemoryStrategy memoryStrategy; private final int memUsageFactor; private final long endTimeCapacityVer; // The list of strings which are interned during iteration private final List stringsToIntern = new ArrayList(); private final RandomStringProducer gp = new RandomStringProducer(); public StringInternSyncWithGC(int memUsage, long endTimeCapVer) { memUsageFactor = memUsage; endTimeCapacityVer = endTimeCapVer; } @Override public void setMemoryStrategy(MemoryStrategy memoryStrategy) { this.memoryStrategy = memoryStrategy; } /** * Verify that we could use certain amount of memory. */ private boolean verifyInternedStringCapacity(long initialSize) { long currentSize = 0; final int STEP = 1000; int iter = 0; char[] template = new char[(int) (initialSize / STEP)]; List tmpList = new ArrayList<>(STEP); try { while (currentSize <= initialSize) { if (endTimeCapacityVer < System.currentTimeMillis()) { log.debug("Too long to verify interned string capacity"); log.debug("Silently pass."); return false; } template[iter]++; if (++iter == template.length) { iter = 0; } String str = new String(template); tmpList.add(str.intern()); currentSize += str.length() * 2; //each char costs 2 bytes } } catch (OutOfMemoryError oome) { log.debug("It is not possible to allocate " + initialSize + " size of interned string."); log.debug("Silently pass."); return false; } return true; } @Override public void run() { long size = runParams.getTestMemory() / memUsageFactor; if (!verifyInternedStringCapacity(size)) { return; } // Approximate size occupied by all interned strings long sizeOfAllInternedStrings = size / 2; maxStringSize = (int) (sizeOfAllInternedStrings / memoryStrategy.getSize(sizeOfAllInternedStrings, Memory.getObjectExtraSize())); // Each thread keeps reference to each created string. long extraConsumption = runParams.getNumberOfThreads() * Memory.getReferenceSize(); log.debug("The overall size of interned strings : " + sizeOfAllInternedStrings / (1024 * 1024) + "M"); log.debug("The count of interned strings : " + sizeOfAllInternedStrings / (maxStringSize + extraConsumption)); for (long currentSize = 0; currentSize <= sizeOfAllInternedStrings; currentSize += maxStringSize + extraConsumption) { stringsToIntern.add(gp.create(maxStringSize)); } super.run(); } @Override protected Runnable createRunnable(int threadId) { return new StringGenerator(threadId, this); } public static void main(String[] args) { int appTimeout = -1; int memUsageFactor = 1; // Part of time that function verifyInternedStringCapacity can take. Time = Application_Timeout / capacityVerTimePart double capacityVerPart = 2; for (int i = 0; i < args.length; ++i) { switch (args[i]) { case "-memUsage": memUsageFactor = Integer.parseInt(args[i + 1]); break; case "-capacityVerPart": capacityVerPart = Double.parseDouble(args[i + 1]); break; case "-appTimeout": appTimeout = Integer.parseInt(args[i + 1]); break; default: } } if (appTimeout == -1) { throw new IllegalArgumentException("Specify -appTimeout."); } long endTimeCapacityVer = System.currentTimeMillis() + (long) (appTimeout / capacityVerPart * 60000); GC.runTest(new StringInternSyncWithGC(memUsageFactor, endTimeCapacityVer), args); } protected List getStringsToIntern() { return stringsToIntern; } protected int getNumberOfThreads() { return runParams.getNumberOfThreads(); } protected RandomStringProducer getGarbageProducer() { return gp; } protected int getMaxStringSize() { return maxStringSize; } protected ExecutionController getExecController() { return getExecutionController(); } }