Hi,
The current implementation of fussion predicates misses some common
fussion cases on zen and more recent cores.  I added knobs for
individual conditionals we test.

 1) I split checks for fusing ALU with conditional operands when the ALU
 has memory operand.  This seems to be supported by zen3+ and by
 tigerlake and coperlake (according to Agner Fog's manual)

 2) znver4 and 5 supports fussion of ALU and conditional even if ALU has
    memory and immediate operands.
    This seems to be relatively important enabling 25% more fusions on
    gcc bootstrap.

 3) no CPU supports fusing when ALU contains IP relative memory
    references.  I added separate knob so we do not forger about this if
    this gets supoorted later.

The patch does not solve the limitation of sched that fuse pairs must be
adjacent on imput and the first operation must be signle-set.  Fixing
single-set is easy (I have separate patch for this), for non-adjacent
pairs we need bigger surgery.

To verify what CPU really does I made simpe test script.

jh@ryzen3:~> cat fuse-test.c
#ifdef IPRELATIVE
        int b;
        const int z = 0;
        const int o = 1;
#endif
        int
main()
{
        int a = 1000000000;
#ifndef IPRELATIVE
        int b;
        int z = 0;
        int o = 1;
#endif
        asm volatile ("\n"
".L1234:\n"
#ifndef FUSE
        "nop\n"
#endif
        "subl   %3, %0\n"

#ifdef CMP
        "movl %0, %1\n"
        "cmpl     %2, %1\n"
#endif
#ifdef TEST
        "movl %0, %1\n"
        "test %1, %1\n"
#endif

#ifndef FUSE
        "nop\n"
#endif
        "jne    .L1234":"=a"(a),
#if (defined(MEM) && !defined (CMP)) || defined (MEMIMM)
        "=m"(b)
#else
        "=r"(b)
#endif
        :
#ifdef MEM
        "m"(z),
        "m"(o),
#else
        "i"(0),
        "i"(1),
#endif
        "0"(a)
                );
}
jh@ryzen3:~> cat fuse-test.sh 
EVENT=ex_ret_fused_instr
#EVENT=ex_ret_fus_brnch_inst
dotest()
{
gcc -O2  fuse-test.c $* -o fuse-cmp-imm-mem-nofuse
perf stat -e $EVENT ./fuse-cmp-imm-mem-nofuse  2>&1 | grep $EVENT
gcc -O2 fuse-test.c -DFUSE $* -o fuse-cmp-imm-mem-fuse 
perf stat  -e $EVENT ./fuse-cmp-imm-mem-fuse 2>&1 | grep $EVENT
}

echo ALU with immediate
dotest 
echo ALU with memory
dotest -D MEM
echo ALU with IP relative memory
dotest -D MEM -D IPRELATIVE
echo CMP with immediate
dotest -D CMP
echo CMP with memory
dotest -D CMP -D MEM
echo CMP with memory and immediate
dotest -D CMP -D MEMIMM
echo CMP with IP relative memory
dotest -D CMP -D MEM -D IPRELATIVE
echo TEST
dotest -D TEST

On zen5 I get:
ALU with immediate
            20,345      ex_ret_fused_instr:u                                    
              
     1,000,020,278      ex_ret_fused_instr:u                                    
              
ALU with memory
            20,367      ex_ret_fused_instr:u                                    
              
     1,000,020,290      ex_ret_fused_instr:u                                    
              
ALU with IP relative memory
            20,395      ex_ret_fused_instr:u                                    
              
            20,403      ex_ret_fused_instr:u                                    
              
CMP with immediate
            20,369      ex_ret_fused_instr:u                                    
              
     1,000,020,301      ex_ret_fused_instr:u                                    
              
CMP with memory
            20,314      ex_ret_fused_instr:u                                    
              
     1,000,020,341      ex_ret_fused_instr:u                                    
              
CMP with memory and immediate
            20,372      ex_ret_fused_instr:u                                    
              
     1,000,020,266      ex_ret_fused_instr:u                                    
              
CMP with IP relative memory
            20,382      ex_ret_fused_instr:u                                    
              
            20,369      ex_ret_fused_instr:u                                    
              
