| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270 | 
							- using System.Buffers.Binary;
 
- using System.Collections.Concurrent;
 
- using System.Collections.Generic;
 
- using System.Runtime.InteropServices;
 
- namespace System.Net.Sockets.Kcp
 
- {
 
-     /// <summary>
 
-     /// 动态申请非托管内存
 
-     /// </summary>
 
-     public class SimpleSegManager : ISegmentManager<KcpSegment>
 
-     {
 
-         public static SimpleSegManager Default { get; } = new SimpleSegManager();
 
-         public KcpSegment Alloc(int appendDateSize)
 
-         {
 
-             return KcpSegment.AllocHGlobal(appendDateSize);
 
-         }
 
-         public void Free(KcpSegment seg)
 
-         {
 
-             KcpSegment.FreeHGlobal(seg);
 
-         }
 
-         public class Kcp : Kcp<KcpSegment>
 
-         {
 
-             public Kcp(uint conv_, IKcpCallback callback, IRentable rentable = null)
 
-                 : base(conv_, callback, rentable)
 
-             {
 
-                 SegmentManager = Default;
 
-             }
 
-         }
 
-         public class KcpIO : KcpIO<KcpSegment>
 
-         {
 
-             public KcpIO(uint conv_)
 
-                 : base(conv_)
 
-             {
 
-                 SegmentManager = Default;
 
-             }
 
-         }
 
-     }
 
-     /// <summary>
 
-     /// 申请固定大小非托管内存。使用这个就不能SetMtu了,大小已经写死。
 
-     /// </summary>
 
-     /// <remarks>需要大量测试</remarks>
 
-     public unsafe class UnSafeSegManager : ISegmentManager<KcpSegment>
 
-     {
 
-         /// <summary>
 
-         /// 因为默认mtu是1400,并且内存需要内存行/内存页对齐。这里直接512对齐。
 
-         /// </summary>
 
-         public const int blockSize = 512 * 3;
 
-         public readonly object locker = new object();
 
-         public Stack<IntPtr> blocks = new Stack<IntPtr>();
 
-         public HashSet<IntPtr> header = new HashSet<IntPtr>();
 
-         public UnSafeSegManager()
 
-         {
 
-             Alloc();
 
-         }
 
-         public static UnSafeSegManager Default { get; } = new UnSafeSegManager();
 
-         public KcpSegment Alloc(int appendDateSize)
 
-         {
 
-             lock (locker)
 
-             {
 
-                 var total = KcpSegment.LocalOffset + KcpSegment.HeadOffset + appendDateSize;
 
-                 if (total > blockSize)
 
-                 {
 
-                     throw new ArgumentOutOfRangeException();
 
-                 }
 
-                 if (blocks.Count > 0)
 
-                 {
 
-                 }
 
-                 else
 
-                 {
 
-                     Alloc();
 
-                 }
 
-                 var ptr = blocks.Pop();
 
-                 Span<byte> span = new Span<byte>(ptr.ToPointer(), blockSize);
 
-                 span.Clear();
 
-                 return new KcpSegment((byte*)ptr.ToPointer(), (uint)appendDateSize);
 
-             }
 
-         }
 
-         public void Free(KcpSegment seg)
 
-         {
 
-             IntPtr ptr = (IntPtr)seg.ptr;
 
-             blocks.Push(ptr);
 
-         }
 
-         void Alloc()
 
-         {
 
-             int count = 50;
 
-             IntPtr intPtr = Marshal.AllocHGlobal(blockSize * count);
 
-             header.Add(intPtr);
 
-             for (int i = 0; i < count; i++)
 
-             {
 
-                 blocks.Push(intPtr + blockSize * i);
 
-             }
 
-         }
 
-         ~UnSafeSegManager()
 
-         {
 
-             foreach (var item in header)
 
-             {
 
-                 Marshal.FreeHGlobal(item);
 
-             }
 
-         }
 
-         public class Kcp : Kcp<KcpSegment>
 
-         {
 
-             public Kcp(uint conv_, IKcpCallback callback, IRentable rentable = null)
 
-                 : base(conv_, callback, rentable)
 
-             {
 
-                 SegmentManager = Default;
 
-             }
 
-         }
 
-         public class KcpIO : KcpIO<KcpSegment>
 
-         {
 
-             public KcpIO(uint conv_)
 
-                 : base(conv_)
 
-             {
 
-                 SegmentManager = Default;
 
-             }
 
-         }
 
-     }
 
-     /// <summary>
 
-     /// 使用内存池,而不是非托管内存,有内存alloc,但是不多。可以解决Marshal.AllocHGlobal 内核调用带来的性能问题
 
-     /// </summary>
 
-     public class PoolSegManager : ISegmentManager<PoolSegManager.Seg>
 
-     {
 
-         /// <summary>
 
-         /// 因为默认mtu是1400,并且内存需要内存行/内存页对齐。这里直接512对齐。
 
-         /// </summary>
 
-         public const int blockSize = 512 * 3;
 
-         ConcurrentStack<Seg> Pool = new ConcurrentStack<Seg>();
 
-         public static PoolSegManager Default { get; } = new PoolSegManager();
 
-         public Seg Alloc(int appendDateSize)
 
-         {
 
-             if (appendDateSize > blockSize)
 
-             {
 
-                 throw new NotSupportedException();
 
-             }
 
-             if (Pool.TryPop(out var ret))
 
-             {
 
-             }
 
-             else
 
-             {
 
-                 ret = new Seg(blockSize);
 
-             }
 
-             ret.len = (uint)appendDateSize;
 
-             return ret;
 
-         }
 
-         public void Free(Seg seg)
 
-         {
 
-             seg.cmd = 0;
 
-             seg.conv = 0;
 
-             seg.fastack = 0;
 
-             seg.frg = 0;
 
-             seg.len = 0;
 
-             seg.resendts = 0;
 
-             seg.rto = 0;
 
-             seg.sn = 0;
 
-             seg.ts = 0;
 
-             seg.una = 0;
 
-             seg.wnd = 0;
 
-             seg.xmit = 0;
 
-             Pool.Push(seg);
 
-         }
 
-         public class Seg : IKcpSegment
 
-         {
 
-             ///以下为需要网络传输的参数
 
-             public const int LocalOffset = 4 * 4;
 
-             public const int HeadOffset = Kcp.IKCP_OVERHEAD;
 
-             byte[] cache;
 
-             public Seg(int blockSize)
 
-             {
 
-                 cache = Buffers.ArrayPool<byte>.Shared.Rent(blockSize);
 
-             }
 
-             public byte cmd { get; set; }
 
-             public uint conv { get; set; }
 
-             public Span<byte> data => cache.AsSpan().Slice(0, (int)len);
 
-             public uint fastack { get; set; }
 
-             public byte frg { get; set; }
 
-             public uint len { get; internal set; }
 
-             public uint resendts { get; set; }
 
-             public uint rto { get; set; }
 
-             public uint sn { get; set; }
 
-             public uint ts { get; set; }
 
-             public uint una { get; set; }
 
-             public ushort wnd { get; set; }
 
-             public uint xmit { get; set; }
 
-             public int Encode(Span<byte> buffer)
 
-             {
 
-                 var datelen = (int)(HeadOffset + len);
 
-                 ///备用偏移值 现阶段没有使用
 
-                 const int offset = 0;
 
-                 if (BitConverter.IsLittleEndian)
 
-                 {
 
-                     BinaryPrimitives.WriteUInt32LittleEndian(buffer.Slice(offset), conv);
 
-                     buffer[offset + 4] = cmd;
 
-                     buffer[offset + 5] = frg;
 
-                     BinaryPrimitives.WriteUInt16LittleEndian(buffer.Slice(offset + 6), wnd);
 
-                     BinaryPrimitives.WriteUInt32LittleEndian(buffer.Slice(offset + 8), ts);
 
-                     BinaryPrimitives.WriteUInt32LittleEndian(buffer.Slice(offset + 12), sn);
 
-                     BinaryPrimitives.WriteUInt32LittleEndian(buffer.Slice(offset + 16), una);
 
-                     BinaryPrimitives.WriteUInt32LittleEndian(buffer.Slice(offset + 20), len);
 
-                     data.CopyTo(buffer.Slice(HeadOffset));
 
-                 }
 
-                 else
 
-                 {
 
-                     BinaryPrimitives.WriteUInt32BigEndian(buffer.Slice(offset), conv);
 
-                     buffer[offset + 4] = cmd;
 
-                     buffer[offset + 5] = frg;
 
-                     BinaryPrimitives.WriteUInt16BigEndian(buffer.Slice(offset + 6), wnd);
 
-                     BinaryPrimitives.WriteUInt32BigEndian(buffer.Slice(offset + 8), ts);
 
-                     BinaryPrimitives.WriteUInt32BigEndian(buffer.Slice(offset + 12), sn);
 
-                     BinaryPrimitives.WriteUInt32BigEndian(buffer.Slice(offset + 16), una);
 
-                     BinaryPrimitives.WriteUInt32BigEndian(buffer.Slice(offset + 20), len);
 
-                     data.CopyTo(buffer.Slice(HeadOffset));
 
-                 }
 
-                 return datelen;
 
-             }
 
-         }
 
-         public class Kcp : Kcp<Seg>
 
-         {
 
-             public Kcp(uint conv_, IKcpCallback callback, IRentable rentable = null)
 
-                 : base(conv_, callback, rentable)
 
-             {
 
-                 SegmentManager = Default;
 
-             }
 
-         }
 
-         public class KcpIO : KcpIO<Seg>
 
-         {
 
-             public KcpIO(uint conv_)
 
-                 : base(conv_)
 
-             {
 
-                 SegmentManager = Default;
 
-             }
 
-         }
 
-     }
 
- }
 
 
  |