Click here to Skip to main content
Licence Ms-PL
First Posted 10 Dec 2001
Views 101,425
Bookmarked 32 times

Obtain the plain text session key using CryptoAPI

By | 10 Dec 2001 | Article
Generally it's very important to obtain the value of session keys. However, the Microsoft Cryptographic Providers (Base and Enhanced) do not support this feature. CryptExportKey() and CryptImportKey() require a valid key handle to encrypt and decrypt the session key, respectively.

Introduction

Generally it's very important to obtain the value of session keys. However, the Microsoft Cryptographic Providers (Base and Enhanced) does not support this feature. CryptExportKey() and CryptImportKey() require a valid key handle to encrypt and decrypt the session key, respectively. MSDN shows a way to do this using a exponent-of-one private key. This article shows a better way to perform the same process. This way is faster and more comprehensive.

It's ready to run, but you must set the following parameters at Project -> Settings (Visual Studio 6.0 ) :

  • Add in C++/Preprocessor definitions: _WIN32_WINNT=0x0500, _CRYPT32_(WIN2K) or _WIN32_WINNT=0x0400, _CRYPT32_(NT4)
  • And Link this library  crypt32.lib

Code Listing

#include <windows.h>
#include <wincrypt.h>


#define KEY_PAIR_SIZE     dwSize - 12
#define SESSION_KEY_SIZE  dwKeyMaterial

void main()
{

    HCRYPTPROV hProv = 0;
    HCRYPTKEY hExchangeKeyPair = 0;
    HCRYPTKEY hSessionKey = 0;
    BYTE *pbKeyMaterial  = NULL;
    DWORD dwKeyMaterial ;   
    BYTE *pbExportedKeyBlob = NULL;
    BYTE *pbEncryptedKey    = NULL;
    DWORD dwSize;
    unsigned int c;

    __try
    {

        if (!CryptAcquireContext( &hProv, "Container Name", 
            MS_ENHANCED_PROV , PROV_RSA_FULL, CRYPT_MACHINE_KEYSET ))
        {
            __leave;
        }

        //---------------------------------------------------
        //Creating a session key. In this sample we'll use a 
        //3DES key with 168 bits

        if (!CryptGenKey(hProv,CALG_3DES,CRYPT_EXPORTABLE,
            &hSessionKey))
        {
            __leave;
        }

        //---------------------------------------------------
        //Getting a handle to the Exchange Key pair

        if (!CryptGetUserKey( hProv, AT_KEYEXCHANGE, 
            &hExchangeKeyPair))
        {
            __leave;
        }

        //--------------------------------------------------------
        //Encrypting the session key with the public key part
        //of the keypair. The first call gets the size necessary to 
        //hold the encrypted session key and the second exports it.

        if (!CryptExportKey( hSessionKey, hExchangeKeyPair, 
            SIMPLEBLOB, 0, NULL, &dwSize))
        {
            __leave;
        }

        pbExportedKeyBlob = new BYTE[dwSize];

        if (!CryptExportKey( hSessionKey, hExchangeKeyPair, SIMPLEBLOB, 
            0, pbExportedKeyBlob,  &dwSize))
        {
            __leave;
        }

        //--------------------------------------------------------
        //Let's  remove the first 12 bytes of Blob information 

        pbEncryptedKey  =  new BYTE [KEY_PAIR_SIZE];  

        for ( c = 0 ; c < KEY_PAIR_SIZE ; c++ )
        {
            pbEncryptedKey[c] =  pbExportedKeyBlob[c+12]; 
        }

        //--------------------------------------------------------
        //Now it's time to get the value of the session key, we'll
        //use the private part of the key pair. 

        if (!CryptDecrypt( hExchangeKeyPair,0, TRUE, 0,  
            pbEncryptedKey, &dwKeyMaterial))
        {
            __leave;
        }

        //-------------------------------------------------------
        //The pbKeyMaterial is the value of the key

        pbKeyMaterial = new BYTE[ SESSION_KEY_SIZE ];

        for ( c = 0; c <  SESSION_KEY_SIZE ; c-- )
        {
            pbKeyMaterial[c] = pbEncryptedKey[c];    
        }

        //-------------------------------------------------------
        //The key is in big-endian format, because that's the way
        //the encryption block is built.

    }
    __finally
    {
        if (pbKeyMaterial ) LocalFree(pbKeyMaterial );
        if (hSessionKey) CryptDestroyKey(hSessionKey);
        if (hExchangeKeyPair) CryptDestroyKey(hExchangeKeyPair);
        if (hProv) 
        {  
            CryptReleaseContext(hProv, 0);
        }

    }

} // End Main

