概要

MD5は任意のデータ長から128bitのほぼ一意の128bitのハッシュ値を出力する関数である。電子署名やパスワードの保存、電子メールのSMTPサーバーのCRAM-MD5認証、ファイルの改ざんチェック等に用いられている。
本プログラムは、WindowsのCryptography APIを用い_tmain関数の変数passで定義されている文字列からMD5ハッシュを算出し16進数で出力します。
コマンドラインで動作します。

テスト環境

コンパイラ

Visual C++ 2013 Express 32bit/64bit
マルチバイト/Unicode
プラットフォームツールセット Visual Studio 2013 - Windows XP (v120_xp)

実行環境

Windows 7 Enterprise Service Pack 1 64bit(Sandy Bridge-E)

MD5のアルゴリズム

MD5のアルゴリズムについてはMD5ハッシュを計算を参照してください。

プログラムソースの概要

_tmain

HASH_UINT128オブジェクトを作成します。
passにMD5ハッシュを算出したい文字列を指定します。
UNICODEでコンパイルした場合は、TCHAR(WCHAR)をchar型に変換します。
md5Hashed関数により文字列からMDハッシュ値を計算しHASH_UINT128に格納します。
hexメンバー関数により16進文字列を取得し_tprintfにより標準出力にハッシュ値を出力します。

mdHashed関数

引数szPasswordで示され、引数dwLengthの長さの文字列からMD5ハッシュを計算しHASH_UINT128に結果を格納します。
CryptAcquireContext APIによりCSP(CryptoGraphy Service Provider)を準備します。
CryptCreateHash APIによりMD5ハッシュ計算のインスタンスを作成します。
CryptHashData APIによりMD5ハッシュを計算します。
CryptGetHashParam APIにより計算されたMD5ハッシュ値を取得しHASH_UINT128に格納します。

HASH_UINT128

ハッシュ値の保存及びハッシュ値から16進数の文字列を取得するための共用体です。

ソースコード

md5_crypt.cpp


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

union HASH_UINT128{
        UINT64  u64[2];
        UINT32  u32[4];
        BYTE    u8[16];
        void hex(TCHAR* s){
                _stprintf_s(s, 33, _TEXT("%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x%0.2x"), u8[0], u8[1], u8[2], u8[3], u8[4], u8[5], u8[6], u8[7], u8[8], u8[9], u8[10], u8[11], u8[12], u8[13], u8[14], u8[15]);
        }
};

bool md5Hashed(char* szPassword, DWORD dwLength, HASH_UINT128* u128){
        HCRYPTPROV hProv = 0;
        HCRYPTHASH hHash = 0;
        if (!CryptAcquireContext(&hProv, NULL, NULL, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT)){
                MessageBox(0, _TEXT("CryptAcquireContext"), _TEXT("Error"), MB_OK);
                return false;
        }
        if (!CryptCreateHash(hProv, CALG_MD5, 0, 0, &hHash)){
                MessageBox(0, _TEXT("CryptCreateHash"), _TEXT("Error"), MB_OK);
                CryptReleaseContext(hProv, 0);
                return false;
        }
        if (!CryptHashData(hHash, (BYTE*)szPassword, dwLength, 0)){
                MessageBox(0, _TEXT("CryptCreateHash"), _TEXT("Error"), MB_OK);
                CryptDestroyHash(hHash);
                CryptReleaseContext(hProv, 0);
                return false;
        }
        DWORD dwRead = 16;
        if (!CryptGetHashParam(hHash, HP_HASHVAL, (BYTE*)u128, &dwRead, 0)){
                MessageBox(0, _TEXT("CryptGetHashParam"), _TEXT("Error"), MB_OK);
                CryptDestroyHash(hHash);
                CryptReleaseContext(hProv, 0);
                return false;
        }
        CryptDestroyHash(hHash);
        CryptReleaseContext(hProv, 0);
        return true;
}

void _tmain(void){
        HASH_UINT128 hash;
        TCHAR* pass = _TEXT("abc");
        TCHAR hexs[34];
#ifdef UNICODE
        char sjis[256];
        WideCharToMultiByte(932, 0, pass, -1, sjis, sizeof(sjis), NULL, NULL);
        md5Hashed(sjis, (int)strlen(sjis), &hash);
#else
        md5Hashed(pass, (int)_tcslen(pass), &hash);
#endif
        hash.hex(hexs);
        _tprintf(_TEXT("%s\nmd5 %s\n"), pass,hexs);
}

実行ファイルとソースファイルのダウンロード(Visual C++ 2013 マルチバイトでコンパイル)