概要

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 マルチバイトでコンパイル)