Sample code for 30+ languages & platforms
Unicode C

ECDSA Sign Data and Get Raw R and S Values

See more ECC Examples

Demonstrates getting the raw R and S value of an ECDSA signature.

Chilkat Unicode C Downloads

Unicode C
#include <C_CkCrypt2W.h>
#include <C_CkPrivateKeyW.h>
#include <C_CkPrngW.h>
#include <C_CkEccW.h>
#include <C_CkAsnW.h>
#include <C_CkXmlW.h>
#include <C_CkBinDataW.h>

void ChilkatSample(void)
    {
    BOOL success;
    HCkCrypt2W crypt;
    const wchar_t *hash1;
    HCkPrivateKeyW privKey;
    HCkPrngW prng;
    HCkEccW ecdsa;
    const wchar_t *ecdsaSigBase64;
    HCkAsnW asn;
    HCkXmlW xml;
    HCkBinDataW bd;
    const wchar_t *r;
    const wchar_t *s;

    success = FALSE;

    // This example requires the Chilkat API to have been previously unlocked.
    // See Global Unlock Sample for sample code.

    // To create an ECDSA signature, the data first needs to be hashed.  Then the hash
    // is signed.

    crypt = CkCrypt2W_Create();
    CkCrypt2W_putHashAlgorithm(crypt,L"SHA256");
    CkCrypt2W_putCharset(crypt,L"utf-8");
    CkCrypt2W_putEncodingMode(crypt,L"base64");

    // Hash a string.
    hash1 = CkCrypt2W_hashStringENC(crypt,L"The quick brown fox jumps over the lazy dog");
    wprintf(L"hash1 = %s\n",hash1);

    // -----------------------------------------------------------
    // An ECDSA private key is used for signing.  The public key is for signature verification.
    // Load our ECC private key.
    // Our private key file contains this:

    // 	// -----BEGIN PRIVATE KEY-----
    // 	MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQg3J8q/24D1sEKGdP9
    // 	72MGYElLGpw/a56Y3t6pfON3uhShRANCAATlSmoizyhAwoYZAOuFBATl07/1RR54
    // 	a1Dzfm16grxJe666AGKR+bSs24hk7TEpaeCTvT8YOOM3l+xKFg7zq6Q9
    // 	-----END PRIVATE KEY-----

    privKey = CkPrivateKeyW_Create();
    success = CkPrivateKeyW_LoadPemFile(privKey,L"qa_data/ecc/secp256r1-key-pkcs8.pem");
    if (success != TRUE) {
        wprintf(L"%s\n",CkPrivateKeyW_lastErrorText(privKey));
        CkCrypt2W_Dispose(crypt);
        CkPrivateKeyW_Dispose(privKey);
        return;
    }

    // Sign the hash..
    prng = CkPrngW_Create();
    ecdsa = CkEccW_Create();
    ecdsaSigBase64 = CkEccW_signHashENC(ecdsa,hash1,L"base64",privKey,prng);
    if (CkEccW_getLastMethodSuccess(ecdsa) != TRUE) {
        wprintf(L"%s\n",CkEccW_lastErrorText(ecdsa));
        CkCrypt2W_Dispose(crypt);
        CkPrivateKeyW_Dispose(privKey);
        CkPrngW_Dispose(prng);
        CkEccW_Dispose(ecdsa);
        return;
    }

    // The ECDSA signature is ASN.1 that contains a sequence of 2 large integers (r and s)
    // For example:
    // SEQUENCE (2 elem)
    //   INTEGER (255 bit) 792134D9B4AD82D5431ED03835A88E2596EB35E5B13054BD9B05A0069281ACC9
    //   INTEGER (255 bit) 481E758CC1E3CBF825537EC3D9A2CA627E5FAD1137BBEA65DF38658DCB0A9ED5

    wprintf(L"Base64 ECDSA signature = %s\n",ecdsaSigBase64);

    // If the raw R and S values are needed, here's how to get them:
    asn = CkAsnW_Create();
    success = CkAsnW_LoadEncoded(asn,ecdsaSigBase64,L"base64");
    if (success == FALSE) {
        wprintf(L"%s\n",CkAsnW_lastErrorText(asn));
        CkCrypt2W_Dispose(crypt);
        CkPrivateKeyW_Dispose(privKey);
        CkPrngW_Dispose(prng);
        CkEccW_Dispose(ecdsa);
        CkAsnW_Dispose(asn);
        return;
    }

    // The R and X will be in hexidecimal in the XML.
    xml = CkXmlW_Create();
    CkXmlW_LoadXml(xml,CkAsnW_asnToXml(asn));
    wprintf(L"%s\n",CkXmlW_getXml(xml));

    // The XML looks like this:
    // <sequence>
    // <int>792134D9B4AD82D5431ED03835A88E2596EB35E5B13054BD9B05A0069281ACC9</int>
    // <int>481E758CC1E3CBF825537EC3D9A2CA627E5FAD1137BBEA65DF38658DCB0A9ED5</int>
    // </sequence>

    // Copy raw R and S hex values into a Chilkat BinData object.
    bd = CkBinDataW_Create();
    r = CkXmlW_getChildContent(xml,L"int[0]");
    s = CkXmlW_getChildContent(xml,L"int[1]");
    CkBinDataW_AppendEncoded(bd,r,L"hex");
    CkBinDataW_AppendEncoded(bd,s,L"hex");

    wprintf(L"Number of bytes in bd: %d\n",CkBinDataW_getNumBytes(bd));


    CkCrypt2W_Dispose(crypt);
    CkPrivateKeyW_Dispose(privKey);
    CkPrngW_Dispose(prng);
    CkEccW_Dispose(ecdsa);
    CkAsnW_Dispose(asn);
    CkXmlW_Dispose(xml);
    CkBinDataW_Dispose(bd);

    }