TEST
            20,346      ex_ret_fused_instr:u                                    
              
     1,000,020,301      ex_ret_fused_instr:u                                    
              

IP relative memory seems to not be documented.

On zen3/4 I get:

ALU with immediate
            20,263      ex_ret_fused_instr:u                                    
              
     1,000,020,051      ex_ret_fused_instr:u                                    
              
ALU with memory
            20,255      ex_ret_fused_instr:u                                    
              
     1,000,020,056      ex_ret_fused_instr:u                                    
              
ALU with IP relative memory
            20,253      ex_ret_fused_instr:u                                    
              
            20,266      ex_ret_fused_instr:u                                    
              
CMP with immediate
            20,264      ex_ret_fused_instr:u                                    
              
     1,000,020,052      ex_ret_fused_instr:u                                    
              
CMP with memory
            20,253      ex_ret_fused_instr:u                                    
              
     1,000,019,794      ex_ret_fused_instr:u                                    
              
CMP with memory and immediate
            20,260      ex_ret_fused_instr:u                                    
              
            20,264      ex_ret_fused_instr:u                                    
              
CMP with IP relative memory
            20,258      ex_ret_fused_instr:u                                    
              
            20,256      ex_ret_fused_instr:u                                    
              
TEST
            20,261      ex_ret_fused_instr:u                                    
              
     1,000,020,048      ex_ret_fused_instr:u                                    
              

zen1 and 2 gets:

ALU with immediate
            21,610      ex_ret_fus_brnch_inst:u                                 
              
            21,697      ex_ret_fus_brnch_inst:u                                 
              
ALU with memory
            21,479      ex_ret_fus_brnch_inst:u                                 
              
            21,747      ex_ret_fus_brnch_inst:u                                 
              
ALU with IP relative memory
            21,623      ex_ret_fus_brnch_inst:u                                 
              
            21,684      ex_ret_fus_brnch_inst:u                                 
              
CMP with immediate
            21,708      ex_ret_fus_brnch_inst:u                                 
              
     1,000,021,288      ex_ret_fus_brnch_inst:u                                 
              
CMP with memory
            21,689      ex_ret_fus_brnch_inst:u                                 
              
     1,000,004,270      ex_ret_fus_brnch_inst:u                                 
              
CMP with memory and immediate
            21,604      ex_ret_fus_brnch_inst:u                                 
              
            21,671      ex_ret_fus_brnch_inst:u                                 
              
CMP with IP relative memory
            21,589      ex_ret_fus_brnch_inst:u                                 
              
            21,602      ex_ret_fus_brnch_inst:u                                 
              
TEST
            21,600      ex_ret_fus_brnch_inst:u                                 
              
     1,000,021,233      ex_ret_fus_brnch_inst:u                                 
              

I tested the patch on zen3 and zen5 and spec2k17 and it seems neutral, however
the number of fussion does go up.

Bootstrapped/regtested x86_64-linux, I plan to commit it tomorrow.

Honza

gcc/ChangeLog:

        * config/i386/i386.h (TARGET_FUSE_ALU_AND_BRANCH_MEM): New macro.
        (TARGET_FUSE_ALU_AND_BRANCH_MEM_IMM): New macro.
        (TARGET_FUSE_ALU_AND_BRANCH_RIP_RELATIVE): New macro.
        * config/i386/x86-tune-sched.cc (ix86_fuse_mov_alu_p): Support
        non-single-set.
        (ix86_macro_fusion_pair_p): Allow ALU which only clobbers;
        be more careful about immediates; check TARGET_FUSE_ALU_AND_BRANCH_MEM,
        TARGET_FUSE_ALU_AND_BRANCH_MEM_IMM, 
TARGET_FUSE_ALU_AND_BRANCH_RIP_RELATIVE;
        verify that we never use unsigned checks with inc/dec.
        * config/i386/x86-tune.def (X86_TUNE_FUSE_ALU_AND_BRANCH): New tune.
        (X86_TUNE_FUSE_ALU_AND_BRANCH_MEM): New tune.
        (X86_TUNE_FUSE_ALU_AND_BRANCH_MEM_IMM): New tune.
        (X86_TUNE_FUSE_ALU_AND_BRANCH_RIP_RELATIVE): New tune.

