diff options
| author | Simon Pilgrim <llvm-dev@redking.me.uk> | 2018-03-13 17:17:15 +0000 |
|---|---|---|
| committer | Simon Pilgrim <llvm-dev@redking.me.uk> | 2018-03-13 17:17:15 +0000 |
| commit | 9855b3938093eb4a3663474a42a1a694327816c0 (patch) | |
| tree | 3045d2a84bc5a492b0afa9277b958b8c0708173d | |
| parent | 8afcd938ed9cc51dd7db79171d1b06d1e479b3dc (diff) | |
| download | bcm5719-llvm-9855b3938093eb4a3663474a42a1a694327816c0.tar.gz bcm5719-llvm-9855b3938093eb4a3663474a42a1a694327816c0.zip | |
[DAGCombine] visitREM - Don't assume that one divrem isn't driving another
Under some circumstances the divrems won't have been combined together before getting to this code.
So replace the assertion with a if() guard to not expand to X-((X/C)*C) to give the other combine chance to happen.
Reduced from OSS-Fuzz #6883
https://bugs.chromium.org/p/oss-fuzz/issues/detail?id=6883
llvm-svn: 327424
| -rw-r--r-- | llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp | 6 | ||||
| -rw-r--r-- | llvm/test/CodeGen/X86/combine-srem.ll | 30 |
2 files changed, 33 insertions, 3 deletions
diff --git a/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp b/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp index d0e2f918278..48cfe4f57a1 100644 --- a/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp +++ b/llvm/lib/CodeGen/SelectionDAG/DAGCombiner.cpp @@ -3068,9 +3068,9 @@ SDValue DAGCombiner::visitREM(SDNode *N) { SDValue Div = DAG.getNode(DivOpcode, DL, VT, N0, N1); AddToWorklist(Div.getNode()); SDValue OptimizedDiv = combine(Div.getNode()); - if (OptimizedDiv.getNode() && OptimizedDiv.getNode() != Div.getNode()) { - assert((OptimizedDiv.getOpcode() != ISD::UDIVREM) && - (OptimizedDiv.getOpcode() != ISD::SDIVREM)); + if (OptimizedDiv.getNode() && OptimizedDiv.getNode() != Div.getNode() && + OptimizedDiv.getOpcode() != ISD::UDIVREM && + OptimizedDiv.getOpcode() != ISD::SDIVREM) { SDValue Mul = DAG.getNode(ISD::MUL, DL, VT, OptimizedDiv, N1); SDValue Sub = DAG.getNode(ISD::SUB, DL, VT, N0, Mul); AddToWorklist(Mul.getNode()); diff --git a/llvm/test/CodeGen/X86/combine-srem.ll b/llvm/test/CodeGen/X86/combine-srem.ll index 0d9b35dbc42..a3ea5efd095 100644 --- a/llvm/test/CodeGen/X86/combine-srem.ll +++ b/llvm/test/CodeGen/X86/combine-srem.ll @@ -131,3 +131,33 @@ define <4 x i32> @combine_vec_srem_by_pos1(<4 x i32> %x) { %2 = srem <4 x i32> %1, <i32 1, i32 4, i32 8, i32 16> ret <4 x i32> %2 } + +; OSS-Fuzz #6883 +; https://bugs.chromium.org/p/oss-fuzz/issues/detail?id=6883 +define i32 @ossfuzz6883() { +; CHECK-LABEL: ossfuzz6883: +; CHECK: # %bb.0: +; CHECK-NEXT: movl (%rax), %ecx +; CHECK-NEXT: movl %ecx, %eax +; CHECK-NEXT: cltd +; CHECK-NEXT: idivl %ecx +; CHECK-NEXT: movl %edx, %esi +; CHECK-NEXT: movl $1, %edi +; CHECK-NEXT: cltd +; CHECK-NEXT: idivl %edi +; CHECK-NEXT: movl %edx, %edi +; CHECK-NEXT: xorl %edx, %edx +; CHECK-NEXT: movl %ecx, %eax +; CHECK-NEXT: divl %edi +; CHECK-NEXT: andl %esi, %eax +; CHECK-NEXT: retq + %B17 = or i32 0, 2147483647 + %L6 = load i32, i32* undef + %B11 = sdiv i32 %L6, %L6 + %B13 = udiv i32 %B17, %B17 + %B14 = srem i32 %B11, %B13 + %B16 = srem i32 %L6, %L6 + %B10 = udiv i32 %L6, %B14 + %B6 = and i32 %B16, %B10 + ret i32 %B6 +} |

