2024-04-24 06:44:14 +00:00
/ *
* Copyright ( c ) 2024 , 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 .
* /
package compiler.c2 ;
import compiler.lib.ir_framework.* ;
import jdk.test.lib.Utils ;
import jdk.internal.misc.Unsafe ;
import java.lang.reflect.Array ;
import java.util.Map ;
import java.util.HashMap ;
import java.util.Random ;
/ *
* @test
2024-11-05 11:46:40 +00:00
* @bug 8318446 8331054 8331311 8335392
2024-04-24 06:44:14 +00:00
* @summary Test merging of consecutive stores
* @modules java . base / jdk . internal . misc
* @library / test / lib /
* @run main compiler . c2 . TestMergeStores aligned
* /
/ *
* @test
2024-11-05 11:46:40 +00:00
* @bug 8318446 8331054 8331311 8335392
2024-04-24 06:44:14 +00:00
* @summary Test merging of consecutive stores
* @modules java . base / jdk . internal . misc
* @library / test / lib /
* @run main compiler . c2 . TestMergeStores unaligned
* /
public class TestMergeStores {
static int RANGE = 1000 ;
private static final Unsafe UNSAFE = Unsafe . getUnsafe ( ) ;
private static final Random RANDOM = Utils . getRandomInstance ( ) ;
// Inputs
byte [ ] aB = new byte [ RANGE ] ;
byte [ ] bB = new byte [ RANGE ] ;
short [ ] aS = new short [ RANGE ] ;
short [ ] bS = new short [ RANGE ] ;
int [ ] aI = new int [ RANGE ] ;
int [ ] bI = new int [ RANGE ] ;
long [ ] aL = new long [ RANGE ] ;
long [ ] bL = new long [ RANGE ] ;
int offset1 ;
int offset2 ;
byte vB1 ;
byte vB2 ;
short vS1 ;
short vS2 ;
int vI1 ;
int vI2 ;
long vL1 ;
long vL2 ;
2024-11-05 11:46:40 +00:00
static int zero0 = 0 ;
static int zero1 = 0 ;
static int zero2 = 0 ;
static int zero3 = 0 ;
static int zero4 = 0 ;
static int zero5 = 0 ;
static int zero6 = 0 ;
static int zero7 = 0 ;
static int zero8 = 0 ;
static int zero9 = 0 ;
2024-04-24 06:44:14 +00:00
interface TestFunction {
Object [ ] run ( boolean isWarmUp , int rnd ) ;
}
Map < String , Map < String , TestFunction > > testGroups = new HashMap < String , Map < String , TestFunction > > ( ) ;
public static void main ( String [ ] args ) {
TestFramework framework = new TestFramework ( TestMergeStores . class ) ;
framework . addFlags ( " --add-modules " , " java.base " , " --add-exports " , " java.base/jdk.internal.misc=ALL-UNNAMED " ) ;
switch ( args [ 0 ] ) {
case " aligned " - > { framework . addFlags ( " -XX:-UseUnalignedAccesses " ) ; }
case " unaligned " - > { framework . addFlags ( " -XX:+UseUnalignedAccesses " ) ; }
default - > { throw new RuntimeException ( " Test argument not recognized: " + args [ 0 ] ) ; }
}
framework . start ( ) ;
}
public TestMergeStores ( ) {
testGroups . put ( " test1 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test1 " ) . put ( " test1R " , ( _ , _ ) - > { return test1R ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1a " , ( _ , _ ) - > { return test1a ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1b " , ( _ , _ ) - > { return test1b ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1c " , ( _ , _ ) - > { return test1c ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1d " , ( _ , _ ) - > { return test1d ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1e " , ( _ , _ ) - > { return test1e ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1f " , ( _ , _ ) - > { return test1f ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1g " , ( _ , _ ) - > { return test1g ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1h " , ( _ , _ ) - > { return test1h ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test1 " ) . put ( " test1i " , ( _ , _ ) - > { return test1i ( aB . clone ( ) ) ; } ) ;
testGroups . put ( " test2 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test2 " ) . put ( " test2R " , ( _ , _ ) - > { return test2R ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2 " ) . put ( " test2a " , ( _ , _ ) - > { return test2a ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2 " ) . put ( " test2b " , ( _ , _ ) - > { return test2b ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2 " ) . put ( " test2c " , ( _ , _ ) - > { return test2c ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2 " ) . put ( " test2d " , ( _ , _ ) - > { return test2d ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . get ( " test2 " ) . put ( " test2e " , ( _ , _ ) - > { return test2e ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . put ( " test2BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test2BE " ) . put ( " test2RBE " , ( _ , _ ) - > { return test2RBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2BE " ) . put ( " test2aBE " , ( _ , _ ) - > { return test2aBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2BE " ) . put ( " test2bBE " , ( _ , _ ) - > { return test2bBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2BE " ) . put ( " test2cBE " , ( _ , _ ) - > { return test2cBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2BE " ) . put ( " test2dBE " , ( _ , _ ) - > { return test2dBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test2BE " ) . put ( " test2eBE " , ( _ , _ ) - > { return test2eBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
2024-04-24 06:44:14 +00:00
testGroups . put ( " test3 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test3 " ) . put ( " test3R " , ( _ , _ ) - > { return test3R ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test3 " ) . put ( " test3a " , ( _ , _ ) - > { return test3a ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . put ( " test3BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test3BE " ) . put ( " test3RBE " , ( _ , _ ) - > { return test3RBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test3BE " ) . put ( " test3aBE " , ( _ , _ ) - > { return test3aBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
2024-04-24 06:44:14 +00:00
testGroups . put ( " test4 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test4 " ) . put ( " test4R " , ( _ , _ ) - > { return test4R ( aB . clone ( ) , offset1 , vL1 , vI1 , vS1 , vB1 ) ; } ) ;
testGroups . get ( " test4 " ) . put ( " test4a " , ( _ , _ ) - > { return test4a ( aB . clone ( ) , offset1 , vL1 , vI1 , vS1 , vB1 ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . put ( " test4BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test4BE " ) . put ( " test4RBE " , ( _ , _ ) - > { return test4RBE ( aB . clone ( ) , offset1 , vL1 , vI1 , vS1 , vB1 ) ; } ) ;
testGroups . get ( " test4BE " ) . put ( " test4aBE " , ( _ , _ ) - > { return test4aBE ( aB . clone ( ) , offset1 , vL1 , vI1 , vS1 , vB1 ) ; } ) ;
2024-04-24 06:44:14 +00:00
testGroups . put ( " test5 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test5 " ) . put ( " test5R " , ( _ , _ ) - > { return test5R ( aB . clone ( ) , offset1 ) ; } ) ;
testGroups . get ( " test5 " ) . put ( " test5a " , ( _ , _ ) - > { return test5a ( aB . clone ( ) , offset1 ) ; } ) ;
testGroups . put ( " test6 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test6 " ) . put ( " test6R " , ( _ , _ ) - > { return test6R ( aB . clone ( ) , bB . clone ( ) , offset1 , offset2 ) ; } ) ;
testGroups . get ( " test6 " ) . put ( " test6a " , ( _ , _ ) - > { return test6a ( aB . clone ( ) , bB . clone ( ) , offset1 , offset2 ) ; } ) ;
testGroups . put ( " test7 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test7 " ) . put ( " test7R " , ( _ , _ ) - > { return test7R ( aB . clone ( ) , offset1 , vI1 ) ; } ) ;
testGroups . get ( " test7 " ) . put ( " test7a " , ( _ , _ ) - > { return test7a ( aB . clone ( ) , offset1 , vI1 ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . put ( " test7BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test7BE " ) . put ( " test7RBE " , ( _ , _ ) - > { return test7RBE ( aB . clone ( ) , offset1 , vI1 ) ; } ) ;
testGroups . get ( " test7BE " ) . put ( " test7aBE " , ( _ , _ ) - > { return test7aBE ( aB . clone ( ) , offset1 , vI1 ) ; } ) ;
2024-11-05 11:46:40 +00:00
testGroups . put ( " test10 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test10 " ) . put ( " test10R " , ( _ , _ ) - > { return test10R ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test10 " ) . put ( " test10a " , ( _ , _ ) - > { return test10a ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test10 " ) . put ( " test10b " , ( _ , _ ) - > { return test10b ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test10 " ) . put ( " test10c " , ( _ , _ ) - > { return test10c ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test10 " ) . put ( " test10d " , ( _ , _ ) - > { return test10d ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test10 " ) . put ( " test10e " , ( _ , _ ) - > { return test10e ( aB . clone ( ) ) ; } ) ;
testGroups . get ( " test10 " ) . put ( " test10f " , ( _ , _ ) - > { return test10f ( aB . clone ( ) ) ; } ) ;
2024-04-24 06:44:14 +00:00
testGroups . put ( " test100 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test100 " ) . put ( " test100R " , ( _ , _ ) - > { return test100R ( aS . clone ( ) , offset1 ) ; } ) ;
testGroups . get ( " test100 " ) . put ( " test100a " , ( _ , _ ) - > { return test100a ( aS . clone ( ) , offset1 ) ; } ) ;
testGroups . put ( " test101 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test101 " ) . put ( " test101R " , ( _ , _ ) - > { return test101R ( aS . clone ( ) , offset1 ) ; } ) ;
testGroups . get ( " test101 " ) . put ( " test101a " , ( _ , _ ) - > { return test101a ( aS . clone ( ) , offset1 ) ; } ) ;
testGroups . put ( " test102 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test102 " ) . put ( " test102R " , ( _ , _ ) - > { return test102R ( aS . clone ( ) , offset1 , vL1 , vI1 , vS1 ) ; } ) ;
testGroups . get ( " test102 " ) . put ( " test102a " , ( _ , _ ) - > { return test102a ( aS . clone ( ) , offset1 , vL1 , vI1 , vS1 ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . put ( " test102BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test102BE " ) . put ( " test102RBE " , ( _ , _ ) - > { return test102RBE ( aS . clone ( ) , offset1 , vL1 , vI1 , vS1 ) ; } ) ;
testGroups . get ( " test102BE " ) . put ( " test102aBE " , ( _ , _ ) - > { return test102aBE ( aS . clone ( ) , offset1 , vL1 , vI1 , vS1 ) ; } ) ;
2024-04-24 06:44:14 +00:00
testGroups . put ( " test200 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test200 " ) . put ( " test200R " , ( _ , _ ) - > { return test200R ( aI . clone ( ) , offset1 ) ; } ) ;
testGroups . get ( " test200 " ) . put ( " test200a " , ( _ , _ ) - > { return test200a ( aI . clone ( ) , offset1 ) ; } ) ;
testGroups . put ( " test201 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test201 " ) . put ( " test201R " , ( _ , _ ) - > { return test201R ( aI . clone ( ) , offset1 ) ; } ) ;
testGroups . get ( " test201 " ) . put ( " test201a " , ( _ , _ ) - > { return test201a ( aI . clone ( ) , offset1 ) ; } ) ;
testGroups . put ( " test202 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test202 " ) . put ( " test202R " , ( _ , _ ) - > { return test202R ( aI . clone ( ) , offset1 , vL1 , vI1 ) ; } ) ;
testGroups . get ( " test202 " ) . put ( " test202a " , ( _ , _ ) - > { return test202a ( aI . clone ( ) , offset1 , vL1 , vI1 ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . put ( " test202BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test202BE " ) . put ( " test202RBE " , ( _ , _ ) - > { return test202RBE ( aI . clone ( ) , offset1 , vL1 , vI1 ) ; } ) ;
testGroups . get ( " test202BE " ) . put ( " test202aBE " , ( _ , _ ) - > { return test202aBE ( aI . clone ( ) , offset1 , vL1 , vI1 ) ; } ) ;
2024-04-24 06:44:14 +00:00
testGroups . put ( " test300 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test300 " ) . put ( " test300R " , ( _ , _ ) - > { return test300R ( aI . clone ( ) ) ; } ) ;
testGroups . get ( " test300 " ) . put ( " test300a " , ( _ , _ ) - > { return test300a ( aI . clone ( ) ) ; } ) ;
testGroups . put ( " test400 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test400 " ) . put ( " test400R " , ( _ , _ ) - > { return test400R ( aI . clone ( ) ) ; } ) ;
testGroups . get ( " test400 " ) . put ( " test400a " , ( _ , _ ) - > { return test400a ( aI . clone ( ) ) ; } ) ;
testGroups . put ( " test500 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test500 " ) . put ( " test500R " , ( _ , _ ) - > { return test500R ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test500 " ) . put ( " test500a " , ( _ , _ ) - > { return test500a ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . put ( " test501 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test501 " ) . put ( " test500R " , ( _ , i ) - > { return test500R ( aB . clone ( ) , RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
testGroups . get ( " test501 " ) . put ( " test501a " , ( _ , i ) - > { return test501a ( aB . clone ( ) , RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
// +-------------------+
// Create offsets that are sometimes going to pass all RangeChecks, and sometimes one, and sometimes none.
// Consequence: all RangeChecks stay in the final compilation.
testGroups . put ( " test502 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test502 " ) . put ( " test500R " , ( w , i ) - > { return test500R ( aB . clone ( ) , w ? offset1 : RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
testGroups . get ( " test502 " ) . put ( " test502a " , ( w , i ) - > { return test502a ( aB . clone ( ) , w ? offset1 : RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
// +-----+ +-------------------+
// First use something in range, and after warmup randomize going outside the range.
// Consequence: all RangeChecks stay in the final compilation.
2024-04-24 19:06:46 +00:00
2024-06-07 06:16:03 +00:00
testGroups . put ( " test500BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test500BE " ) . put ( " test500RBE " , ( _ , _ ) - > { return test500RBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test500BE " ) . put ( " test500aBE " , ( _ , _ ) - > { return test500aBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . put ( " test501BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test501BE " ) . put ( " test500RBE " , ( _ , i ) - > { return test500RBE ( aB . clone ( ) , RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
testGroups . get ( " test501BE " ) . put ( " test501aBE " , ( _ , i ) - > { return test501aBE ( aB . clone ( ) , RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
// +-------------------+
// Create offsets that are sometimes going to pass all RangeChecks, and sometimes one, and sometimes none.
// Consequence: all RangeChecks stay in the final compilation.
testGroups . put ( " test502BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test502BE " ) . put ( " test500RBE " , ( w , i ) - > { return test500RBE ( aB . clone ( ) , w ? offset1 : RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
testGroups . get ( " test502BE " ) . put ( " test502aBE " , ( w , i ) - > { return test502aBE ( aB . clone ( ) , w ? offset1 : RANGE - 20 + ( i % 30 ) , vL1 ) ; } ) ;
// +-----+ +-------------------+
// First use something in range, and after warmup randomize going outside the range.
// Consequence: all RangeChecks stay in the final compilation.
2024-04-24 19:06:46 +00:00
testGroups . put ( " test600 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test600 " ) . put ( " test600R " , ( _ , i ) - > { return test600R ( aB . clone ( ) , aI . clone ( ) , i ) ; } ) ;
testGroups . get ( " test600 " ) . put ( " test600a " , ( _ , i ) - > { return test600a ( aB . clone ( ) , aI . clone ( ) , i ) ; } ) ;
2024-04-30 16:15:07 +00:00
2024-11-05 11:46:40 +00:00
testGroups . put ( " test601 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test601 " ) . put ( " test601R " , ( _ , i ) - > { return test601R ( aB . clone ( ) , aI . clone ( ) , i , offset1 ) ; } ) ;
testGroups . get ( " test601 " ) . put ( " test601a " , ( _ , i ) - > { return test601a ( aB . clone ( ) , aI . clone ( ) , i , offset1 ) ; } ) ;
2024-04-30 16:15:07 +00:00
testGroups . put ( " test700 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test700 " ) . put ( " test700R " , ( _ , i ) - > { return test700R ( aI . clone ( ) , i ) ; } ) ;
testGroups . get ( " test700 " ) . put ( " test700a " , ( _ , i ) - > { return test700a ( aI . clone ( ) , i ) ; } ) ;
2024-06-07 06:16:03 +00:00
testGroups . put ( " test800 " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test800 " ) . put ( " test800R " , ( _ , _ ) - > { return test800R ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test800 " ) . put ( " test800a " , ( _ , _ ) - > { return test800a ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . put ( " test800BE " , new HashMap < String , TestFunction > ( ) ) ;
testGroups . get ( " test800BE " ) . put ( " test800RBE " , ( _ , _ ) - > { return test800RBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
testGroups . get ( " test800BE " ) . put ( " test800aBE " , ( _ , _ ) - > { return test800aBE ( aB . clone ( ) , offset1 , vL1 ) ; } ) ;
2024-04-24 06:44:14 +00:00
}
@Warmup ( 100 )
@Run ( test = { " test1a " ,
" test1b " ,
" test1c " ,
" test1d " ,
" test1e " ,
" test1f " ,
" test1g " ,
" test1h " ,
" test1i " ,
" test2a " ,
" test2b " ,
" test2c " ,
" test2d " ,
" test2e " ,
2024-06-07 06:16:03 +00:00
" test2aBE " ,
" test2bBE " ,
" test2cBE " ,
" test2dBE " ,
" test2eBE " ,
2024-04-24 06:44:14 +00:00
" test3a " ,
2024-06-07 06:16:03 +00:00
" test3aBE " ,
2024-04-24 06:44:14 +00:00
" test4a " ,
2024-06-07 06:16:03 +00:00
" test4aBE " ,
2024-04-24 06:44:14 +00:00
" test5a " ,
" test6a " ,
" test7a " ,
2024-11-05 11:46:40 +00:00
" test10a " ,
" test10b " ,
" test10c " ,
" test10d " ,
" test10e " ,
" test10f " ,
2024-06-07 06:16:03 +00:00
" test7aBE " ,
2024-04-24 06:44:14 +00:00
" test100a " ,
" test101a " ,
" test102a " ,
2024-06-07 06:16:03 +00:00
" test102aBE " ,
2024-04-24 06:44:14 +00:00
" test200a " ,
" test201a " ,
" test202a " ,
2024-06-07 06:16:03 +00:00
" test202aBE " ,
2024-04-24 06:44:14 +00:00
" test300a " ,
" test400a " ,
" test500a " ,
" test501a " ,
2024-04-24 19:06:46 +00:00
" test502a " ,
2024-06-07 06:16:03 +00:00
" test500aBE " ,
" test501aBE " ,
" test502aBE " ,
2024-04-30 16:15:07 +00:00
" test600a " ,
2024-11-05 11:46:40 +00:00
" test601a " ,
2024-06-07 06:16:03 +00:00
" test700a " ,
" test800a " ,
" test800aBE " } )
2024-04-24 06:44:14 +00:00
public void runTests ( RunInfo info ) {
// Repeat many times, so that we also have multiple iterations for post-warmup to potentially recompile
int iters = info . isWarmUp ( ) ? 1_000 : 50_000 ;
for ( int iter = 0 ; iter < iters ; iter + + ) {
// Write random values to inputs
set_random ( aB ) ;
set_random ( bB ) ;
set_random ( aS ) ;
set_random ( bS ) ;
set_random ( aI ) ;
set_random ( bI ) ;
set_random ( aL ) ;
set_random ( bL ) ;
offset1 = Math . abs ( RANDOM . nextInt ( ) ) % 100 ;
offset2 = Math . abs ( RANDOM . nextInt ( ) ) % 100 ;
vB1 = ( byte ) RANDOM . nextInt ( ) ;
vB2 = ( byte ) RANDOM . nextInt ( ) ;
vS1 = ( short ) RANDOM . nextInt ( ) ;
vS2 = ( short ) RANDOM . nextInt ( ) ;
vI1 = RANDOM . nextInt ( ) ;
vI2 = RANDOM . nextInt ( ) ;
vL1 = RANDOM . nextLong ( ) ;
vL2 = RANDOM . nextLong ( ) ;
// Run all tests
for ( Map . Entry < String , Map < String , TestFunction > > group_entry : testGroups . entrySet ( ) ) {
String group_name = group_entry . getKey ( ) ;
Map < String , TestFunction > group = group_entry . getValue ( ) ;
Object [ ] gold = null ;
String gold_name = " NONE " ;
for ( Map . Entry < String , TestFunction > entry : group . entrySet ( ) ) {
String name = entry . getKey ( ) ;
TestFunction test = entry . getValue ( ) ;
Object [ ] result = test . run ( info . isWarmUp ( ) , iter ) ;
if ( gold = = null ) {
gold = result ;
gold_name = name ;
} else {
verify ( " group " + group_name + " , gold " + gold_name + " , test " + name , gold , result ) ;
}
}
}
}
}
static void verify ( String name , Object [ ] gold , Object [ ] result ) {
if ( gold . length ! = result . length ) {
throw new RuntimeException ( " verify " + name + " : not the same number of outputs: gold.length = " +
gold . length + " , result.length = " + result . length ) ;
}
for ( int i = 0 ; i < gold . length ; i + + ) {
Object g = gold [ i ] ;
Object r = result [ i ] ;
if ( g . getClass ( ) ! = r . getClass ( ) | | ! g . getClass ( ) . isArray ( ) | | ! r . getClass ( ) . isArray ( ) ) {
throw new RuntimeException ( " verify " + name + " : must both be array of same type: " +
" gold[ " + i + " ].getClass() = " + g . getClass ( ) . getSimpleName ( ) +
" result[ " + i + " ].getClass() = " + r . getClass ( ) . getSimpleName ( ) ) ;
}
if ( g = = r ) {
throw new RuntimeException ( " verify " + name + " : should be two separate arrays (with identical content): " +
" gold[ " + i + " ] == result[ " + i + " ] " ) ;
}
if ( Array . getLength ( g ) ! = Array . getLength ( r ) ) {
throw new RuntimeException ( " verify " + name + " : arrays must have same length: " +
" gold[ " + i + " ].length = " + Array . getLength ( g ) +
" result[ " + i + " ].length = " + Array . getLength ( r ) ) ;
}
Class c = g . getClass ( ) . getComponentType ( ) ;
if ( c = = byte . class ) {
verifyB ( name , i , ( byte [ ] ) g , ( byte [ ] ) r ) ;
} else if ( c = = short . class ) {
verifyS ( name , i , ( short [ ] ) g , ( short [ ] ) r ) ;
} else if ( c = = int . class ) {
verifyI ( name , i , ( int [ ] ) g , ( int [ ] ) r ) ;
} else if ( c = = long . class ) {
verifyL ( name , i , ( long [ ] ) g , ( long [ ] ) r ) ;
} else {
throw new RuntimeException ( " verify " + name + " : array type not supported for verify: " +
" gold[ " + i + " ].getClass() = " + g . getClass ( ) . getSimpleName ( ) +
" result[ " + i + " ].getClass() = " + r . getClass ( ) . getSimpleName ( ) ) ;
}
}
}
static void verifyB ( String name , int i , byte [ ] g , byte [ ] r ) {
for ( int j = 0 ; j < g . length ; j + + ) {
if ( g [ j ] ! = r [ j ] ) {
throw new RuntimeException ( " verify " + name + " : arrays must have same content: " +
" gold[ " + i + " ][ " + j + " ] = " + g [ j ] +
" = " + String . format ( " %02X " , g [ j ] & 0xFF ) +
" result[ " + i + " ][ " + j + " ] = " + r [ j ] +
" = " + String . format ( " %02X " , r [ j ] & 0xFF ) ) ;
}
}
}
static void verifyS ( String name , int i , short [ ] g , short [ ] r ) {
for ( int j = 0 ; j < g . length ; j + + ) {
if ( g [ j ] ! = r [ j ] ) {
throw new RuntimeException ( " verify " + name + " : arrays must have same content: " +
" gold[ " + i + " ][ " + j + " ] = " + g [ j ] +
" result[ " + i + " ][ " + j + " ] = " + r [ j ] ) ;
}
}
}
static void verifyI ( String name , int i , int [ ] g , int [ ] r ) {
for ( int j = 0 ; j < g . length ; j + + ) {
if ( g [ j ] ! = r [ j ] ) {
throw new RuntimeException ( " verify " + name + " : arrays must have same content: " +
" gold[ " + i + " ][ " + j + " ] = " + g [ j ] +
" result[ " + i + " ][ " + j + " ] = " + r [ j ] ) ;
}
}
}
static void verifyL ( String name , int i , long [ ] g , long [ ] r ) {
for ( int j = 0 ; j < g . length ; j + + ) {
if ( g [ j ] ! = r [ j ] ) {
throw new RuntimeException ( " verify " + name + " : arrays must have same content: " +
" gold[ " + i + " ][ " + j + " ] = " + g [ j ] +
" result[ " + i + " ][ " + j + " ] = " + r [ j ] ) ;
}
}
}
static void set_random ( byte [ ] a ) {
for ( int i = 0 ; i < a . length ; i + + ) {
a [ i ] = ( byte ) RANDOM . nextInt ( ) ;
}
}
static void set_random ( short [ ] a ) {
for ( int i = 0 ; i < a . length ; i + + ) {
a [ i ] = ( short ) RANDOM . nextInt ( ) ;
}
}
static void set_random ( int [ ] a ) {
for ( int i = 0 ; i < a . length ; i + + ) {
a [ i ] = RANDOM . nextInt ( ) ;
}
}
static void set_random ( long [ ] a ) {
for ( int i = 0 ; i < a . length ; i + + ) {
a [ i ] = RANDOM . nextLong ( ) ;
}
}
// -------------------------------------------
// ------- Little-Endian API ----------
// -------------------------------------------
// Note: I had to add @ForceInline because otherwise it would sometimes
// not inline nested method calls.
// Store a short LE into an array using store bytes in an array
@ForceInline
static void storeShortLE ( byte [ ] bytes , int offset , short value ) {
storeBytes ( bytes , offset , ( byte ) ( value > > 0 ) ,
( byte ) ( value > > 8 ) ) ;
}
// Store an int LE into an array using store bytes in an array
@ForceInline
static void storeIntLE ( byte [ ] bytes , int offset , int value ) {
storeBytes ( bytes , offset , ( byte ) ( value > > 0 ) ,
( byte ) ( value > > 8 ) ,
( byte ) ( value > > 16 ) ,
( byte ) ( value > > 24 ) ) ;
}
// Store an int LE into an array using store bytes in an array
@ForceInline
static void storeLongLE ( byte [ ] bytes , int offset , long value ) {
storeBytes ( bytes , offset , ( byte ) ( value > > 0 ) ,
( byte ) ( value > > 8 ) ,
( byte ) ( value > > 16 ) ,
( byte ) ( value > > 24 ) ,
( byte ) ( value > > 32 ) ,
( byte ) ( value > > 40 ) ,
( byte ) ( value > > 48 ) ,
( byte ) ( value > > 56 ) ) ;
}
2024-06-07 06:16:03 +00:00
// -------------------------------------------
// ------- Big-Endian API ----------
// -------------------------------------------
// Store a short BE into an array using store bytes in an array
@ForceInline
static void storeShortBE ( byte [ ] bytes , int offset , short value ) {
storeBytes ( bytes , offset , ( byte ) ( value > > 8 ) ,
( byte ) ( value > > 0 ) ) ;
}
// Store an int BE into an array using store bytes in an array
@ForceInline
static void storeIntBE ( byte [ ] bytes , int offset , int value ) {
storeBytes ( bytes , offset , ( byte ) ( value > > 24 ) ,
( byte ) ( value > > 16 ) ,
( byte ) ( value > > 8 ) ,
( byte ) ( value > > 0 ) ) ;
}
// Store an int BE into an array using store bytes in an array
@ForceInline
static void storeLongBE ( byte [ ] bytes , int offset , long value ) {
storeBytes ( bytes , offset , ( byte ) ( value > > 56 ) ,
( byte ) ( value > > 48 ) ,
( byte ) ( value > > 40 ) ,
( byte ) ( value > > 32 ) ,
( byte ) ( value > > 24 ) ,
( byte ) ( value > > 16 ) ,
( byte ) ( value > > 8 ) ,
( byte ) ( value > > 0 ) ) ;
}
2024-04-24 06:44:14 +00:00
// Store 2 bytes into an array
@ForceInline
static void storeBytes ( byte [ ] bytes , int offset , byte b0 , byte b1 ) {
bytes [ offset + 0 ] = b0 ;
bytes [ offset + 1 ] = b1 ;
}
// Store 4 bytes into an array
@ForceInline
static void storeBytes ( byte [ ] bytes , int offset , byte b0 , byte b1 , byte b2 , byte b3 ) {
bytes [ offset + 0 ] = b0 ;
bytes [ offset + 1 ] = b1 ;
bytes [ offset + 2 ] = b2 ;
bytes [ offset + 3 ] = b3 ;
}
// Store 8 bytes into an array
@ForceInline
static void storeBytes ( byte [ ] bytes , int offset , byte b0 , byte b1 , byte b2 , byte b3 ,
byte b4 , byte b5 , byte b6 , byte b7 ) {
bytes [ offset + 0 ] = b0 ;
bytes [ offset + 1 ] = b1 ;
bytes [ offset + 2 ] = b2 ;
bytes [ offset + 3 ] = b3 ;
bytes [ offset + 4 ] = b4 ;
bytes [ offset + 5 ] = b5 ;
bytes [ offset + 6 ] = b6 ;
bytes [ offset + 7 ] = b7 ;
}
@DontCompile
static Object [ ] test1R ( byte [ ] a ) {
a [ 0 ] = ( byte ) 0xbe ;
a [ 1 ] = ( byte ) 0xba ;
a [ 2 ] = ( byte ) 0xad ;
a [ 3 ] = ( byte ) 0xba ;
a [ 4 ] = ( byte ) 0xef ;
a [ 5 ] = ( byte ) 0xbe ;
a [ 6 ] = ( byte ) 0xad ;
a [ 7 ] = ( byte ) 0xde ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1a ( byte [ ] a ) {
a [ 0 ] = ( byte ) 0xbe ;
a [ 1 ] = ( byte ) 0xba ;
a [ 2 ] = ( byte ) 0xad ;
a [ 3 ] = ( byte ) 0xba ;
a [ 4 ] = ( byte ) 0xef ;
a [ 5 ] = ( byte ) 0xbe ;
a [ 6 ] = ( byte ) 0xad ;
a [ 7 ] = ( byte ) 0xde ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1b ( byte [ ] a ) {
// Add custom null check, to ensure the unsafe access always recognizes its type as an array store
if ( a = = null ) { return null ; }
2024-06-07 06:16:03 +00:00
UNSAFE . putLongUnaligned ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET , 0xdeadbeefbaadbabeL , false /* bigEndian */ ) ;
2024-04-24 06:44:14 +00:00
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1c ( byte [ ] a ) {
storeLongLE ( a , 0 , 0xdeadbeefbaadbabeL ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1d ( byte [ ] a ) {
storeIntLE ( a , 0 , 0xbaadbabe ) ;
storeIntLE ( a , 4 , 0xdeadbeef ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1e ( byte [ ] a ) {
storeShortLE ( a , 0 , ( short ) 0xbabe ) ;
storeShortLE ( a , 2 , ( short ) 0xbaad ) ;
storeShortLE ( a , 4 , ( short ) 0xbeef ) ;
storeShortLE ( a , 6 , ( short ) 0xdead ) ;
return new Object [ ] { a } ;
}
@Test
2024-11-05 11:46:40 +00:00
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test1f ( byte [ ] a ) {
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 1 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 2 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 3 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 4 , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 6 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 7 , ( byte ) 0xde ) ;
return new Object [ ] { a } ;
}
@Test
// Do not optimize these, just to be sure we do not mess with store ordering.
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1g ( byte [ ] a ) {
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 1 , ( byte ) 0xba ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 2 , ( byte ) 0xad ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 3 , ( byte ) 0xba ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 4 , ( byte ) 0xef ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 6 , ( byte ) 0xad ) ;
UNSAFE . putByteRelease ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 7 , ( byte ) 0xde ) ;
return new Object [ ] { a } ;
}
@Test
// Do not optimize these, just to be sure we do not mess with store ordering.
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1h ( byte [ ] a ) {
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 1 , ( byte ) 0xba ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 2 , ( byte ) 0xad ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 3 , ( byte ) 0xba ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 4 , ( byte ) 0xef ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 6 , ( byte ) 0xad ) ;
UNSAFE . putByteVolatile ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 7 , ( byte ) 0xde ) ;
return new Object [ ] { a } ;
}
@Test
// Do not optimize these, just to be sure we do not mess with store ordering.
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test1i ( byte [ ] a ) {
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 1 , ( byte ) 0xba ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 2 , ( byte ) 0xad ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 3 , ( byte ) 0xba ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 4 , ( byte ) 0xef ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 6 , ( byte ) 0xad ) ;
UNSAFE . putByteOpaque ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 7 , ( byte ) 0xde ) ;
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test2R ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 48 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 56 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test2a ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 48 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 56 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test2b ( byte [ ] a , int offset , long v ) {
// Add custom null check, to ensure the unsafe access always recognizes its type as an array store
if ( a = = null ) { return null ; }
2024-06-07 06:16:03 +00:00
UNSAFE . putLongUnaligned ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset , v , false /* bigEndian */ ) ;
2024-04-24 06:44:14 +00:00
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test2c ( byte [ ] a , int offset , long v ) {
storeLongLE ( a , offset , v ) ;
return new Object [ ] { a } ;
}
@Test
// No optimization, casting long -> int -> byte does not work
static Object [ ] test2d ( byte [ ] a , int offset , long v ) {
storeIntLE ( a , offset + 0 , ( int ) ( v > > 0 ) ) ;
storeIntLE ( a , offset + 4 , ( int ) ( v > > 32 ) ) ;
return new Object [ ] { a } ;
}
@Test
// No optimization, casting long -> short -> byte does not work
static Object [ ] test2e ( byte [ ] a , int offset , long v ) {
storeShortLE ( a , offset + 0 , ( short ) ( v > > 0 ) ) ;
storeShortLE ( a , offset + 2 , ( short ) ( v > > 16 ) ) ;
storeShortLE ( a , offset + 4 , ( short ) ( v > > 32 ) ) ;
storeShortLE ( a , offset + 6 , ( short ) ( v > > 48 ) ) ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test2RBE ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 56 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 48 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 40 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 32 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test2aBE ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 56 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 48 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 40 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 32 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test2bBE ( byte [ ] a , int offset , long v ) {
// Add custom null check, to ensure the unsafe access always recognizes its type as an array store
if ( a = = null ) { return null ; }
UNSAFE . putLongUnaligned ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset , v , true /* bigEndian */ ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test2cBE ( byte [ ] a , int offset , long v ) {
storeLongBE ( a , offset , v ) ;
return new Object [ ] { a } ;
}
@Test
// No optimization, casting long -> int -> byte does not work
static Object [ ] test2dBE ( byte [ ] a , int offset , long v ) {
storeIntBE ( a , offset + 0 , ( int ) ( v > > 32 ) ) ;
storeIntBE ( a , offset + 4 , ( int ) ( v > > 0 ) ) ;
return new Object [ ] { a } ;
}
@Test
// No optimization, casting long -> short -> byte does not work
static Object [ ] test2eBE ( byte [ ] a , int offset , long v ) {
storeShortBE ( a , offset + 0 , ( short ) ( v > > 48 ) ) ;
storeShortBE ( a , offset + 2 , ( short ) ( v > > 32 ) ) ;
storeShortBE ( a , offset + 4 , ( short ) ( v > > 16 ) ) ;
storeShortBE ( a , offset + 6 , ( short ) ( v > > 0 ) ) ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
@DontCompile
static Object [ ] test3R ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 24 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " } ,
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test3a ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 24 ) ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test3RBE ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test3aBE ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
@DontCompile
static Object [ ] test4R ( byte [ ] a , int offset , long v1 , int v2 , short v3 , byte v4 ) {
a [ offset + 0 ] = ( byte ) 0x00 ;
a [ offset + 1 ] = ( byte ) 0xFF ;
a [ offset + 2 ] = v4 ;
a [ offset + 3 ] = ( byte ) 0x42 ;
a [ offset + 4 ] = ( byte ) ( v1 > > 0 ) ;
a [ offset + 5 ] = ( byte ) ( v1 > > 8 ) ;
a [ offset + 6 ] = ( byte ) 0xAB ;
a [ offset + 7 ] = ( byte ) 0xCD ;
a [ offset + 8 ] = ( byte ) 0xEF ;
a [ offset + 9 ] = ( byte ) 0x01 ;
a [ offset + 10 ] = ( byte ) ( v2 > > 0 ) ;
a [ offset + 11 ] = ( byte ) ( v2 > > 8 ) ;
a [ offset + 12 ] = ( byte ) ( v2 > > 16 ) ;
a [ offset + 13 ] = ( byte ) ( v2 > > 24 ) ;
a [ offset + 14 ] = ( byte ) ( v3 > > 0 ) ;
a [ offset + 15 ] = ( byte ) ( v3 > > 8 ) ;
a [ offset + 16 ] = ( byte ) 0xEF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 4 " , // 3 (+ 1 for uncommon trap)
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 12 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // Stores of constants can be merged
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test4a ( byte [ ] a , int offset , long v1 , int v2 , short v3 , byte v4 ) {
a [ offset + 0 ] = ( byte ) 0x00 ; // individual load expected to go into state of RC
a [ offset + 1 ] = ( byte ) 0xFF ;
a [ offset + 2 ] = v4 ;
a [ offset + 3 ] = ( byte ) 0x42 ;
a [ offset + 4 ] = ( byte ) ( v1 > > 0 ) ;
a [ offset + 5 ] = ( byte ) ( v1 > > 8 ) ;
a [ offset + 6 ] = ( byte ) 0xAB ;
a [ offset + 7 ] = ( byte ) 0xCD ;
a [ offset + 8 ] = ( byte ) 0xEF ;
a [ offset + 9 ] = ( byte ) 0x01 ;
a [ offset + 10 ] = ( byte ) ( v2 > > 0 ) ;
a [ offset + 11 ] = ( byte ) ( v2 > > 8 ) ;
a [ offset + 12 ] = ( byte ) ( v2 > > 16 ) ;
a [ offset + 13 ] = ( byte ) ( v2 > > 24 ) ;
a [ offset + 14 ] = ( byte ) ( v3 > > 0 ) ;
a [ offset + 15 ] = ( byte ) ( v3 > > 8 ) ;
a [ offset + 16 ] = ( byte ) 0xEF ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test4RBE ( byte [ ] a , int offset , long v1 , int v2 , short v3 , byte v4 ) {
a [ offset + 0 ] = ( byte ) 0x00 ;
a [ offset + 1 ] = ( byte ) 0xFF ;
a [ offset + 2 ] = v4 ;
a [ offset + 3 ] = ( byte ) 0x42 ;
a [ offset + 4 ] = ( byte ) ( v1 > > 8 ) ;
a [ offset + 5 ] = ( byte ) ( v1 > > 0 ) ;
a [ offset + 6 ] = ( byte ) 0xAB ;
a [ offset + 7 ] = ( byte ) 0xCD ;
a [ offset + 8 ] = ( byte ) 0xEF ;
a [ offset + 9 ] = ( byte ) 0x01 ;
a [ offset + 10 ] = ( byte ) ( v2 > > 24 ) ;
a [ offset + 11 ] = ( byte ) ( v2 > > 16 ) ;
a [ offset + 12 ] = ( byte ) ( v2 > > 8 ) ;
a [ offset + 13 ] = ( byte ) ( v2 > > 0 ) ;
a [ offset + 14 ] = ( byte ) ( v3 > > 8 ) ;
a [ offset + 15 ] = ( byte ) ( v3 > > 0 ) ;
a [ offset + 16 ] = ( byte ) 0xEF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 12 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // Stores of constants can be merged
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 4 " , // 3 (+ 1 for uncommon trap)
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test4aBE ( byte [ ] a , int offset , long v1 , int v2 , short v3 , byte v4 ) {
a [ offset + 0 ] = ( byte ) 0x00 ; // individual load expected to go into state of RC
a [ offset + 1 ] = ( byte ) 0xFF ;
a [ offset + 2 ] = v4 ;
a [ offset + 3 ] = ( byte ) 0x42 ;
a [ offset + 4 ] = ( byte ) ( v1 > > 8 ) ;
a [ offset + 5 ] = ( byte ) ( v1 > > 0 ) ;
a [ offset + 6 ] = ( byte ) 0xAB ;
a [ offset + 7 ] = ( byte ) 0xCD ;
a [ offset + 8 ] = ( byte ) 0xEF ;
a [ offset + 9 ] = ( byte ) 0x01 ;
a [ offset + 10 ] = ( byte ) ( v2 > > 24 ) ;
a [ offset + 11 ] = ( byte ) ( v2 > > 16 ) ;
a [ offset + 12 ] = ( byte ) ( v2 > > 8 ) ;
a [ offset + 13 ] = ( byte ) ( v2 > > 0 ) ;
a [ offset + 14 ] = ( byte ) ( v3 > > 8 ) ;
a [ offset + 15 ] = ( byte ) ( v3 > > 0 ) ;
a [ offset + 16 ] = ( byte ) 0xEF ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
@DontCompile
static Object [ ] test5R ( byte [ ] a , int offset ) {
a [ offset + 0 ] = ( byte ) 0x01 ;
a [ offset + 1 ] = ( byte ) 0x02 ;
a [ offset + 2 ] = ( byte ) 0x03 ;
a [ offset + 3 ] = ( byte ) 0x04 ;
a [ offset + 4 ] = ( byte ) 0x11 ;
a [ offset + 5 ] = ( byte ) 0x22 ;
a [ offset + 6 ] = ( byte ) 0x33 ;
a [ offset + 7 ] = ( byte ) 0x44 ;
a [ offset + 8 ] = ( byte ) 0x55 ;
a [ offset + 9 ] = ( byte ) 0x66 ;
a [ offset + 10 ] = ( byte ) 0x77 ;
a [ offset + 11 ] = ( byte ) 0xAA ;
a [ offset + 12 ] = ( byte ) 0xBB ;
a [ offset + 13 ] = ( byte ) 0xCC ;
a [ offset + 14 ] = ( byte ) 0xDD ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test5a ( byte [ ] a , int offset ) {
a [ offset + 0 ] = ( byte ) 0x01 ;
a [ offset + 1 ] = ( byte ) 0x02 ;
a [ offset + 2 ] = ( byte ) 0x03 ;
a [ offset + 3 ] = ( byte ) 0x04 ;
a [ offset + 4 ] = ( byte ) 0x11 ;
a [ offset + 5 ] = ( byte ) 0x22 ;
a [ offset + 6 ] = ( byte ) 0x33 ;
a [ offset + 7 ] = ( byte ) 0x44 ;
a [ offset + 8 ] = ( byte ) 0x55 ;
a [ offset + 9 ] = ( byte ) 0x66 ;
a [ offset + 10 ] = ( byte ) 0x77 ;
a [ offset + 11 ] = ( byte ) 0xAA ;
a [ offset + 12 ] = ( byte ) 0xBB ;
a [ offset + 13 ] = ( byte ) 0xCC ;
a [ offset + 14 ] = ( byte ) 0xDD ;
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test6R ( byte [ ] a , byte [ ] b , int offset1 , int offset2 ) {
a [ offset1 + 1 ] = ( byte ) 0x02 ;
a [ offset1 + 3 ] = ( byte ) 0x04 ;
b [ offset1 + 4 ] = ( byte ) 0x11 ;
a [ offset1 + 5 ] = ( byte ) 0x22 ;
a [ offset2 + 6 ] = ( byte ) 0x33 ;
a [ offset1 + 7 ] = ( byte ) 0x44 ;
b [ offset1 + 8 ] = ( byte ) 0x55 ;
b [ offset1 + 10 ] = ( byte ) 0x66 ;
return new Object [ ] { a , b } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test6a ( byte [ ] a , byte [ ] b , int offset1 , int offset2 ) {
a [ offset1 + 1 ] = ( byte ) 0x02 ;
a [ offset1 + 3 ] = ( byte ) 0x04 ;
b [ offset1 + 4 ] = ( byte ) 0x11 ;
a [ offset1 + 5 ] = ( byte ) 0x22 ;
a [ offset2 + 6 ] = ( byte ) 0x33 ;
a [ offset1 + 7 ] = ( byte ) 0x44 ;
b [ offset1 + 8 ] = ( byte ) 0x55 ;
b [ offset1 + 10 ] = ( byte ) 0x66 ;
return new Object [ ] { a , b } ;
}
@DontCompile
static Object [ ] test7R ( byte [ ] a , int offset1 , int v1 ) {
a [ offset1 + 1 ] = ( byte ) ( v1 > > 8 ) ;
a [ offset1 + 2 ] = ( byte ) ( v1 > > 16 ) ;
a [ offset1 + 3 ] = ( byte ) ( v1 > > 24 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test7a ( byte [ ] a , int offset1 , int v1 ) {
a [ offset1 + 1 ] = ( byte ) ( v1 > > 8 ) ;
a [ offset1 + 2 ] = ( byte ) ( v1 > > 16 ) ;
a [ offset1 + 3 ] = ( byte ) ( v1 > > 24 ) ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test7RBE ( byte [ ] a , int offset1 , int v1 ) {
a [ offset1 + 1 ] = ( byte ) ( v1 > > 24 ) ;
a [ offset1 + 2 ] = ( byte ) ( v1 > > 16 ) ;
a [ offset1 + 3 ] = ( byte ) ( v1 > > 8 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test7aBE ( byte [ ] a , int offset1 , int v1 ) {
a [ offset1 + 1 ] = ( byte ) ( v1 > > 24 ) ;
a [ offset1 + 2 ] = ( byte ) ( v1 > > 16 ) ;
a [ offset1 + 3 ] = ( byte ) ( v1 > > 8 ) ;
return new Object [ ] { a } ;
}
2024-11-05 11:46:40 +00:00
@DontCompile
static Object [ ] test10R ( byte [ ] a ) {
int zero = zero0 + zero1 + zero2 + zero3 + zero4
+ zero5 + zero6 + zero7 + zero8 + zero9 ;
a [ zero + 0 ] = 'h' ;
a [ zero + 1 ] = 'e' ;
a [ zero + 2 ] = 'l' ;
a [ zero + 3 ] = 'l' ;
a [ zero + 4 ] = 'o' ;
a [ zero + 5 ] = ' ' ;
a [ zero + 6 ] = ':' ;
a [ zero + 7 ] = ')' ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " , // no merge
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test10a ( byte [ ] a ) {
// We have 11 summands: 10x zero variable + 1x array base.
// Parsing only allows 10 summands -> does not merge the stores.
int zero = zero0 + zero1 + zero2 + zero3 + zero4
+ zero5 + zero6 + zero7 + zero8 + zero9 ;
a [ zero + 0 ] = 'h' ;
a [ zero + 1 ] = 'e' ;
a [ zero + 2 ] = 'l' ;
a [ zero + 3 ] = 'l' ;
a [ zero + 4 ] = 'o' ;
a [ zero + 5 ] = ' ' ;
a [ zero + 6 ] = ':' ;
a [ zero + 7 ] = ')' ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // 1 left in uncommon trap path of RangeCheck
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test10b ( byte [ ] a ) {
int zero = zero0 + zero1 + zero2 + zero3 + zero4
+ zero5 + zero6 + zero7 + zero8 ;
// We have 10 summands: 9x zero variable + 1x array base.
// Parsing allows 10 summands, so this should merge the stores.
a [ zero + 0 ] = 'h' ;
a [ zero + 1 ] = 'e' ;
a [ zero + 2 ] = 'l' ;
a [ zero + 3 ] = 'l' ;
a [ zero + 4 ] = 'o' ;
a [ zero + 5 ] = ' ' ;
a [ zero + 6 ] = ':' ;
a [ zero + 7 ] = ')' ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // 1 left in uncommon trap path of RangeCheck
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test10c ( byte [ ] a ) {
int zero = 7 * zero0 + 7 * zero1 + 7 * zero2 + 7 * zero3 + 7 * zero4
+ 7 * zero5 + 7 * zero6 + 7 * zero7 + 7 * zero8 ;
// The "7 * zero" is split into "zero << 3 - zero". But the parsing combines it again, lowering the summand count.
// We have 10 summands: 9x zero variable + 1x array base.
// Parsing allows 10 summands, so this should merge the stores.
a [ zero + 0 ] = 'h' ;
a [ zero + 1 ] = 'e' ;
a [ zero + 2 ] = 'l' ;
a [ zero + 3 ] = 'l' ;
a [ zero + 4 ] = 'o' ;
a [ zero + 5 ] = ' ' ;
a [ zero + 6 ] = ':' ;
a [ zero + 7 ] = ')' ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test10d ( byte [ ] a ) {
// Summand is subtracted from itself -> scale = 0 -> should be removed from list.
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 0 ) - zero0 , ( byte ) 'h' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 1 ) - zero0 , ( byte ) 'e' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 2 ) - zero0 , ( byte ) 'l' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 3 ) - zero0 , ( byte ) 'l' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 4 ) - zero0 , ( byte ) 'o' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 5 ) - zero0 , ( byte ) ' ' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 6 ) - zero0 , ( byte ) ':' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 7 ) - zero0 , ( byte ) ')' ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test10e ( byte [ ] a ) {
// Summand is subtracted from itself -> scale = 0 -> should be removed from list. Thus equal to if not present at all.
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 0 ) - zero0 , ( byte ) 'h' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 1 ) - zero0 , ( byte ) 'e' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 2 ) - zero0 , ( byte ) 'l' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + ( long ) ( zero0 + 3 ) - zero0 , ( byte ) 'l' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 4 , ( byte ) 'o' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 5 , ( byte ) ' ' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 6 , ( byte ) ':' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + 7 , ( byte ) ')' ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " , // no merge
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test10f ( byte [ ] a ) {
int big = 1 < < 29 ;
// Adding up the scales overflows -> no merge.
long offset = zero9 * big + zero9 * big + zero9 * big + zero9 * big ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 0 , ( byte ) 'h' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 1 , ( byte ) 'e' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 2 , ( byte ) 'l' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 3 , ( byte ) 'l' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 4 , ( byte ) 'o' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 5 , ( byte ) ' ' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 6 , ( byte ) ':' ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_BYTE_BASE_OFFSET + offset + 7 , ( byte ) ')' ) ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
@DontCompile
static Object [ ] test100R ( short [ ] a , int offset ) {
a [ offset + 0 ] = ( short ) 0x0100 ;
a [ offset + 1 ] = ( short ) 0x0200 ;
a [ offset + 2 ] = ( short ) 0x0311 ;
a [ offset + 3 ] = ( short ) 0x0400 ;
a [ offset + 4 ] = ( short ) 0x1100 ;
a [ offset + 5 ] = ( short ) 0x2233 ;
a [ offset + 6 ] = ( short ) 0x3300 ;
a [ offset + 7 ] = ( short ) 0x4400 ;
a [ offset + 8 ] = ( short ) 0x5599 ;
a [ offset + 9 ] = ( short ) 0x6600 ;
a [ offset + 10 ] = ( short ) 0x7700 ;
a [ offset + 11 ] = ( short ) 0xAACC ;
a [ offset + 12 ] = ( short ) 0xBB00 ;
a [ offset + 13 ] = ( short ) 0xCC00 ;
a [ offset + 14 ] = ( short ) 0xDDFF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_I_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test100a ( short [ ] a , int offset ) {
a [ offset + 0 ] = ( short ) 0x0100 ; // stays unchanged -> both used for RC and Return path
a [ offset + 1 ] = ( short ) 0x0200 ; // I
a [ offset + 2 ] = ( short ) 0x0311 ; // I
a [ offset + 3 ] = ( short ) 0x0400 ; // L
a [ offset + 4 ] = ( short ) 0x1100 ; // L
a [ offset + 5 ] = ( short ) 0x2233 ; // L
a [ offset + 6 ] = ( short ) 0x3300 ; // L
a [ offset + 7 ] = ( short ) 0x4400 ; // L
a [ offset + 8 ] = ( short ) 0x5599 ; // L
a [ offset + 9 ] = ( short ) 0x6600 ; // L
a [ offset + 10 ] = ( short ) 0x7700 ; // L
a [ offset + 11 ] = ( short ) 0xAACC ; // L
a [ offset + 12 ] = ( short ) 0xBB00 ; // L
a [ offset + 13 ] = ( short ) 0xCC00 ; // L
a [ offset + 14 ] = ( short ) 0xDDFF ; // L
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test101R ( short [ ] a , int offset ) {
a [ offset + 0 ] = ( short ) 0x0100 ;
a [ offset + 1 ] = ( short ) 0x0200 ;
a [ offset + 2 ] = ( short ) 0x0311 ;
a [ offset + 3 ] = ( short ) 0x0400 ;
a [ offset + 4 ] = ( short ) 0x1100 ;
a [ offset + 5 ] = ( short ) 0x2233 ;
a [ offset + 6 ] = ( short ) 0x3300 ;
a [ offset + 7 ] = ( short ) 0x4400 ;
a [ offset + 8 ] = ( short ) 0x5599 ;
a [ offset + 9 ] = ( short ) 0x6600 ;
a [ offset + 10 ] = ( short ) 0x7700 ;
a [ offset + 11 ] = ( short ) 0xAACC ;
a [ offset + 12 ] = ( short ) 0xBB00 ;
a [ offset + 13 ] = ( short ) 0xCC00 ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // only for RC
IRNode . STORE_I_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test101a ( short [ ] a , int offset ) {
a [ offset + 0 ] = ( short ) 0x0100 ; // I plus kept unchanged for RC
a [ offset + 1 ] = ( short ) 0x0200 ; // I
a [ offset + 2 ] = ( short ) 0x0311 ; // L
a [ offset + 3 ] = ( short ) 0x0400 ; // L
a [ offset + 4 ] = ( short ) 0x1100 ; // L
a [ offset + 5 ] = ( short ) 0x2233 ; // L
a [ offset + 6 ] = ( short ) 0x3300 ; // L
a [ offset + 7 ] = ( short ) 0x4400 ; // L
a [ offset + 8 ] = ( short ) 0x5599 ; // L
a [ offset + 9 ] = ( short ) 0x6600 ; // L
a [ offset + 10 ] = ( short ) 0x7700 ; // L
a [ offset + 11 ] = ( short ) 0xAACC ; // L
a [ offset + 12 ] = ( short ) 0xBB00 ; // L
a [ offset + 13 ] = ( short ) 0xCC00 ; // L
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test102R ( short [ ] a , int offset , long v1 , int v2 , short v3 ) {
a [ offset + 0 ] = ( short ) 0x0000 ;
a [ offset + 1 ] = ( short ) 0xFFFF ;
a [ offset + 2 ] = v3 ;
a [ offset + 3 ] = ( short ) 0x4242 ;
a [ offset + 4 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 5 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 6 ] = ( short ) 0xAB11 ;
a [ offset + 7 ] = ( short ) 0xCD36 ;
a [ offset + 8 ] = ( short ) 0xEF89 ;
a [ offset + 9 ] = ( short ) 0x0156 ;
a [ offset + 10 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 11 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 12 ] = ( short ) ( v1 > > 32 ) ;
a [ offset + 13 ] = ( short ) ( v1 > > 48 ) ;
a [ offset + 14 ] = ( short ) ( v2 > > 0 ) ;
a [ offset + 15 ] = ( short ) ( v2 > > 16 ) ;
a [ offset + 16 ] = ( short ) 0xEFEF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 4 " , // 3 (+1 that goes into RC)
IRNode . STORE_I_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " ,
IRNode . STORE_L_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " } ,
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 12 " ,
IRNode . STORE_I_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // Stores of constants can be merged
IRNode . STORE_L_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test102a ( short [ ] a , int offset , long v1 , int v2 , short v3 ) {
a [ offset + 0 ] = ( short ) 0x0000 ; // store goes into RC
a [ offset + 1 ] = ( short ) 0xFFFF ;
a [ offset + 2 ] = v3 ;
a [ offset + 3 ] = ( short ) 0x4242 ;
a [ offset + 4 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 5 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 6 ] = ( short ) 0xAB11 ;
a [ offset + 7 ] = ( short ) 0xCD36 ;
a [ offset + 8 ] = ( short ) 0xEF89 ;
a [ offset + 9 ] = ( short ) 0x0156 ;
a [ offset + 10 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 11 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 12 ] = ( short ) ( v1 > > 32 ) ;
a [ offset + 13 ] = ( short ) ( v1 > > 48 ) ;
a [ offset + 14 ] = ( short ) ( v2 > > 0 ) ;
a [ offset + 15 ] = ( short ) ( v2 > > 16 ) ;
a [ offset + 16 ] = ( short ) 0xEFEF ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test102RBE ( short [ ] a , int offset , long v1 , int v2 , short v3 ) {
a [ offset + 0 ] = ( short ) 0x0000 ;
a [ offset + 1 ] = ( short ) 0xFFFF ;
a [ offset + 2 ] = v3 ;
a [ offset + 3 ] = ( short ) 0x4242 ;
a [ offset + 4 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 5 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 6 ] = ( short ) 0xAB11 ;
a [ offset + 7 ] = ( short ) 0xCD36 ;
a [ offset + 8 ] = ( short ) 0xEF89 ;
a [ offset + 9 ] = ( short ) 0x0156 ;
a [ offset + 10 ] = ( short ) ( v1 > > 48 ) ;
a [ offset + 11 ] = ( short ) ( v1 > > 32 ) ;
a [ offset + 12 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 13 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 14 ] = ( short ) ( v2 > > 16 ) ;
a [ offset + 15 ] = ( short ) ( v2 > > 0 ) ;
a [ offset + 16 ] = ( short ) 0xEFEF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 12 " ,
IRNode . STORE_I_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // Stores of constants can be merged
IRNode . STORE_L_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 4 " , // 3 (+1 that goes into RC)
IRNode . STORE_I_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 3 " ,
IRNode . STORE_L_OF_CLASS , " short \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test102aBE ( short [ ] a , int offset , long v1 , int v2 , short v3 ) {
a [ offset + 0 ] = ( short ) 0x0000 ; // store goes into RC
a [ offset + 1 ] = ( short ) 0xFFFF ;
a [ offset + 2 ] = v3 ;
a [ offset + 3 ] = ( short ) 0x4242 ;
a [ offset + 4 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 5 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 6 ] = ( short ) 0xAB11 ;
a [ offset + 7 ] = ( short ) 0xCD36 ;
a [ offset + 8 ] = ( short ) 0xEF89 ;
a [ offset + 9 ] = ( short ) 0x0156 ;
a [ offset + 10 ] = ( short ) ( v1 > > 48 ) ;
a [ offset + 11 ] = ( short ) ( v1 > > 32 ) ;
a [ offset + 12 ] = ( short ) ( v1 > > 16 ) ;
a [ offset + 13 ] = ( short ) ( v1 > > 0 ) ;
a [ offset + 14 ] = ( short ) ( v2 > > 16 ) ;
a [ offset + 15 ] = ( short ) ( v2 > > 0 ) ;
a [ offset + 16 ] = ( short ) 0xEFEF ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
@DontCompile
static Object [ ] test200R ( int [ ] a , int offset ) {
a [ offset + 0 ] = 0x01001236 ;
a [ offset + 1 ] = 0x02001284 ;
a [ offset + 2 ] = 0x03111235 ;
a [ offset + 3 ] = 0x04001294 ;
a [ offset + 4 ] = 0x11001234 ;
a [ offset + 5 ] = 0x22331332 ;
a [ offset + 6 ] = 0x33001234 ;
a [ offset + 7 ] = 0x44001432 ;
a [ offset + 8 ] = 0x55991234 ;
a [ offset + 9 ] = 0x66001233 ;
a [ offset + 10 ] = 0x77001434 ;
a [ offset + 11 ] = 0xAACC1234 ;
a [ offset + 12 ] = 0xBB001434 ;
a [ offset + 13 ] = 0xCC001236 ;
a [ offset + 14 ] = 0xDDFF1534 ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 7 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test200a ( int [ ] a , int offset ) {
a [ offset + 0 ] = 0x01001236 ; // stays unchanged -> both used for RC and Return path
a [ offset + 1 ] = 0x02001284 ; // L
a [ offset + 2 ] = 0x03111235 ; // L
a [ offset + 3 ] = 0x04001294 ; // L
a [ offset + 4 ] = 0x11001234 ; // L
a [ offset + 5 ] = 0x22331332 ; // L
a [ offset + 6 ] = 0x33001234 ; // L
a [ offset + 7 ] = 0x44001432 ; // L
a [ offset + 8 ] = 0x55991234 ; // L
a [ offset + 9 ] = 0x66001233 ; // L
a [ offset + 10 ] = 0x77001434 ; // L
a [ offset + 11 ] = 0xAACC1234 ; // L
a [ offset + 12 ] = 0xBB001434 ; // L
a [ offset + 13 ] = 0xCC001236 ; // L
a [ offset + 14 ] = 0xDDFF1534 ; // L
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test201R ( int [ ] a , int offset ) {
a [ offset + 0 ] = 0x01001236 ;
a [ offset + 1 ] = 0x02001284 ;
a [ offset + 2 ] = 0x03111235 ;
a [ offset + 3 ] = 0x04001294 ;
a [ offset + 4 ] = 0x11001234 ;
a [ offset + 5 ] = 0x22331332 ;
a [ offset + 6 ] = 0x33001234 ;
a [ offset + 7 ] = 0x44001432 ;
a [ offset + 8 ] = 0x55991234 ;
a [ offset + 9 ] = 0x66001233 ;
a [ offset + 10 ] = 0x77001434 ;
a [ offset + 11 ] = 0xAACC1234 ;
a [ offset + 12 ] = 0xBB001434 ;
a [ offset + 13 ] = 0xCC001236 ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // only for RC
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 7 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test201a ( int [ ] a , int offset ) {
a [ offset + 0 ] = 0x01001236 ; // L and also kept unchanged for RC
a [ offset + 1 ] = 0x02001284 ; // L
a [ offset + 2 ] = 0x03111235 ; // L
a [ offset + 3 ] = 0x04001294 ; // L
a [ offset + 4 ] = 0x11001234 ; // L
a [ offset + 5 ] = 0x22331332 ; // L
a [ offset + 6 ] = 0x33001234 ; // L
a [ offset + 7 ] = 0x44001432 ; // L
a [ offset + 8 ] = 0x55991234 ; // L
a [ offset + 9 ] = 0x66001233 ; // L
a [ offset + 10 ] = 0x77001434 ; // L
a [ offset + 11 ] = 0xAACC1234 ; // L
a [ offset + 12 ] = 0xBB001434 ; // L
a [ offset + 13 ] = 0xCC001236 ; // L
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test202R ( int [ ] a , int offset , long v1 , int v2 ) {
a [ offset + 0 ] = 0x00000000 ;
a [ offset + 1 ] = 0xFFFFFFFF ;
a [ offset + 2 ] = v2 ;
a [ offset + 3 ] = 0x42424242 ;
a [ offset + 4 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 5 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 6 ] = 0xAB110129 ;
a [ offset + 7 ] = 0xCD360183 ;
a [ offset + 8 ] = 0xEF890173 ;
a [ offset + 9 ] = 0x01560124 ;
a [ offset + 10 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 11 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 12 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 13 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 14 ] = v2 ;
a [ offset + 15 ] = v2 ;
a [ offset + 16 ] = 0xEFEFEFEF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 6 " , // 5 (+1 that goes into RC)
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 6 " } ,
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 10 " ,
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 4 " } , // Stores of constants can be merged
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test202a ( int [ ] a , int offset , long v1 , int v2 ) {
a [ offset + 0 ] = 0x00000000 ; // merged with store below, but also kept unchanged for RC
a [ offset + 1 ] = 0xFFFFFFFF ;
a [ offset + 2 ] = v2 ;
a [ offset + 3 ] = 0x42424242 ;
a [ offset + 4 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 5 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 6 ] = 0xAB110129 ;
a [ offset + 7 ] = 0xCD360183 ;
a [ offset + 8 ] = 0xEF890173 ;
a [ offset + 9 ] = 0x01560124 ;
a [ offset + 10 ] = ( int ) ( v1 > > 0 ) ;
2024-06-07 06:16:03 +00:00
a [ offset + 11 ] = ( int ) ( v1 > > 32 ) ; // Stores to +11 and +12 can be merged also on big-endian
2024-04-24 06:44:14 +00:00
a [ offset + 12 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 13 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 14 ] = v2 ;
a [ offset + 15 ] = v2 ;
a [ offset + 16 ] = 0xEFEFEFEF ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test202RBE ( int [ ] a , int offset , long v1 , int v2 ) {
a [ offset + 0 ] = 0x00000000 ;
a [ offset + 1 ] = 0xFFFFFFFF ;
a [ offset + 2 ] = v2 ;
a [ offset + 3 ] = 0x42424242 ;
a [ offset + 4 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 5 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 6 ] = 0xAB110129 ;
a [ offset + 7 ] = 0xCD360183 ;
a [ offset + 8 ] = 0xEF890173 ;
a [ offset + 9 ] = 0x01560124 ;
a [ offset + 10 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 11 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 12 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 13 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 14 ] = v2 ;
a [ offset + 15 ] = v2 ;
a [ offset + 16 ] = 0xEFEFEFEF ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 10 " ,
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 4 " } , // Stores of constants can be merged
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 6 " , // 5 (+1 that goes into RC)
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 6 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test202aBE ( int [ ] a , int offset , long v1 , int v2 ) {
a [ offset + 0 ] = 0x00000000 ; // merged with store below, but also kept unchanged for RC
a [ offset + 1 ] = 0xFFFFFFFF ;
a [ offset + 2 ] = v2 ;
a [ offset + 3 ] = 0x42424242 ;
a [ offset + 4 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 5 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 6 ] = 0xAB110129 ;
a [ offset + 7 ] = 0xCD360183 ;
a [ offset + 8 ] = 0xEF890173 ;
a [ offset + 9 ] = 0x01560124 ;
a [ offset + 10 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 11 ] = ( int ) ( v1 > > 0 ) ; // Stores to +11 and +12 can be merged also on little-endian
a [ offset + 12 ] = ( int ) ( v1 > > 32 ) ;
a [ offset + 13 ] = ( int ) ( v1 > > 0 ) ;
a [ offset + 14 ] = v2 ;
a [ offset + 15 ] = v2 ;
a [ offset + 16 ] = 0xEFEFEFEF ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
@DontCompile
static Object [ ] test300R ( int [ ] a ) {
a [ 2 ] = 42 ;
a [ 3 ] = 42 ;
a [ 4 ] = 42 ;
a [ 5 ] = 42 ;
int x = a [ 3 ] ; // dependent load
return new Object [ ] { a , new int [ ] { x } } ;
}
@Test
@IR ( counts = { IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } )
static Object [ ] test300a ( int [ ] a ) {
a [ 2 ] = 42 ;
a [ 3 ] = 42 ;
a [ 4 ] = 42 ;
a [ 5 ] = 42 ;
int x = a [ 3 ] ; // dependent load
return new Object [ ] { a , new int [ ] { x } } ;
}
@DontCompile
static Object [ ] test400R ( int [ ] a ) {
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 1 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 2 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 3 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 4 , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 6 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 7 , ( byte ) 0xde ) ;
return new Object [ ] { a } ;
}
@Test
2024-11-05 11:46:40 +00:00
// All constants are known, and AddI can be converted to AddL safely, hence the stores can be merged.
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
2024-04-24 06:44:14 +00:00
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
2024-11-05 11:46:40 +00:00
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test400a ( int [ ] a ) {
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 1 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 2 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 3 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 4 , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 6 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , UNSAFE . ARRAY_INT_BASE_OFFSET + 7 , ( byte ) 0xde ) ;
return new Object [ ] { a } ;
}
@DontCompile
// The 500-series has all the same code, but is executed with different inputs:
// 500a: never violate a RangeCheck -> expect will always merge stores
// 501a: randomly violate RangeCheck, also during warmup -> never merge stores
// 502a: during warmup never violate RangeCheck -> compile once with merged stores
// but then after warmup violate RangeCheck -> recompile without merged stores
static Object [ ] test500R ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 48 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 56 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // for RangeCheck trap
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // expect merged
2024-06-07 06:16:03 +00:00
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " big-endian " , " true " } )
2024-04-24 06:44:14 +00:00
static Object [ ] test500a ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 48 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 56 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " , // No optimization because of too many RangeChecks
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test501a ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 48 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 56 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " , // No optimization because of too many RangeChecks
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test502a ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 48 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 56 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
2024-04-24 19:06:46 +00:00
2024-06-07 06:16:03 +00:00
@DontCompile
// The 500-series has all the same code, but is executed with different inputs:
// 500a: never violate a RangeCheck -> expect will always merge stores
// 501a: randomly violate RangeCheck, also during warmup -> never merge stores
// 502a: during warmup never violate RangeCheck -> compile once with merged stores
// but then after warmup violate RangeCheck -> recompile without merged stores
static Object [ ] test500RBE ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 56 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 48 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 40 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 32 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " , // for RangeCheck trap
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // expect merged
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test500aBE ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 56 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 48 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 40 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 32 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 7 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test501aBE ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 56 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 48 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 40 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 32 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 7 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test502aBE ( byte [ ] a , int offset , long v ) {
int idx = 0 ;
try {
a [ offset + 0 ] = ( byte ) ( v > > 56 ) ;
idx = 1 ;
a [ offset + 1 ] = ( byte ) ( v > > 48 ) ;
idx = 2 ;
a [ offset + 2 ] = ( byte ) ( v > > 40 ) ;
idx = 3 ;
a [ offset + 3 ] = ( byte ) ( v > > 32 ) ;
idx = 4 ;
a [ offset + 4 ] = ( byte ) ( v > > 24 ) ;
idx = 5 ;
a [ offset + 5 ] = ( byte ) ( v > > 16 ) ;
idx = 6 ;
a [ offset + 6 ] = ( byte ) ( v > > 8 ) ;
idx = 7 ;
a [ offset + 7 ] = ( byte ) ( v > > 0 ) ;
idx = 8 ;
} catch ( ArrayIndexOutOfBoundsException _ ) { }
return new Object [ ] { a , new int [ ] { idx } } ;
}
2024-04-24 19:06:46 +00:00
@DontCompile
static Object [ ] test600R ( byte [ ] aB , int [ ] aI , int i ) {
Object a = null ;
long base = 0 ;
if ( i % 2 = = 0 ) {
a = aB ;
base = UNSAFE . ARRAY_BYTE_BASE_OFFSET ;
} else {
a = aI ;
base = UNSAFE . ARRAY_INT_BASE_OFFSET ;
}
UNSAFE . putByte ( a , base + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + 1 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + 2 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + 3 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + 4 , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , base + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + 6 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + 7 , ( byte ) 0xde ) ;
return new Object [ ] { aB , aI } ;
}
@Test
2024-11-05 11:46:40 +00:00
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } )
2024-04-24 19:06:46 +00:00
static Object [ ] test600a ( byte [ ] aB , int [ ] aI , int i ) {
Object a = null ;
long base = 0 ;
if ( i % 2 = = 0 ) {
a = aB ;
base = UNSAFE . ARRAY_BYTE_BASE_OFFSET ;
} else {
a = aI ;
base = UNSAFE . ARRAY_INT_BASE_OFFSET ;
}
2024-11-05 11:46:40 +00:00
// Array type is unknown, i.e. bottom[]. But all AddI can be safely converted to AddL -> safe to merge.
2024-04-24 19:06:46 +00:00
UNSAFE . putByte ( a , base + 0 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + 1 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + 2 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + 3 , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + 4 , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , base + 5 , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + 6 , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + 7 , ( byte ) 0xde ) ;
return new Object [ ] { aB , aI } ;
}
2024-11-05 11:46:40 +00:00
@DontCompile
static Object [ ] test601R ( byte [ ] aB , int [ ] aI , int i , int offset1 ) {
Object a = null ;
long base = 0 ;
if ( i % 2 = = 0 ) {
a = aB ;
base = UNSAFE . ARRAY_BYTE_BASE_OFFSET ;
} else {
a = aI ;
base = UNSAFE . ARRAY_INT_BASE_OFFSET ;
}
UNSAFE . putByte ( a , base + ( offset1 + 0 ) , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + ( offset1 + 1 ) , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + ( offset1 + 2 ) , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + ( offset1 + 3 ) , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + ( offset1 + 4 ) , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , base + ( offset1 + 5 ) , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + ( offset1 + 6 ) , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + ( offset1 + 7 ) , ( byte ) 0xde ) ;
return new Object [ ] { aB , aI } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 8 " , // nothing merged
IRNode . STORE_C_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " 64-bit " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " bottom \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " } , // all merged
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " 32-bit " , " true " } )
static Object [ ] test601a ( byte [ ] aB , int [ ] aI , int i , int offset1 ) {
Object a = null ;
long base = 0 ;
if ( i % 2 = = 0 ) {
a = aB ;
base = UNSAFE . ARRAY_BYTE_BASE_OFFSET ;
} else {
a = aI ;
base = UNSAFE . ARRAY_INT_BASE_OFFSET ;
}
// Array type is unknown, i.e. bottom[]. Hence we do not know the element size of the array.
// Thus, on 64-bits systems merging is not safe, there could be overflows.
UNSAFE . putByte ( a , base + ( offset1 + 0 ) , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + ( offset1 + 1 ) , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + ( offset1 + 2 ) , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + ( offset1 + 3 ) , ( byte ) 0xba ) ;
UNSAFE . putByte ( a , base + ( offset1 + 4 ) , ( byte ) 0xef ) ;
UNSAFE . putByte ( a , base + ( offset1 + 5 ) , ( byte ) 0xbe ) ;
UNSAFE . putByte ( a , base + ( offset1 + 6 ) , ( byte ) 0xad ) ;
UNSAFE . putByte ( a , base + ( offset1 + 7 ) , ( byte ) 0xde ) ;
return new Object [ ] { aB , aI } ;
}
2024-04-30 16:15:07 +00:00
@DontCompile
static Object [ ] test700R ( int [ ] a , long v1 ) {
a [ 0 ] = ( int ) ( v1 > > - 1 ) ;
a [ 1 ] = ( int ) ( v1 > > - 2 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_C_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " ,
IRNode . STORE_L_OF_CLASS , " int \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test700a ( int [ ] a , long v1 ) {
// Negative shift: cannot optimize
a [ 0 ] = ( int ) ( v1 > > - 1 ) ;
a [ 1 ] = ( int ) ( v1 > > - 2 ) ;
return new Object [ ] { a } ;
}
2024-06-07 06:16:03 +00:00
@DontCompile
static Object [ ] test800R ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 6 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } )
static Object [ ] test800a ( byte [ ] a , int offset , long v ) {
// Merge attempts begin at the lowest store in the Memory chain.
// Candidates are found following the chain. The list is trimmed to a
// power of 2 length by removing higher stores.
a [ offset + 0 ] = ( byte ) ( v > > 0 ) ; // Removed from candidate list
a [ offset + 1 ] = ( byte ) ( v > > 8 ) ; // Removed from candidate list
a [ offset + 2 ] = ( byte ) ( v > > 16 ) ; // The 4 following stores are on the candidate list.
a [ offset + 3 ] = ( byte ) ( v > > 24 ) ; // The current logic does not merge them
a [ offset + 4 ] = ( byte ) ( v > > 32 ) ; // since it would require shifting the input.
a [ offset + 5 ] = ( byte ) ( v > > 40 ) ;
return new Object [ ] { a } ;
}
@DontCompile
static Object [ ] test800RBE ( byte [ ] a , int offset , long v ) {
a [ offset + 0 ] = ( byte ) ( v > > 40 ) ;
a [ offset + 1 ] = ( byte ) ( v > > 32 ) ;
a [ offset + 2 ] = ( byte ) ( v > > 24 ) ;
a [ offset + 3 ] = ( byte ) ( v > > 16 ) ;
a [ offset + 4 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 0 ) ;
return new Object [ ] { a } ;
}
@Test
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 6 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIfPlatform = { " little-endian " , " true " } )
@IR ( counts = { IRNode . STORE_B_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 2 " ,
IRNode . STORE_C_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " ,
IRNode . STORE_I_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 1 " ,
IRNode . STORE_L_OF_CLASS , " byte \\ \\ [int:>=0] \\ \\ (java/lang/Cloneable,java/io/Serializable \\ \\ ) " , " 0 " } ,
applyIf = { " UseUnalignedAccesses " , " true " } ,
applyIfPlatform = { " big-endian " , " true " } )
static Object [ ] test800aBE ( byte [ ] a , int offset , long v ) {
// Merge attempts begin at the lowest store in the Memory chain.
// Candidates are found following the chain. The list is trimmed to a
// power of 2 length by removing higher stores.
a [ offset + 0 ] = ( byte ) ( v > > 40 ) ; // Removed from candidate list
a [ offset + 1 ] = ( byte ) ( v > > 32 ) ; // Removed from candidate list
a [ offset + 2 ] = ( byte ) ( v > > 24 ) ; // The 4 following stores are on the candidate list
a [ offset + 3 ] = ( byte ) ( v > > 16 ) ; // and they are successfully merged on big endian platforms.
a [ offset + 4 ] = ( byte ) ( v > > 8 ) ;
a [ offset + 5 ] = ( byte ) ( v > > 0 ) ;
return new Object [ ] { a } ;
}
2024-04-24 06:44:14 +00:00
}