项目地址Gitee地址

​Gitee地址​

介绍

QT版本基于OpenSSL3的ECDH密钥交换算法

OpenSSL版本

3.0.8

QT版本

5.14.1

主要代码
#ifndef QECDHMGR_H
#define QECDHMGR_H

#include <QString>
#include <QByteArray>
#include <QDebug>
#include <QScopedPointer>
#include <openssl/dh.h>
#include <openssl/err.h>
#include <openssl/core_names.h>
#include <openssl/param_build.h>
#include <openssl/evp.h>
#include <openssl/ec.h>
#include <openssl/param_build.h>
#include <openssl/bn.h>


class QECDHMgr
{
public:
QECDHMgr();

// 获取最后一次错误信息
QString GetLastError();

// 设置CurveName,可以为secp384r1等
// 系统不支持的加密方式返回false
bool SetCurveName(QString curveName);

// 获取CurevName
QString GetCurveName();

// 生成密钥对(包含公钥和私钥)
bool GenerateKeys();

// 获取私钥
QString GetPrivateKey(); // for debug

// 获取公钥
QString GetPublicKey();

// 设置私钥
bool SetPrivateKey(QString hexPrivateKey);

// 设置公钥
bool SetPublicKey(QString hexPublicKey);

// 设置公钥、私钥对
bool SetKeyPair(QString hexPublicKey, QString hexPrivateKey);

// 生成对称密钥
QString DeriveSharedSecret(QString hexPeerPublicKey);

private:

static int GetHexValue(unsigned char hexDigit)
{
static constexpr char hexValues[256] = {
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -1, -1, -1, -1, -1,
-1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
};
return hexValues[hexDigit];
}

// this struct calls "OSSL_PARAM_BLD_free" to delete the pointer
struct ScopedPointerParamBldDeleter
{
static inline void cleanup(OSSL_PARAM_BLD *pointer)
{
OSSL_PARAM_BLD_free(pointer);
}
};

// this struct calls "OSSL_PARAM_BLD_free" to delete the pointer
struct ScopedPointerParamDeleter
{
static inline void cleanup(OSSL_PARAM *pointer)
{
OSSL_PARAM_free(pointer);
}
};

// this struct calls "OSSL_PARAM_BLD_free" to delete the pointer
struct ScopedPointerKeyCtxDeleter
{
static inline void cleanup(EVP_PKEY_CTX *pointer)
{
EVP_PKEY_CTX_free(pointer);
}
};

// this struct calls "OSSL_PARAM_BLD_free" to delete the pointer
struct ScopedPointerKeyDeleter
{
static inline void cleanup(EVP_PKEY *pointer)
{
EVP_PKEY_free(pointer);
}
};

// Converts a hex string to BIGNUM.
BIGNUM* ConvertHexToBigNum(QString hexBigNum);

// Converts BIGNUM to a hex string.
// Returns an empty string if convertion fails.
QString ConvertBigNumToHex(const BIGNUM* bigNum);

// Converts binary data to a hex string.
QString ConvertDataToHex(QByteArray data);

// Converts a hex string to binary data.
QByteArray ConvertHexToData(QString hex);

// Create a Peer public key
bool CreatePeerPublicKey(QString hexPeerPublicKey, QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter>* peerPublicKey);

// Create a Peer private key
bool CreatePeerPrivateKey(QString hexPeerPrivateKey, QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter>* peerPrivateKey);

// Create a Peer key pair
bool CreatePeerKeyPair(QString hexPeerPublicKey, QString hexPeerPrivateKey, QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter>* peerPrivateKey);


private:

QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter> keyPair_;

QString curveName_;
};

#endif // QECDHMGR_H
#include "qdhmgr.h"

QECDHMgr::QECDHMgr()
{

}

QString QECDHMgr::GetCurveName()
{
return this->curveName_;
}

