You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
340 lines
12 KiB
340 lines
12 KiB
// Copyright (c) 2016 Daniel Grunwald
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy of this
|
|
// software and associated documentation files (the "Software"), to deal in the Software
|
|
// without restriction, including without limitation the rights to use, copy, modify, merge,
|
|
// publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons
|
|
// to whom the Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in all copies or
|
|
// substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
|
|
// INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
|
// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
|
|
// FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
using System;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Reflection.PortableExecutable;
|
|
using System.Text.RegularExpressions;
|
|
using System.Threading;
|
|
|
|
using ICSharpCode.Decompiler.CSharp;
|
|
using ICSharpCode.Decompiler.CSharp.ProjectDecompiler;
|
|
using ICSharpCode.Decompiler.Metadata;
|
|
using ICSharpCode.Decompiler.Tests.Helpers;
|
|
|
|
using Microsoft.Build.Locator;
|
|
|
|
using NUnit.Framework;
|
|
|
|
namespace ICSharpCode.Decompiler.Tests
|
|
{
|
|
[TestFixture, Parallelizable(ParallelScope.All)]
|
|
public class RoundtripAssembly
|
|
{
|
|
public static readonly string TestDir = Path.GetFullPath(Path.Combine(Tester.TestCasePath, "../../ILSpy-tests"));
|
|
static readonly string nunit = Path.Combine(TestDir, "nunit", "nunit3-console.exe");
|
|
|
|
[Test]
|
|
public void Cecil_net45()
|
|
{
|
|
RunWithTest("Mono.Cecil-net45", "Mono.Cecil.dll", "Mono.Cecil.Tests.dll");
|
|
}
|
|
|
|
[Test]
|
|
public void NewtonsoftJson_net45()
|
|
{
|
|
RunWithTest("Newtonsoft.Json-net45", "Newtonsoft.Json.dll", "Newtonsoft.Json.Tests.dll");
|
|
}
|
|
|
|
[Test]
|
|
public void NewtonsoftJson_pcl_debug()
|
|
{
|
|
try
|
|
{
|
|
RunWithTest("Newtonsoft.Json-pcl-debug", "Newtonsoft.Json.dll", "Newtonsoft.Json.Tests.dll", useOldProjectFormat: true);
|
|
}
|
|
catch (CompilationFailedException)
|
|
{
|
|
Assert.Ignore("Cannot yet re-compile PCL projects.");
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void NRefactory_CSharp()
|
|
{
|
|
RunWithTest("NRefactory", "ICSharpCode.NRefactory.CSharp.dll", "ICSharpCode.NRefactory.Tests.dll");
|
|
}
|
|
|
|
[Test]
|
|
public void ICSharpCode_Decompiler()
|
|
{
|
|
RunOnly("ICSharpCode.Decompiler", "ICSharpCode.Decompiler.dll");
|
|
}
|
|
|
|
[Test]
|
|
public void ImplicitConversions()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "ImplicitConversions.exe");
|
|
}
|
|
|
|
[Test]
|
|
public void ImplicitConversions_32()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "ImplicitConversions_32.exe");
|
|
}
|
|
|
|
[Test]
|
|
public void ExplicitConversions()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "ExplicitConversions.exe", LanguageVersion.CSharp8_0);
|
|
}
|
|
|
|
[Test]
|
|
public void ExplicitConversions_32()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "ExplicitConversions_32.exe", LanguageVersion.CSharp8_0);
|
|
}
|
|
|
|
[Test]
|
|
[Ignore("Waiting for https://github.com/dotnet/roslyn/issues/45929")]
|
|
public void ExplicitConversions_With_NativeInts()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "ExplicitConversions.exe", LanguageVersion.Preview);
|
|
}
|
|
|
|
[Test]
|
|
[Ignore("Waiting for https://github.com/dotnet/roslyn/issues/45929")]
|
|
public void ExplicitConversions_32_With_NativeInts()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "ExplicitConversions_32.exe", LanguageVersion.Preview);
|
|
}
|
|
|
|
[Test]
|
|
public void Random_TestCase_1()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "TestCase-1.exe", LanguageVersion.CSharp8_0);
|
|
}
|
|
|
|
[Test]
|
|
[Ignore("Waiting for https://github.com/dotnet/roslyn/issues/45929")]
|
|
public void Random_TestCase_1_With_NativeInts()
|
|
{
|
|
RunWithOutput("Random Tests\\TestCases", "TestCase-1.exe", LanguageVersion.Preview);
|
|
}
|
|
|
|
// Let's limit the roundtrip tests to C# 8.0 for now; because 9.0 is still in preview
|
|
// and the generated project doesn't build as-is.
|
|
const LanguageVersion defaultLanguageVersion = LanguageVersion.CSharp8_0;
|
|
|
|
void RunWithTest(string dir, string fileToRoundtrip, string fileToTest, LanguageVersion languageVersion = defaultLanguageVersion, string keyFile = null, bool useOldProjectFormat = false)
|
|
{
|
|
RunInternal(dir, fileToRoundtrip, outputDir => RunTest(outputDir, fileToTest), languageVersion, snkFilePath: keyFile, useOldProjectFormat: useOldProjectFormat);
|
|
}
|
|
|
|
void RunWithOutput(string dir, string fileToRoundtrip, LanguageVersion languageVersion = defaultLanguageVersion)
|
|
{
|
|
string inputDir = Path.Combine(TestDir, dir);
|
|
RunInternal(dir, fileToRoundtrip,
|
|
outputDir => Tester.RunAndCompareOutput(fileToRoundtrip, Path.Combine(inputDir, fileToRoundtrip), Path.Combine(outputDir, fileToRoundtrip)),
|
|
languageVersion);
|
|
}
|
|
|
|
void RunOnly(string dir, string fileToRoundtrip, LanguageVersion languageVersion = defaultLanguageVersion)
|
|
{
|
|
RunInternal(dir, fileToRoundtrip, outputDir => { }, languageVersion);
|
|
}
|
|
|
|
void RunInternal(string dir, string fileToRoundtrip, Action<string> testAction, LanguageVersion languageVersion, string snkFilePath = null, bool useOldProjectFormat = false)
|
|
{
|
|
if (!Directory.Exists(TestDir))
|
|
{
|
|
Assert.Ignore($"Assembly-roundtrip test ignored: test directory '{TestDir}' needs to be checked out separately." + Environment.NewLine +
|
|
$"git clone https://github.com/icsharpcode/ILSpy-tests \"{TestDir}\"");
|
|
}
|
|
string inputDir = Path.Combine(TestDir, dir);
|
|
string decompiledDir = inputDir + "-decompiled";
|
|
string outputDir = inputDir + "-output";
|
|
if (inputDir.EndsWith("TestCases"))
|
|
{
|
|
// make sure output dir names are unique so that we don't get trouble due to parallel test execution
|
|
decompiledDir += Path.GetFileNameWithoutExtension(fileToRoundtrip) + "_" + languageVersion.ToString();
|
|
outputDir += Path.GetFileNameWithoutExtension(fileToRoundtrip) + "_" + languageVersion.ToString();
|
|
}
|
|
ClearDirectory(decompiledDir);
|
|
ClearDirectory(outputDir);
|
|
string projectFile = null;
|
|
foreach (string file in Directory.EnumerateFiles(inputDir, "*", SearchOption.AllDirectories))
|
|
{
|
|
if (!file.StartsWith(inputDir + Path.DirectorySeparatorChar, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
Assert.Fail($"Unexpected file name: {file}");
|
|
}
|
|
string relFile = file.Substring(inputDir.Length + 1);
|
|
Directory.CreateDirectory(Path.Combine(outputDir, Path.GetDirectoryName(relFile)));
|
|
if (relFile.Equals(fileToRoundtrip, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
Console.WriteLine($"Decompiling {fileToRoundtrip}...");
|
|
Stopwatch w = Stopwatch.StartNew();
|
|
using (var fileStream = new FileStream(file, FileMode.Open, FileAccess.Read))
|
|
{
|
|
PEFile module = new PEFile(file, fileStream, PEStreamOptions.PrefetchEntireImage);
|
|
var resolver = new TestAssemblyResolver(file, inputDir, module.Reader.DetectTargetFrameworkId());
|
|
resolver.AddSearchDirectory(inputDir);
|
|
resolver.RemoveSearchDirectory(".");
|
|
|
|
// use a fixed GUID so that we can diff the output between different ILSpy runs without spurious changes
|
|
var projectGuid = Guid.Parse("{127C83E4-4587-4CF9-ADCA-799875F3DFE6}");
|
|
|
|
var settings = new DecompilerSettings(languageVersion);
|
|
if (useOldProjectFormat)
|
|
{
|
|
settings.UseSdkStyleProjectFormat = false;
|
|
}
|
|
|
|
var decompiler = new TestProjectDecompiler(projectGuid, resolver, resolver, settings);
|
|
|
|
if (snkFilePath != null)
|
|
{
|
|
decompiler.StrongNameKeyFile = Path.Combine(inputDir, snkFilePath);
|
|
}
|
|
decompiler.DecompileProject(module, decompiledDir);
|
|
Console.WriteLine($"Decompiled {fileToRoundtrip} in {w.Elapsed.TotalSeconds:f2}");
|
|
projectFile = Path.Combine(decompiledDir, module.Name + ".csproj");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
File.Copy(file, Path.Combine(outputDir, relFile));
|
|
}
|
|
}
|
|
Assert.IsNotNull(projectFile, $"Could not find {fileToRoundtrip}");
|
|
|
|
Compile(projectFile, outputDir);
|
|
testAction(outputDir);
|
|
}
|
|
|
|
static void ClearDirectory(string dir)
|
|
{
|
|
Directory.CreateDirectory(dir);
|
|
foreach (string subdir in Directory.EnumerateDirectories(dir))
|
|
{
|
|
for (int attempt = 0; ; attempt++)
|
|
{
|
|
try
|
|
{
|
|
Directory.Delete(subdir, true);
|
|
break;
|
|
}
|
|
catch (IOException)
|
|
{
|
|
if (attempt >= 10)
|
|
throw;
|
|
Thread.Sleep(100);
|
|
}
|
|
}
|
|
}
|
|
foreach (string file in Directory.EnumerateFiles(dir))
|
|
{
|
|
File.Delete(file);
|
|
}
|
|
}
|
|
|
|
static string FindMSBuild()
|
|
{
|
|
string vsPath = MSBuildLocator.QueryVisualStudioInstances(new VisualStudioInstanceQueryOptions { DiscoveryTypes = DiscoveryType.VisualStudioSetup })
|
|
.OrderByDescending(i => i.Version)
|
|
.FirstOrDefault()
|
|
?.MSBuildPath;
|
|
if (vsPath == null)
|
|
throw new InvalidOperationException("Could not find MSBuild");
|
|
return Path.Combine(vsPath, "msbuild.exe");
|
|
}
|
|
|
|
static void Compile(string projectFile, string outputDir)
|
|
{
|
|
var info = new ProcessStartInfo(FindMSBuild());
|
|
info.Arguments = $"/nologo /v:minimal /restore /p:OutputPath=\"{outputDir}\" \"{projectFile}\"";
|
|
info.CreateNoWindow = true;
|
|
info.UseShellExecute = false;
|
|
info.RedirectStandardOutput = true;
|
|
// Don't let environment variables (e.g. set by AppVeyor) influence the build.
|
|
info.EnvironmentVariables.Remove("Configuration");
|
|
info.EnvironmentVariables.Remove("Platform");
|
|
Console.WriteLine($"\"{info.FileName}\" {info.Arguments}");
|
|
using (var p = Process.Start(info))
|
|
{
|
|
Regex errorRegex = new Regex(@"^[\w\d.\\-]+\(\d+,\d+\):");
|
|
string suffix = $" [{projectFile}]";
|
|
string line;
|
|
while ((line = p.StandardOutput.ReadLine()) != null)
|
|
{
|
|
if (line.EndsWith(suffix, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
line = line.Substring(0, line.Length - suffix.Length);
|
|
}
|
|
Match m = errorRegex.Match(line);
|
|
if (m.Success)
|
|
{
|
|
// Make path absolute so that it gets hyperlinked
|
|
line = Path.GetDirectoryName(projectFile) + Path.DirectorySeparatorChar + line;
|
|
}
|
|
Console.WriteLine(line);
|
|
}
|
|
p.WaitForExit();
|
|
if (p.ExitCode != 0)
|
|
throw new CompilationFailedException($"Compilation of {Path.GetFileName(projectFile)} failed");
|
|
}
|
|
}
|
|
|
|
static void RunTest(string outputDir, string fileToTest)
|
|
{
|
|
var info = new ProcessStartInfo(nunit);
|
|
info.WorkingDirectory = outputDir;
|
|
info.Arguments = $"\"{fileToTest}\"";
|
|
info.CreateNoWindow = true;
|
|
info.UseShellExecute = false;
|
|
info.RedirectStandardOutput = true;
|
|
Console.WriteLine($"\"{info.FileName}\" {info.Arguments}");
|
|
using (var p = Process.Start(info))
|
|
{
|
|
string line;
|
|
while ((line = p.StandardOutput.ReadLine()) != null)
|
|
{
|
|
Console.WriteLine(line);
|
|
}
|
|
p.WaitForExit();
|
|
if (p.ExitCode != 0)
|
|
throw new TestRunFailedException($"Test execution of {Path.GetFileName(fileToTest)} failed");
|
|
}
|
|
}
|
|
|
|
class TestProjectDecompiler : WholeProjectDecompiler
|
|
{
|
|
public TestProjectDecompiler(Guid projecGuid, IAssemblyResolver resolver, AssemblyReferenceClassifier assemblyReferenceClassifier, DecompilerSettings settings)
|
|
: base(settings, projecGuid, resolver, assemblyReferenceClassifier, debugInfoProvider: null)
|
|
{
|
|
}
|
|
}
|
|
|
|
class CompilationFailedException : Exception
|
|
{
|
|
public CompilationFailedException(string message) : base(message)
|
|
{
|
|
}
|
|
}
|
|
|
|
class TestRunFailedException : Exception
|
|
{
|
|
public TestRunFailedException(string message) : base(message)
|
|
{
|
|
}
|
|
}
|
|
}
|
|
}
|