2020-07-10 11:49:02 -07:00

185 lines
6.7 KiB
Java

/*
* 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<String> 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<String> 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<String> getStringsToIntern() {
return stringsToIntern;
}
protected int getNumberOfThreads() {
return runParams.getNumberOfThreads();
}
protected RandomStringProducer getGarbageProducer() {
return gp;
}
protected int getMaxStringSize() {
return maxStringSize;
}
protected ExecutionController getExecController() {
return getExecutionController();
}
}