bool QECDHMgr::SetCurveName(QString curveName)
{
EC_builtin_curve * buffer = nullptr;

size_t count = EC_get_builtin_curves(nullptr, 0);
if (count <= 0) {
return false;
}

buffer = new EC_builtin_curve[count];
if (EC_get_builtin_curves(buffer, count) != count) {
return false;
}

// Check if the curve is in the list.
for (int i = 0; i < count; ++i) {
const char* shortName = OBJ_nid2sn(buffer[i].nid);
const char* comment = buffer[i].comment;
if (shortName && strlen(shortName) > 0) {
// qDebug() << buffer[i].nid << "|"<< shortName<<"|" << comment;
if(curveName == QString(shortName))
{
curveName_ = curveName;
return true;
}
}
}
return false;
}

bool QECDHMgr::GenerateKeys()
{
ERR_clear_error();

// First, create an OSSL_PARAM_BLD.
QScopedPointer <OSSL_PARAM_BLD , ScopedPointerParamBldDeleter> paramBuild(OSSL_PARAM_BLD_new());
if (!paramBuild.data())
{
return false;
}

// Push the curve name to the OSSL_PARAM_BLD.
if (!OSSL_PARAM_BLD_push_utf8_string(paramBuild.data(), OSSL_PKEY_PARAM_GROUP_NAME, curveName_.toStdString().c_str(), 0)) {
return false;
}

// Convert OSSL_PARAM_BLD to OSSL_PARAM.
QScopedPointer <OSSL_PARAM , ScopedPointerParamDeleter> params(OSSL_PARAM_BLD_to_param(paramBuild.data()));
if (!params.data()) {
return false;
}

// Create the EC key generation context.
QScopedPointer <EVP_PKEY_CTX , ScopedPointerKeyCtxDeleter> ctx(EVP_PKEY_CTX_new_from_name(nullptr, "EC", nullptr));
if (!ctx.data())
{
return false;
}

// Initialize the key generation context.
if (EVP_PKEY_keygen_init(ctx.data()) <= 0) {
return false;
}

// Set the parameters which include the curve name.
if (!EVP_PKEY_CTX_set_params(ctx.data(), params.data())) {
return false;
}

// Generate a key pair.
EVP_PKEY* keyPair = nullptr;
if (EVP_PKEY_generate(ctx.data(), &keyPair) <= 0) {
return false;
}
keyPair_.reset(keyPair);
return true;
}

QString QECDHMgr::GetPrivateKey()
{
ERR_clear_error();

if(keyPair_.data())
{
// The private key is stored as a BIGNUM object.
BIGNUM* privateKey = nullptr;
if (!EVP_PKEY_get_bn_param(keyPair_.data(), OSSL_PKEY_PARAM_PRIV_KEY, &privateKey)) {
return "";
}

// Convert the BIGNUM to a hex string.
QString hexPrivateKey = ConvertBigNumToHex(privateKey);
BN_free(privateKey);
return hexPrivateKey;
}else
{
return "";
}
}

QString QECDHMgr::GetPublicKey()
{
ERR_clear_error();
if(keyPair_.data())
{
// The public key is stored as a byte array.
// Get the array size.
size_t keyLength = 0;
if (!EVP_PKEY_get_octet_string_param(keyPair_.data(), OSSL_PKEY_PARAM_PUB_KEY, nullptr, 0, &keyLength)) {
return "";
}

// Get the key.
unsigned char * publicKey = new unsigned char [keyLength];
if (!EVP_PKEY_get_octet_string_param(keyPair_.data(), OSSL_PKEY_PARAM_PUB_KEY, publicKey, keyLength, &keyLength)) {
return "";
}

// Convert the byte array key to a hex string.
QString hexPublicKey = ConvertDataToHex(QByteArray((char *)publicKey, keyLength));
delete [] publicKey;
publicKey = nullptr;
return hexPublicKey;
}else
{
return "";
}
}

bool QECDHMgr::SetPublicKey(QString hexPublicKey)
{
QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter> peerPublicKey;
if(!CreatePeerPublicKey(hexPublicKey, &peerPublicKey))
{
return false;
}

keyPair_.reset(peerPublicKey.take());
return true;
}

