2018-05-17 14:52:47 -07:00
|
|
|
/*
|
2020-06-02 13:04:21 -07:00
|
|
|
* Copyright (c) 2011, 2020, Oracle and/or its affiliates. All rights reserved.
|
2018-05-17 14:52:47 -07:00
|
|
|
* 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
|
2020-06-08 21:27:22 -07:00
|
|
|
* @key stress randomness
|
2018-05-17 14:52:47 -07:00
|
|
|
*
|
|
|
|
* @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();
|
|
|
|
}
|
|
|
|
}
|