Blame crypto/evp/evp_key.c

Packit Service 084de1
/*
Packit Service 084de1
 * Copyright 1995-2016 The OpenSSL Project Authors. All Rights Reserved.
Packit Service 084de1
 *
Packit Service 084de1
 * Licensed under the OpenSSL license (the "License").  You may not use
Packit Service 084de1
 * this file except in compliance with the License.  You can obtain a copy
Packit Service 084de1
 * in the file LICENSE in the source distribution or at
Packit Service 084de1
 * https://www.openssl.org/source/license.html
Packit Service 084de1
 */
Packit Service 084de1
Packit Service 084de1
#include <stdio.h>
Packit Service 084de1
#include "internal/cryptlib.h"
Packit Service 084de1
#include <openssl/x509.h>
Packit Service 084de1
#include <openssl/objects.h>
Packit Service 084de1
#include <openssl/evp.h>
Packit Service 084de1
#include <openssl/ui.h>
Packit Service 084de1
Packit Service 084de1
/* should be init to zeros. */
Packit Service 084de1
static char prompt_string[80];
Packit Service 084de1
Packit Service 084de1
void EVP_set_pw_prompt(const char *prompt)
Packit Service 084de1
{
Packit Service 084de1
    if (prompt == NULL)
Packit Service 084de1
        prompt_string[0] = '\0';
Packit Service 084de1
    else {
Packit Service 084de1
        strncpy(prompt_string, prompt, 79);
Packit Service 084de1
        prompt_string[79] = '\0';
Packit Service 084de1
    }
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
char *EVP_get_pw_prompt(void)
Packit Service 084de1
{
Packit Service 084de1
    if (prompt_string[0] == '\0')
Packit Service 084de1
        return NULL;
Packit Service 084de1
    else
Packit Service 084de1
        return prompt_string;
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
/*
Packit Service 084de1
 * For historical reasons, the standard function for reading passwords is in
Packit Service 084de1
 * the DES library -- if someone ever wants to disable DES, this function
Packit Service 084de1
 * will fail
Packit Service 084de1
 */
Packit Service 084de1
int EVP_read_pw_string(char *buf, int len, const char *prompt, int verify)
Packit Service 084de1
{
Packit Service 084de1
    return EVP_read_pw_string_min(buf, 0, len, prompt, verify);
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
int EVP_read_pw_string_min(char *buf, int min, int len, const char *prompt,
Packit Service 084de1
                           int verify)
Packit Service 084de1
{
Packit Service 084de1
    int ret = -1;
Packit Service 084de1
    char buff[BUFSIZ];
Packit Service 084de1
    UI *ui;
Packit Service 084de1
Packit Service 084de1
    if ((prompt == NULL) && (prompt_string[0] != '\0'))
Packit Service 084de1
        prompt = prompt_string;
Packit Service 084de1
    ui = UI_new();
Packit Service 084de1
    if (ui == NULL)
Packit Service 084de1
        return ret;
Packit Service 084de1
    if (UI_add_input_string(ui, prompt, 0, buf, min,
Packit Service 084de1
                            (len >= BUFSIZ) ? BUFSIZ - 1 : len) < 0
Packit Service 084de1
        || (verify
Packit Service 084de1
            && UI_add_verify_string(ui, prompt, 0, buff, min,
Packit Service 084de1
                                    (len >= BUFSIZ) ? BUFSIZ - 1 : len,
Packit Service 084de1
                                    buf) < 0))
Packit Service 084de1
        goto end;
Packit Service 084de1
    ret = UI_process(ui);
Packit Service 084de1
    OPENSSL_cleanse(buff, BUFSIZ);
Packit Service 084de1
 end:
Packit Service 084de1
    UI_free(ui);
Packit Service 084de1
    return ret;
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
int EVP_BytesToKey(const EVP_CIPHER *type, const EVP_MD *md,
Packit Service 084de1
                   const unsigned char *salt, const unsigned char *data,
Packit Service 084de1
                   int datal, int count, unsigned char *key,
Packit Service 084de1
                   unsigned char *iv)
Packit Service 084de1
{
Packit Service 084de1
    EVP_MD_CTX *c;
Packit Service 084de1
    unsigned char md_buf[EVP_MAX_MD_SIZE];
Packit Service 084de1
    int niv, nkey, addmd = 0;
Packit Service 084de1
    unsigned int mds = 0, i;
Packit Service 084de1
    int rv = 0;
Packit Service 084de1
    nkey = EVP_CIPHER_key_length(type);
Packit Service 084de1
    niv = EVP_CIPHER_iv_length(type);
Packit Service 084de1
    OPENSSL_assert(nkey <= EVP_MAX_KEY_LENGTH);
Packit Service 084de1
    OPENSSL_assert(niv <= EVP_MAX_IV_LENGTH);
Packit Service 084de1
Packit Service 084de1
    if (data == NULL)
Packit Service 084de1
        return nkey;
Packit Service 084de1
Packit Service 084de1
    c = EVP_MD_CTX_new();
Packit Service 084de1
    if (c == NULL)
Packit Service 084de1
        goto err;
Packit Service 084de1
    for (;;) {
Packit Service 084de1
        if (!EVP_DigestInit_ex(c, md, NULL))
Packit Service 084de1
            goto err;
Packit Service 084de1
        if (addmd++)
Packit Service 084de1
            if (!EVP_DigestUpdate(c, &(md_buf[0]), mds))
Packit Service 084de1
                goto err;
Packit Service 084de1
        if (!EVP_DigestUpdate(c, data, datal))
Packit Service 084de1
            goto err;
Packit Service 084de1
        if (salt != NULL)
Packit Service 084de1
            if (!EVP_DigestUpdate(c, salt, PKCS5_SALT_LEN))
Packit Service 084de1
                goto err;
Packit Service 084de1
        if (!EVP_DigestFinal_ex(c, &(md_buf[0]), &mds))
Packit Service 084de1
            goto err;
Packit Service 084de1
Packit Service 084de1
        for (i = 1; i < (unsigned int)count; i++) {
Packit Service 084de1
            if (!EVP_DigestInit_ex(c, md, NULL))
Packit Service 084de1
                goto err;
Packit Service 084de1
            if (!EVP_DigestUpdate(c, &(md_buf[0]), mds))
Packit Service 084de1
                goto err;
Packit Service 084de1
            if (!EVP_DigestFinal_ex(c, &(md_buf[0]), &mds))
Packit Service 084de1
                goto err;
Packit Service 084de1
        }
Packit Service 084de1
        i = 0;
Packit Service 084de1
        if (nkey) {
Packit Service 084de1
            for (;;) {
Packit Service 084de1
                if (nkey == 0)
Packit Service 084de1
                    break;
Packit Service 084de1
                if (i == mds)
Packit Service 084de1
                    break;
Packit Service 084de1
                if (key != NULL)
Packit Service 084de1
                    *(key++) = md_buf[i];
Packit Service 084de1
                nkey--;
Packit Service 084de1
                i++;
Packit Service 084de1
            }
Packit Service 084de1
        }
Packit Service 084de1
        if (niv && (i != mds)) {
Packit Service 084de1
            for (;;) {
Packit Service 084de1
                if (niv == 0)
Packit Service 084de1
                    break;
Packit Service 084de1
                if (i == mds)
Packit Service 084de1
                    break;
Packit Service 084de1
                if (iv != NULL)
Packit Service 084de1
                    *(iv++) = md_buf[i];
Packit Service 084de1
                niv--;
Packit Service 084de1
                i++;
Packit Service 084de1
            }
Packit Service 084de1
        }
Packit Service 084de1
        if ((nkey == 0) && (niv == 0))
Packit Service 084de1
            break;
Packit Service 084de1
    }
Packit Service 084de1
    rv = EVP_CIPHER_key_length(type);
Packit Service 084de1
 err:
Packit Service 084de1
    EVP_MD_CTX_free(c);
Packit Service 084de1
    OPENSSL_cleanse(md_buf, sizeof(md_buf));
Packit Service 084de1
    return rv;
Packit Service 084de1
}