bool QECDHMgr::SetPrivateKey(QString hexPrivateKey)
{
QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter> peerPrivateKey;
if(!CreatePeerPrivateKey(hexPrivateKey, &peerPrivateKey))
{
return false;
}
keyPair_.reset(peerPrivateKey.take());
return true;
}

bool QECDHMgr::SetKeyPair(QString hexPublicKey, QString hexPrivateKey)
{
QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter> peerKeyPair;
if(!CreatePeerKeyPair(hexPublicKey, hexPrivateKey, &peerKeyPair))
{
return false;
}
keyPair_.reset(peerKeyPair.take());
return true;
}

QString QECDHMgr::DeriveSharedSecret(QString hexPeerPublicKey)
{
ERR_clear_error();

// First, you have to create the peer public key object.
// It takes several calls, so it is done in a separate function.
QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter> peerPublicKey;
if(!CreatePeerPublicKey(hexPeerPublicKey, &peerPublicKey))
{
return "";
}

// Create the derivation context.
QScopedPointer <EVP_PKEY_CTX , ScopedPointerKeyCtxDeleter> derivationCtx(EVP_PKEY_CTX_new(keyPair_.data(), nullptr));
if (!derivationCtx.data()) {
return "";
}

// Initialize the derivation context.
if (EVP_PKEY_derive_init(derivationCtx.data()) <= 0) {
return "";
}

// Set the peer public key object.
if (EVP_PKEY_derive_set_peer(derivationCtx.data(), peerPublicKey.data()) <= 0) {
return "";
}

// Get the shared secret length.
size_t sharedSecretLength = 0;
if (EVP_PKEY_derive(derivationCtx.data(), nullptr, &sharedSecretLength) <= 0) {
return "";
}

if (sharedSecretLength == 0) {
return "";
}

unsigned char * sharedSecret = new unsigned char [sharedSecretLength];
// Derive the shared secret.
if (EVP_PKEY_derive(derivationCtx.data(), sharedSecret, &sharedSecretLength) <= 0) {
return "";
}
// Convert to a hex string.
QString hexSharedSecret = ConvertDataToHex(QByteArray((char *)sharedSecret, sharedSecretLength));
delete [] sharedSecret;

return hexSharedSecret;
}

QString QECDHMgr::GetLastError()
{
unsigned long lastError = ERR_peek_last_error();
if (lastError == 0) {
return "";
}
char errorString[256];
ERR_error_string_n(lastError, errorString, sizeof(errorString));
return errorString;
}
// ***************************** 以下是私有函数 **********************************

QString QECDHMgr::ConvertBigNumToHex(const BIGNUM *bigNum)
{
char* tmpHexBigNum = BN_bn2hex(bigNum);
if (!tmpHexBigNum) {
return "";
}
QString hexBigNum(tmpHexBigNum);
OPENSSL_free(tmpHexBigNum);
return hexBigNum;
}

BIGNUM* QECDHMgr::ConvertHexToBigNum(QString hexBigNum)
{
BIGNUM* bn = nullptr;
if (!BN_hex2bn(&bn, hexBigNum.toStdString().c_str())) {
return nullptr;
}
return bn;
}

QString QECDHMgr::ConvertDataToHex(QByteArray data)
{
static const char hexDigits[] = "0123456789ABCDEF";

QString hex;
hex.reserve(data.size() * 2);

for (unsigned char c : data) {
hex.push_back(hexDigits[c >> 4]);
hex.push_back(hexDigits[c & 15]);
}

return hex;
}

// Converts a hex string to binary data.
QByteArray QECDHMgr::ConvertHexToData(QString hex)
{
QByteArray data;
// Must be an even number!
if (hex.size() & 1) {
return data;
}

auto it = hex.begin();
while (it != hex.end()) {
int hi = GetHexValue((*it++).toLatin1());
int lo = GetHexValue((*it++).toLatin1());
if (hi == -1 || lo == -1) {
data.clear();
return data;
}
data.push_back(hi << 4 | lo);
}

return data;
}

