Skip to content

Commit

Permalink
Fix issue #26417 = Incorrect caching of loop variable
Browse files Browse the repository at this point in the history
Contributes to issue  #7147 - JIT: Loop hoisting re-ordering exceptions
Added the Test case for Issue 26417
Updated comments
  • Loading branch information
briansull committed Sep 30, 2019
1 parent 8e4050a commit 7e69c25
Show file tree
Hide file tree
Showing 3 changed files with 115 additions and 29 deletions.
85 changes: 56 additions & 29 deletions src/jit/optimizer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -6795,8 +6795,20 @@ void Compiler::optHoistLoopBlocks(unsigned loopNum, ArrayStack<BasicBlock*>* blo

bool IsTreeVNInvariant(GenTree* tree)
{
return m_compiler->optVNIsLoopInvariant(tree->gtVNPair.GetLiberal(), m_loopNum,
&m_hoistContext->m_curLoopVnInvariantCache);
ValueNum vn = tree->gtVNPair.GetLiberal();
if (m_compiler->vnStore->IsVNConstant(vn))
{
// It is unsafe to allow a GT_CLS_VAR that has been assigned a constant.
// The logic in optVNIsLoopInvariant would consider it to be loop-invariant, even
// if the assignment of the constant to the GT_CLS_VAR was inside the loop.
//
if (tree->OperIs(GT_CLS_VAR))
{
return false;
}
}

return m_compiler->optVNIsLoopInvariant(vn, m_loopNum, &m_hoistContext->m_curLoopVnInvariantCache);
}

public:
Expand Down Expand Up @@ -7007,49 +7019,64 @@ void Compiler::optHoistLoopBlocks(unsigned loopNum, ArrayStack<BasicBlock*>* blo
}
}

// Check if we need to set 'm_beforeSideEffect' to false.
// If we encounter a tree with a call in it
// or if we see an assignment to global we set it to false.
// Next check if we need to set 'm_beforeSideEffect' to false.
//
// If we are already set to false then we can skip these checks
// If we have already set it to false then we can skip these checks
//
if (m_beforeSideEffect)
{
// For this purpose, we only care about memory side effects. We assume that expressions will
// be hoisted so that they are evaluated in the same order as they would have been in the loop,
// and therefore throw exceptions in the same order. (So we don't use GTF_GLOBALLY_VISIBLE_SIDE_EFFECTS
// here, since that includes exceptions.)
if (tree->IsCall())
// If the current tree isn't invariant (and thus prevents hoisting) then we have to check for any side
// effects
if (!treeIsInvariant)
{
// If it's a call, it must be a helper call that does not mutate the heap.
// Further, if it may run a cctor, it must be labeled as "Hoistable"
// (meaning it won't run a cctor because the class is not precise-init).
GenTreeCall* call = tree->AsCall();
if (call->gtCallType != CT_HELPER)
// Here we have a tree that is not invariant and we are cannot hoist
// If it has any side effects, then we need to set m_beforeSideEffect to false.
if ((tree->gtFlags & GTF_SIDE_EFFECT) != 0)
{
m_beforeSideEffect = false;
}
else
}
else // this tree is invariant and can still be part of a tree that gets hoisted
{
// Check if we need to set m_beforeSideEffect here:
//
// For this purpose, we only care about memory side effects. We assume that expressions will
// be hoisted so that they are evaluated in the same order as they would have been in the loop,
// and therefore throw exceptions in the same order.
// (So we don't use GTF_GLOBALLY_VISIBLE_SIDE_EFFECTS here, since that includes exceptions.)
//
if (tree->IsCall())
{
CorInfoHelpFunc helpFunc = eeGetHelperNum(call->gtCallMethHnd);
if (s_helperCallProperties.MutatesHeap(helpFunc))
// If it's a call, it must be a helper call that does not mutate the heap.
// Further, if it may run a cctor, it must be labeled as "Hoistable"
// (meaning it won't run a cctor because the class is not precise-init).
GenTreeCall* call = tree->AsCall();
if (call->gtCallType != CT_HELPER)
{
m_beforeSideEffect = false;
}
else if (s_helperCallProperties.MayRunCctor(helpFunc) &&
(call->gtFlags & GTF_CALL_HOISTABLE) == 0)
else
{
m_beforeSideEffect = false;
CorInfoHelpFunc helpFunc = eeGetHelperNum(call->gtCallMethHnd);
if (s_helperCallProperties.MutatesHeap(helpFunc))
{
m_beforeSideEffect = false;
}
else if (s_helperCallProperties.MayRunCctor(helpFunc) &&
(call->gtFlags & GTF_CALL_HOISTABLE) == 0)
{
m_beforeSideEffect = false;
}
}
}
}
else if (tree->OperIs(GT_ASG))
{
// If the LHS of the assignment has a global reference, then assume it's a global side effect.
GenTree* lhs = tree->gtOp.gtOp1;
if (lhs->gtFlags & GTF_GLOB_REF)
else if (tree->OperIs(GT_ASG))
{
m_beforeSideEffect = false;
// If the LHS of the assignment has a global reference, then assume it's a global side effect.
GenTree* lhs = tree->gtOp.gtOp1;
if (lhs->gtFlags & GTF_GLOB_REF)
{
m_beforeSideEffect = false;
}
}
}
}
Expand Down
47 changes: 47 additions & 0 deletions tests/src/JIT/Regression/JitBlue/GitHub_26417/GitHub_26417.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
// 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.

using System;
using System.Runtime.CompilerServices;

class GitHub_26417
{
static int _a;

[MethodImplAttribute(MethodImplOptions.NoInlining)]
static void MyWriteLine(int v)
{
Console.WriteLine(v);
if (v == 0)
{
throw new Exception();
}
}

[MethodImplAttribute(MethodImplOptions.NoInlining)]
static void Test()
{
_a = 1;

while (_a == 1)
{
MyWriteLine(_a);
_a = 0;
}
}

static int Main()
{
int result = 100;
try {
Test();
}
catch (Exception)
{
Console.WriteLine("FAILED");
result = -1;
}
return result;
}
}
12 changes: 12 additions & 0 deletions tests/src/JIT/Regression/JitBlue/GitHub_26417/GitHub_26417.csproj
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
</PropertyGroup>
<PropertyGroup>
<DebugType>None</DebugType>
<Optimize>True</Optimize>
</PropertyGroup>
<ItemGroup>
<Compile Include="$(MSBuildProjectName).cs" />
</ItemGroup>
</Project>

0 comments on commit 7e69c25

Please sign in to comment.