using System;
using System.IO;
+using System.Linq;
using System.Reflection;
using System.Runtime.Loader;
using System.Collections.Generic;
{
public class AssemblyLoader : AssemblyLoadContext
{
+ private const string NativeAssemblyInfix = ".ni";
+
+ private const string DllAssemblySuffix = ".dll";
+
+ private const string NativeDllAssemblySuffix = NativeAssemblyInfix + DllAssemblySuffix;
+
+ private static readonly string[] s_suffixes = new string[] { NativeDllAssemblySuffix, DllAssemblySuffix };
+
private SortedSet<string> _dllDirectories = new SortedSet<string>();
private SortedSet<string> _nativeDirectories = new SortedSet<string>();
+ private HashSet<FileInfo> _dllCache = new HashSet<FileInfo>();
+
+ public AssemblyLoader()
+ {
+ AssemblyLoadContext.Default.Resolving += OnResolving;
+ }
+
public IEnumerable<string> DllDirectories
{
get { return _dllDirectories; }
{
_dllDirectories.Add(directory);
_nativeDirectories.Add(directory);
+
+ foreach (var file in Directory.GetFiles(directory))
+ {
+ var info = new FileInfo(file);
+
+ if (info.Extension == DllAssemblySuffix)
+ {
+ _dllCache.Add(info);
+ }
+ }
}
}
{
_dllDirectories.Remove(directory);
_nativeDirectories.Remove(directory);
+
+ _dllCache.RemoveWhere(x => x.DirectoryName == directory);
}
- protected override Assembly Load(AssemblyName assemblyName)
+ public Assembly LoadFromPath(string path)
{
- Assembly asm = null;
-
- try
+ if (0 == string.Compare(path, // strA
+ path.Length - NativeDllAssemblySuffix.Length, // indexA
+ NativeAssemblyInfix, // strB
+ 0, // indexB
+ NativeAssemblyInfix.Length, // length
+ StringComparison.OrdinalIgnoreCase)) // options
{
- //asm = AssemblyLoadContext.Default.Load(assemblyName);
- asm = AssemblyLoadContext.Default.LoadFromAssemblyName(assemblyName);
+ return LoadFromNativeImagePath(path, null);
}
- catch (Exception ex)
- when (ex is FileNotFoundException ||
- ex is BadImageFormatException ||
- ex is FileLoadException)
+ else
{
- foreach (string dir in DllDirectories)
- {
- FileInfo f = new FileInfo(Path.Combine(dir, $"{assemblyName.Name}.ni.dll"));
- if (File.Exists(f.FullName))
- {
- asm = LoadFromNativeImagePath(f.FullName, null);
- break;
- }
- f = new FileInfo(Path.Combine(dir, $"{assemblyName.Name}.dll"));
- if (File.Exists(f.FullName))
- {
- asm = LoadFromAssemblyPath(f.FullName);
- break;
- }
- }
+ return LoadFromAssemblyPath(path);
}
- return asm;
+ }
+
+ protected override Assembly Load(AssemblyName assemblyName)
+ {
+ return Resolve(assemblyName);
}
protected override IntPtr LoadUnmanagedDll(string unmanagedDllName)
return native;
}
+
+ private Assembly Resolve(AssemblyName assemblyName)
+ {
+ foreach (string suffix in s_suffixes)
+ {
+ var info = _dllCache.FirstOrDefault(x => x.Name == assemblyName.Name + suffix);
+
+ if (info != null)
+ {
+ return LoadFromPath(info.FullName);
+ }
+ }
+
+ return null;
+ }
+
+ private Assembly OnResolving(AssemblyLoadContext context, AssemblyName assemblyName)
+ {
+ return Resolve(assemblyName);
+ }
}
}
ALog.Debug($"preload dll : {s}");
try
{
- Assembly asm = null;
- if (s.EndsWith(".ni.dll", StringComparison.CurrentCultureIgnoreCase))
- {
- asm = CurrentAssemblyLoaderContext.LoadFromNativeImagePath(s, null);
- }
- else
- {
- asm = CurrentAssemblyLoaderContext.LoadFromAssemblyPath(s);
- }
+ Assembly asm = CurrentAssemblyLoaderContext.LoadFromPath(s);
// this works strange, vm can't load types except loaded in here.
// so user code spit out not found exception.
FileInfo f = new FileInfo(dllPath);
if (File.Exists(f.FullName))
{
- Assembly asm = null;
- if (0 == string.Compare(f.FullName, f.FullName.Length - 7, ".ni", 0, 3, StringComparison.OrdinalIgnoreCase))
- {
- asm = CurrentAssemblyLoaderContext.LoadFromNativeImagePath(f.FullName, null);
- }
- else
- {
- asm = CurrentAssemblyLoaderContext.LoadFromAssemblyPath(f.FullName);
- }
+ Assembly asm = CurrentAssemblyLoaderContext.LoadFromPath(f.FullName);
if (asm == null) throw new FileNotFoundException($"{f.FullName} is not found");
if (asm.EntryPoint == null) throw new ArgumentException($"{f.FullName} did not have EntryPoint");
asm.EntryPoint.Invoke(null, new object[]{argv});
}
+ else
+ {
+ ALog.Debug($"Requested file does not exist: {dllPath}");
+ }
}
catch (Exception e)
{