Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

WI #2039 Remove PERFORM recursivity check from CFG #2100

Merged
merged 1 commit into from
Jan 4, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,3 +1,2 @@
--- Diagnostics ---
Line 28[15,29] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM pararec', recursive instruction is ' perform pararec2'.
Line 29[15,57] <37, Warning, General> - Warning: Unreachable code detected
Original file line number Diff line number Diff line change
@@ -1,5 +1,2 @@
--- Diagnostics ---
Line 31[15,29] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM pararec', recursive instruction is ' perform pararec2'.
Line 32[15,57] <37, Warning, General> - Warning: Unreachable code detected
Line 57[15,64] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM pararec4', recursive instruction is ' perform pararec4 varying n from 1 by 1 until n > 5'.
Line 57[15,64] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM pararec4', recursive instruction is ' perform pararec3 test after varying n from 1 by 1 until n'.
2 changes: 0 additions & 2 deletions TypeCobol.Analysis.Test/BasicCfgInstrs/PerformIdentity.diag

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

4 changes: 0 additions & 4 deletions TypeCobol.Analysis.Test/BasicCfgInstrs/SG102A.diag
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,6 @@ Line 242[13,46] <37, Warning, General> - Warning: "end-if" is missing
Line 250[12,37] <37, Warning, General> - Warning: "end-if" is missing
Line 266[12,43] <37, Warning, General> - Warning: "end-if" is missing
Line 267[12,42] <37, Warning, General> - Warning: "end-if" is missing
Line 269[41,66] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM WRITE-LINE', recursive instruction is ' PERFORM WRT-LN'.
Line 269[41,66] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM WRITE-LINE', recursive instruction is ' PERFORM WRT-LN 2 TIMES'.
Line 269[41,66] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM WRITE-LINE', recursive instruction is ' PERFORM WRT-LN'.
Line 269[41,66] <37, Warning, General> - Warning: A recursive loop has been encountered while analyzing 'PERFORM WRITE-LINE', recursive instruction is ' PERFORM WRT-LN'.
Line 276[12,43] <37, Warning, General> - Warning: "end-if" is missing
Line 277[12,38] <37, Warning, General> - Warning: "end-if" is missing
Line 293[12,16] <37, Warning, General> - Warning: ALTER should not be used
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,17 +63,6 @@ internal bool IsAfterIterativeGroup
set;
}

/// <summary>
/// To detect recursivity on PERFORM Procedure calls.
/// This is a bit array of GroupIndex encountered during the workflow
/// call of other PERFORM.
/// </summary>
internal BitArray RecursivityGroupSet
{
get;
set;
}

/// <summary>
/// Constructor.
/// </summary>
Expand Down
40 changes: 0 additions & 40 deletions TypeCobol.Analysis/Cfg/ControlFlowGraphBuilder.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using TypeCobol.Analysis.Graph;
Expand Down Expand Up @@ -1070,16 +1069,6 @@ void StoreSentences()
{
isPerform = true; //To avoid a second dynamic cast

//Is there a recursion in the graph ?
if (group.RecursivityGroupSet.Get(group0.GroupIndex) && !group0.HasFlag(BasicBlock<Node, D>.Flags.Recursive))
{
//Flag group and store recursive perform
group0.SetFlag(BasicBlock<Node, D>.Flags.Recursive, true);
Node offendingInstruction = group0.Instructions.Last.Value;
System.Diagnostics.Debug.Assert(offendingInstruction != null);
this.Cfg.AddRecursivePerform(p, offendingInstruction);
}

var clonedGroup0 = clonedPerforms
.Select(t => t.Item3)
.SingleOrDefault(g => g.GroupIndex == group0.GroupIndex);
Expand Down Expand Up @@ -1115,9 +1104,6 @@ void StoreSentences()
clonedGroup.Group = new LinkedList<BasicBlock<Node, D>>();
clonedGroup.TerminalBlocks = null;

group.RecursivityGroupSet.Set(clonedGroup.GroupIndex, true);
clonedGroup.RecursivityGroupSet = new BitArray(group.RecursivityGroupSet);

var originalPerform = this.CurrentProgramCfgBuilder.PendingPERFORMProcedures
.Single(t => t.Item3 == block);
clonedPerforms.Add(new Tuple<PerformProcedure, SectionNode, BasicBlockForNodeGroup>(originalPerform.Item1, originalPerform.Item2, clonedGroup));
Expand Down Expand Up @@ -1192,8 +1178,6 @@ private void ResolvePendingPERFORMProcedures()
//First pass: resolve targets of PERFORMs, some new groups may be created during this
foreach (var item in this.CurrentProgramCfgBuilder.PendingPERFORMProcedures)
{
item.Item3.RecursivityGroupSet = new BitArray(GroupCounter + 1);
item.Item3.RecursivityGroupSet.Set(item.Item3.GroupIndex, true);
ResolvePendingPERFORMProcedure(item, clonedPerforms);
}

Expand All @@ -1211,13 +1195,11 @@ private void ResolvePendingPERFORMProcedures()
{
BasicBlockForNodeGroup group = item.Item3;
groupOrder[group.GroupIndex] = group;
group.RecursivityGroupSet = null;
}
foreach (var item in clonedPerforms)
{
BasicBlockForNodeGroup group = item.Item3;
groupOrder[group.GroupIndex] = group;
group.RecursivityGroupSet = null;
}

//Extend groups according to the current building mode
Expand Down Expand Up @@ -1439,28 +1421,6 @@ private void CreateOptionalDiagnostics()
AddDiagnostic(d);
}
}

