EncoderNLS
using System.Runtime.Serialization;
namespace System.Text
{
    internal class EncoderNLS : Encoder, ISerializable
    {
        internal char charLeftOver;
        protected EncodingNLS m_encoding;
        protected bool m_mustFlush;
        internal bool m_throwOnOverflow;
        internal int m_charsUsed;
        internal EncoderFallback m_fallback;
        internal EncoderFallbackBuffer m_fallbackBuffer;
        internal new EncoderFallback Fallback => m_fallback;
        internal bool InternalHasFallbackBuffer => m_fallbackBuffer != null;
        public new EncoderFallbackBuffer FallbackBuffer {
            get {
                if (m_fallbackBuffer == null)
                    m_fallbackBuffer = ((m_fallback != null) ? m_fallback.CreateFallbackBuffer() : EncoderFallback.ReplacementFallback.CreateFallbackBuffer());
                return m_fallbackBuffer;
            }
        }
        public Encoding Encoding => m_encoding;
        public bool MustFlush => m_mustFlush;
        internal virtual bool HasState => charLeftOver != ' ';
        internal EncoderNLS(EncodingNLS encoding)
        {
            m_encoding = encoding;
            m_fallback = m_encoding.EncoderFallback;
            Reset();
        }
        void ISerializable.GetObjectData(SerializationInfo info, StreamingContext context)
        {
            throw new PlatformNotSupportedException();
        }
        public override void Reset()
        {
            charLeftOver = ' ';
            m_fallbackBuffer?.Reset();
        }
        public unsafe override int GetByteCount(char[] chars, int index, int count, bool flush)
        {
            if (chars == null)
                throw new ArgumentNullException("chars");
            if (index < 0 || count < 0)
                throw new ArgumentOutOfRangeException((index < 0) ? "index" : "count", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            if (chars.Length - index >= count) {
                fixed (char* ptr = &CodePagesEncodingProvider.GetNonNullPinnableReference(chars)) {
                    return GetByteCount(ptr + index, count, flush);
                }
            }
            throw new ArgumentOutOfRangeException("chars", System.SR.ArgumentOutOfRange_IndexCountBuffer);
        }
        public unsafe override int GetByteCount(char* chars, int count, bool flush)
        {
            if (chars == null)
                throw new ArgumentNullException("chars");
            if (count < 0)
                throw new ArgumentOutOfRangeException("count", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            m_mustFlush = flush;
            m_throwOnOverflow = true;
            return m_encoding.GetByteCount(chars, count, this);
        }
        public unsafe override int GetBytes(char[] chars, int charIndex, int charCount, byte[] bytes, int byteIndex, bool flush)
        {
            if (chars == null)
                throw new ArgumentNullException("chars");
            if (bytes == null)
                throw new ArgumentNullException("bytes");
            if (charIndex < 0 || charCount < 0)
                throw new ArgumentOutOfRangeException((charIndex < 0) ? "charIndex" : "charCount", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            if (chars.Length - charIndex < charCount)
                throw new ArgumentOutOfRangeException("chars", System.SR.ArgumentOutOfRange_IndexCountBuffer);
            if (byteIndex >= 0 && byteIndex <= bytes.Length) {
                int byteCount = bytes.Length - byteIndex;
                fixed (char* ptr = &CodePagesEncodingProvider.GetNonNullPinnableReference(chars)) {
                    fixed (byte* ptr2 = &CodePagesEncodingProvider.GetNonNullPinnableReference(bytes)) {
                        return GetBytes(ptr + charIndex, charCount, ptr2 + byteIndex, byteCount, flush);
                    }
                }
            }
            throw new ArgumentOutOfRangeException("byteIndex", System.SR.ArgumentOutOfRange_IndexMustBeLessOrEqual);
        }
        public unsafe override int GetBytes(char* chars, int charCount, byte* bytes, int byteCount, bool flush)
        {
            if (chars == null)
                throw new ArgumentNullException("chars");
            if (bytes == null)
                throw new ArgumentNullException("bytes");
            if (byteCount < 0 || charCount < 0)
                throw new ArgumentOutOfRangeException((byteCount < 0) ? "byteCount" : "charCount", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            m_mustFlush = flush;
            m_throwOnOverflow = true;
            return m_encoding.GetBytes(chars, charCount, bytes, byteCount, this);
        }
        public unsafe override void Convert(char[] chars, int charIndex, int charCount, byte[] bytes, int byteIndex, int byteCount, bool flush, out int charsUsed, out int bytesUsed, out bool completed)
        {
            if (chars == null)
                throw new ArgumentNullException("chars");
            if (bytes == null)
                throw new ArgumentNullException("bytes");
            if (charIndex < 0 || charCount < 0)
                throw new ArgumentOutOfRangeException((charIndex < 0) ? "charIndex" : "charCount", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            if (byteIndex < 0 || byteCount < 0)
                throw new ArgumentOutOfRangeException((byteIndex < 0) ? "byteIndex" : "byteCount", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            if (chars.Length - charIndex < charCount)
                throw new ArgumentOutOfRangeException("chars", System.SR.ArgumentOutOfRange_IndexCountBuffer);
            if (bytes.Length - byteIndex < byteCount)
                throw new ArgumentOutOfRangeException("bytes", System.SR.ArgumentOutOfRange_IndexCountBuffer);
            fixed (char* ptr = &CodePagesEncodingProvider.GetNonNullPinnableReference(chars)) {
                fixed (byte* ptr2 = &CodePagesEncodingProvider.GetNonNullPinnableReference(bytes)) {
                    Convert(ptr + charIndex, charCount, ptr2 + byteIndex, byteCount, flush, out charsUsed, out bytesUsed, out completed);
                }
            }
        }
        public unsafe override void Convert(char* chars, int charCount, byte* bytes, int byteCount, bool flush, out int charsUsed, out int bytesUsed, out bool completed)
        {
            if (chars == null)
                throw new ArgumentNullException("chars");
            if (bytes == null)
                throw new ArgumentNullException("bytes");
            if (charCount < 0 || byteCount < 0)
                throw new ArgumentOutOfRangeException((charCount < 0) ? "charCount" : "byteCount", System.SR.ArgumentOutOfRange_NeedNonNegNum);
            m_mustFlush = flush;
            m_throwOnOverflow = false;
            m_charsUsed = 0;
            bytesUsed = m_encoding.GetBytes(chars, charCount, bytes, byteCount, this);
            charsUsed = m_charsUsed;
            completed = (charsUsed == charCount && (!flush || !HasState) && (m_fallbackBuffer == null || m_fallbackBuffer.Remaining == 0));
        }
        internal void ClearMustFlush()
        {
            m_mustFlush = false;
        }
    }
}