#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <openssl/buffer.h>
#include <openssl/evp.h>
#include "bio_local.h"
#include "evp_local.h"
static int enc_write(BIO *h, const char *buf, int num);
static int enc_read(BIO *h, char *buf, int size);
static long enc_ctrl(BIO *h, int cmd, long arg1, void *arg2);
static int enc_new(BIO *h);
static int enc_free(BIO *data);
static long enc_callback_ctrl(BIO *h, int cmd, BIO_info_cb *fps);
#define ENC_BLOCK_SIZE (1024*4)
#define BUF_OFFSET (EVP_MAX_BLOCK_LENGTH*2)
typedef struct enc_struct {
int buf_len;
int buf_off;
int cont;
int finished;
int ok;
EVP_CIPHER_CTX *cipher_ctx;
char buf[ENC_BLOCK_SIZE + BUF_OFFSET + 2];
} BIO_ENC_CTX;
static const BIO_METHOD methods_enc = {
.type = BIO_TYPE_CIPHER,
.name = "cipher",
.bwrite = enc_write,
.bread = enc_read,
.ctrl = enc_ctrl,
.create = enc_new,
.destroy = enc_free,
.callback_ctrl = enc_callback_ctrl
};
const BIO_METHOD *
BIO_f_cipher(void)
{
return &methods_enc;
}
LCRYPTO_ALIAS(BIO_f_cipher);
static void
bio_enc_ctx_free(BIO_ENC_CTX *ctx)
{
if (ctx == NULL)
return;
EVP_CIPHER_CTX_free(ctx->cipher_ctx);
freezero(ctx, sizeof(*ctx));
}
static int
enc_new(BIO *bio)
{
BIO_ENC_CTX *ctx;
int ret = 0;
if ((ctx = calloc(1, sizeof(BIO_ENC_CTX))) == NULL)
goto err;
if ((ctx->cipher_ctx = EVP_CIPHER_CTX_new()) == NULL)
goto err;
ctx->cont = 1;
ctx->ok = 1;
bio->ptr = ctx;
ctx = NULL;
ret = 1;
err:
bio_enc_ctx_free(ctx);
return ret;
}
static int
enc_free(BIO *bio)
{
if (bio == NULL)
return 0;
bio_enc_ctx_free(bio->ptr);
explicit_bzero(bio, sizeof(*bio));
return 1;
}
static int
enc_read(BIO *bio, char *out, int outl)
{
BIO_ENC_CTX *ctx;
int ret = 0, i;
if (out == NULL)
return 0;
ctx = bio->ptr;
if (ctx == NULL || bio->next_bio == NULL)
return 0;
if (ctx->buf_len > 0) {
i = ctx->buf_len - ctx->buf_off;
if (i > outl)
i = outl;
memcpy(out, &(ctx->buf[ctx->buf_off]), i);
ret = i;
out += i;
outl -= i;
ctx->buf_off += i;
if (ctx->buf_len == ctx->buf_off) {
ctx->buf_len = 0;
ctx->buf_off = 0;
}
}
while (outl > 0) {
if (ctx->cont <= 0)
break;
i = BIO_read(bio->next_bio, &ctx->buf[BUF_OFFSET],
ENC_BLOCK_SIZE);
if (i <= 0) {
if (!BIO_should_retry(bio->next_bio)) {
ctx->cont = i;
i = EVP_CipherFinal_ex(ctx->cipher_ctx,
(unsigned char *)ctx->buf,
&(ctx->buf_len));
ctx->ok = i;
ctx->buf_off = 0;
} else {
ret = (ret == 0) ? i : ret;
break;
}
} else {
EVP_CipherUpdate(ctx->cipher_ctx,
(unsigned char *)ctx->buf, &ctx->buf_len,
(unsigned char *)&ctx->buf[BUF_OFFSET], i);
ctx->cont = 1;
if (ctx->buf_len == 0)
continue;
}
if (ctx->buf_len <= outl)
i = ctx->buf_len;
else
i = outl;
if (i <= 0)
break;
memcpy(out, ctx->buf, i);
ret += i;
ctx->buf_off = i;
outl -= i;
out += i;
}
BIO_clear_retry_flags(bio);
BIO_copy_next_retry(bio);
return ret == 0 ? ctx->cont : ret;
}
static int
enc_write(BIO *bio, const char *in, int inl)
{
BIO_ENC_CTX *ctx;
int ret = 0, n, i;
ctx = bio->ptr;
ret = inl;
BIO_clear_retry_flags(bio);
n = ctx->buf_len - ctx->buf_off;
while (n > 0) {
i = BIO_write(bio->next_bio, &(ctx->buf[ctx->buf_off]), n);
if (i <= 0) {
BIO_copy_next_retry(bio);
return i;
}
ctx->buf_off += i;
n -= i;
}
if (in == NULL || inl <= 0)
return 0;
ctx->buf_off = 0;
while (inl > 0) {
n = inl > ENC_BLOCK_SIZE ? ENC_BLOCK_SIZE : inl;
EVP_CipherUpdate(ctx->cipher_ctx,
(unsigned char *)ctx->buf, &ctx->buf_len,
(unsigned char *)in, n);
inl -= n;
in += n;
ctx->buf_off = 0;
n = ctx->buf_len;
while (n > 0) {
i = BIO_write(bio->next_bio, &ctx->buf[ctx->buf_off], n);
if (i <= 0) {
BIO_copy_next_retry(bio);
return ret == inl ? i : ret - inl;
}
n -= i;
ctx->buf_off += i;
}
ctx->buf_len = 0;
ctx->buf_off = 0;
}
BIO_copy_next_retry(bio);
return ret;
}
static long
enc_ctrl(BIO *bio, int cmd, long num, void *ptr)
{
BIO *dbio;
BIO_ENC_CTX *ctx, *dctx;
EVP_CIPHER_CTX **c_ctx;
int i;
long ret = 1;
ctx = bio->ptr;
switch (cmd) {
case BIO_CTRL_RESET:
ctx->ok = 1;
ctx->finished = 0;
EVP_CipherInit_ex(ctx->cipher_ctx, NULL, NULL, NULL, NULL,
ctx->cipher_ctx->encrypt);
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
break;
case BIO_CTRL_EOF:
if (ctx->cont <= 0)
ret = 1;
else
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
break;
case BIO_CTRL_WPENDING:
ret = ctx->buf_len - ctx->buf_off;
if (ret <= 0)
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
break;
case BIO_CTRL_PENDING:
ret = ctx->buf_len - ctx->buf_off;
if (ret <= 0)
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
break;
case BIO_CTRL_FLUSH:
again:
while (ctx->buf_len != ctx->buf_off) {
i = enc_write(bio, NULL, 0);
if (i < 0)
return i;
}
if (!ctx->finished) {
ctx->finished = 1;
ctx->buf_off = 0;
ret = EVP_CipherFinal_ex(ctx->cipher_ctx,
(unsigned char *)ctx->buf,
&ctx->buf_len);
ctx->ok = (int)ret;
if (ret <= 0)
break;
goto again;
}
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
break;
case BIO_C_GET_CIPHER_STATUS:
ret = (long)ctx->ok;
break;
case BIO_C_DO_STATE_MACHINE:
BIO_clear_retry_flags(bio);
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
BIO_copy_next_retry(bio);
break;
case BIO_C_GET_CIPHER_CTX:
c_ctx = ptr;
*c_ctx = ctx->cipher_ctx;
bio->init = 1;
break;
case BIO_CTRL_DUP:
dbio = ptr;
dctx = dbio->ptr;
ret = EVP_CIPHER_CTX_copy(dctx->cipher_ctx, ctx->cipher_ctx);
if (ret)
dbio->init = 1;
break;
default:
ret = BIO_ctrl(bio->next_bio, cmd, num, ptr);
break;
}
return ret;
}
static long
enc_callback_ctrl(BIO *bio, int cmd, BIO_info_cb *fp)
{
long ret = 1;
if (bio->next_bio == NULL)
return 0;
switch (cmd) {
default:
ret = BIO_callback_ctrl(bio->next_bio, cmd, fp);
break;
}
return ret;
}
int
BIO_set_cipher(BIO *bio, const EVP_CIPHER *c, const unsigned char *k,
const unsigned char *i, int e)
{
BIO_ENC_CTX *ctx;
long (*cb)(BIO *, int, const char *, int, long, long);
if (bio == NULL)
return 0;
if ((ctx = BIO_get_data(bio)) == NULL)
return 0;
if ((cb = BIO_get_callback(bio)) != NULL) {
if (cb(bio, BIO_CB_CTRL, (const char *)c, BIO_CTRL_SET, e, 0L) <= 0)
return 0;
}
BIO_set_init(bio, 1);
if (!EVP_CipherInit_ex(ctx->cipher_ctx, c, NULL, k, i, e))
return 0;
if (cb != NULL)
return cb(bio, BIO_CB_CTRL, (const char *)c, BIO_CTRL_SET, e, 1L);
return 1;
}
LCRYPTO_ALIAS(BIO_set_cipher);