Decompiled source of MoreSizesPlugin v2.4.0
MoreSizesPlugin.dll
Decompiled 3 weeks agousing System; using System.CodeDom.Compiler; using System.Collections; using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.Globalization; using System.IO; using System.Linq; using System.Reflection; using System.Resources; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.Versioning; using BepInEx; using BepInEx.Configuration; using Bounce.Unmanaged; using HarmonyLib; using ModdingTales; using MoreSizesPlugin.Consumer.Messages; using MoreSizesPlugin.Patches; using Newtonsoft.Json; using RPCPlugin.Interfaces; using RPCPlugin.RPC; using RadialUI; using UnityEngine; using ZeroFormatter; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)] [assembly: AssemblyTitle("MoreSizesPlugin")] [assembly: AssemblyDescription("")] [assembly: AssemblyConfiguration("")] [assembly: AssemblyCompany("")] [assembly: AssemblyProduct("MoreSizesPlugin")] [assembly: AssemblyCopyright("Copyright © 2024")] [assembly: AssemblyTrademark("")] [assembly: ComVisible(false)] [assembly: Guid("4e4deb5e-97f9-4901-bf67-6748a9c1229a")] [assembly: AssemblyFileVersion("2.4.0.0")] [assembly: TargetFramework(".NETFramework,Version=v4.8", FrameworkDisplayName = ".NET Framework 4.8")] [assembly: AssemblyVersion("2.4.0.0")] namespace MoreSizesPlugin { [BepInPlugin("org.hollofox.plugins.MoreSizesPlugin", "More Sizes Plug-In", "2.4.0.0")] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] [BepInDependency(/*Could not decode attribute arguments.*/)] public class MoreSizesPlugin : BaseUnityPlugin { private const string Guid = "org.hollofox.plugins.MoreSizesPlugin"; private const string Version = "2.4.0.0"; private static CreatureGuid _selectedCreature; private static MoreSizesPlugin _self = null; private readonly float[] coreSizes = new float[5] { 0.5f, 1f, 2f, 3f, 4f }; private static float _restorationDelay = 1f; private MethodInfo setMethod; private MethodInfo clearMethod; private ConfigEntry<string> _customSizes; private void Awake() { //IL_0100: Unknown result type (might be due to invalid IL or missing references) //IL_013a: Unknown result type (might be due to invalid IL or missing references) //IL_013f: Unknown result type (might be due to invalid IL or missing references) //IL_0151: Unknown result type (might be due to invalid IL or missing references) //IL_0158: Unknown result type (might be due to invalid IL or missing references) //IL_0174: Expected O, but got Unknown _self = this; ((BaseUnityPlugin)this).Logger.LogInfo((object)"In Awake for More Sizes"); _customSizes = ((BaseUnityPlugin)this).Config.Bind<string>("Sizes", "List", JsonConvert.SerializeObject((object)new List<float> { 0.5f, 0.75f, 1f, 1.5f, 2f, 3f, 4f, 6f, 8f, 10f, 15f, 20f, 25f, 30f }), (ConfigDescription)null); _restorationDelay = ((BaseUnityPlugin)this).Config.Bind<float>("Settings", "Restoration delay upon board dload", 1f, (ConfigDescription)null).Value; new Harmony("org.hollofox.plugins.MoreSizesPlugin").PatchAll(); ((BaseUnityPlugin)this).Logger.LogDebug((object)"MoreSizes Plug-in loaded"); ModdingUtils.AddPluginToMenuList((BaseUnityPlugin)(object)this, "Hollofoxes'"); RadialUIPlugin.HideDefaultEmotesGMItem("org.hollofox.plugins.MoreSizesPlugin", "Set Size", (ShouldShowMenu)null); RadialUIPlugin.AddCustomButtonGMSubmenu("Set Size", new ItemArgs { Action = HandleSubmenus, CloseMenuOnActivate = false, Title = "Set Size" }, (Func<NGuid, NGuid, bool>)Reporter); AddAssetDataPluginPersistenceIfAvailable(); } private void AddAssetDataPluginPersistenceIfAvailable() { Type type = Type.GetType("LordAshes.AssetDataPlugin, AssetDataPlugin"); if (type != null) { (from m in type.GetRuntimeMethods() where m.Name == "SubscribeViaReflection" select m).ElementAt(0).Invoke(null, new object[3] { "org.hollofox.plugins.MoreSizesPlugin.size", ((object)this).GetType().AssemblyQualifiedName, "RestoreSize" }); setMethod = (from m in type.GetRuntimeMethods() where m.Name == "SetInfo" select m).ElementAt(0); clearMethod = (from m in type.GetRuntimeMethods() where m.Name == "ClearInfo" select m).ElementAt(0); ((BaseUnityPlugin)this).Logger.LogDebug((object)"More Size Plugin size persistence handled by Asset Data Plugin."); } else { ((BaseUnityPlugin)this).Logger.LogDebug((object)"More Size Plugin size persistence unavailable. Missing Asset Data Plugin."); } } public static void RestoreSize(string action, string identity, string key, object previous, object value) { //IL_0055: Unknown result type (might be due to invalid IL or missing references) //IL_005a: Unknown result type (might be due to invalid IL or missing references) ((BaseUnityPlugin)_self).Logger.LogDebug((object)(DateTime.UtcNow.ToString() + ": Preparing to restore custom size " + value?.ToString() + " on creature id " + identity)); MoreSizesPlugin self = _self; NGuid val = new NGuid(identity); ((MonoBehaviour)self).StartCoroutine(DelayedRestoreSize(((NGuid)(ref val)).ToHexString(), float.Parse(value.ToString(), CultureInfo.InvariantCulture), _restorationDelay)); } private static IEnumerator DelayedRestoreSize(string cid, float actualSize, float delay) { yield return (object)new WaitForSeconds(delay); ((BaseUnityPlugin)_self).Logger.LogDebug((object)(DateTime.UtcNow.ToString() + ": Restoring custom size " + actualSize + " on creature id " + cid)); RPCInstance.SendMessage((RpcMessage)(object)new ScaleMini { size = actualSize, cid = cid }, (PhotonTargets)0); } private static Sprite LoadEmbeddedTexture(string texturePath) { //IL_0056: Unknown result type (might be due to invalid IL or missing references) //IL_005c: Expected O, but got Unknown //IL_007d: Unknown result type (might be due to invalid IL or missing references) //IL_008c: Unknown result type (might be due to invalid IL or missing references) Stream manifestResourceStream = Assembly.GetExecutingAssembly().GetManifestResourceStream(typeof(MoreSizesPlugin).Namespace + "." + texturePath.Replace("/", ".")); byte[] array; using (MemoryStream memoryStream = new MemoryStream()) { manifestResourceStream.CopyTo(memoryStream); array = memoryStream.ToArray(); } Texture2D val = new Texture2D(1, 1); ImageConversion.LoadImage(val, array); return Sprite.Create(val, new Rect(0f, 0f, (float)((Texture)val).width, (float)((Texture)val).height), new Vector2(0.5f, 0.5f)); } private void HandleSubmenus(MapMenuItem arg1, object arg2) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) CreatureBoardAsset val = default(CreatureBoardAsset); CreaturePresenter.TryGetAsset(_selectedCreature, ref val); OpenResizeMini(arg1, arg2); } private void OpenResizeMini(MapMenuItem arg1, object arg2) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) //IL_0013: Unknown result type (might be due to invalid IL or missing references) //IL_0018: Unknown result type (might be due to invalid IL or missing references) //IL_0022: Unknown result type (might be due to invalid IL or missing references) //IL_0027: Unknown result type (might be due to invalid IL or missing references) CreatureBoardAsset val = default(CreatureBoardAsset); CreaturePresenter.TryGetAsset(_selectedCreature, ref val); MapMenu mapMenu = MapMenuManager.OpenMenu(((Component)val).transform.position + Vector3.up * CreatureMenuBoardPatch._hitHeightDif, true); foreach (float item in JsonConvert.DeserializeObject<List<float>>(_customSizes.Value)) { if (item < 1f) { AddSize(mapMenu, item, Icons.GetIconSprite("05x05")); } else if (item < 2f) { AddSize(mapMenu, item, Icons.GetIconSprite("1x1")); } else if (item < 3f) { AddSize(mapMenu, item, Icons.GetIconSprite("2x2")); } else if (item < 4f) { AddSize(mapMenu, item, Icons.GetIconSprite("3x3")); } else { AddSize(mapMenu, item, Icons.GetIconSprite("4x4")); } } } private void AddSize(MapMenu mapMenu, float x, Sprite icon = null) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_0022: Unknown result type (might be due to invalid IL or missing references) //IL_0034: Unknown result type (might be due to invalid IL or missing references) //IL_0040: Unknown result type (might be due to invalid IL or missing references) //IL_0047: Unknown result type (might be due to invalid IL or missing references) //IL_0053: Expected O, but got Unknown mapMenu.AddItem(new ItemArgs { Title = $"{x}x{x}", Action = Menu_Scale, Obj = x, CloseMenuOnActivate = true, Icon = icon }); } private bool Reporter(NGuid arg1, NGuid arg2) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) _selectedCreature = new CreatureGuid(arg2); return true; } private void Menu_Scale(MapMenuItem item, object obj) { //IL_00d6: Unknown result type (might be due to invalid IL or missing references) float num = (float)obj; string text = ((NGuid)(ref _selectedCreature.Value)).ToHexString(); if (!coreSizes.Contains(num)) { ((BaseUnityPlugin)this).Logger.LogInfo((object)$"New message: {num}"); RPCInstance.SendMessage((RpcMessage)(object)new ScaleMini { size = num, cid = text }, (PhotonTargets)0); if (setMethod != null) { ((BaseUnityPlugin)this).Logger.LogDebug((object)("Saving custom size " + num + " on creature id " + text)); setMethod.Invoke(null, new object[4] { ((object)(NGuid)(ref _selectedCreature.Value)).ToString(), "org.hollofox.plugins.MoreSizesPlugin.size", num.ToString(), false }); } } else { CreatureManager.SetCreatureScale(_selectedCreature, 0, num); if (clearMethod != null) { ((BaseUnityPlugin)this).Logger.LogDebug((object)("Clearing custom size " + num + " on creature id " + text)); clearMethod.Invoke(null, new object[3] { ((object)(NGuid)(ref _selectedCreature.Value)).ToString(), "org.hollofox.plugins.MoreSizesPlugin", false }); } } } } } namespace MoreSizesPlugin.Properties { [GeneratedCode("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0")] [DebuggerNonUserCode] [CompilerGenerated] internal class Resources { private static ResourceManager resourceMan; private static CultureInfo resourceCulture; [EditorBrowsable(EditorBrowsableState.Advanced)] internal static ResourceManager ResourceManager { get { if (resourceMan == null) { resourceMan = new ResourceManager("MoreSizesPlugin.Properties.Resources", typeof(Resources).Assembly); } return resourceMan; } } [EditorBrowsable(EditorBrowsableState.Advanced)] internal static CultureInfo Culture { get { return resourceCulture; } set { resourceCulture = value; } } internal Resources() { } } } namespace MoreSizesPlugin.Patches { [HarmonyPatch(typeof(CreatureMenuBoardTool), "Begin")] internal class CreatureMenuBoardPatch { internal static float _hitHeightDif; public static void Postfix(ref float ____hitHeightDif) { _hitHeightDif = ____hitHeightDif; } } } namespace MoreSizesPlugin.Consumer { [InitOnLoad] internal class ScaleMiniConsumer : RpcConsumer<ScaleMini> { private static readonly ScaleMiniConsumer instance; public static ScaleMiniConsumer Instance => instance; static ScaleMiniConsumer() { instance = new ScaleMiniConsumer(); } private ScaleMiniConsumer() { } public override void Handle(ScaleMini message) { //IL_002f: Unknown result type (might be due to invalid IL or missing references) //IL_0030: Unknown result type (might be due to invalid IL or missing references) //IL_0066: Unknown result type (might be due to invalid IL or missing references) Debug.Log((object)$"Message Received, {message.size}, {message.cid}"); NGuid val = default(NGuid); if (NGuid.TryParseHexString(message.cid, ref val)) { CreatureBoardAsset val2 = default(CreatureBoardAsset); CreaturePresenter.TryGetAsset(new CreatureGuid(val), ref val2); ((Component)val2).gameObject.transform.GetChild(0).GetChild(0).localScale = new Vector3(message.size, message.size, message.size); } } } } namespace MoreSizesPlugin.Consumer.Messages { [ZeroFormattable] public class ScaleMini : RpcMessage { [Index(0)] public virtual float size { get; set; } [Index(1)] public virtual string cid { get; set; } public ScaleMini() { } public override byte[] Value() { Debug.Log((object)$"size: {size}, cid {cid}"); return ZeroFormatterSerializer.Serialize<ScaleMini>(this); } public ScaleMini(byte[] data) { Debug.Log((object)("data:" + data.ToString())); ScaleMini scaleMini = ZeroFormatterSerializer.Deserialize<ScaleMini>(data); size = scaleMini.size; cid = scaleMini.cid; Debug.Log((object)$"size: {size}, cid {cid}"); } } }
ZeroFormatter.dll
Decompiled 3 weeks ago
The result has been truncated due to the large size, download it to view full contents!
using System; using System.Collections; using System.Collections.Concurrent; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Diagnostics; using System.IO; using System.Linq; using System.Linq.Expressions; using System.Reflection; using System.Reflection.Emit; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.Versioning; using System.Security; using System.Security.Permissions; using System.Text; using ZeroFormatter.Comparers; using ZeroFormatter.Formatters; using ZeroFormatter.Internal; using ZeroFormatter.Segments; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)] [assembly: AssemblyTitle("ZeroFormatter")] [assembly: AssemblyDescription("")] [assembly: AssemblyConfiguration("")] [assembly: AssemblyCompany("")] [assembly: AssemblyProduct("ZeroFormatter")] [assembly: AssemblyCopyright("Copyright © 2016")] [assembly: AssemblyTrademark("")] [assembly: ComVisible(false)] [assembly: Guid("82049afe-0913-4fa4-b924-ad5354952154")] [assembly: AssemblyFileVersion("1.6.4.0")] [assembly: InternalsVisibleTo("ZeroFormatter.Tests")] [assembly: InternalsVisibleTo("ZeroFormatter.NETCore.Tests")] [assembly: TargetFramework(".NETFramework,Version=v4.5", FrameworkDisplayName = ".NET Framework 4.5")] [assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)] [assembly: AssemblyVersion("1.6.4.0")] [module: UnverifiableCode] namespace ZeroFormatter { internal static class EnumerableExtensions { public static T[] StartsWith<T>(this T[] array, T firstValue) { T[] array2 = new T[array.Length + 1]; array2[0] = firstValue; Array.Copy(array, 0, array2, 1, array.Length); return array2; } } public class DirtyTracker { private sealed class NullDirtyTracker : DirtyTracker { public override bool IsDirty => false; public override DirtyTracker CreateChild() { return this; } public override void Dirty() { } } public static readonly DirtyTracker NullTracker = new NullDirtyTracker(); private readonly DirtyTracker parent; public virtual bool IsDirty { get; private set; } public DirtyTracker() { IsDirty = false; } private DirtyTracker(DirtyTracker parent) { this.parent = parent; IsDirty = parent.IsDirty; } public virtual void Dirty() { IsDirty = true; if (parent != null) { parent.Dirty(); } } public virtual DirtyTracker CreateChild() { return new DirtyTracker(this); } } public static class ZeroFormatterSerializer { public static class NonGeneric { private delegate int RefSerialize(ref byte[] bytes, int offset, object obj); private class CompiledMethods { public readonly Func<object, byte[]> serialize1; public readonly RefSerialize serialize2; public readonly Action<Stream, object> serialize3; public readonly Func<byte[], object> deserialize1; public readonly Func<byte[], int, object> deserialize2; public readonly Func<Stream, object> deserialize3; public readonly Func<object, bool, object> convert; public CompiledMethods(Type type) { TypeInfo typeInfo = type.GetTypeInfo(); MethodInfo[] methods = typeof(ZeroFormatterSerializer).GetTypeInfo().GetMethods(); MethodInfo method = methods.First((MethodInfo x) => x.Name == "Serialize" && x.GetParameters().Length == 1).MakeGenericMethod(type); ParameterExpression parameterExpression = Expression.Parameter(typeof(object), "obj"); serialize1 = Expression.Lambda<Func<object, byte[]>>(Expression.Call(method, typeInfo.IsValueType ? Expression.Unbox(parameterExpression, type) : Expression.Convert(parameterExpression, type)), new ParameterExpression[1] { parameterExpression }).Compile(); MethodInfo method2 = methods.First((MethodInfo x) => x.Name == "Serialize" && x.GetParameters().Length == 3).MakeGenericMethod(type); ParameterExpression parameterExpression2 = Expression.Parameter(typeof(byte[]).MakeByRefType(), "bytes"); ParameterExpression parameterExpression3 = Expression.Parameter(typeof(int), "offset"); ParameterExpression parameterExpression4 = Expression.Parameter(typeof(object), "obj"); serialize2 = Expression.Lambda<RefSerialize>(Expression.Call(method2, parameterExpression2, parameterExpression3, typeInfo.IsValueType ? Expression.Unbox(parameterExpression4, type) : Expression.Convert(parameterExpression4, type)), new ParameterExpression[3] { parameterExpression2, parameterExpression3, parameterExpression4 }).Compile(); MethodInfo method3 = methods.First((MethodInfo x) => x.Name == "Serialize" && x.GetParameters().Length == 2).MakeGenericMethod(type); ParameterExpression parameterExpression5 = Expression.Parameter(typeof(Stream), "stream"); ParameterExpression parameterExpression6 = Expression.Parameter(typeof(object), "obj"); serialize3 = Expression.Lambda<Action<Stream, object>>(Expression.Call(method3, parameterExpression5, typeInfo.IsValueType ? Expression.Unbox(parameterExpression6, type) : Expression.Convert(parameterExpression6, type)), new ParameterExpression[2] { parameterExpression5, parameterExpression6 }).Compile(); MethodInfo method4 = methods.First((MethodInfo x) => x.Name == "Deserialize" && x.GetParameters()[0].ParameterType == typeof(byte[])).MakeGenericMethod(type); ParameterExpression parameterExpression7 = Expression.Parameter(typeof(byte[]), "bytes"); Func<byte[], object> func = Expression.Lambda<Func<byte[], object>>(Expression.Convert(Expression.Call(method4, parameterExpression7), typeof(object)), new ParameterExpression[1] { parameterExpression7 }).Compile(); deserialize1 = func; MethodInfo method5 = methods.First((MethodInfo x) => x.Name == "Deserialize" && x.GetParameters().Length == 2).MakeGenericMethod(type); ParameterExpression parameterExpression8 = Expression.Parameter(typeof(byte[]), "bytes"); ParameterExpression parameterExpression9 = Expression.Parameter(typeof(int), "offset"); Func<byte[], int, object> func2 = Expression.Lambda<Func<byte[], int, object>>(Expression.Convert(Expression.Call(method5, parameterExpression8, parameterExpression9), typeof(object)), new ParameterExpression[2] { parameterExpression8, parameterExpression9 }).Compile(); deserialize2 = func2; MethodInfo method6 = methods.First((MethodInfo x) => x.Name == "Deserialize" && x.GetParameters()[0].ParameterType == typeof(Stream)).MakeGenericMethod(type); ParameterExpression parameterExpression10 = Expression.Parameter(typeof(Stream), "stream"); Func<Stream, object> func3 = Expression.Lambda<Func<Stream, object>>(Expression.Convert(Expression.Call(method6, parameterExpression10), typeof(object)), new ParameterExpression[1] { parameterExpression10 }).Compile(); deserialize3 = func3; MethodInfo method7 = methods.First((MethodInfo x) => x.Name == "Convert").MakeGenericMethod(type); ParameterExpression parameterExpression11 = Expression.Parameter(typeof(object), "obj"); ParameterExpression parameterExpression12 = Expression.Parameter(typeof(bool), "forceConvert"); convert = Expression.Lambda<Func<object, bool, object>>(Expression.Convert(Expression.Call(method7, typeInfo.IsValueType ? Expression.Unbox(parameterExpression11, type) : Expression.Convert(parameterExpression11, type), parameterExpression12), typeof(object)), new ParameterExpression[2] { parameterExpression11, parameterExpression12 }).Compile(); } } private static readonly ConcurrentDictionary<Type, CompiledMethods> serializes = new ConcurrentDictionary<Type, CompiledMethods>(); public static byte[] Serialize(Type type, object obj) { return serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).serialize1(obj); } public static int Serialize(Type type, ref byte[] bytes, int offset, object obj) { return serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).serialize2(ref bytes, offset, obj); } public static void Serialize(Type type, Stream stream, object obj) { serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).serialize3(stream, obj); } public static object Deserialize(Type type, byte[] bytes) { return serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).deserialize1(bytes); } public static object Deserialize(Type type, byte[] bytes, int offset) { return serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).deserialize2(bytes, offset); } public static object Deserialize(Type type, Stream stream) { return serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).deserialize3(stream); } public static object Convert(Type type, object obj, bool forceConvert = false) { return serializes.GetOrAdd(type, (Type t) => new CompiledMethods(t)).convert(obj, forceConvert); } public static bool IsFormattedObject(object obj) { return obj is IZeroFormatterSegment; } } public static class CustomSerializer<TTypeResolver> where TTypeResolver : ITypeResolver, new() { public static byte[] Serialize<T>(T obj) { byte[] buffer = null; int num = Serialize(ref buffer, 0, obj); if (buffer.Length != num) { BinaryUtil.FastResize(ref buffer, num); } return buffer; } public static int Serialize<T>(ref byte[] buffer, int offset, T obj) { return (Formatter<TTypeResolver, T>.Default ?? throw new InvalidOperationException("Formatter not found, " + typeof(T).Name)).Serialize(ref buffer, offset, obj); } public static void Serialize<T>(Stream stream, T obj) { if (stream is MemoryStream memoryStream && memoryStream.Position == 0L && (object)obj is IZeroFormatterSegment zeroFormatterSegment && zeroFormatterSegment.CanDirectCopy()) { ArraySegment<byte> bufferReference = zeroFormatterSegment.GetBufferReference(); memoryStream.SetLength(bufferReference.Count); byte[] buffer = memoryStream.GetBuffer(); Buffer.BlockCopy(bufferReference.Array, bufferReference.Offset, buffer, 0, bufferReference.Count); } else { byte[] array = Serialize(obj); stream.Write(array, 0, array.Length); } } public static T Deserialize<T>(byte[] bytes) { Formatter<TTypeResolver, T> obj = Formatter<TTypeResolver, T>.Default ?? throw new InvalidOperationException("Formatter not found, " + typeof(T).Name); DirtyTracker tracker = (obj.NoUseDirtyTracker ? DirtyTracker.NullTracker : new DirtyTracker()); int byteSize; return obj.Deserialize(ref bytes, 0, tracker, out byteSize); } public static T Deserialize<T>(byte[] bytes, int offset) { Formatter<TTypeResolver, T> obj = Formatter<TTypeResolver, T>.Default ?? throw new InvalidOperationException("Formatter not found, " + typeof(T).Name); DirtyTracker tracker = (obj.NoUseDirtyTracker ? DirtyTracker.NullTracker : new DirtyTracker()); int byteSize; return obj.Deserialize(ref bytes, offset, tracker, out byteSize); } public static T Deserialize<T>(Stream stream) { MemoryStream memoryStream = stream as MemoryStream; Formatter<TTypeResolver, T> @default = Formatter<TTypeResolver, T>.Default; DirtyTracker tracker = (@default.NoUseDirtyTracker ? DirtyTracker.NullTracker : new DirtyTracker()); if (memoryStream != null) { byte[] bytes = memoryStream.GetBuffer(); int byteSize; return @default.Deserialize(ref bytes, (int)memoryStream.Position, tracker, out byteSize); } ArraySegment<byte> arraySegment = FillFromStream(stream); byte[] bytes2 = arraySegment.Array; int byteSize2; return @default.Deserialize(ref bytes2, arraySegment.Offset, tracker, out byteSize2); } public static T Convert<T>(T obj, bool forceConvert = false) { IZeroFormatterSegment zeroFormatterSegment = obj as IZeroFormatterSegment; if (!forceConvert && zeroFormatterSegment != null && zeroFormatterSegment.CanDirectCopy()) { return obj; } return Deserialize<T>(Serialize(obj)); } } private const int DefaultMaxSize = 67108864; private static int maxSize = 67108864; public static int MaximumLengthOfDeserialize { get { return maxSize; } set { maxSize = value; } } public static void ValidateNewLength(int length) { if (MaximumLengthOfDeserialize < length) { throw new InvalidOperationException("Reached maximum length:" + MaximumLengthOfDeserialize + " so ensure MaximumLengthOfDeserialize or handle alternate strategy."); } } public static byte[] Serialize<T>(T obj) { return CustomSerializer<DefaultResolver>.Serialize(obj); } public static int Serialize<T>(ref byte[] buffer, int offset, T obj) { return CustomSerializer<DefaultResolver>.Serialize(ref buffer, offset, obj); } public static void Serialize<T>(Stream stream, T obj) { CustomSerializer<DefaultResolver>.Serialize(stream, obj); } public static T Deserialize<T>(byte[] bytes) { return CustomSerializer<DefaultResolver>.Deserialize<T>(bytes); } public static T Deserialize<T>(byte[] bytes, int offset) { return CustomSerializer<DefaultResolver>.Deserialize<T>(bytes, offset); } public static T Deserialize<T>(Stream stream) { return CustomSerializer<DefaultResolver>.Deserialize<T>(stream); } public static T Convert<T>(T obj, bool forceConvert = false) { return CustomSerializer<DefaultResolver>.Convert(obj, forceConvert); } public static bool IsFormattedObject<T>(T obj) { return obj is IZeroFormatterSegment; } private static ArraySegment<byte> FillFromStream(Stream input) { byte[] array = new byte[input.CanSeek ? input.Length : 16]; int num = 0; int num2; while ((num2 = input.Read(array, num, array.Length - num)) > 0) { num += num2; if (num == array.Length) { BinaryUtil.FastResize(ref array, num * 2); } } return new ArraySegment<byte>(array, 0, num); } } } namespace ZeroFormatter.Segments { public interface IZeroFormatterSegment { bool CanDirectCopy(); ArraySegment<byte> GetBufferReference(); int Serialize(ref byte[] targetBytes, int offset); } public sealed class DictionarySegment<TTypeResolver, TKey, TValue> : IDictionary<TKey, TValue>, ICollection<KeyValuePair<TKey, TValue>>, IEnumerable<KeyValuePair<TKey, TValue>>, IEnumerable, IZeroFormatterSegment, ILazyDictionary<TKey, TValue>, IReadOnlyDictionary<TKey, TValue>, IReadOnlyCollection<KeyValuePair<TKey, TValue>>, ILazyReadOnlyDictionary<TKey, TValue> where TTypeResolver : ITypeResolver, new() { private int count; private IList<int> buckets; private IList<DictionaryEntry<TTypeResolver, TKey, TValue>> entries; private int freeCount; private int freeList; private DirtyTracker tracker; private ArraySegment<byte> originalBytes; private readonly IEqualityComparer<TKey> comparer; public int Count => count - freeCount; ICollection<TKey> IDictionary<TKey, TValue>.Keys { get { throw new NotSupportedException("ZeroFormatter Dictionary does not support Keys; use GetEnumerator instead."); } } ICollection<TValue> IDictionary<TKey, TValue>.Values { get { throw new NotSupportedException("ZeroFormatter Dictionary does not support Values; use GetEnumerator instead."); } } public IEnumerable<TKey> Keys { get { throw new NotSupportedException("ZeroFormatter Dictionary does not support Keys; use GetEnumerator instead."); } } public IEnumerable<TValue> Values { get { throw new NotSupportedException("ZeroFormatter Dictionary does not support Values; use GetEnumerator instead."); } } public TValue this[TKey key] { get { int num = FindEntry(key); if (num >= 0) { return entries[num].Value; } throw new KeyNotFoundException(); } set { Insert(key, value, add: false); } } bool ICollection<KeyValuePair<TKey, TValue>>.IsReadOnly => true; internal DictionarySegment(DirtyTracker tracker, int size) { tracker = tracker.CreateChild(); this.tracker = tracker; count = 0; if (size == 0) { size = HashHelpers.GetPrime(size); } int[] array = new int[size]; for (int i = 0; i < array.Length; i++) { array[i] = -1; } buckets = array; entries = new DictionaryEntry<TTypeResolver, TKey, TValue>[size]; comparer = ZeroFormatterEqualityComparer<TKey>.Default; freeList = -1; freeCount = 0; } internal static DictionarySegment<TTypeResolver, TKey, TValue> Create(DirtyTracker tracker, byte[] bytes, int offset, out int byteSize) { int num = BinaryUtil.ReadInt32(ref bytes, offset); if (num == -1) { byteSize = 4; return null; } byteSize = num; return new DictionarySegment<TTypeResolver, TKey, TValue>(tracker, new ArraySegment<byte>(bytes, offset, byteSize)); } private DictionarySegment(DirtyTracker tracker, ArraySegment<byte> originalBytes) { tracker = tracker.CreateChild(); this.tracker = tracker; this.originalBytes = originalBytes; byte[] bytes = originalBytes.Array; int num = originalBytes.Offset + 4; count = BinaryUtil.ReadInt32(ref bytes, num); num += 4; Formatter<TTypeResolver, IList<int>> @default = Formatter<TTypeResolver, IList<int>>.Default; Formatter<TTypeResolver, IList<DictionaryEntry<TTypeResolver, TKey, TValue>>> default2 = Formatter<TTypeResolver, IList<DictionaryEntry<TTypeResolver, TKey, TValue>>>.Default; buckets = @default.Deserialize(ref bytes, num, tracker, out var byteSize); num += byteSize; entries = default2.Deserialize(ref bytes, num, tracker, out byteSize); num += byteSize; if (buckets.Count == 0) { int prime = HashHelpers.GetPrime(0); Resize(prime); } comparer = ZeroFormatterEqualityComparer<TKey>.Default; freeList = -1; freeCount = 0; } public void Add(TKey key, TValue value) { Insert(key, value, add: true); } void ICollection<KeyValuePair<TKey, TValue>>.Add(KeyValuePair<TKey, TValue> keyValuePair) { Add(keyValuePair.Key, keyValuePair.Value); } bool ICollection<KeyValuePair<TKey, TValue>>.Contains(KeyValuePair<TKey, TValue> keyValuePair) { int num = FindEntry(keyValuePair.Key); if (num >= 0 && EqualityComparer<TValue>.Default.Equals(entries[num].Value, keyValuePair.Value)) { return true; } return false; } bool ICollection<KeyValuePair<TKey, TValue>>.Remove(KeyValuePair<TKey, TValue> keyValuePair) { tracker.Dirty(); int num = FindEntry(keyValuePair.Key); if (num >= 0 && EqualityComparer<TValue>.Default.Equals(entries[num].Value, keyValuePair.Value)) { Remove(keyValuePair.Key); return true; } return false; } public void Clear() { tracker.Dirty(); if (count <= 0) { return; } for (int i = 0; i < buckets.Count; i++) { buckets[i] = -1; } ListSegment<TTypeResolver, DictionaryEntry<TTypeResolver, TKey, TValue>> listSegment = entries as ListSegment<TTypeResolver, DictionaryEntry<TTypeResolver, TKey, TValue>>; DictionaryEntry<TTypeResolver, TKey, TValue>[] array = entries as DictionaryEntry<TTypeResolver, TKey, TValue>[]; if (listSegment != null) { listSegment.Clear(); } else { if (array == null) { throw new InvalidOperationException("entries type is invalid. " + entries.GetType().Name); } Array.Clear(array, 0, count); } freeList = -1; count = 0; freeCount = 0; } public bool ContainsKey(TKey key) { return FindEntry(key) >= 0; } public bool ContainsValue(TValue value) { if (value == null) { for (int i = 0; i < count; i++) { if (entries[i].HashCode >= 0 && entries[i].Value == null) { return true; } } } else { EqualityComparer<TValue> @default = EqualityComparer<TValue>.Default; for (int j = 0; j < count; j++) { if (entries[j].HashCode >= 0 && @default.Equals(entries[j].Value, value)) { return true; } } } return false; } private void CopyTo(KeyValuePair<TKey, TValue>[] array, int index) { if (array == null) { throw new ArgumentNullException("array"); } if (index < 0 || index > array.Length) { throw new ArgumentOutOfRangeException("index", index, "ArgumentOutOfRange_Index"); } if (array.Length - index < Count) { throw new ArgumentException("Arg_ArrayPlusOffTooSmall"); } int num = count; for (int i = 0; i < num; i++) { if (entries[i].HashCode >= 0) { array[index++] = new KeyValuePair<TKey, TValue>(entries[i].Key, entries[i].Value); } } } private int FindEntry(TKey key) { if (key == null) { throw new ArgumentNullException("key"); } if (buckets != null) { int num = comparer.GetHashCode(key) & 0x7FFFFFFF; for (int num2 = buckets[num % buckets.Count]; num2 >= 0; num2 = entries[num2].Next) { if (entries[num2].HashCode == num && comparer.Equals(entries[num2].Key, key)) { return num2; } } } return -1; } public bool Remove(TKey key) { tracker.Dirty(); if (key == null) { throw new ArgumentNullException("key"); } if (buckets != null) { int num = comparer.GetHashCode(key) & 0x7FFFFFFF; int index = num % buckets.Count; int num2 = -1; for (int num3 = buckets[index]; num3 >= 0; num3 = entries[num3].Next) { if (entries[num3].HashCode == num && comparer.Equals(entries[num3].Key, key)) { if (num2 < 0) { buckets[index] = entries[num3].Next; } else { entries[num2] = entries[num2].WithNext(entries[num3].Next); } entries[num3] = new DictionaryEntry<TTypeResolver, TKey, TValue>(-1, freeList, default(TKey), default(TValue)); freeList = num3; freeCount++; return true; } num2 = num3; } } return false; } public bool TryGetValue(TKey key, out TValue value) { int num = FindEntry(key); if (num >= 0) { value = entries[num].Value; return true; } value = default(TValue); return false; } void ICollection<KeyValuePair<TKey, TValue>>.CopyTo(KeyValuePair<TKey, TValue>[] array, int index) { CopyTo(array, index); } public IEnumerator<KeyValuePair<TKey, TValue>> GetEnumerator() { int index = 0; while ((uint)index < (uint)count) { if (entries[index].HashCode >= 0) { yield return new KeyValuePair<TKey, TValue>(entries[index].Key, entries[index].Value); index++; } else { index++; } } } IEnumerator<KeyValuePair<TKey, TValue>> IEnumerable<KeyValuePair<TKey, TValue>>.GetEnumerator() { return GetEnumerator(); } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } private void Insert(TKey key, TValue value, bool add) { tracker.Dirty(); if (key == null) { throw new ArgumentNullException("key"); } int num = comparer.GetHashCode(key) & 0x7FFFFFFF; int index = num % buckets.Count; for (int num2 = buckets[index]; num2 >= 0; num2 = entries[num2].Next) { if (entries[num2].HashCode == num && comparer.Equals(entries[num2].Key, key)) { if (add) { throw new ArgumentException(SR.Format("Argument_AddingDuplicate", key)); } entries[num2] = entries[num2].WithValue(value); return; } } int num3; if (freeCount > 0) { num3 = freeList; freeList = entries[num3].Next; freeCount--; } else { if (count == entries.Count) { Resize(); index = num % buckets.Count; } num3 = count; count++; } entries[num3] = new DictionaryEntry<TTypeResolver, TKey, TValue>(num, buckets[index], key, value); buckets[index] = num3; } private void Resize() { Resize(HashHelpers.ExpandPrime(count)); } private void Resize(int newSize) { tracker.Dirty(); int[] array = new int[newSize]; for (int i = 0; i < array.Length; i++) { array[i] = -1; } DictionaryEntry<TTypeResolver, TKey, TValue>[] array2 = new DictionaryEntry<TTypeResolver, TKey, TValue>[newSize]; entries.CopyTo(array2, 0); for (int j = 0; j < count; j++) { if (array2[j].HashCode >= 0) { int num = array2[j].HashCode % newSize; array2[j] = array2[j].WithNext(array[num]); array[num] = j; } } buckets = array; entries = array2; } public bool CanDirectCopy() { if (tracker != null) { if (!tracker.IsDirty) { return originalBytes.Array != null; } return false; } return false; } public ArraySegment<byte> GetBufferReference() { return originalBytes; } public int Serialize(ref byte[] bytes, int offset) { if (CanDirectCopy()) { BinaryUtil.EnsureCapacity(ref bytes, offset, originalBytes.Count); Buffer.BlockCopy(originalBytes.Array, originalBytes.Offset, bytes, offset, originalBytes.Count); return originalBytes.Count; } if (freeCount != 0 || freeList != -1) { Resize(count); } Formatter<TTypeResolver, IList<int>> @default = Formatter<TTypeResolver, IList<int>>.Default; Formatter<TTypeResolver, IList<DictionaryEntry<TTypeResolver, TKey, TValue>>> default2 = Formatter<TTypeResolver, IList<DictionaryEntry<TTypeResolver, TKey, TValue>>>.Default; int num = offset; offset += 4; BinaryUtil.WriteInt32(ref bytes, offset, count); offset += 4; offset += @default.Serialize(ref bytes, offset, buckets); offset += default2.Serialize(ref bytes, offset, entries); int num2 = offset - num; BinaryUtil.WriteInt32(ref bytes, num, num2); return num2; } } public static class DictionaryEntry { public static DictionaryEntry<TTypeResolver, TKey, TValue> Create<TTypeResolver, TKey, TValue>(byte[] bytes, int offset, DirtyTracker tracker, out int byteSize) where TTypeResolver : ITypeResolver, new() { byteSize = 0; int hashCode = BinaryUtil.ReadInt32(ref bytes, offset); offset += 4; byteSize += 4; int next = BinaryUtil.ReadInt32(ref bytes, offset); offset += 4; byteSize += 4; int byteSize2; TKey key = Formatter<TTypeResolver, TKey>.Default.Deserialize(ref bytes, offset, tracker, out byteSize2); offset += byteSize2; byteSize += byteSize2; TValue value = Formatter<TTypeResolver, TValue>.Default.Deserialize(ref bytes, offset, tracker, out byteSize2); byteSize += byteSize2; return new DictionaryEntry<TTypeResolver, TKey, TValue>(hashCode, next, key, value); } } public struct DictionaryEntry<TTypeResolver, TKey, TValue> where TTypeResolver : ITypeResolver, new() { public readonly int HashCode; public readonly int Next; public readonly TKey Key; public readonly TValue Value; public DictionaryEntry(int hashCode, int next, TKey key, TValue value) { HashCode = hashCode; Next = next; Key = key; Value = value; } public DictionaryEntry<TTypeResolver, TKey, TValue> WithNext(int next) { return new DictionaryEntry<TTypeResolver, TKey, TValue>(HashCode, next, Key, Value); } public DictionaryEntry<TTypeResolver, TKey, TValue> WithValue(TValue value) { return new DictionaryEntry<TTypeResolver, TKey, TValue>(HashCode, Next, Key, value); } public int Serialize(ref byte[] bytes, int offset) { int num = 0; num += BinaryUtil.WriteInt32(ref bytes, offset, HashCode); num += BinaryUtil.WriteInt32(ref bytes, offset + num, Next); num += Formatter<TTypeResolver, TKey>.Default.Serialize(ref bytes, offset + num, Key); return num + Formatter<TTypeResolver, TValue>.Default.Serialize(ref bytes, offset + num, Value); } } public class LookupSegment<TTypeResolver, TKey, TElement> : ILookup<TKey, TElement>, IEnumerable<IGrouping<TKey, TElement>>, IEnumerable, ILazyLookup<TKey, TElement>, IZeroFormatterSegment where TTypeResolver : ITypeResolver, new() { private static TElement[] EmptyArray = new TElement[0]; private readonly IEqualityComparer<TKey> comparer; private int count; private IList<IList<GroupingSegment<TTypeResolver, TKey, TElement>>> groupings; internal DirtyTracker tracker; private ArraySegment<byte> originalBytes; public int Count => count; public IEnumerable<TElement> this[TKey key] { get { GroupingSegment<TTypeResolver, TKey, TElement> grouping = GetGrouping(key, create: false); if (grouping != null) { return grouping; } return EmptyArray; } } internal LookupSegment(ILookup<TKey, TElement> source) { comparer = ZeroFormatterEqualityComparer<TKey>.Default; groupings = new List<IList<GroupingSegment<TTypeResolver, TKey, TElement>>>(source.Count); for (int i = 0; i < source.Count; i++) { groupings.Add(null); } foreach (IGrouping<TKey, TElement> item in source) { GroupingSegment<TTypeResolver, TKey, TElement> grouping = GetGrouping(item.Key, create: true); foreach (TElement item2 in item) { grouping.Add(item2); } } count = source.Count; } public static LookupSegment<TTypeResolver, TKey, TElement> Create(DirtyTracker tracker, byte[] bytes, int offset, out int byteSize) { tracker = tracker.CreateChild(); int num = BinaryUtil.ReadInt32(ref bytes, offset); if (num == -1) { byteSize = 4; return null; } LookupSegment<TTypeResolver, TKey, TElement> lookupSegment = new LookupSegment<TTypeResolver, TKey, TElement>(); byteSize = num; lookupSegment.count = BinaryUtil.ReadInt32(ref bytes, offset + 4); Formatter<TTypeResolver, IList<IList<GroupingSegment<TTypeResolver, TKey, TElement>>>> @default = Formatter<TTypeResolver, IList<IList<GroupingSegment<TTypeResolver, TKey, TElement>>>>.Default; lookupSegment.groupings = @default.Deserialize(ref bytes, offset + 8, tracker, out var _); lookupSegment.tracker = tracker; lookupSegment.originalBytes = new ArraySegment<byte>(bytes, offset, byteSize); return lookupSegment; } private LookupSegment() { comparer = ZeroFormatterEqualityComparer<TKey>.Default; } public bool Contains(TKey key) { return GetGrouping(key, create: false) != null; } public IEnumerator<IGrouping<TKey, TElement>> GetEnumerator() { for (int i = 0; i < groupings.Count; i++) { IList<GroupingSegment<TTypeResolver, TKey, TElement>> list = groupings[i]; if (list == null) { continue; } foreach (GroupingSegment<TTypeResolver, TKey, TElement> item in list) { yield return item; } } } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } private GroupingSegment<TTypeResolver, TKey, TElement> GetGrouping(TKey key, bool create) { int num = ((key != null) ? (comparer.GetHashCode(key) & 0x7FFFFFFF) : 0); IList<GroupingSegment<TTypeResolver, TKey, TElement>> list = groupings[num % groupings.Count]; if (list != null) { for (int i = 0; i < list.Count; i++) { GroupingSegment<TTypeResolver, TKey, TElement> groupingSegment = list[i]; if (groupingSegment.hashCode == num && comparer.Equals(groupingSegment.key, key)) { return groupingSegment; } } } if (create) { int index = num % groupings.Count; GroupingSegment<TTypeResolver, TKey, TElement> groupingSegment2 = new GroupingSegment<TTypeResolver, TKey, TElement>(key, num); IList<GroupingSegment<TTypeResolver, TKey, TElement>> list2 = groupings[index]; if (list2 == null) { IList<GroupingSegment<TTypeResolver, TKey, TElement>> list4 = (groupings[index] = new List<GroupingSegment<TTypeResolver, TKey, TElement>>()); list2 = list4; } list2.Add(groupingSegment2); count++; return groupingSegment2; } return null; } public bool CanDirectCopy() { if (tracker != null) { if (!tracker.IsDirty) { return originalBytes.Array != null; } return false; } return false; } public ArraySegment<byte> GetBufferReference() { return originalBytes; } public int Serialize(ref byte[] bytes, int offset) { if (CanDirectCopy()) { BinaryUtil.EnsureCapacity(ref bytes, offset, originalBytes.Count); Buffer.BlockCopy(originalBytes.Array, originalBytes.Offset, bytes, offset, originalBytes.Count); return originalBytes.Count; } int num = 4; num += BinaryUtil.WriteInt32(ref bytes, offset + 4, count); Formatter<TTypeResolver, IList<IList<GroupingSegment<TTypeResolver, TKey, TElement>>>> @default = Formatter<TTypeResolver, IList<IList<GroupingSegment<TTypeResolver, TKey, TElement>>>>.Default; num += @default.Serialize(ref bytes, offset + 8, groupings); BinaryUtil.WriteInt32(ref bytes, offset, num); return num; } } public class GroupingSegment<TTypeResolver, TKey, TElement> : IGrouping<TKey, TElement>, IEnumerable<TElement>, IEnumerable, IList<TElement>, ICollection<TElement>, IZeroFormatterSegment where TTypeResolver : ITypeResolver, new() { internal TKey key; internal int hashCode; internal IList<TElement> elements; internal DirtyTracker tracker; private ArraySegment<byte> originalBytes; public TKey Key => key; int ICollection<TElement>.Count => elements.Count; bool ICollection<TElement>.IsReadOnly => true; TElement IList<TElement>.this[int index] { get { if (index < 0 || index >= elements.Count) { throw new ArgumentOutOfRangeException("index"); } return elements[index]; } set { throw new NotSupportedException(); } } internal static GroupingSegment<TTypeResolver, TKey, TElement> Create(DirtyTracker tracker, byte[] bytes, int offset, out int byteSize) { GroupingSegment<TTypeResolver, TKey, TElement> groupingSegment = new GroupingSegment<TTypeResolver, TKey, TElement>(); tracker = tracker.CreateChild(); groupingSegment.tracker = tracker; Formatter<TTypeResolver, TKey> @default = Formatter<TTypeResolver, TKey>.Default; Formatter<TTypeResolver, IList<TElement>> default2 = Formatter<TTypeResolver, IList<TElement>>.Default; groupingSegment.key = @default.Deserialize(ref bytes, offset, tracker, out var byteSize2); groupingSegment.hashCode = BinaryUtil.ReadInt32(ref bytes, offset + byteSize2); groupingSegment.elements = default2.Deserialize(ref bytes, offset + byteSize2 + 4, tracker, out var byteSize3); byteSize = byteSize2 + 4 + byteSize3; groupingSegment.originalBytes = new ArraySegment<byte>(bytes, offset, byteSize); return groupingSegment; } private GroupingSegment() { } internal GroupingSegment(TKey key, int hashCode) { this.key = key; this.hashCode = hashCode; elements = new List<TElement>(); } internal void Add(TElement element) { elements.Add(element); } public IEnumerator<TElement> GetEnumerator() { return elements.GetEnumerator(); } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } void ICollection<TElement>.Add(TElement item) { throw new NotSupportedException(); } void ICollection<TElement>.Clear() { throw new NotSupportedException(); } bool ICollection<TElement>.Contains(TElement item) { return elements.Contains(item); } void ICollection<TElement>.CopyTo(TElement[] array, int arrayIndex) { elements.CopyTo(array, arrayIndex); } bool ICollection<TElement>.Remove(TElement item) { throw new NotSupportedException(); } int IList<TElement>.IndexOf(TElement item) { return elements.IndexOf(item); } void IList<TElement>.Insert(int index, TElement item) { throw new NotSupportedException(); } void IList<TElement>.RemoveAt(int index) { throw new NotSupportedException(); } public bool CanDirectCopy() { if (tracker != null) { if (!tracker.IsDirty) { return originalBytes.Array != null; } return false; } return false; } public ArraySegment<byte> GetBufferReference() { return originalBytes; } public int Serialize(ref byte[] bytes, int offset) { if (CanDirectCopy()) { BinaryUtil.EnsureCapacity(ref bytes, offset, originalBytes.Count); Buffer.BlockCopy(originalBytes.Array, originalBytes.Offset, bytes, offset, originalBytes.Count); return originalBytes.Count; } Formatter<TTypeResolver, TKey> @default = Formatter<TTypeResolver, TKey>.Default; Formatter<TTypeResolver, IList<TElement>> default2 = Formatter<TTypeResolver, IList<TElement>>.Default; int num = offset; offset += @default.Serialize(ref bytes, offset, key); BinaryUtil.WriteInt32(ref bytes, offset, hashCode); offset += 4; offset += default2.Serialize(ref bytes, offset, elements); return offset - num; } } public abstract class ListSegment<TTypeResolver, T> : IList<T>, ICollection<T>, IEnumerable<T>, IEnumerable, IReadOnlyList<T>, IReadOnlyCollection<T> where TTypeResolver : ITypeResolver, new() { protected readonly ArraySegment<byte> originalBytes; protected readonly Formatter<TTypeResolver, T> formatter; protected DirtyTracker tracker; protected int length; protected T[] cache; protected bool[] isCached; protected bool isAllCached; public abstract T this[int index] { get; set; } public int Count => length; public bool IsReadOnly => false; internal ListSegment(DirtyTracker tracker, int length) { isAllCached = true; cache = new T[length]; this.length = length; this.tracker = tracker.CreateChild(); formatter = Formatter<TTypeResolver, T>.Default; } public ListSegment(DirtyTracker tracker, ArraySegment<byte> originalBytes, int length) { this.originalBytes = originalBytes; formatter = Formatter<TTypeResolver, T>.Default; this.length = length; this.tracker = tracker.CreateChild(); } protected void CreateCacheWhenNotYet() { if (cache == null) { cache = new T[length]; isCached = new bool[length]; } } protected void CacheAllWhenNotYet() { CreateCacheWhenNotYet(); if (isAllCached) { return; } ArraySegment<byte> arraySegment = originalBytes; byte[] bytes = arraySegment.Array; for (int i = 0; i < length; i++) { if (!isCached[i]) { int offset = GetOffset(i); cache[i] = formatter.Deserialize(ref bytes, offset, tracker, out var _); isCached[i] = true; } } isAllCached = true; } protected abstract int GetOffset(int index); public bool Contains(T item) { return IndexOf(item) != -1; } public int IndexOf(T item) { EqualityComparer<T> @default = EqualityComparer<T>.Default; for (int i = 0; i < length; i++) { if (@default.Equals(this[i], item)) { return i; } } return -1; } public void CopyTo(T[] array, int arrayIndex) { if (!isAllCached) { int num = 0; int num2 = arrayIndex; while (num < length) { array[num2] = this[num]; num++; num2++; } } else { Array.Copy(cache, 0, array, arrayIndex, length); } } public IEnumerator<T> GetEnumerator() { for (int i = 0; i < Count; i++) { yield return this[i]; } } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } public void Add(T item) { CacheAllWhenNotYet(); if (cache.Length == length) { Array.Resize(ref cache, (length == 0) ? 4 : (length * 2)); } cache[length] = item; length++; tracker.Dirty(); } public void Clear() { isAllCached = true; if (cache != null) { Array.Clear(cache, 0, cache.Length); } else { cache = new T[0]; } length = 0; tracker.Dirty(); } public void Insert(int index, T item) { if (index > length) { throw new ArgumentOutOfRangeException("index is out of range:" + index); } CacheAllWhenNotYet(); if (cache.Length == length) { Array.Resize(ref cache, (length == 0) ? 4 : (length * 2)); } if (index < length) { Array.Copy(cache, index, cache, index + 1, length - index); } cache[index] = item; length++; tracker.Dirty(); } public bool Remove(T item) { CacheAllWhenNotYet(); int num = IndexOf(item); if (num != -1) { RemoveAt(num); return true; } return false; } public void RemoveAt(int index) { tracker.Dirty(); if (index >= length) { throw new ArgumentOutOfRangeException("index is out of range:" + index); } CacheAllWhenNotYet(); length--; if (index < length) { Array.Copy(cache, index + 1, cache, index, length - index); } cache[length] = default(T); tracker.Dirty(); } } public class FixedListSegment<TTypeResolver, T> : ListSegment<TTypeResolver, T>, IZeroFormatterSegment where TTypeResolver : ITypeResolver, new() { private readonly int elementSize; public override T this[int index] { get { if (index > length) { throw new ArgumentOutOfRangeException("index > Count"); } if (!isAllCached) { ArraySegment<byte> arraySegment = originalBytes; byte[] bytes = arraySegment.Array; int offset = GetOffset(index); int byteSize; return formatter.Deserialize(ref bytes, offset, tracker, out byteSize); } return cache[index]; } set { if (index > base.Count) { throw new ArgumentOutOfRangeException("index > Count"); } if (!isAllCached) { ArraySegment<byte> arraySegment = originalBytes; byte[] bytes = arraySegment.Array; int num = 4 + elementSize * index; Formatter<TTypeResolver, T> obj = formatter; arraySegment = originalBytes; obj.Serialize(ref bytes, arraySegment.Offset + num, value); } else { cache[index] = value; tracker.Dirty(); } } } internal static FixedListSegment<TTypeResolver, T> Create(DirtyTracker tracker, byte[] bytes, int offset, out int byteSize) { int? num = Formatter<TTypeResolver, T>.Default.GetLength(); if (!num.HasValue) { throw new InvalidOperationException("T should be fixed length. Type: " + typeof(T).Name); } int num2 = BinaryUtil.ReadInt32(ref bytes, offset); if (num2 == -1) { byteSize = 4; return null; } byteSize = num.Value * num2 + 4; return new FixedListSegment<TTypeResolver, T>(tracker, new ArraySegment<byte>(bytes, offset, byteSize), num2); } private FixedListSegment(DirtyTracker tracker, ArraySegment<byte> originalBytes, int length) : base(tracker, originalBytes, length) { int? num = formatter.GetLength(); if (!num.HasValue) { throw new InvalidOperationException("T should be fixed length. Type: " + typeof(T).Name); } elementSize = num.Value; } protected override int GetOffset(int index) { ArraySegment<byte> arraySegment = originalBytes; return arraySegment.Offset + 4 + elementSize * index; } public bool CanDirectCopy() { if (tracker != null) { if (!tracker.IsDirty) { ArraySegment<byte> arraySegment = originalBytes; return arraySegment.Array != null; } return false; } return false; } public ArraySegment<byte> GetBufferReference() { return originalBytes; } public int Serialize(ref byte[] bytes, int offset) { if (CanDirectCopy()) { int offset2 = offset; ArraySegment<byte> arraySegment = originalBytes; BinaryUtil.EnsureCapacity(ref bytes, offset2, arraySegment.Count); arraySegment = originalBytes; byte[]? array = arraySegment.Array; arraySegment = originalBytes; int offset3 = arraySegment.Offset; byte[] dst = bytes; int dstOffset = offset; arraySegment = originalBytes; Buffer.BlockCopy(array, offset3, dst, dstOffset, arraySegment.Count); arraySegment = originalBytes; return arraySegment.Count; } int num = base.Count * elementSize + 4; if (bytes == null) { bytes = new byte[num]; } offset += BinaryUtil.WriteInt32(ref bytes, offset, base.Count); for (int i = 0; i < base.Count; i++) { offset += formatter.Serialize(ref bytes, offset, this[i]); } return num; } } public class VariableListSegment<TTypeResolver, T> : ListSegment<TTypeResolver, T>, IZeroFormatterSegment where TTypeResolver : ITypeResolver, new() { public override T this[int index] { get { if (index > length) { throw new ArgumentOutOfRangeException("index > Count"); } CreateCacheWhenNotYet(); if (!isAllCached && !isCached[index]) { ArraySegment<byte> arraySegment = originalBytes; byte[] bytes = arraySegment.Array; int offset = GetOffset(index); cache[index] = formatter.Deserialize(ref bytes, offset, tracker, out var _); isCached[index] = true; } return cache[index]; } set { if (index > base.Count) { throw new ArgumentOutOfRangeException("index > Count"); } CreateCacheWhenNotYet(); cache[index] = value; if (!isAllCached) { isCached[index] = true; } tracker.Dirty(); } } internal static VariableListSegment<TTypeResolver, T> Create(DirtyTracker tracker, byte[] bytes, int offset, out int byteSize) { byteSize = BinaryUtil.ReadInt32(ref bytes, offset); if (byteSize == -1) { byteSize = 4; return null; } int num = BinaryUtil.ReadInt32(ref bytes, offset + 4); return new VariableListSegment<TTypeResolver, T>(tracker, new ArraySegment<byte>(bytes, offset, byteSize), num); } private VariableListSegment(DirtyTracker tracker, ArraySegment<byte> originalBytes, int length) : base(tracker, originalBytes, length) { if (formatter.GetLength().HasValue) { throw new InvalidOperationException("T has fixed length, use FixedListSegement instead. Type: " + typeof(T).Name); } } protected override int GetOffset(int index) { ArraySegment<byte> arraySegment = originalBytes; byte[] bytes = arraySegment.Array; arraySegment = originalBytes; int num = BinaryUtil.ReadInt32(ref bytes, arraySegment.Offset + 8 + 4 * index); arraySegment = originalBytes; return arraySegment.Offset + num; } public bool CanDirectCopy() { if (tracker != null) { if (!tracker.IsDirty) { ArraySegment<byte> arraySegment = originalBytes; return arraySegment.Array != null; } return false; } return false; } public ArraySegment<byte> GetBufferReference() { return originalBytes; } public int Serialize(ref byte[] bytes, int offset) { if (CanDirectCopy()) { int offset2 = offset; ArraySegment<byte> arraySegment = originalBytes; BinaryUtil.EnsureCapacity(ref bytes, offset2, arraySegment.Count); arraySegment = originalBytes; byte[]? array = arraySegment.Array; arraySegment = originalBytes; int offset3 = arraySegment.Offset; byte[] dst = bytes; int dstOffset = offset; arraySegment = originalBytes; Buffer.BlockCopy(array, offset3, dst, dstOffset, arraySegment.Count); arraySegment = originalBytes; return arraySegment.Count; } int num = offset; int num2 = 0; offset = num + 8 + base.Count * 4; for (int i = 0; i < base.Count; i++) { T value = this[i]; int num3 = formatter.Serialize(ref bytes, offset, value); BinaryUtil.WriteInt32(ref bytes, num + 8 + num2 * 4, offset - num); offset += num3; num2++; } BinaryUtil.WriteInt32(ref bytes, num + 4, base.Count); int num4 = offset - num; BinaryUtil.WriteInt32(ref bytes, num, num4); return num4; } } internal enum SegmentState { Original, Cached, Dirty } public struct CacheSegment<TTypeResolver, T> : IZeroFormatterSegment where TTypeResolver : ITypeResolver, new() { private readonly DirtyTracker tracker; private SegmentState state; private ArraySegment<byte> serializedBytes; private T cached; public T Value { get { if (state == SegmentState.Original) { byte[] bytes = serializedBytes.Array; cached = Formatter<TTypeResolver, T>.Default.Deserialize(ref bytes, serializedBytes.Offset, tracker, out var _); state = SegmentState.Cached; return cached; } return cached; } set { tracker.Dirty(); state = SegmentState.Dirty; cached = value; } } public CacheSegment(DirtyTracker tracker, ArraySegment<byte> originalBytes) { this.tracker = tracker.CreateChild(); state = SegmentState.Original; serializedBytes = originalBytes; cached = default(T); if (originalBytes.Array == null) { state = SegmentState.Dirty; } } public bool CanDirectCopy() { if (Formatter<TTypeResolver, T>.Default.NoUseDirtyTracker) { if (state != 0) { return state == SegmentState.Cached; } return true; } return state == SegmentState.Original; } public ArraySegment<byte> GetBufferReference() { return serializedBytes; } public int Serialize(ref byte[] targetBytes, int offset) { if (targetBytes == null) { throw new ArgumentNullException("targetBytes"); } if (!CanDirectCopy()) { Formatter<TTypeResolver, T> @default = Formatter<TTypeResolver, T>.Default; if (!@default.NoUseDirtyTracker) { return @default.Serialize(ref targetBytes, offset, Value); } byte[] bytes = null; @default.Serialize(ref bytes, 0, Value); serializedBytes = new ArraySegment<byte>(bytes, 0, bytes.Length); state = SegmentState.Cached; } BinaryUtil.EnsureCapacity(ref targetBytes, offset, serializedBytes.Count); Buffer.BlockCopy(serializedBytes.Array, serializedBytes.Offset, targetBytes, offset, serializedBytes.Count); return serializedBytes.Count; } } public static class ObjectSegmentHelper { public static int GetByteSize(ArraySegment<byte> originalBytes) { byte[] bytes = originalBytes.Array; return BinaryUtil.ReadInt32(ref bytes, originalBytes.Offset); } public static int GetOffset(ArraySegment<byte> originalBytes, int index, int lastIndex, DirtyTracker tracker) { if (index > lastIndex) { return -1; } byte[] bytes = originalBytes.Array; int num = BinaryUtil.ReadInt32(ref bytes, originalBytes.Offset + 8 + 4 * index); if (num == 0) { return -1; } return originalBytes.Offset + num; } public static ArraySegment<byte> GetSegment(ArraySegment<byte> originalBytes, int index, int lastIndex, DirtyTracker tracker) { int offset = GetOffset(originalBytes, index, lastIndex, tracker); if (offset == -1) { return default(ArraySegment<byte>); } int num = originalBytes.Offset + originalBytes.Count; return new ArraySegment<byte>(originalBytes.Array, offset, num - offset); } public static T DeserializeSegment<TTypeResolver, T>(ArraySegment<byte> originalBytes, int index, int lastIndex, DirtyTracker tracker) where TTypeResolver : ITypeResolver, new() { int offset = GetOffset(originalBytes, index, lastIndex, tracker); if (offset == -1) { return default(T); } byte[] bytes = originalBytes.Array; int byteSize; return Formatter<TTypeResolver, T>.Default.Deserialize(ref bytes, offset, tracker, out byteSize); } public static int SerializeFixedLength<TTypeResolver, T>(ref byte[] targetBytes, int startOffset, int offset, int index, int lastIndex, ArraySegment<byte> originalBytes, byte[] extraBytes, DirtyTracker tracker) where TTypeResolver : ITypeResolver, new() { BinaryUtil.WriteInt32(ref targetBytes, startOffset + (8 + 4 * index), offset - startOffset); int? length = Formatter<TTypeResolver, T>.Default.GetLength(); BinaryUtil.EnsureCapacity(ref targetBytes, offset, length.Value); int offset2 = GetOffset(originalBytes, index, lastIndex, tracker); if (offset2 != -1) { Buffer.BlockCopy(originalBytes.Array, offset2, targetBytes, offset, length.Value); } else { int extraBytesOffset = GetExtraBytesOffset(extraBytes, lastIndex, index); Buffer.BlockCopy(extraBytes, extraBytesOffset, targetBytes, offset, length.Value); } return length.Value; } public static int SerializeSegment<TTypeResolver, T>(ref byte[] targetBytes, int startOffset, int offset, int index, T segment) where TTypeResolver : ITypeResolver, new() { BinaryUtil.WriteInt32(ref targetBytes, startOffset + (8 + 4 * index), offset - startOffset); return Formatter<TTypeResolver, T>.Default.Serialize(ref targetBytes, offset, segment); } public static int SerializeCacheSegment<TTypeResolver, T>(ref byte[] targetBytes, int startOffset, int offset, int index, ref CacheSegment<TTypeResolver, T> segment) where TTypeResolver : ITypeResolver, new() { BinaryUtil.WriteInt32(ref targetBytes, startOffset + (8 + 4 * index), offset - startOffset); return segment.Serialize(ref targetBytes, offset); } public static T GetFixedProperty<TTypeResolver, T>(ArraySegment<byte> bytes, int index, int lastIndex, byte[] extraBytes, DirtyTracker tracker) where TTypeResolver : ITypeResolver, new() { if (index <= lastIndex) { byte[] bytes2 = bytes.Array; int byteSize; return Formatter<TTypeResolver, T>.Default.Deserialize(ref bytes2, GetOffset(bytes, index, lastIndex, tracker), tracker, out byteSize); } int extraBytesOffset = GetExtraBytesOffset(extraBytes, lastIndex, index); int byteSize2; return Formatter<TTypeResolver, T>.Default.Deserialize(ref extraBytes, extraBytesOffset, tracker, out byteSize2); } public static void SetFixedProperty<TTypeResolver, T>(ArraySegment<byte> bytes, int index, int lastIndex, byte[] extraBytes, T value, DirtyTracker tracker) where TTypeResolver : ITypeResolver, new() { if (index <= lastIndex) { byte[] bytes2 = bytes.Array; Formatter<TTypeResolver, T>.Default.Serialize(ref bytes2, GetOffset(bytes, index, lastIndex, tracker), value); } else { int extraBytesOffset = GetExtraBytesOffset(extraBytes, lastIndex, index); Formatter<TTypeResolver, T>.Default.Serialize(ref extraBytes, extraBytesOffset, value); } } public static int WriteSize(ref byte[] targetBytes, int startOffset, int lastOffset, int lastIndex) { BinaryUtil.WriteInt32(ref targetBytes, startOffset + 4, lastIndex); int num = lastOffset - startOffset; BinaryUtil.WriteInt32Unsafe(ref targetBytes, startOffset, num); return num; } public static int DirectCopyAll(ArraySegment<byte> originalBytes, ref byte[] targetBytes, int targetOffset) { byte[] bytes = originalBytes.Array; int num = BinaryUtil.ReadInt32(ref bytes, originalBytes.Offset); BinaryUtil.EnsureCapacity(ref targetBytes, targetOffset, num); Buffer.BlockCopy(bytes, originalBytes.Offset, targetBytes, targetOffset, num); return num; } private static int GetExtraBytesOffset(byte[] extraBytes, int binaryLastIndex, int index) { int offset = (index - binaryLastIndex - 1) * 4; return BinaryUtil.ReadInt32(ref extraBytes, offset); } public static byte[] CreateExtraFixedBytes(int binaryLastIndex, int schemaLastIndex, int[] elementSizes) { if (binaryLastIndex < schemaLastIndex) { int num = (schemaLastIndex - binaryLastIndex) * 4; int num2 = elementSizes.Sum(); byte[] bytes = new byte[num + num2]; int num3 = num + 4; for (int i = binaryLastIndex + 1; i < elementSizes.Length; i++) { if (elementSizes[i] != 0) { BinaryUtil.WriteInt32(ref bytes, (i - binaryLastIndex - 1) * 4, num3); num3 += elementSizes[i]; } } return bytes; } return null; } public static int SerializeFromFormatter<TTypeResolver, T>(ref byte[] bytes, int startOffset, int offset, int index, T value) where TTypeResolver : ITypeResolver, new() { BinaryUtil.WriteInt32(ref bytes, startOffset + (8 + 4 * index), offset - startOffset); return Formatter<TTypeResolver, T>.Default.Serialize(ref bytes, offset, value); } public static Exception GetException1(string msgFormat, object o) { return new Exception(string.Format(msgFormat, o)); } public static bool IsLazySegment(Type type) { TypeInfo typeInfo = type.GetTypeInfo(); if (typeInfo.IsClass && ((MemberInfo)typeInfo).GetCustomAttributes<ZeroFormattableAttribute>().Any()) { return true; } if (typeInfo.IsGenericType) { Type genericTypeDefinition = typeInfo.GetGenericTypeDefinition(); if (genericTypeDefinition == typeof(IList<>)) { return true; } if (genericTypeDefinition == typeof(ILazyLookup<, >)) { return true; } if (genericTypeDefinition == typeof(ILazyDictionary<, >)) { return true; } if (genericTypeDefinition == typeof(ILazyReadOnlyDictionary<, >)) { return true; } } return false; } } internal static class DynamicAssemblyHolder { public const string ModuleName = "ZeroFormatter.DynamicObjectSegments"; private static readonly DynamicAssembly assembly; public static ModuleBuilder Module => assembly.ModuleBuilder; static DynamicAssemblyHolder() { assembly = new DynamicAssembly("ZeroFormatter.DynamicObjectSegments"); } } internal static class DynamicObjectSegmentBuilder<TTypeResolver, T> where TTypeResolver : ITypeResolver, new() { private class PropertyTuple { public int Index; public PropertyInfo PropertyInfo; public FieldInfo SegmentField; public bool IsCacheSegment; public bool IsFixedSize; public int FixedSize; } private static readonly MethodInfo ArraySegmentArrayGet = typeof(ArraySegment<byte>).GetTypeInfo().GetProperty("Array").GetGetMethod(); private static readonly MethodInfo ArraySegmentOffsetGet = typeof(ArraySegment<byte>).GetTypeInfo().GetProperty("Offset").GetGetMethod(); private static readonly MethodInfo ReadInt32 = typeof(BinaryUtil).GetTypeInfo().GetMethod("ReadInt32"); private static readonly MethodInfo CreateChild = typeof(DirtyTracker).GetTypeInfo().GetMethod("CreateChild"); private static readonly MethodInfo Dirty = typeof(DirtyTracker).GetTypeInfo().GetMethod("Dirty"); private static readonly MethodInfo IsDirty = typeof(DirtyTracker).GetTypeInfo().GetProperty("IsDirty").GetGetMethod(); private static readonly MethodInfo GetSegment = typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("GetSegment"); private static readonly MethodInfo GetOffset = typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("GetOffset"); private static readonly Lazy<TypeInfo> lazyBuild = new Lazy<TypeInfo>(() => Build(), isThreadSafe: true); public static TypeInfo GetProxyType() { return lazyBuild.Value; } private static TypeInfo Build() { return GenerateObjectSegmentImplementation(DynamicAssemblyHolder.Module); } private static TypeInfo GenerateObjectSegmentImplementation(ModuleBuilder moduleBuilder) { TypeBuilder typeBuilder = moduleBuilder.DefineType("ZeroFormatter.DynamicObjectSegments." + typeof(TTypeResolver).FullName.Replace(".", "_") + "." + typeof(T).FullName, TypeAttributes.Public, typeof(T)); FieldBuilder originalBytesField = typeBuilder.DefineField("<>_originalBytes", typeof(ArraySegment<byte>), FieldAttributes.Private | FieldAttributes.InitOnly); FieldBuilder trackerField = typeBuilder.DefineField("<>_tracker", typeof(DirtyTracker), FieldAttributes.Private | FieldAttributes.InitOnly); FieldBuilder fieldBuilder = typeBuilder.DefineField("<>_binaryLastIndex", typeof(int), FieldAttributes.Private | FieldAttributes.InitOnly); FieldBuilder fieldBuilder2 = typeBuilder.DefineField("<>_extraFixedBytes", typeof(byte[]), FieldAttributes.Private | FieldAttributes.InitOnly); PropertyTuple[] propertiesWithVerify = GetPropertiesWithVerify(typeBuilder); BuildConstructor(typeBuilder, originalBytesField, trackerField, fieldBuilder, fieldBuilder2, propertiesWithVerify); PropertyTuple[] array = propertiesWithVerify; foreach (PropertyTuple propertyTuple in array) { if (propertyTuple.IsFixedSize) { BuildFixedProperty(typeBuilder, originalBytesField, trackerField, fieldBuilder, fieldBuilder2, propertyTuple); } else if (propertyTuple.IsCacheSegment) { BuildCacheSegmentProperty(typeBuilder, originalBytesField, trackerField, propertyTuple); } else { BuildSegmentProperty(typeBuilder, originalBytesField, trackerField, propertyTuple); } } BuildInterfaceMethod(typeBuilder, originalBytesField, trackerField, fieldBuilder, fieldBuilder2, propertiesWithVerify); return typeBuilder.CreateTypeInfo(); } private static PropertyTuple[] GetPropertiesWithVerify(TypeBuilder typeBuilder) { Type typeFromHandle = typeof(TTypeResolver); List<PropertyTuple> list = new List<PropertyTuple>(); Tuple<int, EmittableMemberInfo>[] members = DynamicObjectDescriptor.GetMembers(typeFromHandle, typeof(T), isClass: true); foreach (Tuple<int, EmittableMemberInfo> obj in members) { EmittableMemberInfo item = obj.Item2; int item2 = obj.Item1; IFormatter formatter = (IFormatter)typeof(Formatter<, >).MakeGenericType(typeFromHandle, item.MemberType).GetTypeInfo().GetProperty("Default") .GetValue(null, Type.EmptyTypes); if (formatter == null) { throw new InvalidOperationException("Circular reference does not supported. " + typeof(T).Name + "." + item.Name); } if (!formatter.GetLength().HasValue) { if (!ObjectSegmentHelper.IsLazySegment(item.MemberType)) { FieldBuilder segmentField = typeBuilder.DefineField("<>_" + item.Name, typeof(CacheSegment<, >).MakeGenericType(typeFromHandle, item.MemberType), FieldAttributes.Private); list.Add(new PropertyTuple { Index = item2, PropertyInfo = item.PropertyInfoUnsafe, IsFixedSize = false, SegmentField = segmentField, IsCacheSegment = true }); } else { FieldBuilder segmentField2 = typeBuilder.DefineField("<>_" + item.Name, item.MemberType, FieldAttributes.Private); list.Add(new PropertyTuple { Index = item2, PropertyInfo = item.PropertyInfoUnsafe, IsFixedSize = false, SegmentField = segmentField2 }); } } else { list.Add(new PropertyTuple { Index = item2, PropertyInfo = item.PropertyInfoUnsafe, IsFixedSize = true, FixedSize = formatter.GetLength().Value }); } } return list.OrderBy((PropertyTuple x) => x.Index).ToArray(); } private static void BuildConstructor(TypeBuilder type, FieldInfo originalBytesField, FieldInfo trackerField, FieldInfo lastIndexField, FieldInfo extraFixedBytes, PropertyTuple[] properties) { MethodBuilder methodBuilder = type.DefineMethod(".ctor", MethodAttributes.Public | MethodAttributes.HideBySig); ConstructorInfo constructor = typeof(T).GetTypeInfo().GetConstructor(Type.EmptyTypes); methodBuilder.SetReturnType(typeof(void)); methodBuilder.SetParameters(typeof(DirtyTracker), typeof(ArraySegment<byte>)); ILGenerator iLGenerator = methodBuilder.GetILGenerator(); iLGenerator.DeclareLocal(typeof(byte[])); iLGenerator.DeclareLocal(typeof(int)); iLGenerator.DeclareLocal(typeof(int[])); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Call, constructor); iLGenerator.Emit(OpCodes.Ldarga_S, (byte)2); iLGenerator.Emit(OpCodes.Call, ArraySegmentArrayGet); iLGenerator.Emit(OpCodes.Stloc_0); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldarg_2); iLGenerator.Emit(OpCodes.Stfld, originalBytesField); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldarg_1); iLGenerator.Emit(OpCodes.Callvirt, CreateChild); iLGenerator.Emit(OpCodes.Dup); iLGenerator.Emit(OpCodes.Starg_S, (byte)1); iLGenerator.Emit(OpCodes.Stfld, trackerField); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldloca_S, (byte)0); iLGenerator.Emit(OpCodes.Ldarga_S, (byte)2); iLGenerator.Emit(OpCodes.Call, ArraySegmentOffsetGet); iLGenerator.Emit(OpCodes.Ldc_I4_4); iLGenerator.Emit(OpCodes.Add); iLGenerator.Emit(OpCodes.Call, ReadInt32); iLGenerator.Emit(OpCodes.Stfld, lastIndexField); int num = properties.Select((PropertyTuple x) => x.Index).LastOrDefault(); Dictionary<int, int> dictionary = properties.Where((PropertyTuple x) => x.IsFixedSize).ToDictionary((PropertyTuple x) => x.Index, (PropertyTuple x) => x.FixedSize); int[] array = new int[num + 1]; for (int i = 0; i < num + 1; i++) { if (!dictionary.TryGetValue(i, out array[i])) { array[i] = 0; } } EmitNewArray(iLGenerator, array); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, lastIndexField); iLGenerator.Emit(OpCodes.Ldc_I4, num); iLGenerator.Emit(OpCodes.Ldloc_2); iLGenerator.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("CreateExtraFixedBytes")); iLGenerator.Emit(OpCodes.Stfld, extraFixedBytes); foreach (PropertyTuple propertyTuple in properties) { if (!propertyTuple.IsFixedSize) { if (propertyTuple.IsCacheSegment) { AssignCacheSegment(iLGenerator, propertyTuple.Index, trackerField, lastIndexField, propertyTuple.SegmentField); } else { AssignSegment(iLGenerator, propertyTuple.Index, trackerField, lastIndexField, propertyTuple.SegmentField); } } } iLGenerator.Emit(OpCodes.Ret); } private static void EmitNewArray(ILGenerator il, int[] array) { il.Emit(OpCodes.Ldc_I4, array.Length); il.Emit(OpCodes.Newarr, typeof(int)); il.Emit(OpCodes.Stloc_2); for (int i = 0; i < array.Length; i++) { il.Emit(OpCodes.Ldloc_2); il.Emit(OpCodes.Ldc_I4, i); il.Emit(OpCodes.Ldc_I4, array[i]); il.Emit(OpCodes.Stelem_I4); } } private static void AssignCacheSegment(ILGenerator il, int index, FieldInfo tracker, FieldInfo lastIndex, FieldInfo field) { il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, tracker); il.Emit(OpCodes.Ldarg_2); il.Emit(OpCodes.Ldc_I4, index); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, lastIndex); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, tracker); il.Emit(OpCodes.Call, GetSegment); il.Emit(OpCodes.Newobj, field.FieldType.GetTypeInfo().GetConstructors().First()); il.Emit(OpCodes.Stfld, field); } private static void AssignSegment(ILGenerator il, int index, FieldInfo tracker, FieldInfo lastIndex, FieldInfo field) { il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_2); il.Emit(OpCodes.Ldc_I4, index); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, lastIndex); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, tracker); il.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("DeserializeSegment").MakeGenericMethod(typeof(TTypeResolver), field.FieldType)); il.Emit(OpCodes.Stfld, field); } private static void BuildFixedProperty(TypeBuilder type, FieldInfo originalBytesField, FieldInfo trackerField, FieldInfo binaryLastIndex, FieldInfo extraBytes, PropertyTuple property) { PropertyBuilder propertyBuilder = type.DefineProperty(property.PropertyInfo.Name, property.PropertyInfo.Attributes, property.PropertyInfo.PropertyType, Type.EmptyTypes); MethodInfo getMethod = property.PropertyInfo.GetGetMethod(); if (getMethod != null) { MethodBuilder methodBuilder = type.DefineMethod(getMethod.Name, getMethod.Attributes & ~MethodAttributes.VtableLayoutMask, getMethod.ReturnType, Type.EmptyTypes); ILGenerator iLGenerator = methodBuilder.GetILGenerator(); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, originalBytesField); iLGenerator.Emit(OpCodes.Ldc_I4, property.Index); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, binaryLastIndex); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, extraBytes); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, trackerField); iLGenerator.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("GetFixedProperty").MakeGenericMethod(typeof(TTypeResolver), getMethod.ReturnType)); iLGenerator.Emit(OpCodes.Ret); propertyBuilder.SetGetMethod(methodBuilder); } MethodInfo setMethod = property.PropertyInfo.GetSetMethod(); if (setMethod != null) { MethodBuilder methodBuilder2 = type.DefineMethod(setMethod.Name, setMethod.Attributes & ~MethodAttributes.VtableLayoutMask, null, new Type[1] { setMethod.GetParameters()[0].ParameterType }); ILGenerator iLGenerator2 = methodBuilder2.GetILGenerator(); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldfld, originalBytesField); iLGenerator2.Emit(OpCodes.Ldc_I4, property.Index); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldfld, binaryLastIndex); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldfld, extraBytes); iLGenerator2.Emit(OpCodes.Ldarg_1); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldfld, trackerField); iLGenerator2.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("SetFixedProperty").MakeGenericMethod(typeof(TTypeResolver), getMethod.ReturnType)); iLGenerator2.Emit(OpCodes.Ret); propertyBuilder.SetSetMethod(methodBuilder2); } } private static void BuildCacheSegmentProperty(TypeBuilder type, FieldInfo originalBytesField, FieldInfo trackerField, PropertyTuple property) { PropertyBuilder propertyBuilder = type.DefineProperty(property.PropertyInfo.Name, property.PropertyInfo.Attributes, property.PropertyInfo.PropertyType, Type.EmptyTypes); MethodInfo getMethod = property.PropertyInfo.GetGetMethod(); if (getMethod != null) { MethodBuilder methodBuilder = type.DefineMethod(getMethod.Name, getMethod.Attributes & ~MethodAttributes.VtableLayoutMask, getMethod.ReturnType, Type.EmptyTypes); ILGenerator iLGenerator = methodBuilder.GetILGenerator(); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldflda, property.SegmentField); iLGenerator.Emit(OpCodes.Call, property.SegmentField.FieldType.GetTypeInfo().GetProperty("Value").GetGetMethod()); iLGenerator.Emit(OpCodes.Ret); propertyBuilder.SetGetMethod(methodBuilder); } MethodInfo setMethod = property.PropertyInfo.GetSetMethod(); if (setMethod != null) { MethodBuilder methodBuilder2 = type.DefineMethod(setMethod.Name, setMethod.Attributes & ~MethodAttributes.VtableLayoutMask, null, new Type[1] { setMethod.GetParameters()[0].ParameterType }); ILGenerator iLGenerator2 = methodBuilder2.GetILGenerator(); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldflda, property.SegmentField); iLGenerator2.Emit(OpCodes.Ldarg_1); iLGenerator2.Emit(OpCodes.Call, property.SegmentField.FieldType.GetTypeInfo().GetProperty("Value").GetSetMethod()); iLGenerator2.Emit(OpCodes.Ret); propertyBuilder.SetSetMethod(methodBuilder2); } } private static void BuildSegmentProperty(TypeBuilder type, FieldInfo originalBytesField, FieldInfo trackerField, PropertyTuple property) { PropertyBuilder propertyBuilder = type.DefineProperty(property.PropertyInfo.Name, property.PropertyInfo.Attributes, property.PropertyInfo.PropertyType, Type.EmptyTypes); MethodInfo getMethod = property.PropertyInfo.GetGetMethod(); if (getMethod != null) { MethodBuilder methodBuilder = type.DefineMethod(getMethod.Name, getMethod.Attributes & ~MethodAttributes.VtableLayoutMask, getMethod.ReturnType, Type.EmptyTypes); ILGenerator iLGenerator = methodBuilder.GetILGenerator(); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, property.SegmentField); iLGenerator.Emit(OpCodes.Ret); propertyBuilder.SetGetMethod(methodBuilder); } MethodInfo setMethod = property.PropertyInfo.GetSetMethod(); if (setMethod != null) { MethodBuilder methodBuilder2 = type.DefineMethod(setMethod.Name, setMethod.Attributes & ~MethodAttributes.VtableLayoutMask, null, new Type[1] { setMethod.GetParameters()[0].ParameterType }); ILGenerator iLGenerator2 = methodBuilder2.GetILGenerator(); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldfld, trackerField); iLGenerator2.Emit(OpCodes.Callvirt, Dirty); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldarg_1); iLGenerator2.Emit(OpCodes.Stfld, property.SegmentField); iLGenerator2.Emit(OpCodes.Ret); propertyBuilder.SetSetMethod(methodBuilder2); } } private static void BuildInterfaceMethod(TypeBuilder type, FieldInfo originalBytesField, FieldInfo trackerField, FieldInfo binaryLastIndexField, FieldInfo extraBytes, PropertyTuple[] properties) { type.AddInterfaceImplementation(typeof(IZeroFormatterSegment)); ILGenerator iLGenerator = type.DefineMethod("CanDirectCopy", MethodAttributes.Public | MethodAttributes.Final | MethodAttributes.Virtual, typeof(bool), Type.EmptyTypes).GetILGenerator(); iLGenerator.Emit(OpCodes.Ldarg_0); iLGenerator.Emit(OpCodes.Ldfld, trackerField); iLGenerator.Emit(OpCodes.Callvirt, IsDirty); iLGenerator.Emit(OpCodes.Ldc_I4_0); iLGenerator.Emit(OpCodes.Ceq); iLGenerator.Emit(OpCodes.Ret); ILGenerator iLGenerator2 = type.DefineMethod("GetBufferReference", MethodAttributes.Public | MethodAttributes.Final | MethodAttributes.Virtual, typeof(ArraySegment<byte>), Type.EmptyTypes).GetILGenerator(); iLGenerator2.Emit(OpCodes.Ldarg_0); iLGenerator2.Emit(OpCodes.Ldfld, originalBytesField); iLGenerator2.Emit(OpCodes.Ret); ILGenerator iLGenerator3 = type.DefineMethod("Serialize", MethodAttributes.Public | MethodAttributes.Final | MethodAttributes.Virtual, typeof(int), new Type[2] { typeof(byte[]).MakeByRefType(), typeof(int) }).GetILGenerator(); iLGenerator3.DeclareLocal(typeof(int)); Label label = iLGenerator3.DefineLabel(); Label label2 = iLGenerator3.DefineLabel(); iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, extraBytes); iLGenerator3.Emit(OpCodes.Brtrue, label); iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, trackerField); iLGenerator3.Emit(OpCodes.Callvirt, IsDirty); iLGenerator3.Emit(OpCodes.Brfalse, label2); int num = properties.Select((PropertyTuple x) => x.Index).LastOrDefault(); int arg = 8 + 4 * (num + 1); iLGenerator3.MarkLabel(label); iLGenerator3.Emit(OpCodes.Ldarg_2); iLGenerator3.Emit(OpCodes.Stloc_0); iLGenerator3.Emit(OpCodes.Ldarg_2); iLGenerator3.Emit(OpCodes.Ldc_I4, arg); iLGenerator3.Emit(OpCodes.Add); foreach (PropertyTuple propertyTuple in properties) { iLGenerator3.Emit(OpCodes.Starg_S, (byte)2); iLGenerator3.Emit(OpCodes.Ldarg_2); iLGenerator3.Emit(OpCodes.Ldarg_1); iLGenerator3.Emit(OpCodes.Ldloc_0); iLGenerator3.Emit(OpCodes.Ldarg_2); iLGenerator3.Emit(OpCodes.Ldc_I4, propertyTuple.Index); if (propertyTuple.IsFixedSize) { iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, binaryLastIndexField); iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, originalBytesField); iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, extraBytes); iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, trackerField); iLGenerator3.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("SerializeFixedLength").MakeGenericMethod(typeof(TTypeResolver), propertyTuple.PropertyInfo.PropertyType)); } else if (propertyTuple.IsCacheSegment) { iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldflda, propertyTuple.SegmentField); iLGenerator3.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("SerializeCacheSegment").MakeGenericMethod(typeof(TTypeResolver), propertyTuple.PropertyInfo.PropertyType)); } else { iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, propertyTuple.SegmentField); iLGenerator3.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("SerializeSegment").MakeGenericMethod(typeof(TTypeResolver), propertyTuple.PropertyInfo.PropertyType)); } iLGenerator3.Emit(OpCodes.Add); } iLGenerator3.Emit(OpCodes.Starg_S, (byte)2); iLGenerator3.Emit(OpCodes.Ldarg_1); iLGenerator3.Emit(OpCodes.Ldloc_0); iLGenerator3.Emit(OpCodes.Ldarg_2); iLGenerator3.Emit(OpCodes.Ldc_I4, num); iLGenerator3.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("WriteSize")); iLGenerator3.Emit(OpCodes.Ret); iLGenerator3.MarkLabel(label2); iLGenerator3.Emit(OpCodes.Ldarg_0); iLGenerator3.Emit(OpCodes.Ldfld, originalBytesField); iLGenerator3.Emit(OpCodes.Ldarg_1); iLGenerator3.Emit(OpCodes.Ldarg_2); iLGenerator3.Emit(OpCodes.Call, typeof(ObjectSegmentHelper).GetTypeInfo().GetMethod("DirectCopyAll")); iLGenerator3.Emit(OpCodes.Ret); } } } namespace ZeroFormatter.Internal { public static class BinaryUtil { internal static class Timestamp { internal static readonly DateTime UnixEpoch = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc); internal const long BclSecondsAtUnixEpoch = 62135596800L; internal const long UnixSecondsAtBclMaxValue = 253402300799L; internal const long UnixSecondsAtBclMinValue = -62135596800L; internal const int MaxNanos = 999999999; internal static bool IsNormalized(long seconds, int nanoseconds) { if (nanoseconds >= 0 && nanoseconds <= 999999999 && seconds >= -62135596800L) { return seconds <= 253402300799L; } return false; } } internal static class Duration { public const int NanosecondsPerSecond = 1000000000; public const int NanosecondsPerTick = 100; public const long MaxSeconds = 315576000000L; public const long MinSeconds = -315576000000L; internal const int MaxNanoseconds = 999999999; internal const int MinNanoseconds = -999999999; internal static bool IsNormalized(long seconds, int nanoseconds) { if (seconds < -315576000000L || seconds > 315576000000L || nanoseconds < -999999999 || nanoseconds > 999999999) { return false; } return Math.Sign(seconds) * Math.Sign(nanoseconds) != -1; } } static BinaryUtil() { if (!BitConverter.IsLittleEndian) { throw new Exception("Currently ZeroFormatter only supports Little-Endian environments. If you need supports, please report your envitonments to https://github.com/neuecc/ZeroFormatter/issues ."); } } public static void EnsureCapacity(ref byte[] bytes, int offset, int appendLength) { int num = offset + appendLength; if (bytes == null) { bytes = new byte[num]; return; } int num2 = bytes.Length; if (num <= num2) { return; } int num3 = num; if (num3 < 256) { num3 = 256; FastResize(ref bytes, num3); return; } if (num3 < num2 * 2) { num3 = num2 * 2; } FastResize(ref bytes, num3); } public static void FastResize(ref byte[] array, int newSize) { if (newSize < 0) { throw new ArgumentOutOfRangeException("newSize"); } byte[] array2 = array; if (array2 == null) { array = new byte[newSize]; } else if (array2.Length != newSize) { byte[] array3 = new byte[newSize]; Buffer.BlockCopy(array2, 0, array3, 0, (array2.Length > newSize) ? newSize : array2.Length); array = array3; } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static int WriteBoolean(ref byte[] bytes, int offset, bool value) { EnsureCapacity(ref bytes, offset, 1); bytes[offset] = (byte)(value ? 1u : 0u); return 1; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void WriteBooleanUnsafe(ref byte[] bytes, int offset, bool value) { bytes[offset] = (byte)(value ? 1u : 0u); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void WriteBooleanTrueUnsafe(ref byte[] bytes, int offset) { bytes[offset] = 1; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void WriteBooleanFalseUnsafe(ref byte[] bytes, int offset) { bytes[offset] = 0; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static bool ReadBoolean(ref byte[] bytes, int offset) { if (bytes[offset] != 0) { return true; } return false; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static int WriteByte(ref byte[] bytes, int offset, byte value) { EnsureCapacity(ref bytes, offset, 1); bytes[offset] = value; return 1; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static byte ReadByte(ref byte[] bytes, int offset) { return bytes[offset]; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static int WriteBytes(ref byte[] bytes, int offset, byte[] value) { EnsureCapacity(ref bytes, offset, value.Length); Buffer.BlockCopy(value, 0, bytes, offset, value.Length); return value.Length; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static byte[] ReadBytes(ref byte[] bytes, int offset, int count) { byte[] array = new byte[count]; Buffer.BlockCopy(bytes, offset, array, 0, count); return array; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static int WriteSByte(ref byte[] bytes, int offset, sbyte value) { EnsureCapacity(ref bytes, offset, 1); bytes[offset] = (byte)value; return 1; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static sbyte ReadSByte(ref byte[] bytes, int offset) { return (sbyte)bytes[offset]; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteSingle(ref byte[] bytes, int offset, float value) { EnsureCapacity(ref bytes, offset, 4); if (offset % 4 == 0) { fixed (byte* ptr = bytes) { *(float*)(ptr + offset) = value; } } else { uint num = *(uint*)(&value); bytes[offset] = (byte)num; bytes[offset + 1] = (byte)(num >> 8); bytes[offset + 2] = (byte)(num >> 16); bytes[offset + 3] = (byte)(num >> 24); } return 4; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static float ReadSingle(ref byte[] bytes, int offset) { if (offset % 4 == 0) { fixed (byte* ptr = bytes) { return *(float*)(ptr + offset); } } uint num = (uint)(bytes[offset] | (bytes[offset + 1] << 8) | (bytes[offset + 2] << 16) | (bytes[offset + 3] << 24)); return *(float*)(&num); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteDouble(ref byte[] bytes, int offset, double value) { EnsureCapacity(ref bytes, offset, 8); if (offset % 8 == 0) { fixed (byte* ptr = bytes) { *(double*)(ptr + offset) = value; } } else { ulong num = *(ulong*)(&value); bytes[offset] = (byte)num; bytes[offset + 1] = (byte)(num >> 8); bytes[offset + 2] = (byte)(num >> 16); bytes[offset + 3] = (byte)(num >> 24); bytes[offset + 4] = (byte)(num >> 32); bytes[offset + 5] = (byte)(num >> 40); bytes[offset + 6] = (byte)(num >> 48); bytes[offset + 7] = (byte)(num >> 56); } return 8; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static double ReadDouble(ref byte[] bytes, int offset) { if (offset % 8 == 0) { fixed (byte* ptr = bytes) { return *(double*)(ptr + offset); } } uint num = (uint)(bytes[offset] | (bytes[offset + 1] << 8) | (bytes[offset + 2] << 16) | (bytes[offset + 3] << 24)); ulong num2 = (ulong)(((long)(bytes[offset + 4] | (bytes[offset + 5] << 8) | (bytes[offset + 6] << 16) | (bytes[offset + 7] << 24)) << 32) | num); return *(double*)(&num2); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteInt16(ref byte[] bytes, int offset, short value) { EnsureCapacity(ref bytes, offset, 2); fixed (byte* ptr = bytes) { *(short*)(ptr + offset) = value; } return 2; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static short ReadInt16(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(short*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteInt32(ref byte[] bytes, int offset, int value) { EnsureCapacity(ref bytes, offset, 4); fixed (byte* ptr = bytes) { *(int*)(ptr + offset) = value; } return 4; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void WriteInt32Unsafe(ref byte[] bytes, int offset, int value) { fixed (byte* ptr = bytes) { *(int*)(ptr + offset) = value; } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int ReadInt32(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(int*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteInt64(ref byte[] bytes, int offset, long value) { EnsureCapacity(ref bytes, offset, 8); fixed (byte* ptr = bytes) { *(long*)(ptr + offset) = value; } return 8; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static long ReadInt64(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(long*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteUInt16(ref byte[] bytes, int offset, ushort value) { EnsureCapacity(ref bytes, offset, 2); fixed (byte* ptr = bytes) { *(ushort*)(ptr + offset) = value; } return 2; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static ushort ReadUInt16(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(ushort*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteUInt32(ref byte[] bytes, int offset, uint value) { EnsureCapacity(ref bytes, offset, 4); fixed (byte* ptr = bytes) { *(uint*)(ptr + offset) = value; } return 4; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static uint ReadUInt32(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(uint*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteUInt64(ref byte[] bytes, int offset, ulong value) { EnsureCapacity(ref bytes, offset, 8); fixed (byte* ptr = bytes) { *(ulong*)(ptr + offset) = value; } return 8; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static ulong ReadUInt64(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(ulong*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static int WriteChar(ref byte[] bytes, int offset, char value) { return WriteUInt16(ref bytes, offset, value); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static char ReadChar(ref byte[] bytes, int offset) { return (char)ReadUInt16(ref bytes, offset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static int WriteString(ref byte[] bytes, int offset, string value) { int maxByteCount = StringEncoding.UTF8.GetMaxByteCount(value.Length); EnsureCapacity(ref bytes, offset, maxByteCount); return StringEncoding.UTF8.GetBytes(value, 0, value.Length, bytes, offset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static string ReadString(ref byte[] bytes, int offset, int count) { return StringEncoding.UTF8.GetString(bytes, offset, count); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteDecimal(ref byte[] bytes, int offset, decimal value) { EnsureCapacity(ref bytes, offset, 16); fixed (byte* ptr = bytes) { *(decimal*)(ptr + offset) = value; } return 16; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static decimal ReadDecimal(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(decimal*)(ptr + offset); } } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int WriteGuid(ref byte[] bytes, int offset, Guid value) { EnsureCapacity(ref bytes, offset, 16); fixed (byte* ptr = bytes) { *(Guid*)(ptr + offset) = value; } return 16; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static Guid ReadGuid(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { return *(Guid*)(ptr + offset); } } public unsafe static int WriteTimeSpan(ref byte[] bytes, int offset, TimeSpan timeSpan) { long ticks = timeSpan.Ticks; long num = ticks / 10000000; int num2; checked { num2 = (int)unchecked(ticks % 10000000) * 100; EnsureCapacity(ref bytes, offset, 12); } fixed (byte* ptr = bytes) { *(long*)checked(unchecked((nuint)ptr) + unchecked((nuint)offset)) = num; *(int*)checked(unchecked((nuint)ptr) + unchecked((nuint)offset) + 8) = num2; } return 12; } public unsafe static TimeSpan ReadTimeSpan(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { long num = *(long*)checked(unchecked((nuint)ptr) + unchecked((nuint)offset)); int num2 = *(int*)checked(unchecked((nuint)ptr) + unchecked((nuint)offset) + 8); if (!Duration.IsNormalized(num, num2)) { throw new InvalidOperationException("Duration was not a valid normalized duration"); } checked { return TimeSpan.FromTicks(num * 10000000 + unchecked(num2 / 100)); } } } public unsafe static int WriteDateTime(ref byte[] bytes, int offset, DateTime dateTime) { dateTime = dateTime.ToUniversalTime(); long num = dateTime.Ticks / 10000000; int num2 = (int)(dateTime.Ticks % 10000000) * 100; EnsureCapacity(ref bytes, offset, 12); fixed (byte* ptr = bytes) { *(long*)(ptr + offset) = num - 62135596800L; *(int*)(ptr + offset + 8) = num2; } return 12; } public unsafe static DateTime ReadDateTime(ref byte[] bytes, int offset) { fixed (byte* ptr = bytes) { long num = *(long*)(ptr + offset); int num2 = *(int*)(ptr + offset + 8); if (!Timestamp.IsNormalized(num, num2)) { throw new InvalidOperationException($"Timestamp contains invalid values: Seconds={num}; Nanos={num2}"); } DateTime unixEpoch = Timestamp.UnixEpoch; return unixEpoch.AddSeconds(num).AddTicks(num2 / 100); } } } internal class DynamicAssembly { private readonly object gate = new object(); private readonly string moduleName; private readonly AssemblyBuilder assemblyBuilder; private readonly ModuleBuilder moduleBuilder; public ModuleBuilder ModuleBuilder => moduleBuilder; public DynamicAssembly(string moduleName) { this.moduleName = moduleName; assemblyBuilder = AssemblyBuilder.DefineDynamicAssembly(new AssemblyName(moduleName), AssemblyBuilderAccess.Run); moduleBuilder = assemblyBuilder.DefineDynamicModule(moduleName); } } internal static class HashHelpers { public static readonly int[] primes = new int[104] { 3, 7, 11, 17, 23, 29, 37, 47, 59, 71, 89, 107, 131, 163, 197, 239, 293, 353, 431, 521, 631, 761, 919, 1103, 1327, 1597, 1931, 2333, 2801, 3371, 4049, 4861, 5839, 7013, 8419, 10103, 12143, 14591, 17519, 21023, 25229, 30293, 36353, 43627, 52361, 62851, 75431, 90523, 108631, 130363, 156437, 187751, 225307, 270371, 324449, 389357, 467237, 560689, 672827, 807403, 968897, 1162687, 1395263, 1674319, 2009191, 2411033, 2893249, 3471899, 4166287, 4999559, 5999471, 7199369, 8639249, 10367101, 12440537, 14928671, 17914409, 21497293, 25796759, 30956117, 37147349, 44576837, 53492207, 64190669, 77028803, 92434613, 110921543, 133105859, 159727031, 191672443, 230006941, 276008387, 331210079, 397452101, 476942527, 572331049, 686797261, 824156741, 988988137, 1186785773, 1424142949, 1708971541, 2050765853, 2146435069 }; public const int MaxPrimeArrayLength = 2146435069; public static int GetPrime(int min) { if (min < 0) { throw new ArgumentException("Arg_HTCapacityOverflow"); } for (int i = 0; i < primes.Length; i++) { int num = primes[i]; if (num >= min) { return num; } } return min; } public static int GetMinPrime() { return primes[0]; } public static int ExpandPrime(int oldSize) { int num = 2 * oldSize; if ((uint)num > 2146435069u && 2146435069 > oldSize) { return 2146435069; } return GetPrime(num); } } internal static class ILGeneratorExtensions { public static void EmitLdloc(this ILGenerator il, int index) { switch (index) { case 0: il.Emit(OpCodes.Ldloc_0); return; case 1: il.Emit(OpCodes.Ldloc_1); return; case 2: il.Emit(OpCodes.Ldloc_2); return; case 3: il.Emit(OpCodes.Ldloc_3); return; } if (index <= 255) { il.Emit(OpCodes.Ldloc_S, (byte)index); } else { il.Emit(OpCodes.Ldloc, (short)index); } } public static void EmitStloc(this ILGenerator il, int index) { switch (index) { case 0: il.Emit(OpCodes.Stloc_0); return; case 1: il.Emit(OpCodes.Stloc_1); return; case 2: il.Emit(OpCodes.Stloc_2); return; case 3: il.Emit(OpCodes.Stloc_3); return; } if (index <= 255) { il.Emit(OpCodes.Stloc_S, (byte)index); } else { il.Emit(OpCodes.Stloc, (short)index); } } public static void EmitLdloca(this ILGenerator il, int index) { if (index <= 255) { il.Emit(OpCodes.Ldloca_S, (byte)index); } else { il.Emit(OpCodes.Ldloca, (short)index); } } public static void EmitLdc_I4(this ILGenerator il, int value) { switch (value) { case -1: il.Emit(OpCodes.Ldc_I4_M1); return; case 0: il.Emit(OpCodes.Ldc_I4_0); return; case 1: il.Emit(OpCodes.Ldc_I4_1); return; case 2: il.Emit(OpCodes.Ldc_I4_2); return; case 3: il.Emit(OpCodes.Ldc_I4_3); return; case 4: il.Emit(OpCodes.Ldc_I4_4); return; case 5: il.Emit(OpCodes.Ldc_I4_5); return; case 6: il.Emit(OpCodes.Ldc_I4_6); return; case 7: il.Emit(OpCodes.Ldc_I4_7); return; case 8: il.Emit(OpCodes.Ldc_I4_8); return; } if (value >= -128 && value <= 127) { il.Emit(OpCodes.Ldc_I4_S, (sbyte)value); } else { il.Emit(OpCodes.Ldc_I4, value); } } } internal static class ReflectionExtensions { public static bool IsNullable(this TypeInfo type) { if (type.IsGenericType) { return type.GetGenericTypeDefinition() == typeof(Nullable<>); } return false; } } internal static class SR { public const string InvalidOperation_EnumFailedVersion = "InvalidOperation_EnumFailedVersion"; public const string InvalidOperation_EnumOpCantHappen = "InvalidOperation_EnumOpCantHappen"; public const string ArgumentOutOfRange_Index = "ArgumentOutOfRange_Index"; public const string Argument_InvalidArrayType = "Argument_InvalidArrayType"; public const string NotSupported_ValueCollectionSet = "NotSupported_ValueCollectionSet"; public const string Arg_RankMultiDimNotSupported = "Arg_RankMultiDimNotSupported"; public const string Arg_ArrayPlusOffTooSmall = "Arg_ArrayPlusOffTooSmall"; public const string Arg_NonZeroLowerBound = "Arg_NonZeroLowerBound"; public const string NotSupported_KeyCollectionSet = "NotSupported_KeyCollectionSet"; public const string Arg_WrongType = "Arg_WrongType"; public const string ArgumentOutOfRange_NeedNonNegNum = "ArgumentOutOfRange_NeedNonNegNum"; public const string Arg_HTCapacityOverflow = "Arg_HTCapacityOverflow"; public const string Argument_AddingDuplicate = "Argument_AddingDuplicate"; public static string Format(string f, params object[] args) { return string.Format(f, args); } } internal static class StringEncoding { public static Encoding UTF8 = new UTF8Encoding(encoderShouldEmitUTF8Identifier: false); } } namespace ZeroFormatter.Formatters { internal class DateTimeOffsetFormatter<TTypeResolver> : Formatter<TTypeResolver, DateTimeOffset> where TTypeResolver : ITypeResolver, new() { public override bool NoUseDirtyTracker => true; public override int? GetLength() { return 14; } public override int Serialize(ref byte[] bytes, int offset, DateTimeOffset value) { BinaryUtil.WriteDateTime(ref bytes, offset, new DateTime(value.Ticks, DateTimeKind.Utc)); BinaryUtil.WriteInt16(ref bytes, offset + 12, (short)value.Offset.TotalMinutes); return 14; } public override DateTimeOffset Deserialize(ref byte[] bytes, int offset, DirtyTracker tracker, out int byteSize) { byteSize = 14; DateTime dateTime = BinaryUtil.ReadDateTime(ref bytes, offset); short num = BinaryUtil.ReadInt16(ref bytes, offset + 12); return new DateTimeOffset(dateTime.Ticks, TimeSpan.FromMinutes(num)); } } internal class NullableDateTimeOffsetFormatter<TTypeResolver> : Formatter<TTypeResolver, DateTimeOffset?> where TTypeResolver : ITypeResolver, new() { public override bool NoUseDirtyTracker => true; public override int? GetLength() { return 15; } public override int Serialize(ref byte[] bytes, int offset, DateTimeOffset? value) { BinaryUtil.EnsureCapacity(ref bytes, offset, 13); if (value.HasValue) { BinaryUtil.WriteBooleanTrueUnsafe(ref bytes, offset); BinaryUtil.WriteDateTime(ref bytes, offset + 1, new DateTime(value.Value.Ticks, DateTimeKind.Utc)); BinaryUtil.WriteInt16(ref bytes, offset + 13, (short)value.Value.Offset.TotalMinutes); } else { BinaryUtil.WriteBooleanFalseUnsafe(ref bytes, offset); } return 15; } public override DateTimeOffset? Deserialize(ref byte[] bytes, int offset, DirtyTracker tracker, out int byteSize) { byteSize = 15; if (!BinaryUtil.ReadBoolean(ref bytes, offset)) { return null; } DateTime dateTime = BinaryUtil.ReadDateTime(ref bytes, offset + 1); short num = BinaryUtil.ReadInt16(ref bytes, offset + 13); return new DateTimeOffset(dateTime.Ticks, TimeSpan.FromMinutes(num)); } } internal class Int16ArrayFormatter<TTypeResolver> : Formatter<TTypeResolver, short[]> where TTypeResolver : ITypeResolver, new() { public override bool NoUseDirtyTracker => true; public override int? GetLength() { return null; } public override int Serialize(ref byte[] bytes, int offset, short[] value) { if (value == null) { BinaryUtil.WriteInt32(ref bytes, offset, -1); return 4; } int num = value.Length * 2; BinaryUtil.EnsureCapacity(ref bytes, offset, num + 4); BinaryUtil.WriteInt32Unsafe(ref bytes, offset, value.Length); Buffer.BlockCopy(value, 0, bytes, offset + 4, num); return num + 4; } public override short[] Deserialize(ref byte[] bytes, int offset, DirtyTracker tracker, out int byteSize) { tracker.Dirty(); int num = BinaryUtil.ReadInt32(ref bytes, offset); if (num == -1) { byteSize = 4; return null; } byteSize = num * 2 + 4; int num2 = num * 2; ZeroFormatterSerializer.ValidateNewLength(num2); short[] array = new short[num]; Buffer.BlockCopy(bytes, offset + 4, array, 0, num2); return array; } } internal class Int32ArrayFormatter<TTypeResolver> : Formatter<TTypeResolver, int[]> where TTypeResolver : ITypeResolver, new() { public override bool NoUseDirtyTracker => true; public override int? GetLength() { return null; } public override int Serialize(ref byte[] bytes, int offset, int[] value) { if (value == null) { BinaryUtil.WriteInt32(ref bytes, offset, -1); return 4; } int num = value.Length * 4; BinaryUtil.EnsureCapacity(ref bytes, offset, num + 4); BinaryUtil.WriteInt32Unsafe(ref bytes, offset, value.Length); Buffer.BlockCopy(value, 0, bytes, offset + 4, num); return num + 4; } public override int[] Deserialize(ref byte[] bytes, int offset, DirtyTracker tracker, out int byteSize) { tracker.Dirty(); int num = BinaryUtil.ReadInt32(ref bytes, offset); if (num == -1) { byteSize = 4; return null; } byteSize = num * 4 + 4; int num2 = num * 4; ZeroFormatterSerializer.ValidateNewLength(num2); int[] array = new int[num]; Buffer.BlockCopy(bytes, offset + 4, array, 0, num2); return array; } } internal class Int64ArrayFormatter<TTypeResolver> : Formatter<TTypeResolver, long[]> where TTypeResolver : ITypeResolver, new() { public override bool NoUseDirtyTracker => true; public override int? GetLength() { return null; } public override int Serialize(ref byte[] bytes, int offset, long[] value) { if (value == null) { BinaryUtil.WriteInt32(ref bytes, offset, -1); return 4; } int num = value.Length * 8; BinaryUtil.EnsureCapacity(ref bytes, offset, num + 4); BinaryUtil.WriteInt32Unsafe(ref bytes, offset, value.Length); Buffer.BlockCopy(value, 0, bytes, offset + 4, num); return num + 4; } public override long[] Deserialize(ref byte[] bytes, int offset, DirtyTracker tracker, out int byteSize) { tracker.Dirty(); int num = BinaryUtil.ReadInt32(ref bytes, offset); if (num == -1) { byteSize = 4; return null; } byteSize = num * 8 + 4; int num2 = num * 8; ZeroFormatterSerializer.ValidateNewLength(num2); long[] array = new long[num]; Buffer.BlockCopy(bytes, offset + 4, array, 0, num2); return array; } } internal class UInt16ArrayFormatter<TTypeResolver> : Formatter<TTypeResolver, ushort[]> where TTypeResolver : ITypeResolver, new() { public override bool NoUseDirtyTracker => true; public override int? GetLength() { return null; } public override int Serialize(ref byte[] bytes, int offset, ushort[] value) { if (value == null) { BinaryUtil.WriteInt32(ref bytes, offset, -1); r
ZeroFormatter.Interfaces.dll
Decompiled 3 weeks agousing System; using System.Collections; using System.Collections.Generic; using System.Diagnostics; using System.Linq; using System.Reflection; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.Versioning; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)] [assembly: AssemblyTitle("ZeroFormatter.Interfaces")] [assembly: AssemblyDescription("")] [assembly: AssemblyConfiguration("")] [assembly: AssemblyCompany("")] [assembly: AssemblyProduct("ZeroFormatter.Interfaces")] [assembly: AssemblyCopyright("Copyright © 2016")] [assembly: AssemblyTrademark("")] [assembly: ComVisible(false)] [assembly: Guid("c051f473-427f-4062-94fb-cae3fa460fc8")] [assembly: AssemblyFileVersion("1.6.4.0")] [assembly: TargetFramework(".NETFramework,Version=v4.5", FrameworkDisplayName = ".NET Framework 4.5")] [assembly: AssemblyVersion("1.6.4.0")] namespace ZeroFormatter; [AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct, AllowMultiple = false, Inherited = true)] public class ZeroFormattableAttribute : Attribute { } [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field, AllowMultiple = false, Inherited = true)] public class IndexAttribute : Attribute { public int Index { get; private set; } public IndexAttribute(int index) { Index = index; } } [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field, AllowMultiple = false, Inherited = true)] public class IgnoreFormatAttribute : Attribute { } [AttributeUsage(AttributeTargets.Class | AttributeTargets.Interface, AllowMultiple = false, Inherited = false)] public class UnionAttribute : Attribute { public Type[] SubTypes { get; private set; } public Type FallbackType { get; private set; } public UnionAttribute(params Type[] subTypes) { SubTypes = subTypes; } public UnionAttribute(Type[] subTypes, Type fallbackType) { SubTypes = subTypes; FallbackType = fallbackType; } } [AttributeUsage(AttributeTargets.Property, AllowMultiple = false, Inherited = true)] public class UnionKeyAttribute : Attribute { } [AttributeUsage(AttributeTargets.Class | AttributeTargets.Interface, AllowMultiple = false, Inherited = false)] public class DynamicUnionAttribute : Attribute { } public sealed class PreserveAttribute : Attribute { public bool AllMembers; public bool Conditional; } public interface ILazyDictionary<TKey, TValue> : IDictionary<TKey, TValue>, ICollection<KeyValuePair<TKey, TValue>>, IEnumerable<KeyValuePair<TKey, TValue>>, IEnumerable { } public interface ILazyLookup<TKey, TElement> : ILookup<TKey, TElement>, IEnumerable<IGrouping<TKey, TElement>>, IEnumerable { } public interface ILazyReadOnlyDictionary<TKey, TValue> : IReadOnlyDictionary<TKey, TValue>, IReadOnlyCollection<KeyValuePair<TKey, TValue>>, IEnumerable<KeyValuePair<TKey, TValue>>, IEnumerable { } public static class LazyCollectionExtensions { internal class DelegateDictionary<TKey, TValue> : IDictionary<TKey, TValue>, ICollection<KeyValuePair<TKey, TValue>>, IEnumerable<KeyValuePair<TKey, TValue>>, IEnumerable, ILazyDictionary<TKey, TValue>, ILazyReadOnlyDictionary<TKey, TValue>, IReadOnlyDictionary<TKey, TValue>, IReadOnlyCollection<KeyValuePair<TKey, TValue>> { private readonly IDictionary<TKey, TValue> dictionary; public TValue this[TKey key] { get { return dictionary[key]; } set { dictionary[key] = value; } } public int Count => dictionary.Count; public bool IsReadOnly => dictionary.IsReadOnly; public ICollection<TKey> Keys => dictionary.Keys; public ICollection<TValue> Values => dictionary.Values; IEnumerable<TKey> IReadOnlyDictionary<TKey, TValue>.Keys => dictionary.Keys; IEnumerable<TValue> IReadOnlyDictionary<TKey, TValue>.Values => dictionary.Values; public DelegateDictionary(IDictionary<TKey, TValue> dictionary) { this.dictionary = dictionary; } public void Add(KeyValuePair<TKey, TValue> item) { dictionary.Add(item); } public void Add(TKey key, TValue value) { dictionary.Add(key, value); } public void Clear() { dictionary.Clear(); } public bool Contains(KeyValuePair<TKey, TValue> item) { return dictionary.Contains(item); } public bool ContainsKey(TKey key) { return dictionary.ContainsKey(key); } public void CopyTo(KeyValuePair<TKey, TValue>[] array, int arrayIndex) { dictionary.CopyTo(array, arrayIndex); } public IEnumerator<KeyValuePair<TKey, TValue>> GetEnumerator() { return dictionary.GetEnumerator(); } public bool Remove(KeyValuePair<TKey, TValue> item) { return dictionary.Remove(item); } public bool Remove(TKey key) { return dictionary.Remove(key); } public bool TryGetValue(TKey key, out TValue value) { return dictionary.TryGetValue(key, out value); } IEnumerator IEnumerable.GetEnumerator() { return dictionary.GetEnumerator(); } } internal class DelegateLookup<TKey, TElement> : ILookup<TKey, TElement>, IEnumerable<IGrouping<TKey, TElement>>, IEnumerable, ILazyLookup<TKey, TElement> { private readonly ILookup<TKey, TElement> lookup; public IEnumerable<TElement> this[TKey key] => lookup[key]; public int Count => lookup.Count; public DelegateLookup(ILookup<TKey, TElement> lookup) { this.lookup = lookup; } public bool Contains(TKey key) { return lookup.Contains(key); } public IEnumerator<IGrouping<TKey, TElement>> GetEnumerator() { return lookup.GetEnumerator(); } IEnumerator IEnumerable.GetEnumerator() { return lookup.GetEnumerator(); } } public static ILazyDictionary<TKey, TValue> AsLazyDictionary<TKey, TValue>(this IDictionary<TKey, TValue> dict) { return new DelegateDictionary<TKey, TValue>(dict); } public static ILazyReadOnlyDictionary<TKey, TValue> AsLazyReadOnlyDictionary<TKey, TValue>(this IDictionary<TKey, TValue> dict) { return new DelegateDictionary<TKey, TValue>(dict); } public static ILazyLookup<TKey, TElement> AsLazyLookup<TKey, TElement>(this ILookup<TKey, TElement> lookup) { return new DelegateLookup<TKey, TElement>(lookup); } } public interface IKeyTuple { new string ToString(); } public static class KeyTuple { public static KeyTuple<T1, T2, T3, T4, T5, T6, T7, KeyTuple<T8>> Create<T1, T2, T3, T4, T5, T6, T7, T8>(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5, T6 item6, T7 item7, T8 item8) { return new KeyTuple<T1, T2, T3, T4, T5, T6, T7, KeyTuple<T8>>(item1, item2, item3, item4, item5, item6, item7, new KeyTuple<T8>(item8)); } public static KeyTuple<T1, T2, T3, T4, T5, T6, T7> Create<T1, T2, T3, T4, T5, T6, T7>(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5, T6 item6, T7 item7) { return new KeyTuple<T1, T2, T3, T4, T5, T6, T7>(item1, item2, item3, item4, item5, item6, item7); } public static KeyTuple<T1, T2, T3, T4, T5, T6> Create<T1, T2, T3, T4, T5, T6>(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5, T6 item6) { return new KeyTuple<T1, T2, T3, T4, T5, T6>(item1, item2, item3, item4, item5, item6); } public static KeyTuple<T1, T2, T3, T4, T5> Create<T1, T2, T3, T4, T5>(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5) { return new KeyTuple<T1, T2, T3, T4, T5>(item1, item2, item3, item4, item5); } public static KeyTuple<T1, T2, T3, T4> Create<T1, T2, T3, T4>(T1 item1, T2 item2, T3 item3, T4 item4) { return new KeyTuple<T1, T2, T3, T4>(item1, item2, item3, item4); } public static KeyTuple<T1, T2, T3> Create<T1, T2, T3>(T1 item1, T2 item2, T3 item3) { return new KeyTuple<T1, T2, T3>(item1, item2, item3); } public static KeyTuple<T1, T2> Create<T1, T2>(T1 item1, T2 item2) { return new KeyTuple<T1, T2>(item1, item2); } public static KeyTuple<T1> Create<T1>(T1 item1) { return new KeyTuple<T1>(item1); } } public struct KeyTuple<T1> : IKeyTuple { private T1 item1; public T1 Item1 => item1; public KeyTuple(T1 item1) { this.item1 = item1; } string IKeyTuple.ToString() { return $"{item1}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2> : IKeyTuple { private T1 item1; private T2 item2; public T1 Item1 => item1; public T2 Item2 => item2; public KeyTuple(T1 item1, T2 item2) { this.item1 = item1; this.item2 = item2; } string IKeyTuple.ToString() { return $"{item1}, {item2}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2, T3> : IKeyTuple { private T1 item1; private T2 item2; private T3 item3; public T1 Item1 => item1; public T2 Item2 => item2; public T3 Item3 => item3; public KeyTuple(T1 item1, T2 item2, T3 item3) { this.item1 = item1; this.item2 = item2; this.item3 = item3; } string IKeyTuple.ToString() { return $"{item1}, {item2}, {item3}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2, T3, T4> : IKeyTuple { private T1 item1; private T2 item2; private T3 item3; private T4 item4; public T1 Item1 => item1; public T2 Item2 => item2; public T3 Item3 => item3; public T4 Item4 => item4; public KeyTuple(T1 item1, T2 item2, T3 item3, T4 item4) { this.item1 = item1; this.item2 = item2; this.item3 = item3; this.item4 = item4; } string IKeyTuple.ToString() { return $"{item1}, {item2}, {item3}, {item4}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2, T3, T4, T5> : IKeyTuple { private T1 item1; private T2 item2; private T3 item3; private T4 item4; private T5 item5; public T1 Item1 => item1; public T2 Item2 => item2; public T3 Item3 => item3; public T4 Item4 => item4; public T5 Item5 => item5; public KeyTuple(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5) { this.item1 = item1; this.item2 = item2; this.item3 = item3; this.item4 = item4; this.item5 = item5; } string IKeyTuple.ToString() { return $"{item1}, {item2}, {item3}, {item4}, {item5}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2, T3, T4, T5, T6> : IKeyTuple { private T1 item1; private T2 item2; private T3 item3; private T4 item4; private T5 item5; private T6 item6; public T1 Item1 => item1; public T2 Item2 => item2; public T3 Item3 => item3; public T4 Item4 => item4; public T5 Item5 => item5; public T6 Item6 => item6; public KeyTuple(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5, T6 item6) { this.item1 = item1; this.item2 = item2; this.item3 = item3; this.item4 = item4; this.item5 = item5; this.item6 = item6; } string IKeyTuple.ToString() { return $"{item1}, {item2}, {item3}, {item4}, {item5}, {item6}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2, T3, T4, T5, T6, T7> : IKeyTuple { private T1 item1; private T2 item2; private T3 item3; private T4 item4; private T5 item5; private T6 item6; private T7 item7; public T1 Item1 => item1; public T2 Item2 => item2; public T3 Item3 => item3; public T4 Item4 => item4; public T5 Item5 => item5; public T6 Item6 => item6; public T7 Item7 => item7; public KeyTuple(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5, T6 item6, T7 item7) { this.item1 = item1; this.item2 = item2; this.item3 = item3; this.item4 = item4; this.item5 = item5; this.item6 = item6; this.item7 = item7; } string IKeyTuple.ToString() { return $"{item1}, {item2}, {item3}, {item4}, {item5}, {item6}, {item7}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public struct KeyTuple<T1, T2, T3, T4, T5, T6, T7, TRest> : IKeyTuple { private T1 item1; private T2 item2; private T3 item3; private T4 item4; private T5 item5; private T6 item6; private T7 item7; private TRest rest; public T1 Item1 => item1; public T2 Item2 => item2; public T3 Item3 => item3; public T4 Item4 => item4; public T5 Item5 => item5; public T6 Item6 => item6; public T7 Item7 => item7; public TRest Rest => rest; public KeyTuple(T1 item1, T2 item2, T3 item3, T4 item4, T5 item5, T6 item6, T7 item7, TRest rest) { this.item1 = item1; this.item2 = item2; this.item3 = item3; this.item4 = item4; this.item5 = item5; this.item6 = item6; this.item7 = item7; this.rest = rest; if (!(rest is IKeyTuple)) { throw new ArgumentException("rest", "The last element of an eight element tuple must be a Tuple."); } } string IKeyTuple.ToString() { return $"{item1}, {item2}, {item3}, {item4}, {item5}, {item6}, {item7}, {((IKeyTuple)(object)rest).ToString()}"; } public override string ToString() { return "(" + ((IKeyTuple)this).ToString() + ")"; } } public static class KeyTupleExtensions { public static IEnumerable<TValue> Get<TKey1, TKey2, TValue>(this ILookup<KeyTuple<TKey1, TKey2>, TValue> lookup, TKey1 tKey1, TKey2 tKey2) { return lookup[KeyTuple.Create(tKey1, tKey2)]; } public static IEnumerable<TValue> Get<TKey1, TKey2, TKey3, TValue>(this ILookup<KeyTuple<TKey1, TKey2, TKey3>, TValue> lookup, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3) { return lookup[KeyTuple.Create(tKey1, tKey2, tKey3)]; } public static IEnumerable<TValue> Get<TKey1, TKey2, TKey3, TKey4, TValue>(this ILookup<KeyTuple<TKey1, TKey2, TKey3, TKey4>, TValue> lookup, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4) { return lookup[KeyTuple.Create(tKey1, tKey2, tKey3, tKey4)]; } public static IEnumerable<TValue> Get<TKey1, TKey2, TKey3, TKey4, TKey5, TValue>(this ILookup<KeyTuple<TKey1, TKey2, TKey3, TKey4, TKey5>, TValue> lookup, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TKey5 tKey5) { return lookup[KeyTuple.Create(tKey1, tKey2, tKey3, tKey4, tKey5)]; } public static IEnumerable<TValue> Get<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6, TValue>(this ILookup<KeyTuple<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6>, TValue> lookup, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TKey5 tKey5, TKey6 tKey6) { return lookup[KeyTuple.Create(tKey1, tKey2, tKey3, tKey4, tKey5, tKey6)]; } public static IEnumerable<TValue> Get<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6, TKey7, TValue>(this ILookup<KeyTuple<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6, TKey7>, TValue> lookup, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TKey5 tKey5, TKey6 tKey6, TKey7 tKey7) { return lookup[KeyTuple.Create(tKey1, tKey2, tKey3, tKey4, tKey5, tKey6, tKey7)]; } public static TValue GetValueOrDefault<TKey1, TKey2, TValue>(this IDictionary<KeyTuple<TKey1, TKey2>, TValue> dictionary, TKey1 tKey1, TKey2 tKey2, TValue defaultValue = default(TValue)) { if (!dictionary.TryGetValue(KeyTuple.Create(tKey1, tKey2), out var value)) { return defaultValue; } return value; } public static TValue GetValueOrDefault<TKey1, TKey2, TKey3, TValue>(this IDictionary<KeyTuple<TKey1, TKey2, TKey3>, TValue> dictionary, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TValue defaultValue = default(TValue)) { if (!dictionary.TryGetValue(KeyTuple.Create(tKey1, tKey2, tKey3), out var value)) { return defaultValue; } return value; } public static TValue GetValueOrDefault<TKey1, TKey2, TKey3, TKey4, TValue>(this IDictionary<KeyTuple<TKey1, TKey2, TKey3, TKey4>, TValue> dictionary, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TValue defaultValue = default(TValue)) { if (!dictionary.TryGetValue(KeyTuple.Create(tKey1, tKey2, tKey3, tKey4), out var value)) { return defaultValue; } return value; } public static TValue GetValueOrDefault<TKey1, TKey2, TKey3, TKey4, TKey5, TValue>(this IDictionary<KeyTuple<TKey1, TKey2, TKey3, TKey4, TKey5>, TValue> dictionary, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TKey5 tKey5, TValue defaultValue = default(TValue)) { if (!dictionary.TryGetValue(KeyTuple.Create(tKey1, tKey2, tKey3, tKey4, tKey5), out var value)) { return defaultValue; } return value; } public static TValue GetValueOrDefault<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6, TValue>(this IDictionary<KeyTuple<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6>, TValue> dictionary, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TKey5 tKey5, TKey6 tKey6, TValue defaultValue = default(TValue)) { if (!dictionary.TryGetValue(KeyTuple.Create(tKey1, tKey2, tKey3, tKey4, tKey5, tKey6), out var value)) { return defaultValue; } return value; } public static TValue GetValueOrDefault<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6, TKey7, TValue>(this IDictionary<KeyTuple<TKey1, TKey2, TKey3, TKey4, TKey5, TKey6, TKey7>, TValue> dictionary, TKey1 tKey1, TKey2 tKey2, TKey3 tKey3, TKey4 tKey4, TKey5 tKey5, TKey6 tKey6, TKey7 tKey7, TValue defaultValue = default(TValue)) { if (!dictionary.TryGetValue(KeyTuple.Create(tKey1, tKey2, tKey3, tKey4, tKey5, tKey6, tKey7), out var value)) { return defaultValue; } return value; } }