//RecursiveBlockOnPerformProcedure
if (_compilerOptions.CheckRecursivePerforms.IsActive && this.CurrentProgramCfgBuilder.Cfg.RecursivePerforms != null)
{
foreach (var recursivePerform in this.CurrentProgramCfgBuilder.Cfg.RecursivePerforms)
{
var perform = recursivePerform.Key;
SymbolReference procedureReference = perform.CodeElement.Procedure;
SymbolReference throughProcedureReference = perform.CodeElement.ThroughProcedure;
foreach (var offendingInstruction in recursivePerform.Value)
{
string performTarget = throughProcedureReference != null
? $"{procedureReference} THRU {throughProcedureReference}"
: procedureReference.ToString();
System.Diagnostics.Debug.Assert(offendingInstruction.CodeElement != null);
string offendingStatement = offendingInstruction.CodeElement.SourceText;
Diagnostic d = new Diagnostic(_compilerOptions.CheckRecursivePerforms.GetMessageCode(), perform.CodeElement.Position(),
string.Format(Resource.RecursiveBlockOnPerformProcedure, performTarget, offendingStatement));
AddDiagnostic(d);
}
}
}
}

/// <summary>
Expand Down
3 changes: 1 addition & 2 deletions TypeCobol.Analysis/Graph/BasicBlock.cs
Original file line number Diff line number Diff line change
Expand Up @@ -81,8 +81,7 @@ public enum Flags : uint
Declaratives = 0x01 << 2, //Flag if this basic block is inside a declaratives section.
Start = 0x01 << 3, //Flag for a start node.
End = 0x01 << 4, //Flag for a end node.
GroupGrafted = 0x01 << 5, //Flag a Grafted Group.
Recursive = 0x01 << 6 //Flag a recursive block
GroupGrafted = 0x01 << 5 //Flag a Grafted Group.
}

/// <summary>
Expand Down
28 changes: 0 additions & 28 deletions TypeCobol.Analysis/Graph/ControlFlowGraph.cs
Original file line number Diff line number Diff line change
Expand Up @@ -134,12 +134,6 @@ public BasicBlock<N, D> RootBlock
/// </summary>
public Dictionary<PerformProcedure, Tuple<N, N>> WrongOrderPerformThrus { get; private set; }

/// <summary>
/// All recursive PERFORMs found in the program. Null if no such perform found.
/// Key is the recursive PERFORM statement, value is the list of all detected instructions that lead to recursion.
/// </summary>
public Dictionary<PerformProcedure, List<N>> RecursivePerforms { get; private set; }

/// <summary>
/// All unreachable blocks of this graph. The property may be null but not empty.
/// </summary>
Expand Down Expand Up @@ -243,28 +237,6 @@ internal void AddWrongOrderPerformThru(PerformProcedure performThru, N procedure
WrongOrderPerformThrus.Add(performThru, new Tuple<N, N>(procedure, throughProcedure));
}

/// <summary>
/// Register a recursive jump for a perform statement.
/// </summary>
/// <param name="perform">Perform node</param>
/// <param name="recursiveJump">Recursive jump node</param>
internal void AddRecursivePerform(PerformProcedure perform, N recursiveJump)
{
if (RecursivePerforms == null)
{
RecursivePerforms = new Dictionary<PerformProcedure, List<N>>();
}

if (RecursivePerforms.TryGetValue(perform, out var nodes))
{
nodes.Add(recursiveJump);
}
else
{
RecursivePerforms.Add(perform, new List<N>() { recursiveJump });
}
}

