2023-12-12 20:55:17 +00:00
|
|
|
/*
|
2024-04-08 17:58:21 +00:00
|
|
|
* Copyright (c) 2023, 2024, Oracle and/or its affiliates. All rights reserved.
|
2023-12-12 20:55:17 +00: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import org.junit.jupiter.api.Test;
|
|
|
|
import static org.junit.jupiter.api.Assertions.*;
|
|
|
|
|
|
|
|
import java.nio.charset.StandardCharsets;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @bug 8077559 8321180
|
|
|
|
* @summary Tests Compact String for maximum size strings
|
2023-12-15 16:13:36 +00:00
|
|
|
* @requires os.maxMemory >= 8g & vm.bits == 64
|
|
|
|
* @requires vm.flagless
|
|
|
|
* @run junit/othervm -XX:+CompactStrings -Xmx8g MaxSizeUTF16String
|
|
|
|
* @run junit/othervm -XX:-CompactStrings -Xmx8g MaxSizeUTF16String
|
|
|
|
* @run junit/othervm -Xcomp -Xmx8g MaxSizeUTF16String
|
2023-12-12 20:55:17 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
public class MaxSizeUTF16String {
|
|
|
|
|
|
|
|
private final static int MAX_UTF16_STRING_LENGTH = Integer.MAX_VALUE / 2;
|
|
|
|
|
|
|
|
private final static String EXPECTED_OOME_MESSAGE = "UTF16 String size is";
|
2023-12-15 16:13:36 +00:00
|
|
|
private final static String EXPECTED_VM_LIMIT_MESSAGE = "Requested array size exceeds VM limit";
|
|
|
|
private final static String UNEXPECTED_JAVA_HEAP_SPACE = "Java heap space";
|
2023-12-12 20:55:17 +00:00
|
|
|
|
|
|
|
// Create a large UTF-8 byte array with a single non-latin1 character
|
|
|
|
private static byte[] generateUTF8Data(int byteSize) {
|
|
|
|
byte[] nonAscii = "\u0100".getBytes(StandardCharsets.UTF_8);
|
|
|
|
byte[] arr = new byte[byteSize];
|
|
|
|
System.arraycopy(nonAscii, 0, arr, 0, nonAscii.length); // non-latin1 at start
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a large char array with a single non-latin1 character
|
|
|
|
private static char[] generateCharData(int size) {
|
|
|
|
char[] nonAscii = "\u0100".toCharArray();
|
|
|
|
char[] arr = new char[size];
|
|
|
|
System.arraycopy(nonAscii, 0, arr, 0, nonAscii.length); // non-latin1 at start
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testMaxUTF8() {
|
|
|
|
// Overly large UTF-8 data with 1 non-latin1 char
|
|
|
|
final byte[] large_utf8_bytes = generateUTF8Data(MAX_UTF16_STRING_LENGTH + 1);
|
|
|
|
int[] sizes = new int[] {
|
|
|
|
MAX_UTF16_STRING_LENGTH + 1,
|
|
|
|
MAX_UTF16_STRING_LENGTH,
|
|
|
|
MAX_UTF16_STRING_LENGTH - 1};
|
|
|
|
for (int size : sizes) {
|
|
|
|
System.err.println("Checking max UTF16 string len: " + size);
|
|
|
|
try {
|
|
|
|
// Use only part of the UTF-8 byte array
|
|
|
|
new String(large_utf8_bytes, 0, size, StandardCharsets.UTF_8);
|
|
|
|
if (size >= MAX_UTF16_STRING_LENGTH) {
|
|
|
|
fail("Expected OutOfMemoryError with message prefix: " + EXPECTED_OOME_MESSAGE);
|
|
|
|
}
|
|
|
|
} catch (OutOfMemoryError ex) {
|
2023-12-15 16:13:36 +00:00
|
|
|
if (ex.getMessage().equals(UNEXPECTED_JAVA_HEAP_SPACE)) {
|
|
|
|
// Insufficient heap size
|
|
|
|
throw ex;
|
|
|
|
}
|
|
|
|
if (!ex.getMessage().startsWith(EXPECTED_OOME_MESSAGE) &&
|
|
|
|
!ex.getMessage().startsWith(EXPECTED_VM_LIMIT_MESSAGE)) {
|
2023-12-12 20:55:17 +00:00
|
|
|
fail("Failed: Not the OutOfMemoryError expected", ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testMaxCharArray() {
|
|
|
|
// Overly large UTF-8 data with 1 non-latin1 char
|
|
|
|
final char[] large_char_array = generateCharData(MAX_UTF16_STRING_LENGTH + 1);
|
|
|
|
int[] sizes = new int[]{
|
|
|
|
MAX_UTF16_STRING_LENGTH + 1,
|
|
|
|
MAX_UTF16_STRING_LENGTH,
|
|
|
|
MAX_UTF16_STRING_LENGTH - 1};
|
|
|
|
for (int size : sizes) {
|
|
|
|
System.err.println("Checking max UTF16 string len: " + size);
|
|
|
|
try {
|
|
|
|
// Large char array with 1 non-latin1 char
|
|
|
|
new String(large_char_array, 0, size);
|
|
|
|
if (size >= MAX_UTF16_STRING_LENGTH) {
|
|
|
|
fail("Expected OutOfMemoryError with message prefix: " + EXPECTED_OOME_MESSAGE);
|
|
|
|
}
|
|
|
|
} catch (OutOfMemoryError ex) {
|
2023-12-15 16:13:36 +00:00
|
|
|
if (ex.getMessage().equals(UNEXPECTED_JAVA_HEAP_SPACE)) {
|
|
|
|
// Insufficient heap size
|
|
|
|
throw ex;
|
|
|
|
}
|
|
|
|
if (!ex.getMessage().startsWith(EXPECTED_OOME_MESSAGE) &&
|
|
|
|
!ex.getMessage().startsWith(EXPECTED_VM_LIMIT_MESSAGE)) {
|
2023-12-12 20:55:17 +00:00
|
|
|
throw new RuntimeException("Wrong exception message: " + ex.getMessage(), ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-04-08 17:58:21 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Test that UTF-8 of too large strings throws OOME, (not NegativeArraySizeException).
|
|
|
|
*/
|
|
|
|
@Test
|
|
|
|
public void testMaxUTF8_UTF16Encode() {
|
|
|
|
String s = "\uFFFF";
|
|
|
|
final byte[] bytes1 = s.getBytes(StandardCharsets.UTF_8);
|
|
|
|
assertEquals(3, bytes1.length, "UTF_8 encoded length of 0xffff");
|
|
|
|
|
|
|
|
int min = Integer.MAX_VALUE / bytes1.length - 1;
|
|
|
|
int max = min + 3;
|
|
|
|
|
|
|
|
// String of size min can be UTF_8 encoded.
|
|
|
|
System.out.println("testing size: " + min);
|
|
|
|
String s1 = s.repeat(min);
|
|
|
|
byte[] bytes = s1.getBytes(StandardCharsets.UTF_8);
|
|
|
|
int remaining = Integer.MAX_VALUE - bytes.length;
|
|
|
|
assertTrue(remaining >= bytes1.length, "remainder too large: " + remaining);
|
|
|
|
|
|
|
|
// Strings of size min+1...min+2, throw OOME
|
|
|
|
// The resulting byte array would exceed implementation limits
|
|
|
|
for (int count = min + 1; count < max; count++) {
|
|
|
|
System.out.println("testing size: " + count);
|
|
|
|
final String s2 = s.repeat(count);
|
|
|
|
OutOfMemoryError ex = assertThrows(OutOfMemoryError.class, () -> s2.getBytes(StandardCharsets.UTF_8));
|
|
|
|
ex.printStackTrace();
|
|
|
|
};
|
|
|
|
}
|
2023-12-12 20:55:17 +00:00
|
|
|
}
|