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;
- }
- }
- }
- }
|