add test_crypto from Steam

Heavily modified, since we only use Symmetric*WithIV in
GameNetworkingSockets.

Signed-off-by: Steven Noonan <steven@valvesoftware.com>
This commit is contained in:
Steven Noonan
2019-01-10 16:34:54 -08:00
parent 9e34ec3da4
commit 045a9b4c52
2 changed files with 693 additions and 0 deletions
+53
View File
@@ -1,8 +1,61 @@
find_package(OpenSSL REQUIRED)
if (USE_LIBSODIUM)
find_package(sodium REQUIRED)
endif()
add_executable(
test_connection
test_connection.cpp)
target_link_libraries(test_connection GameNetworkingSockets)
set(TEST_CRYPTO_SRC
"test_crypto.cpp"
"../src/tier0/dbg.cpp"
"../src/common/crypto.cpp"
"../src/common/crypto_25519.cpp"
"../src/common/crypto_25519_libsodium.cpp"
"../src/common/crypto_textencode.cpp"
"../src/common/keypair.cpp"
"../src/common/opensslwrapper.cpp"
"../src/tier0/cpu.cpp"
"../src/tier0/platformtime.cpp"
"../src/tier1/utlbuffer.cpp"
"../src/tier1/utlmemory.cpp"
"../src/vstdlib/strtools.cpp")
if (NOT USE_LIBSODIUM)
set(TEST_CRYPTO_SRC ${TEST_CRYPTO_SRC}
"../src/external/curve25519-donna/curve25519.c"
"../src/external/curve25519-donna/curve25519_VALVE_sse2.c"
"../src/external/ed25519-donna/ed25519_VALVE.c"
"../src/external/ed25519-donna/ed25519_VALVE_sse2.c")
endif()
add_executable(
test_crypto
${TEST_CRYPTO_SRC}
)
target_compile_definitions(test_crypto PRIVATE POSIX LINUX GNUC)
target_include_directories(test_crypto PRIVATE ../src ../src/public ../src/common ../include)
target_link_libraries(test_crypto OpenSSL::Crypto)
if(CMAKE_CXX_COMPILER_ID MATCHES "GNU"
OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
target_compile_definitions(test_crypto PRIVATE GNUC GNU_COMPILER)
endif()
if(CMAKE_SYSTEM_NAME MATCHES Linux)
target_compile_definitions(test_crypto PRIVATE POSIX LINUX)
elseif(CMAKE_SYSTEM_NAME MATCHES Darwin)
target_compile_definitions(test_crypto PRIVATE POSIX OSX)
elseif(CMAKE_SYSTEM_NAME MATCHES Windows)
target_compile_definitions(test_crypto PRIVATE WIN32)
endif()
if(USE_LIBSODIUM)
target_compile_definitions(test_crypto PRIVATE USE_LIBSODIUM)
target_include_directories(test_crypto PRIVATE ${sodium_INCLUDE_DIR})
target_link_libraries(test_crypto ${sodium_LIBRARY_RELEASE})
endif(USE_LIBSODIUM)
#add_executable(
# test_flat
# test_flat.c)
+640
View File
@@ -0,0 +1,640 @@
#include <assert.h>
#include <tier1/utlbuffer.h>
#include <crypto.h>
#include <crypto_constants.h>
// I copied these tests from the Steam branch.
// A little compatibility glue so I don't have to make any changes to them.
#define CHECK(x) Assert(x)
#define CHECK_EQUAL(a,b) Assert((a)==(b))
const int k_cSmallBuff = 255; // smallish buffer
const int k_cMedBuff = 1024;
static unsigned char Q_nibble( char c )
{
if ( ( c >= '0' ) &&
( c <= '9' ) )
{
return (unsigned char)(c - '0');
}
if ( ( c >= 'A' ) &&
( c <= 'F' ) )
{
return (unsigned char)(c - 'A' + 0x0a);
}
if ( ( c >= 'a' ) &&
( c <= 'f' ) )
{
return (unsigned char)(c - 'a' + 0x0a);
}
// received an invalid character, and no real way to return an error
// AssertMsg1( false, "Q_nibble invalid hex character '%c' ", c );
return 0;
}
#define min(x,y) (((x) > (y)) ? (y) : (x))
//-----------------------------------------------------------------------------
static void V_hextobinary( char const *in, int numchars, byte *out, int maxoutputbytes )
{
int len = V_strlen( in );
numchars = min( len, numchars );
// Make sure it's even
numchars = ( numchars ) & ~0x1;
DbgAssert( V_isvalidhex( in, numchars ) );
memset( out, 0x00, maxoutputbytes );
byte *p;
int i;
p = out;
for ( i = 0;
( i < numchars ) && ( ( p - out ) < maxoutputbytes );
i+=2, p++ )
{
*p = ( Q_nibble( in[i] ) << 4 ) | Q_nibble( in[i+1] );
}
}
//-----------------------------------------------------------------------------
// Purpose: Tests cryptography hex encoding
//-----------------------------------------------------------------------------
void TestCryptoEncoding()
{
bool bRet;
// If you change the source data you'll need to change the expected encoding
// output strings further down!
uint8 rgubData[] = { 0x14, 0xfe, 0x26, 0x19, 0x54, 0x78, 0x00, 0x35, 0x19, 0xa9, 0x54, 0x4e, 0x99 };
uint cubData = V_ARRAYSIZE( rgubData );
char rgchEncodedData[k_cMedBuff];
uint cchEncodedData = V_ARRAYSIZE( rgchEncodedData );
uint8 rgubDecodedData[k_cSmallBuff];
uint cubDecodedData = V_ARRAYSIZE( rgubDecodedData );
// HEX
// encode some data
bRet = CCrypto::HexEncode( rgubData, cubData, rgchEncodedData, cchEncodedData );
CHECK( bRet ); // must succeed
// decode the data
bRet = CCrypto::HexDecode( rgchEncodedData, rgubDecodedData, &cubDecodedData );
CHECK( bRet ); // must succeed
CHECK( cubDecodedData == cubData ); // must be of correct size
bRet = (0 == V_memcmp( rgubData, rgubDecodedData, cubData ));
CHECK( bRet ); // decoded data must match original
// test the documented, if questionable, permissiveness of hexdecode. Note that Crypto++
// documentation claims that the last partial byte should have been parsed as E0, but this
// has been disproven by testing with 5.6.1 and 5.6.2. this test verifies that there is no
// change in the behavior of partial strings, if the algorithm should be updated.
cubDecodedData = V_ARRAYSIZE( rgubDecodedData );
bRet = CCrypto::HexDecode( "x,F\nF1\t ,2\t~E ", rgubDecodedData, &cubDecodedData );
CHECK( bRet ); // must succeed
CHECK( cubDecodedData == 2 && rgubDecodedData[0] == 0xFF && rgubDecodedData[1] == 0x12 );
// this hilarious string is offered up for laughs to verify that we remain as broken as ever.
// Crypto++'s documentation claims that it will "correctly" parse this string as FF 12 E0.
// In reality, it does the dumb/obvious thing and discards the 'x' and ' ' characters and
// parses the '0's, resulting in 4 bytes. There is no evidence that any version of Crypto++
// has ever matched the documentation and actually done "smart" prefix skipping.
cubDecodedData = V_ARRAYSIZE( rgubDecodedData );
bRet = CCrypto::HexDecode( "0xFF 0x12 0xE", rgubDecodedData, &cubDecodedData );
CHECK( bRet );
CHECK( cubDecodedData == 4 && rgubDecodedData[0] == 0x0F && rgubDecodedData[1] == 0xF0 && rgubDecodedData[2] == 0x12 && rgubDecodedData[3] == 0x0E );
// BASE64
bRet = CCrypto::Base64Encode( rgubData, cubData, rgchEncodedData, cchEncodedData );
CHECK( bRet ); // must succeed
// decode the data
cubDecodedData = V_ARRAYSIZE( rgubDecodedData );
bRet = CCrypto::Base64Decode( rgchEncodedData, rgubDecodedData, &cubDecodedData );
CHECK( bRet ); // must succeed
CHECK( cubDecodedData == cubData ); // must be of correct size
bRet = ( 0 == V_memcmp( rgubData, rgubDecodedData, cubData ) );
CHECK( bRet ); // decoded data must match original
// Test empty string handling
{
char buf[4] = { 1, 1, 1, 1 };
uint32 bufSize = sizeof(buf);
bRet = CCrypto::Base64Encode( NULL, 0, buf, &bufSize, NULL );
CHECK( bRet );
CHECK_EQUAL( bufSize, 0 ); // zero characters written
CHECK_EQUAL( buf[0], 0 ); // terminating null written to buffer
}
{
uint8 buf[4] = { 1, 1, 1, 1 };
uint32 bufSize = sizeof(buf);
bRet = CCrypto::Base64Decode( NULL, 0, buf, &bufSize, true );
CHECK( bRet );
CHECK_EQUAL( bufSize, 0 );
CHECK_EQUAL( buf[0], 1 ); // shouldn't have written to buf at all
bufSize = sizeof(buf);
bRet = CCrypto::Base64Decode( "", buf, &bufSize, true );
CHECK( bRet );
CHECK_EQUAL( bufSize, 0 );
CHECK_EQUAL( buf[0], 1 ); // shouldn't have written to buf at all
}
// Test decoding error
{
uint8 buf[4] = { 1, 1, 1, 1 };
uint32 bufSize = sizeof(buf);
bRet = CCrypto::Base64Decode( "AAAA!@#$%^&*()_+|<>?:;'[]{}\\/,.", buf, &bufSize, false );
CHECK( bRet == false );
// Should have decoded 3 null bytes and then failed
CHECK_EQUAL( bufSize, 3 );
CHECK_EQUAL( buf[0], 0 );
CHECK_EQUAL( buf[1], 0 );
CHECK_EQUAL( buf[2], 0 );
CHECK_EQUAL( buf[3], 1 );
}
}
void TestSymmetricCrypto()
{
bool bRet;
uint8 rgubIV[ k_nSymmetricBlockSize ];
uint8 rgubKey[k_nSymmetricKeyLen];
uint8 rgubKey2[k_nSymmetricKeyLen];
// Generate a couple of random keys and an IV.
CCrypto::GenerateRandomBlock( rgubKey, V_ARRAYSIZE( rgubKey ) );
CCrypto::GenerateRandomBlock( rgubKey2, V_ARRAYSIZE( rgubKey2 ) );
CCrypto::GenerateRandomBlock( rgubIV, V_ARRAYSIZE( rgubIV ) );
const char rgchSrc[] = "This is a test of symmetric encryption! It is at least 160 bytes long "
"to trigger at least two iterations of the 64-byte AES-NI optimized path on systems that "
"support AES-NI instructions. Blah blah blah blah blah. Blah blah blah blah blah. 12345 ";
uint8 rgubEncrypted[k_cMedBuff];
uint cubEncrypted = V_ARRAYSIZE( rgubEncrypted );
uint8 rgubOutput[k_cMedBuff];
// Repeat the same test but this time using ...WithIV so as to not prepend the IV at all.
cubEncrypted = V_ARRAYSIZE( rgubEncrypted );
bRet = CCrypto::SymmetricEncryptWithIV( (const uint8*)rgchSrc, V_ARRAYSIZE( rgchSrc ),
rgubIV, sizeof(rgubIV),
rgubEncrypted, &cubEncrypted,
rgubKey, V_ARRAYSIZE( rgubKey ) );
CHECK( bRet ); // must succeed
uint cubOutput = V_ARRAYSIZE( rgubOutput );
bRet = CCrypto::SymmetricDecryptWithIV( rgubEncrypted, cubEncrypted, rgubIV, sizeof(rgubIV),
rgubOutput, &cubOutput, rgubKey, V_ARRAYSIZE( rgubKey ) );
CHECK( bRet ); // must succeed
CHECK( cubOutput == V_ARRAYSIZE( rgchSrc ) ); // output length must be same as input length
CHECK( !V_strcmp( rgchSrc, (const char *) rgubOutput ) ); // output must be the same as input
//
// In-place decryption. Make sure that we get correct results if the destination buffer and
// source buffer are identical.
//
uint8 rgubInplace[k_cMedBuff];
uint cubInplace = V_ARRAYSIZE( rgubInplace );
// Now try ...WithIV in place as well
V_memcpy( rgubInplace, rgchSrc, V_ARRAYSIZE(rgchSrc) );
cubInplace = V_ARRAYSIZE( rgubInplace );
bRet = CCrypto::SymmetricEncryptWithIV( rgubInplace, V_ARRAYSIZE( rgchSrc ), rgubIV, sizeof(rgubIV), rgubInplace,
&cubInplace, rgubKey, V_ARRAYSIZE( rgubKey ) );
CHECK( bRet );
// In place encryption with ...WithIV should result in identical ciphertext again
CHECK( cubEncrypted == cubInplace );
CHECK( !V_memcmp( rgubEncrypted, rgubInplace, cubEncrypted ) );
// In-place ...WithIV decryption
bRet = CCrypto::SymmetricDecryptWithIV( rgubInplace, cubInplace, rgubIV, sizeof( rgubIV ), rgubInplace,
&cubInplace, rgubKey, V_ARRAYSIZE( rgubKey ) );
CHECK( bRet );
CHECK_EQUAL( cubInplace, V_ARRAYSIZE( rgchSrc ) );
CHECK( !V_strcmp( rgchSrc, (const char *)rgubInplace ) );
}
//-----------------------------------------------------------------------------
// Purpose: Test elliptic-curve primitives (ed25519 signing, curve25519 key exchange)
//-----------------------------------------------------------------------------
void TestEllipticCrypto()
{
// test vectors from curve25519 reference impl
const char rgchAlicePriv[] = "77076d0a7318a57d3c16c17251b26645df4c2f87ebc0992ab177fba51db92c2a";
const char rgchAlicePub[] = "8520f0098930a754748b7ddcb43ef75a0dbf3a0d26381af4eba4a98eaa9b4e6a";
const char rgchBobPriv[] = "5dab087e624a8a4b79e17f8b83800ee66f3bb1292618b6fd1c2f8b27ff88e0eb";
const char rgchBobPub[] = "de9edb7d7b7dc1b4d35b61c2ece435373f8343c85b78674dadfc7e146f882b4f";
const char rgchExpectSharedPreHash[] = "4a5d9d5ba4ce2de1728e3bf480350f25e07e21c947d19e3376f09b3c1e161742";
uint8 buf[64];
V_hextobinary( rgchAlicePub, 64, buf, 32 );
V_hextobinary( rgchAlicePriv, 64, buf + 32, 32 );
CECKeyExchangePrivateKey alicePriv, bobPriv;
CECKeyExchangePublicKey alicePub, bobPub;
alicePriv.Set( buf, 64 );
alicePub.SetFromHexEncodedString( rgchAlicePub );
CHECK( alicePriv.MatchesPublicKey( alicePub ) );
CECKeyExchangePrivateKey testRebuildAlice;
testRebuildAlice.RebuildFromPrivateData( buf + 32 );
CHECK( testRebuildAlice == alicePriv && testRebuildAlice.MatchesPublicKey( alicePub ) );
V_hextobinary( rgchBobPub, 64, buf, 32 );
V_hextobinary( rgchBobPriv, 64, buf + 32, 32 );
bobPriv.Set( buf, 64 );
bobPriv.GetPublicKey( &bobPub );
V_hextobinary( rgchExpectSharedPreHash, 64, buf, 32 );
SHA256Digest_t expectedResult;
CCrypto::GenerateSHA256Digest( buf, 32, &expectedResult );
SHA256Digest_t aliceSharedSecret;
SHA256Digest_t bobSharedSecret;
CCrypto::PerformKeyExchange( alicePriv, bobPub, &aliceSharedSecret );
CCrypto::PerformKeyExchange( bobPriv, alicePub, &bobSharedSecret );
CHECK( V_memcmp( expectedResult, aliceSharedSecret, sizeof(SHA256Digest_t) ) == 0 );
CHECK( V_memcmp( expectedResult, bobSharedSecret, sizeof(SHA256Digest_t) ) == 0 );
// test key extraction and comparison operations
CECKeyExchangePublicKey testPubFromPriv;
alicePriv.GetPublicKey( &testPubFromPriv );
CHECK( testPubFromPriv == alicePub );
CHECK( testPubFromPriv != bobPub );
CHECK( alicePriv.MatchesPublicKey( testPubFromPriv ) );
CHECK( !bobPriv.MatchesPublicKey( testPubFromPriv ) );
// test key exchange with random keys
alicePriv.Wipe();
alicePub.Wipe();
bobPriv.Wipe();
bobPub.Wipe();
memset( aliceSharedSecret, 0, sizeof( aliceSharedSecret ) );
memset( bobSharedSecret, 0xFF, sizeof( bobSharedSecret ) );
CCrypto::GenerateKeyExchangeKeyPair( &alicePub, &alicePriv );
CCrypto::GenerateKeyExchangeKeyPair( &bobPub, &bobPriv );
// alice and bob send each other only their public keys.
CCrypto::PerformKeyExchange( alicePriv, bobPub, &aliceSharedSecret );
CCrypto::PerformKeyExchange( bobPriv, alicePub, &bobSharedSecret );
// alice and bob should have computed the same shared secret.
CHECK( V_memcmp( aliceSharedSecret, bobSharedSecret, sizeof( bobSharedSecret ) ) == 0 );
// test vectors from ed25519 reference impl
const char rgchSignPriv[] = "b18e1d0045995ec3d010c387ccfeb984d783af8fbb0f40fa7db126d889f6dadd";
const char rgchSignPub[] = "77f48b59caeda77751ed138b0ec667ff50f8768c25d48309a8f386a2bad187fb";
const char rgchMessageToSign[] = "916c7d1d268fc0e77c1bef238432573c39be577bbea0998936add2b50a653171"
"ce18a542b0b7f96c1691a3be6031522894a8634183eda38798a0c5d5d79fbd01"
"dd04a8646d71873b77b221998a81922d8105f892316369d5224c9983372d2313"
"c6b1f4556ea26ba49d46e8b561e0fc76633ac9766e68e21fba7edca93c4c7460"
"376d7f3ac22ff372c18f613f2ae2e856af40";
const char rgchExpected[] = "6bd710a368c1249923fc7a1610747403040f0cc30815a00f9ff548a896bbda0b"
"4eb2ca19ebcf917f0f34200a9edbad3901b64ab09cc5ef7b9bcc3c40c0ff7509";
V_hextobinary( rgchSignPub, 64, buf, 32 );
V_hextobinary( rgchSignPriv, 64, buf + 32, 32 );
CECSigningPrivateKey signPriv;
CECSigningPublicKey signPub;
signPriv.Set( buf, 64 );
signPub.SetFromHexEncodedString( rgchSignPub );
CECSigningPrivateKey testRebuildSign;
testRebuildSign.RebuildFromPrivateData( buf + 32 );
CHECK( testRebuildSign == signPriv && testRebuildSign.MatchesPublicKey( signPub ) );
uint8 bufMessageToSign[ (sizeof(rgchMessageToSign)-1)/2 ];
V_hextobinary( rgchMessageToSign, sizeof(rgchMessageToSign)-1, bufMessageToSign, sizeof(bufMessageToSign) );
uint8 bufExpected[64];
V_hextobinary( rgchExpected, 128, bufExpected, 64 );
CryptoSignature_t signature;
CCrypto::GenerateSignature( bufMessageToSign, sizeof(bufMessageToSign), signPriv, &signature );
CHECK( V_memcmp( signature, bufExpected, sizeof(CryptoSignature_t) ) == 0 );
bufMessageToSign[5] ^= 1;
CHECK( !CCrypto::VerifySignature( bufMessageToSign, sizeof( bufMessageToSign ), signPub, signature ) );
bufMessageToSign[5] ^= 1;
signature[20] ^= 1;
CHECK( !CCrypto::VerifySignature( bufMessageToSign, sizeof( bufMessageToSign ), signPub, signature ) );
signature[20] ^= 1;
CHECK( CCrypto::VerifySignature( bufMessageToSign, sizeof( bufMessageToSign ), signPub, signature ) );
// test signing with random keys
signPriv.Wipe();
signPub.Wipe();
CCrypto::GenerateSigningKeyPair( &signPub, &signPriv );
CCrypto::GenerateSignature( bufMessageToSign, sizeof( bufMessageToSign ), signPriv, &signature );
bufMessageToSign[5] ^= 1;
CHECK( !CCrypto::VerifySignature( bufMessageToSign, sizeof( bufMessageToSign ), signPub, signature ) );
bufMessageToSign[5] ^= 1;
signature[20] ^= 1;
CHECK( !CCrypto::VerifySignature( bufMessageToSign, sizeof( bufMessageToSign ), signPub, signature ) );
signature[20] ^= 1;
CHECK( CCrypto::VerifySignature( bufMessageToSign, sizeof( bufMessageToSign ), signPub, signature ) );
// test public/private key relationships and operators
CECSigningPublicKey testSignPubFromPriv;
signPriv.GetPublicKey( &testSignPubFromPriv );
CHECK( testSignPubFromPriv == signPub );
CHECK( signPriv.MatchesPublicKey( testSignPubFromPriv ) );
CCrypto::GenerateSigningKeyPair( &signPub, &signPriv );
CHECK( testSignPubFromPriv != signPub );
CHECK( !signPriv.MatchesPublicKey( testSignPubFromPriv ) );
}
//-----------------------------------------------------------------------------
// Purpose: Test parsing and re-writing of keys in OpenSSH formats
//-----------------------------------------------------------------------------
void TestOpenSSHEd25519()
{
char buf[ 2048 ];
// Generate some keys, and make sure we can round trip them
{
CECSigningPublicKey pubKey;
CECSigningPrivateKey privKey;
CCrypto::GenerateSigningKeyPair( &pubKey, &privKey );
uint32 cbBuf;
// Get public key as authorized_keys format.
// Should fail if we tell it the buffer is too small
CHECK( !pubKey.GetAsOpenSSHAuthorizedKeys( buf, 16, &cbBuf, "" ) );
CHECK( pubKey.GetAsOpenSSHAuthorizedKeys( buf, sizeof(buf), &cbBuf, "" ) );
CHECK( cbBuf == V_strlen( buf )+1 );
CHECK( 75 <= cbBuf && cbBuf <= 85 ); // typical size (assuming no password or key comment). Not necessarily a bug if this assert fires, but maybe something suspicious
// Parse it back out, make sure it matches
CECSigningPublicKey pubKey2;
CHECK( pubKey2.LoadFromAndWipeBuffer( buf, cbBuf ) );
CHECK( pubKey2 == pubKey );
// Get private key in openSSH PEM-ish format.
// Should fail if we tell it the buffer is too small
CHECK( !privKey.GetAsPEM( buf, 64, &cbBuf ) );
CHECK( privKey.GetAsPEM( buf, sizeof(buf), &cbBuf ) );
CHECK( cbBuf == V_strlen( buf )+1 );
CHECK( 370 <= cbBuf && cbBuf <= 390 ); // typical size (assuming no password or key comment). Not necessarily a bug if this assert fires, but maybe something suspicious
// Parse it back out, make sure it matches
CECSigningPrivateKey privKey2;
CHECK( privKey2.LoadFromAndWipeBuffer( buf, cbBuf ) );
CHECK( privKey2 == privKey );
CHECK( privKey2.MatchesPublicKey( pubKey2 ) );
}
// Parse some known keys
{
CUtlBuffer bufPrivKeyPEMA;
bufPrivKeyPEMA.PutString(
R"PEM(
-----BEGIN OPENSSH PRIVATE KEY-----
b3BlbnNzaC1rZXktdjEAAAAABG5vbmUAAAAEbm9uZQAAAAAAAAABAAAAMwAAAAtzc2gtZW
QyNTUxOQAAACC3vdN6llE0by4d7aFur0nBXdu5hXJb7LLkiC5UCEPFDgAAAJgJaJG1CWiR
tQAAAAtzc2gtZWQyNTUxOQAAACC3vdN6llE0by4d7aFur0nBXdu5hXJb7LLkiC5UCEPFDg
AAAECpUfg4C0BkgsCO+GlFAbcTQZUeFFQcamXzDA1tx7aNWre903qWUTRvLh3toW6vScFd
27mFclvssuSILlQIQ8UOAAAAEmZsZXRjaGVyZEBzcmNkczAwMwECAw==
-----END OPENSSH PRIVATE KEY-----
)PEM" );
CECSigningPrivateKey privKeyA;
CHECK( privKeyA.LoadFromAndWipeBuffer( bufPrivKeyPEMA.Base(), bufPrivKeyPEMA.TellPut() ) );
CUtlBuffer bufPubKeyA;
bufPubKeyA.PutString( "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAILe903qWUTRvLh3toW6vScFd27mFclvssuSILlQIQ8UO" );
CECSigningPublicKey pubKeyA;
CHECK( pubKeyA.LoadFromAndWipeBuffer( bufPubKeyA.Base(), bufPubKeyA.TellPut() ) );
CHECK( privKeyA.MatchesPublicKey( pubKeyA ) );
}
}
//-----------------------------------------------------------------------------
// Purpose: Tests elliptic crypto perf
//-----------------------------------------------------------------------------
void TestEllipticPerf()
{
uint64 usecStart;
const int k_cubPktBig = 1024*1024*10;
const int k_cubPktSmall = 128;
CUtlBuffer bufData;
bufData.EnsureCapacity( k_cubPktBig );
bufData.SeekPut( CUtlBuffer::SEEK_HEAD, k_cubPktBig );
int k_cIterationsECDH = 500;
int k_cIterationsSignSmall = 500;
int k_cIterationsSignBig = 25;
CECKeyExchangePublicKey lastPub;
CECKeyExchangePrivateKey lastPriv;
CCrypto::GenerateKeyExchangeKeyPair( &lastPub, &lastPriv );
usecStart = Plat_USTime();
int x = 0;
for ( int i = 0; i < k_cIterationsECDH; ++i )
{
SHA256Digest_t sharedsecret;
CECKeyExchangePublicKey pub;
CECKeyExchangePrivateKey priv;
CCrypto::GenerateKeyExchangeKeyPair( &pub, &priv );
CCrypto::PerformKeyExchange( priv, lastPub, &sharedsecret );
x ^= sharedsecret[0] ^ sharedsecret[sizeof(sharedsecret)-1];
}
double dMicrosecPerECDH = double( Plat_USTime() - usecStart ) / k_cIterationsECDH;
CECSigningPublicKey signPub;
CECSigningPrivateKey signPriv;
CCrypto::GenerateSigningKeyPair( &signPub, &signPriv );
CryptoSignature_t signature;
// small data sign
usecStart = Plat_USTime();
for ( int i = 0; i < k_cIterationsSignSmall; ++i )
{
CCrypto::GenerateSignature( (uint8*)bufData.Base(), k_cubPktSmall, signPriv, &signature );
x ^= signature[0] ^ signature[sizeof( signature ) - 1];
}
double dMicrosecPerSignSmall = double( Plat_USTime() - usecStart ) / k_cIterationsSignSmall;
// small data verify
usecStart = Plat_USTime();
for ( int i = 0; i < k_cIterationsSignSmall; ++i )
{
x ^= (int)CCrypto::VerifySignature( (uint8*)bufData.Base(), k_cubPktSmall, signPub, signature );
}
double dMicrosecPerSignCheckSmall = double( Plat_USTime() - usecStart ) / k_cIterationsSignSmall;
// large data sign
usecStart = Plat_USTime();
for ( int i = 0; i < k_cIterationsSignBig; ++i )
{
CCrypto::GenerateSignature( (uint8*)bufData.Base(), k_cubPktBig, signPriv, &signature );
x ^= signature[0] ^ signature[sizeof( signature ) - 1];
}
double elapsed = Plat_USTime() - usecStart;
double dMicrosecPerSignBig = elapsed / k_cIterationsSignBig;
double dRateLargeMBPerSec = double( k_cubPktBig ) * k_cIterationsSignBig / elapsed;
// large data verify
usecStart = Plat_USTime();
for ( int i = 0; i < k_cIterationsSignBig; ++i )
{
x ^= (int)CCrypto::VerifySignature( (uint8*)bufData.Base(), k_cubPktBig, signPub, signature );
}
elapsed = Plat_USTime() - usecStart;
double dMicrosecPerSignCheckBig = elapsed / k_cIterationsSignBig;
double dRateLargeMBPerSecCheck = double( k_cubPktBig ) * k_cIterationsSignBig / elapsed;
printf( "\tEphemeral curve25519 key exchange:\t\t\t%f microseconds each (%d iterations)\n", dMicrosecPerECDH, k_cIterationsSignSmall );
printf( "\tCalculate ed25519 signature (small):\t\t\t%f microseconds each (%d iterations)\n", dMicrosecPerSignSmall, k_cIterationsSignSmall );
printf( "\tCalculate ed25519 signature (big):\t\t\t%f microseconds each (%d iterations)\n", dMicrosecPerSignBig, k_cIterationsSignBig );
printf( "\tCalculate ed25519 signature (big):\t\t\t%f MB/sec (%d iterations)\n", dRateLargeMBPerSec, k_cIterationsSignBig );
printf( "\tVerify ed25519 signature (small):\t\t\t%f microseconds each (%d iterations)\n", dMicrosecPerSignCheckSmall, k_cIterationsSignSmall );
printf( "\tVerify ed25519 signature (big):\t\t\t%f microseconds each (%d iterations)\n", dMicrosecPerSignCheckBig, k_cIterationsSignBig );
printf( "\tVerify ed25519 signature (big):\t\t\t%f MB/sec (%d iterations)\n", dRateLargeMBPerSecCheck, k_cIterationsSignBig );
}
//-----------------------------------------------------------------------------
// Purpose: Performs specified # of symmetric encryptions
//-----------------------------------------------------------------------------
void SymmetricEncryptRepeatedly( int cIterations, uint8 *pubData, int cubToEncrypt, uint8 *pubIV, uint cubIV, uint8 *pubKey, uint cubKey )
{
int nBufSize = cubToEncrypt + 32; // 16 = AES block size.. worst case for padded data
uint8 *pEncrypted = new uint8[ nBufSize ];
// try a bunch of iterations of symmetric encrypting big packets
for ( int iIteration = 0; iIteration < cIterations; iIteration++ )
{
uint cubEncrypted = nBufSize;
bool bRet = CCrypto::SymmetricEncryptWithIV( &pubData[iIteration], cubToEncrypt,
pubIV, cubIV,
pEncrypted, &cubEncrypted,
pubKey, cubKey );
CHECK( bRet ); // must succeed
}
delete [] pEncrypted;
}
//-----------------------------------------------------------------------------
// Purpose: Performs specified # of symmetric descryptions
//-----------------------------------------------------------------------------
void SymmetricDecryptRepeatedly( int cIterations, uint8 *pubEncrypted, int cubEncrypted, uint8 *pubIV, uint cubIV, uint8 *pubKey, uint cubKey )
{
int nBufSize = cubEncrypted + 32; // 16 = AES block size.. worst case for padded data
uint8 *pEncrypted = new uint8[ nBufSize ];
// try a bunch of iterations of symmetric encrypting big packets
for ( int iIteration = 0; iIteration < cIterations; iIteration++ )
{
uint cubOutput = nBufSize;
bool bRet = CCrypto::SymmetricDecryptWithIV( pubEncrypted, cubEncrypted,
pubIV, cubIV,
pEncrypted, &cubOutput,
pubKey, cubKey );
CHECK( bRet ); // must succeed
}
delete [] pEncrypted;
}
//-----------------------------------------------------------------------------
// Purpose: Tests symmetric crypto perf
//-----------------------------------------------------------------------------
void TestSymmetricCryptoPerf()
{
uint64 usecStart;
// generate a random key
uint8 rgubKey[k_nSymmetricKeyLen];
uint8 rgubIV[ k_nSymmetricBlockSize ];
CCrypto::GenerateRandomBlock( rgubKey, V_ARRAYSIZE( rgubKey ) );
const int k_cIterations = 10000;
const int k_cMaxData = 800;
const int k_cBufs = 5;
const int k_cubTestBuf = k_cMaxData * k_cBufs + k_cIterations;
const int k_cubPktBig = 2000;
const int k_cubPktSmall = 40;
uint8 rgubData[k_cubTestBuf];
CCrypto::GenerateRandomBlock( rgubIV, V_ARRAYSIZE( rgubIV ) );
// fill data buffer with arbitrary data
uint8 rgubEncrypted[ k_cubPktBig + 32 ]; // 16 = AES block size.. worst case for padded data
for ( int iubData = 0; iubData < V_ARRAYSIZE( rgubData ); iubData ++ )
rgubData[iubData] = (uint8) iubData;
// try a bunch of iterations of symmetric encrypting small packets
usecStart = Plat_USTime();
SymmetricEncryptRepeatedly( k_cIterations, rgubData, k_cubPktSmall, rgubIV, k_nSymmetricBlockSize, rgubKey, V_ARRAYSIZE( rgubKey ) );
int cMicroSecPerEncryptSmall = Plat_USTime() - usecStart;
// try a bunch of iterations of symmetric encrypting big packets
usecStart = Plat_USTime();
SymmetricEncryptRepeatedly( k_cIterations, rgubData, k_cubPktBig, rgubIV, k_nSymmetricBlockSize, rgubKey, V_ARRAYSIZE( rgubKey ) );
int cMicroSecPerEncryptBig = Plat_USTime() - usecStart;
// try a bunch of iterations decrypting small packets
uint cubEncrypted = V_ARRAYSIZE( rgubEncrypted );
bool bRet = CCrypto::SymmetricEncryptWithIV( rgubData, k_cubPktSmall,
rgubIV, k_nSymmetricBlockSize,
rgubEncrypted, &cubEncrypted,
rgubKey, V_ARRAYSIZE( rgubKey ) );
CHECK( bRet );
usecStart = Plat_USTime();
SymmetricDecryptRepeatedly( k_cIterations, rgubEncrypted, cubEncrypted, rgubIV, k_nSymmetricBlockSize, rgubKey, V_ARRAYSIZE( rgubKey ) );
int cMicroSecPerDecryptSmall = Plat_USTime() - usecStart;
// try a bunch of iterations decrypting big packets
cubEncrypted = V_ARRAYSIZE( rgubEncrypted );
bRet = CCrypto::SymmetricEncryptWithIV( rgubData, k_cubPktBig,
rgubIV, k_nSymmetricBlockSize,
rgubEncrypted, &cubEncrypted,
rgubKey, V_ARRAYSIZE( rgubKey ) );
CHECK( bRet );
usecStart = Plat_USTime();
SymmetricDecryptRepeatedly( k_cIterations, rgubEncrypted, cubEncrypted, rgubIV, k_nSymmetricBlockSize, rgubKey, V_ARRAYSIZE( rgubKey ) );
int cMicroSecPerDecryptBig = Plat_USTime() - usecStart;
printf( "\tSymmetric encrypt (small):\t\t%d microsec (%d iterations)\n", cMicroSecPerEncryptSmall, k_cIterations );
printf( "\tSymmetric encrypt (big):\t\t%d microsec (%d iterations)\n", cMicroSecPerEncryptBig, k_cIterations );
printf( "\tSymmetric decrypt (small):\t\t%d microsec (%d iterations)\n", cMicroSecPerDecryptSmall, k_cIterations );
printf( "\tSymmetric decrypt (big):\t\t%d microsec (%d iterations)\n", cMicroSecPerDecryptBig, k_cIterations );
}
int main()
{
CCrypto::Init();
TestCryptoEncoding();
TestSymmetricCrypto();
TestEllipticCrypto();
TestOpenSSHEd25519();
TestEllipticPerf();
TestSymmetricCryptoPerf();
return 0;
}