8218227: StringBuilder/StringBuffer constructor throws confusing NegativeArraySizeException

Reviewed-by: rriggs
This commit is contained in:
Ivan Gerasimov 2019-02-05 17:05:40 -08:00
parent 621bf58687
commit 076d2267b6
5 changed files with 136 additions and 10 deletions

View file

@ -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.
* *

View file

@ -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);
} }

View file

@ -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);
} }

View 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 ""; }
}
}

View file

@ -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 ""; }
}
} }