bool QECDHMgr::CreatePeerPublicKey(QString hexPeerPublicKey, QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter>* peerPublicKey)
{
ERR_clear_error();

// First, we sould create an OSSL_PARAM_BLD with the curve name
// and the raw peer public key.
QScopedPointer <OSSL_PARAM_BLD, ScopedPointerParamBldDeleter> paramBuild(OSSL_PARAM_BLD_new());
if (!paramBuild.data()) {
return false;
}

// Set the curve name.
if (!OSSL_PARAM_BLD_push_utf8_string(paramBuild.data(), OSSL_PKEY_PARAM_GROUP_NAME,curveName_.toStdString().c_str(), 0)) {
return false;
}

// Convert the peer hex public key to raw data.
QByteArray binPeerPublicKey = ConvertHexToData(hexPeerPublicKey);
if (binPeerPublicKey.size() <= 0) {
return false;
}

// Set the raw peer public key.
if (!OSSL_PARAM_BLD_push_octet_string(paramBuild.data(), OSSL_PKEY_PARAM_PUB_KEY, binPeerPublicKey.data(), binPeerPublicKey.size())) {
return false;
}

// Convert the OSSL_PARAM_BLD to OSSL_PARAM.
QScopedPointer <OSSL_PARAM , ScopedPointerParamDeleter> params(OSSL_PARAM_BLD_to_param(paramBuild.data()));
if (!params.data()) {
return false;
}

// Create a EVP_PKEY context.
QScopedPointer <EVP_PKEY_CTX , ScopedPointerKeyCtxDeleter> peerPublicKeyCtx(EVP_PKEY_CTX_new_from_name(nullptr, "EC", nullptr));
if (!peerPublicKeyCtx.data()) {
return false;
}

// Initialize the context.
if (EVP_PKEY_fromdata_init(peerPublicKeyCtx.data()) <= 0) {
return false;
}

// Create the peer public key object.
EVP_PKEY* tmp = nullptr;
if (EVP_PKEY_fromdata(peerPublicKeyCtx.data(), &tmp, EVP_PKEY_PUBLIC_KEY, params.data()) <= 0) {
return false;
}
peerPublicKey->reset(tmp);
return true;
}

bool QECDHMgr::CreatePeerPrivateKey(QString hexPeerPrivateKey, QScopedPointer <EVP_PKEY , ScopedPointerKeyDeleter>* peerPrivateKey)
{
ERR_clear_error();

// First, we sould create an OSSL_PARAM_BLD with the curve name
// and the raw peer public key.
QScopedPointer <OSSL_PARAM_BLD, ScopedPointerParamBldDeleter> paramBuild(OSSL_PARAM_BLD_new());
if (!paramBuild.data()) {
return false;
}

// Set the curve name.
if (!OSSL_PARAM_BLD_push_utf8_string(paramBuild.data(), OSSL_PKEY_PARAM_GROUP_NAME,curveName_.toStdString().c_str(), 0)) {
return false;
}

// Convert the peer hex private key to raw data.
BIGNUM * binPeerPrivateKey = ConvertHexToBigNum(hexPeerPrivateKey);
if(!binPeerPrivateKey)
{
return false;
}

// Set the raw peer private key.
if (!OSSL_PARAM_BLD_push_BN(paramBuild.data(), OSSL_PKEY_PARAM_PRIV_KEY, binPeerPrivateKey)) {
return false;
}

// Convert the OSSL_PARAM_BLD to OSSL_PARAM.
QScopedPointer <OSSL_PARAM , ScopedPointerParamDeleter> params(OSSL_PARAM_BLD_to_param(paramBuild.data()));
if (!params.data()) {
return false;
}

// Create a EVP_PKEY context.
QScopedPointer <EVP_PKEY_CTX , ScopedPointerKeyCtxDeleter> peerPrivateKeyCtx(EVP_PKEY_CTX_new_from_name(nullptr, "EC", nullptr));
if (!peerPrivateKeyCtx.data()) {
return false;
}

// Initialize the context.
if (EVP_PKEY_fromdata_init(peerPrivateKeyCtx.data()) <= 0) {
return false;
}

// Create the peer private key object.
EVP_PKEY* tmp = nullptr;
if (EVP_PKEY_fromdata(peerPrivateKeyCtx.data(), &tmp, EVP_PKEY_KEYPAIR, params.data()) <= 0) {
return false;
}
peerPrivateKey->reset(tmp);
return true;
}

