gpt4 book ai didi

c - 使用 C 中的 openssl 库进行简单的 AES 加密解密

转载 作者:太空狗 更新时间:2023-10-29 17:18:25 26 4
gpt4 key购买 nike

我想加密一个包含少量字符串的结构,然后对其进行解密。我尝试了以下代码。原始代码是从网上找到的,并且运行良好。我将它的输入更改为结构。以下是代码。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <openssl/aes.h>
#include <openssl/rand.h>

typedef struct ticket { /* test field */
int ticketId;
char username[20];
char date[20];
} USR_TICKET;

// a simple hex-print routine. could be modified to print 16 bytes-per-line
static void hex_print(const void* pv, size_t len)
{
const unsigned char * p = (const unsigned char*)pv;
if (NULL == pv)
printf("NULL");
else
{
size_t i = 0;
for (; i<len;++i)
printf("%02X ", *p++);
}
printf("\n");
}

// main entrypoint
int main(int argc, char **argv)
{
int keylength;
printf("Give a key length [only 128 or 192 or 256!]:\n");
scanf("%d", &keylength);

/* generate a key with a given length */
unsigned char aes_key[keylength/8];
memset(aes_key, 0, keylength/8);
if (!RAND_bytes(aes_key, keylength/8))
exit(-1);

/* input struct creation */
size_t inputslength = sizeof(USR_TICKET);
USR_TICKET ticket;
ticket.ticketId = 1;
time_t now = time(NULL);
strftime(ticket.date, 20, "%Y-%m-%d", localtime(&now));
strcpy(ticket.username, "ravinda");

printf("Username - %s\n", ticket.username);
printf("Ticket Id - %d\n", ticket.ticketId);
printf("Date - %s\n", ticket.date);

/* init vector */
unsigned char iv_enc[AES_BLOCK_SIZE], iv_dec[AES_BLOCK_SIZE];
RAND_bytes(iv_enc, AES_BLOCK_SIZE);
memcpy(iv_dec, iv_enc, AES_BLOCK_SIZE);

// buffers for encryption and decryption
const size_t encslength = ((inputslength + AES_BLOCK_SIZE) / AES_BLOCK_SIZE) * AES_BLOCK_SIZE;
unsigned char enc_out[encslength];
unsigned char dec_out[inputslength];
memset(enc_out, 0, sizeof(enc_out));
memset(dec_out, 0, sizeof(dec_out));

// so i can do with this aes-cbc-128 aes-cbc-192 aes-cbc-256
AES_KEY enc_key, dec_key;
AES_set_encrypt_key(aes_key, keylength, &enc_key);
AES_cbc_encrypt((unsigned char *)&ticket, enc_out, inputslength, &enc_key, iv_enc, AES_ENCRYPT);

AES_set_decrypt_key(aes_key, keylength, &dec_key);
AES_cbc_encrypt(enc_out, dec_out, encslength, &dec_key, iv_dec, AES_DECRYPT);

printf("original:\t");
hex_print((unsigned char *)&ticket, inputslength);

printf("encrypt:\t");
hex_print(enc_out, sizeof(enc_out));

printf("decrypt:\t");
hex_print(dec_out, sizeof(dec_out));

USR_TICKET * dyc = (USR_TICKET *)dec_out;
printf("Username - %s\n", dyc->username);
printf("Ticket Id - %d\n", dyc->ticketId);
printf("Date - %s\n", dyc->date);
return 0;
}

问题是只有结构的前两个成员可以正确解密。在那之后数据被破坏了。我在这里做错了什么?

最佳答案

几乎会说这是 OpenSSL 的问题。似乎当传递给 AES_cbc_encryptlength 参数是 > AES_BLOCK_SIZE 但不是其整数倍时(即 length mod AES_BLOCK_SIZE != 0),然后最后一个 block 使用初始 IV 加密,而不是 CBC 模式应该使用的前一个密文 block 。

您可以通过以下两种方式之一解决此问题:

将您的结构复制到缓冲区,其大小是 AES_BLOCK_SIZE 的整数倍,或分两部分加密 - 完整 block ,后跟单个部分块。后者具有避免额外内存使用的优点,可以按如下方式完成:

size_t fullBlocks = inputslength - (inputslength % AES_BLOCK_SIZE);
size_t remainingBlock = inputslength - fullBlocks;

AES_cbc_encrypt((unsigned char *)&ticket, enc_out, fullBlocks, &enc_key, iv_enc, AES_ENCRYPT);
AES_cbc_encrypt((unsigned char *)&ticket + fullBlocks, enc_out + fullBlocks, remainingBlock, &enc_key, iv_enc, AES_ENCRYPT);

然后您应该能够像目前一样毫无问题地解密。但值得注意的是,您应该将 dec_out 声明为与 enc_out 相同的大小,因为您当前在解密时超出了 dec_out 缓冲区。

编辑:

我将此作为 OpenSSL 中的错误提出:https://rt.openssl.org/Ticket/Display.html?id=3182&user=guest&pass=guest虽然对于这实际上是一个错误还是只是(未记录的)未定义行为存在一些争论,但普遍的共识是应该使用 EVP 例程,而不是这些低级函数。

关于c - 使用 C 中的 openssl 库进行简单的 AES 加密解密,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19889740/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com