RFR(M): 8248188: [PATCH] Add HotSpotIntrinsicCandidate and API for Base64 decoding
Roger Riggs
Roger.Riggs at oracle.com
Wed Aug 19 18:20:13 UTC 2020
Hi Corey,
For changes obviously performance motivated, it is conventional to run a
JMH perf test to demonstate
the improvement and prove it is worthwhile to add code complexity.
I don't see any existing Base64 JMH tests but they would be in the repo
below or near:
test/micro/org/openjdk/bench/java/util/
Please contribute a JMH test and results to show the difference.
Regards, Roger
On 8/19/20 2:10 PM, Corey Ashford wrote:
> Michihiro Horie posted up a new iteration of this webrev for me. This
> time the webrev includes a complete implementation of the intrinsic
> for Power9 and Power10.
>
> You can find it here:
> http://cr.openjdk.java.net/~mhorie/8248188/webrev.02/
>
> Changes in webrev.02 vs. webrev.01:
>
> * The method header for the intrinsic in the Base64 code has been
> rewritten using the Javadoc style. The clarity of the comments has
> been improved and some verbosity has been removed. There are no
> additional functional changes to Base64.java.
>
> * The code needed to martial and check the intrinsic parameters has
> been added, using the base64 encodeBlock intrinsic as a guideline.
>
> * A complete intrinsic implementation for Power9 and Power10 is
> included.
>
> * Adds some Power9 and Power10 assembler instructions needed by the
> intrinsic which hadn't been defined before.
>
> The intrinsic implementation in this patch accelerates the decoding of
> large blocks of base64 data by a factor of about 3.5X on Power9.
>
> I'm attaching two Java test cases I am using for testing and
> benchmarking. The TestBase64_VB encodes and decodes randomly-sized
> buffers of random data and checks that original data matches the
> encoded-then-decoded data. TestBase64Errors encodes a 48K block of
> random bytes, then corrupts each byte of the encoded data, one at a
> time, checking to see if the decoder catches the illegal byte.
>
> Any comments/suggestions would be appreciated.
>
> Thanks,
>
> - Corey
>
> On 7/27/20 6:49 PM, Corey Ashford wrote:
>> Michihiro Horie uploaded a new revision of the Base64 decodeBlock
>> intrinsic API for me:
>>
>> http://cr.openjdk.java.net/~mhorie/8248188/webrev.01/
>>
>> It has the following changes with respect to the original one posted:
>>
>> * In the event of encountering a non-base64 character, instead of
>> having a separate error code of -1, the intrinsic can now just return
>> either 0, or the number of data bytes produced up to the point where
>> the illegal base64 character was encountered. This reduces the number
>> of special cases, and also provides a way to speed up the process of
>> finding the bad character by the slower, pure-Java algorithm.
>>
>> * The isMIME boolean is removed from the API for two reasons:
>> - The current API is not sufficient to handle the isMIME case,
>> because there isn't a strict relationship between the number of input
>> bytes and the number of output bytes, because there can be an
>> arbitrary number of non-base64 characters in the source.
>> - If an intrinsic only implements the (isMIME == false) case as
>> ours does, it will always return 0 bytes processed, which will
>> slightly slow down the normal path of processing an (isMIME == true)
>> instantiation.
>> - We considered adding a separate hotspot candidate for the
>> (isMIME == true) case, but since we don't have an intrinsic
>> implementation to test that, we decided to leave it as a future
>> optimization.
>>
>> Comments and suggestions are welcome. Thanks for your consideration.
>>
>> - Corey
>>
>> On 6/23/20 6:23 PM, Michihiro Horie wrote:
>>> Hi Corey,
>>>
>>> Following is the issue I created.
>>> https://bugs.openjdk.java.net/browse/JDK-8248188
>>>
>>> I will upload a webrev when you're ready as we talked in private.
>>>
>>> Best regards,
>>> Michihiro
>>>
>>> Inactive hide details for "Corey Ashford" ---2020/06/24
>>> 09:40:10---Currently in java.util.Base64, there is a
>>> HotSpotIntrinsicCa"Corey Ashford" ---2020/06/24 09:40:10---Currently
>>> in java.util.Base64, there is a HotSpotIntrinsicCandidate and API
>>> for encodeBlock, but no
>>>
>>> From: "Corey Ashford" <cjashfor at linux.ibm.com>
>>> To: "hotspot-compiler-dev at openjdk.java.net"
>>> <hotspot-compiler-dev at openjdk.java.net>,
>>> "ppc-aix-port-dev at openjdk.java.net" <ppc-aix-port-dev at openjdk.java.net>
>>> Cc: Michihiro Horie/Japan/IBM at IBMJP, Kazunori Ogata/Japan/IBM at IBMJP,
>>> joserz at br.ibm.com
>>> Date: 2020/06/24 09:40
>>> Subject: RFR(S): [PATCH] Add HotSpotIntrinsicCandidate and API for
>>> Base64 decoding
>>>
>>> ------------------------------------------------------------------------
>>>
>>>
>>>
>>>
>>> Currently in java.util.Base64, there is a HotSpotIntrinsicCandidate and
>>> API for encodeBlock, but none for decoding. This means that only
>>> encoding gets acceleration from the underlying CPU's vector hardware.
>>>
>>> I'd like to propose adding a new intrinsic for decodeBlock. The
>>> considerations I have for this new intrinsic's API:
>>>
>>> * Don't make any assumptions about the underlying capability of the
>>> hardware. For example, do not impose any specific block size
>>> granularity.
>>>
>>> * Don't assume the underlying intrinsic can handle isMIME or isURL
>>> modes, but also let them decide if they will process the data
>>> regardless
>>> of the settings of the two booleans.
>>>
>>> * Any remaining data that is not processed by the intrinsic will be
>>> processed by the pure Java implementation. This allows the
>>> intrinsic to
>>> process whatever block sizes it's good at without the complexity of
>>> handling the end fragments.
>>>
>>> * If any illegal character is discovered in the decoding process, the
>>> intrinsic will simply return -1, instead of requiring it to throw a
>>> proper exception from the context of the intrinsic. In the event of
>>> getting a -1 returned from the intrinsic, the Java Base64 library code
>>> simply calls the pure Java implementation to have it find the error and
>>> properly throw an exception. This is a performance trade-off in the
>>> case of an error (which I expect to be very rare).
>>>
>>> * One thought I have for a further optimization (not implemented in
>>> the current patch), is that when the intrinsic decides not to process a
>>> block because of some combination of isURL and isMIME settings it
>>> doesn't handle, it could return extra bits in the return code, encoded
>>> as a negative number. For example:
>>>
>>> Illegal_Base64_char = 0b001;
>>> isMIME_unsupported = 0b010;
>>> isURL_unsupported = 0b100;
>>>
>>> These can be OR'd together as needed and then negated (flip the sign).
>>> The Base64 library code could then cache these flags, so it will know
>>> not to call the intrinsic again when another decodeBlock is requested
>>> but with an unsupported mode. This will save the performance hit of
>>> calling the intrinsic when it is guaranteed to fail.
>>>
>>> I've tested the attached patch with an actual intrinsic coded up for
>>> Power9/Power10, but those runtime intrinsics and arch-specific patches
>>> aren't attached today. I want to get some consensus on the
>>> library-level intrinsic API first.
>>>
>>> Also attached is a simple test case to test that the new intrinsic API
>>> doesn't break anything.
>>>
>>> I'm open to any comments about this.
>>>
>>> Thanks for your consideration,
>>>
>>> - Corey
>>>
>>>
>>> Corey Ashford
>>> IBM Systems, Linux Technology Center, OpenJDK team
>>> cjashfor at us dot ibm dot com
>>> [attachment "decodeBlock_api-20200623.patch" deleted by Michihiro
>>> Horie/Japan/IBM] [attachment "TestBase64.java" deleted by Michihiro
>>> Horie/Japan/IBM]
>>>
>>>
>>
>
More information about the core-libs-dev
mailing list