From 492a5201fdb26eae5288fd6653945a689a601f3a Mon Sep 17 00:00:00 2001 From: Egor Bogatov Date: Tue, 17 Aug 2021 18:27:11 +0300 Subject: [PATCH] JIT: Don't fold NEG(NEG(X)) to X when NEG(x) is a CSE candidate (#57478) --- src/coreclr/jit/morph.cpp | 12 +++-- .../JIT/opt/perf/doublenegate/GitHub_57470.cs | 50 +++++++++++++++++++ .../opt/perf/doublenegate/GitHub_57470.csproj | 9 ++++ 3 files changed, 67 insertions(+), 4 deletions(-) create mode 100644 src/tests/JIT/opt/perf/doublenegate/GitHub_57470.cs create mode 100644 src/tests/JIT/opt/perf/doublenegate/GitHub_57470.csproj diff --git a/src/coreclr/jit/morph.cpp b/src/coreclr/jit/morph.cpp index a00b4bb2f49c1..3c2bb4423c717 100644 --- a/src/coreclr/jit/morph.cpp +++ b/src/coreclr/jit/morph.cpp @@ -13114,13 +13114,17 @@ GenTree* Compiler::fgMorphSmpOp(GenTree* tree, MorphAddrContext* mac) case GT_NEG: // Remove double negation/not. // Note: this is not a safe tranformation if "tree" is a CSE candidate. - // Consider for example the following expression: NEG(NEG(OP)), where the top-level + // Consider for example the following expression: NEG(NEG(OP)), where any // NEG is a CSE candidate. Were we to morph this to just OP, CSE would fail to find // the original NEG in the statement. - if (op1->OperIs(oper) && opts.OptimizationEnabled() && !gtIsActiveCSE_Candidate(tree)) + if (op1->OperIs(oper) && opts.OptimizationEnabled() && !gtIsActiveCSE_Candidate(tree) && + !gtIsActiveCSE_Candidate(op1)) { - GenTree* child = op1->AsOp()->gtGetOp1(); - return child; + JITDUMP("Remove double negation/not\n") + GenTree* op1op1 = op1->gtGetOp1(); + DEBUG_DESTROY_NODE(tree); + DEBUG_DESTROY_NODE(op1); + return op1op1; } // Distribute negation over simple multiplication/division expressions diff --git a/src/tests/JIT/opt/perf/doublenegate/GitHub_57470.cs b/src/tests/JIT/opt/perf/doublenegate/GitHub_57470.cs new file mode 100644 index 0000000000000..e0dbc3534c177 --- /dev/null +++ b/src/tests/JIT/opt/perf/doublenegate/GitHub_57470.cs @@ -0,0 +1,50 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. +// +// This file is auto-generated. +// Seed: -1 +// +using System; +using System.Runtime.CompilerServices; +public class TestClass8505 +{ + static int s_int32_6 = -5; + static sbyte s_sbyte_8 = 2; + static int s_loopInvariant = 3; + public sbyte LeafMethod8() + { + unchecked + { + return s_sbyte_8 <<= s_int32_6 >>= s_int32_6 ^ (-2 - (s_int32_6 &= -5)) / (-1 * s_int32_6 * (2 ^ -2)) + 77; + } + } + public void Method0() + { + unchecked + { + try + { + } + finally + { + { + int __loopvar1 = s_loopInvariant, __loopSecondaryVar1_0 = 15 - 4; + do + { + } + while (15 % 4 > LeafMethod8() / 15 + 4); + } + { + } + } + return; + } + } + public static int Main(string[] args) + { + TestClass8505 objTestClass8505 = new TestClass8505(); + objTestClass8505.Method0(); + return 100; + } +} diff --git a/src/tests/JIT/opt/perf/doublenegate/GitHub_57470.csproj b/src/tests/JIT/opt/perf/doublenegate/GitHub_57470.csproj new file mode 100644 index 0000000000000..e6cddfa42c24e --- /dev/null +++ b/src/tests/JIT/opt/perf/doublenegate/GitHub_57470.csproj @@ -0,0 +1,9 @@ + + + Exe + None + + + + +