diff --git a/gcc/config/i386/i386.h b/gcc/config/i386/i386.h
index 56ef11a58bb..2696bfb3a81 100644
--- a/gcc/config/i386/i386.h
+++ b/gcc/config/i386/i386.h
@@ -432,6 +432,12 @@ extern unsigned char ix86_tune_features[X86_TUNE_LAST];
        ix86_tune_features[X86_TUNE_FUSE_CMP_AND_BRANCH_SOFLAGS]
 #define TARGET_FUSE_ALU_AND_BRANCH \
        ix86_tune_features[X86_TUNE_FUSE_ALU_AND_BRANCH]
+#define TARGET_FUSE_ALU_AND_BRANCH_MEM \
+       ix86_tune_features[X86_TUNE_FUSE_ALU_AND_BRANCH_MEM]
+#define TARGET_FUSE_ALU_AND_BRANCH_MEM_IMM \
+       ix86_tune_features[X86_TUNE_FUSE_ALU_AND_BRANCH_MEM_IMM]
+#define TARGET_FUSE_ALU_AND_BRANCH_RIP_RELATIVE\
+       ix86_tune_features[X86_TUNE_FUSE_ALU_AND_BRANCH_RIP_RELATIVE]
 #define TARGET_FUSE_MOV_AND_ALU \
        ix86_tune_features[X86_TUNE_FUSE_MOV_AND_ALU]
 #define TARGET_OPT_AGU ix86_tune_features[X86_TUNE_OPT_AGU]
diff --git a/gcc/config/i386/x86-tune-sched.cc 
b/gcc/config/i386/x86-tune-sched.cc
index a59d7c229c2..a51764e078c 100644
--- a/gcc/config/i386/x86-tune-sched.cc
+++ b/gcc/config/i386/x86-tune-sched.cc
@@ -32,6 +32,7 @@ along with GCC; see the file COPYING3.  If not see
 #include "insn-attr.h"
 #include "insn-opinit.h"
 #include "recog.h"
+#include "tm-constrs.h"
 
 /* Return the maximum number of instructions a cpu can issue.  */
 
@@ -571,6 +572,9 @@ ix86_macro_fusion_p ()
   return TARGET_FUSE_CMP_AND_BRANCH;
 }
 
+/* Check whether MOV is a reg-reg move and ALU is an
+   ALU operation that allows macro-op fusion.  */
+
 static bool
 ix86_fuse_mov_alu_p (rtx_insn *mov, rtx_insn *alu)
 {
@@ -593,6 +597,16 @@ ix86_fuse_mov_alu_p (rtx_insn *mov, rtx_insn *alu)
   rtx set2 = XVECEXP (PATTERN (alu), 0, 0);
   if (GET_CODE (set2) != SET)
     return false;
+  /* If this is instruction setting both compare and normal
+     register, the first set always sets flags, while
+     second set writes to the output operan.  Pick
+     the second set.  */
+  if (GET_CODE (SET_SRC (set2)) == COMPARE)
+    {
+      set2 = XVECEXP (PATTERN (alu), 0, 1);
+      if (GET_CODE (set2) != SET)
+       return false;
+    }
   /* Match one of:
      ADD ADC AND XOR OR SUB SBB INC DEC NOT SAL SHL SHR SAR
      We also may add insn attribute to handle some of sporadic
@@ -635,10 +649,11 @@ ix86_macro_fusion_pair_p (rtx_insn *condgen, rtx_insn 
*condjmp)
   if (TARGET_FUSE_MOV_AND_ALU
       && ix86_fuse_mov_alu_p (condgen, condjmp))
     return true;
-  rtx src, dest;
+  rtx src, imm = NULL_RTX;
   enum rtx_code ccode;
   rtx compare_set = NULL_RTX, test_if, cond;
   rtx alu_set = NULL_RTX, addr = NULL_RTX;
+  rtx alu_clobber = NULL_RTX;
   enum attr_type condgen_type;
 
   if (!any_condjump_p (condjmp))
@@ -664,6 +679,9 @@ ix86_macro_fusion_pair_p (rtx_insn *condgen, rtx_insn 
*condjmp)
       alu_set = XVECEXP (PATTERN (condgen), 0, 1);
       goto handle_stack_protect_test;
     }
+  /* ??? zen5 can fuse cmp, test, sub, add, inc, dec, or, and xor.
+     Cores can not fuse or and xor which will pass the test below
+     since type is ALU.  */
   else if (condgen_type != TYPE_TEST
           && condgen_type != TYPE_ICMP
           && condgen_type != TYPE_INCDEC
@@ -687,6 +705,11 @@ ix86_macro_fusion_pair_p (rtx_insn *condgen, rtx_insn 
*condjmp)
            else
              alu_set = XVECEXP (pat, 0, i);
          }
+       /* We also possibly generated ALU instruction only to set
+          flags.  In this case there will be clobber.  */
+       else if (GET_CODE (XVECEXP (pat, 0, i)) == CLOBBER
+           && GENERAL_REG_P (XEXP (XVECEXP (pat, 0, i), 0)))
+         alu_clobber = XVECEXP (pat, 0, i);
     }
   if (compare_set == NULL_RTX)
     return false;
