org.bouncycastle.asn1.ConstructedBitStream Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of bcprov-jdk15to18 Show documentation
Show all versions of bcprov-jdk15to18 Show documentation
The Bouncy Castle Crypto package is a Java implementation of cryptographic algorithms. This jar contains JCE provider and lightweight API for the Bouncy Castle Cryptography APIs for JDK 1.5 to JDK 1.8.
package org.bouncycastle.asn1;
import java.io.IOException;
import java.io.InputStream;
class ConstructedBitStream
extends InputStream
{
private final ASN1StreamParser _parser;
private final boolean _octetAligned;
private boolean _first = true;
private int _padBits = 0;
private ASN1BitStringParser _currentParser;
private InputStream _currentStream;
ConstructedBitStream(ASN1StreamParser parser, boolean octetAligned)
{
_parser = parser;
_octetAligned = octetAligned;
}
int getPadBits()
{
return _padBits;
}
public int read(byte[] b, int off, int len) throws IOException
{
if (_currentStream == null)
{
if (!_first)
{
return -1;
}
_currentParser = getNextParser();
if (_currentParser == null)
{
return -1;
}
_first = false;
_currentStream = _currentParser.getBitStream();
}
int totalRead = 0;
for (;;)
{
int numRead = _currentStream.read(b, off + totalRead, len - totalRead);
if (numRead >= 0)
{
totalRead += numRead;
if (totalRead == len)
{
return totalRead;
}
}
else
{
_padBits = _currentParser.getPadBits();
_currentParser = getNextParser();
if (_currentParser == null)
{
_currentStream = null;
return totalRead < 1 ? -1 : totalRead;
}
_currentStream = _currentParser.getBitStream();
}
}
}
public int read()
throws IOException
{
if (_currentStream == null)
{
if (!_first)
{
return -1;
}
_currentParser = getNextParser();
if (_currentParser == null)
{
return -1;
}
_first = false;
_currentStream = _currentParser.getBitStream();
}
for (;;)
{
int b = _currentStream.read();
if (b >= 0)
{
return b;
}
_padBits = _currentParser.getPadBits();
_currentParser = getNextParser();
if (_currentParser == null)
{
_currentStream = null;
return -1;
}
_currentStream = _currentParser.getBitStream();
}
}
private ASN1BitStringParser getNextParser() throws IOException
{
ASN1Encodable asn1Obj = _parser.readObject();
if (asn1Obj == null)
{
if (_octetAligned && _padBits != 0)
{
throw new IOException("expected octet-aligned bitstring, but found padBits: " + _padBits);
}
return null;
}
if (asn1Obj instanceof ASN1BitStringParser)
{
if (_padBits != 0)
{
throw new IOException("only the last nested bitstring can have padding");
}
return (ASN1BitStringParser)asn1Obj;
}
throw new IOException("unknown object encountered: " + asn1Obj.getClass());
}
}