-
-
Notifications
You must be signed in to change notification settings - Fork 276
/
ILTemplateWithTempAssembly.cs
157 lines (131 loc) · 5.27 KB
/
ILTemplateWithTempAssembly.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
using System.Threading;
#if NETCORE
using System.Runtime.Loader;
using System.Runtime.InteropServices;
#else
using System.Runtime.Versioning;
#endif
internal static class ILTemplateWithTempAssembly
{
private static object nullCacheLock = new object();
private static Dictionary<string, bool> nullCache = new Dictionary<string, bool>();
private static string tempBasePath;
private static List<string> preloadList = new List<string>();
private static List<string> preloadWinX86List = new List<string>();
private static List<string> preloadWinX64List = new List<string>();
private static List<string> preloadWinArm64List = new List<string>();
private static Dictionary<string, string> checksums = new Dictionary<string, string>();
private static int isAttached;
public static void Attach(bool subscribe)
{
if (Interlocked.Exchange(ref isAttached, 1) == 1)
{
return;
}
#if !NETCORE
var currentDomain = AppDomain.CurrentDomain;
// Make sure the target framework is set in order not to interfere with AppContext switches initialization
// See https://github.com/Fody/Costura/issues/633 for full explanation
var setupInformation = currentDomain.GetType()?.GetProperty("SetupInformation")?.GetValue(currentDomain);
var targetFrameworkNameProperty = setupInformation?.GetType()?.GetProperty("TargetFrameworkName");
if (targetFrameworkNameProperty is not null && targetFrameworkNameProperty.GetValue(setupInformation) is null)
{
var targetFrameworkAttribute = (TargetFrameworkAttribute)Assembly.GetCallingAssembly()?.GetCustomAttribute(typeof(TargetFrameworkAttribute));
var targetFrameworkName = targetFrameworkAttribute?.FrameworkName;
if (targetFrameworkName is not null)
{
currentDomain.SetData("TargetFrameworkName", targetFrameworkName);
}
}
#endif
//Create a unique Temp directory for the application path.
var md5Hash = "To be replaced at compile time";
var prefixPath = Path.Combine(Path.GetTempPath(), "Costura");
tempBasePath = Path.Combine(prefixPath, md5Hash);
// Preload
var unmanagedAssemblies = GetUnmanagedAssemblies();
var libList = new List<string>();
libList.AddRange(unmanagedAssemblies);
libList.AddRange(preloadList);
Common.PreloadUnmanagedLibraries(md5Hash, tempBasePath, libList, checksums);
if (subscribe)
{
#if NETCORE
AssemblyLoadContext.Default.Resolving += ResolveAssembly;
#else
currentDomain.AssemblyResolve += ResolveAssembly;
#endif
}
}
#if NETCORE
public static Assembly ResolveAssembly(AssemblyLoadContext assemblyLoadContext, AssemblyName assemblyName)
#else
public static Assembly ResolveAssembly(object sender, ResolveEventArgs e)
#endif
{
#if NETCORE
var assemblyNameAsString = assemblyName.Name;
#else
var assemblyNameAsString = e.Name;
var assemblyName = new AssemblyName(assemblyNameAsString);
#endif
lock (nullCacheLock)
{
if (nullCache.ContainsKey(assemblyNameAsString))
{
return null;
}
}
var requestedAssemblyName = new AssemblyName(assemblyNameAsString);
var assembly = Common.ReadExistingAssembly(requestedAssemblyName);
if (assembly is not null)
{
return assembly;
}
Common.Log("Loading assembly '{0}' into the AppDomain", requestedAssemblyName);
assembly = Common.ReadFromDiskCache(tempBasePath, requestedAssemblyName);
if (assembly is null)
{
lock (nullCacheLock)
{
nullCache[assemblyNameAsString] = true;
}
// Handles re-targeted assemblies like PCL
if ((requestedAssemblyName.Flags & AssemblyNameFlags.Retargetable) != 0)
{
assembly = Assembly.Load(requestedAssemblyName);
}
}
return assembly;
}
private static List<string> GetUnmanagedAssemblies()
{
#if NETCORE
var processorArchitecture = RuntimeInformation.ProcessArchitecture;
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
switch (processorArchitecture)
{
case Architecture.Arm64:
return preloadWinArm64List;
case Architecture.X86:
return preloadWinX86List;
case Architecture.X64:
return preloadWinX64List;
default:
// Note: somehow copying string interpolation doesn't work correctly, hence using string.Format instead
//throw new NotSupportedException($"Architecture '{processorArchitecture}' not supported");
throw new NotSupportedException(string.Format("Architecture '{0}' not supported", processorArchitecture));
}
}
throw new NotSupportedException("Platform is not (yet) supported");
#else
// Only support Windows
return IntPtr.Size == 8 ? preloadWinX64List : preloadWinX86List;
#endif
}
}