@@ -694,19 +717,30 @@ ix86_macro_fusion_pair_p (rtx_insn *condgen, rtx_insn 
*condjmp)
   if (GET_CODE (src) != COMPARE)
     return false;
 
-  /* Macro-fusion for cmp/test MEM-IMM + conditional jmp is not
-     supported.  */
-  if ((MEM_P (XEXP (src, 0)) && CONST_INT_P (XEXP (src, 1)))
-      || (MEM_P (XEXP (src, 1)) && CONST_INT_P (XEXP (src, 0))))
-    return false;
-
-  /* No fusion for RIP-relative address.  */
+  /* Check for memory operand.  */
   if (MEM_P (XEXP (src, 0)))
     addr = XEXP (XEXP (src, 0), 0);
   else if (MEM_P (XEXP (src, 1)))
     addr = XEXP (XEXP (src, 1), 0);
+  /* Some CPUs, i.e. tigerlake and cooperlake does not fuse
+     ALU with memory operand.  */
+  if (addr && !TARGET_FUSE_ALU_AND_BRANCH_MEM)
+    return false;
+  if (CONST_INT_P (XEXP (src, 0)))
+    imm = XEXP (src, 0);
+  else if (CONST_INT_P (XEXP (src, 1)))
+    imm = XEXP (src, 1);
+  /* Check that the instruction really has immediate.
+     In particular compare with 0 is done using test with no immediate.  */
+  if (imm && !get_attr_length_immediate (condgen))
+    imm = NULL;
+  /* Macro-fusion for cmp/test MEM-IMM + conditional jmp is not
+     supported.   */
+  if (addr && imm && !TARGET_FUSE_ALU_AND_BRANCH_MEM_IMM)
+    return false;
 
-  if (addr)
+  /* No fusion for RIP-relative address.   */
+  if (addr && !TARGET_FUSE_ALU_AND_BRANCH_RIP_RELATIVE)
     {
       ix86_address parts;
       int ok = ix86_decompose_address (addr, &parts);
@@ -715,6 +749,12 @@ ix86_macro_fusion_pair_p (rtx_insn *condgen, rtx_insn 
*condjmp)
       if (ix86_rip_relative_addr_p (&parts))
        return false;
     }
+  /* Znver5 supports fussion fusion with their reg/reg, reg/imm and
+     reg/mem forms. They are also supported when the instruction has an
+     immediate and displacement that meets the criteria of 4 byte displacement
+     and 2 byte immediate or the case of 2 byte displacement and 4 byte
+     immediate.  We do not know the displacement size, so we ignore this
+     limitation.  */
 
  handle_stack_protect_test:
   test_if = SET_SRC (pc_set (condjmp));
@@ -730,19 +770,19 @@ ix86_macro_fusion_pair_p (rtx_insn *condgen, rtx_insn 
*condjmp)
     return true;
 
   /* The following is the case that macro-fusion for alu + jmp.  */
