diff options
author | Sanjay Patel <spatel@rotateright.com> | 2015-04-28 21:03:22 +0000 |
---|---|---|
committer | Sanjay Patel <spatel@rotateright.com> | 2015-04-28 21:03:22 +0000 |
commit | 2fbc4e5c4942a3a816c2d13913db0ee65990f448 (patch) | |
tree | 64ec8eecd85d44b264ab61fc3a73e32b692fbfff /llvm/lib | |
parent | 659ece9ddbb6bb8a351bfd6d9337e093af2248be (diff) | |
download | bcm5719-llvm-2fbc4e5c4942a3a816c2d13913db0ee65990f448.tar.gz bcm5719-llvm-2fbc4e5c4942a3a816c2d13913db0ee65990f448.zip |
transform fadd chains to increase parallelism
This is a compromise: with this simple patch, we should always handle a chain of exactly 3
operations optimally, but we're not generating the optimal balanced binary tree for a longer
sequence.
In general, this transform will reduce the dependency chain for a sequence of instructions
using N operands from a worst case N-1 dependent operations to N/2 dependent operations.
The optimal balanced binary tree would reduce the chain to log2(N).
The trade-off for not dealing with longer sequences is: (1) we have less complexity in the
compiler, (2) we avoid unknown compile-time blowup calculating a balanced tree, and (3) we
don't need to worry about the increased register pressure required to parallelize longer
sequences. It also seems unlikely that we would ever encounter really long strings of
dependent ops like that in the wild, but I'm not sure how to verify that speculation.
FWIW, I see no perf difference for test-suite running on btver2 (x86-64) with -ffast-math
and this patch.
We can extend this patch to cover other associative operations such as fmul, fmax, fmin,
integer add, integer mul.
This is a partial fix for:
https://llvm.org/bugs/show_bug.cgi?id=17305
and if extended:
https://llvm.org/bugs/show_bug.cgi?id=21768
https://llvm.org/bugs/show_bug.cgi?id=23116
The issue also came up in:
http://reviews.llvm.org/D8941
Differential Revision: http://reviews.llvm.org/D9232
llvm-svn: 236031
Diffstat (limited to 'llvm/lib')
-rw-r--r-- | llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp | 18 |
1 files changed, 18 insertions, 0 deletions
diff --git a/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp b/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp index 629313c7120..66c1356152d 100644 --- a/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp +++ b/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp @@ -7801,6 +7801,24 @@ SDValue DAGCombiner::visitFADD(SDNode *N) { N0.getOperand(0), DAG.getConstantFP(4.0, DL, VT)); } } + + // Canonicalize chains of adds to LHS to simplify the following transform. + if (N0.getOpcode() != ISD::FADD && N1.getOpcode() == ISD::FADD) + return DAG.getNode(ISD::FADD, SDLoc(N), VT, N1, N0); + + // Convert a chain of 3 dependent operations into 2 independent operations + // and 1 dependent operation: + // (fadd N0: (fadd N00: (fadd z, w), N01: y), N1: x) -> + // (fadd N00: (fadd z, w), (fadd N1: x, N01: y)) + if (N0.getOpcode() == ISD::FADD && N0.hasOneUse() && + N1.getOpcode() != ISD::FADD) { + SDValue N00 = N0.getOperand(0); + if (N00.getOpcode() == ISD::FADD) { + SDValue N01 = N0.getOperand(1); + SDValue NewAdd = DAG.getNode(ISD::FADD, SDLoc(N), VT, N1, N01); + return DAG.getNode(ISD::FADD, SDLoc(N), VT, N00, NewAdd); + } + } } // enable-unsafe-fp-math // FADD -> FMA combines: |