Browse Source

Merge branch 'roslyn-430'

pull/2749/head
Siegfried Pammer 3 years ago
parent
commit
1afcec2af1
  1. 26
      ICSharpCode.Decompiler.Tests/Helpers/RoslynToolset.cs
  2. 6
      ICSharpCode.Decompiler.Tests/Helpers/Tester.cs
  3. 21
      ICSharpCode.Decompiler/CSharp/RecordDecompiler.cs
  4. 16
      ICSharpCode.Decompiler/IL/ControlFlow/AsyncAwaitDecompiler.cs
  5. 17
      ICSharpCode.Decompiler/IL/Instructions/PatternMatching.cs
  6. 2
      packages.props

26
ICSharpCode.Decompiler.Tests/Helpers/RoslynToolset.cs

@ -48,7 +48,7 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
this.baseDir = baseDir; this.baseDir = baseDir;
} }
protected async Task FetchPackage(string packageName, string version, string outputPath) protected async Task FetchPackage(string packageName, string version, string sourcePath, string outputPath)
{ {
ILogger logger = NullLogger.Instance; ILogger logger = NullLogger.Instance;
CancellationToken cancellationToken = CancellationToken.None; CancellationToken cancellationToken = CancellationToken.None;
@ -65,8 +65,8 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
using PackageArchiveReader packageReader = new PackageArchiveReader(packageStream); using PackageArchiveReader packageReader = new PackageArchiveReader(packageStream);
NuspecReader nuspecReader = await packageReader.GetNuspecReaderAsync(cancellationToken).ConfigureAwait(false); NuspecReader nuspecReader = await packageReader.GetNuspecReaderAsync(cancellationToken).ConfigureAwait(false);
var files = await packageReader.GetFilesAsync(cancellationToken).ConfigureAwait(false); var files = (await packageReader.GetFilesAsync(cancellationToken).ConfigureAwait(false)).ToArray();
files = files.Where(f => f.StartsWith("tools", StringComparison.OrdinalIgnoreCase)); files = files.Where(f => f.StartsWith(sourcePath, StringComparison.OrdinalIgnoreCase)).ToArray();
await packageReader.CopyFilesAsync(outputPath, files, await packageReader.CopyFilesAsync(outputPath, files,
(sourceFile, targetPath, fileStream) => { (sourceFile, targetPath, fileStream) => {
fileStream.CopyToFile(targetPath); fileStream.CopyToFile(targetPath);
@ -87,15 +87,15 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
{ {
} }
public async Task Fetch(string version) public async Task Fetch(string version, string packageName = "Microsoft.Net.Compilers.Toolset", string sourcePath = "tasks/net472")
{ {
string path = Path.Combine(baseDir, version, "tools"); string path = Path.Combine(baseDir, version, sourcePath);
if (!Directory.Exists(path)) if (!Directory.Exists(path))
{ {
await FetchPackage("Microsoft.Net.Compilers", version, Path.Combine(baseDir, version)).ConfigureAwait(false); await FetchPackage(packageName, version, sourcePath, Path.Combine(baseDir, version)).ConfigureAwait(false);
} }
installedCompilers.Add(version, path); installedCompilers.Add(SanitizeVersion(version), path);
} }
public string GetCSharpCompiler(string version) public string GetCSharpCompiler(string version)
@ -110,10 +110,18 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
string GetCompiler(string compiler, string version) string GetCompiler(string compiler, string version)
{ {
if (installedCompilers.TryGetValue(version, out var path)) if (installedCompilers.TryGetValue(SanitizeVersion(version), out var path))
return Path.Combine(path, compiler); return Path.Combine(path, compiler);
throw new NotSupportedException($"Cannot find {compiler} {version}, please add it to the initialization."); throw new NotSupportedException($"Cannot find {compiler} {version}, please add it to the initialization.");
} }
internal static string SanitizeVersion(string version)
{
int index = version.IndexOf("-");
if (index > 0)
return version.Remove(index);
return version;
}
} }
class VsWhereToolset : AbstractToolset class VsWhereToolset : AbstractToolset
@ -130,7 +138,7 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
string path = Path.Combine(baseDir, "tools"); string path = Path.Combine(baseDir, "tools");
if (!Directory.Exists(path)) if (!Directory.Exists(path))
{ {
await FetchPackage("vswhere", "2.8.4", baseDir).ConfigureAwait(false); await FetchPackage("vswhere", "2.8.4", "tools", baseDir).ConfigureAwait(false);
} }
vswherePath = Path.Combine(path, "vswhere.exe"); vswherePath = Path.Combine(path, "vswhere.exe");
} }

6
ICSharpCode.Decompiler.Tests/Helpers/Tester.cs

@ -113,8 +113,8 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
internal static async Task Initialize() internal static async Task Initialize()
{ {
await roslynToolset.Fetch("1.3.2").ConfigureAwait(false); await roslynToolset.Fetch("1.3.2", "Microsoft.Net.Compilers", "tools").ConfigureAwait(false);
await roslynToolset.Fetch("2.10.0").ConfigureAwait(false); await roslynToolset.Fetch("2.10.0", "Microsoft.Net.Compilers", "tools").ConfigureAwait(false);
await roslynToolset.Fetch("3.11.0").ConfigureAwait(false); await roslynToolset.Fetch("3.11.0").ConfigureAwait(false);
await roslynToolset.Fetch(roslynLatestVersion).ConfigureAwait(false); await roslynToolset.Fetch(roslynLatestVersion).ConfigureAwait(false);
@ -432,7 +432,7 @@ namespace ICSharpCode.Decompiler.Tests.Helpers
if (roslynVersion != "legacy") if (roslynVersion != "legacy")
{ {
otherOptions += "/shared "; otherOptions += "/shared ";
if (!targetNet40 && Version.Parse(roslynVersion).Major > 2) if (!targetNet40 && Version.Parse(RoslynToolset.SanitizeVersion(roslynVersion)).Major > 2)
{ {
if (flags.HasFlag(CompilerOptions.NullableEnable)) if (flags.HasFlag(CompilerOptions.NullableEnable))
otherOptions += "/nullable+ "; otherOptions += "/nullable+ ";

21
ICSharpCode.Decompiler/CSharp/RecordDecompiler.cs

@ -354,6 +354,17 @@ namespace ICSharpCode.Decompiler.CSharp
&& IsRecordType(method.Parameters[0].Type); && IsRecordType(method.Parameters[0].Type);
} }
private bool IsAllowedAttribute(IAttribute attribute)
{
switch (attribute.AttributeType.ReflectionName)
{
case "System.Runtime.CompilerServices.CompilerGeneratedAttribute":
return true;
default:
return false;
}
}
private bool IsGeneratedCopyConstructor(IMethod method) private bool IsGeneratedCopyConstructor(IMethod method)
{ {
/* /*
@ -362,7 +373,7 @@ namespace ICSharpCode.Decompiler.CSharp
leave IL_0000 (nop) leave IL_0000 (nop)
*/ */
Debug.Assert(method.IsConstructor && method.Parameters.Count == 1); Debug.Assert(method.IsConstructor && method.Parameters.Count == 1);
if (method.GetAttributes().Any() || method.GetReturnTypeAttributes().Any()) if (method.GetAttributes().Any(attr => !IsAllowedAttribute(attr)) || method.GetReturnTypeAttributes().Any())
return false; return false;
if (method.Accessibility != Accessibility.Protected && (!isSealed || method.Accessibility != Accessibility.Private)) if (method.Accessibility != Accessibility.Protected && (!isSealed || method.Accessibility != Accessibility.Private))
return false; return false;
@ -461,7 +472,7 @@ namespace ICSharpCode.Decompiler.CSharp
return false; return false;
if (!isSealed && !method.IsOverridable) if (!isSealed && !method.IsOverridable)
return false; return false;
if (method.GetAttributes().Any() || method.GetReturnTypeAttributes().Any()) if (method.GetAttributes().Any(attr => !IsAllowedAttribute(attr)) || method.GetReturnTypeAttributes().Any())
return false; return false;
if (method.Accessibility != Accessibility.Protected && (!isSealed || method.Accessibility != Accessibility.Private)) if (method.Accessibility != Accessibility.Protected && (!isSealed || method.Accessibility != Accessibility.Private))
return false; return false;
@ -637,7 +648,7 @@ namespace ICSharpCode.Decompiler.CSharp
return false; return false;
if (method.IsSealed) if (method.IsSealed)
return false; return false;
if (method.GetAttributes().Any() || method.GetReturnTypeAttributes().Any()) if (method.GetAttributes().Any(attr => !IsAllowedAttribute(attr)) || method.GetReturnTypeAttributes().Any())
return false; return false;
var body = DecompileBody(method); var body = DecompileBody(method);
if (body == null) if (body == null)
@ -711,7 +722,7 @@ namespace ICSharpCode.Decompiler.CSharp
return false; return false;
if (!isSealed && !method.IsOverridable) if (!isSealed && !method.IsOverridable)
return false; return false;
if (method.GetAttributes().Any() || method.GetReturnTypeAttributes().Any()) if (method.GetAttributes().Any(attr => !IsAllowedAttribute(attr)) || method.GetReturnTypeAttributes().Any())
return false; return false;
if (orderedMembers == null) if (orderedMembers == null)
return false; return false;
@ -901,7 +912,7 @@ namespace ICSharpCode.Decompiler.CSharp
return false; return false;
if (!method.IsOverride || method.IsSealed) if (!method.IsOverride || method.IsSealed)
return false; return false;
if (method.GetAttributes().Any() || method.GetReturnTypeAttributes().Any()) if (method.GetAttributes().Any(attr => !IsAllowedAttribute(attr)) || method.GetReturnTypeAttributes().Any())
return false; return false;
if (orderedMembers == null) if (orderedMembers == null)
return false; return false;

16
ICSharpCode.Decompiler/IL/ControlFlow/AsyncAwaitDecompiler.cs

@ -725,6 +725,7 @@ namespace ICSharpCode.Decompiler.IL.ControlFlow
throw new SymbolicAnalysisFailedException(); throw new SymbolicAnalysisFailedException();
finalStateKnown = true; finalStateKnown = true;
pos++; pos++;
if (pos + 2 == block.Instructions.Count && block.MatchIfAtEndOfBlock(out var condition, out var trueInst, out var falseInst)) if (pos + 2 == block.Instructions.Count && block.MatchIfAtEndOfBlock(out var condition, out var trueInst, out var falseInst))
{ {
if (MatchDisposeCombinedTokens(blockContainer, condition, trueInst, falseInst, blocksAnalyzed, out var setResultAndExitBlock)) if (MatchDisposeCombinedTokens(blockContainer, condition, trueInst, falseInst, blocksAnalyzed, out var setResultAndExitBlock))
@ -807,7 +808,7 @@ namespace ICSharpCode.Decompiler.IL.ControlFlow
// https://github.com/dotnet/roslyn/pull/39735 hoisted local cleanup // https://github.com/dotnet/roslyn/pull/39735 hoisted local cleanup
if (!target.MatchLdThis()) if (!target.MatchLdThis())
throw new SymbolicAnalysisFailedException(); throw new SymbolicAnalysisFailedException();
if (!(value.MatchLdNull() || value is DefaultValue)) if (!value.MatchDefaultOrNullOrZero())
throw new SymbolicAnalysisFailedException(); throw new SymbolicAnalysisFailedException();
pos++; pos++;
} }
@ -1441,13 +1442,20 @@ namespace ICSharpCode.Decompiler.IL.ControlFlow
pos--; pos--;
} }
if (pos < 0 || !block.Instructions[pos].MatchStFld(out var target, out var field, out yieldValue)) if (pos < 0 || !MatchCurrentAssignment(block.Instructions[pos], out yieldValue))
return false;
block.Instructions.RemoveRange(pos, block.Instructions.Count - pos);
return true;
}
bool MatchCurrentAssignment(ILInstruction inst, out ILInstruction value)
{
if (!inst.MatchStFld(out var target, out var field, out value))
return false; return false;
if (!StackSlotValue(target).MatchLdThis()) if (!StackSlotValue(target).MatchLdThis())
return false; return false;
// TODO: check that we are accessing the current field (compare with get_Current) // TODO: check that we are accessing the current field (compare with get_Current)
block.Instructions.RemoveRange(pos, block.Instructions.Count - pos);
return true; return true;
} }
#endregion #endregion

17
ICSharpCode.Decompiler/IL/Instructions/PatternMatching.cs

@ -553,6 +553,23 @@ namespace ICSharpCode.Decompiler.IL
return false; return false;
} }
public bool MatchDefaultOrNullOrZero()
{
switch (this)
{
case LdNull _:
case LdcF4 { Value: 0 }:
case LdcF8 { Value: 0 }:
case LdcI4 { Value: 0 }:
case LdcI8 { Value: 0 }:
case DefaultValue _:
return true;
default:
return false;
}
}
/// <summary> /// <summary>
/// If this instruction is a conversion of the specified kind, return its argument. /// If this instruction is a conversion of the specified kind, return its argument.
/// Otherwise, return the instruction itself. /// Otherwise, return the instruction itself.

2
packages.props

@ -14,7 +14,7 @@
<!-- Microsoft.NETCore.ILDAsm --> <!-- Microsoft.NETCore.ILDAsm -->
<ILDAsmVersion>6.0.0</ILDAsmVersion> <ILDAsmVersion>6.0.0</ILDAsmVersion>
<!-- Microsoft.CodeAnalysis.* --> <!-- Microsoft.CodeAnalysis.* -->
<RoslynVersion>4.1.0</RoslynVersion> <RoslynVersion>4.3.0-2.final</RoslynVersion>
<MonoCecilVersion>0.11.4</MonoCecilVersion> <MonoCecilVersion>0.11.4</MonoCecilVersion>
<AvalonEditVersion>6.1.3.50</AvalonEditVersion> <AvalonEditVersion>6.1.3.50</AvalonEditVersion>
<WpfStylesToolboxVersion>2.7.4</WpfStylesToolboxVersion> <WpfStylesToolboxVersion>2.7.4</WpfStylesToolboxVersion>

Loading…
Cancel
Save