-
Notifications
You must be signed in to change notification settings - Fork 5.5k
Expand file tree
/
Copy pathComputeManagedAssembliesToCompileToNative.cs
More file actions
225 lines (196 loc) · 8.18 KB
/
Copy pathComputeManagedAssembliesToCompileToNative.cs
File metadata and controls
225 lines (196 loc) · 8.18 KB
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
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using Microsoft.Build.Framework;
using Microsoft.Build.Utilities;
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Reflection.Metadata;
using System.Reflection.PortableExecutable;
namespace Build.Tasks
{
public class ComputeManagedAssembliesToCompileToNative : Task
{
[Required]
public ITaskItem[] Assemblies
{
get;
set;
}
/// <summary>
/// The NativeAOT-specific System.Private.* assemblies that must be used instead of the default CoreCLR versions.
/// </summary>
[Required]
public ITaskItem[] SdkAssemblies
{
get;
set;
}
/// <summary>
/// The set of NativeAOT-specific framework assemblies we currently need to use which will replace the same-named ones
/// in the app's closure.
/// </summary>
[Required]
public ITaskItem[] FrameworkAssemblies
{
get;
set;
}
/// <summary>
/// The native apphost (whose name ends up colliding with the native output binary)
/// </summary>
[Required]
public string DotNetAppHostExecutableName
{
get;
set;
}
/// <summary>
/// The CoreCLR dotnet host fixer library that can be skipped during publish
/// </summary>
[Required]
public string DotNetHostFxrLibraryName
{
get;
set;
}
/// <summary>
/// The CoreCLR dotnet host policy library that can be skipped during publish
/// </summary>
[Required]
public string DotNetHostPolicyLibraryName
{
get;
set;
}
[Output]
public ITaskItem[] ManagedAssemblies
{
get;
set;
}
[Output]
public ITaskItem[] SatelliteAssemblies
{
get;
set;
}
[Output]
public ITaskItem[] AssembliesToSkipPublish
{
get;
set;
}
public override bool Execute()
{
var list = new List<ITaskItem>();
var assembliesToSkipPublish = new List<ITaskItem>();
var satelliteAssemblies = new List<ITaskItem>();
var nativeAotFrameworkAssembliesToUse = new Dictionary<string, ITaskItem>();
foreach (ITaskItem taskItem in SdkAssemblies)
{
var fileName = Path.GetFileName(taskItem.ItemSpec);
if (!nativeAotFrameworkAssembliesToUse.ContainsKey(fileName))
nativeAotFrameworkAssembliesToUse.Add(fileName, taskItem);
}
foreach (ITaskItem taskItem in FrameworkAssemblies)
{
var fileName = Path.GetFileName(taskItem.ItemSpec);
if (!nativeAotFrameworkAssembliesToUse.ContainsKey(fileName))
nativeAotFrameworkAssembliesToUse.Add(fileName, taskItem);
}
foreach (ITaskItem taskItem in Assemblies)
{
// In the case of disk-based assemblies, this holds the file path
string itemSpec = taskItem.ItemSpec;
// Skip the native apphost (whose name ends up colliding with the native output binary) and supporting libraries
if (itemSpec.EndsWith(DotNetAppHostExecutableName, StringComparison.OrdinalIgnoreCase) || itemSpec.Contains(DotNetHostFxrLibraryName) || itemSpec.Contains(DotNetHostPolicyLibraryName))
{
assembliesToSkipPublish.Add(taskItem);
continue;
}
// Prototype aid - remove the native CoreCLR runtime pieces from the publish folder
if (itemSpec.IndexOf("microsoft.netcore.app", StringComparison.OrdinalIgnoreCase) != -1 && (itemSpec.Contains("\\native\\") || itemSpec.Contains("/native/")))
{
assembliesToSkipPublish.Add(taskItem);
continue;
}
var assemblyFileName = Path.GetFileName(itemSpec);
if (assemblyFileName == "WindowsBase.dll")
{
// There are two instances of WindowsBase.dll, one small one, in the NativeAOT framework
// and real one in WindowsDesktop SDK. We want to make sure that if both are present,
// we will use the one from WindowsDesktop SDK, and not from NativeAOT framework.
foreach (ITaskItem taskItemToSkip in FrameworkAssemblies)
{
if (Path.GetFileName(taskItemToSkip.ItemSpec) == assemblyFileName)
{
assembliesToSkipPublish.Add(taskItemToSkip);
break;
}
}
assembliesToSkipPublish.Add(taskItem);
list.Add(taskItem);
continue;
}
// Remove any assemblies whose implementation we want to come from NativeAOT's package.
// Currently that's System.Private.* SDK assemblies and a bunch of framework assemblies.
if (nativeAotFrameworkAssembliesToUse.TryGetValue(assemblyFileName, out ITaskItem frameworkItem))
{
if (GetFileVersion(itemSpec).CompareTo(GetFileVersion(frameworkItem.ItemSpec)) > 0)
{
if (assemblyFileName == "System.Private.CoreLib.dll")
{
Log.LogError($"Overriding System.Private.CoreLib.dll with a newer version is not supported. Attempted to use {itemSpec} instead of {frameworkItem.ItemSpec}.");
}
else
{
// Allow OOB references with higher version to take precedence over the framework assemblies.
list.Add(taskItem);
}
}
assembliesToSkipPublish.Add(taskItem);
continue;
}
try
{
using (FileStream moduleStream = File.OpenRead(itemSpec))
using (var module = new PEReader(moduleStream))
{
if (module.HasMetadata)
{
MetadataReader moduleMetadataReader = module.GetMetadataReader();
if (moduleMetadataReader.IsAssembly)
{
string culture = moduleMetadataReader.GetString(moduleMetadataReader.GetAssemblyDefinition().Culture);
assembliesToSkipPublish.Add(taskItem);
// Split satellite assemblies from normal assemblies
if (culture == "" || culture.Equals("neutral", StringComparison.OrdinalIgnoreCase))
{
list.Add(taskItem);
}
else
{
satelliteAssemblies.Add(taskItem);
}
}
}
}
}
catch (BadImageFormatException)
{
}
}
ManagedAssemblies = list.ToArray();
AssembliesToSkipPublish = assembliesToSkipPublish.ToArray();
SatelliteAssemblies = satelliteAssemblies.ToArray();
return true;
static Version GetFileVersion(string path)
{
var versionInfo = FileVersionInfo.GetVersionInfo(path);
return new Version(versionInfo.FileMajorPart, versionInfo.FileMinorPart, versionInfo.FileBuildPart, versionInfo.FilePrivatePart);
}
}
}
}