C++国密SM2算法加解密的使用

目录

效果

在线校验

代码实现参考

项目

下载


效果

加密字符串:lxw 123abcD 2024-09-01:12:00

加密后信息:042E82EE8ACE2BD56FA71DC6A0C34190627AA365F8EEE6261903BEE327A85EB5E1D6E78F2D79AD6F6DC9E45C0829625DC3165BB78BD897F99044A640F930653747939CF9D5A10C8216F945A55949D8B759FAC93638AD24321017C83331F213C7599802EA216083D6E6C1372C838B9F1AA756B11E8D3BFF6A294C7FCA61

解密后信息:lxw 123abcD 2024-09-01:12:00

在线校验

地址:https://the-x.cn/cryptography/Sm2.aspx

代码实现参考

https://github.com/yaqiangxue/Test_SM2_encrypt_and_decrypt/tree/master

项目

代码

#include "StdAfx.h"

#include <iostream>

#include <string>

#include <cstring>

#include <memory>

#include <openssl/bio.h>

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <iostream>

#include "sm2_cipher_error_codes.h"

#include "sm2_create_key_pair.h"

#include "sm2_encrypt_and_decrypt.h"

using namespace std;

// 将16进制的string字符串,转成16进制的arr

int hexCharStr2unsignedCharStr(char *src, unsigned long lsrc, int flag, unsigned char * out, unsigned long * lout)

{

if((0 == flag && 0 !=lsrc%2) || (0 != flag && 0 !=lsrc%3) ||NULL == src || NULL == out )

{

return 1;//param err

}

int j = 0;//index of out buff

if(0 == flag)

{ //int i;

for (int i=0; i<lsrc; i += 2)

{

int tmp = 0;

int HIGH_HALF_BYTE = 0;

int LOW_HALF_BYTE = 0;

if (src[i]>= 0x30 && src[i]<=0x39)

{

HIGH_HALF_BYTE = src[i] - 0x30;

}

else if (src[i]>= 0x41 && src[i]<=0x46)

{

HIGH_HALF_BYTE = src[i] - 0x37;

}

else if( src[i]>= 0x61 && src[i]<=0x66)

{

HIGH_HALF_BYTE = src[i] - 0x57;

}

else if( src[i] == 0x20)

{

HIGH_HALF_BYTE = 0x00;

}

else

{

return -1;

}

if (src[i+1]>= 0x30 && src[i+1]<=0x39)

{

LOW_HALF_BYTE = src[i+1] - 0x30;

}

else if (src[i+1]>= 0x41 && src[i+1]<=0x46)

{

LOW_HALF_BYTE = src[i+1] - 0x37;

}

else if( src[i+1]>= 0x61 && src[i+1]<=0x66)

{

LOW_HALF_BYTE = src[i+1] - 0x57;

}

else if( src[i+1] == 0x20)

{

LOW_HALF_BYTE = 0x00;

}

else

{

return -1;

}

tmp = (HIGH_HALF_BYTE<<4) + LOW_HALF_BYTE;

out [j] = tmp;

j++;

}

}

else

{ //int i;

for (int i=0; i<lsrc; i += 3)

{

int tmp = 0;

int HIGH_HALF_BYTE = 0;

int LOW_HALF_BYTE = 0;

if ((i+2<= lsrc) && (src[i+2] != flag))

{

return 1;

}

if (src[i]>= 0x30 && src[i]<=0x39 )

{

HIGH_HALF_BYTE = src[i] - 0x30;

}

else if (src[i]>= 0x41 && src[i]<=0x46)

{

HIGH_HALF_BYTE = src[i] - 0x37;

}

else if( src[i]>= 0x61 && src[i]<=0x66)

{

HIGH_HALF_BYTE = src[i] - 0x57;

}

else

{

return -1;

}

if (src[i+1]>= 0x30 && src[i+1]<=0x39)

{

LOW_HALF_BYTE = src[i+1] - 0x30;

}

else if (src[i+1]>= 0x41 && src[i+1]<=0x46)

{

LOW_HALF_BYTE = src[i+1] - 0x37;

}

else if( src[i+1]>= 0x61 && src[i+1]<=0x66)

{

LOW_HALF_BYTE = src[i+1] - 0x57;

}

else

{

return -1;

}

tmp = (HIGH_HALF_BYTE<<4) + LOW_HALF_BYTE;

out [j] = tmp;

j++;

}

}

* lout = j;

return 0;

}

