OpenSSL aes加解密簡單實例+base64編解碼編碼
#include <stdio.h> #include <string.h> #include <memory.h> #include <openssl/aes.h> #include <stdlib.h> #define AES_BLOCK_SIZE 16 #define BUF_SIZE 1024 #define CRYPT_USER_KEY "d1ea806ec3c35643" static const unsigned char en_table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; static unsigned char de_table[128] = {0}; static void init_de_table() { if (de_table[0] != 0) { return; } memset(de_table, 0xff, sizeof(de_table)); const unsigned char* p; for (p = en_table; *p != 0; p++) { de_table[*p] = p - en_table; } de_table['='] = 0; } int get_encode_len(int len) { return (((len/3 + (len%3==0 ? 0:1)) * 4) + 1); } int get_decode_len(int len) { return ((len/4)*3 + 1); } // base64 編碼 int base64_encode(const unsigned char* data, int data_len, char* dst, int dst_len) { if (data == NULL || data_len == 0 || dst == NULL || dst_len == 0) { return -1; } if (dst_len < get_encode_len(data_len)) { return -2; } int mod_len = data_len%3; char* dst_tmp = dst; int pos = 0; for (pos = 0; pos+2 < data_len; pos+=3) { *(dst_tmp++) = en_table[data[pos] >> 2]; *(dst_tmp++) = en_table[((data[pos] & 0x03) << 4) | (data[pos+1] >> 4)]; *(dst_tmp++) = en_table[((data[pos+1] & 0x0f) << 2) | (data[pos+2] >> 6)]; *(dst_tmp++) = en_table[data[pos+2] & 0x3f]; } if (mod_len == 2) { *(dst_tmp++) = en_table[data[pos] >> 2]; *(dst_tmp++) = en_table[((data[pos] & 0x03) << 4) | (data[pos+1] >> 4)]; *(dst_tmp++) = en_table[((data[pos+1] & 0x0f) << 2)]; *(dst_tmp++) = '='; } else if (mod_len == 1) { *(dst_tmp++) = en_table[data[pos] >> 2]; *(dst_tmp++) = en_table[((data[pos] & 0x03) << 4)]; *(dst_tmp++) = '='; *(dst_tmp++) = '='; } *dst_tmp = 0; return (dst_tmp - dst); } // base64 解碼 int base64_decode(const unsigned char* data, int data_len, char* dst, int dst_len) { if (data == NULL || data_len < 4 || dst == NULL || dst_len == 0) { return -1; } if (dst_len < get_decode_len(data_len)) { return -2; } init_de_table(); char* dst_tmp = dst; int pos; for (pos = 0; pos < data_len; pos+=4) { if (de_table[data[pos]]==0xff || de_table[data[pos+1]]==0xff || de_table[data[pos+2]]==0xff || de_table[data[pos+3]]==0xff) { return -3; } *(dst_tmp++) = (de_table[data[pos]]<<2) | (de_table[data[pos+1]]>>4)&0x3; *(dst_tmp++) = (de_table[data[pos+1]]<<4) | (de_table[data[pos+2]]>>2)&0x0f; *(dst_tmp++) = (de_table[data[pos+2]]<<6) | (de_table[data[pos+3]])&0x3f; } while (*(dst_tmp-1) == 0) { dst_tmp--; } *dst_tmp = 0; return (dst_tmp - dst); } // 根據數據字節數 獲取加密字節數 int get_crypt_size(int data_len) { return ((data_len/AES_BLOCK_SIZE)+((data_len%AES_BLOCK_SIZE) == 0 ? 0:1))*AES_BLOCK_SIZE; } // 加密 int aes_encrypt(char* in, char* key, char* out)//, int olen) { if(NULL == in || NULL == key || NULL == out) return -1; AES_KEY aes; if(AES_set_encrypt_key((unsigned char*)key, 128, &aes) < 0) { return -2; } int len=strlen(in); int en_len=0; while(en_len<len)//輸入輸出字符串夠長。並且是AES_BLOCK_SIZE的整數倍,需要嚴格限制 { AES_encrypt((unsigned char*)in, (unsigned char*)out, &aes); in+=AES_BLOCK_SIZE; out+=AES_BLOCK_SIZE; en_len+=AES_BLOCK_SIZE; } return get_crypt_size(len); } // 解密 int aes_decrypt(char* in, int in_len, char* key, char* out) { if(NULL == in || NULL == key || NULL == out) return -1; AES_KEY aes; if(AES_set_decrypt_key((unsigned char*)key, 128, &aes) < 0) { return -2; } int en_len=0; while(en_len<in_len) { AES_decrypt((unsigned char*)in, (unsigned char*)out, &aes); in+=AES_BLOCK_SIZE; out+=AES_BLOCK_SIZE; en_len+=AES_BLOCK_SIZE; } return 0; } int main(int argc, char* argv[]) { if (argc < 2) return -1; printf("src: %s\n", argv[1]); char in[BUF_SIZE] = {0}; memcpy(in, argv[1], strlen(argv[1])); char aes_en[BUF_SIZE] = {0}; int aes_en_len = aes_encrypt(in, CRYPT_USER_KEY, aes_en); // aes-128 加密 if (aes_en_len < 0) { printf("aes_encrypt error ret: %d\n", aes_en_len); return -1; } int i=0; printf("aes_en: {"); for(i=0; i<aes_en_len; i++) printf(" %d",aes_en[i]); printf(" } aes_en_len: %d\n", aes_en_len); char base64_en[BUF_SIZE] = {0}; int base64_en_len = base64_encode(aes_en, aes_en_len, base64_en, BUF_SIZE); // base64 編碼 if (base64_en_len < 0) { printf("base64_encode error ret: %d\n", base64_en_len); return -1; } printf("base64_en_len: %d base64_en: %s\n", base64_en_len, base64_en); char base64_de[BUF_SIZE] = {0}; int base64_de_len = base64_decode(base64_en, base64_en_len, base64_de, BUF_SIZE); // base64 解碼 if (base64_de_len < 0) { printf("base64_encode error ret: %d\n", base64_de_len); return -1; } printf("base64_de: {"); for(i=0; i<base64_de_len; i++) printf(" %d",base64_de[i]); printf(" } base64_de_len: %d\n", base64_de_len); char aes_de[BUF_SIZE] = {0}; int aes_de_len = aes_decrypt(base64_de, base64_de_len, CRYPT_USER_KEY, aes_de); // aes-128 解密 if (aes_de_len < 0) { printf("aes_encrypt error ret: %d\n", aes_de_len); return -1; } printf("aes_de_len: %d aes_en: %s\n", strlen(aes_de), aes_de); return 0; }
編譯時要 libcrypto.so加密