summary refs log tree commit diff
path: root/crypto/src/asn1/IndefiniteLengthInputStream.cs
blob: e192e9e8b18a788e22af53fe041538d16711fbcb (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
using System;
using System.IO;

namespace Org.BouncyCastle.Asn1
{
	class IndefiniteLengthInputStream
		: LimitedInputStream
	{
        private int _lookAhead;
        private bool _eofOn00 = true;

		internal IndefiniteLengthInputStream(Stream	inStream, int limit)
			: base(inStream, limit)
		{
            _lookAhead = RequireByte();

            if (0 == _lookAhead)
            {
                CheckEndOfContents();
            }
        }

		internal void SetEofOn00(bool eofOn00)
		{
			_eofOn00 = eofOn00;
            if (_eofOn00 && 0 == _lookAhead)
            {
                CheckEndOfContents();
            }
        }

        private void CheckEndOfContents()
        {
            if (0 != RequireByte())
                throw new IOException("malformed end-of-contents marker");

            _lookAhead = -1;
            SetParentEofDetect();
        }

        public override int Read(byte[] buffer, int offset, int count)
        {
            // Only use this optimisation if we aren't checking for 00
            if (_eofOn00 || count <= 1)
				return base.Read(buffer, offset, count);

			if (_lookAhead < 0)
				return 0;

			int numRead = _in.Read(buffer, offset + 1, count - 1);
			if (numRead <= 0)
				throw new EndOfStreamException();

			buffer[offset] = (byte)_lookAhead;
			_lookAhead = RequireByte();

			return numRead + 1;
		}

#if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER
        public override int Read(Span<byte> buffer)
        {
            // Only use this optimisation if we aren't checking for 00
            if (_eofOn00 || buffer.Length <= 1)
                return base.Read(buffer);

            if (_lookAhead < 0)
                return 0;

            int numRead = _in.Read(buffer[1..]);
            if (numRead <= 0)
                throw new EndOfStreamException();

            buffer[0] = (byte)_lookAhead;
            _lookAhead = RequireByte();

            return numRead + 1;
        }
#endif

        public override int ReadByte()
		{
            if (_eofOn00 && _lookAhead <= 0)
            {
                if (0 == _lookAhead)
                {
                    CheckEndOfContents();
                }
                return -1;
            }

            int result = _lookAhead;
            _lookAhead = RequireByte();
            return result;
		}

        private int RequireByte()
        {
            int b = _in.ReadByte();
            if (b < 0)
                throw new EndOfStreamException();

            return b;
        }
	}
}