Fix SSL_peek and SSL_pending.

This commit is contained in:
Bodo Möller 2000-12-25 18:40:46 +00:00
parent 57d2f21782
commit a0aae68cf6
7 changed files with 65 additions and 61 deletions

18
CHANGES
View file

@ -144,11 +144,19 @@
512 bits], about 30% for larger ones [1024 or 2048 bits].) 512 bits], about 30% for larger ones [1024 or 2048 bits].)
[Bodo Moeller] [Bodo Moeller]
*) Disable ssl2_peek and ssl3_peek (i.e., both implementations *) Fix ssl3_pending: If the record in s->s3->rrec is not of type
of SSL_peek) because they both are completely broken. SSL3_RT_APPLICATION_DATA, return 0.
For fixing this, the internal read functions now have an additional [Bodo Moeller]
'peek' parameter, but the actual peek functionality has not
yet been implemented. *) Fix SSL_peek:
Both ssl2_peek and ssl3_peek, which were totally broken in earlier
releases, have been re-implemented by renaming the previous
implementations of ssl2_read and ssl3_read to ssl2_read_internal
and ssl3_read_internal, respectively, and adding 'peek' parameters
to them. The new ssl[23]_{read,peek} functions are calls to
ssl[23]_read_internal with the 'peek' flag set appropriately.
A 'peek' parameter has also been added to ssl3_read_bytes, which
does the actual work for ssl3_read_internal.
[Bodo Moeller] [Bodo Moeller]
*) New function BN_kronecker. *) New function BN_kronecker.

View file

@ -14,8 +14,8 @@ SSL_get_error - obtain result code for TLS/SSL I/O operation
SSL_get_error() returns a result code (suitable for the C "switch" SSL_get_error() returns a result code (suitable for the C "switch"
statement) for a preceding call to SSL_connect(), SSL_accept(), statement) for a preceding call to SSL_connect(), SSL_accept(),
SSL_read(), or SSL_write() on B<ssl>. The value returned by that SSL_read(), SSL_peek(), or SSL_write() on B<ssl>. The value returned by
TLS/SSL I/O function must be passed to SSL_get_error() in parameter that TLS/SSL I/O function must be passed to SSL_get_error() in parameter
B<ret>. B<ret>.
In addition to B<ssl> and B<ret>, SSL_get_error() inspects the In addition to B<ssl> and B<ret>, SSL_get_error() inspects the
@ -64,10 +64,10 @@ TLS/SSL I/O function should be retried.
Caveat: Any TLS/SSL I/O function can lead to either of Caveat: Any TLS/SSL I/O function can lead to either of
B<SSL_ERROR_WANT_READ> and B<SSL_ERROR_WANT_WRITE>. In particular, B<SSL_ERROR_WANT_READ> and B<SSL_ERROR_WANT_WRITE>. In particular,
SSL_read() may want to write data and SSL_write() may want to read SSL_read() or SSL_peek() may want to write data and SSL_write() may want
data. This is mainly because TLS/SSL handshakes may occur at any time to read data. This is mainly because TLS/SSL handshakes may occur at any
during the protocol (initiated by either the client or the server); time during the protocol (initiated by either the client or the server);
SSL_read() and SSL_write() will handle any pending handshakes. SSL_read(), SSL_peek(), and SSL_write() will handle any pending handshakes.
=item SSL_ERROR_WANT_X509_LOOKUP =item SSL_ERROR_WANT_X509_LOOKUP

View file

@ -33,8 +33,8 @@ I<read_ahead> flag is set, additional protocol bytes may have been
read containing more TLS/SSL records; these are ignored by read containing more TLS/SSL records; these are ignored by
SSL_pending(). SSL_pending().
SSL_pending() does not check if the record type of pending data is Up to OpenSSL 0.9.6, SSL_pending() does not check if the record type
application data. of pending data is application data.
=head1 SEE ALSO =head1 SEE ALSO

View file

@ -260,9 +260,6 @@ SSL_CIPHER *ssl2_get_cipher(unsigned int u)
int ssl2_pending(SSL *s) int ssl2_pending(SSL *s)
{ {
/* Unlike ssl2_pending, this one probably works (if read-ahead
* is disabled), but it should be examined
* XXX */
return(s->s2->ract_data_length); return(s->s2->ract_data_length);
} }

View file