-  if (!TARGET_FUSE_ALU_AND_BRANCH || !alu_set)
+  if (!TARGET_FUSE_ALU_AND_BRANCH || (!alu_set && !alu_clobber))
     return false;
 
   /* No fusion for alu op with memory destination operand.  */
-  dest = SET_DEST (alu_set);
-  if (MEM_P (dest))
+  if (alu_set && MEM_P (SET_DEST (alu_set)))
     return false;
 
+
   /* Macro-fusion for inc/dec + unsigned conditional jump is not
-     supported.  */
-  if (condgen_type == TYPE_INCDEC
-      && (ccode == GEU || ccode == GTU || ccode == LEU || ccode == LTU))
-    return false;
+     supported on some CPUs while supported on others (znver5 and core_avx512).
+     We however never generate it, so we do not need a specific tune for it.  
*/
+  gcc_checking_assert (!(condgen_type == TYPE_INCDEC
+                      && (ccode == GEU || ccode == GTU || ccode == LEU || 
ccode == LTU)));
 
   return true;
 }
diff --git a/gcc/config/i386/x86-tune.def b/gcc/config/i386/x86-tune.def
index 0bdad7234a6..b6e39f642e8 100644
--- a/gcc/config/i386/x86-tune.def
+++ b/gcc/config/i386/x86-tune.def
@@ -149,7 +149,7 @@ DEF_TUNE (X86_TUNE_FUSE_CMP_AND_BRANCH_SOFLAGS, 
"fuse_cmp_and_branch_soflags",
    TODO: znver5 supports fusing with SUB, ADD, INC, DEC, OR, AND,
    There is also limitation for immediate and displacement supported.  */
 DEF_TUNE (X86_TUNE_FUSE_ALU_AND_BRANCH, "fuse_alu_and_branch",
-         m_SANDYBRIDGE | m_CORE_AVX2 | m_ZHAOXIN | m_GENERIC | m_ZNVER5)
+         m_SANDYBRIDGE | m_CORE_AVX2 | m_ZHAOXIN | m_GENERIC | m_ZNVER3 | 
m_ZNVER4 | m_ZNVER5)
 
 /* X86_TUNE_FUSE_MOV_AND_ALU: mov and alu in case mov is reg-reg mov
    and the destination is used by alu.  alu must be one of
@@ -157,6 +157,22 @@ DEF_TUNE (X86_TUNE_FUSE_ALU_AND_BRANCH, 
"fuse_alu_and_branch",
 DEF_TUNE (X86_TUNE_FUSE_MOV_AND_ALU, "fuse_mov_and_alu",
         m_ZNVER5 | m_GRANITERAPIDS | m_GRANITERAPIDS_D)
 
+/* X86_TUNE_FUSE_AND_BRANCH_MEM: Fuse alu with a subsequent conditional
+   jump instruction when alu contains memory operand.
+   TODO: Not suported by TIGERLAKE and COPERLAKE, so m_CORE_AVX2 is wrong.  */
+DEF_TUNE (X86_TUNE_FUSE_ALU_AND_BRANCH_MEM, "fuse_alu_and_branch_mem",
+         m_SANDYBRIDGE | m_CORE_AVX2 | m_ZHAOXIN | m_GENERIC | m_ZNVER3 | 
m_ZNVER4 | m_ZNVER5)
+
+/* X86_TUNE_FUSE_AND_BRANCH_MEM_IMM: Fuse alu with a subsequent conditional
+   jump instruction when alu contains both immediate and displacement.  */
+DEF_TUNE (X86_TUNE_FUSE_ALU_AND_BRANCH_MEM_IMM, "fuse_alu_and_branch_mem_imm",
+         m_GENERIC | m_ZNVER4 | m_ZNVER5)
+
+/* X86_TUNE_FUSE_AND_BRANCH_RIP_RELATIVE: Fuse alu with a subsequent
+   conditional jump instruction when alu contains IP relative address.  */
+DEF_TUNE (X86_TUNE_FUSE_ALU_AND_BRANCH_RIP_RELATIVE,
+         "fuse_alu_and_branch_rip_relative", 0)
+
 /*****************************************************************************/
 /* Function prologue, epilogue and function calling sequences.               */
 /*****************************************************************************/

Reply via email to