์ด ์์ค๋ GitHub ์์ ๊ฐ์ ธ์์.
https://github.com/insanum/ecies
/*
* Home: https://github.com/insanum/ecies
* Author: Eric Davis <edavis@insanum.com>
*/
#include <stdio.h>
#include <string.h>
#include <openssl/bn.h>
#include <openssl/ec.h>
#include <openssl/ecdh.h>
#include <openssl/pem.h>
#include <openssl/rand.h>
#include <openssl/evp.h>
#define err(fmt, ...) \
do { \
printf("ERROR:" fmt, ##__VA_ARGS__); \
exit(1); \
} while (0)
#define log(fmt, ...) \
do { \
printf(fmt, ##__VA_ARGS__); \
} while (0)
/*
* This function takes a buffer with binary data and dumps
* out a hex string prefixed with a label.
*/
void dump_hex(char *label, uint8_t *buf, int len)
{
int i;
log("%-10s: ", label);
for (i = 0; i < len; ++i) { log("%02x", buf[i]); }
log("\n");
}
/* Convert an EC key`s public key to a binary array. */
int ec_key_public_key_to_bin(const EC_KEY *ec_key,
uint8_t **pubk, // out (must free)
size_t *pubk_len) // out
{
const EC_GROUP *ec_group = EC_KEY_get0_group(ec_key);
const EC_POINT *pub = EC_KEY_get0_public_key(ec_key);
BIGNUM *pub_bn = BN_new();
BN_CTX *pub_bn_ctx = BN_CTX_new();
BN_CTX_start(pub_bn_ctx);
EC_POINT_point2bn(ec_group, pub, POINT_CONVERSION_UNCOMPRESSED,
pub_bn, pub_bn_ctx);
*pubk_len = BN_num_bytes(pub_bn);
*pubk = OPENSSL_malloc(*pubk_len);
if (BN_bn2bin(pub_bn, *pubk) != *pubk_len)
err("Failed to decode pubkey\n");
BN_CTX_end(pub_bn_ctx);
BN_CTX_free(pub_bn_ctx);
BN_clear_free(pub_bn);
return 0;
}
/* Convert an EC key`s private key to a binary array. */
int ec_key_private_key_to_bin(const EC_KEY *ec_key,
uint8_t **privk, // out (must free)
size_t *privk_len) // out
{
const BIGNUM *priv = EC_KEY_get0_private_key(ec_key);
*privk_len = BN_num_bytes(priv);
*privk = OPENSSL_malloc(*privk_len);
if (BN_bn2bin(priv, *privk) != *privk_len)
err("Failed to decode privkey\n");
return 0;
}
/* Convert a public key binary array to an EC point. */
int ec_key_public_key_bin_to_point(const EC_GROUP *ec_group,
const uint8_t *pubk,
const size_t pubk_len,
EC_POINT **pubk_point) // out
{
BIGNUM *pubk_bn;
BN_CTX *pubk_bn_ctx;
*pubk_point = EC_POINT_new(ec_group);
pubk_bn = BN_bin2bn(pubk, pubk_len, NULL);
pubk_bn_ctx = BN_CTX_new();
BN_CTX_start(pubk_bn_ctx);
EC_POINT_bn2point(ec_group, pubk_bn, *pubk_point, pubk_bn_ctx);
BN_CTX_end(pubk_bn_ctx);
BN_CTX_free(pubk_bn_ctx);
BN_clear_free(pubk_bn);
return 0;
}
/* (TX) Generate an ephemeral EC key and associated shared symmetric key. */
int ecies_transmitter_generate_symkey(const int curve,
const uint8_t *peer_pubk,
const size_t peer_pubk_len,
uint8_t **epubk, // out (must free)
size_t *epubk_len, // out
uint8_t **skey, // out (must free)
size_t *skey_len) // out
{
EC_KEY *ec_key = NULL; /* ephemeral keypair */
const EC_GROUP *ec_group = NULL;
EC_POINT *peer_pubk_point = NULL;
/* Create and initialize a new empty key pair on the curve. */
ec_key = EC_KEY_new_by_curve_name(curve);
EC_KEY_generate_key(ec_key);
ec_group = EC_KEY_get0_group(ec_key);
/* Allocate a buffer to hold the shared symmetric key. */
*skey_len = ((EC_GROUP_get_degree(ec_group) + 7) / 8);
*skey = OPENSSL_malloc(*skey_len);
/* Convert the peer public key to an EC point. */
ec_key_public_key_bin_to_point(ec_group, peer_pubk, peer_pubk_len,
&peer_pubk_point);
/* Generate the shared symmetric key (diffie-hellman primitive). */
*skey_len = ECDH_compute_key(*skey, *skey_len, peer_pubk_point,
ec_key, NULL);
/* Write the ephemeral key`s public key to the output buffer. */
ec_key_public_key_to_bin(ec_key, epubk, epubk_len);
/*
* NOTE: The private key is thrown away here...
* With ECIES the transmitter EC key pair is a one time use only.
*/
return 0;
}
/* (RX) Generate the shared symmetric key. */
int ecies_receiver_generate_symkey(const EC_KEY *ec_key,
const uint8_t *peer_pubk,
const size_t peer_pubk_len,
uint8_t **skey, // out (must free)
size_t *skey_len) // out
{
const EC_GROUP *ec_group = EC_KEY_get0_group(ec_key);
EC_POINT *peer_pubk_point = NULL;
/* Allocate a buffer to hold the shared symmetric key. */
*skey_len = ((EC_GROUP_get_degree(ec_group) + 7) / 8);
*skey = OPENSSL_malloc(*skey_len);
/* Convert the peer public key to an EC point. */
ec_key_public_key_bin_to_point(ec_group, peer_pubk, peer_pubk_len,
&peer_pubk_point);
/* Generate the shared symmetric key (diffie-hellman primitive). */
*skey_len = ECDH_compute_key(*skey, *skey_len, peer_pubk_point,
(EC_KEY *)ec_key, NULL);
return 0;
}
/* Encrypt plaintext data using 256b AES-GCM. */
int aes_gcm_256b_encrypt(uint8_t *plaintext,
size_t plaintext_len,
uint8_t *skey,
uint8_t *aad,
size_t aad_len,
uint8_t **iv, // out (must free)
uint8_t *iv_len, // out
uint8_t **tag, // out (must free)
uint8_t *tag_len, // out
uint8_t **ciphertext, // out (must free)
uint8_t *ciphertext_len) // out
{
EVP_CIPHER_CTX *ctx;
int len;
/* Allocate buffers for the IV, tag, and ciphertext. */
*iv_len = 12;
*iv = OPENSSL_malloc(*iv_len);
*tag_len = 12;
*tag = OPENSSL_malloc(*tag_len);
*ciphertext = OPENSSL_malloc((plaintext_len + 0xf) & ~0xf);
/* Initialize the context and encryption operation. */
ctx = EVP_CIPHER_CTX_new();
EVP_EncryptInit_ex(ctx, EVP_aes_256_gcm(), NULL, NULL, NULL);
/* Generate a new random IV. */
RAND_pseudo_bytes(*iv, *iv_len);
/* Prime the key and IV. */
EVP_EncryptInit_ex(ctx, NULL, NULL, skey, *iv);
/* Prime with any additional authentication data. */
if (aad && aad_len)
EVP_EncryptUpdate(ctx, NULL, &len, aad, aad_len);
/* Encrypt the data. */
EVP_EncryptUpdate(ctx, *ciphertext, &len, plaintext, plaintext_len);
*ciphertext_len = len;
/* Finalize the encryption session. */
EVP_EncryptFinal_ex(ctx, (*ciphertext + len), &len);
*ciphertext_len += len;
/* Get the authentication tag. */
EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_GET_TAG, *tag_len, *tag);
EVP_CIPHER_CTX_free(ctx);
return 0;
}
/* Decrypt ciphertext data using 256b AES-GCM. */
int aes_gcm_256b_decrypt(uint8_t *ciphertext,
size_t ciphertext_len,
uint8_t *skey,
uint8_t *aad,
size_t aad_len,
uint8_t *iv,
uint8_t iv_len,
uint8_t *tag,
size_t tag_len,
uint8_t **plaintext, // out (must free)
uint8_t *plaintext_len) // out
{
EVP_CIPHER_CTX *ctx;
int len, rc;
/* Allocate a buffer for the plaintext. */
*plaintext = OPENSSL_malloc(ciphertext_len);
/* Initialize the context and encryption operation. */
ctx = EVP_CIPHER_CTX_new();
EVP_EncryptInit_ex(ctx, EVP_aes_256_gcm(), NULL, NULL, NULL);
/* Prime the key and IV (+length). */
EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_SET_IVLEN, iv_len, NULL);
EVP_DecryptInit_ex(ctx, NULL, NULL, skey, iv);
/* Prime with any additional authentication data. */
if (aad && aad_len)
EVP_DecryptUpdate(ctx, NULL, &len, aad, aad_len);
/* Decrypt the data. */
EVP_DecryptUpdate(ctx, *plaintext, &len, ciphertext, ciphertext_len);
*plaintext_len = len;
/* Set the expected tag value. */
EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_SET_TAG, tag_len, tag);
/* Finalize the decryption session. Returns 0 with a bad tag! */
rc = EVP_DecryptFinal_ex(ctx, (*plaintext + len), &len);
EVP_CIPHER_CTX_free(ctx);
if (rc > 0)
{
*plaintext_len += len;
return 0;
}
/* verification failed */
err("Decryption verification failed\n");
}
int ecies_receiver_load_key(char *filename,
EC_KEY **ec_key, // out
int *curve, // out
uint8_t **pubk, // out (must free)
size_t *pubk_len, // out
uint8_t **privk, // out (must free)
size_t *privk_len) // out
{
const EC_GROUP *ec_group = NULL;
BIO *bio_key = NULL;
BIO *bio_out = NULL; /* stdout */
/*
* Create a BIO object wrapping a file pointer to read the EC key file
* in DER format. Then read in and parse the EC key from the file.
*/
bio_key = BIO_new_file(filename, "r");
if (bio_key == NULL)
err("Failed to read EC key file `%s`\n", filename);
*ec_key = d2i_ECPrivateKey_bio(bio_key, NULL);
if (*ec_key == NULL)
err("Failed to parse EC key file `%s`\n", filename);
BIO_free(bio_key);
/* Get the curve parameters from the EC key. */
ec_group = EC_KEY_get0_group(*ec_key);
/* Create a BIO object wrapping stdout. */
bio_out = BIO_new_fp(stdout, BIO_NOCLOSE);
/* Set the point conversion outputs to always be `uncompressed`. */
EC_KEY_set_conv_form(*ec_key, POINT_CONVERSION_UNCOMPRESSED);
/* Dump the EC key (public/private). */
//EC_KEY_print(bio_out, *ec_key, 0);
/* Dump the EC key`s curve parameters (either by name or explicit). */
//EC_KEY_set_asn1_flag(*ec_key, OPENSSL_EC_NAMED_CURVE);
//EC_KEY_set_asn1_flag(*ec_key, 0 /* OPENSSL_EC_EXPLICIT_CURVE */);
//ECPKParameters_print(bio_out, ec_group, 0);
/* Get the EC key`s public key in a binary array format. */
ec_key_public_key_to_bin(*ec_key, pubk, pubk_len);
/* Get the EC key`s private key in a binary array format. */
ec_key_private_key_to_bin(*ec_key, privk, privk_len);
/* Get the EC key`s curve name. */
*curve = EC_GROUP_get_curve_name(ec_group);
log("**********************************************\n"
"* (ECIES RECEIVER) EC KEY LOADED FROM FILE *\n"
"**********************************************\n");
dump_hex("pubkey", *pubk, *pubk_len);
dump_hex("privkey", *privk, *privk_len);
log("%-10s: %s(%d)\n", "curve", OBJ_nid2sn(*curve), *curve);
return 0;
}
int ecies_transmitter_send_message(uint8_t *msg,
size_t msg_len,
int curve,
const uint8_t *peer_pubk,
const uint8_t peer_pubk_len,
uint8_t **epubk, // out (must free)
size_t *epubk_len, // out
uint8_t **iv, // out (must free)
uint8_t *iv_len, // out
uint8_t **tag, // out (must free)
uint8_t *tag_len, // out
uint8_t **ciphertext, // out (must free)
uint8_t *ciphertext_len) // out
{
uint8_t *skey = NULL; // DH generated shared symmetric key
size_t skey_len = 0;
/* Generate the shared symmetric key (transmitter). */
ecies_transmitter_generate_symkey(curve, peer_pubk, peer_pubk_len,
epubk, epubk_len, &skey, &skey_len);
if (skey_len != 32)
err("Invalid symkey length %lub (expecting 256b)\n",
(skey_len * 8));
log("*******************************************************************\n"
"* (ECIES TRANSMITTER) EPHEMERAL EC PUBLIC KEY AND SYMMETRIC KEY *\n"
"*******************************************************************\n");
dump_hex("epubkey", *epubk, *epubk_len);
dump_hex("symkey", skey, skey_len);
/* Encrypt the data using 256b AES-GCM. */
aes_gcm_256b_encrypt(msg, msg_len, skey, NULL, 0,
iv, iv_len, tag, tag_len,
ciphertext, ciphertext_len);
log("************************************************\n"
"* (ECIES TRANSMITTER) AES-GCM ENCRYPTED DATA *\n"
"************************************************\n");
log("%-10s: (%lu) %s\n", "plain-tx", msg_len, msg); // it`s a string
dump_hex("iv", *iv, *iv_len);
dump_hex("tag", *tag, *tag_len);
dump_hex("cipher", *ciphertext, *ciphertext_len);
free(skey);
return 0;
}
int ecies_receiver_recv_message(const EC_KEY *ec_key,
const uint8_t *peer_pubk,
const uint8_t peer_pubk_len,
uint8_t *iv,
uint32_t iv_len,
uint8_t *tag,
uint32_t tag_len,
uint8_t *ciphertext,
uint32_t ciphertext_len)
{
// Shared symmetric encryption key (DH generated)
uint8_t *skey = NULL;
size_t skey_len = 0;
// Decrypted data (plaintext)
uint8_t *plaintext = NULL;
uint8_t plaintext_len = 0;
/* Generate the shared symmetric key (receiver). */
ecies_receiver_generate_symkey(ec_key, peer_pubk, peer_pubk_len,
&skey, &skey_len);
if (skey_len != 32)
err("Invalid symkey length %lub (expecting 256b)\n",
(skey_len * 8));
log("************************************\n"
"* (ECIES RECEIVER) SYMMETRIC KEY *\n"
"************************************\n");
dump_hex("symkey", skey, skey_len);
/* Decrypt the data using 256b AES-GCM. */
aes_gcm_256b_decrypt(ciphertext, ciphertext_len, skey, NULL, 0,
iv, iv_len, tag, tag_len,
&plaintext, &plaintext_len);
log("*********************************************\n"
"* (ECIES RECEIVER) AES-GCM DECRYPTED DATA *\n"
"*********************************************\n");
log("%-10s: (%d) %s\n", "plain-rx", plaintext_len, plaintext);
free(skey);
free(plaintext);
return 0;
}
int main(int argc, char * argv[])
{
EC_KEY *ec_key = NULL; // EC key from key file
// Receiver`s EC Key (public, private, curve)
uint8_t *pubk = NULL;
size_t pubk_len = 0;
uint8_t *privk = NULL;
size_t privk_len = 0;
int curve;
// Transmitter`s ephemeral public EC Key
uint8_t *epubk = NULL;
size_t epubk_len = 0;
// AES-GCM encrypted data (IV, authentication tag, ciphertext)
uint8_t *iv = NULL;
uint8_t iv_len = 0;
uint8_t *tag = NULL;
uint8_t tag_len = 0;
uint8_t *ciphertext = NULL;
uint8_t ciphertext_len = 0;
if (argc != 2)
err("Must specify EC key file in DER format\n"
"Usage: %s <file.der>\n", argv[0]);
/* ECIES Receiver loads the EC key. */
ecies_receiver_load_key(argv[1], &ec_key, &curve,
&pubk, &pubk_len, &privk, &privk_len);
/*
* At this point (receiver private key data loaded):
* - `ppub` holds the public key in uncompressed binary format
* - `ppriv` holds the private key in binary format
* - `curve` holds the curve name in ID format
*/
log("\n-> (receiver) sends public key, curve name...\n\n");
/* ECIES Transmitter sends encrypted message to the Receiver. */
#define MSG "The quick brown fox jumps over the lazy dog!"
ecies_transmitter_send_message((uint8_t *)MSG, (strlen(MSG) + 1),
curve, pubk, pubk_len,
&epubk, &epubk_len,
&iv, &iv_len, &tag, &tag_len,
&ciphertext, &ciphertext_len);
/*
* At this point (transmitter encrypted messsage):
* - `epubk` holds the ephemeral public key in uncompressed
* binary format, generated by the transmitter
* - `iv` holds the IV used for the AES-GCM encrypted data
* - `tag` holds the AES-GCM auth tag of the encrypted data
* - `ciphertext` holds encrypted message data
*/
log("\n-> (transmitter) sends ephemeral public key, IV, tag, ciphertxt...\n\n");
/* ECIES Receiver receives encrypted message from the Transmitter. */
ecies_receiver_recv_message(ec_key, epubk, epubk_len,
iv, iv_len, tag, tag_len,
ciphertext, ciphertext_len);
free(iv);
free(tag);
free(ciphertext);
free(epubk);
free(pubk);
free(privk);
return 0;
}
๋ฐ์ํ
'Source Code' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
[OpenSSL] EC_POINT example source (0) | 2016.11.25 |
---|