Blame crypto/evp/evp_enc.c

Packit c4476c
/*
Packit c4476c
 * Copyright 1995-2018 The OpenSSL Project Authors. All Rights Reserved.
Packit c4476c
 *
Packit c4476c
 * Licensed under the OpenSSL license (the "License").  You may not use
Packit c4476c
 * this file except in compliance with the License.  You can obtain a copy
Packit c4476c
 * in the file LICENSE in the source distribution or at
Packit c4476c
 * https://www.openssl.org/source/license.html
Packit c4476c
 */
Packit c4476c
Packit c4476c
#include <stdio.h>
Packit c4476c
#include <assert.h>
Packit c4476c
#include "internal/cryptlib.h"
Packit c4476c
#include <openssl/evp.h>
Packit c4476c
#include <openssl/err.h>
Packit c4476c
#include <openssl/rand.h>
Packit c4476c
#include <openssl/rand_drbg.h>
Packit c4476c
#include <openssl/engine.h>
Packit c4476c
#include "crypto/evp.h"
Packit c4476c
#include "evp_local.h"
Packit c4476c
#ifdef OPENSSL_FIPS
Packit c4476c
# include <openssl/fips.h>
Packit c4476c
#endif
Packit c4476c
Packit c4476c
int EVP_CIPHER_CTX_reset(EVP_CIPHER_CTX *c)
Packit c4476c
{
Packit c4476c
#ifdef OPENSSL_FIPS
Packit c4476c
    if (FIPS_selftest_failed()) {
Packit c4476c
        FIPSerr(FIPS_F_EVP_CIPHER_CTX_RESET, FIPS_R_FIPS_SELFTEST_FAILED);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
#endif
Packit c4476c
    if (c == NULL)
Packit c4476c
        return 1;
Packit c4476c
    if (c->cipher != NULL) {
Packit c4476c
        if (c->cipher->cleanup && !c->cipher->cleanup(c))
Packit c4476c
            return 0;
Packit c4476c
        /* Cleanse cipher context data */
Packit c4476c
        if (c->cipher_data && c->cipher->ctx_size)
Packit c4476c
            OPENSSL_cleanse(c->cipher_data, c->cipher->ctx_size);
Packit c4476c
    }
Packit c4476c
    OPENSSL_free(c->cipher_data);
Packit c4476c
#ifndef OPENSSL_NO_ENGINE
Packit c4476c
    ENGINE_finish(c->engine);
Packit c4476c
#endif
Packit c4476c
    memset(c, 0, sizeof(*c));
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
EVP_CIPHER_CTX *EVP_CIPHER_CTX_new(void)
Packit c4476c
{
Packit c4476c
#ifdef OPENSSL_FIPS
Packit c4476c
    if (FIPS_selftest_failed()) {
Packit c4476c
        FIPSerr(FIPS_F_EVP_CIPHER_CTX_NEW, FIPS_R_FIPS_SELFTEST_FAILED);
Packit c4476c
        return NULL;
Packit c4476c
    }
Packit c4476c
#endif
Packit c4476c
    return OPENSSL_zalloc(sizeof(EVP_CIPHER_CTX));
Packit c4476c
}
Packit c4476c
Packit c4476c
void EVP_CIPHER_CTX_free(EVP_CIPHER_CTX *ctx)
Packit c4476c
{
Packit c4476c
    EVP_CIPHER_CTX_reset(ctx);
Packit c4476c
    OPENSSL_free(ctx);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CipherInit(EVP_CIPHER_CTX *ctx, const EVP_CIPHER *cipher,
Packit c4476c
                   const unsigned char *key, const unsigned char *iv, int enc)
Packit c4476c
{
Packit c4476c
    if (cipher != NULL)
Packit c4476c
        EVP_CIPHER_CTX_reset(ctx);
Packit c4476c
    return EVP_CipherInit_ex(ctx, cipher, NULL, key, iv, enc);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CipherInit_ex(EVP_CIPHER_CTX *ctx, const EVP_CIPHER *cipher,
Packit c4476c
                      ENGINE *impl, const unsigned char *key,
Packit c4476c
                      const unsigned char *iv, int enc)
Packit c4476c
{
Packit c4476c
    if (enc == -1)
Packit c4476c
        enc = ctx->encrypt;
Packit c4476c
    else {
Packit c4476c
        if (enc)
Packit c4476c
            enc = 1;
Packit c4476c
        ctx->encrypt = enc;
Packit c4476c
    }
Packit c4476c
#ifdef OPENSSL_FIPS
Packit c4476c
    if (FIPS_selftest_failed()) {
Packit c4476c
        FIPSerr(FIPS_F_EVP_CIPHERINIT_EX, FIPS_R_FIPS_SELFTEST_FAILED);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
#endif
Packit c4476c
#ifndef OPENSSL_NO_ENGINE
Packit c4476c
    /*
Packit c4476c
     * Whether it's nice or not, "Inits" can be used on "Final"'d contexts so
Packit c4476c
     * this context may already have an ENGINE! Try to avoid releasing the
Packit c4476c
     * previous handle, re-querying for an ENGINE, and having a
Packit c4476c
     * reinitialisation, when it may all be unnecessary.
Packit c4476c
     */
Packit c4476c
    if (ctx->engine && ctx->cipher
Packit c4476c
        && (cipher == NULL || cipher->nid == ctx->cipher->nid))
Packit c4476c
        goto skip_to_init;
Packit c4476c
#endif
Packit c4476c
    if (cipher) {
Packit c4476c
        /*
Packit c4476c
         * Ensure a context left lying around from last time is cleared (the
Packit c4476c
         * previous check attempted to avoid this if the same ENGINE and
Packit c4476c
         * EVP_CIPHER could be used).
Packit c4476c
         */
Packit c4476c
        if (ctx->cipher) {
Packit c4476c
            unsigned long flags = ctx->flags;
Packit c4476c
            EVP_CIPHER_CTX_reset(ctx);
Packit c4476c
            /* Restore encrypt and flags */
Packit c4476c
            ctx->encrypt = enc;
Packit c4476c
            ctx->flags = flags;
Packit c4476c
        }
Packit c4476c
#ifndef OPENSSL_NO_ENGINE
Packit c4476c
        if (impl) {
Packit c4476c
            if (!ENGINE_init(impl)) {
Packit c4476c
                EVPerr(EVP_F_EVP_CIPHERINIT_EX, EVP_R_INITIALIZATION_ERROR);
Packit c4476c
                return 0;
Packit c4476c
            }
Packit c4476c
        } else
Packit c4476c
            /* Ask if an ENGINE is reserved for this job */
Packit c4476c
            impl = ENGINE_get_cipher_engine(cipher->nid);
Packit c4476c
        if (impl) {
Packit c4476c
            /* There's an ENGINE for this job ... (apparently) */
Packit c4476c
            const EVP_CIPHER *c = ENGINE_get_cipher(impl, cipher->nid);
Packit c4476c
            if (!c) {
Packit c4476c
                /*
Packit c4476c
                 * One positive side-effect of US's export control history,
Packit c4476c
                 * is that we should at least be able to avoid using US
Packit c4476c
                 * misspellings of "initialisation"?
Packit c4476c
                 */
Packit c4476c
                EVPerr(EVP_F_EVP_CIPHERINIT_EX, EVP_R_INITIALIZATION_ERROR);
Packit c4476c
                return 0;
Packit c4476c
            }
Packit c4476c
            /* We'll use the ENGINE's private cipher definition */
Packit c4476c
            cipher = c;
Packit c4476c
            /*
Packit c4476c
             * Store the ENGINE functional reference so we know 'cipher' came
Packit c4476c
             * from an ENGINE and we need to release it when done.
Packit c4476c
             */
Packit c4476c
            ctx->engine = impl;
Packit c4476c
        } else
Packit c4476c
            ctx->engine = NULL;
Packit c4476c
#endif
Packit c4476c
Packit c4476c
        ctx->cipher = cipher;
Packit c4476c
        if (ctx->cipher->ctx_size) {
Packit c4476c
            ctx->cipher_data = OPENSSL_zalloc(ctx->cipher->ctx_size);
Packit c4476c
            if (ctx->cipher_data == NULL) {
Packit c4476c
                ctx->cipher = NULL;
Packit c4476c
                EVPerr(EVP_F_EVP_CIPHERINIT_EX, ERR_R_MALLOC_FAILURE);
Packit c4476c
                return 0;
Packit c4476c
            }
Packit c4476c
        } else {
Packit c4476c
            ctx->cipher_data = NULL;
Packit c4476c
        }
Packit c4476c
        ctx->key_len = cipher->key_len;
Packit c4476c
        /* Preserve wrap enable flag, zero everything else */
Packit c4476c
        ctx->flags &= EVP_CIPHER_CTX_FLAG_WRAP_ALLOW | EVP_CIPH_FLAG_NON_FIPS_ALLOW;
Packit c4476c
        if (ctx->cipher->flags & EVP_CIPH_CTRL_INIT) {
Packit c4476c
            if (!EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_INIT, 0, NULL)) {
Packit c4476c
                ctx->cipher = NULL;
Packit c4476c
                EVPerr(EVP_F_EVP_CIPHERINIT_EX, EVP_R_INITIALIZATION_ERROR);
Packit c4476c
                return 0;
Packit c4476c
            }
Packit c4476c
        }
Packit c4476c
    } else if (!ctx->cipher) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHERINIT_EX, EVP_R_NO_CIPHER_SET);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
#ifndef OPENSSL_NO_ENGINE
Packit c4476c
 skip_to_init:
Packit c4476c
#endif
Packit c4476c
    /* we assume block size is a power of 2 in *cryptUpdate */
Packit c4476c
    OPENSSL_assert(ctx->cipher->block_size == 1
Packit c4476c
                   || ctx->cipher->block_size == 8
Packit c4476c
                   || ctx->cipher->block_size == 16);
Packit c4476c
Packit c4476c
    if (!(ctx->flags & EVP_CIPHER_CTX_FLAG_WRAP_ALLOW)
Packit c4476c
        && EVP_CIPHER_CTX_mode(ctx) == EVP_CIPH_WRAP_MODE) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHERINIT_EX, EVP_R_WRAP_MODE_NOT_ALLOWED);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (!(EVP_CIPHER_flags(EVP_CIPHER_CTX_cipher(ctx)) & EVP_CIPH_CUSTOM_IV)) {
Packit c4476c
        switch (EVP_CIPHER_CTX_mode(ctx)) {
Packit c4476c
Packit c4476c
        case EVP_CIPH_STREAM_CIPHER:
Packit c4476c
        case EVP_CIPH_ECB_MODE:
Packit c4476c
            break;
Packit c4476c
Packit c4476c
        case EVP_CIPH_CFB_MODE:
Packit c4476c
        case EVP_CIPH_OFB_MODE:
Packit c4476c
Packit c4476c
            ctx->num = 0;
Packit c4476c
            /* fall-through */
Packit c4476c
Packit c4476c
        case EVP_CIPH_CBC_MODE:
Packit c4476c
Packit c4476c
            OPENSSL_assert(EVP_CIPHER_CTX_iv_length(ctx) <=
Packit c4476c
                           (int)sizeof(ctx->iv));
Packit c4476c
            if (iv)
Packit c4476c
                memcpy(ctx->oiv, iv, EVP_CIPHER_CTX_iv_length(ctx));
Packit c4476c
            memcpy(ctx->iv, ctx->oiv, EVP_CIPHER_CTX_iv_length(ctx));
Packit c4476c
            break;
Packit c4476c
Packit c4476c
        case EVP_CIPH_CTR_MODE:
Packit c4476c
            ctx->num = 0;
Packit c4476c
            /* Don't reuse IV for CTR mode */
Packit c4476c
            if (iv)
Packit c4476c
                memcpy(ctx->iv, iv, EVP_CIPHER_CTX_iv_length(ctx));
Packit c4476c
            break;
Packit c4476c
Packit c4476c
        default:
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
    }
Packit c4476c
#ifdef OPENSSL_FIPS
Packit c4476c
    /* After 'key' is set no further parameters changes are permissible.
Packit c4476c
     * So only check for non FIPS enabling at this point.
Packit c4476c
     */
Packit c4476c
    if (key && FIPS_mode()) {
Packit c4476c
        if (!(ctx->cipher->flags & EVP_CIPH_FLAG_FIPS)
Packit c4476c
            & !(ctx->flags & EVP_CIPH_FLAG_NON_FIPS_ALLOW)) {
Packit c4476c
            EVPerr(EVP_F_EVP_CIPHERINIT_EX, EVP_R_DISABLED_FOR_FIPS);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
    }
Packit c4476c
#endif
Packit c4476c
Packit c4476c
    if (key || (ctx->cipher->flags & EVP_CIPH_ALWAYS_CALL_INIT)) {
Packit c4476c
        if (!ctx->cipher->init(ctx, key, iv, enc))
Packit c4476c
            return 0;
Packit c4476c
    }
Packit c4476c
    ctx->buf_len = 0;
Packit c4476c
    ctx->final_used = 0;
Packit c4476c
    ctx->block_mask = ctx->cipher->block_size - 1;
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CipherUpdate(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl,
Packit c4476c
                     const unsigned char *in, int inl)
Packit c4476c
{
Packit c4476c
    if (ctx->encrypt)
Packit c4476c
        return EVP_EncryptUpdate(ctx, out, outl, in, inl);
Packit c4476c
    else
Packit c4476c
        return EVP_DecryptUpdate(ctx, out, outl, in, inl);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CipherFinal_ex(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl)
Packit c4476c
{
Packit c4476c
    if (ctx->encrypt)
Packit c4476c
        return EVP_EncryptFinal_ex(ctx, out, outl);
Packit c4476c
    else
Packit c4476c
        return EVP_DecryptFinal_ex(ctx, out, outl);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CipherFinal(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl)
Packit c4476c
{
Packit c4476c
    if (ctx->encrypt)
Packit c4476c
        return EVP_EncryptFinal(ctx, out, outl);
Packit c4476c
    else
Packit c4476c
        return EVP_DecryptFinal(ctx, out, outl);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_EncryptInit(EVP_CIPHER_CTX *ctx, const EVP_CIPHER *cipher,
Packit c4476c
                    const unsigned char *key, const unsigned char *iv)
Packit c4476c
{
Packit c4476c
    return EVP_CipherInit(ctx, cipher, key, iv, 1);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_EncryptInit_ex(EVP_CIPHER_CTX *ctx, const EVP_CIPHER *cipher,
Packit c4476c
                       ENGINE *impl, const unsigned char *key,
Packit c4476c
                       const unsigned char *iv)
Packit c4476c
{
Packit c4476c
    return EVP_CipherInit_ex(ctx, cipher, impl, key, iv, 1);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_DecryptInit(EVP_CIPHER_CTX *ctx, const EVP_CIPHER *cipher,
Packit c4476c
                    const unsigned char *key, const unsigned char *iv)
Packit c4476c
{
Packit c4476c
    return EVP_CipherInit(ctx, cipher, key, iv, 0);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_DecryptInit_ex(EVP_CIPHER_CTX *ctx, const EVP_CIPHER *cipher,
Packit c4476c
                       ENGINE *impl, const unsigned char *key,
Packit c4476c
                       const unsigned char *iv)
Packit c4476c
{
Packit c4476c
    return EVP_CipherInit_ex(ctx, cipher, impl, key, iv, 0);
Packit c4476c
}
Packit c4476c
Packit c4476c
/*
Packit c4476c
 * According to the letter of standard difference between pointers
Packit c4476c
 * is specified to be valid only within same object. This makes
Packit c4476c
 * it formally challenging to determine if input and output buffers
Packit c4476c
 * are not partially overlapping with standard pointer arithmetic.
Packit c4476c
 */
Packit c4476c
#ifdef PTRDIFF_T
Packit c4476c
# undef PTRDIFF_T
Packit c4476c
#endif
Packit c4476c
#if defined(OPENSSL_SYS_VMS) && __INITIAL_POINTER_SIZE==64
Packit c4476c
/*
Packit c4476c
 * Then we have VMS that distinguishes itself by adhering to
Packit c4476c
 * sizeof(size_t)==4 even in 64-bit builds, which means that
Packit c4476c
 * difference between two pointers might be truncated to 32 bits.
Packit c4476c
 * In the context one can even wonder how comparison for
Packit c4476c
 * equality is implemented. To be on the safe side we adhere to
Packit c4476c
 * PTRDIFF_T even for comparison for equality.
Packit c4476c
 */
Packit c4476c
# define PTRDIFF_T uint64_t
Packit c4476c
#else
Packit c4476c
# define PTRDIFF_T size_t
Packit c4476c
#endif
Packit c4476c
Packit c4476c
int is_partially_overlapping(const void *ptr1, const void *ptr2, int len)
Packit c4476c
{
Packit c4476c
    PTRDIFF_T diff = (PTRDIFF_T)ptr1-(PTRDIFF_T)ptr2;
Packit c4476c
    /*
Packit c4476c
     * Check for partially overlapping buffers. [Binary logical
Packit c4476c
     * operations are used instead of boolean to minimize number
Packit c4476c
     * of conditional branches.]
Packit c4476c
     */
Packit c4476c
    int overlapped = (len > 0) & (diff != 0) & ((diff < (PTRDIFF_T)len) |
Packit c4476c
                                                (diff > (0 - (PTRDIFF_T)len)));
Packit c4476c
Packit c4476c
    return overlapped;
Packit c4476c
}
Packit c4476c
Packit c4476c
static int evp_EncryptDecryptUpdate(EVP_CIPHER_CTX *ctx,
Packit c4476c
                                    unsigned char *out, int *outl,
Packit c4476c
                                    const unsigned char *in, int inl)
Packit c4476c
{
Packit c4476c
    int i, j, bl, cmpl = inl;
Packit c4476c
Packit c4476c
    if (EVP_CIPHER_CTX_test_flags(ctx, EVP_CIPH_FLAG_LENGTH_BITS))
Packit c4476c
        cmpl = (cmpl + 7) / 8;
Packit c4476c
Packit c4476c
    bl = ctx->cipher->block_size;
Packit c4476c
Packit c4476c
    /*
Packit c4476c
     * CCM mode needs to know about the case where inl == 0 && in == NULL - it
Packit c4476c
     * means the plaintext/ciphertext length is 0
Packit c4476c
     */
Packit c4476c
    if (inl < 0
Packit c4476c
            || (inl == 0
Packit c4476c
                && EVP_CIPHER_mode(ctx->cipher) != EVP_CIPH_CCM_MODE)) {
Packit c4476c
        *outl = 0;
Packit c4476c
        return inl == 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (ctx->cipher->flags & EVP_CIPH_FLAG_CUSTOM_CIPHER) {
Packit c4476c
        /* If block size > 1 then the cipher will have to do this check */
Packit c4476c
        if (bl == 1 && is_partially_overlapping(out, in, cmpl)) {
Packit c4476c
            EVPerr(EVP_F_EVP_ENCRYPTDECRYPTUPDATE, EVP_R_PARTIALLY_OVERLAPPING);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
Packit c4476c
        i = ctx->cipher->do_cipher(ctx, out, in, inl);
Packit c4476c
        if (i < 0)
Packit c4476c
            return 0;
Packit c4476c
        else
Packit c4476c
            *outl = i;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (is_partially_overlapping(out + ctx->buf_len, in, cmpl)) {
Packit c4476c
        EVPerr(EVP_F_EVP_ENCRYPTDECRYPTUPDATE, EVP_R_PARTIALLY_OVERLAPPING);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (ctx->buf_len == 0 && (inl & (ctx->block_mask)) == 0) {
Packit c4476c
        if (ctx->cipher->do_cipher(ctx, out, in, inl)) {
Packit c4476c
            *outl = inl;
Packit c4476c
            return 1;
Packit c4476c
        } else {
Packit c4476c
            *outl = 0;
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
    }
Packit c4476c
    i = ctx->buf_len;
Packit c4476c
    OPENSSL_assert(bl <= (int)sizeof(ctx->buf));
Packit c4476c
    if (i != 0) {
Packit c4476c
        if (bl - i > inl) {
Packit c4476c
            memcpy(&(ctx->buf[i]), in, inl);
Packit c4476c
            ctx->buf_len += inl;
Packit c4476c
            *outl = 0;
Packit c4476c
            return 1;
Packit c4476c
        } else {
Packit c4476c
            j = bl - i;
Packit c4476c
            memcpy(&(ctx->buf[i]), in, j);
Packit c4476c
            inl -= j;
Packit c4476c
            in += j;
Packit c4476c
            if (!ctx->cipher->do_cipher(ctx, out, ctx->buf, bl))
Packit c4476c
                return 0;
Packit c4476c
            out += bl;
Packit c4476c
            *outl = bl;
Packit c4476c
        }
Packit c4476c
    } else
Packit c4476c
        *outl = 0;
Packit c4476c
    i = inl & (bl - 1);
Packit c4476c
    inl -= i;
Packit c4476c
    if (inl > 0) {
Packit c4476c
        if (!ctx->cipher->do_cipher(ctx, out, in, inl))
Packit c4476c
            return 0;
Packit c4476c
        *outl += inl;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (i != 0)
Packit c4476c
        memcpy(ctx->buf, &(in[inl]), i);
Packit c4476c
    ctx->buf_len = i;
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
Packit c4476c
int EVP_EncryptUpdate(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl,
Packit c4476c
                      const unsigned char *in, int inl)
Packit c4476c
{
Packit c4476c
    /* Prevent accidental use of decryption context when encrypting */
Packit c4476c
    if (!ctx->encrypt) {
Packit c4476c
        EVPerr(EVP_F_EVP_ENCRYPTUPDATE, EVP_R_INVALID_OPERATION);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    return evp_EncryptDecryptUpdate(ctx, out, outl, in, inl);
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_EncryptFinal(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl)
Packit c4476c
{
Packit c4476c
    int ret;
Packit c4476c
    ret = EVP_EncryptFinal_ex(ctx, out, outl);
Packit c4476c
    return ret;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_EncryptFinal_ex(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl)
Packit c4476c
{
Packit c4476c
    int n, ret;
Packit c4476c
    unsigned int i, b, bl;
Packit c4476c
Packit c4476c
    /* Prevent accidental use of decryption context when encrypting */
Packit c4476c
    if (!ctx->encrypt) {
Packit c4476c
        EVPerr(EVP_F_EVP_ENCRYPTFINAL_EX, EVP_R_INVALID_OPERATION);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (ctx->cipher->flags & EVP_CIPH_FLAG_CUSTOM_CIPHER) {
Packit c4476c
        ret = ctx->cipher->do_cipher(ctx, out, NULL, 0);
Packit c4476c
        if (ret < 0)
Packit c4476c
            return 0;
Packit c4476c
        else
Packit c4476c
            *outl = ret;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    b = ctx->cipher->block_size;
Packit c4476c
    OPENSSL_assert(b <= sizeof(ctx->buf));
Packit c4476c
    if (b == 1) {
Packit c4476c
        *outl = 0;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
    bl = ctx->buf_len;
Packit c4476c
    if (ctx->flags & EVP_CIPH_NO_PADDING) {
Packit c4476c
        if (bl) {
Packit c4476c
            EVPerr(EVP_F_EVP_ENCRYPTFINAL_EX,
Packit c4476c
                   EVP_R_DATA_NOT_MULTIPLE_OF_BLOCK_LENGTH);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
        *outl = 0;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    n = b - bl;
Packit c4476c
    for (i = bl; i < b; i++)
Packit c4476c
        ctx->buf[i] = n;
Packit c4476c
    ret = ctx->cipher->do_cipher(ctx, out, ctx->buf, b);
Packit c4476c
Packit c4476c
    if (ret)
Packit c4476c
        *outl = b;
Packit c4476c
Packit c4476c
    return ret;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_DecryptUpdate(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl,
Packit c4476c
                      const unsigned char *in, int inl)
Packit c4476c
{
Packit c4476c
    int fix_len, cmpl = inl;
Packit c4476c
    unsigned int b;
Packit c4476c
Packit c4476c
    /* Prevent accidental use of encryption context when decrypting */
Packit c4476c
    if (ctx->encrypt) {
Packit c4476c
        EVPerr(EVP_F_EVP_DECRYPTUPDATE, EVP_R_INVALID_OPERATION);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    b = ctx->cipher->block_size;
Packit c4476c
Packit c4476c
    if (EVP_CIPHER_CTX_test_flags(ctx, EVP_CIPH_FLAG_LENGTH_BITS))
Packit c4476c
        cmpl = (cmpl + 7) / 8;
Packit c4476c
Packit c4476c
    /*
Packit c4476c
     * CCM mode needs to know about the case where inl == 0 - it means the
Packit c4476c
     * plaintext/ciphertext length is 0
Packit c4476c
     */
Packit c4476c
    if (inl < 0
Packit c4476c
            || (inl == 0
Packit c4476c
                && EVP_CIPHER_mode(ctx->cipher) != EVP_CIPH_CCM_MODE)) {
Packit c4476c
        *outl = 0;
Packit c4476c
        return inl == 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (ctx->cipher->flags & EVP_CIPH_FLAG_CUSTOM_CIPHER) {
Packit c4476c
        if (b == 1 && is_partially_overlapping(out, in, cmpl)) {
Packit c4476c
            EVPerr(EVP_F_EVP_DECRYPTUPDATE, EVP_R_PARTIALLY_OVERLAPPING);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
Packit c4476c
        fix_len = ctx->cipher->do_cipher(ctx, out, in, inl);
Packit c4476c
        if (fix_len < 0) {
Packit c4476c
            *outl = 0;
Packit c4476c
            return 0;
Packit c4476c
        } else
Packit c4476c
            *outl = fix_len;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (ctx->flags & EVP_CIPH_NO_PADDING)
Packit c4476c
        return evp_EncryptDecryptUpdate(ctx, out, outl, in, inl);
Packit c4476c
Packit c4476c
    OPENSSL_assert(b <= sizeof(ctx->final));
Packit c4476c
Packit c4476c
    if (ctx->final_used) {
Packit c4476c
        /* see comment about PTRDIFF_T comparison above */
Packit c4476c
        if (((PTRDIFF_T)out == (PTRDIFF_T)in)
Packit c4476c
            || is_partially_overlapping(out, in, b)) {
Packit c4476c
            EVPerr(EVP_F_EVP_DECRYPTUPDATE, EVP_R_PARTIALLY_OVERLAPPING);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
        memcpy(out, ctx->final, b);
Packit c4476c
        out += b;
Packit c4476c
        fix_len = 1;
Packit c4476c
    } else
Packit c4476c
        fix_len = 0;
Packit c4476c
Packit c4476c
    if (!evp_EncryptDecryptUpdate(ctx, out, outl, in, inl))
Packit c4476c
        return 0;
Packit c4476c
Packit c4476c
    /*
Packit c4476c
     * if we have 'decrypted' a multiple of block size, make sure we have a
Packit c4476c
     * copy of this last block
Packit c4476c
     */
Packit c4476c
    if (b > 1 && !ctx->buf_len) {
Packit c4476c
        *outl -= b;
Packit c4476c
        ctx->final_used = 1;
Packit c4476c
        memcpy(ctx->final, &out[*outl], b);
Packit c4476c
    } else
Packit c4476c
        ctx->final_used = 0;
Packit c4476c
Packit c4476c
    if (fix_len)
Packit c4476c
        *outl += b;
Packit c4476c
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_DecryptFinal(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl)
Packit c4476c
{
Packit c4476c
    int ret;
Packit c4476c
    ret = EVP_DecryptFinal_ex(ctx, out, outl);
Packit c4476c
    return ret;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_DecryptFinal_ex(EVP_CIPHER_CTX *ctx, unsigned char *out, int *outl)
Packit c4476c
{
Packit c4476c
    int i, n;
Packit c4476c
    unsigned int b;
Packit c4476c
Packit c4476c
    /* Prevent accidental use of encryption context when decrypting */
Packit c4476c
    if (ctx->encrypt) {
Packit c4476c
        EVPerr(EVP_F_EVP_DECRYPTFINAL_EX, EVP_R_INVALID_OPERATION);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    *outl = 0;
Packit c4476c
Packit c4476c
    if (ctx->cipher->flags & EVP_CIPH_FLAG_CUSTOM_CIPHER) {
Packit c4476c
        i = ctx->cipher->do_cipher(ctx, out, NULL, 0);
Packit c4476c
        if (i < 0)
Packit c4476c
            return 0;
Packit c4476c
        else
Packit c4476c
            *outl = i;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    b = ctx->cipher->block_size;
Packit c4476c
    if (ctx->flags & EVP_CIPH_NO_PADDING) {
Packit c4476c
        if (ctx->buf_len) {
Packit c4476c
            EVPerr(EVP_F_EVP_DECRYPTFINAL_EX,
Packit c4476c
                   EVP_R_DATA_NOT_MULTIPLE_OF_BLOCK_LENGTH);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
        *outl = 0;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
    if (b > 1) {
Packit c4476c
        if (ctx->buf_len || !ctx->final_used) {
Packit c4476c
            EVPerr(EVP_F_EVP_DECRYPTFINAL_EX, EVP_R_WRONG_FINAL_BLOCK_LENGTH);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
        OPENSSL_assert(b <= sizeof(ctx->final));
Packit c4476c
Packit c4476c
        /*
Packit c4476c
         * The following assumes that the ciphertext has been authenticated.
Packit c4476c
         * Otherwise it provides a padding oracle.
Packit c4476c
         */
Packit c4476c
        n = ctx->final[b - 1];
Packit c4476c
        if (n == 0 || n > (int)b) {
Packit c4476c
            EVPerr(EVP_F_EVP_DECRYPTFINAL_EX, EVP_R_BAD_DECRYPT);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
        for (i = 0; i < n; i++) {
Packit c4476c
            if (ctx->final[--b] != n) {
Packit c4476c
                EVPerr(EVP_F_EVP_DECRYPTFINAL_EX, EVP_R_BAD_DECRYPT);
Packit c4476c
                return 0;
Packit c4476c
            }
Packit c4476c
        }
Packit c4476c
        n = ctx->cipher->block_size - n;
Packit c4476c
        for (i = 0; i < n; i++)
Packit c4476c
            out[i] = ctx->final[i];
Packit c4476c
        *outl = n;
Packit c4476c
    } else
Packit c4476c
        *outl = 0;
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CIPHER_CTX_set_key_length(EVP_CIPHER_CTX *c, int keylen)
Packit c4476c
{
Packit c4476c
    if (c->cipher->flags & EVP_CIPH_CUSTOM_KEY_LENGTH)
Packit c4476c
        return EVP_CIPHER_CTX_ctrl(c, EVP_CTRL_SET_KEY_LENGTH, keylen, NULL);
Packit c4476c
    if (c->key_len == keylen)
Packit c4476c
        return 1;
Packit c4476c
    if ((keylen > 0) && (c->cipher->flags & EVP_CIPH_VARIABLE_LENGTH)) {
Packit c4476c
        c->key_len = keylen;
Packit c4476c
        return 1;
Packit c4476c
    }
Packit c4476c
    EVPerr(EVP_F_EVP_CIPHER_CTX_SET_KEY_LENGTH, EVP_R_INVALID_KEY_LENGTH);
Packit c4476c
    return 0;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CIPHER_CTX_set_padding(EVP_CIPHER_CTX *ctx, int pad)
Packit c4476c
{
Packit c4476c
    if (pad)
Packit c4476c
        ctx->flags &= ~EVP_CIPH_NO_PADDING;
Packit c4476c
    else
Packit c4476c
        ctx->flags |= EVP_CIPH_NO_PADDING;
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CIPHER_CTX_ctrl(EVP_CIPHER_CTX *ctx, int type, int arg, void *ptr)
Packit c4476c
{
Packit c4476c
    int ret;
Packit c4476c
Packit c4476c
    if (!ctx->cipher) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHER_CTX_CTRL, EVP_R_NO_CIPHER_SET);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (!ctx->cipher->ctrl) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHER_CTX_CTRL, EVP_R_CTRL_NOT_IMPLEMENTED);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
Packit c4476c
    ret = ctx->cipher->ctrl(ctx, type, arg, ptr);
Packit c4476c
    if (ret == -1) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHER_CTX_CTRL,
Packit c4476c
               EVP_R_CTRL_OPERATION_NOT_IMPLEMENTED);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
    return ret;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CIPHER_CTX_rand_key(EVP_CIPHER_CTX *ctx, unsigned char *key)
Packit c4476c
{
Packit c4476c
    if (ctx->cipher->flags & EVP_CIPH_RAND_KEY)
Packit c4476c
        return EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_RAND_KEY, 0, key);
Packit c4476c
    if (RAND_priv_bytes(key, ctx->key_len) <= 0)
Packit c4476c
        return 0;
Packit c4476c
    return 1;
Packit c4476c
}
Packit c4476c
Packit c4476c
int EVP_CIPHER_CTX_copy(EVP_CIPHER_CTX *out, const EVP_CIPHER_CTX *in)
Packit c4476c
{
Packit c4476c
    if ((in == NULL) || (in->cipher == NULL)) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHER_CTX_COPY, EVP_R_INPUT_NOT_INITIALIZED);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
#ifndef OPENSSL_NO_ENGINE
Packit c4476c
    /* Make sure it's safe to copy a cipher context using an ENGINE */
Packit c4476c
    if (in->engine && !ENGINE_init(in->engine)) {
Packit c4476c
        EVPerr(EVP_F_EVP_CIPHER_CTX_COPY, ERR_R_ENGINE_LIB);
Packit c4476c
        return 0;
Packit c4476c
    }
Packit c4476c
#endif
Packit c4476c
Packit c4476c
    EVP_CIPHER_CTX_reset(out);
Packit c4476c
    memcpy(out, in, sizeof(*out));
Packit c4476c
Packit c4476c
    if (in->cipher_data && in->cipher->ctx_size) {
Packit c4476c
        out->cipher_data = OPENSSL_malloc(in->cipher->ctx_size);
Packit c4476c
        if (out->cipher_data == NULL) {
Packit c4476c
            out->cipher = NULL;
Packit c4476c
            EVPerr(EVP_F_EVP_CIPHER_CTX_COPY, ERR_R_MALLOC_FAILURE);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
        memcpy(out->cipher_data, in->cipher_data, in->cipher->ctx_size);
Packit c4476c
    }
Packit c4476c
Packit c4476c
    if (in->cipher->flags & EVP_CIPH_CUSTOM_COPY)
Packit c4476c
        if (!in->cipher->ctrl((EVP_CIPHER_CTX *)in, EVP_CTRL_COPY, 0, out)) {
Packit c4476c
            out->cipher = NULL;
Packit c4476c
            EVPerr(EVP_F_EVP_CIPHER_CTX_COPY, EVP_R_INITIALIZATION_ERROR);
Packit c4476c
            return 0;
Packit c4476c
        }
Packit c4476c
    return 1;
Packit c4476c
}