@ -138,7 +138,7 @@ static int ssl2_read_internal(SSL *s, void *buf, int len, int peek)
return -1; return -1;
} }
ssl2_read_again: ssl2_read_again:
if (SSL_in_init(s) && !s->in_handshake) if (SSL_in_init(s) && !s->in_handshake)
{ {
n=s->handshake_func(s); n=s->handshake_func(s);
@ -162,13 +162,22 @@ ssl2_read_again:
n=len; n=len;
memcpy(buf,s->s2->ract_data,(unsigned int)n); memcpy(buf,s->s2->ract_data,(unsigned int)n);
s->s2->ract_data_length-=n; if (!peek)
s->s2->ract_data+=n; {
if (s->s2->ract_data_length == 0) s->s2->ract_data_length-=n;
s->rstate=SSL_ST_READ_HEADER; s->s2->ract_data+=n;
if (s->s2->ract_data_length == 0)
s->rstate=SSL_ST_READ_HEADER;
}
return(n); return(n);
} }
/* s->s2->ract_data_length == 0
*
* Fill the buffer, then goto ssl2_read_again.
*/
if (s->rstate == SSL_ST_READ_HEADER) if (s->rstate == SSL_ST_READ_HEADER)
{ {
if (s->first_packet) if (s->first_packet)
@ -266,33 +275,24 @@ ssl2_read_again:
INC32(s->s2->read_sequence); /* expect next number */ INC32(s->s2->read_sequence); /* expect next number */
/* s->s2->ract_data is now available for processing */ /* s->s2->ract_data is now available for processing */
#if 1 /* Possibly the packet that we just read had 0 actual data bytes.
/* How should we react when a packet containing 0 * (SSLeay/OpenSSL itself never sends such packets; see ssl2_write.)
* bytes is received? (Note that SSLeay/OpenSSL itself * In this case, returning 0 would be interpreted by the caller
* never sends such packets; see ssl2_write.) * as indicating EOF, so it's not a good idea. Instead, we just
* Returning 0 would be interpreted by the caller as * continue reading; thus ssl2_read_internal may have to process
* indicating EOF, so it's not a good idea. * multiple packets before it can return.
* Instead, we just continue reading. Note that using *
* select() for blocking sockets *never* guarantees * [Note that using select() for blocking sockets *never* guarantees
* that the next SSL_read will not block -- the available * that the next SSL_read will not block -- the available
* data may contain incomplete packets, and except for SSL 2 * data may contain incomplete packets, and except for SSL 2,
* renegotiation can confuse things even more. */ * renegotiation can confuse things even more.] */
goto ssl2_read_again; /* This should really be goto ssl2_read_again; /* This should really be
* "return ssl2_read(s,buf,len)", * "return ssl2_read(s,buf,len)",
* but that would allow for * but that would allow for
* denial-of-service attacks if a * denial-of-service attacks if a
* C compiler is used that does not * C compiler is used that does not
* recognize end-recursion. */ * recognize end-recursion. */
#else
/* If a 0 byte packet was sent, return 0, otherwise
* we play havoc with people using select with
* blocking sockets. Let them handle a packet at a time,
* they should really be using non-blocking sockets. */
if (s->s2->ract_data_length == 0)
return(0);
return(ssl2_read(s,buf,len));
#endif
} }
else else
{ {

View file

@ -782,8 +782,7 @@ SSL_CIPHER *ssl3_get_cipher(unsigned int u)
int ssl3_pending(SSL *s) int ssl3_pending(SSL *s)
{ {
/* The problem is that it may not be the correct record type */ return (s->s3->rrec.type == SSL3_RT_APPLICATION_DATA) ? s->s3->rrec.length : 0;
return(s->s3->rrec.length); /* FIXME */
} }
int ssl3_new(SSL *s) int ssl3_new(SSL *s)

View file

@ -711,17 +711,12 @@ int ssl3_read_bytes(SSL *s, int type, unsigned char *buf, int len, int peek)
SSL3_RECORD *rr; SSL3_RECORD *rr;
void (*cb)()=NULL; void (*cb)()=NULL;
if (peek)
{
SSLerr(SSL_F_SSL3_READ_BYTES, SSL_R_FIXME); /* proper implementation not yet completed */
return -1;
}
if (s->s3->rbuf.buf == NULL) /* Not initialized yet */ if (s->s3->rbuf.buf == NULL) /* Not initialized yet */
if (!ssl3_setup_buffers(s)) if (!ssl3_setup_buffers(s))
return(-1); return(-1);
if ((type != SSL3_RT_APPLICATION_DATA) && (type != SSL3_RT_HANDSHAKE) && type) if ((type && (type != SSL3_RT_APPLICATION_DATA) && (type != SSL3_RT_HANDSHAKE) && type) ||
(peek && (type != SSL3_RT_APPLICATION_DATA)))
{ {
SSLerr(SSL_F_SSL3_READ_BYTES, SSL_R_INTERNAL_ERROR); SSLerr(SSL_F_SSL3_READ_BYTES, SSL_R_INTERNAL_ERROR);
return -1; return -1;
@ -734,6 +729,7 @@ int ssl3_read_bytes(SSL *s, int type, unsigned char *buf, int len, int peek)
unsigned char *dst = buf; unsigned char *dst = buf;
unsigned int k; unsigned int k;
/* peek == 0 */
n = 0; n = 0;
while ((len > 0) && (s->s3->handshake_fragment_len > 0)) while ((len > 0) && (s->s3->handshake_fragment_len > 0))
{ {
@ -769,7 +765,7 @@ start:
* s->s3->rrec.length, - number of bytes. */ * s->s3->rrec.length, - number of bytes. */
rr = &(s->s3->rrec); rr = &(s->s3->rrec);
/* get new packet */ /* get new packet if necessary */
if ((rr->length == 0) || (s->rstate == SSL_ST_READ_BODY)) if ((rr->length == 0) || (s->rstate == SSL_ST_READ_BODY))
{ {
ret=ssl3_get_record(s); ret=ssl3_get_record(s);
@ -787,7 +783,8 @@ start:
goto err; goto err;
} }
/* If the other end has shutdown, throw anything we read away */ /* If the other end has shut down, throw anything we read away
* (even in 'peek' mode) */
if (s->shutdown & SSL_RECEIVED_SHUTDOWN) if (s->shutdown & SSL_RECEIVED_SHUTDOWN)
{ {
rr->length=0; rr->length=0;
@ -816,12 +813,15 @@ start:
n = (unsigned int)len; n = (unsigned int)len;
memcpy(buf,&(rr->data[rr->off]),n); memcpy(buf,&(rr->data[rr->off]),n);
rr->length-=n; if (!peek)
rr->off+=n;
if (rr->length == 0)
{ {
s->rstate=SSL_ST_READ_HEADER; rr->length-=n;
rr->off=0; rr->off+=n;
if (rr->length == 0)
{
s->rstate=SSL_ST_READ_HEADER;
rr->off=0;
}
} }
return(n); return(n);
} }