// 将hexarr 转成16进制的字符串 如 0x11 0x22 转了之后是 "1122"

string array2hex(const unsigned char *arr, size_t len)

{

size_t i;

string res;

char tmp[3];

const char *tab = "0123456789ABCDEF";

res.reserve(len * 2 + 1);

for(i = 0; i < len; ++i) {

tmp[0] = tab[arr[i] >> 4];

tmp[1] = tab[arr[i] & 0xf];

tmp[2] = '\0';

res.append(tmp);

}

return res;

}

int main() {

int error_code;

//生成密钥对

/*SM2_KEY_PAIR key_pair;

if ( error_code = sm2_create_key_pair(&key_pair) )

{

printf("Create SM2 key pair failed!\n");

return (-1);

}

std::string pubKeyStr2=array2hex(key_pair.pri_key,32);

std::string priKeyStr2=array2hex(key_pair.pub_key,65);

std::cout<<"pubKeyStr:"<<pubKeyStr2<<std::endl;

std::cout<<"priKeyStr:"<<priKeyStr2<<std::endl;*/

//公钥是加04的。

std::string pubKeyStr = "04FDFB7C93565AB39E1D8178429632EEC914F6A347AE9A0CE9B201FFAEA81A80CC4D81036191209B21CDBAD8A4BCD5C9A776FEDB771D6D2D8DAC0F1E5941C0F63C";

std::string priKeyStr = "832B9C649C63B376DBD1D858C4D1B804CCFF6F7B6B588A9F30A54AF821F80E86";

std::string msg = "lxw 123abcD 2024-09-01:12:00";

std::cout<<"加密字符串:"<<msg<<std::endl<<std::endl;

int msg_len = msg.length();

//私钥

unsigned char pri_key[32] = {0};

unsigned long pri_key_len = 32;

//公钥

unsigned char pub_key[65] = {0};

unsigned long pub_key_len = 65;

unsigned char c1[65], c3[32];

unsigned long c1Len = 65;

unsigned long c3Len = 32;

unsigned char *c2, *plaintext;

int b = hexCharStr2unsignedCharStr((char*)priKeyStr.c_str(), priKeyStr.length(), 0, pri_key, &pri_key_len);

if(b != 0)

{

printf("转换priKeyStr失败\n");

}

b = hexCharStr2unsignedCharStr((char*)pubKeyStr.c_str(), pubKeyStr.length(), 0, pub_key, &pub_key_len);

if(b != 0)

{

printf("转换pubKeyStr失败\n");

}

if ( !(c2 = (unsigned char *)malloc(msg_len)) )

{

printf("Memory allocation failed!\n");

return ALLOCATION_MEMORY_FAIL;

}

//加密

if ( error_code = sm2_encrypt((unsigned char *)msg.c_str(),

msg_len,

pub_key,

c1,

c3,

c2) )

{

printf("Create SM2 ciphertext by using input defined in standard failed!\n");

free(c2);

return error_code;

}

std::string c1str=array2hex(c1,c1Len);

std::string c2str=array2hex(c2,msg_len);

std::string c3str=array2hex(c3,c3Len);

std::string result=c1str+c2str+c3str;

std::cout<<"加密后信息:"<<result<<std::endl<<std::endl;

//解密

if ( !(plaintext = (unsigned char *)malloc(c2str.length())) )

{

printf("Memory allocation failed!\n");

return ALLOCATION_MEMORY_FAIL;

}

if ( error_code = sm2_decrypt(c1,

c3,

c2,

msg_len,

pri_key,

plaintext) )

{

printf("Decrypt SM2 ciphertext by using private key defined in standard failed!\n");

}

std::string plainTextStr((char*)plaintext);

plainTextStr = plainTextStr.substr(0, msg_len);

std::cout<<"解密后信息:"<<plainTextStr<<std::endl;

free(plaintext);

free(c2);

getchar();

return 0;

}

#include "StdAfx.h"
#include <iostream>
#include <string>
#include <cstring>
#include <memory>
#include <openssl/bio.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <iostream>
#include "sm2_cipher_error_codes.h"
#include "sm2_create_key_pair.h"
#include "sm2_encrypt_and_decrypt.h"

using namespace std;