/// <summary>
/// Register one unreachable block of this graph
/// </summary>
Expand Down
11 changes: 1 addition & 10 deletions TypeCobol.Analysis/Resource.Designer.cs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 0 additions & 3 deletions TypeCobol.Analysis/Resource.resx
Original file line number Diff line number Diff line change
Expand Up @@ -126,7 +126,4 @@
<data name="BasicBlockGroupGoesBeyondTheLimit" xml:space="preserve">
<value>Statement '{0}' located at line {1}, column {2} prevents this PERFORM statement to reach its exit.</value>
</data>
<data name="RecursiveBlockOnPerformProcedure" xml:space="preserve">
<value>A recursive loop has been encountered while analyzing 'PERFORM {0}', recursive instruction is '{1}'.</value>
</data>
</root>
6 changes: 0 additions & 6 deletions TypeCobol/Compiler/Directives/TypeCobolOptions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -81,11 +81,6 @@ public bool UseEuroInformationLegacyReplacingSyntax
/// </summary>
public TypeCobolCheckOption CheckPerformThruOrder { get; set; }

/// <summary>
/// Check that perform statement are not recursive, requires CFG.
/// </summary>
public TypeCobolCheckOption CheckRecursivePerforms { get; set; }

public TypeCobolOptions(TypeCobolConfiguration config)
{
HaltOnMissingCopy = config.HaltOnMissingCopyFilePath != null;
Expand All @@ -104,7 +99,6 @@ public TypeCobolOptions(TypeCobolConfiguration config)
CheckEndProgram = config.CheckEndProgram;
CheckPerformPrematureExits = config.CheckPerformPrematureExits;
CheckPerformThruOrder = config.CheckPerformThruOrder;
CheckRecursivePerforms = config.CheckRecursivePerforms;
}

public TypeCobolOptions()
Expand Down
4 changes: 0 additions & 4 deletions TypeCobol/Tools/Options-Config/TypeCobolConfiguration.cs
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,6 @@ public class TypeCobolConfiguration : ITypeCobolCheckOptions
public TypeCobolCheckOption CheckEndProgram { get; set; }
public TypeCobolCheckOption CheckPerformPrematureExits { get; set; }
public TypeCobolCheckOption CheckPerformThruOrder { get; set; }
public TypeCobolCheckOption CheckRecursivePerforms { get; set; }

public List<string> Copies = new List<string>();
public List<string> Dependencies = new List<string>();
Expand Down Expand Up @@ -246,7 +245,6 @@ public interface ITypeCobolCheckOptions
TypeCobolCheckOption CheckEndProgram { get; set; }
TypeCobolCheckOption CheckPerformPrematureExits { get; set; }
TypeCobolCheckOption CheckPerformThruOrder { get; set; }
TypeCobolCheckOption CheckRecursivePerforms { get; set; }
}

public static class TypeCobolCheckOptionsInitializer
Expand All @@ -257,7 +255,6 @@ public static void SetDefaultValues(ITypeCobolCheckOptions checkOptions)
checkOptions.CheckEndProgram = new TypeCobolCheckOption(Severity.Error);
checkOptions.CheckPerformPrematureExits = new TypeCobolCheckOption(Severity.Warning);
checkOptions.CheckPerformThruOrder = new TypeCobolCheckOption(Severity.Warning);
checkOptions.CheckRecursivePerforms = new TypeCobolCheckOption(Severity.Warning);
}
}

Expand Down Expand Up @@ -293,7 +290,6 @@ public static OptionSet GetCommonTypeCobolOptions(TypeCobolConfiguration typeCob
{ "diag.cep|diagnostic.checkEndProgram=", "Indicate level of check end program: warning, error, info, ignore.", v => typeCobolConfig.CheckEndProgram = TypeCobolCheckOption.Parse(v) },
{ "diag.cppe|diagnostic.checkPerformPrematureExits=", "Indicate level of check perform premature exits: warning, error, info, ignore.", v => typeCobolConfig.CheckPerformPrematureExits = TypeCobolCheckOption.Parse(v) },
{ "diag.cpto|diagnostic.checkPerformThruOrder=", "Indicate level of check perform thru procedure order: warning, error, info, ignore.", v => typeCobolConfig.CheckPerformThruOrder = TypeCobolCheckOption.Parse(v) },
{ "diag.crp|diagnostic.checkRecursivePerforms=", "Indicate level of check recursive performs: warning, error, info, ignore.", v => typeCobolConfig.CheckRecursivePerforms = TypeCobolCheckOption.Parse(v) },
{ "log|logfilepath=", "{PATH} to TypeCobol.CLI.log log file", v => typeCobolConfig.LogFile = Path.Combine(v, TypeCobolConfiguration.DefaultLogFileName)},
{ "cfg|cfgbuild=", "CFG build option, recognized values are: None/0, Standard/1, Extended/2, WithDfa/3.", v => typeCobolConfig.RawCfgBuildingMode = v },
{ "cob|cobol", "Indicate that it's a pure Cobol85 input file.", v => typeCobolConfig.IsCobolLanguage = true },
Expand Down