Adapt BIO_f_base64 to the opaque EVP_ENCODE_CTX

Reviewed-by: Rich Salz <rsalz@openssl.org>
This commit is contained in:
Richard Levitte 2015-12-11 16:09:52 +01:00
parent a0be4fd17b
commit b518d2d5f8

View File

@ -89,7 +89,7 @@ typedef struct b64_struct {
int encode; int encode;
int start; /* have we started decoding yet? */ int start; /* have we started decoding yet? */
int cont; /* <= 0 when finished */ int cont; /* <= 0 when finished */
EVP_ENCODE_CTX base64; EVP_ENCODE_CTX *base64;
char buf[EVP_ENCODE_LENGTH(B64_BLOCK_SIZE) + 10]; char buf[EVP_ENCODE_LENGTH(B64_BLOCK_SIZE) + 10];
char tmp[B64_BLOCK_SIZE]; char tmp[B64_BLOCK_SIZE];
} BIO_B64_CTX; } BIO_B64_CTX;
@ -121,6 +121,7 @@ static int b64_new(BIO *bi)
ctx->cont = 1; ctx->cont = 1;
ctx->start = 1; ctx->start = 1;
ctx->base64 = EVP_ENCODE_CTX_new();
bi->init = 1; bi->init = 1;
bi->ptr = (char *)ctx; bi->ptr = (char *)ctx;
bi->flags = 0; bi->flags = 0;
@ -132,6 +133,7 @@ static int b64_free(BIO *a)
{ {
if (a == NULL) if (a == NULL)
return (0); return (0);
EVP_ENCODE_CTX_free(((BIO_B64_CTX *)a->ptr)->base64);
OPENSSL_free(a->ptr); OPENSSL_free(a->ptr);
a->ptr = NULL; a->ptr = NULL;
a->init = 0; a->init = 0;
@ -159,7 +161,7 @@ static int b64_read(BIO *b, char *out, int outl)
ctx->buf_len = 0; ctx->buf_len = 0;
ctx->buf_off = 0; ctx->buf_off = 0;
ctx->tmp_len = 0; ctx->tmp_len = 0;
EVP_DecodeInit(&(ctx->base64)); EVP_DecodeInit(ctx->base64);
} }
/* First check if there are bytes decoded/encoded */ /* First check if there are bytes decoded/encoded */
@ -238,11 +240,11 @@ static int b64_read(BIO *b, char *out, int outl)
continue; continue;
} }
k = EVP_DecodeUpdate(&(ctx->base64), k = EVP_DecodeUpdate(ctx->base64,
(unsigned char *)ctx->buf, (unsigned char *)ctx->buf,
&num, p, q - p); &num, p, q - p);
if ((k <= 0) && (num == 0) && (ctx->start)) if ((k <= 0) && (num == 0) && (ctx->start))
EVP_DecodeInit(&ctx->base64); EVP_DecodeInit(ctx->base64);
else { else {
if (p != (unsigned char *) if (p != (unsigned char *)
&(ctx->tmp[0])) { &(ctx->tmp[0])) {
@ -251,7 +253,7 @@ static int b64_read(BIO *b, char *out, int outl)
for (x = 0; x < i; x++) for (x = 0; x < i; x++)
ctx->tmp[x] = p[x]; ctx->tmp[x] = p[x];
} }
EVP_DecodeInit(&ctx->base64); EVP_DecodeInit(ctx->base64);
ctx->start = 0; ctx->start = 0;
break; break;
} }
@ -315,7 +317,7 @@ static int b64_read(BIO *b, char *out, int outl)
} }
i = z; i = z;
} else { } else {
i = EVP_DecodeUpdate(&(ctx->base64), i = EVP_DecodeUpdate(ctx->base64,
(unsigned char *)ctx->buf, &ctx->buf_len, (unsigned char *)ctx->buf, &ctx->buf_len,
(unsigned char *)ctx->tmp, i); (unsigned char *)ctx->tmp, i);
ctx->tmp_len = 0; ctx->tmp_len = 0;
@ -362,7 +364,7 @@ static int b64_write(BIO *b, const char *in, int inl)
ctx->buf_len = 0; ctx->buf_len = 0;
ctx->buf_off = 0; ctx->buf_off = 0;
ctx->tmp_len = 0; ctx->tmp_len = 0;
EVP_EncodeInit(&(ctx->base64)); EVP_EncodeInit(ctx->base64);
} }
OPENSSL_assert(ctx->buf_off < (int)sizeof(ctx->buf)); OPENSSL_assert(ctx->buf_off < (int)sizeof(ctx->buf));
@ -431,7 +433,7 @@ static int b64_write(BIO *b, const char *in, int inl)
ret += n; ret += n;
} }
} else { } else {
EVP_EncodeUpdate(&(ctx->base64), EVP_EncodeUpdate(ctx->base64,
(unsigned char *)ctx->buf, &ctx->buf_len, (unsigned char *)ctx->buf, &ctx->buf_len,
(unsigned char *)in, n); (unsigned char *)in, n);
OPENSSL_assert(ctx->buf_len <= (int)sizeof(ctx->buf)); OPENSSL_assert(ctx->buf_len <= (int)sizeof(ctx->buf));
@ -486,7 +488,7 @@ static long b64_ctrl(BIO *b, int cmd, long num, void *ptr)
OPENSSL_assert(ctx->buf_len >= ctx->buf_off); OPENSSL_assert(ctx->buf_len >= ctx->buf_off);
ret = ctx->buf_len - ctx->buf_off; ret = ctx->buf_len - ctx->buf_off;
if ((ret == 0) && (ctx->encode != B64_NONE) if ((ret == 0) && (ctx->encode != B64_NONE)
&& (ctx->base64.num != 0)) && (EVP_ENCODE_CTX_num(ctx->base64) != 0))
ret = 1; ret = 1;
else if (ret <= 0) else if (ret <= 0)
ret = BIO_ctrl(b->next_bio, cmd, num, ptr); ret = BIO_ctrl(b->next_bio, cmd, num, ptr);
@ -514,9 +516,10 @@ static long b64_ctrl(BIO *b, int cmd, long num, void *ptr)
ctx->tmp_len = 0; ctx->tmp_len = 0;
goto again; goto again;
} }
} else if (ctx->encode != B64_NONE && ctx->base64.num != 0) { } else if (ctx->encode != B64_NONE
&& EVP_ENCODE_CTX_num(ctx->base64) != 0) {
ctx->buf_off = 0; ctx->buf_off = 0;
EVP_EncodeFinal(&(ctx->base64), EVP_EncodeFinal(ctx->base64,
(unsigned char *)ctx->buf, &(ctx->buf_len)); (unsigned char *)ctx->buf, &(ctx->buf_len));
/* push out the bytes */ /* push out the bytes */
goto again; goto again;