C++使用CNG从模数和指数导入RSA公钥

标签 c++ cryptography cng

我正在尝试使用 CNG 通过作为参数提供的公钥来加密一些数据。调用 NCryptImportKey 函数时,出现 NTE_BAD_DATA 错误 which isn't listed in the msdn page.

我的代码:

#include <iostream>
#include <Windows.h>
#include <Bcrypt.h>
#include <Ntstatus.h>
#include <string>
#include <vector>
#include "base64.h"

using std::string;
using std::vector;

struct MyRSAPublicBlob {
    BCRYPT_RSAKEY_BLOB blob;
    BYTE exponent[3];
    BYTE modulus[128];
    MyRSAPublicBlob(const vector<BYTE>& mod, const vector<BYTE>& exp)
    {
        blob.BitLength = (128 + 3) * 8;
        blob.Magic = BCRYPT_RSAPUBLIC_MAGIC;
        blob.cbModulus = 128;
        blob.cbPublicExp = 3;
        for (size_t i = 0; i < mod.size(); ++i) //copy BigEndian
            modulus[i] = mod[mod.size() - 1 - i];
        for (size_t i = 0; i < exp.size(); ++i) //copy BigEndian
            exponent[i] = exp[exp.size() - 1 - i];
    }
    MyRSAPublicBlob() { ; }

};

MyRSAPublicBlob b;

bool RSA_encrypt() {
    SECURITY_STATUS stat;
    NCRYPT_PROV_HANDLE hProv;
    NCRYPT_KEY_HANDLE hKey;

    stat = NCryptOpenStorageProvider(&hProv, MS_KEY_STORAGE_PROVIDER, 0);
    if (ERROR_SUCCESS != stat) {
        std::cout << "failed in NCryptOpenStorageProvider: " << GetLastError() << std::endl;
        return false;
    }
    stat = NCryptImportKey(hProv,
        NULL,
        BCRYPT_RSAPUBLIC_BLOB,
        NULL,
        &hKey,
        (PBYTE)&b.blob,
        sizeof(b),
        0);

    if (ERROR_SUCCESS != stat) {
        std::cout << "failed in NCryptImportKey: " << GetLastError() << std::endl;
        return false;
    }

我如何构建 MyRSAPublicBlob 的示例:

string PubKeyModulus = "yVUndgQFuB5Z5FgC0/WgWCg6Y8VuB582avGjQDdeoJDa1+RBKCyXo700sAMSGjM/bVakOlFqvCsVFNBysx1CH731CDb2DR1a0bsmYmDQ9d0ZHX+AOohVDIx9mc7bkDQZoEFpe9NqFsu95Y9yktpl1JKPmKyLOFgufGJYYvQyoOM=";
string PubKeyExp = "AQAB";

vector<BYTE> PubKeyModulus_bin = base64_decode(PubKeyModulus);
vector<BYTE> PubKeyExp_bin = base64_decode(PubKeyExp);
struct MyRSAPublicBlob c(PubKeyModulus_bin, PubKeyExp_bin);
b = c;

我做错了什么吗?

最佳答案

RSA 公钥 BLOB (BCRYPT_RSAPUBLIC_BLOB) 在连续内存中具有以下格式。尝试使用#pragma pack 来避免任何填充问题。例如,

#pragma pack(push, 1)
struct MyRSAPublicBlob {
    BCRYPT_RSAKEY_BLOB blob;
    BYTE exponent[3];
    BYTE modulus[128];
...};
#pragma pack(pop)

关于C++使用CNG从模数和指数导入RSA公钥,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44266617/

相关文章:

c++ - 使用 Gloox C/C++ 的 XMPP 握手流程

c++ - 在 Qt 4.10.2 中创建 .exe 文件

python - 如何修复 scrypt 的无效参数组合?

java - 使用KSP/CNG登录java

encryption - Microsoft CNG BCryptEncrypt 返回密文 == 明文

c# - 与 ECDsaCng 算法一起使用的 key 必须具有 ECDsa 算法组

c++ - Make - 无所事事(另一个)

c++ - STL 排序进入无限循环?

javascript - 在客户端使用私钥签署数据(javascript)

java - 使用 DES 算法对 Java 中的文件进行加密和解密