Sometimes constraints are questionable
    Jim Laskey 
    james.laskey at oracle.com
       
    Mon Jun  1 12:21:40 UTC 2020
    
    
  
My mistake it's ArraysSupport.newLength that's failing here,
String::replace
                pos = Arrays.copyOf(pos, ArraysSupport.newLength(p, 1, p >> 1));
newLength should be causing the exception but it in turn is passing the OOM size on to Arrays.copyOf
The logic in ArraysSupport::newLength is similar to that in the first part of this discussion:
    public static int newLength(int oldLength, int minGrowth, int prefGrowth) {
        // assert oldLength >= 0
        // assert minGrowth > 0
        int newLength = Math.max(minGrowth, prefGrowth) + oldLength;
        if (newLength - MAX_ARRAY_LENGTH <= 0) {
            return newLength;
        }
        return hugeLength(oldLength, minGrowth);
    }
    private static int hugeLength(int oldLength, int minGrowth) {
        int minLength = oldLength + minGrowth;
        if (minLength < 0) { // overflow
            throw new OutOfMemoryError("Required array length too large");
        }
        if (minLength <= MAX_ARRAY_LENGTH) {
            return MAX_ARRAY_LENGTH;
        }
        return Integer.MAX_VALUE;
    }
That is, pass Integer.MAX_VALUE on to the allocator and let it fail.
> On Jun 1, 2020, at 8:56 AM, Jim Laskey <james.laskey at oracle.com> wrote:
> 
> In the same light the test in String::replace
> 
>        int resultLen;
>        try {
>            resultLen = Math.addExact(valLen,
>                    Math.multiplyExact(++p, replLen - targLen));
>        } catch (ArithmeticException ignored) {
>            throw new
>            OutOfMemoryError("String resulting from replace exceeds maximum string size");
>        }
> 
> Is useless in this situation (from the bug report):
> 
>        int missingToIntMax = 10;
>        String tooLarge = "a".repeat(Integer.MAX_VALUE - missingToIntMax);
>        tooLarge.replace("a", "a".repeat(missingToIntMax + 2));
> 
> The replace "test" does not fail and the exception comes from further down the stack:
> 
> Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
> 	at java.base/java.util.Arrays.copyOf(Arrays.java:3584)
> 	at java.base/java.lang.StringLatin1.replace(StringLatin1.java:349)
> 	at java.base/java.lang.String.replace(String.java:2173)
> 	at JI9062163.main(JI9062163.java:6)
> 
> 
> 
> 
>> On May 31, 2020, at 10:15 PM, Martin Buchholz <martinrb at google.com> wrote:
>> 
>> I'm still not deeply reading the code in java.nio but I see in
>> AbstractStringBuilder
>> 
>>    * Will not return a capacity greater than
>>    * {@code (MAX_ARRAY_SIZE >> coder)} unless the given minimum capacity
>>    * is greater than that.
>> 
>> My intent was that a garden variety grow method would first grow to
>> MAX_ARRAY_SIZE, then to Integer.MAX_VALUE.
>> 
>> (this doesn't always work, e.g. hash tables want to have a
>> power-of-two sized backing array)
>> 
>> (these methods have evolved recently without much help from me)
>> 
>> (Library code should know as little about VM implementation
>> idiosyncrasies as possible)
>> 
>> On Sun, May 31, 2020 at 5:56 PM David Holmes <david.holmes at oracle.com> wrote:
>>> 
>>> On 1/06/2020 9:52 am, Martin Buchholz wrote:
>>>> On Sun, May 31, 2020 at 4:35 PM David Holmes <david.holmes at oracle.com> wrote:
>>>> 
>>>>> Not sure how we could have minCapacity < 0 at this point. It should have
>>>>> been checked before the call to grow, and grow will not make it negative.
>>>> 
>>>> At least some of the grow methods were designed so that callers did
>>>> not have to do the checking, and grow would interpret negative values
>>>> as int overflow
>>>> 
>>>>>> It just seems that it's pushing the inevitable off to Arrays.copyOf.  Shouldn't it be:
>>>>>> 
>>>>>>     private static int hugeCapacity(int minCapacity) {
>>>>>>         if (minCapacity < 0 || minCapacity > MAX_ARRAY_SIZE) {
>>>>>>             throw
>>>>>>                 new OutOfMemoryError("ByteArrayChannel exceeds maximum size: " +
>>>>>>                                       MAX_ARRAY_SIZE);
>>>>>>         }
>>>>>> 
>>>>>>         return MAX_ARRAY_SIZE;
>>>>>>     }
>>>>> 
>>>>> That seems more appropriate to me - modulo the question mark over
>>>>> minCapacity being negative.
>>>> 
>>>> Again, the original design was to allow a capacity with MAX_ARRAY_SIZE
>>>> < capacity <= Integer.MAX_VALUE if and when the VM also allowed it.
>>> 
>>> The only way to know that is to try and create the array, catch the OOME
>>> and then adjust the value used. None of the code does that so I can't
>>> see how your claim here can be correct. My understanding, having been on
>>> the VM side, was that Hotspot's internal limits were reflected on the
>>> Java side so that the Java code could avoid the VM exception.
>>> 
>>> David
> 
    
    
More information about the core-libs-dev
mailing list