License

This article, along with any associated source code and files, is licensed under The Microsoft Public License (Ms-PL)

About the Author

Raphael Amorim

Software Developer (Senior)
Instituto Atlântico
Brazil Brazil

Member

Follow on Twitter Follow on Twitter
Senior Software Developer Engineer @ Instituto Atlântico
 
Summary:
 
• Has Acted as Software Architect and Technical Leader in many projects
• Good knowledge of Windows Internals
• More than 9 years of software design and development experience in Visual C/C++.
• More than 8 years of experience in Security/Cryptography software design and programming.
• Good knowledge of .NET framework and C# language, also using ASP.NET.
• Good knowledge of Lua Scripting Language
• Good knowledge of Objective C and Cocoa.
• Good knowledge of Java language.
• Good knowledge of MacOSX/Linux programming in C++
• Good knowledge of Brew Framework for CDMA cell phone software development (using C)
• Some knowledge of Carbon and Qt application development C++.
 
Specialist in:
Visual C++® and C, MFC, ATL, COM, STL, ActiveX, Lua, Cocoa, Carbon and Qt application development, C# and Java (J2SE and J2ME), Brew Framework for CDMA, Linux Programming in C++, PocketPC (embedded C++ and .NET Compact Framework) , Device Driver Development , SQL Server 6.5/7/2000/2005, MySQL, IBM DB2 and Access Visual C++ 6.0, Visual Studio .NET (all versions), Eclipse, NetBeans, XCode, Qt Creator and Carbide. Microsoft Windows NT to 7, Windows CE 3.0/5.0/6.0, Mac OS X and Linux.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board. (secure sign-in)
 
Search this forum  
 FAQ
    Noise  Layout  Per page   
  Refresh
GeneralHere is a link to some associated documentation PinmemberPaul Sanders (AlpineSoft)3:52 21 Oct '07  
GeneralKudos PinmemberAshutosh Arya19:59 2 Oct '07  
QuestionHow to use publickey of another user Pinmemberchakra_r10:21 24 Mar '04  
AnswerRe: How to use publickey of another user PinsussAnonymous15:26 24 Mar '04  
Generalerror include wincrypt.h PinmemberLoLem23:18 23 Mar '04  
GeneralRe: error include wincrypt.h PinmemberRaphael Amorim1:07 22 Apr '04  
GeneralThanks, Bug & Suggestion PinsussC. Jayachandran21:57 20 Nov '03  
GeneralRe: Thanks, Bug & Suggestion PinmemberRaphael Amorim16:02 19 Apr '05  
QuestionWhy CryptGetUserKey returned the same key every time called at one machine? Pinmemberzy_ball15:02 10 Nov '03  
AnswerRe: Why CryptGetUserKey returned the same key every time called at one machine? PinmemberRaphael Amorim9:33 24 Nov '03  
GeneralRe: Why CryptGetUserKey returned the same key every time called at one machine? Pinmemberzy_ball13:38 24 Nov '03  
GeneralWindows XP != Windows 2000 PinmemberWilliam huang12:10 8 Aug '03  
GeneralImporting it back PinmemberThomas George3:41 11 Mar '03  
GeneralRe: Importing it back PinmemberRaphael Amorim5:30 11 Mar '03  
GeneralCAPI in a vc++ app PinmemberThe Eclypse3:27 12 Dec '01  
GeneralRe: Could you be an more lamer? Pinmemberd0:04 15 Dec '01  
GeneralRe: Could you be an more lamer? PinmemberJohn Simmons / outlaw programmer1:30 15 Dec '01  
GeneralRe: Could you be an more lamer? PinmemberRaphael Amorim9:23 15 Dec '01  

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

Permalink | Advertise | Privacy | Mobile
Web01 | 2.5.120517.1 | Last Updated 11 Dec 2001
Article Copyright 2001 by Raphael Amorim
Everything else Copyright © CodeProject, 1999-2012
Terms of Use
Layout: fixed | fluid