// 将16进制的string字符串,转成16进制的arr
int hexCharStr2unsignedCharStr(char *src, unsigned long lsrc, int flag, unsigned char * out, unsigned long * lout)
{
	if((0 == flag && 0 !=lsrc%2) || (0 != flag && 0 !=lsrc%3) ||NULL == src || NULL == out )
	{
		return 1;//param err
	}

	int j = 0;//index of out buff
	if(0 == flag)
	{	//int i;
		for (int i=0; i<lsrc; i += 2)
		{
			int tmp = 0;
			int HIGH_HALF_BYTE = 0;
			int LOW_HALF_BYTE = 0;
			if (src[i]>= 0x30 && src[i]<=0x39)
			{
				HIGH_HALF_BYTE = src[i] - 0x30;
			}
			else if (src[i]>= 0x41 && src[i]<=0x46)
			{
				HIGH_HALF_BYTE = src[i] - 0x37;
			}
			else if( src[i]>= 0x61 && src[i]<=0x66)
			{
				HIGH_HALF_BYTE = src[i] - 0x57;
			}
			else if( src[i] == 0x20)
			{
				HIGH_HALF_BYTE = 0x00;
			}
			else
			{
				return -1;
			}

			if (src[i+1]>= 0x30 && src[i+1]<=0x39)
			{
				LOW_HALF_BYTE = src[i+1] - 0x30;
			}
			else if (src[i+1]>= 0x41 && src[i+1]<=0x46)
			{
				LOW_HALF_BYTE = src[i+1] - 0x37;
			}
			else if( src[i+1]>= 0x61 && src[i+1]<=0x66)
			{
				LOW_HALF_BYTE = src[i+1] - 0x57;
			}
			else if( src[i+1] == 0x20)
			{
				LOW_HALF_BYTE = 0x00;
			}
			else
			{
				return -1;
			}

			tmp = (HIGH_HALF_BYTE<<4) + LOW_HALF_BYTE;
			out [j] = tmp;
			j++;
		}
	}
	else
	{	//int i;
		for (int i=0; i<lsrc; i += 3)
		{
			int tmp = 0;
			int HIGH_HALF_BYTE = 0;
			int LOW_HALF_BYTE = 0;
			if ((i+2<= lsrc) && (src[i+2] != flag))
			{
				return 1;
			}

			if (src[i]>= 0x30 && src[i]<=0x39 )
			{
				HIGH_HALF_BYTE = src[i] - 0x30;
			}
			else if (src[i]>= 0x41 && src[i]<=0x46)
			{
				HIGH_HALF_BYTE = src[i] - 0x37;
			}
			else if( src[i]>= 0x61 && src[i]<=0x66)
			{
				HIGH_HALF_BYTE = src[i] - 0x57;
			}
			else
			{
				return -1;
			}

			if (src[i+1]>= 0x30 && src[i+1]<=0x39)
			{
				LOW_HALF_BYTE = src[i+1] - 0x30;
			}
			else if (src[i+1]>= 0x41 && src[i+1]<=0x46)
			{
				LOW_HALF_BYTE = src[i+1] - 0x37;
			}
			else if( src[i+1]>= 0x61 && src[i+1]<=0x66)
			{
				LOW_HALF_BYTE = src[i+1] - 0x57;
			}
			else
			{
				return -1;
			}

			tmp = (HIGH_HALF_BYTE<<4) + LOW_HALF_BYTE;
			out [j] = tmp;
			j++;
		}
	}

	* lout = j;
	return 0;

}

// 将hexarr 转成16进制的字符串  如 0x11 0x22  转了之后是 "1122"
string array2hex(const unsigned char *arr, size_t len)
{
	size_t i;
	string res;
	char tmp[3];
	const char *tab = "0123456789ABCDEF";
	res.reserve(len * 2 + 1);
	for(i = 0; i < len; ++i) {
		tmp[0] = tab[arr[i] >> 4];
		tmp[1] = tab[arr[i] & 0xf];
		tmp[2] = '\0';
		res.append(tmp);
	}
	return res;
}

int main() {

	int error_code;
	//生成密钥对
	/*SM2_KEY_PAIR key_pair;
	if ( error_code = sm2_create_key_pair(&key_pair) )
	{
	printf("Create SM2 key pair failed!\n");
	return (-1);
	}
	std::string pubKeyStr2=array2hex(key_pair.pri_key,32);
	std::string priKeyStr2=array2hex(key_pair.pub_key,65);
	std::cout<<"pubKeyStr:"<<pubKeyStr2<<std::endl;
	std::cout<<"priKeyStr:"<<priKeyStr2<<std::endl;*/

	//公钥是加04的。
	std::string pubKeyStr = "04FDFB7C93565AB39E1D8178429632EEC914F6A347AE9A0CE9B201FFAEA81A80CC4D81036191209B21CDBAD8A4BCD5C9A776FEDB771D6D2D8DAC0F1E5941C0F63C";
	std::string priKeyStr = "832B9C649C63B376DBD1D858C4D1B804CCFF6F7B6B588A9F30A54AF821F80E86";

	std::string msg = "lxw 123abcD 2024-09-01:12:00";
	std::cout<<"加密字符串:"<<msg<<std::endl<<std::endl;
	int msg_len = msg.length();

	//私钥
	unsigned char pri_key[32] = {0};
	unsigned long pri_key_len = 32;

	//公钥
	unsigned char pub_key[65] = {0}; 
	unsigned long pub_key_len = 65;

	unsigned char c1[65], c3[32];
	unsigned long c1Len = 65;
	unsigned long c3Len = 32;
	unsigned char *c2, *plaintext;

	int b = hexCharStr2unsignedCharStr((char*)priKeyStr.c_str(), priKeyStr.length(), 0, pri_key, &pri_key_len);
	if(b != 0)
	{
		printf("转换priKeyStr失败\n");
	}

	b = hexCharStr2unsignedCharStr((char*)pubKeyStr.c_str(), pubKeyStr.length(), 0, pub_key, &pub_key_len);
	if(b != 0)
	{
		printf("转换pubKeyStr失败\n");
	}

	if ( !(c2 = (unsigned char *)malloc(msg_len)) )
	{
		printf("Memory allocation failed!\n");
		return ALLOCATION_MEMORY_FAIL;
	}

	//加密
	if ( error_code = sm2_encrypt((unsigned char *)msg.c_str(),
		msg_len,
		pub_key,
		c1,
		c3,
		c2) )
	{
		printf("Create SM2 ciphertext by using input defined in standard failed!\n");
		free(c2);
		return error_code;
	}
	std::string c1str=array2hex(c1,c1Len);
	std::string c2str=array2hex(c2,msg_len);
	std::string c3str=array2hex(c3,c3Len);

	std::string result=c1str+c2str+c3str;
	std::cout<<"加密后信息:"<<result<<std::endl<<std::endl;

	//解密
	if ( !(plaintext = (unsigned char *)malloc(c2str.length())) )
	{
		printf("Memory allocation failed!\n");
		return ALLOCATION_MEMORY_FAIL;
	}
	if ( error_code = sm2_decrypt(c1,
		c3,
		c2,
		msg_len,
		pri_key,
		plaintext) )
	{
		printf("Decrypt SM2 ciphertext by using private key defined in standard failed!\n");
	}

	std::string plainTextStr((char*)plaintext);
	plainTextStr = plainTextStr.substr(0, msg_len);
	std::cout<<"解密后信息:"<<plainTextStr<<std::endl;

	free(plaintext);
	free(c2);

	getchar();
	return 0;
}

下载

源码下载

相关推荐
tyler_download2 分钟前
手撸 chatgpt 大模型:简述 LLM 的架构,算法和训练流程
算法·chatgpt
何曾参静谧8 分钟前
「Py」Python基础篇 之 Python都可以做哪些自动化?
开发语言·python·自动化
Prejudices12 分钟前
C++如何调用Python脚本
开发语言·c++·python
单音GG15 分钟前
推荐一个基于协程的C++(lua)游戏服务器
服务器·c++·游戏·lua
SoraLuna22 分钟前
「Mac玩转仓颉内测版7」入门篇7 - Cangjie控制结构(下)
算法·macos·动态规划·cangjie
我狠狠地刷刷刷刷刷25 分钟前
中文分词模拟器
开发语言·python·算法
鸽鸽程序猿25 分钟前
【算法】【优选算法】前缀和(上)
java·算法·前缀和
wyh要好好学习28 分钟前
C# WPF 记录DataGrid的表头顺序,下次打开界面时应用到表格中
开发语言·c#·wpf
AitTech29 分钟前
C#实现:电脑系统信息的全面获取与监控
开发语言·c#
qing_04060331 分钟前
C++——多态
开发语言·c++·多态