bool QECDHMgr::CreatePeerKeyPair(QString hexPeerPublicKey, QString hexPeerPrivateKey, QScopedPointer<EVP_PKEY, ScopedPointerKeyDeleter> *peerPrivateKey)
{
ERR_clear_error();

// First, we sould create an OSSL_PARAM_BLD with the curve name
// and the raw peer public key.
QScopedPointer <OSSL_PARAM_BLD, ScopedPointerParamBldDeleter> paramBuild(OSSL_PARAM_BLD_new());
if (!paramBuild.data()) {
return false;
}

// Set the curve name.
if (!OSSL_PARAM_BLD_push_utf8_string(paramBuild.data(), OSSL_PKEY_PARAM_GROUP_NAME,curveName_.toStdString().c_str(), 0)) {
return false;
}

// Convert the peer hex private key to raw data.
BIGNUM * binPeerPrivateKey = ConvertHexToBigNum(hexPeerPrivateKey);
if(!binPeerPrivateKey)
{
return false;
}

// Set the raw peer private key.
if (!OSSL_PARAM_BLD_push_BN(paramBuild.data(), OSSL_PKEY_PARAM_PRIV_KEY, binPeerPrivateKey)) {
return false;
}

// Convert the peer hex public key to raw data.
QByteArray binPeerPublicKey = ConvertHexToData(hexPeerPublicKey);
if (binPeerPublicKey.size() <= 0) {
return false;
}

// Set the raw peer public key.
if (!OSSL_PARAM_BLD_push_octet_string(paramBuild.data(), OSSL_PKEY_PARAM_PUB_KEY, binPeerPublicKey.data(), binPeerPublicKey.size())) {
return false;
}

// Convert the OSSL_PARAM_BLD to OSSL_PARAM.
QScopedPointer <OSSL_PARAM , ScopedPointerParamDeleter> params(OSSL_PARAM_BLD_to_param(paramBuild.data()));
if (!params.data()) {
return false;
}

// Create a EVP_PKEY context.
QScopedPointer <EVP_PKEY_CTX , ScopedPointerKeyCtxDeleter> peerPrivateKeyCtx(EVP_PKEY_CTX_new_from_name(nullptr, "EC", nullptr));
if (!peerPrivateKeyCtx.data()) {
return false;
}

// Initialize the context.
if (EVP_PKEY_fromdata_init(peerPrivateKeyCtx.data()) <= 0) {
return false;
}

// Create the peer private key object.
EVP_PKEY* tmp = nullptr;
if (EVP_PKEY_fromdata(peerPrivateKeyCtx.data(), &tmp, EVP_PKEY_KEYPAIR, params.data()) <= 0) {
return false;
}
peerPrivateKey->reset(tmp);
return true;
}
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QDebug>


MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this);


QECDHMgr qECDHMgrA,qECDHMgrB;
QString pubA, priA, pubB, priB;

// *****************
// * 【测试1】
// * 产生A的密钥对
// *****************
if(qECDHMgrA.SetCurveName("secp384r1"))
{
if(qECDHMgrA.GenerateKeys())
{
pubA = qECDHMgrA.GetPublicKey();
priA = qECDHMgrA.GetPrivateKey();
qDebug() << "PublicKeyA" << pubA;
qDebug() << "PrivateKeyA" << priA;
}else
{
qDebug() << "GenerateKeys fail" ;
}
}else
{
qDebug() << "The curve is not supported.";
}

