mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-27 14:54:52 +02:00
8218227: StringBuilder/StringBuffer constructor throws confusing NegativeArraySizeException
Reviewed-by: rriggs
This commit is contained in:
parent
621bf58687
commit
076d2267b6
5 changed files with 136 additions and 10 deletions
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2003, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
@ -91,6 +91,19 @@ abstract class AbstractStringBuilder implements Appendable, CharSequence {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an AbstractStringBuilder with the specified coder and with
|
||||||
|
* the initial capacity equal to the smaller of (capacity + addition)
|
||||||
|
* and Integer.MAX_VALUE.
|
||||||
|
*/
|
||||||
|
AbstractStringBuilder(byte coder, int capacity, int addition) {
|
||||||
|
this.coder = coder;
|
||||||
|
capacity = (capacity < Integer.MAX_VALUE - addition)
|
||||||
|
? capacity + addition : Integer.MAX_VALUE;
|
||||||
|
value = (coder == LATIN1)
|
||||||
|
? new byte[capacity] : StringUTF16.newBytesFor(capacity);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compares the objects of two AbstractStringBuilder implementations lexicographically.
|
* Compares the objects of two AbstractStringBuilder implementations lexicographically.
|
||||||
*
|
*
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 1994, 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1994, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
@ -148,7 +148,7 @@ import jdk.internal.HotSpotIntrinsicCandidate;
|
||||||
*/
|
*/
|
||||||
@HotSpotIntrinsicCandidate
|
@HotSpotIntrinsicCandidate
|
||||||
public StringBuffer(String str) {
|
public StringBuffer(String str) {
|
||||||
super(str.length() + 16);
|
super(str.coder(), str.length(), 16);
|
||||||
append(str);
|
append(str);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,7 +166,7 @@ import jdk.internal.HotSpotIntrinsicCandidate;
|
||||||
* @since 1.5
|
* @since 1.5
|
||||||
*/
|
*/
|
||||||
public StringBuffer(CharSequence seq) {
|
public StringBuffer(CharSequence seq) {
|
||||||
this(seq.length() + 16);
|
super(String.LATIN1, seq.length(), 16);
|
||||||
append(seq);
|
append(seq);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2003, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
@ -121,7 +121,7 @@ public final class StringBuilder
|
||||||
*/
|
*/
|
||||||
@HotSpotIntrinsicCandidate
|
@HotSpotIntrinsicCandidate
|
||||||
public StringBuilder(String str) {
|
public StringBuilder(String str) {
|
||||||
super(str.length() + 16);
|
super(str.coder(), str.length(), 16);
|
||||||
append(str);
|
append(str);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -134,7 +134,7 @@ public final class StringBuilder
|
||||||
* @param seq the sequence to copy.
|
* @param seq the sequence to copy.
|
||||||
*/
|
*/
|
||||||
public StringBuilder(CharSequence seq) {
|
public StringBuilder(CharSequence seq) {
|
||||||
this(seq.length() + 16);
|
super(String.LATIN1, seq.length(), 16);
|
||||||
append(seq);
|
append(seq);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
77
test/jdk/java/lang/StringBuffer/HugeCapacity.java
Normal file
77
test/jdk/java/lang/StringBuffer/HugeCapacity.java
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2019, 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
|
||||||
|
* @bug 8218227
|
||||||
|
* @summary StringBuilder/StringBuffer constructor throws confusing
|
||||||
|
* NegativeArraySizeException
|
||||||
|
* @requires os.maxMemory >= 6G
|
||||||
|
* @run main/othervm -Xms5G -Xmx5G HugeCapacity
|
||||||
|
* @ignore This test has huge memory requirements
|
||||||
|
*/
|
||||||
|
|
||||||
|
public class HugeCapacity {
|
||||||
|
private static int failures = 0;
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
testHugeInitialString();
|
||||||
|
testHugeInitialCharSequence();
|
||||||
|
if (failures > 0) {
|
||||||
|
throw new RuntimeException(failures + " tests failed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void testHugeInitialString() {
|
||||||
|
try {
|
||||||
|
String str = "Z".repeat(Integer.MAX_VALUE - 8);
|
||||||
|
StringBuffer sb = new StringBuffer(str);
|
||||||
|
} catch (OutOfMemoryError ignore) {
|
||||||
|
} catch (Throwable unexpected) {
|
||||||
|
unexpected.printStackTrace();
|
||||||
|
failures++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void testHugeInitialCharSequence() {
|
||||||
|
try {
|
||||||
|
CharSequence seq = new MyHugeCharSeq();
|
||||||
|
StringBuffer sb = new StringBuffer(seq);
|
||||||
|
} catch (OutOfMemoryError ignore) {
|
||||||
|
} catch (Throwable unexpected) {
|
||||||
|
unexpected.printStackTrace();
|
||||||
|
failures++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MyHugeCharSeq implements CharSequence {
|
||||||
|
public char charAt(int i) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
public int length() { return Integer.MAX_VALUE; }
|
||||||
|
public CharSequence subSequence(int st, int e) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
public String toString() { return ""; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2016, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
@ -23,10 +23,11 @@
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @test
|
* @test
|
||||||
* @bug 8149330
|
* @bug 8149330 8218227
|
||||||
* @summary Capacity should not get close to Integer.MAX_VALUE unless
|
* @summary Capacity should not get close to Integer.MAX_VALUE unless
|
||||||
* necessary
|
* necessary
|
||||||
* @run main/othervm -Xmx5G HugeCapacity
|
* @requires os.maxMemory >= 6G
|
||||||
|
* @run main/othervm -Xms5G -Xmx5G HugeCapacity
|
||||||
* @ignore This test has huge memory requirements
|
* @ignore This test has huge memory requirements
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@ -36,6 +37,8 @@ public class HugeCapacity {
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
testLatin1();
|
testLatin1();
|
||||||
testUtf16();
|
testUtf16();
|
||||||
|
testHugeInitialString();
|
||||||
|
testHugeInitialCharSequence();
|
||||||
if (failures > 0) {
|
if (failures > 0) {
|
||||||
throw new RuntimeException(failures + " tests failed");
|
throw new RuntimeException(failures + " tests failed");
|
||||||
}
|
}
|
||||||
|
@ -63,4 +66,37 @@ public class HugeCapacity {
|
||||||
failures++;
|
failures++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void testHugeInitialString() {
|
||||||
|
try {
|
||||||
|
String str = "Z".repeat(Integer.MAX_VALUE - 8);
|
||||||
|
StringBuilder sb = new StringBuilder(str);
|
||||||
|
} catch (OutOfMemoryError ignore) {
|
||||||
|
} catch (Throwable unexpected) {
|
||||||
|
unexpected.printStackTrace();
|
||||||
|
failures++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void testHugeInitialCharSequence() {
|
||||||
|
try {
|
||||||
|
CharSequence seq = new MyHugeCharSeq();
|
||||||
|
StringBuilder sb = new StringBuilder(seq);
|
||||||
|
} catch (OutOfMemoryError ignore) {
|
||||||
|
} catch (Throwable unexpected) {
|
||||||
|
unexpected.printStackTrace();
|
||||||
|
failures++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MyHugeCharSeq implements CharSequence {
|
||||||
|
public char charAt(int i) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
public int length() { return Integer.MAX_VALUE; }
|
||||||
|
public CharSequence subSequence(int st, int e) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
public String toString() { return ""; }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue