From 77c4adbeb5add29a2429bfa1059794b8ea343dad Mon Sep 17 00:00:00 2001 From: Boris Ulasevich Date: Fri, 24 Jul 2020 09:07:10 -0400 Subject: [PATCH] 8249189: AARCH64: more L2I conversions can be skipped Reviewed-by: aph --- src/hotspot/cpu/aarch64/aarch64.ad | 77 +++++++++++++++++++++++++++ src/hotspot/cpu/aarch64/aarch64_ad.m4 | 40 ++++++++------ 2 files changed, 101 insertions(+), 16 deletions(-) diff --git a/src/hotspot/cpu/aarch64/aarch64.ad b/src/hotspot/cpu/aarch64/aarch64.ad index bcffa622a0f..840cbdd80fb 100644 --- a/src/hotspot/cpu/aarch64/aarch64.ad +++ b/src/hotspot/cpu/aarch64/aarch64.ad @@ -4053,6 +4053,18 @@ operand immI_bitmask() interface(CONST_INTER); %} +operand immL_positive_bitmaskI() +%{ + predicate((n->get_long() != 0) + && ((julong)n->get_long() < 0x80000000ULL) + && is_power_of_2(n->get_long() + 1)); + match(ConL); + + op_cost(0); + format %{ %} + interface(CONST_INTER); +%} + // Scale values for scaled offset addressing modes (up to long but not quad) operand immIScale() %{ @@ -12149,6 +12161,50 @@ instruct ubfizL(iRegLNoSp dst, iRegL src, immI lshift, immL_bitmask mask) ins_pipe(ialu_reg_shift); %} +// This pattern is automatically generated from aarch64_ad.m4 +// DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE + +// We can use ubfiz when masking by a positive number and then left shifting the result. +// We know that the mask is positive because immI_bitmask guarantees it. +instruct ubfizwIConvI2L(iRegLNoSp dst, iRegIorL2I src, immI lshift, immI_bitmask mask) +%{ + match(Set dst (ConvI2L (LShiftI (AndI src mask) lshift))); + predicate((exact_log2(n->in(1)->in(1)->in(2)->get_int() + 1) + (n->in(1)->in(2)->get_int() & 31)) <= 31); + + ins_cost(INSN_COST); + format %{ "ubfizw $dst, $src, $lshift, $mask" %} + ins_encode %{ + int lshift = $lshift$$constant & 31; + intptr_t mask = $mask$$constant; + int width = exact_log2(mask+1); + __ ubfizw(as_Register($dst$$reg), + as_Register($src$$reg), lshift, width); + %} + ins_pipe(ialu_reg_shift); +%} + +// This pattern is automatically generated from aarch64_ad.m4 +// DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE + +// We can use ubfiz when masking by a positive number and then left shifting the result. +// We know that the mask is positive because immL_bitmask guarantees it. +instruct ubfizLConvL2I(iRegINoSp dst, iRegL src, immI lshift, immL_positive_bitmaskI mask) +%{ + match(Set dst (ConvL2I (LShiftL (AndL src mask) lshift))); + predicate((exact_log2_long(n->in(1)->in(1)->in(2)->get_long() + 1) + (n->in(1)->in(2)->get_int() & 63)) <= 31); + + ins_cost(INSN_COST); + format %{ "ubfiz $dst, $src, $lshift, $mask" %} + ins_encode %{ + int lshift = $lshift$$constant & 63; + intptr_t mask = $mask$$constant; + int width = exact_log2_long(mask+1); + __ ubfiz(as_Register($dst$$reg), + as_Register($src$$reg), lshift, width); + %} + ins_pipe(ialu_reg_shift); +%} + // This pattern is automatically generated from aarch64_ad.m4 // DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE @@ -12174,6 +12230,27 @@ instruct ubfizIConvI2L(iRegLNoSp dst, iRegIorL2I src, immI lshift, immI_bitmask // This pattern is automatically generated from aarch64_ad.m4 // DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE +// If there is a convert L to I block between and AndL and a LShiftI, we can also match ubfiz +instruct ubfizLConvL2Ix(iRegINoSp dst, iRegL src, immI lshift, immL_positive_bitmaskI mask) +%{ + match(Set dst (LShiftI (ConvL2I (AndL src mask)) lshift)); + predicate((exact_log2_long(n->in(1)->in(1)->in(2)->get_long() + 1) + (n->in(2)->get_int() & 31)) <= 31); + + ins_cost(INSN_COST); + format %{ "ubfiz $dst, $src, $lshift, $mask" %} + ins_encode %{ + int lshift = $lshift$$constant & 31; + intptr_t mask = $mask$$constant; + int width = exact_log2(mask+1); + __ ubfiz(as_Register($dst$$reg), + as_Register($src$$reg), lshift, width); + %} + ins_pipe(ialu_reg_shift); +%} + +// This pattern is automatically generated from aarch64_ad.m4 +// DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE + // Can skip int2long conversions after AND with small bitmask instruct ubfizIConvI2LAndI(iRegLNoSp dst, iRegI src, immI_bitmask msk) %{ diff --git a/src/hotspot/cpu/aarch64/aarch64_ad.m4 b/src/hotspot/cpu/aarch64/aarch64_ad.m4 index f626424e0f8..5893f451459 100644 --- a/src/hotspot/cpu/aarch64/aarch64_ad.m4 +++ b/src/hotspot/cpu/aarch64/aarch64_ad.m4 @@ -237,39 +237,45 @@ define(`UBFIZ_INSN', `// This pattern is automatically generated from aarch64_ad // We can use ubfiz when masking by a positive number and then left shifting the result. // We know that the mask is positive because imm$1_bitmask guarantees it. -instruct $2$1(iReg$1NoSp dst, iReg$1`'ORL2I($1) src, immI lshift, imm$1_bitmask mask) +instruct $3$1$8(iReg$2NoSp dst, iReg$1`'ORL2I($1) src, immI lshift, $7 mask) %{ - match(Set dst (LShift$1 (And$1 src mask) lshift)); - predicate((exact_log2$5(n->in(1)->in(2)->get_$4() + 1) + (n->in(2)->get_int() & $3)) <= ($3 + 1)); + ifelse($8,, + match(Set dst (LShift$1 (And$1 src mask) lshift));, + match(Set dst ($8 (LShift$1 (And$1 src mask) lshift)));) + ifelse($8,, + predicate(($6(n->in(1)->in(2)->get_$5() + 1) + (n->in(2)->get_int() & $4)) <= ($4 + 1));, + predicate(($6(n->in(1)->in(1)->in(2)->get_$5() + 1) + (n->in(1)->in(2)->get_int() & $4)) <= 31);) ins_cost(INSN_COST); - format %{ "$2 $dst, $src, $lshift, $mask" %} + format %{ "$3 $dst, $src, $lshift, $mask" %} ins_encode %{ - int lshift = $lshift$$constant & $3; + int lshift = $lshift$$constant & $4; intptr_t mask = $mask$$constant; - int width = exact_log2$5(mask+1); - __ $2(as_Register($dst$$reg), + int width = $6(mask+1); + __ $3(as_Register($dst$$reg), as_Register($src$$reg), lshift, width); %} ins_pipe(ialu_reg_shift); %} ') -UBFIZ_INSN(I, ubfizw, 31, int) -UBFIZ_INSN(L, ubfiz, 63, long, _long) +UBFIZ_INSN(I, I, ubfizw, 31, int, exact_log2, immI_bitmask) +UBFIZ_INSN(L, L, ubfiz, 63, long, exact_log2_long, immL_bitmask) +UBFIZ_INSN(I, L, ubfizw, 31, int, exact_log2, immI_bitmask, ConvI2L) +UBFIZ_INSN(L, I, ubfiz, 63, long, exact_log2_long, immL_positive_bitmaskI, ConvL2I) -// This pattern is automatically generated from aarch64_ad.m4 +define(`BFX1_INSN', `// This pattern is automatically generated from aarch64_ad.m4 // DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE -// If there is a convert I to L block between and AndI and a LShiftL, we can also match ubfiz -instruct ubfizIConvI2L(iRegLNoSp dst, iRegIorL2I src, immI lshift, immI_bitmask mask) +// If there is a convert $1 to $2 block between and And$1 and a LShift$2, we can also match ubfiz +instruct ubfiz$1Conv$3$9(iReg$2NoSp dst, iReg$1`'ORL2I($1) src, immI lshift, $8 mask) %{ - match(Set dst (LShiftL (ConvI2L (AndI src mask)) lshift)); - predicate((exact_log2(n->in(1)->in(1)->in(2)->get_int() + 1) + (n->in(2)->get_int() & 63)) <= (63 + 1)); + match(Set dst (LShift$2 (Conv$3 (And$1 src mask)) lshift)); + predicate(($4(n->in(1)->in(1)->in(2)->$5() + 1) + (n->in(2)->get_int() & $6)) <= $7); ins_cost(INSN_COST); format %{ "ubfiz $dst, $src, $lshift, $mask" %} ins_encode %{ - int lshift = $lshift$$constant & 63; + int lshift = $lshift$$constant & $6; intptr_t mask = $mask$$constant; int width = exact_log2(mask+1); __ ubfiz(as_Register($dst$$reg), @@ -277,7 +283,9 @@ instruct ubfizIConvI2L(iRegLNoSp dst, iRegIorL2I src, immI lshift, immI_bitmask %} ins_pipe(ialu_reg_shift); %} - +')dnl +BFX1_INSN(I, L, I2L, exact_log2, get_int, 63, (63 + 1), immI_bitmask) +BFX1_INSN(L, I, L2I, exact_log2_long, get_long, 31, 31, immL_positive_bitmaskI, x) // This pattern is automatically generated from aarch64_ad.m4 // DO NOT EDIT ANYTHING IN THIS SECTION OF THE FILE