// *****************
// * 【测试2】
// * 产生B的密钥对
// *****************
if(qECDHMgrB.SetCurveName("secp384r1"))
{
if(qECDHMgrB.GenerateKeys())
{
pubB = qECDHMgrB.GetPublicKey();
priB = qECDHMgrB.GetPrivateKey();
qDebug() << "PublicKeyB" << pubB;
qDebug() << "PrivateKeyB" << priB;
}else
{
qDebug() << "GenerateKeys fail" ;
}
}else
{
qDebug() << "The curve is not supported.";
}

qDebug() << "*********************************";

// *****************
// * 【测试3】
// * 分别载入公钥、私钥
// *****************
QECDHMgr qECDHMgrA1;
if(qECDHMgrA1.SetCurveName("secp384r1"))
{
// 写入公钥
if(qECDHMgrA1.SetPublicKey(pubA))
{
qDebug() << "PublicKeyA1" << qECDHMgrA1.GetPublicKey();
}else
{
qDebug() << "set pubkeyA1 Error" << qECDHMgrA1.GetLastError();
}

// 写入私钥
if(qECDHMgrA1.SetPrivateKey(priA))
{
qDebug() << "PrivateKeyA1" << qECDHMgrA1.GetPrivateKey();
QString secA1 = qECDHMgrA1.DeriveSharedSecret(pubB);
qDebug() << "SecKeyA1" << secA1;
}else
{
qDebug() << "set prikeyA1 Error" << qECDHMgrA1.GetLastError();
}
}else
{
qDebug() << "The curve is not supported.";
}

qDebug() << "*********************************";


// *****************
// * 【测试4】
// * 一次性载入密钥对(包含公钥和私钥)
// *****************
QECDHMgr qECDHMgrA2;
if(qECDHMgrA2.SetCurveName("secp384r1"))
{
if(qECDHMgrA2.SetKeyPair(pubA, priA))
{
qDebug() << "PrivateKeyA3" << qECDHMgrA2.GetPrivateKey();
qDebug() << "PublicKeyA3" << qECDHMgrA2.GetPublicKey();
}else
{
qDebug() << "set prikeyA3 Error" << qECDHMgrA2.GetLastError();
}
}else
{
qDebug() << "The curve is not supported.";
}


qDebug() << "*********************************";

// *****************
// * 【测试5】
// * 生成对称密钥
// *****************
QECDHMgr qECDHMgrA3, qECDHMgrB3;
if(qECDHMgrA3.SetCurveName("secp384r1"))
{
if(qECDHMgrA3.SetKeyPair(pubA, priA))
{
qDebug() << "PrivateKeyA3" << qECDHMgrA3.GetPrivateKey();
qDebug() << "PublicKeyA3" << qECDHMgrA3.GetPublicKey();
}else
{
qDebug() << "set prikeyA3 Error" << qECDHMgrA3.GetLastError();
}
}else
{
qDebug() << "The curve is not supported.";
}
if(qECDHMgrB3.SetCurveName("secp384r1"))
{
if(qECDHMgrB3.SetKeyPair(pubB, priB))
{
qDebug() << "PrivateKeyB3" << qECDHMgrB3.GetPrivateKey();
qDebug() << "PublicKeyB3" << qECDHMgrB3.GetPublicKey();
}else
{
qDebug() << "set prikeyB3 Error" << qECDHMgrB3.GetLastError();
}
}else
{
qDebug() << "The curve is not supported.";
}
qDebug() << "SecKeyA3" << qECDHMgrA3.DeriveSharedSecret(pubB);
qDebug() << "SecKeyB3" << qECDHMgrB3.DeriveSharedSecret(pubA);
}

MainWindow::~MainWindow()
{
delete ui;
}

执行效果

QT版本基于OpenSSL3的ECDH密钥交换算法_QT