diff --git a/src/Shared/AssemblyNameExtension.cs b/src/Shared/AssemblyNameExtension.cs index 529335c8bea..482bccd1bba 100644 --- a/src/Shared/AssemblyNameExtension.cs +++ b/src/Shared/AssemblyNameExtension.cs @@ -57,7 +57,7 @@ internal enum PartialComparisonFlags : int /// between the two is done lazily on demand. /// [Serializable] - sealed internal class AssemblyNameExtension + internal sealed class AssemblyNameExtension { private AssemblyName asAssemblyName = null; private string asString = null; @@ -592,11 +592,7 @@ internal AssemblyNameExtension Clone() if (asAssemblyName != null) { -#if FEATURE_ASSEMBLYNAME_CLONE - newExtension.asAssemblyName = (AssemblyName)asAssemblyName.Clone(); -#else - newExtension.asAssemblyName = new AssemblyName(asAssemblyName.FullName); -#endif + newExtension.asAssemblyName = asAssemblyName.CloneIfPossible(); } newExtension.asString = asString; diff --git a/src/Shared/AssemblyUtilities.cs b/src/Shared/AssemblyUtilities.cs index a40a6d46268..cb166d248e2 100644 --- a/src/Shared/AssemblyUtilities.cs +++ b/src/Shared/AssemblyUtilities.cs @@ -11,6 +11,13 @@ namespace Microsoft.Build.Shared /// internal static class AssemblyUtilities { + // True when the cached method info objects have been set. + private static bool s_initialized; + + // Cached method info + private static MethodInfo s_assemblyNameCloneMethod; + private static PropertyInfo s_assemblylocationProperty; + public static string GetAssemblyLocation(Assembly assembly) { #if FEATURE_ASSEMBLY_LOCATION @@ -18,15 +25,14 @@ public static string GetAssemblyLocation(Assembly assembly) #else // Assembly.Location is only available in .netstandard1.5, but MSBuild needs to target 1.3. // use reflection to access the property + Initialize(); - var locationProperty = assembly.GetType().GetProperty("Location", typeof(string)); - - if (locationProperty == null) + if (s_assemblylocationProperty == null) { throw new NotSupportedException("Type Assembly does not have the Location property"); } - return (string)locationProperty.GetValue(assembly); + return (string)s_assemblylocationProperty.GetValue(assembly); #endif } @@ -39,5 +45,37 @@ public static Type GetTypeInfo(this Type t) return t; } #endif + + public static AssemblyName CloneIfPossible(this AssemblyName assemblyNameToClone) + { +#if FEATURE_ASSEMBLYNAME_CLONE + return (AssemblyName) assemblyNameToClone.Clone(); +#else + + Initialize(); + + if (s_assemblyNameCloneMethod == null) + { + return new AssemblyName(assemblyNameToClone.FullName); + } + + // Try to Invoke the Clone method via reflection. If the method exists (it will on .NET + // Core 2.0 or later) use that result, otherwise use new AssemblyName(FullName). + return (AssemblyName) s_assemblyNameCloneMethod.Invoke(assemblyNameToClone, null) ?? + new AssemblyName(assemblyNameToClone.FullName); +#endif + } + + /// + /// Initialize static fields. Doesn't need to be thread safe. + /// + private static void Initialize() + { + if (s_initialized) return; + + s_assemblyNameCloneMethod = typeof(AssemblyName).GetMethod("Clone"); + s_assemblylocationProperty = typeof(Assembly).GetProperty("Location", typeof(string)); + s_initialized = true; + } } } diff --git a/src/Tasks/AppConfig/DependentAssembly.cs b/src/Tasks/AppConfig/DependentAssembly.cs index 3c70399c50b..c7252c95b3f 100644 --- a/src/Tasks/AppConfig/DependentAssembly.cs +++ b/src/Tasks/AppConfig/DependentAssembly.cs @@ -22,35 +22,23 @@ internal sealed class DependentAssembly private BindingRedirect[] _bindingRedirects = null; /// - /// The partial assemblyname, there should be no version. + /// The partial , there should be no version. /// - private AssemblyName _partialAssemblyName = null; + private AssemblyName _partialAssemblyName; /// - /// The partial assemblyname, there should be no version. + /// The partial , there should be no version. /// internal AssemblyName PartialAssemblyName { set { -#if FEATURE_ASSEMBLYNAME_CLONE - _partialAssemblyName = (AssemblyName)value.Clone(); -#else - _partialAssemblyName = new AssemblyName(value.FullName); -#endif + _partialAssemblyName = value.CloneIfPossible(); _partialAssemblyName.Version = null; } get { - if (_partialAssemblyName == null) - { - return null; - } -#if FEATURE_ASSEMBLYNAME_CLONE - return (AssemblyName)_partialAssemblyName.Clone(); -#else - return new AssemblyName(_partialAssemblyName.FullName); -#endif + return _partialAssemblyName?.CloneIfPossible(); } } diff --git a/src/Tasks/AssemblyDependency/ReferenceTable.cs b/src/Tasks/AssemblyDependency/ReferenceTable.cs index cd4bbc230e3..e6082536856 100644 --- a/src/Tasks/AssemblyDependency/ReferenceTable.cs +++ b/src/Tasks/AssemblyDependency/ReferenceTable.cs @@ -1070,14 +1070,9 @@ IEnumerable preUnificationAssemblyNames { foreach (AssemblyNameExtension preUnificationAssemblyName in preUnificationAssemblyNames) { - string name = preUnificationAssemblyName.Name; // First, unify the assembly name so that we're dealing with the right version. // Not AssemblyNameExtension because we're going to write to it. -#if FEATURE_ASSEMBLYNAME_CLONE - AssemblyNameExtension dependentAssembly = new AssemblyNameExtension((AssemblyName)preUnificationAssemblyName.AssemblyName.Clone()); -#else - AssemblyNameExtension dependentAssembly = new AssemblyNameExtension(new AssemblyName(preUnificationAssemblyName.AssemblyName.FullName)); -#endif + AssemblyNameExtension dependentAssembly = new AssemblyNameExtension(preUnificationAssemblyName.AssemblyName.CloneIfPossible()); Version unifiedVersion; bool isPrerequisite; @@ -1883,11 +1878,7 @@ out AssemblyNameReference[] conflictingReferences byte[] pkt = assemblyName.GetPublicKeyToken(); if (pkt != null && pkt.Length > 0) { -#if FEATURE_ASSEMBLYNAME_CLONE - AssemblyName baseKey = (AssemblyName)assemblyName.AssemblyName.Clone(); -#else - AssemblyName baseKey = new AssemblyName(assemblyName.AssemblyName.FullName); -#endif + AssemblyName baseKey = assemblyName.AssemblyName.CloneIfPossible(); Version version = baseKey.Version; baseKey.Version = null; string key = baseKey.ToString(); @@ -2377,11 +2368,7 @@ out string redistName foreach (DependentAssembly remappedAssembly in _remappedAssemblies) { // First, exclude anything without the simple name match -#if FEATURE_ASSEMBLYNAME_CLONE - AssemblyNameExtension comparisonAssembly = new AssemblyNameExtension((AssemblyName)remappedAssembly.PartialAssemblyName.Clone()); -#else - AssemblyNameExtension comparisonAssembly = new AssemblyNameExtension(new AssemblyName(remappedAssembly.PartialAssemblyName.FullName)); -#endif + AssemblyNameExtension comparisonAssembly = new AssemblyNameExtension(remappedAssembly.PartialAssemblyName.CloneIfPossible()); if (assemblyName.CompareBaseNameTo(comparisonAssembly) == 0) { // Comparison assembly is a partial name. Give it our version.