1e5b75505Sopenharmony_ci/* 2e5b75505Sopenharmony_ci * Wrapper functions for OpenSSL libcrypto 3e5b75505Sopenharmony_ci * Copyright (c) 2004-2017, Jouni Malinen <j@w1.fi> 4e5b75505Sopenharmony_ci * 5e5b75505Sopenharmony_ci * This software may be distributed under the terms of the BSD license. 6e5b75505Sopenharmony_ci * See README for more details. 7e5b75505Sopenharmony_ci */ 8e5b75505Sopenharmony_ci 9e5b75505Sopenharmony_ci#include "includes.h" 10e5b75505Sopenharmony_ci#include <openssl/opensslv.h> 11e5b75505Sopenharmony_ci#include <openssl/err.h> 12e5b75505Sopenharmony_ci#include <openssl/des.h> 13e5b75505Sopenharmony_ci#include <openssl/aes.h> 14e5b75505Sopenharmony_ci#include <openssl/bn.h> 15e5b75505Sopenharmony_ci#include <openssl/evp.h> 16e5b75505Sopenharmony_ci#include <openssl/dh.h> 17e5b75505Sopenharmony_ci#include <openssl/hmac.h> 18e5b75505Sopenharmony_ci#include <openssl/rand.h> 19e5b75505Sopenharmony_ci#ifdef CONFIG_OPENSSL_CMAC 20e5b75505Sopenharmony_ci#include <openssl/cmac.h> 21e5b75505Sopenharmony_ci#endif /* CONFIG_OPENSSL_CMAC */ 22e5b75505Sopenharmony_ci#ifdef CONFIG_ECC 23e5b75505Sopenharmony_ci#include <openssl/ec.h> 24e5b75505Sopenharmony_ci#endif /* CONFIG_ECC */ 25e5b75505Sopenharmony_ci 26e5b75505Sopenharmony_ci#include "common.h" 27e5b75505Sopenharmony_ci#include "utils/const_time.h" 28e5b75505Sopenharmony_ci#include "wpabuf.h" 29e5b75505Sopenharmony_ci#include "dh_group5.h" 30e5b75505Sopenharmony_ci#include "sha1.h" 31e5b75505Sopenharmony_ci#include "sha256.h" 32e5b75505Sopenharmony_ci#include "sha384.h" 33e5b75505Sopenharmony_ci#include "sha512.h" 34e5b75505Sopenharmony_ci#include "md5.h" 35e5b75505Sopenharmony_ci#include "aes_wrap.h" 36e5b75505Sopenharmony_ci#include "crypto.h" 37e5b75505Sopenharmony_ci 38e5b75505Sopenharmony_ci#if OPENSSL_VERSION_NUMBER < 0x10100000L || \ 39e5b75505Sopenharmony_ci (defined(LIBRESSL_VERSION_NUMBER) && \ 40e5b75505Sopenharmony_ci LIBRESSL_VERSION_NUMBER < 0x20700000L) 41e5b75505Sopenharmony_ci/* Compatibility wrappers for older versions. */ 42e5b75505Sopenharmony_ci 43e5b75505Sopenharmony_cistatic HMAC_CTX * HMAC_CTX_new(void) 44e5b75505Sopenharmony_ci{ 45e5b75505Sopenharmony_ci HMAC_CTX *ctx; 46e5b75505Sopenharmony_ci 47e5b75505Sopenharmony_ci ctx = os_zalloc(sizeof(*ctx)); 48e5b75505Sopenharmony_ci if (ctx) 49e5b75505Sopenharmony_ci HMAC_CTX_init(ctx); 50e5b75505Sopenharmony_ci return ctx; 51e5b75505Sopenharmony_ci} 52e5b75505Sopenharmony_ci 53e5b75505Sopenharmony_ci 54e5b75505Sopenharmony_cistatic void HMAC_CTX_free(HMAC_CTX *ctx) 55e5b75505Sopenharmony_ci{ 56e5b75505Sopenharmony_ci if (!ctx) 57e5b75505Sopenharmony_ci return; 58e5b75505Sopenharmony_ci HMAC_CTX_cleanup(ctx); 59e5b75505Sopenharmony_ci bin_clear_free(ctx, sizeof(*ctx)); 60e5b75505Sopenharmony_ci} 61e5b75505Sopenharmony_ci 62e5b75505Sopenharmony_ci 63e5b75505Sopenharmony_cistatic EVP_MD_CTX * EVP_MD_CTX_new(void) 64e5b75505Sopenharmony_ci{ 65e5b75505Sopenharmony_ci EVP_MD_CTX *ctx; 66e5b75505Sopenharmony_ci 67e5b75505Sopenharmony_ci ctx = os_zalloc(sizeof(*ctx)); 68e5b75505Sopenharmony_ci if (ctx) 69e5b75505Sopenharmony_ci EVP_MD_CTX_init(ctx); 70e5b75505Sopenharmony_ci return ctx; 71e5b75505Sopenharmony_ci} 72e5b75505Sopenharmony_ci 73e5b75505Sopenharmony_ci 74e5b75505Sopenharmony_cistatic void EVP_MD_CTX_free(EVP_MD_CTX *ctx) 75e5b75505Sopenharmony_ci{ 76e5b75505Sopenharmony_ci if (!ctx) 77e5b75505Sopenharmony_ci return; 78e5b75505Sopenharmony_ci EVP_MD_CTX_cleanup(ctx); 79e5b75505Sopenharmony_ci bin_clear_free(ctx, sizeof(*ctx)); 80e5b75505Sopenharmony_ci} 81e5b75505Sopenharmony_ci 82e5b75505Sopenharmony_ci#endif /* OpenSSL version < 1.1.0 */ 83e5b75505Sopenharmony_ci 84e5b75505Sopenharmony_cistatic BIGNUM * get_group5_prime(void) 85e5b75505Sopenharmony_ci{ 86e5b75505Sopenharmony_ci#if OPENSSL_VERSION_NUMBER >= 0x10100000L && \ 87e5b75505Sopenharmony_ci !(defined(LIBRESSL_VERSION_NUMBER) && \ 88e5b75505Sopenharmony_ci LIBRESSL_VERSION_NUMBER < 0x20700000L) 89e5b75505Sopenharmony_ci return BN_get_rfc3526_prime_1536(NULL); 90e5b75505Sopenharmony_ci#elif !defined(OPENSSL_IS_BORINGSSL) 91e5b75505Sopenharmony_ci return get_rfc3526_prime_1536(NULL); 92e5b75505Sopenharmony_ci#else 93e5b75505Sopenharmony_ci static const unsigned char RFC3526_PRIME_1536[] = { 94e5b75505Sopenharmony_ci 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xC9,0x0F,0xDA,0xA2, 95e5b75505Sopenharmony_ci 0x21,0x68,0xC2,0x34,0xC4,0xC6,0x62,0x8B,0x80,0xDC,0x1C,0xD1, 96e5b75505Sopenharmony_ci 0x29,0x02,0x4E,0x08,0x8A,0x67,0xCC,0x74,0x02,0x0B,0xBE,0xA6, 97e5b75505Sopenharmony_ci 0x3B,0x13,0x9B,0x22,0x51,0x4A,0x08,0x79,0x8E,0x34,0x04,0xDD, 98e5b75505Sopenharmony_ci 0xEF,0x95,0x19,0xB3,0xCD,0x3A,0x43,0x1B,0x30,0x2B,0x0A,0x6D, 99e5b75505Sopenharmony_ci 0xF2,0x5F,0x14,0x37,0x4F,0xE1,0x35,0x6D,0x6D,0x51,0xC2,0x45, 100e5b75505Sopenharmony_ci 0xE4,0x85,0xB5,0x76,0x62,0x5E,0x7E,0xC6,0xF4,0x4C,0x42,0xE9, 101e5b75505Sopenharmony_ci 0xA6,0x37,0xED,0x6B,0x0B,0xFF,0x5C,0xB6,0xF4,0x06,0xB7,0xED, 102e5b75505Sopenharmony_ci 0xEE,0x38,0x6B,0xFB,0x5A,0x89,0x9F,0xA5,0xAE,0x9F,0x24,0x11, 103e5b75505Sopenharmony_ci 0x7C,0x4B,0x1F,0xE6,0x49,0x28,0x66,0x51,0xEC,0xE4,0x5B,0x3D, 104e5b75505Sopenharmony_ci 0xC2,0x00,0x7C,0xB8,0xA1,0x63,0xBF,0x05,0x98,0xDA,0x48,0x36, 105e5b75505Sopenharmony_ci 0x1C,0x55,0xD3,0x9A,0x69,0x16,0x3F,0xA8,0xFD,0x24,0xCF,0x5F, 106e5b75505Sopenharmony_ci 0x83,0x65,0x5D,0x23,0xDC,0xA3,0xAD,0x96,0x1C,0x62,0xF3,0x56, 107e5b75505Sopenharmony_ci 0x20,0x85,0x52,0xBB,0x9E,0xD5,0x29,0x07,0x70,0x96,0x96,0x6D, 108e5b75505Sopenharmony_ci 0x67,0x0C,0x35,0x4E,0x4A,0xBC,0x98,0x04,0xF1,0x74,0x6C,0x08, 109e5b75505Sopenharmony_ci 0xCA,0x23,0x73,0x27,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, 110e5b75505Sopenharmony_ci }; 111e5b75505Sopenharmony_ci return BN_bin2bn(RFC3526_PRIME_1536, sizeof(RFC3526_PRIME_1536), NULL); 112e5b75505Sopenharmony_ci#endif 113e5b75505Sopenharmony_ci} 114e5b75505Sopenharmony_ci 115e5b75505Sopenharmony_ci 116e5b75505Sopenharmony_cistatic BIGNUM * get_group5_order(void) 117e5b75505Sopenharmony_ci{ 118e5b75505Sopenharmony_ci static const unsigned char RFC3526_ORDER_1536[] = { 119e5b75505Sopenharmony_ci 0x7F,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xE4,0x87,0xED,0x51, 120e5b75505Sopenharmony_ci 0x10,0xB4,0x61,0x1A,0x62,0x63,0x31,0x45,0xC0,0x6E,0x0E,0x68, 121e5b75505Sopenharmony_ci 0x94,0x81,0x27,0x04,0x45,0x33,0xE6,0x3A,0x01,0x05,0xDF,0x53, 122e5b75505Sopenharmony_ci 0x1D,0x89,0xCD,0x91,0x28,0xA5,0x04,0x3C,0xC7,0x1A,0x02,0x6E, 123e5b75505Sopenharmony_ci 0xF7,0xCA,0x8C,0xD9,0xE6,0x9D,0x21,0x8D,0x98,0x15,0x85,0x36, 124e5b75505Sopenharmony_ci 0xF9,0x2F,0x8A,0x1B,0xA7,0xF0,0x9A,0xB6,0xB6,0xA8,0xE1,0x22, 125e5b75505Sopenharmony_ci 0xF2,0x42,0xDA,0xBB,0x31,0x2F,0x3F,0x63,0x7A,0x26,0x21,0x74, 126e5b75505Sopenharmony_ci 0xD3,0x1B,0xF6,0xB5,0x85,0xFF,0xAE,0x5B,0x7A,0x03,0x5B,0xF6, 127e5b75505Sopenharmony_ci 0xF7,0x1C,0x35,0xFD,0xAD,0x44,0xCF,0xD2,0xD7,0x4F,0x92,0x08, 128e5b75505Sopenharmony_ci 0xBE,0x25,0x8F,0xF3,0x24,0x94,0x33,0x28,0xF6,0x72,0x2D,0x9E, 129e5b75505Sopenharmony_ci 0xE1,0x00,0x3E,0x5C,0x50,0xB1,0xDF,0x82,0xCC,0x6D,0x24,0x1B, 130e5b75505Sopenharmony_ci 0x0E,0x2A,0xE9,0xCD,0x34,0x8B,0x1F,0xD4,0x7E,0x92,0x67,0xAF, 131e5b75505Sopenharmony_ci 0xC1,0xB2,0xAE,0x91,0xEE,0x51,0xD6,0xCB,0x0E,0x31,0x79,0xAB, 132e5b75505Sopenharmony_ci 0x10,0x42,0xA9,0x5D,0xCF,0x6A,0x94,0x83,0xB8,0x4B,0x4B,0x36, 133e5b75505Sopenharmony_ci 0xB3,0x86,0x1A,0xA7,0x25,0x5E,0x4C,0x02,0x78,0xBA,0x36,0x04, 134e5b75505Sopenharmony_ci 0x65,0x11,0xB9,0x93,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF 135e5b75505Sopenharmony_ci }; 136e5b75505Sopenharmony_ci return BN_bin2bn(RFC3526_ORDER_1536, sizeof(RFC3526_ORDER_1536), NULL); 137e5b75505Sopenharmony_ci} 138e5b75505Sopenharmony_ci 139e5b75505Sopenharmony_ci 140e5b75505Sopenharmony_ci#ifdef OPENSSL_NO_SHA256 141e5b75505Sopenharmony_ci#define NO_SHA256_WRAPPER 142e5b75505Sopenharmony_ci#endif 143e5b75505Sopenharmony_ci#ifdef OPENSSL_NO_SHA512 144e5b75505Sopenharmony_ci#define NO_SHA384_WRAPPER 145e5b75505Sopenharmony_ci#endif 146e5b75505Sopenharmony_ci 147e5b75505Sopenharmony_cistatic int openssl_digest_vector(const EVP_MD *type, size_t num_elem, 148e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 149e5b75505Sopenharmony_ci{ 150e5b75505Sopenharmony_ci EVP_MD_CTX *ctx; 151e5b75505Sopenharmony_ci size_t i; 152e5b75505Sopenharmony_ci unsigned int mac_len; 153e5b75505Sopenharmony_ci 154e5b75505Sopenharmony_ci if (TEST_FAIL()) 155e5b75505Sopenharmony_ci return -1; 156e5b75505Sopenharmony_ci 157e5b75505Sopenharmony_ci ctx = EVP_MD_CTX_new(); 158e5b75505Sopenharmony_ci if (!ctx) 159e5b75505Sopenharmony_ci return -1; 160e5b75505Sopenharmony_ci if (!EVP_DigestInit_ex(ctx, type, NULL)) { 161e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_DigestInit_ex failed: %s", 162e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 163e5b75505Sopenharmony_ci EVP_MD_CTX_free(ctx); 164e5b75505Sopenharmony_ci return -1; 165e5b75505Sopenharmony_ci } 166e5b75505Sopenharmony_ci for (i = 0; i < num_elem; i++) { 167e5b75505Sopenharmony_ci if (!EVP_DigestUpdate(ctx, addr[i], len[i])) { 168e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_DigestUpdate " 169e5b75505Sopenharmony_ci "failed: %s", 170e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 171e5b75505Sopenharmony_ci EVP_MD_CTX_free(ctx); 172e5b75505Sopenharmony_ci return -1; 173e5b75505Sopenharmony_ci } 174e5b75505Sopenharmony_ci } 175e5b75505Sopenharmony_ci if (!EVP_DigestFinal(ctx, mac, &mac_len)) { 176e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_DigestFinal failed: %s", 177e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 178e5b75505Sopenharmony_ci EVP_MD_CTX_free(ctx); 179e5b75505Sopenharmony_ci return -1; 180e5b75505Sopenharmony_ci } 181e5b75505Sopenharmony_ci EVP_MD_CTX_free(ctx); 182e5b75505Sopenharmony_ci 183e5b75505Sopenharmony_ci return 0; 184e5b75505Sopenharmony_ci} 185e5b75505Sopenharmony_ci 186e5b75505Sopenharmony_ci 187e5b75505Sopenharmony_ci#ifndef CONFIG_FIPS 188e5b75505Sopenharmony_ciint md4_vector(size_t num_elem, const u8 *addr[], const size_t *len, u8 *mac) 189e5b75505Sopenharmony_ci{ 190e5b75505Sopenharmony_ci return openssl_digest_vector(EVP_md4(), num_elem, addr, len, mac); 191e5b75505Sopenharmony_ci} 192e5b75505Sopenharmony_ci#endif /* CONFIG_FIPS */ 193e5b75505Sopenharmony_ci 194e5b75505Sopenharmony_ci 195e5b75505Sopenharmony_ciint des_encrypt(const u8 *clear, const u8 *key, u8 *cypher) 196e5b75505Sopenharmony_ci{ 197e5b75505Sopenharmony_ci u8 pkey[8], next, tmp; 198e5b75505Sopenharmony_ci int i; 199e5b75505Sopenharmony_ci DES_key_schedule ks; 200e5b75505Sopenharmony_ci 201e5b75505Sopenharmony_ci /* Add parity bits to the key */ 202e5b75505Sopenharmony_ci next = 0; 203e5b75505Sopenharmony_ci for (i = 0; i < 7; i++) { 204e5b75505Sopenharmony_ci tmp = key[i]; 205e5b75505Sopenharmony_ci pkey[i] = (tmp >> i) | next | 1; 206e5b75505Sopenharmony_ci next = tmp << (7 - i); 207e5b75505Sopenharmony_ci } 208e5b75505Sopenharmony_ci pkey[i] = next | 1; 209e5b75505Sopenharmony_ci 210e5b75505Sopenharmony_ci DES_set_key((DES_cblock *) &pkey, &ks); 211e5b75505Sopenharmony_ci DES_ecb_encrypt((DES_cblock *) clear, (DES_cblock *) cypher, &ks, 212e5b75505Sopenharmony_ci DES_ENCRYPT); 213e5b75505Sopenharmony_ci return 0; 214e5b75505Sopenharmony_ci} 215e5b75505Sopenharmony_ci 216e5b75505Sopenharmony_ci 217e5b75505Sopenharmony_ci#ifndef CONFIG_NO_RC4 218e5b75505Sopenharmony_ciint rc4_skip(const u8 *key, size_t keylen, size_t skip, 219e5b75505Sopenharmony_ci u8 *data, size_t data_len) 220e5b75505Sopenharmony_ci{ 221e5b75505Sopenharmony_ci#ifdef OPENSSL_NO_RC4 222e5b75505Sopenharmony_ci return -1; 223e5b75505Sopenharmony_ci#else /* OPENSSL_NO_RC4 */ 224e5b75505Sopenharmony_ci EVP_CIPHER_CTX *ctx; 225e5b75505Sopenharmony_ci int outl; 226e5b75505Sopenharmony_ci int res = -1; 227e5b75505Sopenharmony_ci unsigned char skip_buf[16]; 228e5b75505Sopenharmony_ci 229e5b75505Sopenharmony_ci ctx = EVP_CIPHER_CTX_new(); 230e5b75505Sopenharmony_ci if (!ctx || 231e5b75505Sopenharmony_ci !EVP_CIPHER_CTX_set_padding(ctx, 0) || 232e5b75505Sopenharmony_ci !EVP_CipherInit_ex(ctx, EVP_rc4(), NULL, NULL, NULL, 1) || 233e5b75505Sopenharmony_ci !EVP_CIPHER_CTX_set_key_length(ctx, keylen) || 234e5b75505Sopenharmony_ci !EVP_CipherInit_ex(ctx, NULL, NULL, key, NULL, 1)) 235e5b75505Sopenharmony_ci goto out; 236e5b75505Sopenharmony_ci 237e5b75505Sopenharmony_ci while (skip >= sizeof(skip_buf)) { 238e5b75505Sopenharmony_ci size_t len = skip; 239e5b75505Sopenharmony_ci if (len > sizeof(skip_buf)) 240e5b75505Sopenharmony_ci len = sizeof(skip_buf); 241e5b75505Sopenharmony_ci if (!EVP_CipherUpdate(ctx, skip_buf, &outl, skip_buf, len)) 242e5b75505Sopenharmony_ci goto out; 243e5b75505Sopenharmony_ci skip -= len; 244e5b75505Sopenharmony_ci } 245e5b75505Sopenharmony_ci 246e5b75505Sopenharmony_ci if (EVP_CipherUpdate(ctx, data, &outl, data, data_len)) 247e5b75505Sopenharmony_ci res = 0; 248e5b75505Sopenharmony_ci 249e5b75505Sopenharmony_ciout: 250e5b75505Sopenharmony_ci if (ctx) 251e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx); 252e5b75505Sopenharmony_ci return res; 253e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_RC4 */ 254e5b75505Sopenharmony_ci} 255e5b75505Sopenharmony_ci#endif /* CONFIG_NO_RC4 */ 256e5b75505Sopenharmony_ci 257e5b75505Sopenharmony_ci 258e5b75505Sopenharmony_ci#ifndef CONFIG_FIPS 259e5b75505Sopenharmony_ciint md5_vector(size_t num_elem, const u8 *addr[], const size_t *len, u8 *mac) 260e5b75505Sopenharmony_ci{ 261e5b75505Sopenharmony_ci return openssl_digest_vector(EVP_md5(), num_elem, addr, len, mac); 262e5b75505Sopenharmony_ci} 263e5b75505Sopenharmony_ci#endif /* CONFIG_FIPS */ 264e5b75505Sopenharmony_ci 265e5b75505Sopenharmony_ci 266e5b75505Sopenharmony_ciint sha1_vector(size_t num_elem, const u8 *addr[], const size_t *len, u8 *mac) 267e5b75505Sopenharmony_ci{ 268e5b75505Sopenharmony_ci return openssl_digest_vector(EVP_sha1(), num_elem, addr, len, mac); 269e5b75505Sopenharmony_ci} 270e5b75505Sopenharmony_ci 271e5b75505Sopenharmony_ci 272e5b75505Sopenharmony_ci#ifndef NO_SHA256_WRAPPER 273e5b75505Sopenharmony_ciint sha256_vector(size_t num_elem, const u8 *addr[], const size_t *len, 274e5b75505Sopenharmony_ci u8 *mac) 275e5b75505Sopenharmony_ci{ 276e5b75505Sopenharmony_ci return openssl_digest_vector(EVP_sha256(), num_elem, addr, len, mac); 277e5b75505Sopenharmony_ci} 278e5b75505Sopenharmony_ci#endif /* NO_SHA256_WRAPPER */ 279e5b75505Sopenharmony_ci 280e5b75505Sopenharmony_ci 281e5b75505Sopenharmony_ci#ifndef NO_SHA384_WRAPPER 282e5b75505Sopenharmony_ciint sha384_vector(size_t num_elem, const u8 *addr[], const size_t *len, 283e5b75505Sopenharmony_ci u8 *mac) 284e5b75505Sopenharmony_ci{ 285e5b75505Sopenharmony_ci return openssl_digest_vector(EVP_sha384(), num_elem, addr, len, mac); 286e5b75505Sopenharmony_ci} 287e5b75505Sopenharmony_ci#endif /* NO_SHA384_WRAPPER */ 288e5b75505Sopenharmony_ci 289e5b75505Sopenharmony_ci 290e5b75505Sopenharmony_ci#ifndef NO_SHA512_WRAPPER 291e5b75505Sopenharmony_ciint sha512_vector(size_t num_elem, const u8 *addr[], const size_t *len, 292e5b75505Sopenharmony_ci u8 *mac) 293e5b75505Sopenharmony_ci{ 294e5b75505Sopenharmony_ci return openssl_digest_vector(EVP_sha512(), num_elem, addr, len, mac); 295e5b75505Sopenharmony_ci} 296e5b75505Sopenharmony_ci#endif /* NO_SHA512_WRAPPER */ 297e5b75505Sopenharmony_ci 298e5b75505Sopenharmony_ci 299e5b75505Sopenharmony_cistatic const EVP_CIPHER * aes_get_evp_cipher(size_t keylen) 300e5b75505Sopenharmony_ci{ 301e5b75505Sopenharmony_ci switch (keylen) { 302e5b75505Sopenharmony_ci case 16: 303e5b75505Sopenharmony_ci return EVP_aes_128_ecb(); 304e5b75505Sopenharmony_ci case 24: 305e5b75505Sopenharmony_ci return EVP_aes_192_ecb(); 306e5b75505Sopenharmony_ci case 32: 307e5b75505Sopenharmony_ci return EVP_aes_256_ecb(); 308e5b75505Sopenharmony_ci } 309e5b75505Sopenharmony_ci 310e5b75505Sopenharmony_ci return NULL; 311e5b75505Sopenharmony_ci} 312e5b75505Sopenharmony_ci 313e5b75505Sopenharmony_ci 314e5b75505Sopenharmony_civoid * aes_encrypt_init(const u8 *key, size_t len) 315e5b75505Sopenharmony_ci{ 316e5b75505Sopenharmony_ci EVP_CIPHER_CTX *ctx; 317e5b75505Sopenharmony_ci const EVP_CIPHER *type; 318e5b75505Sopenharmony_ci 319e5b75505Sopenharmony_ci if (TEST_FAIL()) 320e5b75505Sopenharmony_ci return NULL; 321e5b75505Sopenharmony_ci 322e5b75505Sopenharmony_ci type = aes_get_evp_cipher(len); 323e5b75505Sopenharmony_ci if (!type) { 324e5b75505Sopenharmony_ci wpa_printf(MSG_INFO, "%s: Unsupported len=%u", 325e5b75505Sopenharmony_ci __func__, (unsigned int) len); 326e5b75505Sopenharmony_ci return NULL; 327e5b75505Sopenharmony_ci } 328e5b75505Sopenharmony_ci 329e5b75505Sopenharmony_ci ctx = EVP_CIPHER_CTX_new(); 330e5b75505Sopenharmony_ci if (ctx == NULL) 331e5b75505Sopenharmony_ci return NULL; 332e5b75505Sopenharmony_ci if (EVP_EncryptInit_ex(ctx, type, NULL, key, NULL) != 1) { 333e5b75505Sopenharmony_ci os_free(ctx); 334e5b75505Sopenharmony_ci return NULL; 335e5b75505Sopenharmony_ci } 336e5b75505Sopenharmony_ci EVP_CIPHER_CTX_set_padding(ctx, 0); 337e5b75505Sopenharmony_ci return ctx; 338e5b75505Sopenharmony_ci} 339e5b75505Sopenharmony_ci 340e5b75505Sopenharmony_ci 341e5b75505Sopenharmony_ciint aes_encrypt(void *ctx, const u8 *plain, u8 *crypt) 342e5b75505Sopenharmony_ci{ 343e5b75505Sopenharmony_ci EVP_CIPHER_CTX *c = ctx; 344e5b75505Sopenharmony_ci int clen = 16; 345e5b75505Sopenharmony_ci if (EVP_EncryptUpdate(c, crypt, &clen, plain, 16) != 1) { 346e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_EncryptUpdate failed: %s", 347e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 348e5b75505Sopenharmony_ci return -1; 349e5b75505Sopenharmony_ci } 350e5b75505Sopenharmony_ci return 0; 351e5b75505Sopenharmony_ci} 352e5b75505Sopenharmony_ci 353e5b75505Sopenharmony_ci 354e5b75505Sopenharmony_civoid aes_encrypt_deinit(void *ctx) 355e5b75505Sopenharmony_ci{ 356e5b75505Sopenharmony_ci EVP_CIPHER_CTX *c = ctx; 357e5b75505Sopenharmony_ci u8 buf[16]; 358e5b75505Sopenharmony_ci int len = sizeof(buf); 359e5b75505Sopenharmony_ci if (EVP_EncryptFinal_ex(c, buf, &len) != 1) { 360e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_EncryptFinal_ex failed: " 361e5b75505Sopenharmony_ci "%s", ERR_error_string(ERR_get_error(), NULL)); 362e5b75505Sopenharmony_ci } 363e5b75505Sopenharmony_ci if (len != 0) { 364e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: Unexpected padding length %d " 365e5b75505Sopenharmony_ci "in AES encrypt", len); 366e5b75505Sopenharmony_ci } 367e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(c); 368e5b75505Sopenharmony_ci} 369e5b75505Sopenharmony_ci 370e5b75505Sopenharmony_ci 371e5b75505Sopenharmony_civoid * aes_decrypt_init(const u8 *key, size_t len) 372e5b75505Sopenharmony_ci{ 373e5b75505Sopenharmony_ci EVP_CIPHER_CTX *ctx; 374e5b75505Sopenharmony_ci const EVP_CIPHER *type; 375e5b75505Sopenharmony_ci 376e5b75505Sopenharmony_ci if (TEST_FAIL()) 377e5b75505Sopenharmony_ci return NULL; 378e5b75505Sopenharmony_ci 379e5b75505Sopenharmony_ci type = aes_get_evp_cipher(len); 380e5b75505Sopenharmony_ci if (!type) { 381e5b75505Sopenharmony_ci wpa_printf(MSG_INFO, "%s: Unsupported len=%u", 382e5b75505Sopenharmony_ci __func__, (unsigned int) len); 383e5b75505Sopenharmony_ci return NULL; 384e5b75505Sopenharmony_ci } 385e5b75505Sopenharmony_ci 386e5b75505Sopenharmony_ci ctx = EVP_CIPHER_CTX_new(); 387e5b75505Sopenharmony_ci if (ctx == NULL) 388e5b75505Sopenharmony_ci return NULL; 389e5b75505Sopenharmony_ci if (EVP_DecryptInit_ex(ctx, type, NULL, key, NULL) != 1) { 390e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx); 391e5b75505Sopenharmony_ci return NULL; 392e5b75505Sopenharmony_ci } 393e5b75505Sopenharmony_ci EVP_CIPHER_CTX_set_padding(ctx, 0); 394e5b75505Sopenharmony_ci return ctx; 395e5b75505Sopenharmony_ci} 396e5b75505Sopenharmony_ci 397e5b75505Sopenharmony_ci 398e5b75505Sopenharmony_ciint aes_decrypt(void *ctx, const u8 *crypt, u8 *plain) 399e5b75505Sopenharmony_ci{ 400e5b75505Sopenharmony_ci EVP_CIPHER_CTX *c = ctx; 401e5b75505Sopenharmony_ci int plen = 16; 402e5b75505Sopenharmony_ci if (EVP_DecryptUpdate(c, plain, &plen, crypt, 16) != 1) { 403e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_DecryptUpdate failed: %s", 404e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 405e5b75505Sopenharmony_ci return -1; 406e5b75505Sopenharmony_ci } 407e5b75505Sopenharmony_ci return 0; 408e5b75505Sopenharmony_ci} 409e5b75505Sopenharmony_ci 410e5b75505Sopenharmony_ci 411e5b75505Sopenharmony_civoid aes_decrypt_deinit(void *ctx) 412e5b75505Sopenharmony_ci{ 413e5b75505Sopenharmony_ci EVP_CIPHER_CTX *c = ctx; 414e5b75505Sopenharmony_ci u8 buf[16]; 415e5b75505Sopenharmony_ci int len = sizeof(buf); 416e5b75505Sopenharmony_ci if (EVP_DecryptFinal_ex(c, buf, &len) != 1) { 417e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_DecryptFinal_ex failed: " 418e5b75505Sopenharmony_ci "%s", ERR_error_string(ERR_get_error(), NULL)); 419e5b75505Sopenharmony_ci } 420e5b75505Sopenharmony_ci if (len != 0) { 421e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: Unexpected padding length %d " 422e5b75505Sopenharmony_ci "in AES decrypt", len); 423e5b75505Sopenharmony_ci } 424e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(c); 425e5b75505Sopenharmony_ci} 426e5b75505Sopenharmony_ci 427e5b75505Sopenharmony_ci 428e5b75505Sopenharmony_ci#ifndef CONFIG_FIPS 429e5b75505Sopenharmony_ci#ifndef CONFIG_OPENSSL_INTERNAL_AES_WRAP 430e5b75505Sopenharmony_ci 431e5b75505Sopenharmony_ciint aes_wrap(const u8 *kek, size_t kek_len, int n, const u8 *plain, u8 *cipher) 432e5b75505Sopenharmony_ci{ 433e5b75505Sopenharmony_ci AES_KEY actx; 434e5b75505Sopenharmony_ci int res; 435e5b75505Sopenharmony_ci 436e5b75505Sopenharmony_ci if (TEST_FAIL()) 437e5b75505Sopenharmony_ci return -1; 438e5b75505Sopenharmony_ci if (AES_set_encrypt_key(kek, kek_len << 3, &actx)) 439e5b75505Sopenharmony_ci return -1; 440e5b75505Sopenharmony_ci res = AES_wrap_key(&actx, NULL, cipher, plain, n * 8); 441e5b75505Sopenharmony_ci OPENSSL_cleanse(&actx, sizeof(actx)); 442e5b75505Sopenharmony_ci return res <= 0 ? -1 : 0; 443e5b75505Sopenharmony_ci} 444e5b75505Sopenharmony_ci 445e5b75505Sopenharmony_ci 446e5b75505Sopenharmony_ciint aes_unwrap(const u8 *kek, size_t kek_len, int n, const u8 *cipher, 447e5b75505Sopenharmony_ci u8 *plain) 448e5b75505Sopenharmony_ci{ 449e5b75505Sopenharmony_ci AES_KEY actx; 450e5b75505Sopenharmony_ci int res; 451e5b75505Sopenharmony_ci 452e5b75505Sopenharmony_ci if (TEST_FAIL()) 453e5b75505Sopenharmony_ci return -1; 454e5b75505Sopenharmony_ci if (AES_set_decrypt_key(kek, kek_len << 3, &actx)) 455e5b75505Sopenharmony_ci return -1; 456e5b75505Sopenharmony_ci res = AES_unwrap_key(&actx, NULL, plain, cipher, (n + 1) * 8); 457e5b75505Sopenharmony_ci OPENSSL_cleanse(&actx, sizeof(actx)); 458e5b75505Sopenharmony_ci return res <= 0 ? -1 : 0; 459e5b75505Sopenharmony_ci} 460e5b75505Sopenharmony_ci 461e5b75505Sopenharmony_ci#endif /* CONFIG_OPENSSL_INTERNAL_AES_WRAP */ 462e5b75505Sopenharmony_ci#endif /* CONFIG_FIPS */ 463e5b75505Sopenharmony_ci 464e5b75505Sopenharmony_ci 465e5b75505Sopenharmony_ciint aes_128_cbc_encrypt(const u8 *key, const u8 *iv, u8 *data, size_t data_len) 466e5b75505Sopenharmony_ci{ 467e5b75505Sopenharmony_ci EVP_CIPHER_CTX *ctx; 468e5b75505Sopenharmony_ci int clen, len; 469e5b75505Sopenharmony_ci u8 buf[16]; 470e5b75505Sopenharmony_ci int res = -1; 471e5b75505Sopenharmony_ci 472e5b75505Sopenharmony_ci if (TEST_FAIL()) 473e5b75505Sopenharmony_ci return -1; 474e5b75505Sopenharmony_ci 475e5b75505Sopenharmony_ci ctx = EVP_CIPHER_CTX_new(); 476e5b75505Sopenharmony_ci if (!ctx) 477e5b75505Sopenharmony_ci return -1; 478e5b75505Sopenharmony_ci clen = data_len; 479e5b75505Sopenharmony_ci len = sizeof(buf); 480e5b75505Sopenharmony_ci if (EVP_EncryptInit_ex(ctx, EVP_aes_128_cbc(), NULL, key, iv) == 1 && 481e5b75505Sopenharmony_ci EVP_CIPHER_CTX_set_padding(ctx, 0) == 1 && 482e5b75505Sopenharmony_ci EVP_EncryptUpdate(ctx, data, &clen, data, data_len) == 1 && 483e5b75505Sopenharmony_ci clen == (int) data_len && 484e5b75505Sopenharmony_ci EVP_EncryptFinal_ex(ctx, buf, &len) == 1 && len == 0) 485e5b75505Sopenharmony_ci res = 0; 486e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx); 487e5b75505Sopenharmony_ci 488e5b75505Sopenharmony_ci return res; 489e5b75505Sopenharmony_ci} 490e5b75505Sopenharmony_ci 491e5b75505Sopenharmony_ci 492e5b75505Sopenharmony_ciint aes_128_cbc_decrypt(const u8 *key, const u8 *iv, u8 *data, size_t data_len) 493e5b75505Sopenharmony_ci{ 494e5b75505Sopenharmony_ci EVP_CIPHER_CTX *ctx; 495e5b75505Sopenharmony_ci int plen, len; 496e5b75505Sopenharmony_ci u8 buf[16]; 497e5b75505Sopenharmony_ci int res = -1; 498e5b75505Sopenharmony_ci 499e5b75505Sopenharmony_ci if (TEST_FAIL()) 500e5b75505Sopenharmony_ci return -1; 501e5b75505Sopenharmony_ci 502e5b75505Sopenharmony_ci ctx = EVP_CIPHER_CTX_new(); 503e5b75505Sopenharmony_ci if (!ctx) 504e5b75505Sopenharmony_ci return -1; 505e5b75505Sopenharmony_ci plen = data_len; 506e5b75505Sopenharmony_ci len = sizeof(buf); 507e5b75505Sopenharmony_ci if (EVP_DecryptInit_ex(ctx, EVP_aes_128_cbc(), NULL, key, iv) == 1 && 508e5b75505Sopenharmony_ci EVP_CIPHER_CTX_set_padding(ctx, 0) == 1 && 509e5b75505Sopenharmony_ci EVP_DecryptUpdate(ctx, data, &plen, data, data_len) == 1 && 510e5b75505Sopenharmony_ci plen == (int) data_len && 511e5b75505Sopenharmony_ci EVP_DecryptFinal_ex(ctx, buf, &len) == 1 && len == 0) 512e5b75505Sopenharmony_ci res = 0; 513e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx); 514e5b75505Sopenharmony_ci 515e5b75505Sopenharmony_ci return res; 516e5b75505Sopenharmony_ci 517e5b75505Sopenharmony_ci} 518e5b75505Sopenharmony_ci 519e5b75505Sopenharmony_ci 520e5b75505Sopenharmony_ciint crypto_dh_init(u8 generator, const u8 *prime, size_t prime_len, u8 *privkey, 521e5b75505Sopenharmony_ci u8 *pubkey) 522e5b75505Sopenharmony_ci{ 523e5b75505Sopenharmony_ci size_t pubkey_len, pad; 524e5b75505Sopenharmony_ci 525e5b75505Sopenharmony_ci if (os_get_random(privkey, prime_len) < 0) 526e5b75505Sopenharmony_ci return -1; 527e5b75505Sopenharmony_ci if (os_memcmp(privkey, prime, prime_len) > 0) { 528e5b75505Sopenharmony_ci /* Make sure private value is smaller than prime */ 529e5b75505Sopenharmony_ci privkey[0] = 0; 530e5b75505Sopenharmony_ci } 531e5b75505Sopenharmony_ci 532e5b75505Sopenharmony_ci pubkey_len = prime_len; 533e5b75505Sopenharmony_ci if (crypto_mod_exp(&generator, 1, privkey, prime_len, prime, prime_len, 534e5b75505Sopenharmony_ci pubkey, &pubkey_len) < 0) 535e5b75505Sopenharmony_ci return -1; 536e5b75505Sopenharmony_ci if (pubkey_len < prime_len) { 537e5b75505Sopenharmony_ci pad = prime_len - pubkey_len; 538e5b75505Sopenharmony_ci os_memmove(pubkey + pad, pubkey, pubkey_len); 539e5b75505Sopenharmony_ci os_memset(pubkey, 0, pad); 540e5b75505Sopenharmony_ci } 541e5b75505Sopenharmony_ci 542e5b75505Sopenharmony_ci return 0; 543e5b75505Sopenharmony_ci} 544e5b75505Sopenharmony_ci 545e5b75505Sopenharmony_ci 546e5b75505Sopenharmony_ciint crypto_dh_derive_secret(u8 generator, const u8 *prime, size_t prime_len, 547e5b75505Sopenharmony_ci const u8 *order, size_t order_len, 548e5b75505Sopenharmony_ci const u8 *privkey, size_t privkey_len, 549e5b75505Sopenharmony_ci const u8 *pubkey, size_t pubkey_len, 550e5b75505Sopenharmony_ci u8 *secret, size_t *len) 551e5b75505Sopenharmony_ci{ 552e5b75505Sopenharmony_ci BIGNUM *pub, *p; 553e5b75505Sopenharmony_ci int res = -1; 554e5b75505Sopenharmony_ci 555e5b75505Sopenharmony_ci pub = BN_bin2bn(pubkey, pubkey_len, NULL); 556e5b75505Sopenharmony_ci p = BN_bin2bn(prime, prime_len, NULL); 557e5b75505Sopenharmony_ci if (!pub || !p || BN_is_zero(pub) || BN_is_one(pub) || 558e5b75505Sopenharmony_ci BN_cmp(pub, p) >= 0) 559e5b75505Sopenharmony_ci goto fail; 560e5b75505Sopenharmony_ci 561e5b75505Sopenharmony_ci if (order) { 562e5b75505Sopenharmony_ci BN_CTX *ctx; 563e5b75505Sopenharmony_ci BIGNUM *q, *tmp; 564e5b75505Sopenharmony_ci int failed; 565e5b75505Sopenharmony_ci 566e5b75505Sopenharmony_ci /* verify: pubkey^q == 1 mod p */ 567e5b75505Sopenharmony_ci q = BN_bin2bn(order, order_len, NULL); 568e5b75505Sopenharmony_ci ctx = BN_CTX_new(); 569e5b75505Sopenharmony_ci tmp = BN_new(); 570e5b75505Sopenharmony_ci failed = !q || !ctx || !tmp || 571e5b75505Sopenharmony_ci !BN_mod_exp(tmp, pub, q, p, ctx) || 572e5b75505Sopenharmony_ci !BN_is_one(tmp); 573e5b75505Sopenharmony_ci BN_clear_free(q); 574e5b75505Sopenharmony_ci BN_clear_free(tmp); 575e5b75505Sopenharmony_ci BN_CTX_free(ctx); 576e5b75505Sopenharmony_ci if (failed) 577e5b75505Sopenharmony_ci goto fail; 578e5b75505Sopenharmony_ci } 579e5b75505Sopenharmony_ci 580e5b75505Sopenharmony_ci res = crypto_mod_exp(pubkey, pubkey_len, privkey, privkey_len, 581e5b75505Sopenharmony_ci prime, prime_len, secret, len); 582e5b75505Sopenharmony_cifail: 583e5b75505Sopenharmony_ci BN_clear_free(pub); 584e5b75505Sopenharmony_ci BN_clear_free(p); 585e5b75505Sopenharmony_ci return res; 586e5b75505Sopenharmony_ci} 587e5b75505Sopenharmony_ci 588e5b75505Sopenharmony_ci 589e5b75505Sopenharmony_ciint crypto_mod_exp(const u8 *base, size_t base_len, 590e5b75505Sopenharmony_ci const u8 *power, size_t power_len, 591e5b75505Sopenharmony_ci const u8 *modulus, size_t modulus_len, 592e5b75505Sopenharmony_ci u8 *result, size_t *result_len) 593e5b75505Sopenharmony_ci{ 594e5b75505Sopenharmony_ci BIGNUM *bn_base, *bn_exp, *bn_modulus, *bn_result; 595e5b75505Sopenharmony_ci int ret = -1; 596e5b75505Sopenharmony_ci BN_CTX *ctx; 597e5b75505Sopenharmony_ci 598e5b75505Sopenharmony_ci ctx = BN_CTX_new(); 599e5b75505Sopenharmony_ci if (ctx == NULL) 600e5b75505Sopenharmony_ci return -1; 601e5b75505Sopenharmony_ci 602e5b75505Sopenharmony_ci bn_base = BN_bin2bn(base, base_len, NULL); 603e5b75505Sopenharmony_ci bn_exp = BN_bin2bn(power, power_len, NULL); 604e5b75505Sopenharmony_ci bn_modulus = BN_bin2bn(modulus, modulus_len, NULL); 605e5b75505Sopenharmony_ci bn_result = BN_new(); 606e5b75505Sopenharmony_ci 607e5b75505Sopenharmony_ci if (bn_base == NULL || bn_exp == NULL || bn_modulus == NULL || 608e5b75505Sopenharmony_ci bn_result == NULL) 609e5b75505Sopenharmony_ci goto error; 610e5b75505Sopenharmony_ci 611e5b75505Sopenharmony_ci if (BN_mod_exp_mont_consttime(bn_result, bn_base, bn_exp, bn_modulus, 612e5b75505Sopenharmony_ci ctx, NULL) != 1) 613e5b75505Sopenharmony_ci goto error; 614e5b75505Sopenharmony_ci 615e5b75505Sopenharmony_ci *result_len = BN_bn2bin(bn_result, result); 616e5b75505Sopenharmony_ci ret = 0; 617e5b75505Sopenharmony_ci 618e5b75505Sopenharmony_cierror: 619e5b75505Sopenharmony_ci BN_clear_free(bn_base); 620e5b75505Sopenharmony_ci BN_clear_free(bn_exp); 621e5b75505Sopenharmony_ci BN_clear_free(bn_modulus); 622e5b75505Sopenharmony_ci BN_clear_free(bn_result); 623e5b75505Sopenharmony_ci BN_CTX_free(ctx); 624e5b75505Sopenharmony_ci return ret; 625e5b75505Sopenharmony_ci} 626e5b75505Sopenharmony_ci 627e5b75505Sopenharmony_ci 628e5b75505Sopenharmony_cistruct crypto_cipher { 629e5b75505Sopenharmony_ci EVP_CIPHER_CTX *enc; 630e5b75505Sopenharmony_ci EVP_CIPHER_CTX *dec; 631e5b75505Sopenharmony_ci}; 632e5b75505Sopenharmony_ci 633e5b75505Sopenharmony_ci 634e5b75505Sopenharmony_cistruct crypto_cipher * crypto_cipher_init(enum crypto_cipher_alg alg, 635e5b75505Sopenharmony_ci const u8 *iv, const u8 *key, 636e5b75505Sopenharmony_ci size_t key_len) 637e5b75505Sopenharmony_ci{ 638e5b75505Sopenharmony_ci struct crypto_cipher *ctx; 639e5b75505Sopenharmony_ci const EVP_CIPHER *cipher; 640e5b75505Sopenharmony_ci 641e5b75505Sopenharmony_ci ctx = os_zalloc(sizeof(*ctx)); 642e5b75505Sopenharmony_ci if (ctx == NULL) 643e5b75505Sopenharmony_ci return NULL; 644e5b75505Sopenharmony_ci 645e5b75505Sopenharmony_ci switch (alg) { 646e5b75505Sopenharmony_ci#ifndef CONFIG_NO_RC4 647e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_RC4 648e5b75505Sopenharmony_ci case CRYPTO_CIPHER_ALG_RC4: 649e5b75505Sopenharmony_ci cipher = EVP_rc4(); 650e5b75505Sopenharmony_ci break; 651e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_RC4 */ 652e5b75505Sopenharmony_ci#endif /* CONFIG_NO_RC4 */ 653e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_AES 654e5b75505Sopenharmony_ci case CRYPTO_CIPHER_ALG_AES: 655e5b75505Sopenharmony_ci switch (key_len) { 656e5b75505Sopenharmony_ci case 16: 657e5b75505Sopenharmony_ci cipher = EVP_aes_128_cbc(); 658e5b75505Sopenharmony_ci break; 659e5b75505Sopenharmony_ci#ifndef OPENSSL_IS_BORINGSSL 660e5b75505Sopenharmony_ci case 24: 661e5b75505Sopenharmony_ci cipher = EVP_aes_192_cbc(); 662e5b75505Sopenharmony_ci break; 663e5b75505Sopenharmony_ci#endif /* OPENSSL_IS_BORINGSSL */ 664e5b75505Sopenharmony_ci case 32: 665e5b75505Sopenharmony_ci cipher = EVP_aes_256_cbc(); 666e5b75505Sopenharmony_ci break; 667e5b75505Sopenharmony_ci default: 668e5b75505Sopenharmony_ci os_free(ctx); 669e5b75505Sopenharmony_ci return NULL; 670e5b75505Sopenharmony_ci } 671e5b75505Sopenharmony_ci break; 672e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_AES */ 673e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_DES 674e5b75505Sopenharmony_ci case CRYPTO_CIPHER_ALG_3DES: 675e5b75505Sopenharmony_ci cipher = EVP_des_ede3_cbc(); 676e5b75505Sopenharmony_ci break; 677e5b75505Sopenharmony_ci case CRYPTO_CIPHER_ALG_DES: 678e5b75505Sopenharmony_ci cipher = EVP_des_cbc(); 679e5b75505Sopenharmony_ci break; 680e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_DES */ 681e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_RC2 682e5b75505Sopenharmony_ci case CRYPTO_CIPHER_ALG_RC2: 683e5b75505Sopenharmony_ci cipher = EVP_rc2_ecb(); 684e5b75505Sopenharmony_ci break; 685e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_RC2 */ 686e5b75505Sopenharmony_ci default: 687e5b75505Sopenharmony_ci os_free(ctx); 688e5b75505Sopenharmony_ci return NULL; 689e5b75505Sopenharmony_ci } 690e5b75505Sopenharmony_ci 691e5b75505Sopenharmony_ci if (!(ctx->enc = EVP_CIPHER_CTX_new()) || 692e5b75505Sopenharmony_ci !EVP_CIPHER_CTX_set_padding(ctx->enc, 0) || 693e5b75505Sopenharmony_ci !EVP_EncryptInit_ex(ctx->enc, cipher, NULL, NULL, NULL) || 694e5b75505Sopenharmony_ci !EVP_CIPHER_CTX_set_key_length(ctx->enc, key_len) || 695e5b75505Sopenharmony_ci !EVP_EncryptInit_ex(ctx->enc, NULL, NULL, key, iv)) { 696e5b75505Sopenharmony_ci if (ctx->enc) 697e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx->enc); 698e5b75505Sopenharmony_ci os_free(ctx); 699e5b75505Sopenharmony_ci return NULL; 700e5b75505Sopenharmony_ci } 701e5b75505Sopenharmony_ci 702e5b75505Sopenharmony_ci if (!(ctx->dec = EVP_CIPHER_CTX_new()) || 703e5b75505Sopenharmony_ci !EVP_CIPHER_CTX_set_padding(ctx->dec, 0) || 704e5b75505Sopenharmony_ci !EVP_DecryptInit_ex(ctx->dec, cipher, NULL, NULL, NULL) || 705e5b75505Sopenharmony_ci !EVP_CIPHER_CTX_set_key_length(ctx->dec, key_len) || 706e5b75505Sopenharmony_ci !EVP_DecryptInit_ex(ctx->dec, NULL, NULL, key, iv)) { 707e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx->enc); 708e5b75505Sopenharmony_ci if (ctx->dec) 709e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx->dec); 710e5b75505Sopenharmony_ci os_free(ctx); 711e5b75505Sopenharmony_ci return NULL; 712e5b75505Sopenharmony_ci } 713e5b75505Sopenharmony_ci 714e5b75505Sopenharmony_ci return ctx; 715e5b75505Sopenharmony_ci} 716e5b75505Sopenharmony_ci 717e5b75505Sopenharmony_ci 718e5b75505Sopenharmony_ciint crypto_cipher_encrypt(struct crypto_cipher *ctx, const u8 *plain, 719e5b75505Sopenharmony_ci u8 *crypt, size_t len) 720e5b75505Sopenharmony_ci{ 721e5b75505Sopenharmony_ci int outl; 722e5b75505Sopenharmony_ci if (!EVP_EncryptUpdate(ctx->enc, crypt, &outl, plain, len)) 723e5b75505Sopenharmony_ci return -1; 724e5b75505Sopenharmony_ci return 0; 725e5b75505Sopenharmony_ci} 726e5b75505Sopenharmony_ci 727e5b75505Sopenharmony_ci 728e5b75505Sopenharmony_ciint crypto_cipher_decrypt(struct crypto_cipher *ctx, const u8 *crypt, 729e5b75505Sopenharmony_ci u8 *plain, size_t len) 730e5b75505Sopenharmony_ci{ 731e5b75505Sopenharmony_ci int outl; 732e5b75505Sopenharmony_ci outl = len; 733e5b75505Sopenharmony_ci if (!EVP_DecryptUpdate(ctx->dec, plain, &outl, crypt, len)) 734e5b75505Sopenharmony_ci return -1; 735e5b75505Sopenharmony_ci return 0; 736e5b75505Sopenharmony_ci} 737e5b75505Sopenharmony_ci 738e5b75505Sopenharmony_ci 739e5b75505Sopenharmony_civoid crypto_cipher_deinit(struct crypto_cipher *ctx) 740e5b75505Sopenharmony_ci{ 741e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx->enc); 742e5b75505Sopenharmony_ci EVP_CIPHER_CTX_free(ctx->dec); 743e5b75505Sopenharmony_ci os_free(ctx); 744e5b75505Sopenharmony_ci} 745e5b75505Sopenharmony_ci 746e5b75505Sopenharmony_ci 747e5b75505Sopenharmony_civoid * dh5_init(struct wpabuf **priv, struct wpabuf **publ) 748e5b75505Sopenharmony_ci{ 749e5b75505Sopenharmony_ci#if OPENSSL_VERSION_NUMBER < 0x10100000L || \ 750e5b75505Sopenharmony_ci (defined(LIBRESSL_VERSION_NUMBER) && \ 751e5b75505Sopenharmony_ci LIBRESSL_VERSION_NUMBER < 0x20700000L) 752e5b75505Sopenharmony_ci DH *dh; 753e5b75505Sopenharmony_ci struct wpabuf *pubkey = NULL, *privkey = NULL; 754e5b75505Sopenharmony_ci size_t publen, privlen; 755e5b75505Sopenharmony_ci 756e5b75505Sopenharmony_ci *priv = NULL; 757e5b75505Sopenharmony_ci wpabuf_free(*publ); 758e5b75505Sopenharmony_ci *publ = NULL; 759e5b75505Sopenharmony_ci 760e5b75505Sopenharmony_ci dh = DH_new(); 761e5b75505Sopenharmony_ci if (dh == NULL) 762e5b75505Sopenharmony_ci return NULL; 763e5b75505Sopenharmony_ci 764e5b75505Sopenharmony_ci dh->g = BN_new(); 765e5b75505Sopenharmony_ci if (dh->g == NULL || BN_set_word(dh->g, 2) != 1) 766e5b75505Sopenharmony_ci goto err; 767e5b75505Sopenharmony_ci 768e5b75505Sopenharmony_ci dh->p = get_group5_prime(); 769e5b75505Sopenharmony_ci if (dh->p == NULL) 770e5b75505Sopenharmony_ci goto err; 771e5b75505Sopenharmony_ci 772e5b75505Sopenharmony_ci dh->q = get_group5_order(); 773e5b75505Sopenharmony_ci if (!dh->q) 774e5b75505Sopenharmony_ci goto err; 775e5b75505Sopenharmony_ci 776e5b75505Sopenharmony_ci if (DH_generate_key(dh) != 1) 777e5b75505Sopenharmony_ci goto err; 778e5b75505Sopenharmony_ci 779e5b75505Sopenharmony_ci publen = BN_num_bytes(dh->pub_key); 780e5b75505Sopenharmony_ci pubkey = wpabuf_alloc(publen); 781e5b75505Sopenharmony_ci if (pubkey == NULL) 782e5b75505Sopenharmony_ci goto err; 783e5b75505Sopenharmony_ci privlen = BN_num_bytes(dh->priv_key); 784e5b75505Sopenharmony_ci privkey = wpabuf_alloc(privlen); 785e5b75505Sopenharmony_ci if (privkey == NULL) 786e5b75505Sopenharmony_ci goto err; 787e5b75505Sopenharmony_ci 788e5b75505Sopenharmony_ci BN_bn2bin(dh->pub_key, wpabuf_put(pubkey, publen)); 789e5b75505Sopenharmony_ci BN_bn2bin(dh->priv_key, wpabuf_put(privkey, privlen)); 790e5b75505Sopenharmony_ci 791e5b75505Sopenharmony_ci *priv = privkey; 792e5b75505Sopenharmony_ci *publ = pubkey; 793e5b75505Sopenharmony_ci return dh; 794e5b75505Sopenharmony_ci 795e5b75505Sopenharmony_cierr: 796e5b75505Sopenharmony_ci wpabuf_clear_free(pubkey); 797e5b75505Sopenharmony_ci wpabuf_clear_free(privkey); 798e5b75505Sopenharmony_ci DH_free(dh); 799e5b75505Sopenharmony_ci return NULL; 800e5b75505Sopenharmony_ci#else 801e5b75505Sopenharmony_ci DH *dh; 802e5b75505Sopenharmony_ci struct wpabuf *pubkey = NULL, *privkey = NULL; 803e5b75505Sopenharmony_ci size_t publen, privlen; 804e5b75505Sopenharmony_ci BIGNUM *p, *g, *q; 805e5b75505Sopenharmony_ci const BIGNUM *priv_key = NULL, *pub_key = NULL; 806e5b75505Sopenharmony_ci 807e5b75505Sopenharmony_ci *priv = NULL; 808e5b75505Sopenharmony_ci wpabuf_free(*publ); 809e5b75505Sopenharmony_ci *publ = NULL; 810e5b75505Sopenharmony_ci 811e5b75505Sopenharmony_ci dh = DH_new(); 812e5b75505Sopenharmony_ci if (dh == NULL) 813e5b75505Sopenharmony_ci return NULL; 814e5b75505Sopenharmony_ci 815e5b75505Sopenharmony_ci g = BN_new(); 816e5b75505Sopenharmony_ci p = get_group5_prime(); 817e5b75505Sopenharmony_ci q = get_group5_order(); 818e5b75505Sopenharmony_ci if (!g || BN_set_word(g, 2) != 1 || !p || !q || 819e5b75505Sopenharmony_ci DH_set0_pqg(dh, p, q, g) != 1) 820e5b75505Sopenharmony_ci goto err; 821e5b75505Sopenharmony_ci p = NULL; 822e5b75505Sopenharmony_ci q = NULL; 823e5b75505Sopenharmony_ci g = NULL; 824e5b75505Sopenharmony_ci 825e5b75505Sopenharmony_ci if (DH_generate_key(dh) != 1) 826e5b75505Sopenharmony_ci goto err; 827e5b75505Sopenharmony_ci 828e5b75505Sopenharmony_ci DH_get0_key(dh, &pub_key, &priv_key); 829e5b75505Sopenharmony_ci publen = BN_num_bytes(pub_key); 830e5b75505Sopenharmony_ci pubkey = wpabuf_alloc(publen); 831e5b75505Sopenharmony_ci if (!pubkey) 832e5b75505Sopenharmony_ci goto err; 833e5b75505Sopenharmony_ci privlen = BN_num_bytes(priv_key); 834e5b75505Sopenharmony_ci privkey = wpabuf_alloc(privlen); 835e5b75505Sopenharmony_ci if (!privkey) 836e5b75505Sopenharmony_ci goto err; 837e5b75505Sopenharmony_ci 838e5b75505Sopenharmony_ci BN_bn2bin(pub_key, wpabuf_put(pubkey, publen)); 839e5b75505Sopenharmony_ci BN_bn2bin(priv_key, wpabuf_put(privkey, privlen)); 840e5b75505Sopenharmony_ci 841e5b75505Sopenharmony_ci *priv = privkey; 842e5b75505Sopenharmony_ci *publ = pubkey; 843e5b75505Sopenharmony_ci return dh; 844e5b75505Sopenharmony_ci 845e5b75505Sopenharmony_cierr: 846e5b75505Sopenharmony_ci BN_free(p); 847e5b75505Sopenharmony_ci BN_free(q); 848e5b75505Sopenharmony_ci BN_free(g); 849e5b75505Sopenharmony_ci wpabuf_clear_free(pubkey); 850e5b75505Sopenharmony_ci wpabuf_clear_free(privkey); 851e5b75505Sopenharmony_ci DH_free(dh); 852e5b75505Sopenharmony_ci return NULL; 853e5b75505Sopenharmony_ci#endif 854e5b75505Sopenharmony_ci} 855e5b75505Sopenharmony_ci 856e5b75505Sopenharmony_ci 857e5b75505Sopenharmony_civoid * dh5_init_fixed(const struct wpabuf *priv, const struct wpabuf *publ) 858e5b75505Sopenharmony_ci{ 859e5b75505Sopenharmony_ci#if OPENSSL_VERSION_NUMBER < 0x10100000L || \ 860e5b75505Sopenharmony_ci (defined(LIBRESSL_VERSION_NUMBER) && \ 861e5b75505Sopenharmony_ci LIBRESSL_VERSION_NUMBER < 0x20700000L) 862e5b75505Sopenharmony_ci DH *dh; 863e5b75505Sopenharmony_ci 864e5b75505Sopenharmony_ci dh = DH_new(); 865e5b75505Sopenharmony_ci if (dh == NULL) 866e5b75505Sopenharmony_ci return NULL; 867e5b75505Sopenharmony_ci 868e5b75505Sopenharmony_ci dh->g = BN_new(); 869e5b75505Sopenharmony_ci if (dh->g == NULL || BN_set_word(dh->g, 2) != 1) 870e5b75505Sopenharmony_ci goto err; 871e5b75505Sopenharmony_ci 872e5b75505Sopenharmony_ci dh->p = get_group5_prime(); 873e5b75505Sopenharmony_ci if (dh->p == NULL) 874e5b75505Sopenharmony_ci goto err; 875e5b75505Sopenharmony_ci 876e5b75505Sopenharmony_ci dh->priv_key = BN_bin2bn(wpabuf_head(priv), wpabuf_len(priv), NULL); 877e5b75505Sopenharmony_ci if (dh->priv_key == NULL) 878e5b75505Sopenharmony_ci goto err; 879e5b75505Sopenharmony_ci 880e5b75505Sopenharmony_ci dh->pub_key = BN_bin2bn(wpabuf_head(publ), wpabuf_len(publ), NULL); 881e5b75505Sopenharmony_ci if (dh->pub_key == NULL) 882e5b75505Sopenharmony_ci goto err; 883e5b75505Sopenharmony_ci 884e5b75505Sopenharmony_ci if (DH_generate_key(dh) != 1) 885e5b75505Sopenharmony_ci goto err; 886e5b75505Sopenharmony_ci 887e5b75505Sopenharmony_ci return dh; 888e5b75505Sopenharmony_ci 889e5b75505Sopenharmony_cierr: 890e5b75505Sopenharmony_ci DH_free(dh); 891e5b75505Sopenharmony_ci return NULL; 892e5b75505Sopenharmony_ci#else 893e5b75505Sopenharmony_ci DH *dh; 894e5b75505Sopenharmony_ci BIGNUM *p = NULL, *g, *priv_key = NULL, *pub_key = NULL; 895e5b75505Sopenharmony_ci 896e5b75505Sopenharmony_ci dh = DH_new(); 897e5b75505Sopenharmony_ci if (dh == NULL) 898e5b75505Sopenharmony_ci return NULL; 899e5b75505Sopenharmony_ci 900e5b75505Sopenharmony_ci g = BN_new(); 901e5b75505Sopenharmony_ci p = get_group5_prime(); 902e5b75505Sopenharmony_ci if (!g || BN_set_word(g, 2) != 1 || !p || 903e5b75505Sopenharmony_ci DH_set0_pqg(dh, p, NULL, g) != 1) 904e5b75505Sopenharmony_ci goto err; 905e5b75505Sopenharmony_ci p = NULL; 906e5b75505Sopenharmony_ci g = NULL; 907e5b75505Sopenharmony_ci 908e5b75505Sopenharmony_ci priv_key = BN_bin2bn(wpabuf_head(priv), wpabuf_len(priv), NULL); 909e5b75505Sopenharmony_ci pub_key = BN_bin2bn(wpabuf_head(publ), wpabuf_len(publ), NULL); 910e5b75505Sopenharmony_ci if (!priv_key || !pub_key || DH_set0_key(dh, pub_key, priv_key) != 1) 911e5b75505Sopenharmony_ci goto err; 912e5b75505Sopenharmony_ci pub_key = NULL; 913e5b75505Sopenharmony_ci priv_key = NULL; 914e5b75505Sopenharmony_ci 915e5b75505Sopenharmony_ci if (DH_generate_key(dh) != 1) 916e5b75505Sopenharmony_ci goto err; 917e5b75505Sopenharmony_ci 918e5b75505Sopenharmony_ci return dh; 919e5b75505Sopenharmony_ci 920e5b75505Sopenharmony_cierr: 921e5b75505Sopenharmony_ci BN_free(p); 922e5b75505Sopenharmony_ci BN_free(g); 923e5b75505Sopenharmony_ci BN_free(pub_key); 924e5b75505Sopenharmony_ci BN_clear_free(priv_key); 925e5b75505Sopenharmony_ci DH_free(dh); 926e5b75505Sopenharmony_ci return NULL; 927e5b75505Sopenharmony_ci#endif 928e5b75505Sopenharmony_ci} 929e5b75505Sopenharmony_ci 930e5b75505Sopenharmony_ci 931e5b75505Sopenharmony_cistruct wpabuf * dh5_derive_shared(void *ctx, const struct wpabuf *peer_public, 932e5b75505Sopenharmony_ci const struct wpabuf *own_private) 933e5b75505Sopenharmony_ci{ 934e5b75505Sopenharmony_ci BIGNUM *pub_key; 935e5b75505Sopenharmony_ci struct wpabuf *res = NULL; 936e5b75505Sopenharmony_ci size_t rlen; 937e5b75505Sopenharmony_ci DH *dh = ctx; 938e5b75505Sopenharmony_ci int keylen; 939e5b75505Sopenharmony_ci 940e5b75505Sopenharmony_ci if (ctx == NULL) 941e5b75505Sopenharmony_ci return NULL; 942e5b75505Sopenharmony_ci 943e5b75505Sopenharmony_ci pub_key = BN_bin2bn(wpabuf_head(peer_public), wpabuf_len(peer_public), 944e5b75505Sopenharmony_ci NULL); 945e5b75505Sopenharmony_ci if (pub_key == NULL) 946e5b75505Sopenharmony_ci return NULL; 947e5b75505Sopenharmony_ci 948e5b75505Sopenharmony_ci rlen = DH_size(dh); 949e5b75505Sopenharmony_ci res = wpabuf_alloc(rlen); 950e5b75505Sopenharmony_ci if (res == NULL) 951e5b75505Sopenharmony_ci goto err; 952e5b75505Sopenharmony_ci 953e5b75505Sopenharmony_ci keylen = DH_compute_key(wpabuf_mhead(res), pub_key, dh); 954e5b75505Sopenharmony_ci if (keylen < 0) 955e5b75505Sopenharmony_ci goto err; 956e5b75505Sopenharmony_ci wpabuf_put(res, keylen); 957e5b75505Sopenharmony_ci BN_clear_free(pub_key); 958e5b75505Sopenharmony_ci 959e5b75505Sopenharmony_ci return res; 960e5b75505Sopenharmony_ci 961e5b75505Sopenharmony_cierr: 962e5b75505Sopenharmony_ci BN_clear_free(pub_key); 963e5b75505Sopenharmony_ci wpabuf_clear_free(res); 964e5b75505Sopenharmony_ci return NULL; 965e5b75505Sopenharmony_ci} 966e5b75505Sopenharmony_ci 967e5b75505Sopenharmony_ci 968e5b75505Sopenharmony_civoid dh5_free(void *ctx) 969e5b75505Sopenharmony_ci{ 970e5b75505Sopenharmony_ci DH *dh; 971e5b75505Sopenharmony_ci if (ctx == NULL) 972e5b75505Sopenharmony_ci return; 973e5b75505Sopenharmony_ci dh = ctx; 974e5b75505Sopenharmony_ci DH_free(dh); 975e5b75505Sopenharmony_ci} 976e5b75505Sopenharmony_ci 977e5b75505Sopenharmony_ci 978e5b75505Sopenharmony_cistruct crypto_hash { 979e5b75505Sopenharmony_ci HMAC_CTX *ctx; 980e5b75505Sopenharmony_ci}; 981e5b75505Sopenharmony_ci 982e5b75505Sopenharmony_ci 983e5b75505Sopenharmony_cistruct crypto_hash * crypto_hash_init(enum crypto_hash_alg alg, const u8 *key, 984e5b75505Sopenharmony_ci size_t key_len) 985e5b75505Sopenharmony_ci{ 986e5b75505Sopenharmony_ci struct crypto_hash *ctx; 987e5b75505Sopenharmony_ci const EVP_MD *md; 988e5b75505Sopenharmony_ci 989e5b75505Sopenharmony_ci switch (alg) { 990e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_MD5 991e5b75505Sopenharmony_ci case CRYPTO_HASH_ALG_HMAC_MD5: 992e5b75505Sopenharmony_ci md = EVP_md5(); 993e5b75505Sopenharmony_ci break; 994e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_MD5 */ 995e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_SHA 996e5b75505Sopenharmony_ci case CRYPTO_HASH_ALG_HMAC_SHA1: 997e5b75505Sopenharmony_ci md = EVP_sha1(); 998e5b75505Sopenharmony_ci break; 999e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_SHA */ 1000e5b75505Sopenharmony_ci#ifndef OPENSSL_NO_SHA256 1001e5b75505Sopenharmony_ci#ifdef CONFIG_SHA256 1002e5b75505Sopenharmony_ci case CRYPTO_HASH_ALG_HMAC_SHA256: 1003e5b75505Sopenharmony_ci md = EVP_sha256(); 1004e5b75505Sopenharmony_ci break; 1005e5b75505Sopenharmony_ci#endif /* CONFIG_SHA256 */ 1006e5b75505Sopenharmony_ci#endif /* OPENSSL_NO_SHA256 */ 1007e5b75505Sopenharmony_ci default: 1008e5b75505Sopenharmony_ci return NULL; 1009e5b75505Sopenharmony_ci } 1010e5b75505Sopenharmony_ci 1011e5b75505Sopenharmony_ci ctx = os_zalloc(sizeof(*ctx)); 1012e5b75505Sopenharmony_ci if (ctx == NULL) 1013e5b75505Sopenharmony_ci return NULL; 1014e5b75505Sopenharmony_ci ctx->ctx = HMAC_CTX_new(); 1015e5b75505Sopenharmony_ci if (!ctx->ctx) { 1016e5b75505Sopenharmony_ci os_free(ctx); 1017e5b75505Sopenharmony_ci return NULL; 1018e5b75505Sopenharmony_ci } 1019e5b75505Sopenharmony_ci 1020e5b75505Sopenharmony_ci if (HMAC_Init_ex(ctx->ctx, key, key_len, md, NULL) != 1) { 1021e5b75505Sopenharmony_ci HMAC_CTX_free(ctx->ctx); 1022e5b75505Sopenharmony_ci bin_clear_free(ctx, sizeof(*ctx)); 1023e5b75505Sopenharmony_ci return NULL; 1024e5b75505Sopenharmony_ci } 1025e5b75505Sopenharmony_ci 1026e5b75505Sopenharmony_ci return ctx; 1027e5b75505Sopenharmony_ci} 1028e5b75505Sopenharmony_ci 1029e5b75505Sopenharmony_ci 1030e5b75505Sopenharmony_civoid crypto_hash_update(struct crypto_hash *ctx, const u8 *data, size_t len) 1031e5b75505Sopenharmony_ci{ 1032e5b75505Sopenharmony_ci if (ctx == NULL) 1033e5b75505Sopenharmony_ci return; 1034e5b75505Sopenharmony_ci HMAC_Update(ctx->ctx, data, len); 1035e5b75505Sopenharmony_ci} 1036e5b75505Sopenharmony_ci 1037e5b75505Sopenharmony_ci 1038e5b75505Sopenharmony_ciint crypto_hash_finish(struct crypto_hash *ctx, u8 *mac, size_t *len) 1039e5b75505Sopenharmony_ci{ 1040e5b75505Sopenharmony_ci unsigned int mdlen; 1041e5b75505Sopenharmony_ci int res; 1042e5b75505Sopenharmony_ci 1043e5b75505Sopenharmony_ci if (ctx == NULL) 1044e5b75505Sopenharmony_ci return -2; 1045e5b75505Sopenharmony_ci 1046e5b75505Sopenharmony_ci if (mac == NULL || len == NULL) { 1047e5b75505Sopenharmony_ci HMAC_CTX_free(ctx->ctx); 1048e5b75505Sopenharmony_ci bin_clear_free(ctx, sizeof(*ctx)); 1049e5b75505Sopenharmony_ci return 0; 1050e5b75505Sopenharmony_ci } 1051e5b75505Sopenharmony_ci 1052e5b75505Sopenharmony_ci mdlen = *len; 1053e5b75505Sopenharmony_ci res = HMAC_Final(ctx->ctx, mac, &mdlen); 1054e5b75505Sopenharmony_ci HMAC_CTX_free(ctx->ctx); 1055e5b75505Sopenharmony_ci bin_clear_free(ctx, sizeof(*ctx)); 1056e5b75505Sopenharmony_ci 1057e5b75505Sopenharmony_ci if (TEST_FAIL()) 1058e5b75505Sopenharmony_ci return -1; 1059e5b75505Sopenharmony_ci 1060e5b75505Sopenharmony_ci if (res == 1) { 1061e5b75505Sopenharmony_ci *len = mdlen; 1062e5b75505Sopenharmony_ci return 0; 1063e5b75505Sopenharmony_ci } 1064e5b75505Sopenharmony_ci 1065e5b75505Sopenharmony_ci return -1; 1066e5b75505Sopenharmony_ci} 1067e5b75505Sopenharmony_ci 1068e5b75505Sopenharmony_ci 1069e5b75505Sopenharmony_cistatic int openssl_hmac_vector(const EVP_MD *type, const u8 *key, 1070e5b75505Sopenharmony_ci size_t key_len, size_t num_elem, 1071e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac, 1072e5b75505Sopenharmony_ci unsigned int mdlen) 1073e5b75505Sopenharmony_ci{ 1074e5b75505Sopenharmony_ci HMAC_CTX *ctx; 1075e5b75505Sopenharmony_ci size_t i; 1076e5b75505Sopenharmony_ci int res; 1077e5b75505Sopenharmony_ci 1078e5b75505Sopenharmony_ci if (TEST_FAIL()) 1079e5b75505Sopenharmony_ci return -1; 1080e5b75505Sopenharmony_ci 1081e5b75505Sopenharmony_ci ctx = HMAC_CTX_new(); 1082e5b75505Sopenharmony_ci if (!ctx) 1083e5b75505Sopenharmony_ci return -1; 1084e5b75505Sopenharmony_ci res = HMAC_Init_ex(ctx, key, key_len, type, NULL); 1085e5b75505Sopenharmony_ci if (res != 1) 1086e5b75505Sopenharmony_ci goto done; 1087e5b75505Sopenharmony_ci 1088e5b75505Sopenharmony_ci for (i = 0; i < num_elem; i++) 1089e5b75505Sopenharmony_ci HMAC_Update(ctx, addr[i], len[i]); 1090e5b75505Sopenharmony_ci 1091e5b75505Sopenharmony_ci res = HMAC_Final(ctx, mac, &mdlen); 1092e5b75505Sopenharmony_cidone: 1093e5b75505Sopenharmony_ci HMAC_CTX_free(ctx); 1094e5b75505Sopenharmony_ci 1095e5b75505Sopenharmony_ci return res == 1 ? 0 : -1; 1096e5b75505Sopenharmony_ci} 1097e5b75505Sopenharmony_ci 1098e5b75505Sopenharmony_ci 1099e5b75505Sopenharmony_ci#ifndef CONFIG_FIPS 1100e5b75505Sopenharmony_ci 1101e5b75505Sopenharmony_ciint hmac_md5_vector(const u8 *key, size_t key_len, size_t num_elem, 1102e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1103e5b75505Sopenharmony_ci{ 1104e5b75505Sopenharmony_ci return openssl_hmac_vector(EVP_md5(), key ,key_len, num_elem, addr, len, 1105e5b75505Sopenharmony_ci mac, 16); 1106e5b75505Sopenharmony_ci} 1107e5b75505Sopenharmony_ci 1108e5b75505Sopenharmony_ci 1109e5b75505Sopenharmony_ciint hmac_md5(const u8 *key, size_t key_len, const u8 *data, size_t data_len, 1110e5b75505Sopenharmony_ci u8 *mac) 1111e5b75505Sopenharmony_ci{ 1112e5b75505Sopenharmony_ci return hmac_md5_vector(key, key_len, 1, &data, &data_len, mac); 1113e5b75505Sopenharmony_ci} 1114e5b75505Sopenharmony_ci 1115e5b75505Sopenharmony_ci#endif /* CONFIG_FIPS */ 1116e5b75505Sopenharmony_ci 1117e5b75505Sopenharmony_ci 1118e5b75505Sopenharmony_ciint pbkdf2_sha1(const char *passphrase, const u8 *ssid, size_t ssid_len, 1119e5b75505Sopenharmony_ci int iterations, u8 *buf, size_t buflen) 1120e5b75505Sopenharmony_ci{ 1121e5b75505Sopenharmony_ci if (PKCS5_PBKDF2_HMAC_SHA1(passphrase, os_strlen(passphrase), ssid, 1122e5b75505Sopenharmony_ci ssid_len, iterations, buflen, buf) != 1) 1123e5b75505Sopenharmony_ci return -1; 1124e5b75505Sopenharmony_ci return 0; 1125e5b75505Sopenharmony_ci} 1126e5b75505Sopenharmony_ci 1127e5b75505Sopenharmony_ci 1128e5b75505Sopenharmony_ciint hmac_sha1_vector(const u8 *key, size_t key_len, size_t num_elem, 1129e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1130e5b75505Sopenharmony_ci{ 1131e5b75505Sopenharmony_ci return openssl_hmac_vector(EVP_sha1(), key, key_len, num_elem, addr, 1132e5b75505Sopenharmony_ci len, mac, 20); 1133e5b75505Sopenharmony_ci} 1134e5b75505Sopenharmony_ci 1135e5b75505Sopenharmony_ci 1136e5b75505Sopenharmony_ciint hmac_sha1(const u8 *key, size_t key_len, const u8 *data, size_t data_len, 1137e5b75505Sopenharmony_ci u8 *mac) 1138e5b75505Sopenharmony_ci{ 1139e5b75505Sopenharmony_ci return hmac_sha1_vector(key, key_len, 1, &data, &data_len, mac); 1140e5b75505Sopenharmony_ci} 1141e5b75505Sopenharmony_ci 1142e5b75505Sopenharmony_ci 1143e5b75505Sopenharmony_ci#ifdef CONFIG_SHA256 1144e5b75505Sopenharmony_ci 1145e5b75505Sopenharmony_ciint hmac_sha256_vector(const u8 *key, size_t key_len, size_t num_elem, 1146e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1147e5b75505Sopenharmony_ci{ 1148e5b75505Sopenharmony_ci return openssl_hmac_vector(EVP_sha256(), key, key_len, num_elem, addr, 1149e5b75505Sopenharmony_ci len, mac, 32); 1150e5b75505Sopenharmony_ci} 1151e5b75505Sopenharmony_ci 1152e5b75505Sopenharmony_ci 1153e5b75505Sopenharmony_ciint hmac_sha256(const u8 *key, size_t key_len, const u8 *data, 1154e5b75505Sopenharmony_ci size_t data_len, u8 *mac) 1155e5b75505Sopenharmony_ci{ 1156e5b75505Sopenharmony_ci return hmac_sha256_vector(key, key_len, 1, &data, &data_len, mac); 1157e5b75505Sopenharmony_ci} 1158e5b75505Sopenharmony_ci 1159e5b75505Sopenharmony_ci#endif /* CONFIG_SHA256 */ 1160e5b75505Sopenharmony_ci 1161e5b75505Sopenharmony_ci 1162e5b75505Sopenharmony_ci#ifdef CONFIG_SHA384 1163e5b75505Sopenharmony_ci 1164e5b75505Sopenharmony_ciint hmac_sha384_vector(const u8 *key, size_t key_len, size_t num_elem, 1165e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1166e5b75505Sopenharmony_ci{ 1167e5b75505Sopenharmony_ci return openssl_hmac_vector(EVP_sha384(), key, key_len, num_elem, addr, 1168e5b75505Sopenharmony_ci len, mac, 48); 1169e5b75505Sopenharmony_ci} 1170e5b75505Sopenharmony_ci 1171e5b75505Sopenharmony_ci 1172e5b75505Sopenharmony_ciint hmac_sha384(const u8 *key, size_t key_len, const u8 *data, 1173e5b75505Sopenharmony_ci size_t data_len, u8 *mac) 1174e5b75505Sopenharmony_ci{ 1175e5b75505Sopenharmony_ci return hmac_sha384_vector(key, key_len, 1, &data, &data_len, mac); 1176e5b75505Sopenharmony_ci} 1177e5b75505Sopenharmony_ci 1178e5b75505Sopenharmony_ci#endif /* CONFIG_SHA384 */ 1179e5b75505Sopenharmony_ci 1180e5b75505Sopenharmony_ci 1181e5b75505Sopenharmony_ci#ifdef CONFIG_SHA512 1182e5b75505Sopenharmony_ci 1183e5b75505Sopenharmony_ciint hmac_sha512_vector(const u8 *key, size_t key_len, size_t num_elem, 1184e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1185e5b75505Sopenharmony_ci{ 1186e5b75505Sopenharmony_ci return openssl_hmac_vector(EVP_sha512(), key, key_len, num_elem, addr, 1187e5b75505Sopenharmony_ci len, mac, 64); 1188e5b75505Sopenharmony_ci} 1189e5b75505Sopenharmony_ci 1190e5b75505Sopenharmony_ci 1191e5b75505Sopenharmony_ciint hmac_sha512(const u8 *key, size_t key_len, const u8 *data, 1192e5b75505Sopenharmony_ci size_t data_len, u8 *mac) 1193e5b75505Sopenharmony_ci{ 1194e5b75505Sopenharmony_ci return hmac_sha512_vector(key, key_len, 1, &data, &data_len, mac); 1195e5b75505Sopenharmony_ci} 1196e5b75505Sopenharmony_ci 1197e5b75505Sopenharmony_ci#endif /* CONFIG_SHA512 */ 1198e5b75505Sopenharmony_ci 1199e5b75505Sopenharmony_ci 1200e5b75505Sopenharmony_ciint crypto_get_random(void *buf, size_t len) 1201e5b75505Sopenharmony_ci{ 1202e5b75505Sopenharmony_ci if (RAND_bytes(buf, len) != 1) 1203e5b75505Sopenharmony_ci return -1; 1204e5b75505Sopenharmony_ci return 0; 1205e5b75505Sopenharmony_ci} 1206e5b75505Sopenharmony_ci 1207e5b75505Sopenharmony_ci 1208e5b75505Sopenharmony_ci#ifdef CONFIG_OPENSSL_CMAC 1209e5b75505Sopenharmony_ciint omac1_aes_vector(const u8 *key, size_t key_len, size_t num_elem, 1210e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1211e5b75505Sopenharmony_ci{ 1212e5b75505Sopenharmony_ci CMAC_CTX *ctx; 1213e5b75505Sopenharmony_ci int ret = -1; 1214e5b75505Sopenharmony_ci size_t outlen, i; 1215e5b75505Sopenharmony_ci 1216e5b75505Sopenharmony_ci if (TEST_FAIL()) 1217e5b75505Sopenharmony_ci return -1; 1218e5b75505Sopenharmony_ci 1219e5b75505Sopenharmony_ci ctx = CMAC_CTX_new(); 1220e5b75505Sopenharmony_ci if (ctx == NULL) 1221e5b75505Sopenharmony_ci return -1; 1222e5b75505Sopenharmony_ci 1223e5b75505Sopenharmony_ci if (key_len == 32) { 1224e5b75505Sopenharmony_ci if (!CMAC_Init(ctx, key, 32, EVP_aes_256_cbc(), NULL)) 1225e5b75505Sopenharmony_ci goto fail; 1226e5b75505Sopenharmony_ci } else if (key_len == 16) { 1227e5b75505Sopenharmony_ci if (!CMAC_Init(ctx, key, 16, EVP_aes_128_cbc(), NULL)) 1228e5b75505Sopenharmony_ci goto fail; 1229e5b75505Sopenharmony_ci } else { 1230e5b75505Sopenharmony_ci goto fail; 1231e5b75505Sopenharmony_ci } 1232e5b75505Sopenharmony_ci for (i = 0; i < num_elem; i++) { 1233e5b75505Sopenharmony_ci if (!CMAC_Update(ctx, addr[i], len[i])) 1234e5b75505Sopenharmony_ci goto fail; 1235e5b75505Sopenharmony_ci } 1236e5b75505Sopenharmony_ci if (!CMAC_Final(ctx, mac, &outlen) || outlen != 16) 1237e5b75505Sopenharmony_ci goto fail; 1238e5b75505Sopenharmony_ci 1239e5b75505Sopenharmony_ci ret = 0; 1240e5b75505Sopenharmony_cifail: 1241e5b75505Sopenharmony_ci CMAC_CTX_free(ctx); 1242e5b75505Sopenharmony_ci return ret; 1243e5b75505Sopenharmony_ci} 1244e5b75505Sopenharmony_ci 1245e5b75505Sopenharmony_ci 1246e5b75505Sopenharmony_ciint omac1_aes_128_vector(const u8 *key, size_t num_elem, 1247e5b75505Sopenharmony_ci const u8 *addr[], const size_t *len, u8 *mac) 1248e5b75505Sopenharmony_ci{ 1249e5b75505Sopenharmony_ci return omac1_aes_vector(key, 16, num_elem, addr, len, mac); 1250e5b75505Sopenharmony_ci} 1251e5b75505Sopenharmony_ci 1252e5b75505Sopenharmony_ci 1253e5b75505Sopenharmony_ciint omac1_aes_128(const u8 *key, const u8 *data, size_t data_len, u8 *mac) 1254e5b75505Sopenharmony_ci{ 1255e5b75505Sopenharmony_ci return omac1_aes_128_vector(key, 1, &data, &data_len, mac); 1256e5b75505Sopenharmony_ci} 1257e5b75505Sopenharmony_ci 1258e5b75505Sopenharmony_ci 1259e5b75505Sopenharmony_ciint omac1_aes_256(const u8 *key, const u8 *data, size_t data_len, u8 *mac) 1260e5b75505Sopenharmony_ci{ 1261e5b75505Sopenharmony_ci return omac1_aes_vector(key, 32, 1, &data, &data_len, mac); 1262e5b75505Sopenharmony_ci} 1263e5b75505Sopenharmony_ci#endif /* CONFIG_OPENSSL_CMAC */ 1264e5b75505Sopenharmony_ci 1265e5b75505Sopenharmony_ci 1266e5b75505Sopenharmony_cistruct crypto_bignum * crypto_bignum_init(void) 1267e5b75505Sopenharmony_ci{ 1268e5b75505Sopenharmony_ci if (TEST_FAIL()) 1269e5b75505Sopenharmony_ci return NULL; 1270e5b75505Sopenharmony_ci return (struct crypto_bignum *) BN_new(); 1271e5b75505Sopenharmony_ci} 1272e5b75505Sopenharmony_ci 1273e5b75505Sopenharmony_ci 1274e5b75505Sopenharmony_cistruct crypto_bignum * crypto_bignum_init_set(const u8 *buf, size_t len) 1275e5b75505Sopenharmony_ci{ 1276e5b75505Sopenharmony_ci BIGNUM *bn; 1277e5b75505Sopenharmony_ci 1278e5b75505Sopenharmony_ci if (TEST_FAIL()) 1279e5b75505Sopenharmony_ci return NULL; 1280e5b75505Sopenharmony_ci 1281e5b75505Sopenharmony_ci bn = BN_bin2bn(buf, len, NULL); 1282e5b75505Sopenharmony_ci return (struct crypto_bignum *) bn; 1283e5b75505Sopenharmony_ci} 1284e5b75505Sopenharmony_ci 1285e5b75505Sopenharmony_ci 1286e5b75505Sopenharmony_cistruct crypto_bignum * crypto_bignum_init_uint(unsigned int val) 1287e5b75505Sopenharmony_ci{ 1288e5b75505Sopenharmony_ci BIGNUM *bn; 1289e5b75505Sopenharmony_ci 1290e5b75505Sopenharmony_ci if (TEST_FAIL()) 1291e5b75505Sopenharmony_ci return NULL; 1292e5b75505Sopenharmony_ci 1293e5b75505Sopenharmony_ci bn = BN_new(); 1294e5b75505Sopenharmony_ci if (!bn) 1295e5b75505Sopenharmony_ci return NULL; 1296e5b75505Sopenharmony_ci if (BN_set_word(bn, val) != 1) { 1297e5b75505Sopenharmony_ci BN_free(bn); 1298e5b75505Sopenharmony_ci return NULL; 1299e5b75505Sopenharmony_ci } 1300e5b75505Sopenharmony_ci return (struct crypto_bignum *) bn; 1301e5b75505Sopenharmony_ci} 1302e5b75505Sopenharmony_ci 1303e5b75505Sopenharmony_ci 1304e5b75505Sopenharmony_civoid crypto_bignum_deinit(struct crypto_bignum *n, int clear) 1305e5b75505Sopenharmony_ci{ 1306e5b75505Sopenharmony_ci if (clear) 1307e5b75505Sopenharmony_ci BN_clear_free((BIGNUM *) n); 1308e5b75505Sopenharmony_ci else 1309e5b75505Sopenharmony_ci BN_free((BIGNUM *) n); 1310e5b75505Sopenharmony_ci} 1311e5b75505Sopenharmony_ci 1312e5b75505Sopenharmony_ci 1313e5b75505Sopenharmony_ciint crypto_bignum_to_bin(const struct crypto_bignum *a, 1314e5b75505Sopenharmony_ci u8 *buf, size_t buflen, size_t padlen) 1315e5b75505Sopenharmony_ci{ 1316e5b75505Sopenharmony_ci int num_bytes, offset; 1317e5b75505Sopenharmony_ci 1318e5b75505Sopenharmony_ci if (TEST_FAIL()) 1319e5b75505Sopenharmony_ci return -1; 1320e5b75505Sopenharmony_ci 1321e5b75505Sopenharmony_ci if (padlen > buflen) 1322e5b75505Sopenharmony_ci return -1; 1323e5b75505Sopenharmony_ci 1324e5b75505Sopenharmony_ci if (padlen) { 1325e5b75505Sopenharmony_ci#ifdef OPENSSL_IS_BORINGSSL 1326e5b75505Sopenharmony_ci if (BN_bn2bin_padded(buf, padlen, (const BIGNUM *) a) == 0) 1327e5b75505Sopenharmony_ci return -1; 1328e5b75505Sopenharmony_ci return padlen; 1329e5b75505Sopenharmony_ci#else /* OPENSSL_IS_BORINGSSL */ 1330e5b75505Sopenharmony_ci#if OPENSSL_VERSION_NUMBER >= 0x10100000L && !defined(LIBRESSL_VERSION_NUMBER) 1331e5b75505Sopenharmony_ci return BN_bn2binpad((const BIGNUM *) a, buf, padlen); 1332e5b75505Sopenharmony_ci#endif 1333e5b75505Sopenharmony_ci#endif 1334e5b75505Sopenharmony_ci } 1335e5b75505Sopenharmony_ci 1336e5b75505Sopenharmony_ci num_bytes = BN_num_bytes((const BIGNUM *) a); 1337e5b75505Sopenharmony_ci if ((size_t) num_bytes > buflen) 1338e5b75505Sopenharmony_ci return -1; 1339e5b75505Sopenharmony_ci if (padlen > (size_t) num_bytes) 1340e5b75505Sopenharmony_ci offset = padlen - num_bytes; 1341e5b75505Sopenharmony_ci else 1342e5b75505Sopenharmony_ci offset = 0; 1343e5b75505Sopenharmony_ci 1344e5b75505Sopenharmony_ci os_memset(buf, 0, offset); 1345e5b75505Sopenharmony_ci BN_bn2bin((const BIGNUM *) a, buf + offset); 1346e5b75505Sopenharmony_ci 1347e5b75505Sopenharmony_ci return num_bytes + offset; 1348e5b75505Sopenharmony_ci} 1349e5b75505Sopenharmony_ci 1350e5b75505Sopenharmony_ci 1351e5b75505Sopenharmony_ciint crypto_bignum_rand(struct crypto_bignum *r, const struct crypto_bignum *m) 1352e5b75505Sopenharmony_ci{ 1353e5b75505Sopenharmony_ci if (TEST_FAIL()) 1354e5b75505Sopenharmony_ci return -1; 1355e5b75505Sopenharmony_ci return BN_rand_range((BIGNUM *) r, (const BIGNUM *) m) == 1 ? 0 : -1; 1356e5b75505Sopenharmony_ci} 1357e5b75505Sopenharmony_ci 1358e5b75505Sopenharmony_ci 1359e5b75505Sopenharmony_ciint crypto_bignum_add(const struct crypto_bignum *a, 1360e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1361e5b75505Sopenharmony_ci struct crypto_bignum *c) 1362e5b75505Sopenharmony_ci{ 1363e5b75505Sopenharmony_ci return BN_add((BIGNUM *) c, (const BIGNUM *) a, (const BIGNUM *) b) ? 1364e5b75505Sopenharmony_ci 0 : -1; 1365e5b75505Sopenharmony_ci} 1366e5b75505Sopenharmony_ci 1367e5b75505Sopenharmony_ci 1368e5b75505Sopenharmony_ciint crypto_bignum_mod(const struct crypto_bignum *a, 1369e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1370e5b75505Sopenharmony_ci struct crypto_bignum *c) 1371e5b75505Sopenharmony_ci{ 1372e5b75505Sopenharmony_ci int res; 1373e5b75505Sopenharmony_ci BN_CTX *bnctx; 1374e5b75505Sopenharmony_ci 1375e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1376e5b75505Sopenharmony_ci if (bnctx == NULL) 1377e5b75505Sopenharmony_ci return -1; 1378e5b75505Sopenharmony_ci res = BN_mod((BIGNUM *) c, (const BIGNUM *) a, (const BIGNUM *) b, 1379e5b75505Sopenharmony_ci bnctx); 1380e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1381e5b75505Sopenharmony_ci 1382e5b75505Sopenharmony_ci return res ? 0 : -1; 1383e5b75505Sopenharmony_ci} 1384e5b75505Sopenharmony_ci 1385e5b75505Sopenharmony_ci 1386e5b75505Sopenharmony_ciint crypto_bignum_exptmod(const struct crypto_bignum *a, 1387e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1388e5b75505Sopenharmony_ci const struct crypto_bignum *c, 1389e5b75505Sopenharmony_ci struct crypto_bignum *d) 1390e5b75505Sopenharmony_ci{ 1391e5b75505Sopenharmony_ci int res; 1392e5b75505Sopenharmony_ci BN_CTX *bnctx; 1393e5b75505Sopenharmony_ci 1394e5b75505Sopenharmony_ci if (TEST_FAIL()) 1395e5b75505Sopenharmony_ci return -1; 1396e5b75505Sopenharmony_ci 1397e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1398e5b75505Sopenharmony_ci if (bnctx == NULL) 1399e5b75505Sopenharmony_ci return -1; 1400e5b75505Sopenharmony_ci res = BN_mod_exp_mont_consttime((BIGNUM *) d, (const BIGNUM *) a, 1401e5b75505Sopenharmony_ci (const BIGNUM *) b, (const BIGNUM *) c, 1402e5b75505Sopenharmony_ci bnctx, NULL); 1403e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1404e5b75505Sopenharmony_ci 1405e5b75505Sopenharmony_ci return res ? 0 : -1; 1406e5b75505Sopenharmony_ci} 1407e5b75505Sopenharmony_ci 1408e5b75505Sopenharmony_ci 1409e5b75505Sopenharmony_ciint crypto_bignum_inverse(const struct crypto_bignum *a, 1410e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1411e5b75505Sopenharmony_ci struct crypto_bignum *c) 1412e5b75505Sopenharmony_ci{ 1413e5b75505Sopenharmony_ci BIGNUM *res; 1414e5b75505Sopenharmony_ci BN_CTX *bnctx; 1415e5b75505Sopenharmony_ci 1416e5b75505Sopenharmony_ci if (TEST_FAIL()) 1417e5b75505Sopenharmony_ci return -1; 1418e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1419e5b75505Sopenharmony_ci if (bnctx == NULL) 1420e5b75505Sopenharmony_ci return -1; 1421e5b75505Sopenharmony_ci#ifdef OPENSSL_IS_BORINGSSL 1422e5b75505Sopenharmony_ci /* TODO: use BN_mod_inverse_blinded() ? */ 1423e5b75505Sopenharmony_ci#else /* OPENSSL_IS_BORINGSSL */ 1424e5b75505Sopenharmony_ci BN_set_flags((BIGNUM *) a, BN_FLG_CONSTTIME); 1425e5b75505Sopenharmony_ci#endif /* OPENSSL_IS_BORINGSSL */ 1426e5b75505Sopenharmony_ci res = BN_mod_inverse((BIGNUM *) c, (const BIGNUM *) a, 1427e5b75505Sopenharmony_ci (const BIGNUM *) b, bnctx); 1428e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1429e5b75505Sopenharmony_ci 1430e5b75505Sopenharmony_ci return res ? 0 : -1; 1431e5b75505Sopenharmony_ci} 1432e5b75505Sopenharmony_ci 1433e5b75505Sopenharmony_ci 1434e5b75505Sopenharmony_ciint crypto_bignum_sub(const struct crypto_bignum *a, 1435e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1436e5b75505Sopenharmony_ci struct crypto_bignum *c) 1437e5b75505Sopenharmony_ci{ 1438e5b75505Sopenharmony_ci if (TEST_FAIL()) 1439e5b75505Sopenharmony_ci return -1; 1440e5b75505Sopenharmony_ci return BN_sub((BIGNUM *) c, (const BIGNUM *) a, (const BIGNUM *) b) ? 1441e5b75505Sopenharmony_ci 0 : -1; 1442e5b75505Sopenharmony_ci} 1443e5b75505Sopenharmony_ci 1444e5b75505Sopenharmony_ci 1445e5b75505Sopenharmony_ciint crypto_bignum_div(const struct crypto_bignum *a, 1446e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1447e5b75505Sopenharmony_ci struct crypto_bignum *c) 1448e5b75505Sopenharmony_ci{ 1449e5b75505Sopenharmony_ci int res; 1450e5b75505Sopenharmony_ci 1451e5b75505Sopenharmony_ci BN_CTX *bnctx; 1452e5b75505Sopenharmony_ci 1453e5b75505Sopenharmony_ci if (TEST_FAIL()) 1454e5b75505Sopenharmony_ci return -1; 1455e5b75505Sopenharmony_ci 1456e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1457e5b75505Sopenharmony_ci if (bnctx == NULL) 1458e5b75505Sopenharmony_ci return -1; 1459e5b75505Sopenharmony_ci#ifndef OPENSSL_IS_BORINGSSL 1460e5b75505Sopenharmony_ci BN_set_flags((BIGNUM *) a, BN_FLG_CONSTTIME); 1461e5b75505Sopenharmony_ci#endif /* OPENSSL_IS_BORINGSSL */ 1462e5b75505Sopenharmony_ci res = BN_div((BIGNUM *) c, NULL, (const BIGNUM *) a, 1463e5b75505Sopenharmony_ci (const BIGNUM *) b, bnctx); 1464e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1465e5b75505Sopenharmony_ci 1466e5b75505Sopenharmony_ci return res ? 0 : -1; 1467e5b75505Sopenharmony_ci} 1468e5b75505Sopenharmony_ci 1469e5b75505Sopenharmony_ci 1470e5b75505Sopenharmony_ciint crypto_bignum_addmod(const struct crypto_bignum *a, 1471e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1472e5b75505Sopenharmony_ci const struct crypto_bignum *c, 1473e5b75505Sopenharmony_ci struct crypto_bignum *d) 1474e5b75505Sopenharmony_ci{ 1475e5b75505Sopenharmony_ci int res; 1476e5b75505Sopenharmony_ci BN_CTX *bnctx; 1477e5b75505Sopenharmony_ci 1478e5b75505Sopenharmony_ci if (TEST_FAIL()) 1479e5b75505Sopenharmony_ci return -1; 1480e5b75505Sopenharmony_ci 1481e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1482e5b75505Sopenharmony_ci if (!bnctx) 1483e5b75505Sopenharmony_ci return -1; 1484e5b75505Sopenharmony_ci res = BN_mod_add((BIGNUM *) d, (const BIGNUM *) a, (const BIGNUM *) b, 1485e5b75505Sopenharmony_ci (const BIGNUM *) c, bnctx); 1486e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1487e5b75505Sopenharmony_ci 1488e5b75505Sopenharmony_ci return res ? 0 : -1; 1489e5b75505Sopenharmony_ci} 1490e5b75505Sopenharmony_ci 1491e5b75505Sopenharmony_ci 1492e5b75505Sopenharmony_ciint crypto_bignum_mulmod(const struct crypto_bignum *a, 1493e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1494e5b75505Sopenharmony_ci const struct crypto_bignum *c, 1495e5b75505Sopenharmony_ci struct crypto_bignum *d) 1496e5b75505Sopenharmony_ci{ 1497e5b75505Sopenharmony_ci int res; 1498e5b75505Sopenharmony_ci 1499e5b75505Sopenharmony_ci BN_CTX *bnctx; 1500e5b75505Sopenharmony_ci 1501e5b75505Sopenharmony_ci if (TEST_FAIL()) 1502e5b75505Sopenharmony_ci return -1; 1503e5b75505Sopenharmony_ci 1504e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1505e5b75505Sopenharmony_ci if (bnctx == NULL) 1506e5b75505Sopenharmony_ci return -1; 1507e5b75505Sopenharmony_ci res = BN_mod_mul((BIGNUM *) d, (const BIGNUM *) a, (const BIGNUM *) b, 1508e5b75505Sopenharmony_ci (const BIGNUM *) c, bnctx); 1509e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1510e5b75505Sopenharmony_ci 1511e5b75505Sopenharmony_ci return res ? 0 : -1; 1512e5b75505Sopenharmony_ci} 1513e5b75505Sopenharmony_ci 1514e5b75505Sopenharmony_ci 1515e5b75505Sopenharmony_ciint crypto_bignum_sqrmod(const struct crypto_bignum *a, 1516e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1517e5b75505Sopenharmony_ci struct crypto_bignum *c) 1518e5b75505Sopenharmony_ci{ 1519e5b75505Sopenharmony_ci int res; 1520e5b75505Sopenharmony_ci BN_CTX *bnctx; 1521e5b75505Sopenharmony_ci 1522e5b75505Sopenharmony_ci if (TEST_FAIL()) 1523e5b75505Sopenharmony_ci return -1; 1524e5b75505Sopenharmony_ci 1525e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1526e5b75505Sopenharmony_ci if (!bnctx) 1527e5b75505Sopenharmony_ci return -1; 1528e5b75505Sopenharmony_ci res = BN_mod_sqr((BIGNUM *) c, (const BIGNUM *) a, (const BIGNUM *) b, 1529e5b75505Sopenharmony_ci bnctx); 1530e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1531e5b75505Sopenharmony_ci 1532e5b75505Sopenharmony_ci return res ? 0 : -1; 1533e5b75505Sopenharmony_ci} 1534e5b75505Sopenharmony_ci 1535e5b75505Sopenharmony_ci 1536e5b75505Sopenharmony_ciint crypto_bignum_sqrtmod(const struct crypto_bignum *a, 1537e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1538e5b75505Sopenharmony_ci struct crypto_bignum *c) 1539e5b75505Sopenharmony_ci{ 1540e5b75505Sopenharmony_ci BN_CTX *bnctx; 1541e5b75505Sopenharmony_ci BIGNUM *res; 1542e5b75505Sopenharmony_ci 1543e5b75505Sopenharmony_ci if (TEST_FAIL()) 1544e5b75505Sopenharmony_ci return -1; 1545e5b75505Sopenharmony_ci 1546e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1547e5b75505Sopenharmony_ci if (!bnctx) 1548e5b75505Sopenharmony_ci return -1; 1549e5b75505Sopenharmony_ci res = BN_mod_sqrt((BIGNUM *) c, (const BIGNUM *) a, (const BIGNUM *) b, 1550e5b75505Sopenharmony_ci bnctx); 1551e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1552e5b75505Sopenharmony_ci 1553e5b75505Sopenharmony_ci return res ? 0 : -1; 1554e5b75505Sopenharmony_ci} 1555e5b75505Sopenharmony_ci 1556e5b75505Sopenharmony_ci 1557e5b75505Sopenharmony_ciint crypto_bignum_rshift(const struct crypto_bignum *a, int n, 1558e5b75505Sopenharmony_ci struct crypto_bignum *r) 1559e5b75505Sopenharmony_ci{ 1560e5b75505Sopenharmony_ci /* Note: BN_rshift() does not modify the first argument even though it 1561e5b75505Sopenharmony_ci * has not been marked const. */ 1562e5b75505Sopenharmony_ci return BN_rshift((BIGNUM *) a, (BIGNUM *) r, n) == 1 ? 0 : -1; 1563e5b75505Sopenharmony_ci} 1564e5b75505Sopenharmony_ci 1565e5b75505Sopenharmony_ci 1566e5b75505Sopenharmony_ciint crypto_bignum_cmp(const struct crypto_bignum *a, 1567e5b75505Sopenharmony_ci const struct crypto_bignum *b) 1568e5b75505Sopenharmony_ci{ 1569e5b75505Sopenharmony_ci return BN_cmp((const BIGNUM *) a, (const BIGNUM *) b); 1570e5b75505Sopenharmony_ci} 1571e5b75505Sopenharmony_ci 1572e5b75505Sopenharmony_ci 1573e5b75505Sopenharmony_ciint crypto_bignum_is_zero(const struct crypto_bignum *a) 1574e5b75505Sopenharmony_ci{ 1575e5b75505Sopenharmony_ci return BN_is_zero((const BIGNUM *) a); 1576e5b75505Sopenharmony_ci} 1577e5b75505Sopenharmony_ci 1578e5b75505Sopenharmony_ci 1579e5b75505Sopenharmony_ciint crypto_bignum_is_one(const struct crypto_bignum *a) 1580e5b75505Sopenharmony_ci{ 1581e5b75505Sopenharmony_ci return BN_is_one((const BIGNUM *) a); 1582e5b75505Sopenharmony_ci} 1583e5b75505Sopenharmony_ci 1584e5b75505Sopenharmony_ci 1585e5b75505Sopenharmony_ciint crypto_bignum_is_odd(const struct crypto_bignum *a) 1586e5b75505Sopenharmony_ci{ 1587e5b75505Sopenharmony_ci return BN_is_odd((const BIGNUM *) a); 1588e5b75505Sopenharmony_ci} 1589e5b75505Sopenharmony_ci 1590e5b75505Sopenharmony_ci 1591e5b75505Sopenharmony_ciint crypto_bignum_legendre(const struct crypto_bignum *a, 1592e5b75505Sopenharmony_ci const struct crypto_bignum *p) 1593e5b75505Sopenharmony_ci{ 1594e5b75505Sopenharmony_ci BN_CTX *bnctx; 1595e5b75505Sopenharmony_ci BIGNUM *exp = NULL, *tmp = NULL; 1596e5b75505Sopenharmony_ci int res = -2; 1597e5b75505Sopenharmony_ci unsigned int mask; 1598e5b75505Sopenharmony_ci 1599e5b75505Sopenharmony_ci if (TEST_FAIL()) 1600e5b75505Sopenharmony_ci return -2; 1601e5b75505Sopenharmony_ci 1602e5b75505Sopenharmony_ci bnctx = BN_CTX_new(); 1603e5b75505Sopenharmony_ci if (bnctx == NULL) 1604e5b75505Sopenharmony_ci return -2; 1605e5b75505Sopenharmony_ci 1606e5b75505Sopenharmony_ci exp = BN_new(); 1607e5b75505Sopenharmony_ci tmp = BN_new(); 1608e5b75505Sopenharmony_ci if (!exp || !tmp || 1609e5b75505Sopenharmony_ci /* exp = (p-1) / 2 */ 1610e5b75505Sopenharmony_ci !BN_sub(exp, (const BIGNUM *) p, BN_value_one()) || 1611e5b75505Sopenharmony_ci !BN_rshift1(exp, exp) || 1612e5b75505Sopenharmony_ci !BN_mod_exp_mont_consttime(tmp, (const BIGNUM *) a, exp, 1613e5b75505Sopenharmony_ci (const BIGNUM *) p, bnctx, NULL)) 1614e5b75505Sopenharmony_ci goto fail; 1615e5b75505Sopenharmony_ci 1616e5b75505Sopenharmony_ci /* Return 1 if tmp == 1, 0 if tmp == 0, or -1 otherwise. Need to use 1617e5b75505Sopenharmony_ci * constant time selection to avoid branches here. */ 1618e5b75505Sopenharmony_ci res = -1; 1619e5b75505Sopenharmony_ci mask = const_time_eq(BN_is_word(tmp, 1), 1); 1620e5b75505Sopenharmony_ci res = const_time_select_int(mask, 1, res); 1621e5b75505Sopenharmony_ci mask = const_time_eq(BN_is_zero(tmp), 1); 1622e5b75505Sopenharmony_ci res = const_time_select_int(mask, 0, res); 1623e5b75505Sopenharmony_ci 1624e5b75505Sopenharmony_cifail: 1625e5b75505Sopenharmony_ci BN_clear_free(tmp); 1626e5b75505Sopenharmony_ci BN_clear_free(exp); 1627e5b75505Sopenharmony_ci BN_CTX_free(bnctx); 1628e5b75505Sopenharmony_ci return res; 1629e5b75505Sopenharmony_ci} 1630e5b75505Sopenharmony_ci 1631e5b75505Sopenharmony_ci 1632e5b75505Sopenharmony_ci#ifdef CONFIG_ECC 1633e5b75505Sopenharmony_ci 1634e5b75505Sopenharmony_cistruct crypto_ec { 1635e5b75505Sopenharmony_ci EC_GROUP *group; 1636e5b75505Sopenharmony_ci int nid; 1637e5b75505Sopenharmony_ci BN_CTX *bnctx; 1638e5b75505Sopenharmony_ci BIGNUM *prime; 1639e5b75505Sopenharmony_ci BIGNUM *order; 1640e5b75505Sopenharmony_ci BIGNUM *a; 1641e5b75505Sopenharmony_ci BIGNUM *b; 1642e5b75505Sopenharmony_ci}; 1643e5b75505Sopenharmony_ci 1644e5b75505Sopenharmony_cistruct crypto_ec * crypto_ec_init(int group) 1645e5b75505Sopenharmony_ci{ 1646e5b75505Sopenharmony_ci struct crypto_ec *e; 1647e5b75505Sopenharmony_ci int nid; 1648e5b75505Sopenharmony_ci 1649e5b75505Sopenharmony_ci /* Map from IANA registry for IKE D-H groups to OpenSSL NID */ 1650e5b75505Sopenharmony_ci switch (group) { 1651e5b75505Sopenharmony_ci case 19: 1652e5b75505Sopenharmony_ci nid = NID_X9_62_prime256v1; 1653e5b75505Sopenharmony_ci break; 1654e5b75505Sopenharmony_ci case 20: 1655e5b75505Sopenharmony_ci nid = NID_secp384r1; 1656e5b75505Sopenharmony_ci break; 1657e5b75505Sopenharmony_ci case 21: 1658e5b75505Sopenharmony_ci nid = NID_secp521r1; 1659e5b75505Sopenharmony_ci break; 1660e5b75505Sopenharmony_ci case 25: 1661e5b75505Sopenharmony_ci nid = NID_X9_62_prime192v1; 1662e5b75505Sopenharmony_ci break; 1663e5b75505Sopenharmony_ci case 26: 1664e5b75505Sopenharmony_ci nid = NID_secp224r1; 1665e5b75505Sopenharmony_ci break; 1666e5b75505Sopenharmony_ci#ifdef NID_brainpoolP224r1 1667e5b75505Sopenharmony_ci case 27: 1668e5b75505Sopenharmony_ci nid = NID_brainpoolP224r1; 1669e5b75505Sopenharmony_ci break; 1670e5b75505Sopenharmony_ci#endif /* NID_brainpoolP224r1 */ 1671e5b75505Sopenharmony_ci#ifdef NID_brainpoolP256r1 1672e5b75505Sopenharmony_ci case 28: 1673e5b75505Sopenharmony_ci nid = NID_brainpoolP256r1; 1674e5b75505Sopenharmony_ci break; 1675e5b75505Sopenharmony_ci#endif /* NID_brainpoolP256r1 */ 1676e5b75505Sopenharmony_ci#ifdef NID_brainpoolP384r1 1677e5b75505Sopenharmony_ci case 29: 1678e5b75505Sopenharmony_ci nid = NID_brainpoolP384r1; 1679e5b75505Sopenharmony_ci break; 1680e5b75505Sopenharmony_ci#endif /* NID_brainpoolP384r1 */ 1681e5b75505Sopenharmony_ci#ifdef NID_brainpoolP512r1 1682e5b75505Sopenharmony_ci case 30: 1683e5b75505Sopenharmony_ci nid = NID_brainpoolP512r1; 1684e5b75505Sopenharmony_ci break; 1685e5b75505Sopenharmony_ci#endif /* NID_brainpoolP512r1 */ 1686e5b75505Sopenharmony_ci default: 1687e5b75505Sopenharmony_ci return NULL; 1688e5b75505Sopenharmony_ci } 1689e5b75505Sopenharmony_ci 1690e5b75505Sopenharmony_ci e = os_zalloc(sizeof(*e)); 1691e5b75505Sopenharmony_ci if (e == NULL) 1692e5b75505Sopenharmony_ci return NULL; 1693e5b75505Sopenharmony_ci 1694e5b75505Sopenharmony_ci e->nid = nid; 1695e5b75505Sopenharmony_ci e->bnctx = BN_CTX_new(); 1696e5b75505Sopenharmony_ci e->group = EC_GROUP_new_by_curve_name(nid); 1697e5b75505Sopenharmony_ci e->prime = BN_new(); 1698e5b75505Sopenharmony_ci e->order = BN_new(); 1699e5b75505Sopenharmony_ci e->a = BN_new(); 1700e5b75505Sopenharmony_ci e->b = BN_new(); 1701e5b75505Sopenharmony_ci if (e->group == NULL || e->bnctx == NULL || e->prime == NULL || 1702e5b75505Sopenharmony_ci e->order == NULL || e->a == NULL || e->b == NULL || 1703e5b75505Sopenharmony_ci !EC_GROUP_get_curve_GFp(e->group, e->prime, e->a, e->b, e->bnctx) || 1704e5b75505Sopenharmony_ci !EC_GROUP_get_order(e->group, e->order, e->bnctx)) { 1705e5b75505Sopenharmony_ci crypto_ec_deinit(e); 1706e5b75505Sopenharmony_ci e = NULL; 1707e5b75505Sopenharmony_ci } 1708e5b75505Sopenharmony_ci 1709e5b75505Sopenharmony_ci return e; 1710e5b75505Sopenharmony_ci} 1711e5b75505Sopenharmony_ci 1712e5b75505Sopenharmony_ci 1713e5b75505Sopenharmony_civoid crypto_ec_deinit(struct crypto_ec *e) 1714e5b75505Sopenharmony_ci{ 1715e5b75505Sopenharmony_ci if (e == NULL) 1716e5b75505Sopenharmony_ci return; 1717e5b75505Sopenharmony_ci BN_clear_free(e->b); 1718e5b75505Sopenharmony_ci BN_clear_free(e->a); 1719e5b75505Sopenharmony_ci BN_clear_free(e->order); 1720e5b75505Sopenharmony_ci BN_clear_free(e->prime); 1721e5b75505Sopenharmony_ci EC_GROUP_free(e->group); 1722e5b75505Sopenharmony_ci BN_CTX_free(e->bnctx); 1723e5b75505Sopenharmony_ci os_free(e); 1724e5b75505Sopenharmony_ci} 1725e5b75505Sopenharmony_ci 1726e5b75505Sopenharmony_ci 1727e5b75505Sopenharmony_cistruct crypto_ec_point * crypto_ec_point_init(struct crypto_ec *e) 1728e5b75505Sopenharmony_ci{ 1729e5b75505Sopenharmony_ci if (TEST_FAIL()) 1730e5b75505Sopenharmony_ci return NULL; 1731e5b75505Sopenharmony_ci if (e == NULL) 1732e5b75505Sopenharmony_ci return NULL; 1733e5b75505Sopenharmony_ci return (struct crypto_ec_point *) EC_POINT_new(e->group); 1734e5b75505Sopenharmony_ci} 1735e5b75505Sopenharmony_ci 1736e5b75505Sopenharmony_ci 1737e5b75505Sopenharmony_cisize_t crypto_ec_prime_len(struct crypto_ec *e) 1738e5b75505Sopenharmony_ci{ 1739e5b75505Sopenharmony_ci return BN_num_bytes(e->prime); 1740e5b75505Sopenharmony_ci} 1741e5b75505Sopenharmony_ci 1742e5b75505Sopenharmony_ci 1743e5b75505Sopenharmony_cisize_t crypto_ec_prime_len_bits(struct crypto_ec *e) 1744e5b75505Sopenharmony_ci{ 1745e5b75505Sopenharmony_ci return BN_num_bits(e->prime); 1746e5b75505Sopenharmony_ci} 1747e5b75505Sopenharmony_ci 1748e5b75505Sopenharmony_ci 1749e5b75505Sopenharmony_cisize_t crypto_ec_order_len(struct crypto_ec *e) 1750e5b75505Sopenharmony_ci{ 1751e5b75505Sopenharmony_ci return BN_num_bytes(e->order); 1752e5b75505Sopenharmony_ci} 1753e5b75505Sopenharmony_ci 1754e5b75505Sopenharmony_ci 1755e5b75505Sopenharmony_ciconst struct crypto_bignum * crypto_ec_get_prime(struct crypto_ec *e) 1756e5b75505Sopenharmony_ci{ 1757e5b75505Sopenharmony_ci return (const struct crypto_bignum *) e->prime; 1758e5b75505Sopenharmony_ci} 1759e5b75505Sopenharmony_ci 1760e5b75505Sopenharmony_ci 1761e5b75505Sopenharmony_ciconst struct crypto_bignum * crypto_ec_get_order(struct crypto_ec *e) 1762e5b75505Sopenharmony_ci{ 1763e5b75505Sopenharmony_ci return (const struct crypto_bignum *) e->order; 1764e5b75505Sopenharmony_ci} 1765e5b75505Sopenharmony_ci 1766e5b75505Sopenharmony_ci 1767e5b75505Sopenharmony_ciconst struct crypto_bignum * crypto_ec_get_a(struct crypto_ec *e) 1768e5b75505Sopenharmony_ci{ 1769e5b75505Sopenharmony_ci return (const struct crypto_bignum *) e->a; 1770e5b75505Sopenharmony_ci} 1771e5b75505Sopenharmony_ci 1772e5b75505Sopenharmony_ci 1773e5b75505Sopenharmony_ciconst struct crypto_bignum * crypto_ec_get_b(struct crypto_ec *e) 1774e5b75505Sopenharmony_ci{ 1775e5b75505Sopenharmony_ci return (const struct crypto_bignum *) e->b; 1776e5b75505Sopenharmony_ci} 1777e5b75505Sopenharmony_ci 1778e5b75505Sopenharmony_ci 1779e5b75505Sopenharmony_civoid crypto_ec_point_deinit(struct crypto_ec_point *p, int clear) 1780e5b75505Sopenharmony_ci{ 1781e5b75505Sopenharmony_ci if (clear) 1782e5b75505Sopenharmony_ci EC_POINT_clear_free((EC_POINT *) p); 1783e5b75505Sopenharmony_ci else 1784e5b75505Sopenharmony_ci EC_POINT_free((EC_POINT *) p); 1785e5b75505Sopenharmony_ci} 1786e5b75505Sopenharmony_ci 1787e5b75505Sopenharmony_ci 1788e5b75505Sopenharmony_ciint crypto_ec_point_x(struct crypto_ec *e, const struct crypto_ec_point *p, 1789e5b75505Sopenharmony_ci struct crypto_bignum *x) 1790e5b75505Sopenharmony_ci{ 1791e5b75505Sopenharmony_ci return EC_POINT_get_affine_coordinates_GFp(e->group, 1792e5b75505Sopenharmony_ci (const EC_POINT *) p, 1793e5b75505Sopenharmony_ci (BIGNUM *) x, NULL, 1794e5b75505Sopenharmony_ci e->bnctx) == 1 ? 0 : -1; 1795e5b75505Sopenharmony_ci} 1796e5b75505Sopenharmony_ci 1797e5b75505Sopenharmony_ci 1798e5b75505Sopenharmony_ciint crypto_ec_point_to_bin(struct crypto_ec *e, 1799e5b75505Sopenharmony_ci const struct crypto_ec_point *point, u8 *x, u8 *y) 1800e5b75505Sopenharmony_ci{ 1801e5b75505Sopenharmony_ci BIGNUM *x_bn, *y_bn; 1802e5b75505Sopenharmony_ci int ret = -1; 1803e5b75505Sopenharmony_ci int len = BN_num_bytes(e->prime); 1804e5b75505Sopenharmony_ci 1805e5b75505Sopenharmony_ci if (TEST_FAIL()) 1806e5b75505Sopenharmony_ci return -1; 1807e5b75505Sopenharmony_ci 1808e5b75505Sopenharmony_ci x_bn = BN_new(); 1809e5b75505Sopenharmony_ci y_bn = BN_new(); 1810e5b75505Sopenharmony_ci 1811e5b75505Sopenharmony_ci if (x_bn && y_bn && 1812e5b75505Sopenharmony_ci EC_POINT_get_affine_coordinates_GFp(e->group, (EC_POINT *) point, 1813e5b75505Sopenharmony_ci x_bn, y_bn, e->bnctx)) { 1814e5b75505Sopenharmony_ci if (x) { 1815e5b75505Sopenharmony_ci crypto_bignum_to_bin((struct crypto_bignum *) x_bn, 1816e5b75505Sopenharmony_ci x, len, len); 1817e5b75505Sopenharmony_ci } 1818e5b75505Sopenharmony_ci if (y) { 1819e5b75505Sopenharmony_ci crypto_bignum_to_bin((struct crypto_bignum *) y_bn, 1820e5b75505Sopenharmony_ci y, len, len); 1821e5b75505Sopenharmony_ci } 1822e5b75505Sopenharmony_ci ret = 0; 1823e5b75505Sopenharmony_ci } 1824e5b75505Sopenharmony_ci 1825e5b75505Sopenharmony_ci BN_clear_free(x_bn); 1826e5b75505Sopenharmony_ci BN_clear_free(y_bn); 1827e5b75505Sopenharmony_ci return ret; 1828e5b75505Sopenharmony_ci} 1829e5b75505Sopenharmony_ci 1830e5b75505Sopenharmony_ci 1831e5b75505Sopenharmony_cistruct crypto_ec_point * crypto_ec_point_from_bin(struct crypto_ec *e, 1832e5b75505Sopenharmony_ci const u8 *val) 1833e5b75505Sopenharmony_ci{ 1834e5b75505Sopenharmony_ci BIGNUM *x, *y; 1835e5b75505Sopenharmony_ci EC_POINT *elem; 1836e5b75505Sopenharmony_ci int len = BN_num_bytes(e->prime); 1837e5b75505Sopenharmony_ci 1838e5b75505Sopenharmony_ci if (TEST_FAIL()) 1839e5b75505Sopenharmony_ci return NULL; 1840e5b75505Sopenharmony_ci 1841e5b75505Sopenharmony_ci x = BN_bin2bn(val, len, NULL); 1842e5b75505Sopenharmony_ci y = BN_bin2bn(val + len, len, NULL); 1843e5b75505Sopenharmony_ci elem = EC_POINT_new(e->group); 1844e5b75505Sopenharmony_ci if (x == NULL || y == NULL || elem == NULL) { 1845e5b75505Sopenharmony_ci BN_clear_free(x); 1846e5b75505Sopenharmony_ci BN_clear_free(y); 1847e5b75505Sopenharmony_ci EC_POINT_clear_free(elem); 1848e5b75505Sopenharmony_ci return NULL; 1849e5b75505Sopenharmony_ci } 1850e5b75505Sopenharmony_ci 1851e5b75505Sopenharmony_ci if (!EC_POINT_set_affine_coordinates_GFp(e->group, elem, x, y, 1852e5b75505Sopenharmony_ci e->bnctx)) { 1853e5b75505Sopenharmony_ci EC_POINT_clear_free(elem); 1854e5b75505Sopenharmony_ci elem = NULL; 1855e5b75505Sopenharmony_ci } 1856e5b75505Sopenharmony_ci 1857e5b75505Sopenharmony_ci BN_clear_free(x); 1858e5b75505Sopenharmony_ci BN_clear_free(y); 1859e5b75505Sopenharmony_ci 1860e5b75505Sopenharmony_ci return (struct crypto_ec_point *) elem; 1861e5b75505Sopenharmony_ci} 1862e5b75505Sopenharmony_ci 1863e5b75505Sopenharmony_ci 1864e5b75505Sopenharmony_ciint crypto_ec_point_add(struct crypto_ec *e, const struct crypto_ec_point *a, 1865e5b75505Sopenharmony_ci const struct crypto_ec_point *b, 1866e5b75505Sopenharmony_ci struct crypto_ec_point *c) 1867e5b75505Sopenharmony_ci{ 1868e5b75505Sopenharmony_ci if (TEST_FAIL()) 1869e5b75505Sopenharmony_ci return -1; 1870e5b75505Sopenharmony_ci return EC_POINT_add(e->group, (EC_POINT *) c, (const EC_POINT *) a, 1871e5b75505Sopenharmony_ci (const EC_POINT *) b, e->bnctx) ? 0 : -1; 1872e5b75505Sopenharmony_ci} 1873e5b75505Sopenharmony_ci 1874e5b75505Sopenharmony_ci 1875e5b75505Sopenharmony_ciint crypto_ec_point_mul(struct crypto_ec *e, const struct crypto_ec_point *p, 1876e5b75505Sopenharmony_ci const struct crypto_bignum *b, 1877e5b75505Sopenharmony_ci struct crypto_ec_point *res) 1878e5b75505Sopenharmony_ci{ 1879e5b75505Sopenharmony_ci if (TEST_FAIL()) 1880e5b75505Sopenharmony_ci return -1; 1881e5b75505Sopenharmony_ci return EC_POINT_mul(e->group, (EC_POINT *) res, NULL, 1882e5b75505Sopenharmony_ci (const EC_POINT *) p, (const BIGNUM *) b, e->bnctx) 1883e5b75505Sopenharmony_ci ? 0 : -1; 1884e5b75505Sopenharmony_ci} 1885e5b75505Sopenharmony_ci 1886e5b75505Sopenharmony_ci 1887e5b75505Sopenharmony_ciint crypto_ec_point_invert(struct crypto_ec *e, struct crypto_ec_point *p) 1888e5b75505Sopenharmony_ci{ 1889e5b75505Sopenharmony_ci if (TEST_FAIL()) 1890e5b75505Sopenharmony_ci return -1; 1891e5b75505Sopenharmony_ci return EC_POINT_invert(e->group, (EC_POINT *) p, e->bnctx) ? 0 : -1; 1892e5b75505Sopenharmony_ci} 1893e5b75505Sopenharmony_ci 1894e5b75505Sopenharmony_ci 1895e5b75505Sopenharmony_ciint crypto_ec_point_solve_y_coord(struct crypto_ec *e, 1896e5b75505Sopenharmony_ci struct crypto_ec_point *p, 1897e5b75505Sopenharmony_ci const struct crypto_bignum *x, int y_bit) 1898e5b75505Sopenharmony_ci{ 1899e5b75505Sopenharmony_ci if (TEST_FAIL()) 1900e5b75505Sopenharmony_ci return -1; 1901e5b75505Sopenharmony_ci if (!EC_POINT_set_compressed_coordinates_GFp(e->group, (EC_POINT *) p, 1902e5b75505Sopenharmony_ci (const BIGNUM *) x, y_bit, 1903e5b75505Sopenharmony_ci e->bnctx) || 1904e5b75505Sopenharmony_ci !EC_POINT_is_on_curve(e->group, (EC_POINT *) p, e->bnctx)) 1905e5b75505Sopenharmony_ci return -1; 1906e5b75505Sopenharmony_ci return 0; 1907e5b75505Sopenharmony_ci} 1908e5b75505Sopenharmony_ci 1909e5b75505Sopenharmony_ci 1910e5b75505Sopenharmony_cistruct crypto_bignum * 1911e5b75505Sopenharmony_cicrypto_ec_point_compute_y_sqr(struct crypto_ec *e, 1912e5b75505Sopenharmony_ci const struct crypto_bignum *x) 1913e5b75505Sopenharmony_ci{ 1914e5b75505Sopenharmony_ci BIGNUM *tmp, *tmp2, *y_sqr = NULL; 1915e5b75505Sopenharmony_ci 1916e5b75505Sopenharmony_ci if (TEST_FAIL()) 1917e5b75505Sopenharmony_ci return NULL; 1918e5b75505Sopenharmony_ci 1919e5b75505Sopenharmony_ci tmp = BN_new(); 1920e5b75505Sopenharmony_ci tmp2 = BN_new(); 1921e5b75505Sopenharmony_ci 1922e5b75505Sopenharmony_ci /* y^2 = x^3 + ax + b */ 1923e5b75505Sopenharmony_ci if (tmp && tmp2 && 1924e5b75505Sopenharmony_ci BN_mod_sqr(tmp, (const BIGNUM *) x, e->prime, e->bnctx) && 1925e5b75505Sopenharmony_ci BN_mod_mul(tmp, tmp, (const BIGNUM *) x, e->prime, e->bnctx) && 1926e5b75505Sopenharmony_ci BN_mod_mul(tmp2, e->a, (const BIGNUM *) x, e->prime, e->bnctx) && 1927e5b75505Sopenharmony_ci BN_mod_add_quick(tmp2, tmp2, tmp, e->prime) && 1928e5b75505Sopenharmony_ci BN_mod_add_quick(tmp2, tmp2, e->b, e->prime)) { 1929e5b75505Sopenharmony_ci y_sqr = tmp2; 1930e5b75505Sopenharmony_ci tmp2 = NULL; 1931e5b75505Sopenharmony_ci } 1932e5b75505Sopenharmony_ci 1933e5b75505Sopenharmony_ci BN_clear_free(tmp); 1934e5b75505Sopenharmony_ci BN_clear_free(tmp2); 1935e5b75505Sopenharmony_ci 1936e5b75505Sopenharmony_ci return (struct crypto_bignum *) y_sqr; 1937e5b75505Sopenharmony_ci} 1938e5b75505Sopenharmony_ci 1939e5b75505Sopenharmony_ci 1940e5b75505Sopenharmony_ciint crypto_ec_point_is_at_infinity(struct crypto_ec *e, 1941e5b75505Sopenharmony_ci const struct crypto_ec_point *p) 1942e5b75505Sopenharmony_ci{ 1943e5b75505Sopenharmony_ci return EC_POINT_is_at_infinity(e->group, (const EC_POINT *) p); 1944e5b75505Sopenharmony_ci} 1945e5b75505Sopenharmony_ci 1946e5b75505Sopenharmony_ci 1947e5b75505Sopenharmony_ciint crypto_ec_point_is_on_curve(struct crypto_ec *e, 1948e5b75505Sopenharmony_ci const struct crypto_ec_point *p) 1949e5b75505Sopenharmony_ci{ 1950e5b75505Sopenharmony_ci return EC_POINT_is_on_curve(e->group, (const EC_POINT *) p, 1951e5b75505Sopenharmony_ci e->bnctx) == 1; 1952e5b75505Sopenharmony_ci} 1953e5b75505Sopenharmony_ci 1954e5b75505Sopenharmony_ci 1955e5b75505Sopenharmony_ciint crypto_ec_point_cmp(const struct crypto_ec *e, 1956e5b75505Sopenharmony_ci const struct crypto_ec_point *a, 1957e5b75505Sopenharmony_ci const struct crypto_ec_point *b) 1958e5b75505Sopenharmony_ci{ 1959e5b75505Sopenharmony_ci return EC_POINT_cmp(e->group, (const EC_POINT *) a, 1960e5b75505Sopenharmony_ci (const EC_POINT *) b, e->bnctx); 1961e5b75505Sopenharmony_ci} 1962e5b75505Sopenharmony_ci 1963e5b75505Sopenharmony_ci 1964e5b75505Sopenharmony_cistruct crypto_ecdh { 1965e5b75505Sopenharmony_ci struct crypto_ec *ec; 1966e5b75505Sopenharmony_ci EVP_PKEY *pkey; 1967e5b75505Sopenharmony_ci}; 1968e5b75505Sopenharmony_ci 1969e5b75505Sopenharmony_cistruct crypto_ecdh * crypto_ecdh_init(int group) 1970e5b75505Sopenharmony_ci{ 1971e5b75505Sopenharmony_ci struct crypto_ecdh *ecdh; 1972e5b75505Sopenharmony_ci EVP_PKEY *params = NULL; 1973e5b75505Sopenharmony_ci EC_KEY *ec_params = NULL; 1974e5b75505Sopenharmony_ci EVP_PKEY_CTX *kctx = NULL; 1975e5b75505Sopenharmony_ci 1976e5b75505Sopenharmony_ci ecdh = os_zalloc(sizeof(*ecdh)); 1977e5b75505Sopenharmony_ci if (!ecdh) 1978e5b75505Sopenharmony_ci goto fail; 1979e5b75505Sopenharmony_ci 1980e5b75505Sopenharmony_ci ecdh->ec = crypto_ec_init(group); 1981e5b75505Sopenharmony_ci if (!ecdh->ec) 1982e5b75505Sopenharmony_ci goto fail; 1983e5b75505Sopenharmony_ci 1984e5b75505Sopenharmony_ci ec_params = EC_KEY_new_by_curve_name(ecdh->ec->nid); 1985e5b75505Sopenharmony_ci if (!ec_params) { 1986e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 1987e5b75505Sopenharmony_ci "OpenSSL: Failed to generate EC_KEY parameters"); 1988e5b75505Sopenharmony_ci goto fail; 1989e5b75505Sopenharmony_ci } 1990e5b75505Sopenharmony_ci EC_KEY_set_asn1_flag(ec_params, OPENSSL_EC_NAMED_CURVE); 1991e5b75505Sopenharmony_ci params = EVP_PKEY_new(); 1992e5b75505Sopenharmony_ci if (!params || EVP_PKEY_set1_EC_KEY(params, ec_params) != 1) { 1993e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 1994e5b75505Sopenharmony_ci "OpenSSL: Failed to generate EVP_PKEY parameters"); 1995e5b75505Sopenharmony_ci goto fail; 1996e5b75505Sopenharmony_ci } 1997e5b75505Sopenharmony_ci 1998e5b75505Sopenharmony_ci kctx = EVP_PKEY_CTX_new(params, NULL); 1999e5b75505Sopenharmony_ci if (!kctx) 2000e5b75505Sopenharmony_ci goto fail; 2001e5b75505Sopenharmony_ci 2002e5b75505Sopenharmony_ci if (EVP_PKEY_keygen_init(kctx) != 1) { 2003e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2004e5b75505Sopenharmony_ci "OpenSSL: EVP_PKEY_keygen_init failed: %s", 2005e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2006e5b75505Sopenharmony_ci goto fail; 2007e5b75505Sopenharmony_ci } 2008e5b75505Sopenharmony_ci 2009e5b75505Sopenharmony_ci if (EVP_PKEY_keygen(kctx, &ecdh->pkey) != 1) { 2010e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, "OpenSSL: EVP_PKEY_keygen failed: %s", 2011e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2012e5b75505Sopenharmony_ci goto fail; 2013e5b75505Sopenharmony_ci } 2014e5b75505Sopenharmony_ci 2015e5b75505Sopenharmony_cidone: 2016e5b75505Sopenharmony_ci EC_KEY_free(ec_params); 2017e5b75505Sopenharmony_ci EVP_PKEY_free(params); 2018e5b75505Sopenharmony_ci EVP_PKEY_CTX_free(kctx); 2019e5b75505Sopenharmony_ci 2020e5b75505Sopenharmony_ci return ecdh; 2021e5b75505Sopenharmony_cifail: 2022e5b75505Sopenharmony_ci crypto_ecdh_deinit(ecdh); 2023e5b75505Sopenharmony_ci ecdh = NULL; 2024e5b75505Sopenharmony_ci goto done; 2025e5b75505Sopenharmony_ci} 2026e5b75505Sopenharmony_ci 2027e5b75505Sopenharmony_ci 2028e5b75505Sopenharmony_cistruct wpabuf * crypto_ecdh_get_pubkey(struct crypto_ecdh *ecdh, int inc_y) 2029e5b75505Sopenharmony_ci{ 2030e5b75505Sopenharmony_ci struct wpabuf *buf = NULL; 2031e5b75505Sopenharmony_ci EC_KEY *eckey; 2032e5b75505Sopenharmony_ci const EC_POINT *pubkey; 2033e5b75505Sopenharmony_ci BIGNUM *x, *y = NULL; 2034e5b75505Sopenharmony_ci int len = BN_num_bytes(ecdh->ec->prime); 2035e5b75505Sopenharmony_ci int res; 2036e5b75505Sopenharmony_ci 2037e5b75505Sopenharmony_ci eckey = EVP_PKEY_get1_EC_KEY(ecdh->pkey); 2038e5b75505Sopenharmony_ci if (!eckey) 2039e5b75505Sopenharmony_ci return NULL; 2040e5b75505Sopenharmony_ci 2041e5b75505Sopenharmony_ci pubkey = EC_KEY_get0_public_key(eckey); 2042e5b75505Sopenharmony_ci if (!pubkey) 2043e5b75505Sopenharmony_ci return NULL; 2044e5b75505Sopenharmony_ci 2045e5b75505Sopenharmony_ci x = BN_new(); 2046e5b75505Sopenharmony_ci if (inc_y) { 2047e5b75505Sopenharmony_ci y = BN_new(); 2048e5b75505Sopenharmony_ci if (!y) 2049e5b75505Sopenharmony_ci goto fail; 2050e5b75505Sopenharmony_ci } 2051e5b75505Sopenharmony_ci buf = wpabuf_alloc(inc_y ? 2 * len : len); 2052e5b75505Sopenharmony_ci if (!x || !buf) 2053e5b75505Sopenharmony_ci goto fail; 2054e5b75505Sopenharmony_ci 2055e5b75505Sopenharmony_ci if (EC_POINT_get_affine_coordinates_GFp(ecdh->ec->group, pubkey, 2056e5b75505Sopenharmony_ci x, y, ecdh->ec->bnctx) != 1) { 2057e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2058e5b75505Sopenharmony_ci "OpenSSL: EC_POINT_get_affine_coordinates_GFp failed: %s", 2059e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2060e5b75505Sopenharmony_ci goto fail; 2061e5b75505Sopenharmony_ci } 2062e5b75505Sopenharmony_ci 2063e5b75505Sopenharmony_ci res = crypto_bignum_to_bin((struct crypto_bignum *) x, 2064e5b75505Sopenharmony_ci wpabuf_put(buf, len), len, len); 2065e5b75505Sopenharmony_ci if (res < 0) 2066e5b75505Sopenharmony_ci goto fail; 2067e5b75505Sopenharmony_ci 2068e5b75505Sopenharmony_ci if (inc_y) { 2069e5b75505Sopenharmony_ci res = crypto_bignum_to_bin((struct crypto_bignum *) y, 2070e5b75505Sopenharmony_ci wpabuf_put(buf, len), len, len); 2071e5b75505Sopenharmony_ci if (res < 0) 2072e5b75505Sopenharmony_ci goto fail; 2073e5b75505Sopenharmony_ci } 2074e5b75505Sopenharmony_ci 2075e5b75505Sopenharmony_cidone: 2076e5b75505Sopenharmony_ci BN_clear_free(x); 2077e5b75505Sopenharmony_ci BN_clear_free(y); 2078e5b75505Sopenharmony_ci EC_KEY_free(eckey); 2079e5b75505Sopenharmony_ci 2080e5b75505Sopenharmony_ci return buf; 2081e5b75505Sopenharmony_cifail: 2082e5b75505Sopenharmony_ci wpabuf_free(buf); 2083e5b75505Sopenharmony_ci buf = NULL; 2084e5b75505Sopenharmony_ci goto done; 2085e5b75505Sopenharmony_ci} 2086e5b75505Sopenharmony_ci 2087e5b75505Sopenharmony_ci 2088e5b75505Sopenharmony_cistruct wpabuf * crypto_ecdh_set_peerkey(struct crypto_ecdh *ecdh, int inc_y, 2089e5b75505Sopenharmony_ci const u8 *key, size_t len) 2090e5b75505Sopenharmony_ci{ 2091e5b75505Sopenharmony_ci BIGNUM *x, *y = NULL; 2092e5b75505Sopenharmony_ci EVP_PKEY_CTX *ctx = NULL; 2093e5b75505Sopenharmony_ci EVP_PKEY *peerkey = NULL; 2094e5b75505Sopenharmony_ci struct wpabuf *secret = NULL; 2095e5b75505Sopenharmony_ci size_t secret_len; 2096e5b75505Sopenharmony_ci EC_POINT *pub; 2097e5b75505Sopenharmony_ci EC_KEY *eckey = NULL; 2098e5b75505Sopenharmony_ci 2099e5b75505Sopenharmony_ci x = BN_bin2bn(key, inc_y ? len / 2 : len, NULL); 2100e5b75505Sopenharmony_ci pub = EC_POINT_new(ecdh->ec->group); 2101e5b75505Sopenharmony_ci if (!x || !pub) 2102e5b75505Sopenharmony_ci goto fail; 2103e5b75505Sopenharmony_ci 2104e5b75505Sopenharmony_ci if (inc_y) { 2105e5b75505Sopenharmony_ci y = BN_bin2bn(key + len / 2, len / 2, NULL); 2106e5b75505Sopenharmony_ci if (!y) 2107e5b75505Sopenharmony_ci goto fail; 2108e5b75505Sopenharmony_ci if (!EC_POINT_set_affine_coordinates_GFp(ecdh->ec->group, pub, 2109e5b75505Sopenharmony_ci x, y, 2110e5b75505Sopenharmony_ci ecdh->ec->bnctx)) { 2111e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2112e5b75505Sopenharmony_ci "OpenSSL: EC_POINT_set_affine_coordinates_GFp failed: %s", 2113e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2114e5b75505Sopenharmony_ci goto fail; 2115e5b75505Sopenharmony_ci } 2116e5b75505Sopenharmony_ci } else if (!EC_POINT_set_compressed_coordinates_GFp(ecdh->ec->group, 2117e5b75505Sopenharmony_ci pub, x, 0, 2118e5b75505Sopenharmony_ci ecdh->ec->bnctx)) { 2119e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2120e5b75505Sopenharmony_ci "OpenSSL: EC_POINT_set_compressed_coordinates_GFp failed: %s", 2121e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2122e5b75505Sopenharmony_ci goto fail; 2123e5b75505Sopenharmony_ci } 2124e5b75505Sopenharmony_ci 2125e5b75505Sopenharmony_ci if (!EC_POINT_is_on_curve(ecdh->ec->group, pub, ecdh->ec->bnctx)) { 2126e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2127e5b75505Sopenharmony_ci "OpenSSL: ECDH peer public key is not on curve"); 2128e5b75505Sopenharmony_ci goto fail; 2129e5b75505Sopenharmony_ci } 2130e5b75505Sopenharmony_ci 2131e5b75505Sopenharmony_ci eckey = EC_KEY_new_by_curve_name(ecdh->ec->nid); 2132e5b75505Sopenharmony_ci if (!eckey || EC_KEY_set_public_key(eckey, pub) != 1) { 2133e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2134e5b75505Sopenharmony_ci "OpenSSL: EC_KEY_set_public_key failed: %s", 2135e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2136e5b75505Sopenharmony_ci goto fail; 2137e5b75505Sopenharmony_ci } 2138e5b75505Sopenharmony_ci 2139e5b75505Sopenharmony_ci peerkey = EVP_PKEY_new(); 2140e5b75505Sopenharmony_ci if (!peerkey || EVP_PKEY_set1_EC_KEY(peerkey, eckey) != 1) 2141e5b75505Sopenharmony_ci goto fail; 2142e5b75505Sopenharmony_ci 2143e5b75505Sopenharmony_ci ctx = EVP_PKEY_CTX_new(ecdh->pkey, NULL); 2144e5b75505Sopenharmony_ci if (!ctx || EVP_PKEY_derive_init(ctx) != 1 || 2145e5b75505Sopenharmony_ci EVP_PKEY_derive_set_peer(ctx, peerkey) != 1 || 2146e5b75505Sopenharmony_ci EVP_PKEY_derive(ctx, NULL, &secret_len) != 1) { 2147e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2148e5b75505Sopenharmony_ci "OpenSSL: EVP_PKEY_derive(1) failed: %s", 2149e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2150e5b75505Sopenharmony_ci goto fail; 2151e5b75505Sopenharmony_ci } 2152e5b75505Sopenharmony_ci 2153e5b75505Sopenharmony_ci secret = wpabuf_alloc(secret_len); 2154e5b75505Sopenharmony_ci if (!secret) 2155e5b75505Sopenharmony_ci goto fail; 2156e5b75505Sopenharmony_ci if (EVP_PKEY_derive(ctx, wpabuf_put(secret, 0), &secret_len) != 1) { 2157e5b75505Sopenharmony_ci wpa_printf(MSG_ERROR, 2158e5b75505Sopenharmony_ci "OpenSSL: EVP_PKEY_derive(2) failed: %s", 2159e5b75505Sopenharmony_ci ERR_error_string(ERR_get_error(), NULL)); 2160e5b75505Sopenharmony_ci goto fail; 2161e5b75505Sopenharmony_ci } 2162e5b75505Sopenharmony_ci if (secret->size != secret_len) 2163e5b75505Sopenharmony_ci wpa_printf(MSG_DEBUG, 2164e5b75505Sopenharmony_ci "OpenSSL: EVP_PKEY_derive(2) changed secret_len %d -> %d", 2165e5b75505Sopenharmony_ci (int) secret->size, (int) secret_len); 2166e5b75505Sopenharmony_ci wpabuf_put(secret, secret_len); 2167e5b75505Sopenharmony_ci 2168e5b75505Sopenharmony_cidone: 2169e5b75505Sopenharmony_ci BN_free(x); 2170e5b75505Sopenharmony_ci BN_free(y); 2171e5b75505Sopenharmony_ci EC_KEY_free(eckey); 2172e5b75505Sopenharmony_ci EC_POINT_free(pub); 2173e5b75505Sopenharmony_ci EVP_PKEY_CTX_free(ctx); 2174e5b75505Sopenharmony_ci EVP_PKEY_free(peerkey); 2175e5b75505Sopenharmony_ci return secret; 2176e5b75505Sopenharmony_cifail: 2177e5b75505Sopenharmony_ci wpabuf_free(secret); 2178e5b75505Sopenharmony_ci secret = NULL; 2179e5b75505Sopenharmony_ci goto done; 2180e5b75505Sopenharmony_ci} 2181e5b75505Sopenharmony_ci 2182e5b75505Sopenharmony_ci 2183e5b75505Sopenharmony_civoid crypto_ecdh_deinit(struct crypto_ecdh *ecdh) 2184e5b75505Sopenharmony_ci{ 2185e5b75505Sopenharmony_ci if (ecdh) { 2186e5b75505Sopenharmony_ci crypto_ec_deinit(ecdh->ec); 2187e5b75505Sopenharmony_ci EVP_PKEY_free(ecdh->pkey); 2188e5b75505Sopenharmony_ci os_free(ecdh); 2189e5b75505Sopenharmony_ci } 2190e5b75505Sopenharmony_ci} 2191e5b75505Sopenharmony_ci 2192e5b75505Sopenharmony_ci#endif /* CONFIG_ECC */ 2193